@shopify/hydrogen-react 2025.7.0 → 2025.7.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (115) hide show
  1. package/dist/browser-dev/ShopifyProvider.mjs +18 -1
  2. package/dist/browser-dev/ShopifyProvider.mjs.map +1 -1
  3. package/dist/browser-dev/analytics-schema-custom-storefront-customer-tracking.mjs +4 -3
  4. package/dist/browser-dev/analytics-schema-custom-storefront-customer-tracking.mjs.map +1 -1
  5. package/dist/browser-dev/analytics.mjs +4 -5
  6. package/dist/browser-dev/analytics.mjs.map +1 -1
  7. package/dist/browser-dev/cart-hooks.mjs +25 -7
  8. package/dist/browser-dev/cart-hooks.mjs.map +1 -1
  9. package/dist/browser-dev/cookies-utils.mjs +4 -4
  10. package/dist/browser-dev/cookies-utils.mjs.map +1 -1
  11. package/dist/browser-dev/index.mjs +4 -0
  12. package/dist/browser-dev/index.mjs.map +1 -1
  13. package/dist/browser-dev/packages/hydrogen-react/package.json.mjs +1 -1
  14. package/dist/browser-dev/tracking-utils.mjs +92 -0
  15. package/dist/browser-dev/tracking-utils.mjs.map +1 -0
  16. package/dist/browser-dev/useShopifyCookies.mjs +96 -9
  17. package/dist/browser-dev/useShopifyCookies.mjs.map +1 -1
  18. package/dist/browser-prod/ShopifyProvider.mjs +18 -1
  19. package/dist/browser-prod/ShopifyProvider.mjs.map +1 -1
  20. package/dist/browser-prod/analytics-schema-custom-storefront-customer-tracking.mjs +4 -3
  21. package/dist/browser-prod/analytics-schema-custom-storefront-customer-tracking.mjs.map +1 -1
  22. package/dist/browser-prod/analytics.mjs +4 -5
  23. package/dist/browser-prod/analytics.mjs.map +1 -1
  24. package/dist/browser-prod/cart-hooks.mjs +25 -7
  25. package/dist/browser-prod/cart-hooks.mjs.map +1 -1
  26. package/dist/browser-prod/cookies-utils.mjs +4 -4
  27. package/dist/browser-prod/cookies-utils.mjs.map +1 -1
  28. package/dist/browser-prod/index.mjs +4 -0
  29. package/dist/browser-prod/index.mjs.map +1 -1
  30. package/dist/browser-prod/packages/hydrogen-react/package.json.mjs +1 -1
  31. package/dist/browser-prod/tracking-utils.mjs +92 -0
  32. package/dist/browser-prod/tracking-utils.mjs.map +1 -0
  33. package/dist/browser-prod/useShopifyCookies.mjs +96 -9
  34. package/dist/browser-prod/useShopifyCookies.mjs.map +1 -1
  35. package/dist/node-dev/ShopifyProvider.js +18 -1
  36. package/dist/node-dev/ShopifyProvider.js.map +1 -1
  37. package/dist/node-dev/ShopifyProvider.mjs +18 -1
  38. package/dist/node-dev/ShopifyProvider.mjs.map +1 -1
  39. package/dist/node-dev/analytics-schema-custom-storefront-customer-tracking.js +4 -3
  40. package/dist/node-dev/analytics-schema-custom-storefront-customer-tracking.js.map +1 -1
  41. package/dist/node-dev/analytics-schema-custom-storefront-customer-tracking.mjs +4 -3
  42. package/dist/node-dev/analytics-schema-custom-storefront-customer-tracking.mjs.map +1 -1
  43. package/dist/node-dev/analytics.js +4 -5
  44. package/dist/node-dev/analytics.js.map +1 -1
  45. package/dist/node-dev/analytics.mjs +4 -5
  46. package/dist/node-dev/analytics.mjs.map +1 -1
  47. package/dist/node-dev/cart-hooks.js +24 -6
  48. package/dist/node-dev/cart-hooks.js.map +1 -1
  49. package/dist/node-dev/cart-hooks.mjs +25 -7
  50. package/dist/node-dev/cart-hooks.mjs.map +1 -1
  51. package/dist/node-dev/cookies-utils.js +4 -4
  52. package/dist/node-dev/cookies-utils.js.map +1 -1
  53. package/dist/node-dev/cookies-utils.mjs +4 -4
  54. package/dist/node-dev/cookies-utils.mjs.map +1 -1
  55. package/dist/node-dev/index.js +4 -0
  56. package/dist/node-dev/index.js.map +1 -1
  57. package/dist/node-dev/index.mjs +4 -0
  58. package/dist/node-dev/index.mjs.map +1 -1
  59. package/dist/node-dev/packages/hydrogen-react/package.json.js +1 -1
  60. package/dist/node-dev/packages/hydrogen-react/package.json.mjs +1 -1
  61. package/dist/node-dev/tracking-utils.js +92 -0
  62. package/dist/node-dev/tracking-utils.js.map +1 -0
  63. package/dist/node-dev/tracking-utils.mjs +92 -0
  64. package/dist/node-dev/tracking-utils.mjs.map +1 -0
  65. package/dist/node-dev/useShopifyCookies.js +94 -7
  66. package/dist/node-dev/useShopifyCookies.js.map +1 -1
  67. package/dist/node-dev/useShopifyCookies.mjs +96 -9
  68. package/dist/node-dev/useShopifyCookies.mjs.map +1 -1
  69. package/dist/node-prod/ShopifyProvider.js +18 -1
  70. package/dist/node-prod/ShopifyProvider.js.map +1 -1
  71. package/dist/node-prod/ShopifyProvider.mjs +18 -1
  72. package/dist/node-prod/ShopifyProvider.mjs.map +1 -1
  73. package/dist/node-prod/analytics-schema-custom-storefront-customer-tracking.js +4 -3
  74. package/dist/node-prod/analytics-schema-custom-storefront-customer-tracking.js.map +1 -1
  75. package/dist/node-prod/analytics-schema-custom-storefront-customer-tracking.mjs +4 -3
  76. package/dist/node-prod/analytics-schema-custom-storefront-customer-tracking.mjs.map +1 -1
  77. package/dist/node-prod/analytics.js +4 -5
  78. package/dist/node-prod/analytics.js.map +1 -1
  79. package/dist/node-prod/analytics.mjs +4 -5
  80. package/dist/node-prod/analytics.mjs.map +1 -1
  81. package/dist/node-prod/cart-hooks.js +24 -6
  82. package/dist/node-prod/cart-hooks.js.map +1 -1
  83. package/dist/node-prod/cart-hooks.mjs +25 -7
  84. package/dist/node-prod/cart-hooks.mjs.map +1 -1
  85. package/dist/node-prod/cookies-utils.js +4 -4
  86. package/dist/node-prod/cookies-utils.js.map +1 -1
  87. package/dist/node-prod/cookies-utils.mjs +4 -4
  88. package/dist/node-prod/cookies-utils.mjs.map +1 -1
  89. package/dist/node-prod/index.js +4 -0
  90. package/dist/node-prod/index.js.map +1 -1
  91. package/dist/node-prod/index.mjs +4 -0
  92. package/dist/node-prod/index.mjs.map +1 -1
  93. package/dist/node-prod/packages/hydrogen-react/package.json.js +1 -1
  94. package/dist/node-prod/packages/hydrogen-react/package.json.mjs +1 -1
  95. package/dist/node-prod/tracking-utils.js +92 -0
  96. package/dist/node-prod/tracking-utils.js.map +1 -0
  97. package/dist/node-prod/tracking-utils.mjs +92 -0
  98. package/dist/node-prod/tracking-utils.mjs.map +1 -0
  99. package/dist/node-prod/useShopifyCookies.js +94 -7
  100. package/dist/node-prod/useShopifyCookies.js.map +1 -1
  101. package/dist/node-prod/useShopifyCookies.mjs +96 -9
  102. package/dist/node-prod/useShopifyCookies.mjs.map +1 -1
  103. package/dist/types/ShopifyProvider.d.ts +5 -0
  104. package/dist/types/analytics-schema-custom-storefront-customer-tracking.d.ts +3 -3
  105. package/dist/types/analytics-types.d.ts +24 -4
  106. package/dist/types/cookies-utils.d.ts +4 -0
  107. package/dist/types/index.d.cts +1 -0
  108. package/dist/types/index.d.ts +1 -0
  109. package/dist/types/tracking-utils.d.ts +21 -0
  110. package/dist/types/useShopifyCookies.d.ts +28 -2
  111. package/dist/umd/hydrogen-react.dev.js +279 -92
  112. package/dist/umd/hydrogen-react.dev.js.map +1 -1
  113. package/dist/umd/hydrogen-react.prod.js +18 -18
  114. package/dist/umd/hydrogen-react.prod.js.map +1 -1
  115. package/package.json +1 -1
@@ -1,9 +1,14 @@
1
1
  import { useCallback } from "react";
2
2
  import { useShop } from "./ShopifyProvider.mjs";
