Booking Experts data layer overview

Overview, setup checklist, and implementation guidance for the Booking Experts data layer.

This document describes the data layer events emitted by the website and checkout portal, plus how to map them in Google Tag Manager for GA4 and other platforms.

Event reference: See Booking Experts data layer events for the full event list and payloads.

Legacy reference: See Deprecated data layer events for legacy payloads and removals.

Example GTM container: See Example GTM container for a starter configuration.

Pre-release notice: These docs reflect the schema that becomes active on 2 Mar 2026. Events marked active are part of that upcoming schema. Events marked deprecated are legacy payloads still emitted today and will stop after 2 Mar 2026.

Changelog

Legacy: Existing events/payloads currently emitted in production, but superseded by the new schema.

Deprecated: Legacy items scheduled for removal on 2 Mar 2026 (breaking change). They are still emitted today but will stop after that date.

  • Breaking changes: Legacy payload variants stop being emitted (same event name, new schema required): view_item_list, view_item, add_to_cart, remove_from_cart, select_promotion, search, purchase, failed_payment.
  • Breaking changes: Legacy-only event names stop being emitted: productImpression, productDetail, promotionClick, propertyDetail, formSubmit, view_price_information, visit_checkout_step, select_amenity, deselect_amenity, ga4purchase, ga4option, ga4optionConfirm, checkout, addToCart, option.
  • Breaking changes: Legacy remarketing objects stop being emitted: hrental_* objects and return_from_payment.
  • Breaking changes: Schema enforcement: ecommerce payloads must use ecommerce.* (ecommerce.items, ecommerce.currency, ecommerce.value, etc.). Top-level legacy keys are no longer emitted.
  • Breaking changes: Migration required: update GTM triggers and variables to the new event payload structure and replacement events listed in each deprecated section.

Website types and flows

We support two website types: Sales sites (property inquiry forms) and Booking sites (availability search + booking). The checkout portal only exists on Booking sites, and only fires when the guest reaches those steps.

  • Sales sites: listings + property detail + inquiry forms; no cart or checkout portal.
  • Booking sites: availability search + booking cart + checkout portal (with online payment return).

Sales site flow (example)

view_item_listselect_itemview_item

form_startform_submitgenerate_lead

Booking site flow (example)

search_results_loadedview_item_listselect_itemview_item

add_to_cartview_cart (booking site cart)

begin_checkout (checkout portal stay step)

checkout_progress (stay step + guest group/license plates updates)

view_cart (upgrades step) → checkout_progress (upgrades step)

add_to_cart/remove_from_cart (upgrades)

add_shipping_info (after details submit)

checkout_progress (details step) → checkout_progress (confirm step)

add_payment_info (payment portal) → purchase/failed_payment (after payment return)

Booking option flow (example)

generate_leadclose_convert_lead

Site search flow (example)

searchview_search_results

What you need before you start

  • A GTM container present on all involved BEX apps/domains (site + checkout + payment return).
  • A GA4 property with a web data stream.
  • A decision on your Ads/Meta/affiliate stack (and which tags you will run).

Quick start checklist

  • Install the GA4 base tag on all pages.
  • Enable “Send ecommerce data” in the GA4 tag.
  • Create data layer variables for required custom fields.
  • Configure cross-domain measurement if multiple domains are involved.
  • Configure Consent Mode v2 if using Ads/remarketing.
  • Add a Conversion Linker tag if using Google Ads tags.
  • Trigger purchase and verify in DebugView.
  • Trigger lead events (generate_lead/close_convert_lead) if used.
  • Validate tags with GTM Preview + Tag Assistant.
  • Confirm reporting variables are mapped to GA4 custom definitions.

Glossary

  • Administration: The PMS administration entity for the booking (used for reporting by park/site/brand).
  • Portal settings: Checkout portal configuration snapshot included with events to analyze form requirements.
  • Preferences: Guest preferences/amenities selected during the checkout flow.
  • Special period: Seasonal or promotional period used in availability/search reporting.
  • Customer type: new or returning, derived from booking history in the last 540 days.