3
- import { SHOPIFY_STOREFRONT_ID_HEADER, SHOPIFY_STOREFRONT_Y_HEADER, SHOPIFY_Y, SHOPIFY_STOREFRONT_S_HEADER, SHOPIFY_S } from "./cart-constants.mjs";
4
- import { getShopifyCookies } from "./cookies-utils.mjs";
3
+ import { SHOPIFY_STOREFRONT_ID_HEADER, SHOPIFY_STOREFRONT_Y_HEADER, SHOPIFY_STOREFRONT_S_HEADER } from "./cart-constants.mjs";
4
+ import { getTrackingValues, SHOPIFY_UNIQUE_TOKEN_HEADER, SHOPIFY_VISIT_TOKEN_HEADER } from "./tracking-utils.mjs";
5
5
  function useCartFetch() {
6
- const { storefrontId, getPublicTokenHeaders, getStorefrontApiUrl } = useShop();
6
+ const {
7
+ storefrontId,
8
+ getPublicTokenHeaders,
9
+ getStorefrontApiUrl,
10
+ sameDomainForStorefrontApi
11
+ } = useShop();
7
12
  return useCallback(
8
13
  ({
9
14
  query,
@@ -13,9 +18,17 @@ function useCartFetch() {
13
18
  if (storefrontId) {
14
19
  headers[SHOPIFY_STOREFRONT_ID_HEADER] = storefrontId;
15
20
  }
16
- const cookieData = getShopifyCookies(document.cookie);
17
- headers[SHOPIFY_STOREFRONT_Y_HEADER] = cookieData[SHOPIFY_Y];
18
- headers[SHOPIFY_STOREFRONT_S_HEADER] = cookieData[SHOPIFY_S];
21
+ if (!sameDomainForStorefrontApi) {
22
+ const { uniqueToken, visitToken } = getTrackingValues();
23
+ if (uniqueToken) {
24
+ headers[SHOPIFY_STOREFRONT_Y_HEADER] = uniqueToken;
25
+ headers[SHOPIFY_UNIQUE_TOKEN_HEADER] = uniqueToken;
26
+ }
27
+ if (visitToken) {
28
+ headers[SHOPIFY_STOREFRONT_S_HEADER] = visitToken;
29
+ headers[SHOPIFY_VISIT_TOKEN_HEADER] = visitToken;
30
+ }
31
+ }
19
32
  return fetch(getStorefrontApiUrl(), {
20
33
  method: "POST",
21
34
  headers,
@@ -33,7 +46,12 @@ function useCartFetch() {
33
46
  };
34
47
  });
35
48
  },
36
- [getPublicTokenHeaders, storefrontId, getStorefrontApiUrl]
49
+ [
50
+ getPublicTokenHeaders,
51
+ storefrontId,
52
+ getStorefrontApiUrl,
53
+ sameDomainForStorefrontApi
54
+ ]
37
55
  );
38
56
  }
39
57
  export {
@@ -1 +1 @@
1
- {"version":3,"file":"cart-hooks.mjs","sources":["../../src/cart-hooks.tsx"],"sourcesContent":["import {useState, useCallback} from 'react';\nimport {useShop} from './ShopifyProvider.js';\nimport {flattenConnection} from './flatten-connection.js';\nimport {CartInput, Cart as CartType} from './storefront-api-types.js';\nimport {CartCreate, defaultCartFragment} from './cart-queries.js';\nimport {Cart} from './cart-types.js';\nimport {\n SHOPIFY_STOREFRONT_ID_HEADER,\n SHOPIFY_STOREFRONT_Y_HEADER,\n SHOPIFY_STOREFRONT_S_HEADER,\n SHOPIFY_Y,\n SHOPIFY_S,\n} from './cart-constants.js';\nimport type {StorefrontApiResponseOkPartial} from './storefront-api-response.types.js';\nimport {getShopifyCookies} from './cookies-utils.js';\n\n// eslint-disable-next-line @typescript-eslint/explicit-function-return-type\nexport function useCartFetch() {\n const {storefrontId, getPublicTokenHeaders, getStorefrontApiUrl} = useShop();\n\n return useCallback(\n <ReturnDataGeneric,>({\n query,\n variables,\n }: {\n query: string;\n variables: Record<string, unknown>;\n }): Promise<StorefrontApiResponseOkPartial<ReturnDataGeneric>> => {\n const headers = getPublicTokenHeaders({contentType: 'json'});\n\n if (storefrontId) {\n headers[SHOPIFY_STOREFRONT_ID_HEADER] = storefrontId;\n }\n\n // Find Shopify cookies\n const cookieData = getShopifyCookies(document.cookie);\n headers[SHOPIFY_STOREFRONT_Y_HEADER] = cookieData[SHOPIFY_Y];\n headers[SHOPIFY_STOREFRONT_S_HEADER] = cookieData[SHOPIFY_S];\n\n return fetch(getStorefrontApiUrl(), {\n method: 'POST',\n headers,\n body: JSON.stringify({\n query: query.toString(),\n variables,\n }),\n })\n .then(\n (res) =>\n res.json() as StorefrontApiResponseOkPartial<ReturnDataGeneric>,\n )\n .catch((error) => {\n return {\n data: undefined,\n // eslint-disable-next-line @typescript-eslint/no-unsafe-assignment, @typescript-eslint/no-unsafe-member-access, @typescript-eslint/no-unsafe-call\n errors: error?.toString(),\n };\n });\n },\n [getPublicTokenHeaders, storefrontId, getStorefrontApiUrl],\n );\n}\n\nexport function useInstantCheckout(): {\n cart: Cart | undefined;\n checkoutUrl: Cart['checkoutUrl'];\n error: string | undefined;\n createInstantCheckout: (cartInput: CartInput) => Promise<void>;\n} {\n const [cart, updateCart] = useState<Cart | undefined>();\n const [checkoutUrl, updateCheckoutUrl] = useState<Cart['checkoutUrl']>();\n const [error, updateError] = useState<string | undefined>();\n\n const fetch = useCartFetch();\n\n const createInstantCheckout = useCallback(\n async (cartInput: CartInput) => {\n const {data, errors} = await fetch<{\n cartCreate: {cart: CartType};\n }>({\n query: CartCreate(defaultCartFragment),\n variables: {\n input: cartInput,\n },\n });\n\n if (errors) {\n // eslint-disable-next-line @typescript-eslint/no-base-to-string\n updateError(errors.toString());\n updateCart(undefined);\n updateCheckoutUrl(undefined);\n }\n\n if (data?.cartCreate?.cart) {\n const dataCart = data.cartCreate.cart;\n updateCart({\n ...dataCart,\n lines: flattenConnection(dataCart.lines),\n note: dataCart.note ?? undefined,\n });\n updateCheckoutUrl(dataCart.checkoutUrl);\n }\n },\n [fetch],\n );\n\n return {cart, checkoutUrl, error, createInstantCheckout};\n}\n"],"names":[],"mappings":";;;;AAiBO,SAAS,eAAe;AAC7B,QAAM,EAAC,cAAc,uBAAuB,oBAAA,IAAuB,QAAA;AAEnE,SAAO;AAAA,IACL,CAAqB;AAAA,MACnB;AAAA,MACA;AAAA,IAAA,MAIgE;AAChE,YAAM,UAAU,sBAAsB,EAAC,aAAa,QAAO;AAE3D,UAAI,cAAc;AAChB,gBAAQ,4BAA4B,IAAI;AAAA,MAC1C;AAGA,YAAM,aAAa,kBAAkB,SAAS,MAAM;AACpD,cAAQ,2BAA2B,IAAI,WAAW,SAAS;AAC3D,cAAQ,2BAA2B,IAAI,WAAW,SAAS;AAE3D,aAAO,MAAM,uBAAuB;AAAA,QAClC,QAAQ;AAAA,QACR;AAAA,QACA,MAAM,KAAK,UAAU;AAAA,UACnB,OAAO,MAAM,SAAA;AAAA,UACb;AAAA,QAAA,CACD;AAAA,MAAA,CACF,EACE;AAAA,QACC,CAAC,QACC,IAAI,KAAA;AAAA,MAAK,EAEZ,MAAM,CAAC,UAAU;AAChB,eAAO;AAAA,UACL,MAAM;AAAA;AAAA,UAEN,QAAQ,+BAAO;AAAA,QAAS;AAAA,MAE5B,CAAC;AAAA,IACL;AAAA,IACA,CAAC,uBAAuB,cAAc,mBAAmB;AAAA,EAAA;AAE7D;"}
1
+ {"version":3,"file":"cart-hooks.mjs","sources":["../../src/cart-hooks.tsx"],"sourcesContent":["import {useState, useCallback} from 'react';\nimport {useShop} from './ShopifyProvider.js';\nimport {flattenConnection} from './flatten-connection.js';\nimport {CartInput, Cart as CartType} from './storefront-api-types.js';\nimport {CartCreate, defaultCartFragment} from './cart-queries.js';\nimport {Cart} from './cart-types.js';\nimport {\n SHOPIFY_STOREFRONT_ID_HEADER,\n SHOPIFY_STOREFRONT_Y_HEADER,\n SHOPIFY_STOREFRONT_S_HEADER,\n} from './cart-constants.js';\nimport type {StorefrontApiResponseOkPartial} from './storefront-api-response.types.js';\nimport {\n getTrackingValues,\n SHOPIFY_UNIQUE_TOKEN_HEADER,\n SHOPIFY_VISIT_TOKEN_HEADER,\n} from './tracking-utils.js';\n\n// eslint-disable-next-line @typescript-eslint/explicit-function-return-type\nexport function useCartFetch() {\n const {\n storefrontId,\n getPublicTokenHeaders,\n getStorefrontApiUrl,\n sameDomainForStorefrontApi,\n } = useShop();\n\n return useCallback(\n <ReturnDataGeneric,>({\n query,\n variables,\n }: {\n query: string;\n variables: Record<string, unknown>;\n }): Promise<StorefrontApiResponseOkPartial<ReturnDataGeneric>> => {\n const headers = getPublicTokenHeaders({contentType: 'json'});\n\n if (storefrontId) {\n headers[SHOPIFY_STOREFRONT_ID_HEADER] = storefrontId;\n }\n\n if (!sameDomainForStorefrontApi) {\n // If we are in cross-domain mode, add tracking headers manually.\n // Otherwise, for same-domain we rely on the browser to attach cookies automatically.\n const {uniqueToken, visitToken} = getTrackingValues();\n if (uniqueToken) {\n headers[SHOPIFY_STOREFRONT_Y_HEADER] = uniqueToken;\n headers[SHOPIFY_UNIQUE_TOKEN_HEADER] = uniqueToken;\n }\n if (visitToken) {\n headers[SHOPIFY_STOREFRONT_S_HEADER] = visitToken;\n headers[SHOPIFY_VISIT_TOKEN_HEADER] = visitToken;\n }\n }\n\n return fetch(getStorefrontApiUrl(), {\n method: 'POST',\n headers,\n body: JSON.stringify({\n query: query.toString(),\n variables,\n }),\n })\n .then(\n (res) =>\n res.json() as StorefrontApiResponseOkPartial<ReturnDataGeneric>,\n )\n .catch((error) => {\n return {\n data: undefined,\n // eslint-disable-next-line @typescript-eslint/no-unsafe-assignment, @typescript-eslint/no-unsafe-member-access, @typescript-eslint/no-unsafe-call\n errors: error?.toString(),\n };\n });\n },\n [\n getPublicTokenHeaders,\n storefrontId,\n getStorefrontApiUrl,\n sameDomainForStorefrontApi,\n ],\n );\n}\n\nexport function useInstantCheckout(): {\n cart: Cart | undefined;\n checkoutUrl: Cart['checkoutUrl'];\n error: string | undefined;\n createInstantCheckout: (cartInput: CartInput) => Promise<void>;\n} {\n const [cart, updateCart] = useState<Cart | undefined>();\n const [checkoutUrl, updateCheckoutUrl] = useState<Cart['checkoutUrl']>();\n const [error, updateError] = useState<string | undefined>();\n\n const fetch = useCartFetch();\n\n const createInstantCheckout = useCallback(\n async (cartInput: CartInput) => {\n const {data, errors} = await fetch<{\n cartCreate: {cart: CartType};\n }>({\n query: CartCreate(defaultCartFragment),\n variables: {\n input: cartInput,\n },\n });\n\n if (errors) {\n // eslint-disable-next-line @typescript-eslint/no-base-to-string\n updateError(errors.toString());\n updateCart(undefined);\n updateCheckoutUrl(undefined);\n }\n\n if (data?.cartCreate?.cart) {\n const dataCart = data.cartCreate.cart;\n updateCart({\n ...dataCart,\n lines: flattenConnection(dataCart.lines),\n note: dataCart.note ?? undefined,\n });\n updateCheckoutUrl(dataCart.checkoutUrl);\n }\n },\n [fetch],\n );\n\n return {cart, checkoutUrl, error, createInstantCheckout};\n}\n"],"names":[],"mappings":";;;;AAmBO,SAAS,eAAe;AAC7B,QAAM;AAAA,IACJ;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,EAAA,IACE,QAAA;AAEJ,SAAO;AAAA,IACL,CAAqB;AAAA,MACnB;AAAA,MACA;AAAA,IAAA,MAIgE;AAChE,YAAM,UAAU,sBAAsB,EAAC,aAAa,QAAO;AAE3D,UAAI,cAAc;AAChB,gBAAQ,4BAA4B,IAAI;AAAA,MAC1C;AAEA,UAAI,CAAC,4BAA4B;AAG/B,cAAM,EAAC,aAAa,WAAA,IAAc,kBAAA;AAClC,YAAI,aAAa;AACf,kBAAQ,2BAA2B,IAAI;AACvC,kBAAQ,2BAA2B,IAAI;AAAA,QACzC;AACA,YAAI,YAAY;AACd,kBAAQ,2BAA2B,IAAI;AACvC,kBAAQ,0BAA0B,IAAI;AAAA,QACxC;AAAA,MACF;AAEA,aAAO,MAAM,uBAAuB;AAAA,QAClC,QAAQ;AAAA,QACR;AAAA,QACA,MAAM,KAAK,UAAU;AAAA,UACnB,OAAO,MAAM,SAAA;AAAA,UACb;AAAA,QAAA,CACD;AAAA,MAAA,CACF,EACE;AAAA,QACC,CAAC,QACC,IAAI,KAAA;AAAA,MAAK,EAEZ,MAAM,CAAC,UAAU;AAChB,eAAO;AAAA,UACL,MAAM;AAAA;AAAA,UAEN,QAAQ,+BAAO;AAAA,QAAS;AAAA,MAE5B,CAAC;AAAA,IACL;AAAA,IACA;AAAA,MACE;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,IAAA;AAAA,EACF;AAEJ;"}
@@ -1,7 +1,7 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, Symbol.toStringTag, { value: "Module" });
3
- const cookie = require("worktop/cookie");
4
3
  const cartConstants = require("./cart-constants.js");
4
+ const trackingUtils = require("./tracking-utils.js");
5
5
  const tokenHash = "xxxx-4xxx-xxxx-xxxxxxxxxxxx";
6
6
  function buildUUID() {
7
7
  let hash = "";
@@ -38,10 +38,10 @@ function hexTime() {
38
38
  return output.padStart(8, "0");
39
39
  }
40
40
  function getShopifyCookies(cookies) {
41
- const cookieData = cookie.parse(cookies);
41
+ const trackingValues = trackingUtils.getTrackingValues(cookies);
42
42
  return {
43
- [cartConstants.SHOPIFY_Y]: cookieData[cartConstants.SHOPIFY_Y] || "",
44
- [cartConstants.SHOPIFY_S]: cookieData[cartConstants.SHOPIFY_S] || ""
43
+ [cartConstants.SHOPIFY_Y]: trackingValues.uniqueToken,
44
+ [cartConstants.SHOPIFY_S]: trackingValues.visitToken
45
45
  };
46
46
  }
47
47
  exports.buildUUID = buildUUID;
@@ -1 +1 @@
1
- {"version":3,"file":"cookies-utils.js","sources":["../../src/cookies-utils.tsx"],"sourcesContent":["// @ts-ignore - worktop/cookie types not properly exported\nimport {parse} from 'worktop/cookie';\nimport {ShopifyCookies} from './analytics-types.js';\nimport {SHOPIFY_Y, SHOPIFY_S} from './cart-constants.js';\n\nconst tokenHash = 'xxxx-4xxx-xxxx-xxxxxxxxxxxx';\n\nexport function buildUUID(): string {\n let hash = '';\n\n try {\n const crypto: Crypto = window.crypto;\n const randomValuesArray = new Uint16Array(31);\n crypto.getRandomValues(randomValuesArray);\n\n // Generate a strong UUID\n let i = 0;\n hash = tokenHash\n .replace(/[x]/g, (c: string): string => {\n const r = randomValuesArray[i] % 16;\n const v = c === 'x' ? r : (r & 0x3) | 0x8;\n i++;\n return v.toString(16);\n })\n .toUpperCase();\n } catch (err) {\n // crypto not available, generate weak UUID\n hash = tokenHash\n .replace(/[x]/g, (c: string): string => {\n const r = (Math.random() * 16) | 0;\n const v = c === 'x' ? r : (r & 0x3) | 0x8;\n return v.toString(16);\n })\n .toUpperCase();\n }\n\n return `${hexTime()}-${hash}`;\n}\n\nexport function hexTime(): string {\n // 32 bit representations of new Date().getTime() and performance.now()\n let dateNumber = 0;\n let perfNumber = 0;\n\n // Result of zero-fill right shift is always positive\n dateNumber = new Date().getTime() >>> 0;\n\n try {\n perfNumber = performance.now() >>> 0;\n } catch (err) {\n perfNumber = 0;\n }\n\n const output = Math.abs(dateNumber + perfNumber)\n .toString(16)\n .toLowerCase();\n\n // Ensure the output is exactly 8 characters\n return output.padStart(8, '0');\n}\n\nexport function getShopifyCookies(cookies: string): ShopifyCookies {\n const cookieData = parse(cookies);\n return {\n [SHOPIFY_Y]: cookieData[SHOPIFY_Y] || '',\n [SHOPIFY_S]: cookieData[SHOPIFY_S] || '',\n };\n}\n"],"names":["parse","SHOPIFY_Y","SHOPIFY_S"],"mappings":";;;;AAKA,MAAM,YAAY;AAEX,SAAS,YAAoB;AAClC,MAAI,OAAO;AAEX,MAAI;AACF,UAAM,SAAiB,OAAO;AAC9B,UAAM,oBAAoB,IAAI,YAAY,EAAE;AAC5C,WAAO,gBAAgB,iBAAiB;AAGxC,QAAI,IAAI;AACR,WAAO,UACJ,QAAQ,QAAQ,CAAC,MAAsB;AACtC,YAAM,IAAI,kBAAkB,CAAC,IAAI;AACjC,YAAM,IAAI,MAAM,MAAM,IAAK,IAAI,IAAO;AACtC;AACA,aAAO,EAAE,SAAS,EAAE;AAAA,IACtB,CAAC,EACA,YAAA;AAAA,EACL,SAAS,KAAK;AAEZ,WAAO,UACJ,QAAQ,QAAQ,CAAC,MAAsB;AACtC,YAAM,IAAK,KAAK,OAAA,IAAW,KAAM;AACjC,YAAM,IAAI,MAAM,MAAM,IAAK,IAAI,IAAO;AACtC,aAAO,EAAE,SAAS,EAAE;AAAA,IACtB,CAAC,EACA,YAAA;AAAA,EACL;AAEA,SAAO,GAAG,QAAA,CAAS,IAAI,IAAI;AAC7B;AAEO,SAAS,UAAkB;AAEhC,MAAI,aAAa;AACjB,MAAI,aAAa;AAGjB,gBAAa,oBAAI,QAAO,QAAA,MAAc;AAEtC,MAAI;AACF,iBAAa,YAAY,UAAU;AAAA,EACrC,SAAS,KAAK;AACZ,iBAAa;AAAA,EACf;AAEA,QAAM,SAAS,KAAK,IAAI,aAAa,UAAU,EAC5C,SAAS,EAAE,EACX,YAAA;AAGH,SAAO,OAAO,SAAS,GAAG,GAAG;AAC/B;AAEO,SAAS,kBAAkB,SAAiC;AACjE,QAAM,aAAaA,OAAAA,MAAM,OAAO;AAChC,SAAO;AAAA,IACL,CAACC,uBAAS,GAAG,WAAWA,cAAAA,SAAS,KAAK;AAAA,IACtC,CAACC,cAAAA,SAAS,GAAG,WAAWA,cAAAA,SAAS,KAAK;AAAA,EAAA;AAE1C;;;;"}
1
+ {"version":3,"file":"cookies-utils.js","sources":["../../src/cookies-utils.tsx"],"sourcesContent":["import {ShopifyCookies} from './analytics-types.js';\nimport {SHOPIFY_Y, SHOPIFY_S} from './cart-constants.js';\nimport {getTrackingValues} from './tracking-utils.js';\n\nconst tokenHash = 'xxxx-4xxx-xxxx-xxxxxxxxxxxx';\n\nexport function buildUUID(): string {\n let hash = '';\n\n try {\n const crypto: Crypto = window.crypto;\n const randomValuesArray = new Uint16Array(31);\n crypto.getRandomValues(randomValuesArray);\n\n // Generate a strong UUID\n let i = 0;\n hash = tokenHash\n .replace(/[x]/g, (c: string): string => {\n const r = randomValuesArray[i] % 16;\n const v = c === 'x' ? r : (r & 0x3) | 0x8;\n i++;\n return v.toString(16);\n })\n .toUpperCase();\n } catch (err) {\n // crypto not available, generate weak UUID\n hash = tokenHash\n .replace(/[x]/g, (c: string): string => {\n const r = (Math.random() * 16) | 0;\n const v = c === 'x' ? r : (r & 0x3) | 0x8;\n return v.toString(16);\n })\n .toUpperCase();\n }\n\n return `${hexTime()}-${hash}`;\n}\n\nexport function hexTime(): string {\n // 32 bit representations of new Date().getTime() and performance.now()\n let dateNumber = 0;\n let perfNumber = 0;\n\n // Result of zero-fill right shift is always positive\n dateNumber = new Date().getTime() >>> 0;\n\n try {\n perfNumber = performance.now() >>> 0;\n } catch (err) {\n perfNumber = 0;\n }\n\n const output = Math.abs(dateNumber + perfNumber)\n .toString(16)\n .toLowerCase();\n\n // Ensure the output is exactly 8 characters\n return output.padStart(8, '0');\n}\n\n/**\n * Gets the values of _shopify_y and _shopify_s cookies from the provided cookie string.\n * @deprecated Use getTrackingValues instead.\n */\nexport function getShopifyCookies(cookies: string): ShopifyCookies {\n // @ts-expect-error - Undeclared argument type\n const trackingValues = getTrackingValues(cookies);\n\n return {\n [SHOPIFY_Y]: trackingValues.uniqueToken,\n [SHOPIFY_S]: trackingValues.visitToken,\n };\n}\n"],"names":["getTrackingValues","SHOPIFY_Y","SHOPIFY_S"],"mappings":";;;;AAIA,MAAM,YAAY;AAEX,SAAS,YAAoB;AAClC,MAAI,OAAO;AAEX,MAAI;AACF,UAAM,SAAiB,OAAO;AAC9B,UAAM,oBAAoB,IAAI,YAAY,EAAE;AAC5C,WAAO,gBAAgB,iBAAiB;AAGxC,QAAI,IAAI;AACR,WAAO,UACJ,QAAQ,QAAQ,CAAC,MAAsB;AACtC,YAAM,IAAI,kBAAkB,CAAC,IAAI;AACjC,YAAM,IAAI,MAAM,MAAM,IAAK,IAAI,IAAO;AACtC;AACA,aAAO,EAAE,SAAS,EAAE;AAAA,IACtB,CAAC,EACA,YAAA;AAAA,EACL,SAAS,KAAK;AAEZ,WAAO,UACJ,QAAQ,QAAQ,CAAC,MAAsB;AACtC,YAAM,IAAK,KAAK,OAAA,IAAW,KAAM;AACjC,YAAM,IAAI,MAAM,MAAM,IAAK,IAAI,IAAO;AACtC,aAAO,EAAE,SAAS,EAAE;AAAA,IACtB,CAAC,EACA,YAAA;AAAA,EACL;AAEA,SAAO,GAAG,QAAA,CAAS,IAAI,IAAI;AAC7B;AAEO,SAAS,UAAkB;AAEhC,MAAI,aAAa;AACjB,MAAI,aAAa;AAGjB,gBAAa,oBAAI,QAAO,QAAA,MAAc;AAEtC,MAAI;AACF,iBAAa,YAAY,UAAU;AAAA,EACrC,SAAS,KAAK;AACZ,iBAAa;AAAA,EACf;AAEA,QAAM,SAAS,KAAK,IAAI,aAAa,UAAU,EAC5C,SAAS,EAAE,EACX,YAAA;AAGH,SAAO,OAAO,SAAS,GAAG,GAAG;AAC/B;AAMO,SAAS,kBAAkB,SAAiC;AAEjE,QAAM,iBAAiBA,cAAAA,kBAAkB,OAAO;AAEhD,SAAO;AAAA,IACL,CAACC,cAAAA,SAAS,GAAG,eAAe;AAAA,IAC5B,CAACC,cAAAA,SAAS,GAAG,eAAe;AAAA,EAAA;AAEhC;;;;"}
@@ -1,5 +1,5 @@
1
- import { parse } from "worktop/cookie";
2
1
  import { SHOPIFY_S, SHOPIFY_Y } from "./cart-constants.mjs";
2
+ import { getTrackingValues } from "./tracking-utils.mjs";
3
3
  const tokenHash = "xxxx-4xxx-xxxx-xxxxxxxxxxxx";
4
4
  function buildUUID() {
5
5
  let hash = "";
@@ -36,10 +36,10 @@ function hexTime() {
36
36
  return output.padStart(8, "0");
37
37
  }
38
38
  function getShopifyCookies(cookies) {
39
- const cookieData = parse(cookies);
39
+ const trackingValues = getTrackingValues(cookies);
40
40
  return {
41
- [SHOPIFY_Y]: cookieData[SHOPIFY_Y] || "",
42
- [SHOPIFY_S]: cookieData[SHOPIFY_S] || ""
41
+ [SHOPIFY_Y]: trackingValues.uniqueToken,
42
+ [SHOPIFY_S]: trackingValues.visitToken
43
43
  };
44
44
  }
45
45
  export {
@@ -1 +1 @@
1
- {"version":3,"file":"cookies-utils.mjs","sources":["../../src/cookies-utils.tsx"],"sourcesContent":["// @ts-ignore - worktop/cookie types not properly exported\nimport {parse} from 'worktop/cookie';\nimport {ShopifyCookies} from './analytics-types.js';\nimport {SHOPIFY_Y, SHOPIFY_S} from './cart-constants.js';\n\nconst tokenHash = 'xxxx-4xxx-xxxx-xxxxxxxxxxxx';\n\nexport function buildUUID(): string {\n let hash = '';\n\n try {\n const crypto: Crypto = window.crypto;\n const randomValuesArray = new Uint16Array(31);\n crypto.getRandomValues(randomValuesArray);\n\n // Generate a strong UUID\n let i = 0;\n hash = tokenHash\n .replace(/[x]/g, (c: string): string => {\n const r = randomValuesArray[i] % 16;\n const v = c === 'x' ? r : (r & 0x3) | 0x8;\n i++;\n return v.toString(16);\n })\n .toUpperCase();\n } catch (err) {\n // crypto not available, generate weak UUID\n hash = tokenHash\n .replace(/[x]/g, (c: string): string => {\n const r = (Math.random() * 16) | 0;\n const v = c === 'x' ? r : (r & 0x3) | 0x8;\n return v.toString(16);\n })\n .toUpperCase();\n }\n\n return `${hexTime()}-${hash}`;\n}\n\nexport function hexTime(): string {\n // 32 bit representations of new Date().getTime() and performance.now()\n let dateNumber = 0;\n let perfNumber = 0;\n\n // Result of zero-fill right shift is always positive\n dateNumber = new Date().getTime() >>> 0;\n\n try {\n perfNumber = performance.now() >>> 0;\n } catch (err) {\n perfNumber = 0;\n }\n\n const output = Math.abs(dateNumber + perfNumber)\n .toString(16)\n .toLowerCase();\n\n // Ensure the output is exactly 8 characters\n return output.padStart(8, '0');\n}\n\nexport function getShopifyCookies(cookies: string): ShopifyCookies {\n const cookieData = parse(cookies);\n return {\n [SHOPIFY_Y]: cookieData[SHOPIFY_Y] || '',\n [SHOPIFY_S]: cookieData[SHOPIFY_S] || '',\n };\n}\n"],"names":[],"mappings":";;AAKA,MAAM,YAAY;AAEX,SAAS,YAAoB;AAClC,MAAI,OAAO;AAEX,MAAI;AACF,UAAM,SAAiB,OAAO;AAC9B,UAAM,oBAAoB,IAAI,YAAY,EAAE;AAC5C,WAAO,gBAAgB,iBAAiB;AAGxC,QAAI,IAAI;AACR,WAAO,UACJ,QAAQ,QAAQ,CAAC,MAAsB;AACtC,YAAM,IAAI,kBAAkB,CAAC,IAAI;AACjC,YAAM,IAAI,MAAM,MAAM,IAAK,IAAI,IAAO;AACtC;AACA,aAAO,EAAE,SAAS,EAAE;AAAA,IACtB,CAAC,EACA,YAAA;AAAA,EACL,SAAS,KAAK;AAEZ,WAAO,UACJ,QAAQ,QAAQ,CAAC,MAAsB;AACtC,YAAM,IAAK,KAAK,OAAA,IAAW,KAAM;AACjC,YAAM,IAAI,MAAM,MAAM,IAAK,IAAI,IAAO;AACtC,aAAO,EAAE,SAAS,EAAE;AAAA,IACtB,CAAC,EACA,YAAA;AAAA,EACL;AAEA,SAAO,GAAG,QAAA,CAAS,IAAI,IAAI;AAC7B;AAEO,SAAS,UAAkB;AAEhC,MAAI,aAAa;AACjB,MAAI,aAAa;AAGjB,gBAAa,oBAAI,QAAO,QAAA,MAAc;AAEtC,MAAI;AACF,iBAAa,YAAY,UAAU;AAAA,EACrC,SAAS,KAAK;AACZ,iBAAa;AAAA,EACf;AAEA,QAAM,SAAS,KAAK,IAAI,aAAa,UAAU,EAC5C,SAAS,EAAE,EACX,YAAA;AAGH,SAAO,OAAO,SAAS,GAAG,GAAG;AAC/B;AAEO,SAAS,kBAAkB,SAAiC;AACjE,QAAM,aAAa,MAAM,OAAO;AAChC,SAAO;AAAA,IACL,CAAC,SAAS,GAAG,WAAW,SAAS,KAAK;AAAA,IACtC,CAAC,SAAS,GAAG,WAAW,SAAS,KAAK;AAAA,EAAA;AAE1C;"}
1
+ {"version":3,"file":"cookies-utils.mjs","sources":["../../src/cookies-utils.tsx"],"sourcesContent":["import {ShopifyCookies} from './analytics-types.js';\nimport {SHOPIFY_Y, SHOPIFY_S} from './cart-constants.js';\nimport {getTrackingValues} from './tracking-utils.js';\n\nconst tokenHash = 'xxxx-4xxx-xxxx-xxxxxxxxxxxx';\n\nexport function buildUUID(): string {\n let hash = '';\n\n try {\n const crypto: Crypto = window.crypto;\n const randomValuesArray = new Uint16Array(31);\n crypto.getRandomValues(randomValuesArray);\n\n // Generate a strong UUID\n let i = 0;\n hash = tokenHash\n .replace(/[x]/g, (c: string): string => {\n const r = randomValuesArray[i] % 16;\n const v = c === 'x' ? r : (r & 0x3) | 0x8;\n i++;\n return v.toString(16);\n })\n .toUpperCase();\n } catch (err) {\n // crypto not available, generate weak UUID\n hash = tokenHash\n .replace(/[x]/g, (c: string): string => {\n const r = (Math.random() * 16) | 0;\n const v = c === 'x' ? r : (r & 0x3) | 0x8;\n return v.toString(16);\n })\n .toUpperCase();\n }\n\n return `${hexTime()}-${hash}`;\n}\n\nexport function hexTime(): string {\n // 32 bit representations of new Date().getTime() and performance.now()\n let dateNumber = 0;\n let perfNumber = 0;\n\n // Result of zero-fill right shift is always positive\n dateNumber = new Date().getTime() >>> 0;\n\n try {\n perfNumber = performance.now() >>> 0;\n } catch (err) {\n perfNumber = 0;\n }\n\n const output = Math.abs(dateNumber + perfNumber)\n .toString(16)\n .toLowerCase();\n\n // Ensure the output is exactly 8 characters\n return output.padStart(8, '0');\n}\n\n/**\n * Gets the values of _shopify_y and _shopify_s cookies from the provided cookie string.\n * @deprecated Use getTrackingValues instead.\n */\nexport function getShopifyCookies(cookies: string): ShopifyCookies {\n // @ts-expect-error - Undeclared argument type\n const trackingValues = getTrackingValues(cookies);\n\n return {\n [SHOPIFY_Y]: trackingValues.uniqueToken,\n [SHOPIFY_S]: trackingValues.visitToken,\n };\n}\n"],"names":[],"mappings":";;AAIA,MAAM,YAAY;AAEX,SAAS,YAAoB;AAClC,MAAI,OAAO;AAEX,MAAI;AACF,UAAM,SAAiB,OAAO;AAC9B,UAAM,oBAAoB,IAAI,YAAY,EAAE;AAC5C,WAAO,gBAAgB,iBAAiB;AAGxC,QAAI,IAAI;AACR,WAAO,UACJ,QAAQ,QAAQ,CAAC,MAAsB;AACtC,YAAM,IAAI,kBAAkB,CAAC,IAAI;AACjC,YAAM,IAAI,MAAM,MAAM,IAAK,IAAI,IAAO;AACtC;AACA,aAAO,EAAE,SAAS,EAAE;AAAA,IACtB,CAAC,EACA,YAAA;AAAA,EACL,SAAS,KAAK;AAEZ,WAAO,UACJ,QAAQ,QAAQ,CAAC,MAAsB;AACtC,YAAM,IAAK,KAAK,OAAA,IAAW,KAAM;AACjC,YAAM,IAAI,MAAM,MAAM,IAAK,IAAI,IAAO;AACtC,aAAO,EAAE,SAAS,EAAE;AAAA,IACtB,CAAC,EACA,YAAA;AAAA,EACL;AAEA,SAAO,GAAG,QAAA,CAAS,IAAI,IAAI;AAC7B;AAEO,SAAS,UAAkB;AAEhC,MAAI,aAAa;AACjB,MAAI,aAAa;AAGjB,gBAAa,oBAAI,QAAO,QAAA,MAAc;AAEtC,MAAI;AACF,iBAAa,YAAY,UAAU;AAAA,EACrC,SAAS,KAAK;AACZ,iBAAa;AAAA,EACf;AAEA,QAAM,SAAS,KAAK,IAAI,aAAa,UAAU,EAC5C,SAAS,EAAE,EACX,YAAA;AAGH,SAAO,OAAO,SAAS,GAAG,GAAG;AAC/B;AAMO,SAAS,kBAAkB,SAAiC;AAEjE,QAAM,iBAAiB,kBAAkB,OAAO;AAEhD,SAAO;AAAA,IACL,CAAC,SAAS,GAAG,eAAe;AAAA,IAC5B,CAAC,SAAS,GAAG,eAAe;AAAA,EAAA;AAEhC;"}
@@ -30,6 +30,7 @@ const RichText = require("./RichText.js");
30
30
  const ShopifyProvider = require("./ShopifyProvider.js");
31
31
  const ShopPayButton = require("./ShopPayButton.js");
32
32
  const storefrontClient = require("./storefront-client.js");
33
+ const trackingUtils = require("./tracking-utils.js");
33
34
  const useMoney = require("./useMoney.js");
34
35
  const useSelectedOptionInUrlParam = require("./useSelectedOptionInUrlParam.js");
35
36
  const useShopifyCookies = require("./useShopifyCookies.js");
@@ -80,6 +81,9 @@ exports.ShopifyProvider = ShopifyProvider.ShopifyProvider;
80
81
  exports.useShop = ShopifyProvider.useShop;
81
82
  exports.ShopPayButton = ShopPayButton.ShopPayButton;
82
83
  exports.createStorefrontClient = storefrontClient.createStorefrontClient;
84
+ exports.SHOPIFY_UNIQUE_TOKEN_HEADER = trackingUtils.SHOPIFY_UNIQUE_TOKEN_HEADER;
85
+ exports.SHOPIFY_VISIT_TOKEN_HEADER = trackingUtils.SHOPIFY_VISIT_TOKEN_HEADER;
86
+ exports.getTrackingValues = trackingUtils.getTrackingValues;
83
87
  exports.useMoney = useMoney.useMoney;
84
88
  exports.useSelectedOptionInUrlParam = useSelectedOptionInUrlParam.useSelectedOptionInUrlParam;
85
89
  exports.useShopifyCookies = useShopifyCookies.useShopifyCookies;
@@ -1 +1 @@
1
- {"version":3,"file":"index.js","sources":[],"sourcesContent":[],"names":[],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;"}
1
+ {"version":3,"file":"index.js","sources":[],"sourcesContent":[],"names":[],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;"}
@@ -28,6 +28,7 @@ import { RichText } from "./RichText.mjs";
28
28
  import { ShopifyProvider, useShop } from "./ShopifyProvider.mjs";
29
29
  import { ShopPayButton } from "./ShopPayButton.mjs";
30
30
  import { createStorefrontClient } from "./storefront-client.mjs";
31
+ import { SHOPIFY_UNIQUE_TOKEN_HEADER, SHOPIFY_VISIT_TOKEN_HEADER, getTrackingValues } from "./tracking-utils.mjs";
31
32
  import { useMoney } from "./useMoney.mjs";
32
33
  import { useSelectedOptionInUrlParam } from "./useSelectedOptionInUrlParam.mjs";
33
34
  import { useShopifyCookies } from "./useShopifyCookies.mjs";
@@ -56,6 +57,8 @@ export {
56
57
  SHOPIFY_STOREFRONT_ID_HEADER,
57
58
  SHOPIFY_STOREFRONT_S_HEADER,
58
59
  SHOPIFY_STOREFRONT_Y_HEADER,
60
+ SHOPIFY_UNIQUE_TOKEN_HEADER,
61
+ SHOPIFY_VISIT_TOKEN_HEADER,
59
62
  SHOPIFY_Y,
60
63
  ShopPayButton,
61
64
  ShopifyProvider,
@@ -69,6 +72,7 @@ export {
69
72
  getClientBrowserParameters,
70
73
  getProductOptions,
71
74
  getShopifyCookies,
75
+ getTrackingValues,
72
76
  isOptionValueCombinationInEncodedVariant,
73
77
  mapSelectedProductOptionToObject,
74
78
  parseGid,
@@ -1 +1 @@
1
- {"version":3,"file":"index.mjs","sources":[],"sourcesContent":[],"names":[],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;"}
1
+ {"version":3,"file":"index.mjs","sources":[],"sourcesContent":[],"names":[],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;"}
@@ -1,5 +1,5 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, Symbol.toStringTag, { value: "Module" });
3
- const version = "2025.7.0";
3
+ const version = "2025.7.1";
4
4
  exports.version = version;
5
5
  //# sourceMappingURL=package.json.js.map
@@ -1,4 +1,4 @@
1
- const version = "2025.7.0";
1
+ const version = "2025.7.1";
2
2
  export {
3
3
  version
4
4
  };
@@ -0,0 +1,92 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, Symbol.toStringTag, { value: "Module" });
3
+ const SHOPIFY_VISIT_TOKEN_HEADER = "X-Shopify-VisitToken";
4
+ const SHOPIFY_UNIQUE_TOKEN_HEADER = "X-Shopify-UniqueToken";
5
+ const cachedTrackingValues = { current: null };
6
+ function getTrackingValues() {
7
+ var _a, _b, _c;
8
+ let trackingValues;
9
+ if (typeof window !== "undefined" && typeof window.performance !== "undefined") {
10
+ try {
11
+ const resourceRE = /^https?:\/\/([^/]+)(\/api\/(?:unstable|2\d{3}-\d{2})\/graphql\.json(?=$|\?))?/;
12
+ const entries = performance.getEntriesByType(
13
+ "resource"
14
+ );
15
+ let matchedValues;
16
+ for (let i = entries.length - 1; i >= 0; i--) {
17
+ const entry = entries[i];
18
+ if (entry.initiatorType !== "fetch") continue;
19
+ const currentHost = window.location.host;
20
+ const match = entry.name.match(resourceRE);
21
+ if (!match) continue;
22
+ const [, matchedHost, sfapiPath] = match;
23
+ const isMatch = (
24
+ // Same origin (exact host match)
25
+ matchedHost === currentHost || // Subdomain with SFAPI path
26
+ sfapiPath && (matchedHost == null ? void 0 : matchedHost.endsWith(`.${currentHost}`))
27
+ );
28
+ if (isMatch) {
29
+ const values = extractFromPerformanceEntry(entry);
30
+ if (values) {
31
+ matchedValues = values;
32
+ break;
33
+ }
34
+ }
35
+ }
36
+ if (matchedValues) {
37
+ trackingValues = matchedValues;
38
+ }
39
+ if (trackingValues) {
40
+ cachedTrackingValues.current = trackingValues;
41
+ } else if (cachedTrackingValues.current) {
42
+ trackingValues = cachedTrackingValues.current;
43
+ }
44
+ if (!trackingValues) {
45
+ const navigationEntries = performance.getEntriesByType(
46
+ "navigation"
47
+ )[0];
48
+ trackingValues = extractFromPerformanceEntry(navigationEntries, false);
49
+ }
50
+ } catch {
51
+ }
52
+ }
53
+ if (!trackingValues) {
54
+ const cookie = (
55
+ // Read from arguments to avoid declaring parameters in this function signature.
56
+ // This logic is only used internally from `getShopifyCookies` and will be deprecated.
57
+ typeof arguments[0] === "string" ? arguments[0] : typeof document !== "undefined" ? document.cookie : ""
58
+ );
59
+ trackingValues = {
60
+ uniqueToken: ((_a = cookie.match(/\b_shopify_y=([^;]+)/)) == null ? void 0 : _a[1]) || "",
61
+ visitToken: ((_b = cookie.match(/\b_shopify_s=([^;]+)/)) == null ? void 0 : _b[1]) || "",
62
+ consent: ((_c = cookie.match(/\b_tracking_consent=([^;]+)/)) == null ? void 0 : _c[1]) || ""
63
+ };
64
+ }
65
+ return trackingValues;
66
+ }
67
+ function extractFromPerformanceEntry(entry, isConsentRequired = true) {
68
+ let uniqueToken = "";
69
+ let visitToken = "";
70
+ let consent = "";
71
+ const serverTiming = entry.serverTiming;
72
+ if (serverTiming && serverTiming.length >= 3) {
73
+ for (let i = serverTiming.length - 1; i >= 0; i--) {
74
+ const { name, description } = serverTiming[i];
75
+ if (!name || !description) continue;
76
+ if (name === "_y") {
77
+ uniqueToken = description;
78
+ } else if (name === "_s") {
79
+ visitToken = description;
80
+ } else if (name === "_cmp") {
81
+ consent = description;
82
+ }
83
+ if (uniqueToken && visitToken && consent) break;
84
+ }
85
+ }
86
+ return uniqueToken && visitToken && (isConsentRequired ? consent : true) ? { uniqueToken, visitToken, consent } : void 0;
87
+ }
88
+ exports.SHOPIFY_UNIQUE_TOKEN_HEADER = SHOPIFY_UNIQUE_TOKEN_HEADER;
89
+ exports.SHOPIFY_VISIT_TOKEN_HEADER = SHOPIFY_VISIT_TOKEN_HEADER;
90
+ exports.cachedTrackingValues = cachedTrackingValues;
91
+ exports.getTrackingValues = getTrackingValues;
92
+ //# sourceMappingURL=tracking-utils.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"tracking-utils.js","sources":["../../src/tracking-utils.ts"],"sourcesContent":["/** Storefront API header for VisitToken */\nexport const SHOPIFY_VISIT_TOKEN_HEADER = 'X-Shopify-VisitToken';\n/** Storefront API header for UniqueToken */\nexport const SHOPIFY_UNIQUE_TOKEN_HEADER = 'X-Shopify-UniqueToken';\n\ntype TrackingValues = {\n /** Identifier for the unique user. Equivalent to the deprecated _shopify_y cookie */\n uniqueToken: string;\n /** Identifier for the current visit. Equivalent to the deprecated _shopify_s cookie */\n visitToken: string;\n /** Represents the consent given by the user or the default region consent configured in Admin */\n consent: string;\n};\n\n// Cache values to avoid losing them when performance\n// entries are cleared from the buffer over time.\nexport const cachedTrackingValues: {\n current: null | TrackingValues;\n} = {current: null};\n\n/**\n * Retrieves user session tracking values for analytics\n * and marketing from the browser environment.\n */\nexport function getTrackingValues(): TrackingValues {\n // Overall behavior: Tracking values are returned in Server-Timing headers from\n // Storefront API responses, and we want to find and return these tracking values.\n //\n // Search recent fetches for SFAPI requests matching either: same origin (proxy case)\n // or a subdomain of the current host (eg: checkout subdomain, if there is no proxy).\n // We consider SF API-like endpoints (/api/.../graphql.json) on subdomains, as well as\n // any same-origin request. The reason for the latter is that Hydrogen server collects\n // tracking values and returns them in any non-cached response, not just direct SF API\n // responses. For example, a cart mutation in a server action could return tracking values.\n //\n // If we didn't find tracking values in fetch requests, we fall back to checking cached values,\n // then the initial page navigation entry, and finally the deprecated `_shopify_s` and `_shopify_y`.\n\n let trackingValues: TrackingValues | undefined;\n\n if (\n typeof window !== 'undefined' &&\n typeof window.performance !== 'undefined'\n ) {\n try {\n // RE to extract host and optionally match SFAPI pathname.\n // Group 1: host (e.g. \"checkout.mystore.com\")\n // Group 2: SFAPI path if present (e.g. \"/api/2024-01/graphql.json\")\n const resourceRE =\n /^https?:\\/\\/([^/]+)(\\/api\\/(?:unstable|2\\d{3}-\\d{2})\\/graphql\\.json(?=$|\\?))?/;\n\n // Search backwards through resource entries to find the most recent match.\n // Match criteria (first one with _y and _s values wins):\n // - Same origin (exact host match) with tracking values, OR\n // - Subdomain + SFAPI pathname with tracking values\n const entries = performance.getEntriesByType(\n 'resource',\n ) as PerformanceResourceTiming[];\n\n let matchedValues: ReturnType<typeof extractFromPerformanceEntry>;\n\n for (let i = entries.length - 1; i >= 0; i--) {\n const entry = entries[i];\n\n if (entry.initiatorType !== 'fetch') continue;\n\n const currentHost = window.location.host;\n const match = entry.name.match(resourceRE);\n if (!match) continue;\n\n const [, matchedHost, sfapiPath] = match;\n\n const isMatch =\n // Same origin (exact host match)\n matchedHost === currentHost ||\n // Subdomain with SFAPI path\n (sfapiPath && matchedHost?.endsWith(`.${currentHost}`));\n\n if (isMatch) {\n const values = extractFromPerformanceEntry(entry);\n if (values) {\n matchedValues = values;\n break;\n }\n }\n }\n\n if (matchedValues) {\n trackingValues = matchedValues;\n }\n\n // Resource entries have a limited buffer and are removed over time.\n // Cache the latest values for future calls if we find them.\n // A cached resource entry is always newer than a navigation entry.\n if (trackingValues) {\n cachedTrackingValues.current = trackingValues;\n } else if (cachedTrackingValues.current) {\n // Fallback to cached values from previous calls:\n trackingValues = cachedTrackingValues.current;\n }\n\n if (!trackingValues) {\n // Fallback to navigation entry from full page rendering load:\n const navigationEntries = performance.getEntriesByType(\n 'navigation',\n )[0] as PerformanceNavigationTiming;\n\n // Navigation entries might omit consent when the Hydrogen server generates it.\n // In this case, we skip consent requirement and only extract _y and _s values.\n trackingValues = extractFromPerformanceEntry(navigationEntries, false);\n }\n } catch {}\n }\n\n // Fallback to deprecated cookies to support transitioning:\n if (!trackingValues) {\n const cookie =\n // Read from arguments to avoid declaring parameters in this function signature.\n // This logic is only used internally from `getShopifyCookies` and will be deprecated.\n typeof arguments[0] === 'string'\n ? arguments[0]\n : typeof document !== 'undefined'\n ? document.cookie\n : '';\n\n trackingValues = {\n uniqueToken: cookie.match(/\\b_shopify_y=([^;]+)/)?.[1] || '',\n visitToken: cookie.match(/\\b_shopify_s=([^;]+)/)?.[1] || '',\n consent: cookie.match(/\\b_tracking_consent=([^;]+)/)?.[1] || '',\n };\n }\n\n return trackingValues;\n}\n\nfunction extractFromPerformanceEntry(\n entry: PerformanceNavigationTiming | PerformanceResourceTiming,\n isConsentRequired = true,\n): TrackingValues | undefined {\n let uniqueToken = '';\n let visitToken = '';\n let consent = '';\n\n const serverTiming = entry.serverTiming;\n // Quick check: we need at least 3 entries (_y, _s, _cmp)\n if (serverTiming && serverTiming.length >= 3) {\n // Iterate backwards since our headers are typically at the end\n for (let i = serverTiming.length - 1; i >= 0; i--) {\n const {name, description} = serverTiming[i];\n if (!name || !description) continue;\n\n if (name === '_y') {\n uniqueToken = description;\n } else if (name === '_s') {\n visitToken = description;\n } else if (name === '_cmp') {\n // _cmp (consent management platform) holds the consent value\n // used by consent-tracking-api and privacy-banner scripts.\n consent = description;\n }\n\n if (uniqueToken && visitToken && consent) break;\n }\n }\n\n return uniqueToken && visitToken && (isConsentRequired ? consent : true)\n ? {uniqueToken, visitToken, consent}\n : undefined;\n}\n"],"names":[],"mappings":";;AACO,MAAM,6BAA6B;AAEnC,MAAM,8BAA8B;AAapC,MAAM,uBAET,EAAC,SAAS,KAAA;AAMP,SAAS,oBAAoC;;AAclD,MAAI;AAEJ,MACE,OAAO,WAAW,eAClB,OAAO,OAAO,gBAAgB,aAC9B;AACA,QAAI;AAIF,YAAM,aACJ;AAMF,YAAM,UAAU,YAAY;AAAA,QAC1B;AAAA,MAAA;AAGF,UAAI;AAEJ,eAAS,IAAI,QAAQ,SAAS,GAAG,KAAK,GAAG,KAAK;AAC5C,cAAM,QAAQ,QAAQ,CAAC;AAEvB,YAAI,MAAM,kBAAkB,QAAS;AAErC,cAAM,cAAc,OAAO,SAAS;AACpC,cAAM,QAAQ,MAAM,KAAK,MAAM,UAAU;AACzC,YAAI,CAAC,MAAO;AAEZ,cAAM,CAAA,EAAG,aAAa,SAAS,IAAI;AAEnC,cAAM;AAAA;AAAA,UAEJ,gBAAgB;AAAA,UAEf,cAAa,2CAAa,SAAS,IAAI,WAAW;AAAA;AAErD,YAAI,SAAS;AACX,gBAAM,SAAS,4BAA4B,KAAK;AAChD,cAAI,QAAQ;AACV,4BAAgB;AAChB;AAAA,UACF;AAAA,QACF;AAAA,MACF;AAEA,UAAI,eAAe;AACjB,yBAAiB;AAAA,MACnB;AAKA,UAAI,gBAAgB;AAClB,6BAAqB,UAAU;AAAA,MACjC,WAAW,qBAAqB,SAAS;AAEvC,yBAAiB,qBAAqB;AAAA,MACxC;AAEA,UAAI,CAAC,gBAAgB;AAEnB,cAAM,oBAAoB,YAAY;AAAA,UACpC;AAAA,QAAA,EACA,CAAC;AAIH,yBAAiB,4BAA4B,mBAAmB,KAAK;AAAA,MACvE;AAAA,IACF,QAAQ;AAAA,IAAC;AAAA,EACX;AAGA,MAAI,CAAC,gBAAgB;AACnB,UAAM;AAAA;AAAA;AAAA,MAGJ,OAAO,UAAU,CAAC,MAAM,WACpB,UAAU,CAAC,IACX,OAAO,aAAa,cAClB,SAAS,SACT;AAAA;AAER,qBAAiB;AAAA,MACf,eAAa,YAAO,MAAM,sBAAsB,MAAnC,mBAAuC,OAAM;AAAA,MAC1D,cAAY,YAAO,MAAM,sBAAsB,MAAnC,mBAAuC,OAAM;AAAA,MACzD,WAAS,YAAO,MAAM,6BAA6B,MAA1C,mBAA8C,OAAM;AAAA,IAAA;AAAA,EAEjE;AAEA,SAAO;AACT;AAEA,SAAS,4BACP,OACA,oBAAoB,MACQ;AAC5B,MAAI,cAAc;AAClB,MAAI,aAAa;AACjB,MAAI,UAAU;AAEd,QAAM,eAAe,MAAM;AAE3B,MAAI,gBAAgB,aAAa,UAAU,GAAG;AAE5C,aAAS,IAAI,aAAa,SAAS,GAAG,KAAK,GAAG,KAAK;AACjD,YAAM,EAAC,MAAM,gBAAe,aAAa,CAAC;AAC1C,UAAI,CAAC,QAAQ,CAAC,YAAa;AAE3B,UAAI,SAAS,MAAM;AACjB,sBAAc;AAAA,MAChB,WAAW,SAAS,MAAM;AACxB,qBAAa;AAAA,MACf,WAAW,SAAS,QAAQ;AAG1B,kBAAU;AAAA,MACZ;AAEA,UAAI,eAAe,cAAc,QAAS;AAAA,IAC5C;AAAA,EACF;AAEA,SAAO,eAAe,eAAe,oBAAoB,UAAU,QAC/D,EAAC,aAAa,YAAY,QAAA,IAC1B;AACN;;;;;"}
@@ -0,0 +1,92 @@
1
+ const SHOPIFY_VISIT_TOKEN_HEADER = "X-Shopify-VisitToken";
2
+ const SHOPIFY_UNIQUE_TOKEN_HEADER = "X-Shopify-UniqueToken";
3
+ const cachedTrackingValues = { current: null };
4
+ function getTrackingValues() {
5
+ var _a, _b, _c;
6
+ let trackingValues;
7
+ if (typeof window !== "undefined" && typeof window.performance !== "undefined") {
8
+ try {
9
+ const resourceRE = /^https?:\/\/([^/]+)(\/api\/(?:unstable|2\d{3}-\d{2})\/graphql\.json(?=$|\?))?/;
10
+ const entries = performance.getEntriesByType(
11
+ "resource"
12
+ );
13
+ let matchedValues;
14
+ for (let i = entries.length - 1; i >= 0; i--) {
15
+ const entry = entries[i];
16
+ if (entry.initiatorType !== "fetch") continue;
17
+ const currentHost = window.location.host;
18
+ const match = entry.name.match(resourceRE);
19
+ if (!match) continue;
20
+ const [, matchedHost, sfapiPath] = match;
21
+ const isMatch = (
22
+ // Same origin (exact host match)
23
+ matchedHost === currentHost || // Subdomain with SFAPI path
24
+ sfapiPath && (matchedHost == null ? void 0 : matchedHost.endsWith(`.${currentHost}`))
25
+ );
26
+ if (isMatch) {
27
+ const values = extractFromPerformanceEntry(entry);
28
+ if (values) {
29
+ matchedValues = values;
30
+ break;
31
+ }
32
+ }
33
+ }
34
+ if (matchedValues) {
35
+ trackingValues = matchedValues;
36
+ }
37
+ if (trackingValues) {
38
+ cachedTrackingValues.current = trackingValues;
39
+ } else if (cachedTrackingValues.current) {
40
+ trackingValues = cachedTrackingValues.current;
41
+ }
42
+ if (!trackingValues) {
43
+ const navigationEntries = performance.getEntriesByType(
44
+ "navigation"
45
+ )[0];
46
+ trackingValues = extractFromPerformanceEntry(navigationEntries, false);
47
+ }
48
+ } catch {
49
+ }
50
+ }
51
+ if (!trackingValues) {
52
+ const cookie = (
53
+ // Read from arguments to avoid declaring parameters in this function signature.
54
+ // This logic is only used internally from `getShopifyCookies` and will be deprecated.
55
+ typeof arguments[0] === "string" ? arguments[0] : typeof document !== "undefined" ? document.cookie : ""
56
+ );
57
+ trackingValues = {
58
+ uniqueToken: ((_a = cookie.match(/\b_shopify_y=([^;]+)/)) == null ? void 0 : _a[1]) || "",
59
+ visitToken: ((_b = cookie.match(/\b_shopify_s=([^;]+)/)) == null ? void 0 : _b[1]) || "",
60
+ consent: ((_c = cookie.match(/\b_tracking_consent=([^;]+)/)) == null ? void 0 : _c[1]) || ""
61
+ };
62
+ }
63
+ return trackingValues;
64
+ }
65
+ function extractFromPerformanceEntry(entry, isConsentRequired = true) {
66
+ let uniqueToken = "";
67
+ let visitToken = "";
68
+ let consent = "";
69
+ const serverTiming = entry.serverTiming;
70
+ if (serverTiming && serverTiming.length >= 3) {
71
+ for (let i = serverTiming.length - 1; i >= 0; i--) {
72
+ const { name, description } = serverTiming[i];
73
+ if (!name || !description) continue;
74
+ if (name === "_y") {
75
+ uniqueToken = description;
76
+ } else if (name === "_s") {
77
+ visitToken = description;
78
+ } else if (name === "_cmp") {
79
+ consent = description;
80
+ }
81
+ if (uniqueToken && visitToken && consent) break;
82
+ }
83
+ }
84
+ return uniqueToken && visitToken && (isConsentRequired ? consent : true) ? { uniqueToken, visitToken, consent } : void 0;
85
+ }
86
+ export {
87
+ SHOPIFY_UNIQUE_TOKEN_HEADER,
88
+ SHOPIFY_VISIT_TOKEN_HEADER,
89
+ cachedTrackingValues,
90
+ getTrackingValues
91
+ };
92
+ //# sourceMappingURL=tracking-utils.mjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"tracking-utils.mjs","sources":["../../src/tracking-utils.ts"],"sourcesContent":["/** Storefront API header for VisitToken */\nexport const SHOPIFY_VISIT_TOKEN_HEADER = 'X-Shopify-VisitToken';\n/** Storefront API header for UniqueToken */\nexport const SHOPIFY_UNIQUE_TOKEN_HEADER = 'X-Shopify-UniqueToken';\n\ntype TrackingValues = {\n /** Identifier for the unique user. Equivalent to the deprecated _shopify_y cookie */\n uniqueToken: string;\n /** Identifier for the current visit. Equivalent to the deprecated _shopify_s cookie */\n visitToken: string;\n /** Represents the consent given by the user or the default region consent configured in Admin */\n consent: string;\n};\n\n// Cache values to avoid losing them when performance\n// entries are cleared from the buffer over time.\nexport const cachedTrackingValues: {\n current: null | TrackingValues;\n} = {current: null};\n\n/**\n * Retrieves user session tracking values for analytics\n * and marketing from the browser environment.\n */\nexport function getTrackingValues(): TrackingValues {\n // Overall behavior: Tracking values are returned in Server-Timing headers from\n // Storefront API responses, and we want to find and return these tracking values.\n //\n // Search recent fetches for SFAPI requests matching either: same origin (proxy case)\n // or a subdomain of the current host (eg: checkout subdomain, if there is no proxy).\n // We consider SF API-like endpoints (/api/.../graphql.json) on subdomains, as well as\n // any same-origin request. The reason for the latter is that Hydrogen server collects\n // tracking values and returns them in any non-cached response, not just direct SF API\n // responses. For example, a cart mutation in a server action could return tracking values.\n //\n // If we didn't find tracking values in fetch requests, we fall back to checking cached values,\n // then the initial page navigation entry, and finally the deprecated `_shopify_s` and `_shopify_y`.\n\n let trackingValues: TrackingValues | undefined;\n\n if (\n typeof window !== 'undefined' &&\n typeof window.performance !== 'undefined'\n ) {\n try {\n // RE to extract host and optionally match SFAPI pathname.\n // Group 1: host (e.g. \"checkout.mystore.com\")\n // Group 2: SFAPI path if present (e.g. \"/api/2024-01/graphql.json\")\n const resourceRE =\n /^https?:\\/\\/([^/]+)(\\/api\\/(?:unstable|2\\d{3}-\\d{2})\\/graphql\\.json(?=$|\\?))?/;\n\n // Search backwards through resource entries to find the most recent match.\n // Match criteria (first one with _y and _s values wins):\n // - Same origin (exact host match) with tracking values, OR\n // - Subdomain + SFAPI pathname with tracking values\n const entries = performance.getEntriesByType(\n 'resource',\n ) as PerformanceResourceTiming[];\n\n let matchedValues: ReturnType<typeof extractFromPerformanceEntry>;\n\n for (let i = entries.length - 1; i >= 0; i--) {\n const entry = entries[i];\n\n if (entry.initiatorType !== 'fetch') continue;\n\n const currentHost = window.location.host;\n const match = entry.name.match(resourceRE);\n if (!match) continue;\n\n const [, matchedHost, sfapiPath] = match;\n\n const isMatch =\n // Same origin (exact host match)\n matchedHost === currentHost ||\n // Subdomain with SFAPI path\n (sfapiPath && matchedHost?.endsWith(`.${currentHost}`));\n\n if (isMatch) {\n const values = extractFromPerformanceEntry(entry);\n if (values) {\n matchedValues = values;\n break;\n }\n }\n }\n\n if (matchedValues) {\n trackingValues = matchedValues;\n }\n\n // Resource entries have a limited buffer and are removed over time.\n // Cache the latest values for future calls if we find them.\n // A cached resource entry is always newer than a navigation entry.\n if (trackingValues) {\n cachedTrackingValues.current = trackingValues;\n } else if (cachedTrackingValues.current) {\n // Fallback to cached values from previous calls:\n trackingValues = cachedTrackingValues.current;\n }\n\n if (!trackingValues) {\n // Fallback to navigation entry from full page rendering load:\n const navigationEntries = performance.getEntriesByType(\n 'navigation',\n )[0] as PerformanceNavigationTiming;\n\n // Navigation entries might omit consent when the Hydrogen server generates it.\n // In this case, we skip consent requirement and only extract _y and _s values.\n trackingValues = extractFromPerformanceEntry(navigationEntries, false);\n }\n } catch {}\n }\n\n // Fallback to deprecated cookies to support transitioning:\n if (!trackingValues) {\n const cookie =\n // Read from arguments to avoid declaring parameters in this function signature.\n // This logic is only used internally from `getShopifyCookies` and will be deprecated.\n typeof arguments[0] === 'string'\n ? arguments[0]\n : typeof document !== 'undefined'\n ? document.cookie\n : '';\n\n trackingValues = {\n uniqueToken: cookie.match(/\\b_shopify_y=([^;]+)/)?.[1] || '',\n visitToken: cookie.match(/\\b_shopify_s=([^;]+)/)?.[1] || '',\n consent: cookie.match(/\\b_tracking_consent=([^;]+)/)?.[1] || '',\n };\n }\n\n return trackingValues;\n}\n\nfunction extractFromPerformanceEntry(\n entry: PerformanceNavigationTiming | PerformanceResourceTiming,\n isConsentRequired = true,\n): TrackingValues | undefined {\n let uniqueToken = '';\n let visitToken = '';\n let consent = '';\n\n const serverTiming = entry.serverTiming;\n // Quick check: we need at least 3 entries (_y, _s, _cmp)\n if (serverTiming && serverTiming.length >= 3) {\n // Iterate backwards since our headers are typically at the end\n for (let i = serverTiming.length - 1; i >= 0; i--) {\n const {name, description} = serverTiming[i];\n if (!name || !description) continue;\n\n if (name === '_y') {\n uniqueToken = description;\n } else if (name === '_s') {\n visitToken = description;\n } else if (name === '_cmp') {\n // _cmp (consent management platform) holds the consent value\n // used by consent-tracking-api and privacy-banner scripts.\n consent = description;\n }\n\n if (uniqueToken && visitToken && consent) break;\n }\n }\n\n return uniqueToken && visitToken && (isConsentRequired ? consent : true)\n ? {uniqueToken, visitToken, consent}\n : undefined;\n}\n"],"names":[],"mappings":"AACO,MAAM,6BAA6B;AAEnC,MAAM,8BAA8B;AAapC,MAAM,uBAET,EAAC,SAAS,KAAA;AAMP,SAAS,oBAAoC;AAvB7C;AAqCL,MAAI;AAEJ,MACE,OAAO,WAAW,eAClB,OAAO,OAAO,gBAAgB,aAC9B;AACA,QAAI;AAIF,YAAM,aACJ;AAMF,YAAM,UAAU,YAAY;AAAA,QAC1B;AAAA,MAAA;AAGF,UAAI;AAEJ,eAAS,IAAI,QAAQ,SAAS,GAAG,KAAK,GAAG,KAAK;AAC5C,cAAM,QAAQ,QAAQ,CAAC;AAEvB,YAAI,MAAM,kBAAkB,QAAS;AAErC,cAAM,cAAc,OAAO,SAAS;AACpC,cAAM,QAAQ,MAAM,KAAK,MAAM,UAAU;AACzC,YAAI,CAAC,MAAO;AAEZ,cAAM,CAAA,EAAG,aAAa,SAAS,IAAI;AAEnC,cAAM;AAAA;AAAA,UAEJ,gBAAgB;AAAA,UAEf,cAAa,2CAAa,SAAS,IAAI,WAAW;AAAA;AAErD,YAAI,SAAS;AACX,gBAAM,SAAS,4BAA4B,KAAK;AAChD,cAAI,QAAQ;AACV,4BAAgB;AAChB;AAAA,UACF;AAAA,QACF;AAAA,MACF;AAEA,UAAI,eAAe;AACjB,yBAAiB;AAAA,MACnB;AAKA,UAAI,gBAAgB;AAClB,6BAAqB,UAAU;AAAA,MACjC,WAAW,qBAAqB,SAAS;AAEvC,yBAAiB,qBAAqB;AAAA,MACxC;AAEA,UAAI,CAAC,gBAAgB;AAEnB,cAAM,oBAAoB,YAAY;AAAA,UACpC;AAAA,QAAA,EACA,CAAC;AAIH,yBAAiB,4BAA4B,mBAAmB,KAAK;AAAA,MACvE;AAAA,IACF,QAAQ;AAAA,IAAC;AAAA,EACX;AAGA,MAAI,CAAC,gBAAgB;AACnB,UAAM;AAAA;AAAA;AAAA,MAGJ,OAAO,UAAU,CAAC,MAAM,WACpB,UAAU,CAAC,IACX,OAAO,aAAa,cAClB,SAAS,SACT;AAAA;AAER,qBAAiB;AAAA,MACf,eAAa,YAAO,MAAM,sBAAsB,MAAnC,mBAAuC,OAAM;AAAA,MAC1D,cAAY,YAAO,MAAM,sBAAsB,MAAnC,mBAAuC,OAAM;AAAA,MACzD,WAAS,YAAO,MAAM,6BAA6B,MAA1C,mBAA8C,OAAM;AAAA,IAAA;AAAA,EAEjE;AAEA,SAAO;AACT;AAEA,SAAS,4BACP,OACA,oBAAoB,MACQ;AAC5B,MAAI,cAAc;AAClB,MAAI,aAAa;AACjB,MAAI,UAAU;AAEd,QAAM,eAAe,MAAM;AAE3B,MAAI,gBAAgB,aAAa,UAAU,GAAG;AAE5C,aAAS,IAAI,aAAa,SAAS,GAAG,KAAK,GAAG,KAAK;AACjD,YAAM,EAAC,MAAM,gBAAe,aAAa,CAAC;AAC1C,UAAI,CAAC,QAAQ,CAAC,YAAa;AAE3B,UAAI,SAAS,MAAM;AACjB,sBAAc;AAAA,MAChB,WAAW,SAAS,MAAM;AACxB,qBAAa;AAAA,MACf,WAAW,SAAS,QAAQ;AAG1B,kBAAU;AAAA,MACZ;AAEA,UAAI,eAAe,cAAc,QAAS;AAAA,IAC5C;AAAA,EACF;AAEA,SAAO,eAAe,eAAe,oBAAoB,UAAU,QAC/D,EAAC,aAAa,YAAY,QAAA,IAC1B;AACN;"}
@@ -4,17 +4,26 @@ const React = require("react");
4
4
  const cookie = require("worktop/cookie");
5
5
  const cartConstants = require("./cart-constants.js");
6
6
  const cookiesUtils = require("./cookies-utils.js");
7
+ const trackingUtils = require("./tracking-utils.js");
7
8
  const longTermLength = 60 * 60 * 24 * 360 * 1;
8
9
  const shortTermLength = 60 * 30;
9
10
  function useShopifyCookies(options) {
10
11
  const {
11
- hasUserConsent = false,
12
+ hasUserConsent,
12
13
  domain = "",
13
- checkoutDomain = ""
14
+ checkoutDomain = "",
15
+ storefrontAccessToken,
16
+ fetchTrackingValues,
17
+ ignoreDeprecatedCookies = false
14
18
  } = options || {};
19
+ const coreCookiesReady = useCoreShopifyCookies({
20
+ storefrontAccessToken,
21
+ fetchTrackingValues,
22
+ checkoutDomain
23
+ });
15
24
  React.useEffect(() => {
16
- const cookies = cookiesUtils.getShopifyCookies(document.cookie);
17
- let currentDomain = domain || window.document.location.host;
25
+ if (ignoreDeprecatedCookies || !coreCookiesReady) return;
26
+ let currentDomain = domain || window.location.host;
18
27
  if (checkoutDomain) {
19
28
  const checkoutDomainParts = checkoutDomain.split(".").reverse();
20
29
  const currentDomainParts = currentDomain.split(".").reverse();
@@ -29,15 +38,19 @@ function useShopifyCookies(options) {
29
38
  if (/^localhost/.test(currentDomain)) currentDomain = "";
30
39
  const domainWithLeadingDot = currentDomain ? /^\./.test(currentDomain) ? currentDomain : `.${currentDomain}` : "";
31
40
  if (hasUserConsent) {
41
+ const trackingValues = trackingUtils.getTrackingValues();
42
+ if ((trackingValues.uniqueToken || trackingValues.visitToken || "").startsWith("00000000-")) {
43
+ return;
44
+ }
32
45
  setCookie(
33
46
  cartConstants.SHOPIFY_Y,
34
- cookies[cartConstants.SHOPIFY_Y] || cookiesUtils.buildUUID(),
47
+ trackingValues.uniqueToken || cookiesUtils.buildUUID(),
35
48
  longTermLength,
36
49
  domainWithLeadingDot
37
50
  );
38
51
  setCookie(
39
52
  cartConstants.SHOPIFY_S,
40
- cookies[cartConstants.SHOPIFY_S] || cookiesUtils.buildUUID(),
53
+ trackingValues.visitToken || cookiesUtils.buildUUID(),
41
54
  shortTermLength,
42
55
  domainWithLeadingDot
43
56
  );
@@ -45,7 +58,14 @@ function useShopifyCookies(options) {
45
58
  setCookie(cartConstants.SHOPIFY_Y, "", 0, domainWithLeadingDot);
46
59
  setCookie(cartConstants.SHOPIFY_S, "", 0, domainWithLeadingDot);
47
60
  }
48
- }, [options, hasUserConsent, domain, checkoutDomain]);
61
+ }, [
62
+ coreCookiesReady,
63
+ hasUserConsent,
64
+ domain,
65
+ checkoutDomain,
66
+ ignoreDeprecatedCookies
67
+ ]);
68
+ return coreCookiesReady;
49
69
  }
50
70
  function setCookie(name, value, maxage, domain) {
51
71
  document.cookie = cookie.stringify(name, value, {
@@ -55,5 +75,72 @@ function setCookie(name, value, maxage, domain) {
55
75
  path: "/"
56
76
  });
57
77
  }
78
+ async function fetchTrackingValuesFromBrowser(storefrontAccessToken, storefrontApiDomain = "") {
79
+ const { uniqueToken, visitToken } = trackingUtils.getTrackingValues();
80
+ const response = await fetch(
81
+ // TODO: update this endpoint when it becomes stable
82
+ `${storefrontApiDomain.replace(/\/+$/, "")}/api/unstable/graphql.json`,
83
+ {
84
+ method: "POST",
85
+ headers: {
86
+ "Content-Type": "application/json",
87
+ ...storefrontAccessToken && {
88
+ "X-Shopify-Storefront-Access-Token": storefrontAccessToken
89
+ },
90
+ ...visitToken || uniqueToken ? {
91
+ [trackingUtils.SHOPIFY_VISIT_TOKEN_HEADER]: visitToken,
92
+ [trackingUtils.SHOPIFY_UNIQUE_TOKEN_HEADER]: uniqueToken
93
+ } : void 0
94
+ },
95
+ body: JSON.stringify({
96
+ query: (
97
+ // This query ensures we get _cmp (consent) server-timing header, which is not available in other queries.
98
+ // This value can be passed later to consent-tracking-api and privacy-banner scripts to avoid extra requests.
99
+ "query ensureCookies { consentManagement { cookies(visitorConsent:{}) { cookieDomain } } }"
100
+ )
101
+ })
102
+ }
103
+ );
104
+ if (!response.ok) {
105
+ throw new Error(
106
+ `Failed to fetch consent from browser: ${response.status} ${response.statusText}`
107
+ );
108
+ }
109
+ await response.json();
110
+ trackingUtils.getTrackingValues();
111
+ }
112
+ function useCoreShopifyCookies({
113
+ checkoutDomain,
114
+ storefrontAccessToken,
115
+ fetchTrackingValues = false
116
+ }) {
117
+ const [cookiesReady, setCookiesReady] = React.useState(!fetchTrackingValues);
118
+ const hasFetchedTrackingValues = React.useRef(false);
119
+ React.useEffect(() => {
120
+ if (!fetchTrackingValues) {
121
+ setCookiesReady(true);
122
+ return;
123
+ }
124
+ if (hasFetchedTrackingValues.current) return;
125
+ hasFetchedTrackingValues.current = true;
126
+ fetchTrackingValuesFromBrowser(storefrontAccessToken).catch(
127
+ (error) => checkoutDomain ? (
128
+ // Retry with checkout domain if available to at least
129
+ // get the server-timing values for tracking.
130
+ fetchTrackingValuesFromBrowser(
131
+ storefrontAccessToken,
132
+ checkoutDomain
133
+ )
134
+ ) : Promise.reject(error)
135
+ ).catch((error) => {
136
+ console.warn(
137
+ "[h2:warn:useShopifyCookies] Failed to fetch tracking values from browser: " + (error instanceof Error ? error.message : String(error))
138
+ );
139
+ }).finally(() => {
140
+ setCookiesReady(true);
141
+ });
142
+ }, [checkoutDomain, fetchTrackingValues, storefrontAccessToken]);
143
+ return cookiesReady;
144
+ }
58
145
  exports.useShopifyCookies = useShopifyCookies;
59
146
  //# sourceMappingURL=useShopifyCookies.js.map