Objects vs GA4 parameters

  • GA4 has strict naming/length limits for event/parameter names and values.
  • BEX emits rich objects like booking and portal_settings for GTM variable extraction.
  • Do not send entire objects to GA4 as a parameter; extract only the scalar fields you need for reporting.

Event availability matrix

Definitions: Sales/Booking site refer to CMS website pages; Checkout portal refers to the separate checkout portal app/domain used after cart. Flow describes where the event occurs in the journey.

EventStatusSales siteBooking siteCheckout portal
add_to_cartActiveNoYesYes
remove_from_cartActiveNoYesYes
view_cartActiveNoYesYes
begin_checkoutActiveNoNoYes
add_shipping_infoActiveNoNoYes
add_payment_infoActiveNoNoYes
purchaseActiveNoNoYes
generate_leadActiveYesNoYes
close_convert_leadActiveNoNoYes
view_item_listActiveYesYesNo
select_itemActiveYesYesNo
view_itemActiveYesYesNo
view_promotionActiveNoYesYes
select_promotionActiveNoYesYes
view_search_resultsActiveYesYesNo
searchActiveYesYesNo
search_results_loadedActiveNoYesNo
checkout_progressActiveNoNoYes
form_startActiveYesYesNo
form_submitActiveYesYesNo
failed_paymentActiveNoNoYes
user_identity_updateActiveNoNoYes
view_item_listDeprecated (removal 2 Mar 2026)NoYesNo
view_itemDeprecated (removal 2 Mar 2026)NoYesNo
add_to_cartDeprecated (removal 2 Mar 2026)NoYesNo
remove_from_cartDeprecated (removal 2 Mar 2026)NoYesNo
select_promotionDeprecated (removal 2 Mar 2026)YesYesNo
searchDeprecated (removal 2 Mar 2026)YesYesNo
productImpressionDeprecated (removal 2 Mar 2026)NoYesNo
productDetailDeprecated (removal 2 Mar 2026)NoYesNo
promotionClickDeprecated (removal 2 Mar 2026)YesYesNo
propertyDetailDeprecated (removal 2 Mar 2026)YesNoNo
formSubmitDeprecated (removal 2 Mar 2026)YesNoNo
view_price_informationDeprecated (removal 2 Mar 2026)NoYesYes
visit_checkout_stepDeprecated (removal 2 Mar 2026)NoYesYes
select_amenityDeprecated (removal 2 Mar 2026)NoYesYes
deselect_amenityDeprecated (removal 2 Mar 2026)NoYesYes
ga4purchaseDeprecated (removal 2 Mar 2026)NoYesNo
ga4optionDeprecated (removal 2 Mar 2026)NoYesYes
ga4optionConfirmDeprecated (removal 2 Mar 2026)NoYesYes
purchaseDeprecated (removal 2 Mar 2026)NoYesNo
checkoutDeprecated (removal 2 Mar 2026)NoYesYes
addToCartDeprecated (removal 2 Mar 2026)NoYesYes
optionDeprecated (removal 2 Mar 2026)NoYesYes
failed_paymentDeprecated (removal 2 Mar 2026)NoYesNo
hrental_homeDeprecated (removal 2 Mar 2026)NoYesNo
hrental_home_subsiteDeprecated (removal 2 Mar 2026)NoYesNo
hrental_searchresultsDeprecated (removal 2 Mar 2026)NoYesNo
hrental_offerdetailDeprecated (removal 2 Mar 2026)NoYesNo
hrental_conversionDeprecated (removal 2 Mar 2026)NoYesNo
return_from_paymentDeprecated (removal 2 Mar 2026)NoYesNo

Core GTM configuration

  • Create a GA4 configuration tag (Google tag / GA4 config) that fires on all pages to initialize the data stream.
  • Create a Google Analytics: GA4 Event tag with Event Name {{Event}}.
  • In More Settings → Ecommerce, enable Send Ecommerce data with Data Source Data Layer.
  • Create data layer variables for every custom parameter you want to report on.
  • Trigger the GA4 Event tag with Custom Event triggers matching the data layer event name (for example purchase), not on gtm.js or gtm.load.
  • Publish your GTM container and validate in GA4 realtime/debug view.

Parameter types: ga4 fields are standard GA4 parameters. custom fields need data layer variables in GTM and custom definitions in GA4 to be reportable.

Copy/paste safety: do not quote numeric values for value, price, quantity, or index.

Reference: GA4 ecommerce via GTM

GA4 enhanced measurement (avoid duplicates)

If you rely on BEX events, disable overlapping GA4 Enhanced Measurement features in the data stream settings to avoid double counting.

  • Form interactions: disable if you use form_start and form_submit from BEX.
  • Site search: disable if you use view_search_results (and search if mapped) from BEX.

GA4 reporting

What to register in GA4

Register custom parameters as custom dimensions or metrics in GA4 (Admin → Custom definitions) only if you need them in reports or audiences.

Recommended GA4 reporting dimensions for BEX

GA4 parameter names may contain only letters, numbers, and underscores, and must start with a letter. Keep dotted data layer paths in GTM, but map them to underscore parameter names in GA4.

Register custom definitions with the correct scope. Item fields live inside ecommerce.items[] and require item-scoped custom dimensions.

BEX provides user.user_id in the data layer. If you want GA4 User-ID, map that value to the GA4 tag User-ID field (built-in user_id, not a user property or custom definition).

If you prefer user properties, use GA4-safe names like bex_user_id and bex_returning_customer. Treat these identifiers as pseudonymous and only set them when your privacy policy and consent allow it.

Data layer pathGA4 parameter nameScopeGTM mapping
ecommerce.customer_typecustomer_typeEVENTGA4 Event tag -> Event Parameters
search.special_period_idsearch_special_period_idEVENTGA4 Event tag -> Event Parameters
search.special_period_namesearch_special_period_nameEVENTGA4 Event tag -> Event Parameters
search.amenities_countsearch_amenities_countEVENTGA4 Event tag -> Event Parameters
search.nightssearch_nightsEVENTGA4 Event tag -> Event Parameters
portal_settings.required_fields_countportal_settings_required_fields_countEVENTGA4 Event tag -> Event Parameters
portal_settings.optional_fields_countportal_settings_optional_fields_countEVENTGA4 Event tag -> Event Parameters
booking.statusbooking_statusEVENTGA4 Event tag -> Event Parameters
ecommerce.items[].administration_idadministration_idITEMItems array (item-scoped custom dimension)
ecommerce.items[].administration_nameadministration_nameITEMItems array (item-scoped custom dimension)
ecommerce.items[].item_countryitem_countryITEMItems array (item-scoped custom dimension)
ecommerce.items[].item_regionitem_regionITEMItems array (item-scoped custom dimension)
ecommerce.items[].item_cityitem_cityITEMItems array (item-scoped custom dimension)
ecommerce.items[].item_variantitem_variantITEMItems array (built-in GA4 item parameter; no custom definition)
user.user_iduser_id (built-in)USER-IDGA4 tag User-ID field
user.user_idbex_user_idUSERGA4 Event tag -> User Properties
user.returning_customerbex_returning_customerUSERGA4 Event tag -> User Properties

PII and user_data (important)

Do NOT register or report the following in GA4:

  • user_data.sha256_email_address
  • user_data.sha256_phone_number

Never send email or phone numbers to GA4, even when hashed. GA4 does not allow PII collection.

user_data is hashed identifiers for marketing platforms (for example Google Ads Enhanced Conversions and Meta Advanced Matching). It must be sent only when marketing consent is granted and must not be mapped into GA4 tags.

Google Ads conversions

BEX emits GA4-recommended ecommerce events (e.g. purchase, begin_checkout, add_to_cart). In Google Ads you typically choose between:

  • Option A — Import GA4 key events into Google Ads: simpler setup if GA4 is your source of truth.
  • Option B — Implement Google Ads tags via GTM (recommended): Google Ads-native conversion measurement and advanced features (Enhanced Conversions, remarketing).

Minimum Google Ads tag set in GTM (Option B)

1) Conversion Linker (required)

Create a Conversion Linker tag and trigger it on All Pages.

If your flow spans multiple domains:

  • Enable Link across domains.
  • Fill Auto Link Domains with all involved domains.
  • Optional: enable Decorate Forms if form submits navigate across domains.

2) Google Ads Conversion Tracking (purchase)

  • Trigger: Custom Event = purchase
  • Conversion ID + Conversion Label from Google Ads
  • Conversion Value: {{DLV - ecommerce.value}}
  • Currency Code: {{DLV - ecommerce.currency}}
  • Transaction ID: {{DLV - ecommerce.transaction_id}} (recommended)

BEX mapping: ecommerce.transaction_id → Order ID, ecommerce.value → conversion value, ecommerce.currency → currency.

3) Optional: secondary conversions

  • generate_lead (lead submitted)
  • close_convert_lead (booking option converted)

Mark these as secondary in Google Ads unless you want them as a main bidding signal.

Dynamic remarketing (optional, recommended if you use feeds)

If you run dynamic remarketing, maintain a matching feed and send event data with items[] so Google Ads can match items[].id to the feed Property ID.

For BEX, map:

  • items[].id = ecommerce.items[].item_id
  • items[].google_business_vertical = hotel_rental
  • items[].start_date = ecommerce.items[].start_date
  • items[].end_date = ecommerce.items[].end_date

Recommended event names: view_item, view_item_list, view_search_results, add_to_cart, purchase.

Enhanced Conversions (recommended; requires Consent Mode v2)

Enhanced Conversions uses consented user-provided data to improve conversion matching.

Consent requirements

Only send user-provided data when ad_user_data = granted (and marketing consent is granted).

BEX data available

  • user_data.sha256_email_address
  • user_data.sha256_phone_number

GTM implementation approach (recommended with BEX)

  1. Create a data layer variable named user_data (data layer variable name: user_data).
  2. Create a User-Provided Data variable in GTM using Code configuration from the user_data object.
  3. Assign that user-provided data variable to Google tags via the user_data parameter or use the Google Ads User-Provided Data Event tag.

Triggering

Fire user-provided data collection on the same events where BEX provides user_data:

  • purchase
  • generate_lead (optional)

Implementation diagram

  1. BEX user action triggers purchase.
  2. dataLayer.push({ event: "purchase", ecommerce: {...}, user_data: {...} })
  3. GTM Trigger: Custom Event purchase.
  4. Conversion Linker runs on All Pages (with cross-domain linking enabled).
  5. Google Ads Conversion Tracking tag fires for purchase.
  6. Google Ads conversion is recorded.

How to configure in GTM (short)

  1. Tags → New → Conversion Linker → Trigger: All Pages (enable link across domains + Auto Link Domains).
  2. Tags → New → Google Ads Conversion Tracking → ID/Label from Google Ads → map ecommerce.* → Trigger: purchase.
  3. Optional: Remarketing tag with dynamic remarketing parameters and items[].
  4. Optional: Enhanced Conversions using User-Provided Data variable (only when ad_user_data = granted).

Meta Ads (Pixel / Conversions API)

Event mapping (BEX → Meta)

BEX eventMeta event
view_itemViewContent
view_item_listtrackCustom (ViewItemList)
select_itemtrackCustom (SelectItem)
view_search_resultsSearch
add_to_cartAddToCart
begin_checkoutInitiateCheckout
add_payment_infoAddPaymentInfo
purchasePurchase
generate_leadLead

Map ViewContent only for view_item (product detail views). Do not map list or click events to ViewContent.

Fire Meta Search only once per user search. Use view_search_results as the canonical event and do not map search to avoid double counting.

Required parameters

  • valueecommerce.value
  • currencyecommerce.currency
  • content_ids → derived from ecommerce.items[].item_id
  • content_typeproduct

Search events: set Meta search_string from the BEX/GA4 search_term value (from search or view_search_results).

Event deduplication (Pixel + CAPI)

  • Pixel uses eventID and CAPI uses event_id; the values must match for deduplication.
  • Recommended defaults: event_id = ecommerce.transaction_id for purchases, event_id = lead_id for leads (checkout portal uses the booking number). Use the same value for Pixel eventID.

Advanced matching

When marketing consent is granted:

  • Pass user_data.sha256_email_address
  • Pass user_data.sha256_phone_number

Notes: only send with marketing consent; never send raw email or phone values; do not reuse these fields for GA4 reporting.

TradeTracker

TradeTracker conversion tracking should fire only on the final conversion event: purchase. Do not fire on pageviews or intermediate checkout steps.

Required TradeTracker configuration (GTM template)

  • Campaign ID (static; provided by TradeTracker)
  • ProductGroup ID (static; provided by TradeTracker)
  • Transaction ID (dynamic; from BEX)
  • Transaction Amount (dynamic; from BEX)
  • Optional fields depending on program configuration (for example voucher)

BEX mapping (Sales tag)

  • Transaction ID → ecommerce.transaction_id
  • Transaction Amount → booking.base_rent_total (ensure this matches your TradeTracker commission agreement)
  • Currency → ecommerce.currency
  • Voucher code → ecommerce.coupon (if used)

Rule: TradeTracker amount must match the affiliate agreement definition (base rent vs gross booking value). Default: booking.base_rent_total.

Implementation diagram

  1. BEX emits purchase.
  2. dataLayer.push({ event: "purchase", ecommerce: { transaction_id, value, currency, ... }, booking: { base_rent_total } })
  3. GTM Trigger: Custom Event purchase.
  4. TradeTracker Sales Tag fires with Campaign ID, ProductGroup ID, and transaction variables.
  5. TradeTracker conversion is recorded.

How to configure in GTM (short)

  1. Import the TradeTracker template.
  2. Create constants: TT - Campaign ID, TT - ProductGroup ID.
  3. Create data layer variables: DLV - ecommerce.transaction_id, DLV - booking.base_rent_total, DLV - ecommerce.currency.
  4. Create the Sales tag with the above variables and trigger on purchase.
  5. Test in GTM Preview and validate the conversion once per transaction_id.

Cross-domain tracking & Consent

BEX flows often span multiple domains. Configure both GA4 cross-domain measurement and Conversion Linker cross-domain linking.

Cross-domain measurement (GA4)

  • Use one GA4 property + one Web data stream.
  • Use the same tag ID across all domains.
  • Admin → Data Streams → select your Web stream → Configure tag settings → Configure your domains.
  • Add all involved domains and verify the _gl linker parameter on navigation.
  • If redirects drop query parameters, cross-domain will fail; preserve _gl.
  • Reference: GA4 cross-domain measurement

Google Ads cross-domain (Conversion Linker)

  • Add a Conversion Linker (All Pages).
  • Enable link across domains and set Auto Link Domains.
  • Optional: Decorate Forms if you submit forms across domains.

Consent Mode v2 (Google tags)

  • Consent types: analytics_storage, ad_storage, ad_user_data, ad_personalization.
  • When analytics_storage = denied, GA4 can send cookieless pings in Advanced Consent Mode; Basic mode blocks tags until consent.
  • When ad_user_data = denied, user-provided-data use cases (Enhanced Conversions) are disabled.

Recommended consent enforcement model

Definitions: Marketing consent means analytics + ads consent (grant analytics_storage, ad_storage, ad_user_data, ad_personalization). Analytics only means grant analytics_storage while ads consent stays denied.

Recommended mode: Advanced Consent Mode v2. In GTM, set default consent to denied in a Consent Initialization tag, enable consent checks on GA4/Ads tags, and update consent when the user makes a choice. Use Basic mode only if you must block all tags until consent.

User choiceGA4 (analytics)Google Ads conversion tagsuser_data
Marketing consentAllowedAllowedAllowed
Analytics onlyAllowedBlockedBlocked
No consentCookieless (Advanced) / Blocked (Basic)Cookieless/limited (Advanced) / Blocked (Basic)Blocked

Implementation diagram

Domain A (site)Domain B (checkout/payment)
Page loadNavigation to domain B
Consent defaults set earlyGA4 cross-domain preserves IDs via _gl
GTM tags evaluate consentConversion Linker preserves click IDs
GA4 / Ads tags fire with consent signalspurchase event attributed correctly

How to configure in GTM (short)

  1. Consent defaults: trigger Consent Initialization on All Pages.
  2. Consent update: run consent update before marketing tags fire.
  3. Cross-domain: configure GA4 domains list + Conversion Linker link across domains.