@deriv-com/analytics 1.38.8 → 1.38.10

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 (77) hide show
  1. package/README.md +38 -8
  2. package/dist/browser/analytics.bundle.global.js +12 -12
  3. package/dist/browser/analytics.bundle.global.js.map +1 -1
  4. package/dist/browser/analytics.esm.mjs +1 -7
  5. package/dist/browser/analytics.esm.mjs.map +1 -1
  6. package/dist/chunk-7CT4XR74.mjs +3 -0
  7. package/dist/chunk-7CT4XR74.mjs.map +1 -0
  8. package/dist/chunk-BVGYK54Z.js +4 -0
  9. package/dist/chunk-BVGYK54Z.js.map +1 -0
  10. package/dist/chunk-D2MKQCJB.mjs +3 -0
  11. package/dist/chunk-D2MKQCJB.mjs.map +1 -0
  12. package/dist/chunk-FE5YWK6A.js +3 -0
  13. package/dist/chunk-FE5YWK6A.js.map +1 -0
  14. package/dist/chunk-ID2GN2PS.mjs +4 -0
  15. package/dist/chunk-ID2GN2PS.mjs.map +1 -0
  16. package/dist/chunk-ILN7DKKL.js +3 -0
  17. package/dist/chunk-ILN7DKKL.js.map +1 -0
  18. package/dist/chunk-OB6HH25Z.mjs +9 -0
  19. package/dist/chunk-OB6HH25Z.mjs.map +1 -0
  20. package/dist/chunk-YGU4ZMIH.mjs +14 -0
  21. package/dist/chunk-YGU4ZMIH.mjs.map +1 -0
  22. package/dist/growthbook-3K6VUME4.mjs +3 -0
  23. package/dist/growthbook-3K6VUME4.mjs.map +1 -0
  24. package/dist/index.d.mts +9 -5
  25. package/dist/index.d.ts +9 -5
  26. package/dist/index.js +2 -3
  27. package/dist/index.js.map +1 -1
  28. package/dist/index.mjs +2 -3
  29. package/dist/index.mjs.map +1 -1
  30. package/dist/metafile-cjs.json +1 -1
  31. package/dist/metafile-esm.json +1 -1
  32. package/dist/metafile-iife.json +1 -1
  33. package/dist/posthog-ANAV4RFJ.mjs +6 -0
  34. package/dist/posthog-ANAV4RFJ.mjs.map +1 -0
  35. package/dist/{posthog-Df2P2mKj.d.mts → posthog-Ds4g1sJf.d.mts} +21 -7
  36. package/dist/{posthog-BXTVRi7y.d.ts → posthog-DvDj4azD.d.ts} +21 -7
  37. package/dist/providers/growthbook/index.d.mts +5 -3
  38. package/dist/providers/growthbook/index.d.ts +5 -3
  39. package/dist/providers/growthbook/index.js +2 -2
  40. package/dist/providers/growthbook/index.js.map +1 -1
  41. package/dist/providers/growthbook/index.mjs +2 -2
  42. package/dist/providers/growthbook/index.mjs.map +1 -1
  43. package/dist/providers/posthog/index.d.mts +1 -1
  44. package/dist/providers/posthog/index.d.ts +1 -1
  45. package/dist/providers/posthog/index.js +2 -2
  46. package/dist/providers/posthog/index.js.map +1 -1
  47. package/dist/providers/posthog/index.mjs +2 -2
  48. package/dist/providers/posthog/index.mjs.map +1 -1
  49. package/dist/providers/rudderstack/index.d.mts +5 -2
  50. package/dist/providers/rudderstack/index.d.ts +5 -2
  51. package/dist/providers/rudderstack/index.js +1 -1
  52. package/dist/providers/rudderstack/index.mjs +1 -1
  53. package/dist/utils/analytics-cache/index.d.mts +14 -5
  54. package/dist/utils/analytics-cache/index.d.ts +14 -5
  55. package/dist/utils/analytics-cache/index.js +1 -1
  56. package/dist/utils/analytics-cache/index.mjs +1 -1
  57. package/package.json +1 -1
  58. package/dist/chunk-3LFZFQL4.mjs +0 -3
  59. package/dist/chunk-3LFZFQL4.mjs.map +0 -1
  60. package/dist/chunk-CAKVOFLO.js +0 -3
  61. package/dist/chunk-CAKVOFLO.js.map +0 -1
  62. package/dist/chunk-DNCZM4KN.mjs +0 -4
  63. package/dist/chunk-DNCZM4KN.mjs.map +0 -1
  64. package/dist/chunk-EMF3RT6E.mjs +0 -14
  65. package/dist/chunk-EMF3RT6E.mjs.map +0 -1
  66. package/dist/chunk-HQX3Z7PF.mjs +0 -3
  67. package/dist/chunk-HQX3Z7PF.mjs.map +0 -1
  68. package/dist/chunk-LKLVBL24.js +0 -3
  69. package/dist/chunk-LKLVBL24.js.map +0 -1
  70. package/dist/chunk-NOEKD4DT.js +0 -4
  71. package/dist/chunk-NOEKD4DT.js.map +0 -1
  72. package/dist/chunk-UDVSKE3Y.mjs +0 -3
  73. package/dist/chunk-UDVSKE3Y.mjs.map +0 -1
  74. package/dist/growthbook-NJGOOJH4.mjs +0 -3
  75. package/dist/growthbook-NJGOOJH4.mjs.map +0 -1
  76. package/dist/posthog-72HGN74H.mjs +0 -6
  77. package/dist/posthog-72HGN74H.mjs.map +0 -1
@@ -1,9 +1,3 @@
1
1
  /* @deriv-com/analytics - Browser ESM Bundle (RudderStack + PostHog) - Built with tsup */
2
- import{a as Z}from"../chunk-EMF3RT6E.mjs";import{b as Y,d as ee,h as x}from"../chunk-HQX3Z7PF.mjs";var ke=2*365*24*60*60,E=class E{constructor(e,t){this.analytics=new Z;this.has_identified=!1;this.has_initialized=!1;this.current_page="";this.rudderstack_anonymous_cookie_key="rudder_anonymous_id";this.getAnonymousId=()=>document.cookie.match("(^|;)\\s*"+this.rudderstack_anonymous_cookie_key+"\\s*=\\s*([^;]+)")?.pop();this.setCookieIfNotExists=()=>{if(!this.getAnonymousId())try{let t=window.location.hostname,a=["webflow.io"].some(h=>t.endsWith(h))?t:t.split(".").slice(-2).join("."),r;if(crypto?.randomUUID)r=crypto.randomUUID();else if(crypto?.getRandomValues){let h=new Uint8Array(16);crypto.getRandomValues(h),h[6]=h[6]&15|64,h[8]=h[8]&63|128;let y=Array.from(h,k=>k.toString(16).padStart(2,"0")).join("");r=`${y.slice(0,8)}-${y.slice(8,12)}-${y.slice(12,16)}-${y.slice(16,20)}-${y.slice(20)}`}else throw new Error("Crypto API not available for secure random UUID generation");let u=window.location.protocol==="https:"?"; Secure":"";document.cookie=`${this.rudderstack_anonymous_cookie_key}=${r}; path=/; Domain=${a}; max-age=${ke}; SameSite=Lax${u}`}catch(t){console.warn("RudderStack: Failed to set anonymous ID cookie",t)}};this.getUserId=()=>this.analytics.getUserId();this.init=e=>{if(!e){console.warn("RudderStack: Initialization skipped - no key provided");return}try{this.setCookieIfNotExists(),this.analytics.load(e,ee,{externalAnonymousIdCookieName:this.rudderstack_anonymous_cookie_key,lockIntegrationsVersion:!0,onLoaded:()=>{this.has_initialized=!0,this.has_identified=!!this.getUserId(),this.onLoadedCallback?.()}})}catch(t){console.error("RudderStack: Failed to initialize",t)}};this.identifyEvent=(e,t)=>{if(!this.has_initialized){console.warn("RudderStack: Cannot identify - not initialized");return}let n=this.getUserId();if(!n||n!==e)try{this.analytics.identify(e,t||{}),this.has_identified=!0}catch(i){console.error("RudderStack: Failed to identify user",i)}else this.has_identified=!0};this.pageView=(e,t="Deriv App",n,i)=>{if(!(!this.has_initialized||e===this.current_page))try{let a={...n&&{user_id:n},...i};this.analytics.page(t,e,a),this.current_page=e}catch(a){console.error("RudderStack: Failed to track page view",a)}};this.reset=()=>{if(this.has_initialized)try{this.analytics.reset(),this.has_identified=!1}catch(e){console.error("RudderStack: Failed to reset",e)}};this.track=(e,t)=>{if(this.has_initialized)try{this.analytics.track(e,t)}catch(n){console.warn("RudderStack: Failed to track event",n)}};this.onLoadedCallback=t,this.init(e)}};E.getRudderStackInstance=(e,t)=>(E._instance||(E._instance=new E(e,t)),E._instance);var b=E;function P(s){for(var e=1;e<arguments.length;e++){var t=arguments[e];for(var n in t)s[n]=t[n]}return s}var we={read:function(s){return s[0]==='"'&&(s=s.slice(1,-1)),s.replace(/(%[\dA-F]{2})+/gi,decodeURIComponent)},write:function(s){return encodeURIComponent(s).replace(/%(2[346BF]|3[AC-F]|40|5[BDE]|60|7[BCD])/g,decodeURIComponent)}};function N(s,e){function t(i,a,r){if(!(typeof document>"u")){r=P({},e,r),typeof r.expires=="number"&&(r.expires=new Date(Date.now()+r.expires*864e5)),r.expires&&(r.expires=r.expires.toUTCString()),i=encodeURIComponent(i).replace(/%(2[346B]|5E|60|7C)/g,decodeURIComponent).replace(/[()]/g,escape);var d="";for(var u in r)r[u]&&(d+="; "+u,r[u]!==!0&&(d+="="+r[u].split(";")[0]));return document.cookie=i+"="+s.write(a,i)+d}}function n(i){if(!(typeof document>"u"||arguments.length&&!i)){for(var a=document.cookie?document.cookie.split("; "):[],r={},d=0;d<a.length;d++){var u=a[d].split("="),h=u.slice(1).join("=");try{var y=decodeURIComponent(u[0]);if(r[y]=s.read(h,y),i===y)break}catch{}}return i?r[i]:r}}return Object.create({set:t,get:n,remove:function(i,a){t(i,"",P({},a,{expires:-1}))},withAttributes:function(i){return N(this.converter,P({},this.attributes,i))},withConverter:function(i){return N(P({},this.converter,i),this.attributes)}},{attributes:{value:Object.freeze(e)},converter:{value:Object.freeze(s)}})}var w=N(we,{path:"/"});var O="cached_analytics_events",L="cached_analytics_page_views",te=(s,e)=>{try{let t=x(),n=w.get(O),i=n?JSON.parse(n):[];i.push({name:s,properties:e,timestamp:Date.now()});let a={expires:1};t&&(a.domain=t),w.set(O,JSON.stringify(i),a)}catch(t){console.warn("Analytics: Failed to cache event",t)}},ne=(s,e)=>{try{let t=x(),n=w.get(L),i=n?JSON.parse(n):[];i.push({name:s,properties:e,timestamp:Date.now()});let a={expires:1};t&&(a.domain=t),w.set(L,JSON.stringify(i),a)}catch(t){console.warn("Analytics: Failed to cache page view",t)}},ie=()=>{try{let s=w.get(O);if(s){let e=JSON.parse(s);return Array.isArray(e)?e:[]}}catch(s){console.warn("Analytics: Failed to get cached events",s)}return[]},se=()=>{try{let s=w.get(L);if(s){let e=JSON.parse(s);return Array.isArray(e)?e:[]}}catch(s){console.warn("Analytics: Failed to get cached pages",s)}return[]},re=()=>{let s=x(),e=s?{domain:s}:{};w.remove(O,e)},oe=()=>{let s=x(),e=s?{domain:s}:{};w.remove(L,e)};var T=s=>/^[0-9a-f]{8}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{12}$/i.test(s),U=null,ae=async()=>{if(U)return U;let s=JSON.parse(w.get("website_status")||"{}")?.clients_country;return U=(async()=>{try{let e=await fetch(Y).catch(()=>null);if(!e)return s||"";let t=await e.text().catch(()=>"");return t?Object.fromEntries(t.split(`
3
- `).map(i=>i.split("=",2))).loc?.toLowerCase()||s||"":s||""}catch{return s||""}})(),U},R=s=>{if(s==null||typeof s!="object")return s;if(Array.isArray(s)){let t=s.map(R).filter(n=>n!=null);return t.length?t:void 0}let e={};return Object.entries(s).forEach(([t,n])=>{let i=R(n);i==null||i===""||typeof i=="object"&&!Array.isArray(i)&&Object.keys(i).length===0||Array.isArray(i)&&i.length===0||(e[t]=i)}),Object.keys(e).length?e:void 0},j=s=>{if(s==null||typeof s!="object"||Array.isArray(s))return s;let e={};return Object.entries(s).forEach(([t,n])=>{n&&typeof n=="object"&&!Array.isArray(n)?Object.assign(e,j(n)):e[t]=n}),e};function _e(s){let e,t,n,i={},a={},r=[],d=[],u=!1,h=()=>{if(!u&&t?.has_initialized){u=!0;try{let o=ie();o.length>0&&(o.forEach(f=>{t?.track(f.name,f.properties)}),re());let p=se();p.length>0&&(p.forEach(f=>{t?.pageView(f.name,"Deriv App",A(),f.properties)}),oe())}catch(o){console.warn("Analytics: Failed to process cookie cache",o)}}},y=()=>{h(),d.forEach(({userId:o,traits:p})=>{o&&t?.identifyEvent(o,p)}),d=[]},k=async({growthbookKey:o,growthbookDecryptionKey:p,rudderstackKey:f,growthbookOptions:l,posthogOptions:_})=>{var g,m;try{let v=l?.attributes?.country||(o?await ae():void 0);if(f&&(t=b.getRudderStackInstance(f,y)),l?.attributes&&Object.keys(l.attributes).length>0){let c=l.attributes,C=t?.getAnonymousId();i={...i,country:v,...c.user_language&&{user_language:c.user_language},...c.account_type&&{account_type:c.account_type},...c.app_id&&{app_id:c.app_id},...c.residence_country&&{residence_country:c.residence_country},...c.device_type&&{device_type:c.device_type},...c.url&&{url:c.url},...c.email_hash&&{email_hash:c.email_hash},...c.network_type&&{network_type:c.network_type},...c.network_rtt&&{network_rtt:c.network_rtt},...c.network_downlink&&{network_downlink:c.network_downlink},...c.account_currency&&{account_currency:c.account_currency},...c.account_mode&&{account_mode:c.account_mode},loggedIn:!!c.loggedIn,...c.user_id&&!T(c.user_id)&&{user_id:c.user_id},...C&&{anonymous_id:C}}}l??(l={}),l.attributes??(l.attributes={});let I=t?.getAnonymousId();if((g=l.attributes).id??(g.id=I),(m=l.attributes).country??(m.country=v),o){let{Growthbook:c}=await import("../growthbook-NJGOOJH4.mjs");e=c.getGrowthBookInstance(o,p,l);let C=setInterval(()=>{Object.keys(a).length>0?clearInterval(C):a=J("tracking-buttons-config",{})},1e3)}if(_){let{Posthog:c}=await import("../posthog-72HGN74H.mjs");n=c.getPosthogInstance(_)}}catch(v){console.warn("Analytics: Failed to initialize",v)}},S=({country:o,user_language:p,device_language:f,device_type:l,account_type:_,user_id:g,anonymous_id:m,app_id:v,utm_source:I,utm_medium:c,utm_campaign:C,is_authorised:me,residence_country:V,url:D,domain:ve,geo_location:q,loggedIn:z,network_downlink:B,network_rtt:M,network_type:K,account_currency:W,account_mode:Q})=>{let F=g??A();if(e){let $={country:o,residence_country:V,user_language:p,device_language:f,device_type:l,utm_source:I,utm_medium:c,utm_campaign:C,is_authorised:me,url:D,domain:ve,loggedIn:z,...g&&!T(g)&&{user_id:g},anonymous_id:m};F&&($.id=F,$.user_id=F),e.setAttributes($)}i={...i,...o!==void 0&&{country:o},...q!==void 0&&{geo_location:q},...p!==void 0&&{user_language:p},..._!==void 0&&{account_type:_},...v!==void 0&&{app_id:v},...V!==void 0&&{residence_country:V},...l!==void 0&&{device_type:l},...D!==void 0&&{url:D},...z!==void 0&&{loggedIn:z},...B!==void 0&&{network_downlink:B},...M!==void 0&&{network_rtt:M},...K!==void 0&&{network_type:K},...g!==void 0&&!T(g)&&{user_id:g},...m!==void 0&&{anonymous_id:m},...W!==void 0&&{account_currency:W},...Q!==void 0&&{account_mode:Q}}},ce=o=>e?.getFeatureState(o)?.experimentResult?.name,J=(o,p)=>e?.getFeatureValue(o,p),de=async()=>await e?.getStatus(),le=o=>e?.isOn(o),ue=o=>e?.setUrl(o),A=()=>{let o=t?.getUserId()||"";return o&&!T(o)?o:""},pe=()=>t?.getAnonymousId()||"",fe=(o,p="Deriv App",f)=>{let l=A();t&&(t.has_initialized?t.pageView(o,p,l,f):ne(o,{platform:p,...f}))},he=(o,p)=>{let f=o||A();if(!f)return;let l=p?.rudderstack!==void 0||p?.posthog!==void 0,_=l?p?.rudderstack:p,g=l?p?.posthog:p;t&&(t.has_initialized?t.identifyEvent(f,_):d.some(m=>m.userId===f)||d.push({userId:f,traits:_})),n?.has_initialized&&n.identifyEvent(f,g)},ye=()=>{t?.has_initialized&&t.reset(),n?.has_initialized&&n.reset()},ge=o=>"event_metadata"in o||"cta_information"in o||"error"in o,X={initialise:k,setAttributes:S,identifyEvent:he,getFeatureState:ce,getFeatureValue:J,getGrowthbookStatus:de,isFeatureOn:le,setUrl:ue,getId:A,getAnonymousId:pe,trackEvent:(o,p)=>{let f=A(),l={};if(ge(p)){let m=p;l={...m,event_metadata:{...i,...f&&!i.user_id&&{user_id:f},...m.event_metadata}}}else l={...i,...p,...f&&!i.user_id&&{user_id:f}};if(!(!(o in a)||a[o]))return;let g=t?.has_initialized;if(!navigator.onLine||!g)g?r.push({event:o,payload:l}):te(o,l);else{r.length>0&&(r.forEach(v=>{let I=R(v.payload);t?.track(v.event,I)}),r=[]);let m=R(l);t?.track(o,m)}if(n?.has_initialized){let m=j(l),v=R(m);n.capture(o,v)}},getInstances:()=>({ab:e,tracking:t,posthog:n}),pageView:fe,reset:ye};return typeof window<"u"&&(window.AnalyticsInstance=X),X}var Ee=_e();var G=class{constructor(){this.interval=null;this.responses=[];this.isTrackingResponses=!1;this.delegatedSelectors=new Set;this.addEventhandler=this.addEventHandler.bind(this)}hash(e,t=32){let n=d=>{let u=2166136261;for(let h=0;h<d.length;h++)u^=d.charCodeAt(h),u=u*16777619>>>0;return u.toString(16)},i=d=>btoa(d),a=n(e),r=i(a);for(;r.length<t;)r+=i(n(r));return r.substring(0,t)}getCookies(e){let n=`; ${document.cookie}`.split(`; ${e}=`);if(n.length===2){let i=decodeURIComponent(n.pop().split(";").shift());try{return JSON.parse(i)}catch{return i}}return null}trackPageUnload(){typeof window>"u"||window.addEventListener("beforeunload",()=>{this.isPageViewSent()||this.push("cached_analytics_page_views",{name:window.location.href,properties:{url:window.location.href}})})}trackResponses(){if(typeof window>"u"||typeof XMLHttpRequest>"u")return;let e=XMLHttpRequest.prototype.open,t=XMLHttpRequest.prototype.send;XMLHttpRequest.prototype.open=function(n,i){return this._url=i,this._method=n,e.apply(this,arguments)},XMLHttpRequest.prototype.send=function(n){return this.addEventListener("load",function(){let i=null;if(typeof n=="string")try{i=JSON.parse(n)}catch{i=n}let a={url:this._url,method:this._method,status:this.status,headers:this.getAllResponseHeaders(),data:this.responseText,payload:i};H.responses.push(a)}),t.apply(this,arguments)},this.isTrackingResponses=!0}isReady(){if(typeof window>"u")return!1;let e=window.Analytics;return typeof e>"u"||e===null?!1:!!e.Analytics?.getInstances?.()?.tracking}parseCookies(e){if(typeof document>"u")return null;let t=document.cookie.split("; ").reduce((n,i)=>{let[a,r]=i.split("=");return a&&r&&(n[decodeURIComponent(a)]=decodeURIComponent(r)),n},{});try{return t[e]?JSON.parse(t[e]):null}catch{return null}}isPageViewSent(){return!!this.responses.find(e=>e.payload?.type==="page"&&e.payload?.anonymousId)}set(e){this.push("cached_analytics_events",e)}push(e,t){if(typeof document>"u")return;let n=[],i=this.parseCookies(e);i&&Array.isArray(i)&&(n=i),n.push(t);let a=this.getAllowedDomain(),r=365*24*60*60,d=`${e}=${encodeURIComponent(JSON.stringify(n))}; path=/; Domain=${a}; max-age=${r}; SameSite=Lax`;document.cookie=d}getAllowedDomain(){if(typeof window>"u")return"";let e=window.location.hostname;if(e==="localhost"||/^\d+\.\d+\.\d+\.\d+$/.test(e))return e;let t=e.split(".");return t.length===1?e:`.${t.slice(-2).join(".")}`}processEvent(e){let t=this.getCookies("client_information");if(t){let{email:n=null}=t;n&&(e.properties.email_hash=this.hash(n))}if(e?.properties?.email){let n=e.properties.email;delete e.properties.email,e.properties.email_hash=this.hash(n)}return e}track(e,t=!1){if(typeof window>"u")return;let n=this.processEvent(e),i=window.Analytics;this.isReady()&&!t?i.Analytics.trackEvent(n.name,n.properties):this.set(n)}pageView(){typeof window>"u"||(this.isTrackingResponses||(this.trackResponses(),this.trackPageUnload()),this.interval=setInterval(()=>{let e=window.Analytics;typeof e<"u"&&typeof e.Analytics?.pageView=="function"&&this.isReady()&&e.Analytics.pageView(window.location.href,"Trader's hub"),this.isPageViewSent()&&this.interval&&clearInterval(this.interval)},1e3))}listen(e,{name:t="",properties:n={}},i=!1,a=null){let r=u=>{u.dataset?.clickEventTracking||(u.addEventListener("click",h=>{let y={name:t,properties:n,cache:i};if(typeof a=="function"){let k=a(h);y={...k,cache:k.cache??i}}this.track(y,y.cache??!1)}),u.dataset.clickEventTracking="true")};(e instanceof NodeList?Array.from(e):[e]).forEach(r)}addEventHandler(e){return typeof window>"u"?this:(e.forEach(({element:t,event:n={name:"",properties:{}},cache:i=!1,callback:a=null})=>{if(typeof t=="string"){let r=t;if(!this.delegatedSelectors.has(r)){let d=u=>{let h=u.target;if(!h)return;let y=h.closest(r);if(y&&!y.dataset?.clickEventTracking){let k={name:n.name,properties:n.properties,cache:i};if(typeof a=="function"){let S=a(u);k={...S,cache:S.cache??i}}y.dataset.clickEventTracking="true",this.track(k,k.cache??!1)}};document.addEventListener("click",d),this.delegatedSelectors.add(r)}}else(t instanceof NodeList?Array.from(t):[t]).forEach(d=>{d&&!d.dataset?.clickEventTracking&&this.listen(d,n,i,a)})}),this)}loadEvent(e){return e.forEach(({event:t})=>{let{name:n,properties:i}=t;this.track({name:n,properties:i})}),this}pageLoadEvent(e){if(typeof window>"u")return this;let t=window.location.pathname.slice(1);return e.forEach(({pages:n=[],excludedPages:i=[],event:a,callback:r=null})=>{let d=!1;if(n.length?n.includes(t)&&(d=!0):i.length&&i.includes(t)||(d=!0),d){let u=r?r():a;this.loadEvent([{event:u}])}}),this}clearInterval(){this.interval&&(clearInterval(this.interval),this.interval=null)}cleanup(){this.clearInterval(),this.responses=[],this.isTrackingResponses=!1}},H=new G;typeof window<"u"&&(window.cacheTrackEvents=H);export{Ee as Analytics,H as cacheTrackEvents};
4
- /*! Bundled license information:
5
-
6
- js-cookie/dist/js.cookie.mjs:
7
- (*! js-cookie v3.0.5 | MIT *)
8
- */
2
+ import{a as Z}from"../chunk-YGU4ZMIH.mjs";import{c as M,g as I,h as _,i as R,k as S,l as Y,m as O,n as ee}from"../chunk-OB6HH25Z.mjs";var ke=2*365*24*60*60,b=class b{constructor(t,e,n=!1){this.analytics=new Z;this.has_identified=!1;this.has_initialized=!1;this.current_page="";this.rudderstack_anonymous_cookie_key="rudder_anonymous_id";this.debug=!1;this.log=R("[RudderStack]",()=>this.debug);this.getAnonymousId=()=>document.cookie.match("(^|;)\\s*"+this.rudderstack_anonymous_cookie_key+"\\s*=\\s*([^;]+)")?.pop();this.setCookieIfNotExists=()=>{if(!this.getAnonymousId())try{let e=window.location.hostname,a=["webflow.io"].some(g=>e.endsWith(g))?e:e.split(".").slice(-2).join("."),c;if(crypto?.randomUUID)c=crypto.randomUUID();else if(crypto?.getRandomValues){let g=new Uint8Array(16);crypto.getRandomValues(g),g[6]=g[6]&15|64,g[8]=g[8]&63|128;let v=Array.from(g,w=>w.toString(16).padStart(2,"0")).join("");c=`${v.slice(0,8)}-${v.slice(8,12)}-${v.slice(12,16)}-${v.slice(16,20)}-${v.slice(20)}`}else throw new Error("Crypto API not available for secure random UUID generation");let u=window.location.protocol==="https:"?"; Secure":"";document.cookie=`${this.rudderstack_anonymous_cookie_key}=${c}; path=/; Domain=${a}; max-age=${ke}; SameSite=Lax${u}`}catch(e){console.warn("RudderStack: Failed to set anonymous ID cookie",e)}};this.getUserId=()=>this.analytics.getUserId();this.init=t=>{if(!t){console.warn("RudderStack: Initialization skipped - no key provided");return}this.log("init | loading SDK",{dataplane:M});try{this.setCookieIfNotExists(),this.analytics.load(t,M,{externalAnonymousIdCookieName:this.rudderstack_anonymous_cookie_key,lockIntegrationsVersion:!0,onLoaded:()=>{this.has_initialized=!0,this.has_identified=!!this.getUserId(),this.log("init | SDK loaded successfully",{userId:this.getUserId(),anonymousId:this.getAnonymousId()}),this.onLoadedCallback?.()}})}catch(e){console.error("RudderStack: Failed to initialize",e)}};this.identifyEvent=(t,e)=>{if(!this.has_initialized){console.warn("RudderStack: Cannot identify - not initialized");return}let n=this.getUserId();if(!n||n!==t)try{this.log("identifyEvent | identifying user",{user_id:t,traits:e}),this.analytics.identify(t,e||{}),this.has_identified=!0}catch(i){console.error("RudderStack: Failed to identify user",i)}else this.log("identifyEvent | user already identified",{user_id:t}),this.has_identified=!0};this.pageView=(t,e="Deriv App",n,i)=>{if(!(!this.has_initialized||t===this.current_page))try{let a={...n&&{user_id:n},...i};this.log("pageView | tracking page view",{platform:e,current_page:t,properties:a}),this.analytics.page(e,t,a),this.current_page=t}catch(a){console.error("RudderStack: Failed to track page view",a)}};this.reset=()=>{if(this.has_initialized)try{this.log("reset | resetting RudderStack session"),this.analytics.reset(),this.has_identified=!1}catch(t){console.error("RudderStack: Failed to reset",t)}};this.track=(t,e)=>{if(this.has_initialized)try{this.log("track | sending event to RudderStack",{event:t,payload:e}),this.analytics.track(t,e)}catch(n){console.warn("RudderStack: Failed to track event",n)}};this.onLoadedCallback=e,this.debug=n,this.init(t)}};b.getRudderStackInstance=(t,e,n=!1)=>(b._instance||(b._instance=new b(t,e,n)),b._instance);var U=b;var $="cached_analytics_events",H="cached_analytics_page_views",te=(h,t)=>{try{let e=I(),n=_.get($),i=n?JSON.parse(n):[];i.push({name:h,properties:t,timestamp:Date.now()});let a={expires:1};e&&(a.domain=e),_.set($,JSON.stringify(i),a)}catch(e){console.warn("Analytics: Failed to cache event",e)}},ne=(h,t)=>{try{let e=I(),n=_.get(H),i=n?JSON.parse(n):[];i.push({name:h,properties:t,timestamp:Date.now()});let a={expires:1};e&&(a.domain=e),_.set(H,JSON.stringify(i),a)}catch(e){console.warn("Analytics: Failed to cache page view",e)}},ie=()=>{try{let h=_.get($);if(h){let t=JSON.parse(h);return Array.isArray(t)?t:[]}}catch(h){console.warn("Analytics: Failed to get cached events",h)}return[]},se=()=>{try{let h=_.get(H);if(h){let t=JSON.parse(h);return Array.isArray(t)?t:[]}}catch(h){console.warn("Analytics: Failed to get cached pages",h)}return[]},ae=()=>{let h=I(),t=h?{domain:h}:{};_.remove($,t)},oe=()=>{let h=I(),t=h?{domain:h}:{};_.remove(H,t)};var B=class{constructor(){this.interval=null;this.responses=[];this.isTrackingResponses=!1;this.delegatedSelectors=new Set;this.debug=!1;this.log=R("[CacheManager]",()=>this.debug);this.addEventhandler=this.addEventHandler.bind(this)}setDebug(t){this.debug=t}hash(t,e=32){let n=p=>{let u=2166136261;for(let g=0;g<p.length;g++)u^=p.charCodeAt(g),u=u*16777619>>>0;return u.toString(16)},i=p=>btoa(p),a=n(t),c=i(a);for(;c.length<e;)c+=i(n(c));return c.substring(0,e)}getCookies(t){let n=`; ${document.cookie}`.split(`; ${t}=`);if(n.length===2){let i=decodeURIComponent(n.pop().split(";").shift());try{return JSON.parse(i)}catch{return i}}return null}trackPageUnload(){typeof window>"u"||window.addEventListener("beforeunload",()=>{this.isPageViewSent()||this.push("cached_analytics_page_views",{name:window.location.href,properties:{url:window.location.href}})})}trackResponses(){if(typeof window>"u"||typeof XMLHttpRequest>"u")return;let t=XMLHttpRequest.prototype.open,e=XMLHttpRequest.prototype.send;XMLHttpRequest.prototype.open=function(n,i){return this._url=i,this._method=n,t.apply(this,arguments)},XMLHttpRequest.prototype.send=function(n){return this.addEventListener("load",function(){let i=null;if(typeof n=="string")try{i=JSON.parse(n)}catch{i=n}let a={url:this._url,method:this._method,status:this.status,headers:this.getAllResponseHeaders(),data:this.responseText,payload:i};T.responses.push(a)}),e.apply(this,arguments)},this.isTrackingResponses=!0}getAnalyticsInstance(){return typeof window>"u"?null:window.AnalyticsInstance??window.DerivAnalytics?.Analytics??window.Analytics?.Analytics}isReady(){if(typeof window>"u")return!1;let t=this.getAnalyticsInstance();return t?!!t.getInstances?.()?.tracking:!1}parseCookies(t){if(typeof document>"u")return null;let e=document.cookie.split("; ");for(let n of e){let i=n.indexOf("=");if(i!==-1&&decodeURIComponent(n.slice(0,i))===t){let a=n.slice(i+1);try{return JSON.parse(decodeURIComponent(a))}catch{return decodeURIComponent(a)}}}return null}isPageViewSent(){return!!this.responses.find(t=>t.payload?.type==="page"&&t.payload?.anonymousId)}set(t){this.log("set | caching event to cookie",t),this.push("cached_analytics_events",t)}push(t,e){if(typeof document>"u")return;let n=[],i=this.parseCookies(t);i&&Array.isArray(i)&&(n=i),n.push(e);let a=this.getAllowedDomain(),c=365*24*60*60,p=`${t}=${encodeURIComponent(JSON.stringify(n))}; path=/; Domain=${a}; max-age=${c}; SameSite=Lax`;document.cookie=p}getAllowedDomain(){if(typeof window>"u")return"";let t=window.location.hostname;if(t==="localhost"||/^\d+\.\d+\.\d+\.\d+$/.test(t))return t;let e=t.split(".");return e.length===1?t:`.${e.slice(-2).join(".")}`}processEvent(t){let e=this.getCookies("client_information");if(e){let{email:n=null}=e;n&&(t.properties.email_hash=this.hash(n))}if(t?.properties?.email){let n=t.properties.email;delete t.properties.email,t.properties.email_hash=this.hash(n)}return t}track(t,e=!1){if(typeof window>"u")return;let n=this.processEvent(t),i=this.getAnalyticsInstance();this.isReady()&&!e?(this.log("track | analytics ready \u2014 calling trackEvent",{event:n.name,properties:n.properties}),i.trackEvent(n.name,n.properties)):(this.log("track | analytics not ready or cache=true \u2014 storing event",{event:n.name,cache:e}),this.set(n))}pageView(){typeof window>"u"||(this.log("pageView | starting page view polling"),this.isTrackingResponses||(this.trackResponses(),this.trackPageUnload()),this.interval=setInterval(()=>{let t=window.Analytics;typeof t<"u"&&typeof t.Analytics?.pageView=="function"&&this.isReady()&&(this.log("pageView | analytics ready \u2014 sending page view",{href:window.location.href}),t.Analytics.pageView(window.location.href,"Trader's hub")),this.isPageViewSent()&&(this.log("pageView | page view confirmed sent \u2014 clearing interval"),this.interval&&clearInterval(this.interval))},1e3))}listen(t,{name:e="",properties:n={}},i=!1,a=null){let c=u=>{u.dataset?.clickEventTracking||(u.addEventListener("click",g=>{let v={name:e,properties:n,cache:i};if(typeof a=="function"){let w=a(g);v={...w,cache:w.cache??i}}this.track(v,v.cache??!1)}),u.dataset.clickEventTracking="true")};(t instanceof NodeList?Array.from(t):[t]).forEach(c)}addEventHandler(t){return typeof window>"u"?this:(t.forEach(({element:e,event:n={name:"",properties:{}},cache:i=!1,callback:a=null})=>{if(typeof e=="string"){let c=e;if(!this.delegatedSelectors.has(c)){let p=u=>{let g=u.target;if(!g)return;let v=g.closest(c);if(v&&!v.dataset?.clickEventTracking){let w={name:n.name,properties:n.properties,cache:i};if(typeof a=="function"){let P=a(u);w={...P,cache:P.cache??i}}v.dataset.clickEventTracking="true",this.track(w,w.cache??!1)}};document.addEventListener("click",p),this.delegatedSelectors.add(c)}}else(e instanceof NodeList?Array.from(e):[e]).forEach(p=>{p&&!p.dataset?.clickEventTracking&&this.listen(p,n,i,a)})}),this)}loadEvent(t){return this.log("loadEvent | firing load events",t.map(e=>e.event.name)),t.forEach(({event:e})=>{let{name:n,properties:i}=e;this.track({name:n,properties:i})}),this}pageLoadEvent(t){if(typeof window>"u")return this;let e=window.location.pathname.slice(1);return this.log("pageLoadEvent | checking page load events",{pathname:e}),t.forEach(({pages:n=[],excludedPages:i=[],event:a,callback:c=null})=>{let p=!1;if(n.length?n.includes(e)&&(p=!0):i.length&&i.includes(e)||(p=!0),p){let u=c?c():a;this.log("pageLoadEvent | dispatching event for page",{pathname:e,event:u.name}),this.loadEvent([{event:u}])}else this.log("pageLoadEvent | skipped event for page",{pathname:e,event:a.name,pages:n,excludedPages:i})}),this}clearInterval(){this.interval&&(clearInterval(this.interval),this.interval=null)}cleanup(){this.clearInterval(),this.responses=[],this.isTrackingResponses=!1}},T=new B;typeof window<"u"&&(window.cacheTrackEvents=T);function me(h){let t=h?.debug??!1,e=R("",()=>t),n,i,a,c={},p={},u=[],g=[],v=!1,w=()=>{if(!v&&i?.has_initialized){v=!0;try{let s=ie();s.length>0&&(e(`processCookieCache | replaying ${s.length} cached event(s)`,s),s.forEach(d=>{i?.track(d.name,d.properties)}),ae());let o=se();o.length>0&&(e(`processCookieCache | replaying ${o.length} cached page view(s)`,o),o.forEach(d=>{i?.pageView(d.name,"Deriv App",C(),d.properties)}),oe())}catch(s){console.warn("Analytics: Failed to process cookie cache",s)}}},P=()=>{e("onSdkLoaded | RudderStack SDK loaded"),w(),g.length>0&&e(`onSdkLoaded | flushing ${g.length} pending identify call(s)`),g.forEach(({userId:s,traits:o})=>{s&&i?.identifyEvent(s,o)}),g=[]},re=async({growthbookKey:s,growthbookDecryptionKey:o,rudderstackKey:d,growthbookOptions:l,posthogOptions:k,debug:f})=>{var y,m;f!==void 0&&(t=f),T.setDebug(t),e("initialise | starting analytics initialization",{rudderstack:!!d,growthbook:!!s,posthog:!!k});try{let E=l?.attributes?.country||(s?await Y():void 0);if(d&&(e("initialise | initializing RudderStack"),i=U.getRudderStackInstance(d,P,t)),l?.attributes&&Object.keys(l.attributes).length>0){let r=l.attributes,A=i?.getAnonymousId();c={...c,country:E,...r.user_language&&{user_language:r.user_language},...r.account_type&&{account_type:r.account_type},...r.app_id&&{app_id:r.app_id},...r.residence_country&&{residence_country:r.residence_country},...r.device_type&&{device_type:r.device_type},...r.url&&{url:r.url},...r.email_hash&&{email_hash:r.email_hash},...r.network_type&&{network_type:r.network_type},...r.network_rtt&&{network_rtt:r.network_rtt},...r.network_downlink&&{network_downlink:r.network_downlink},...r.account_currency&&{account_currency:r.account_currency},...r.account_mode&&{account_mode:r.account_mode},loggedIn:!!r.loggedIn,...r.user_id&&!S(r.user_id)&&{user_id:r.user_id},...A&&{anonymous_id:A}}}l??(l={}),l.attributes??(l.attributes={});let x=i?.getAnonymousId();if((y=l.attributes).id??(y.id=x),(m=l.attributes).country??(m.country=E),s){e("initialise | initializing GrowthBook");let{Growthbook:r}=await import("../growthbook-3K6VUME4.mjs");n=r.getGrowthBookInstance(s,o,l,t),e("initialise | GrowthBook initialized");let A=setInterval(()=>{Object.keys(p).length>0?clearInterval(A):p=K("tracking-buttons-config",{})},1e3)}if(k){e("initialise | initializing PostHog");let{Posthog:r}=await import("../posthog-ANAV4RFJ.mjs");a=r.getPosthogInstance(k,t),e("initialise | PostHog initialized")}e("initialise | analytics initialization complete")}catch(E){console.warn("Analytics: Failed to initialize",E)}},ce=({country:s,user_language:o,device_language:d,device_type:l,account_type:k,user_id:f,anonymous_id:y,app_id:m,utm_source:E,utm_medium:x,utm_campaign:r,is_authorised:A,residence_country:V,url:L,domain:Q,geo_location:N,loggedIn:z,network_downlink:F,network_rtt:G,network_type:J,account_currency:X,account_mode:q})=>{let j=f??C();if(e("setAttributes | received attributes",{country:s,user_language:o,device_language:d,device_type:l,account_type:k,user_id:f,anonymous_id:y,app_id:m,utm_source:E,utm_medium:x,utm_campaign:r,is_authorised:A,residence_country:V,url:L,domain:Q,geo_location:N,loggedIn:z,network_downlink:F,network_rtt:G,network_type:J,account_currency:X,account_mode:q}),n){let D={country:s,residence_country:V,user_language:o,device_language:d,device_type:l,utm_source:E,utm_medium:x,utm_campaign:r,is_authorised:A,url:L,domain:Q,loggedIn:z,...f&&!S(f)&&{user_id:f},anonymous_id:y};j&&(D.id=j,D.user_id=j),e("setAttributes | called GrowthBook setAttributes",D),n.setAttributes(D)}c={...c,...s!==void 0&&{country:s},...N!==void 0&&{geo_location:N},...o!==void 0&&{user_language:o},...k!==void 0&&{account_type:k},...m!==void 0&&{app_id:m},...V!==void 0&&{residence_country:V},...l!==void 0&&{device_type:l},...L!==void 0&&{url:L},...z!==void 0&&{loggedIn:z},...F!==void 0&&{network_downlink:F},...G!==void 0&&{network_rtt:G},...J!==void 0&&{network_type:J},...f!==void 0&&!S(f)&&{user_id:f},...y!==void 0&&{anonymous_id:y},...X!==void 0&&{account_currency:X},...q!==void 0&&{account_mode:q}},e("setAttributes | updated core_data",c)},de=s=>n?.getFeatureState(s)?.experimentResult?.name,K=(s,o)=>n?.getFeatureValue(s,o),le=async()=>await n?.getStatus(),pe=s=>n?.isOn(s),ge=s=>n?.setUrl(s),C=()=>{let s=i?.getUserId()||"";return s&&!S(s)?s:""},he=()=>i?.getAnonymousId()||"",ue=(s,o="Deriv App",d)=>{let l=C();e("pageView | called",{current_page:s,platform:o,properties:d,userId:l}),i&&(i.has_initialized?(e("pageView | sending page view to RudderStack",{current_page:s,platform:o}),i.pageView(s,o,l,d)):(e("pageView | RudderStack not initialized \u2014 caching page view to cookie",{current_page:s}),ne(s,{platform:o,...d})))},fe=(s,o)=>{let d=s||C();if(!d){e("identifyEvent | skipped \u2014 no user_id available");return}e("identifyEvent | called",{user_id:d,traits:o});let l=o?.rudderstack!==void 0||o?.posthog!==void 0,k=l?o?.rudderstack:o,f=l?o?.posthog:o;i&&(i.has_initialized?(e("identifyEvent | calling RudderStack identify",{user_id:d,traits:k}),i.identifyEvent(d,k)):g.some(y=>y.userId===d)||(e("identifyEvent | RudderStack not initialized \u2014 queuing identify call",{user_id:d}),g.push({userId:d,traits:k}))),a?.has_initialized&&f&&(e("identifyEvent | calling PostHog identify",{user_id:d,traits:f}),a.identifyEvent(d,f))},ye=()=>{e("reset | resetting all providers"),i?.has_initialized&&(e("reset | resetting RudderStack"),i.reset()),a?.has_initialized&&(e("reset | resetting PostHog"),a.reset())},ve=s=>"event_metadata"in s||"cta_information"in s||"error"in s,W={initialise:re,setAttributes:ce,identifyEvent:fe,backfillPersonProperties:(s,o)=>{e("backfillPersonProperties | called",{user_id:s}),a?.has_initialized?(e("backfillPersonProperties | backfilling person properties in PostHog",{user_id:s}),a.backfillPersonProperties(s,o)):e("backfillPersonProperties | skipped \u2014 PostHog not initialized")},getFeatureState:de,getFeatureValue:K,getGrowthbookStatus:le,isFeatureOn:pe,setUrl:ge,getId:C,getAnonymousId:he,trackEvent:(s,o)=>{let d=C(),l={};if(e("trackEvent | called",{event:s,analytics_data:o,userId:d,core_data:c}),ve(o)){let y=o;l={...y,event_metadata:{...c,...d&&!c.user_id&&{user_id:d},...y.event_metadata}},e("trackEvent | built V2 payload",{event:s,final_payload:l})}else l={...c,...o,...d&&!c.user_id&&{user_id:d}},e("trackEvent | built V1 payload",{event:s,final_payload:l});if(!(!(s in p)||p[s])){e("trackEvent | skipped \u2014 event disabled by tracking_config",{event:s});return}let f=i?.has_initialized;if(!navigator.onLine||!f)f?(e("trackEvent | offline \u2014 caching event to memory",{event:s}),u.push({event:s,payload:l})):(e("trackEvent | RudderStack not initialized \u2014 caching event to cookie",{event:s}),te(s,l));else{u.length>0&&(e(`trackEvent | flushing ${u.length} offline cached event(s) to RudderStack`),u.forEach(m=>{let E=O(m.payload);i?.track(m.event,E)}),u=[]);let y=O(l);e("trackEvent | sending event to RudderStack",{event:s,payload:y}),i?.track(s,y)}if(a?.has_initialized){let y=ee(l),m=O(y);e("trackEvent | sending event to PostHog",{event:s,payload:m}),a.capture(s,m)}},getInstances:()=>({ab:n,tracking:i,posthog:a}),pageView:ue,reset:ye};return typeof window<"u"&&(window.AnalyticsInstance=W),W}var we=me();export{we as Analytics,T as cacheTrackEvents};
9
3
  //# sourceMappingURL=analytics.esm.mjs.map
@@ -1 +1 @@
1
- {"version":3,"sources":["../../src/providers/rudderstack.ts","../../node_modules/js-cookie/dist/js.cookie.mjs","../../src/utils/cookie.ts","../../src/utils/helpers.ts","../../src/analytics.ts","../../src/utils/analytics-cache.ts"],"sourcesContent":["import { RudderAnalytics } from '@rudderstack/analytics-js'\nimport type { TCoreAttributes, TAllEvents } from '../types'\nimport { rudderstackDataplane } from '../utils/urls'\n\n// Constants\nconst COOKIE_MAX_AGE_SECONDS = 2 * 365 * 24 * 60 * 60 // 2 years\n\n/**\n * RudderStack analytics wrapper with singleton pattern.\n * Handles user tracking, page views, and event analytics.\n */\nexport class RudderStack {\n analytics = new RudderAnalytics()\n has_identified = false\n has_initialized = false\n current_page = ''\n rudderstack_anonymous_cookie_key = 'rudder_anonymous_id'\n private static _instance: RudderStack\n private onLoadedCallback?: () => void\n\n constructor(RUDDERSTACK_KEY: string, onLoaded?: () => void) {\n this.onLoadedCallback = onLoaded\n this.init(RUDDERSTACK_KEY)\n }\n\n /**\n * Get or create the singleton instance of RudderStack\n * @param RUDDERSTACK_KEY - RudderStack write key\n * @param onLoaded - Optional callback when RudderStack is loaded\n * @returns The RudderStack singleton instance\n */\n public static getRudderStackInstance = (RUDDERSTACK_KEY: string, onLoaded?: () => void): RudderStack => {\n if (!RudderStack._instance) {\n RudderStack._instance = new RudderStack(RUDDERSTACK_KEY, onLoaded)\n }\n return RudderStack._instance\n }\n\n /**\n * Get the anonymous ID from cookies\n * @returns The anonymous ID or undefined if not found\n */\n getAnonymousId = (): string | undefined => {\n return document.cookie.match('(^|;)\\\\s*' + this.rudderstack_anonymous_cookie_key + '\\\\s*=\\\\s*([^;]+)')?.pop()\n }\n\n /**\n * Set anonymous ID cookie if it doesn't exist\n * Creates a secure cookie with proper domain and security attributes\n */\n setCookieIfNotExists = (): void => {\n const anonymous_id = this.getAnonymousId()\n\n if (!anonymous_id) {\n try {\n const hostname = window.location.hostname\n const external_domains = ['webflow.io']\n const is_external_domain = external_domains.some(domain => hostname.endsWith(domain))\n const domain_name = is_external_domain ? hostname : hostname.split('.').slice(-2).join('.')\n\n // Generate cryptographically secure UUID\n let uuid: string\n if (crypto?.randomUUID) {\n uuid = crypto.randomUUID()\n } else if (crypto?.getRandomValues) {\n // Fallback: Generate UUID v4 using crypto.getRandomValues\n const bytes = new Uint8Array(16)\n crypto.getRandomValues(bytes)\n // Set version (4) and variant bits\n bytes[6] = (bytes[6]! & 0x0f) | 0x40\n bytes[8] = (bytes[8]! & 0x3f) | 0x80\n // Convert to UUID string format\n const hex = Array.from(bytes, byte => byte.toString(16).padStart(2, '0')).join('')\n uuid = `${hex.slice(0, 8)}-${hex.slice(8, 12)}-${hex.slice(12, 16)}-${hex.slice(16, 20)}-${hex.slice(20)}`\n } else {\n // Crypto API not available - this should not happen in modern browsers\n throw new Error('Crypto API not available for secure random UUID generation')\n }\n\n const isSecure = window.location.protocol === 'https:'\n const secureFlag = isSecure ? '; Secure' : ''\n\n document.cookie = `${this.rudderstack_anonymous_cookie_key}=${uuid}; path=/; Domain=${domain_name}; max-age=${COOKIE_MAX_AGE_SECONDS}; SameSite=Lax${secureFlag}`\n } catch (error) {\n console.warn('RudderStack: Failed to set anonymous ID cookie', error)\n }\n }\n }\n\n /**\n * Get the current user ID\n * @returns The user ID, null, or undefined if not identified\n */\n getUserId = () => this.analytics.getUserId()\n\n /**\n * Initialize RudderStack with the provided key\n * @param RUDDERSTACK_KEY - RudderStack write key\n */\n init = (RUDDERSTACK_KEY: string): void => {\n if (!RUDDERSTACK_KEY) {\n console.warn('RudderStack: Initialization skipped - no key provided')\n return\n }\n\n try {\n this.setCookieIfNotExists()\n\n this.analytics.load(RUDDERSTACK_KEY, rudderstackDataplane, {\n externalAnonymousIdCookieName: this.rudderstack_anonymous_cookie_key,\n // Performance optimizations\n lockIntegrationsVersion: true,\n onLoaded: () => {\n this.has_initialized = true\n this.has_identified = !!this.getUserId()\n this.onLoadedCallback?.()\n },\n })\n } catch (error) {\n console.error('RudderStack: Failed to initialize', error)\n }\n }\n\n /**\n * Identify a user with RudderStack\n * Only identifies if user hasn't been identified yet\n * @param user_id - The user ID to identify\n * @param payload - Optional user traits (e.g., language, custom properties)\n */\n identifyEvent = (user_id: string, payload?: Record<string, any>): void => {\n if (!this.has_initialized) {\n console.warn('RudderStack: Cannot identify - not initialized')\n return\n }\n\n const currentUserId = this.getUserId()\n if (!currentUserId || currentUserId !== user_id) {\n try {\n this.analytics.identify(user_id, payload || {})\n this.has_identified = true\n } catch (error) {\n console.error('RudderStack: Failed to identify user', error)\n }\n } else {\n this.has_identified = true\n }\n }\n\n /**\n * Track a page view event\n * @param current_page - The page name/path\n * @param platform - The platform name (default: 'Deriv App')\n * @param user_id - The user ID\n * @param properties - Additional page properties\n */\n pageView = (\n current_page: string,\n platform = 'Deriv App',\n user_id: string,\n properties?: Record<string, unknown>\n ): void => {\n if (!this.has_initialized || current_page === this.current_page) return\n\n try {\n const pageProperties = {\n ...(user_id && { user_id }),\n ...properties,\n }\n\n // Type assertion needed due to RudderStack's type definitions\n this.analytics.page(platform, current_page, pageProperties as any)\n this.current_page = current_page\n } catch (error) {\n console.error('RudderStack: Failed to track page view', error)\n }\n }\n\n /**\n * Reset the RudderStack instance\n * Clears user identification and resets tracking state\n */\n reset = (): void => {\n if (!this.has_initialized) return\n\n try {\n this.analytics.reset()\n this.has_identified = false\n } catch (error) {\n console.error('RudderStack: Failed to reset', error)\n }\n }\n\n /**\n * Track a custom event with payload\n * Payload is pre-cleaned by analytics.ts using cleanObject before being passed here\n * @param event - The event name\n * @param payload - The event payload with core attributes\n */\n track = <T extends keyof TAllEvents>(event: T, payload: TAllEvents[T] & Partial<TCoreAttributes>): void => {\n if (!this.has_initialized) return\n\n try {\n // Type assertion needed to match RudderStack's ApiObject type\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n this.analytics.track(event as string, payload as any)\n } catch (err) {\n console.warn('RudderStack: Failed to track event', err)\n }\n }\n}\n","/*! js-cookie v3.0.5 | MIT */\n/* eslint-disable no-var */\nfunction assign (target) {\n for (var i = 1; i < arguments.length; i++) {\n var source = arguments[i];\n for (var key in source) {\n target[key] = source[key];\n }\n }\n return target\n}\n/* eslint-enable no-var */\n\n/* eslint-disable no-var */\nvar defaultConverter = {\n read: function (value) {\n if (value[0] === '\"') {\n value = value.slice(1, -1);\n }\n return value.replace(/(%[\\dA-F]{2})+/gi, decodeURIComponent)\n },\n write: function (value) {\n return encodeURIComponent(value).replace(\n /%(2[346BF]|3[AC-F]|40|5[BDE]|60|7[BCD])/g,\n decodeURIComponent\n )\n }\n};\n/* eslint-enable no-var */\n\n/* eslint-disable no-var */\n\nfunction init (converter, defaultAttributes) {\n function set (name, value, attributes) {\n if (typeof document === 'undefined') {\n return\n }\n\n attributes = assign({}, defaultAttributes, attributes);\n\n if (typeof attributes.expires === 'number') {\n attributes.expires = new Date(Date.now() + attributes.expires * 864e5);\n }\n if (attributes.expires) {\n attributes.expires = attributes.expires.toUTCString();\n }\n\n name = encodeURIComponent(name)\n .replace(/%(2[346B]|5E|60|7C)/g, decodeURIComponent)\n .replace(/[()]/g, escape);\n\n var stringifiedAttributes = '';\n for (var attributeName in attributes) {\n if (!attributes[attributeName]) {\n continue\n }\n\n stringifiedAttributes += '; ' + attributeName;\n\n if (attributes[attributeName] === true) {\n continue\n }\n\n // Considers RFC 6265 section 5.2:\n // ...\n // 3. If the remaining unparsed-attributes contains a %x3B (\";\")\n // character:\n // Consume the characters of the unparsed-attributes up to,\n // not including, the first %x3B (\";\") character.\n // ...\n stringifiedAttributes += '=' + attributes[attributeName].split(';')[0];\n }\n\n return (document.cookie =\n name + '=' + converter.write(value, name) + stringifiedAttributes)\n }\n\n function get (name) {\n if (typeof document === 'undefined' || (arguments.length && !name)) {\n return\n }\n\n // To prevent the for loop in the first place assign an empty array\n // in case there are no cookies at all.\n var cookies = document.cookie ? document.cookie.split('; ') : [];\n var jar = {};\n for (var i = 0; i < cookies.length; i++) {\n var parts = cookies[i].split('=');\n var value = parts.slice(1).join('=');\n\n try {\n var found = decodeURIComponent(parts[0]);\n jar[found] = converter.read(value, found);\n\n if (name === found) {\n break\n }\n } catch (e) {}\n }\n\n return name ? jar[name] : jar\n }\n\n return Object.create(\n {\n set,\n get,\n remove: function (name, attributes) {\n set(\n name,\n '',\n assign({}, attributes, {\n expires: -1\n })\n );\n },\n withAttributes: function (attributes) {\n return init(this.converter, assign({}, this.attributes, attributes))\n },\n withConverter: function (converter) {\n return init(assign({}, this.converter, converter), this.attributes)\n }\n },\n {\n attributes: { value: Object.freeze(defaultAttributes) },\n converter: { value: Object.freeze(converter) }\n }\n )\n}\n\nvar api = init(defaultConverter, { path: '/' });\n/* eslint-enable no-var */\n\nexport { api as default };\n","import Cookies from 'js-cookie'\nimport { getAllowedDomain } from './urls'\n\nexport const CACHE_COOKIE_EVENTS = 'cached_analytics_events'\nexport const CACHE_COOKIE_PAGES = 'cached_analytics_page_views'\n\nexport type CachedEvent = {\n name: string\n properties: Record<string, unknown>\n timestamp: number\n}\n\nexport type CachedPageView = {\n name: string\n properties?: Record<string, unknown>\n timestamp: number\n}\n\nexport const cacheEventToCookie = (eventName: string, properties: Record<string, unknown>): void => {\n try {\n const domain = getAllowedDomain()\n const existingCache = Cookies.get(CACHE_COOKIE_EVENTS)\n const events: CachedEvent[] = existingCache ? JSON.parse(existingCache) : []\n events.push({ name: eventName, properties, timestamp: Date.now() })\n const cookieOptions: Cookies.CookieAttributes = { expires: 1 }\n if (domain) cookieOptions.domain = domain\n Cookies.set(CACHE_COOKIE_EVENTS, JSON.stringify(events), cookieOptions)\n } catch (err) {\n console.warn('Analytics: Failed to cache event', err)\n }\n}\n\nexport const cachePageViewToCookie = (pageName: string, properties?: Record<string, unknown>): void => {\n try {\n const domain = getAllowedDomain()\n const existingCache = Cookies.get(CACHE_COOKIE_PAGES)\n const pages: CachedPageView[] = existingCache ? JSON.parse(existingCache) : []\n pages.push({ name: pageName, properties, timestamp: Date.now() })\n const cookieOptions: Cookies.CookieAttributes = { expires: 1 }\n if (domain) cookieOptions.domain = domain\n Cookies.set(CACHE_COOKIE_PAGES, JSON.stringify(pages), cookieOptions)\n } catch (err) {\n console.warn('Analytics: Failed to cache page view', err)\n }\n}\n\nexport const getCachedEvents = (): CachedEvent[] => {\n try {\n const storedEventsString = Cookies.get(CACHE_COOKIE_EVENTS)\n if (storedEventsString) {\n const events = JSON.parse(storedEventsString)\n return Array.isArray(events) ? events : []\n }\n } catch (err) {\n console.warn('Analytics: Failed to get cached events', err)\n }\n return []\n}\n\nexport const getCachedPageViews = (): CachedPageView[] => {\n try {\n const storedPagesString = Cookies.get(CACHE_COOKIE_PAGES)\n if (storedPagesString) {\n const pages = JSON.parse(storedPagesString)\n return Array.isArray(pages) ? pages : []\n }\n } catch (err) {\n console.warn('Analytics: Failed to get cached pages', err)\n }\n return []\n}\n\nexport const clearCachedEvents = (): void => {\n const domain = getAllowedDomain()\n const cookieOptions = domain ? { domain } : {}\n Cookies.remove(CACHE_COOKIE_EVENTS, cookieOptions)\n}\n\nexport const clearCachedPageViews = (): void => {\n const domain = getAllowedDomain()\n const cookieOptions = domain ? { domain } : {}\n Cookies.remove(CACHE_COOKIE_PAGES, cookieOptions)\n}\n","import Cookies from 'js-cookie'\nimport { cloudflareTrace } from './urls'\n\nexport const isUUID = (str: string): boolean => {\n const uuidRegex = /^[0-9a-f]{8}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{12}$/i\n return uuidRegex.test(str)\n}\n\ntype TraceData = {\n loc?: string\n}\n\nlet countryPromise: Promise<string> | null = null\n\n/**\n * Fetches the country information based on Cloudflare's trace data or a fallback from cookies.\n * This function attempts to retrieve the country location by first fetching trace data from Cloudflare\n * and then falling back to the location stored in the cookies if the fetch fails.\n *\n * @returns {Promise<string>} A Promise that resolves to a string representing the country code in lowercase.\n * Returns an empty string if no country data is available or if an error occurs.\n *\n * @example\n * // Returns the country code in lowercase based on Cloudflare's trace data or cookies.\n * getCountry().then(country => console.log(country));\n */\nexport const getCountry = async (): Promise<string> => {\n if (countryPromise) return countryPromise\n\n const cookieCountry = JSON.parse(Cookies.get('website_status') || '{}')?.clients_country\n\n countryPromise = (async () => {\n try {\n const response = await fetch(cloudflareTrace).catch(() => null)\n if (!response) return cookieCountry || ''\n\n const text = await response.text().catch(() => '')\n if (!text) return cookieCountry || ''\n\n const data: TraceData = Object.fromEntries(text.split('\\n').map(v => v.split('=', 2)))\n return data.loc?.toLowerCase() || cookieCountry || ''\n } catch {\n return cookieCountry || ''\n }\n })()\n\n return countryPromise\n}\n\n/**\n * Recursively cleans an object by removing undefined, null, empty strings, empty objects, and empty arrays\n * Used to sanitize event properties before sending to analytics providers\n *\n * @param obj - The object to clean\n * @returns The cleaned object, or undefined if the result would be empty\n */\nexport const cleanObject = (obj: any): any => {\n if (obj == null || typeof obj !== 'object') return obj\n\n if (Array.isArray(obj)) {\n const cleanedArr = obj.map(cleanObject).filter(v => v !== undefined && v !== null)\n return cleanedArr.length ? cleanedArr : undefined\n }\n\n const cleaned: Record<string, any> = {}\n Object.entries(obj).forEach(([key, value]) => {\n const v = cleanObject(value)\n if (\n v === undefined ||\n v === null ||\n v === '' ||\n (typeof v === 'object' && !Array.isArray(v) && Object.keys(v).length === 0) ||\n (Array.isArray(v) && v.length === 0)\n ) {\n return\n }\n\n cleaned[key] = v\n })\n\n return Object.keys(cleaned).length ? cleaned : undefined\n}\n\n/**\n * Flattens a nested object structure into a single-level object\n * Lifts all nested properties to the top level without prefixing\n *\n * @param obj - The object to flatten\n * @returns A flattened object with all nested properties at the top level\n *\n * @example\n * flattenObject({ action: 'click', event_metadata: { version: 2, user_language: 'en' } })\n * // Returns: { action: 'click', version: 2, user_language: 'en' }\n *\n * flattenObject({ form_name: 'signup', cta_information: { cta_name: 'signup', section_name: 'header' } })\n * // Returns: { form_name: 'signup', cta_name: 'signup', section_name: 'header' }\n */\nexport const flattenObject = (obj: any): Record<string, any> => {\n if (obj == null || typeof obj !== 'object' || Array.isArray(obj)) {\n return obj\n }\n\n const flattened: Record<string, any> = {}\n\n Object.entries(obj).forEach(([key, value]) => {\n if (value && typeof value === 'object' && !Array.isArray(value)) {\n // Recursively flatten nested objects and merge them at the top level\n Object.assign(flattened, flattenObject(value))\n } else {\n flattened[key] = value\n }\n })\n\n return flattened\n}\n","import { RudderStack } from './providers/rudderstack'\nimport type { TCoreAttributes, TAllEvents, TV2EventPayload } from './types'\nimport {\n cacheEventToCookie,\n cachePageViewToCookie,\n getCachedEvents,\n getCachedPageViews,\n clearCachedEvents,\n clearCachedPageViews,\n} from './utils/cookie'\nimport { isUUID, getCountry, cleanObject, flattenObject } from './utils/helpers'\n\n// Optional Growthbook types - only import if using Growthbook\nimport type { Growthbook, GrowthbookConfigs } from './providers/growthbook'\nimport type { TGrowthbookAttributes, TGrowthbookOptions } from './providers/growthbookTypes'\n\n// Optional Posthog types - only import if using Posthog\nimport type { Posthog } from './providers/posthog'\nimport type { TPosthogOptions } from './providers/posthogTypes'\n\ndeclare global {\n interface Window {\n AnalyticsInstance: ReturnType<typeof createAnalyticsInstance>\n }\n}\n\n/**\n * Configuration options for initializing the analytics instance\n */\ntype Options = {\n /** GrowthBook client API key for A/B testing and feature flags */\n growthbookKey?: string\n /** GrowthBook decryption key for encrypted feature payloads */\n growthbookDecryptionKey?: string\n /** RudderStack write key for event tracking */\n rudderstackKey?: string\n /** Additional configuration options for GrowthBook */\n growthbookOptions?: TGrowthbookOptions\n /** PostHog configuration options including API keys and settings */\n posthogOptions?: TPosthogOptions\n}\n\n/**\n * Creates a unified analytics instance that integrates RudderStack and GrowthBook.\n *\n * This function provides a centralized interface for:\n * - Event tracking across multiple analytics platforms\n * - A/B testing and feature flag management via GrowthBook\n * - Offline event caching with automatic replay\n *\n * @param {Options} _options - Optional initialization configuration\n * @returns {Object} Analytics instance with methods for tracking, identification, and feature management\n *\n * @example\n * ```typescript\n * const analytics = createAnalyticsInstance();\n *\n * // Initialize with providers\n * await analytics.initialise({\n * rudderstackKey: 'YOUR_RS_KEY',\n * growthbookKey: 'YOUR_GB_KEY',\n * growthbookDecryptionKey: 'YOUR_GB_DECRYPT_KEY'\n * });\n *\n * // Set user attributes\n * analytics.setAttributes({\n * user_id: 'user123',\n * country: 'US',\n * user_language: 'en'\n * });\n *\n * // Track events\n * analytics.trackEvent('button_clicked', { button_name: 'signup' });\n *\n * // Track page views\n * analytics.pageView('/dashboard', 'Deriv App');\n * ```\n */\nexport function createAnalyticsInstance(_options?: Options) {\n let _growthbook: Growthbook | undefined,\n _rudderstack: RudderStack,\n _posthog: Posthog | undefined,\n core_data: Partial<TCoreAttributes> = {},\n tracking_config: { [key: string]: boolean } = {},\n offline_event_cache: Array<{ event: keyof TAllEvents; payload: TAllEvents[keyof TAllEvents] }> = [],\n _pending_identify_calls: Array<{ userId: string; traits?: Record<string, any> }> = [],\n _cookie_cache_processed = false\n\n const processCookieCache = () => {\n if (_cookie_cache_processed) return\n if (!_rudderstack?.has_initialized) return\n\n _cookie_cache_processed = true\n\n try {\n const storedEvents = getCachedEvents()\n if (storedEvents.length > 0) {\n storedEvents.forEach(event => {\n _rudderstack?.track(event.name as keyof TAllEvents, event.properties as any)\n })\n clearCachedEvents()\n }\n\n const storedPages = getCachedPageViews()\n if (storedPages.length > 0) {\n storedPages.forEach(page => {\n _rudderstack?.pageView(page.name, 'Deriv App', getId(), page.properties)\n })\n clearCachedPageViews()\n }\n } catch (err) {\n console.warn('Analytics: Failed to process cookie cache', err)\n }\n }\n\n const onSdkLoaded = () => {\n processCookieCache()\n\n _pending_identify_calls.forEach(({ userId, traits }) => {\n if (userId) {\n _rudderstack?.identifyEvent(userId, traits)\n }\n })\n _pending_identify_calls = []\n }\n\n /**\n * Initializes the analytics instance with specified provider configurations.\n * This method should be called before tracking any events.\n *\n * Features:\n * - Lazy-loads providers (GrowthBook, PostHog) only when configured\n * - Automatically fetches user's country for GrowthBook targeting\n * - Processes any cached events from previous sessions\n * - Sets up event tracking callback for GrowthBook experiments\n *\n * @param {Options} options - Configuration options for analytics providers\n * @returns {Promise<void>} Resolves when initialization is complete\n *\n * @example\n * ```typescript\n * await analytics.initialise({\n * rudderstackKey: 'YOUR_RS_KEY',\n * growthbookKey: 'YOUR_GB_KEY',\n * growthbookDecryptionKey: 'YOUR_GB_DECRYPT_KEY',\n * posthogOptions: {\n * apiKey: 'YOUR_POSTHOG_API_KEY',\n * config: {\n * session_recording: {\n * recordCrossOriginIframes: true,\n * minimumDurationMilliseconds: 30000\n * }\n * }\n * }\n * });\n * ```\n */\n const initialise = async ({\n growthbookKey,\n growthbookDecryptionKey,\n rudderstackKey,\n growthbookOptions,\n posthogOptions,\n }: Options) => {\n try {\n // Only fetch country if GrowthBook is enabled and country not provided\n const country = growthbookOptions?.attributes?.country || (growthbookKey ? await getCountry() : undefined)\n\n if (rudderstackKey) {\n _rudderstack = RudderStack.getRudderStackInstance(rudderstackKey, onSdkLoaded)\n }\n\n if (growthbookOptions?.attributes && Object.keys(growthbookOptions.attributes).length > 0) {\n const attrs = growthbookOptions.attributes\n const anonymousId = _rudderstack?.getAnonymousId()\n\n core_data = {\n ...core_data,\n country,\n ...(attrs.user_language && { user_language: attrs.user_language }),\n ...(attrs.account_type && { account_type: attrs.account_type }),\n ...(attrs.app_id && { app_id: attrs.app_id }),\n ...(attrs.residence_country && { residence_country: attrs.residence_country }),\n ...(attrs.device_type && { device_type: attrs.device_type }),\n ...(attrs.url && { url: attrs.url }),\n ...(attrs.email_hash && { email_hash: attrs.email_hash }),\n ...(attrs.network_type && { network_type: attrs.network_type }),\n ...(attrs.network_rtt && { network_rtt: attrs.network_rtt }),\n ...(attrs.network_downlink && { network_downlink: attrs.network_downlink }),\n ...(attrs.account_currency && { account_currency: attrs.account_currency }),\n ...(attrs.account_mode && { account_mode: attrs.account_mode }),\n loggedIn: !!attrs.loggedIn,\n ...(attrs.user_id && !isUUID(attrs.user_id) && { user_id: attrs.user_id }),\n ...(anonymousId && { anonymous_id: anonymousId }),\n }\n }\n\n growthbookOptions ??= {}\n growthbookOptions.attributes ??= {}\n const anonId = _rudderstack?.getAnonymousId()\n growthbookOptions.attributes.id ??= anonId\n growthbookOptions.attributes.country ??= country\n\n if (growthbookKey) {\n // Dynamically import Growthbook only when needed\n const { Growthbook } = await import('./providers/growthbook')\n _growthbook = Growthbook.getGrowthBookInstance(\n growthbookKey,\n growthbookDecryptionKey,\n growthbookOptions\n )\n\n const interval = setInterval(() => {\n if (Object.keys(tracking_config).length > 0) clearInterval(interval)\n else tracking_config = getFeatureValue('tracking-buttons-config', {}) as { [key: string]: boolean }\n }, 1000)\n }\n\n if (posthogOptions) {\n // Dynamically import Posthog only when needed\n const { Posthog } = await import('./providers/posthog')\n _posthog = Posthog.getPosthogInstance(posthogOptions)\n }\n } catch (err) {\n console.warn('Analytics: Failed to initialize', err)\n }\n }\n\n /**\n * Sets user and context attributes for analytics tracking and targeting.\n * These attributes are automatically included in all subsequent events.\n *\n * Attributes are used for:\n * - Event enrichment (added to all tracked events)\n * - GrowthBook targeting (feature flags and A/B tests)\n * - User segmentation across analytics platforms\n *\n * @param {TCoreAttributes} attributes - User and context attributes\n *\n * @example\n * ```typescript\n * analytics.setAttributes({\n * user_id: 'CR123456',\n * country: 'US',\n * user_language: 'en',\n * device_type: 'desktop',\n * account_type: 'real',\n * loggedIn: true\n * });\n * ```\n */\n const setAttributes = ({\n country,\n user_language,\n device_language,\n device_type,\n account_type,\n user_id,\n anonymous_id,\n app_id,\n utm_source,\n utm_medium,\n utm_campaign,\n is_authorised,\n residence_country,\n url,\n domain,\n geo_location,\n loggedIn,\n network_downlink,\n network_rtt,\n network_type,\n account_currency,\n account_mode,\n }: TCoreAttributes) => {\n const user_identity = user_id ?? getId()\n\n if (_growthbook) {\n const config: TGrowthbookAttributes = {\n country,\n residence_country,\n user_language,\n device_language,\n device_type,\n utm_source,\n utm_medium,\n utm_campaign,\n is_authorised,\n url,\n domain,\n loggedIn,\n ...(user_id && !isUUID(user_id) && { user_id }),\n anonymous_id,\n }\n if (user_identity) {\n config.id = user_identity\n config.user_id = user_identity\n }\n _growthbook.setAttributes(config)\n }\n\n core_data = {\n ...core_data,\n ...(country !== undefined && { country }),\n ...(geo_location !== undefined && { geo_location }),\n ...(user_language !== undefined && { user_language }),\n ...(account_type !== undefined && { account_type }),\n ...(app_id !== undefined && { app_id }),\n ...(residence_country !== undefined && { residence_country }),\n ...(device_type !== undefined && { device_type }),\n ...(url !== undefined && { url }),\n ...(loggedIn !== undefined && { loggedIn }),\n ...(network_downlink !== undefined && { network_downlink }),\n ...(network_rtt !== undefined && { network_rtt }),\n ...(network_type !== undefined && { network_type }),\n ...(user_id !== undefined && !isUUID(user_id) && { user_id }),\n ...(anonymous_id !== undefined && { anonymous_id }),\n ...(account_currency !== undefined && { account_currency }),\n ...(account_mode !== undefined && { account_mode }),\n }\n }\n\n const getFeatureState = (id: string) => _growthbook?.getFeatureState(id)?.experimentResult?.name\n\n const getFeatureValue = <K extends keyof GrowthbookConfigs, V extends GrowthbookConfigs[K]>(\n id: K,\n defaultValue: V\n ) => _growthbook?.getFeatureValue(id as string, defaultValue)\n\n const getGrowthbookStatus = async () => await _growthbook?.getStatus()\n const isFeatureOn = (key: string) => _growthbook?.isOn(key)\n const setUrl = (href: string) => _growthbook?.setUrl(href)\n\n const getId = () => {\n const userId = _rudderstack?.getUserId() || ''\n return userId && !isUUID(userId) ? userId : ''\n }\n\n const getAnonymousId = () => _rudderstack?.getAnonymousId() || ''\n\n /**\n * Tracks a page view event.\n *\n * Features:\n * - Automatically includes user ID if available\n * - Caches page views when offline or provider not initialized\n *\n * @param {string} current_page - The current page URL or path\n * @param {string} [platform='Deriv App'] - The platform name\n * @param {Record<string, unknown>} [properties] - Additional page properties\n *\n * @example\n * ```typescript\n * analytics.pageView('/dashboard');\n * analytics.pageView('/trade', 'Deriv Trader', { section: 'multipliers' });\n * ```\n */\n const pageView = (current_page: string, platform = 'Deriv App', properties?: Record<string, unknown>) => {\n const userId = getId()\n\n // Handle RudderStack pageView independently\n if (_rudderstack) {\n if (_rudderstack.has_initialized) {\n _rudderstack.pageView(current_page, platform, userId, properties)\n } else {\n cachePageViewToCookie(current_page, { platform, ...properties })\n }\n }\n\n // PostHog handles page views automatically via autocapture\n // No need to manually send page views to PostHog\n }\n\n /**\n * Identifies a user across analytics platforms.\n * This method should be called after user login or when user identity is known.\n *\n * Features:\n * - Queues identify calls if provider not yet initialized\n * - Allows custom traits for each provider or shared traits for both\n * - Identifies user in PostHog if configured\n *\n * @param {string} [user_id] - The user ID to identify. If not provided, uses stored user ID\n * @param {Record<string, any>} [traits] - Optional traits to send to both providers, or provider-specific traits\n *\n * @example\n * ```typescript\n * // Simple identify\n * analytics.identifyEvent('CR123456');\n *\n * // Identify with same traits for both providers\n * analytics.identifyEvent('CR123456', {\n * language: 'en',\n * country_of_residence: 'US'\n * });\n *\n * // Identify with provider-specific traits\n * analytics.identifyEvent('CR123456', {\n * rudderstack: { language: 'en', custom_field: 'value' },\n * posthog: { language: 'en', country_of_residence: 'US' }\n * });\n * ```\n */\n const identifyEvent = (user_id?: string, traits?: Record<string, any>) => {\n const stored_user_id = user_id || getId()\n if (!stored_user_id) return\n\n // Check if traits has provider-specific structure\n const hasProviderStructure = traits?.rudderstack !== undefined || traits?.posthog !== undefined\n const rudderstackTraits = hasProviderStructure ? traits?.rudderstack : traits\n const posthogTraits = hasProviderStructure ? traits?.posthog : traits\n\n // Handle RudderStack identification independently\n if (_rudderstack) {\n if (_rudderstack.has_initialized) {\n _rudderstack.identifyEvent(stored_user_id, rudderstackTraits)\n } else {\n if (!_pending_identify_calls.some(call => call.userId === stored_user_id)) {\n _pending_identify_calls.push({ userId: stored_user_id, traits: rudderstackTraits })\n }\n }\n }\n\n // Handle PostHog identification independently\n if (_posthog?.has_initialized) {\n _posthog.identifyEvent(stored_user_id, posthogTraits)\n }\n }\n\n const reset = () => {\n // Reset each provider independently\n if (_rudderstack?.has_initialized) {\n _rudderstack.reset()\n }\n if (_posthog?.has_initialized) {\n _posthog.reset()\n }\n }\n\n const isV2Payload = (payload: any): payload is TV2EventPayload => {\n return 'event_metadata' in payload || 'cta_information' in payload || 'error' in payload\n }\n\n /**\n * Tracks a custom event with associated data.\n *\n * Features:\n * - Automatically enriches events with core attributes\n * - Supports both V1 and V2 event payload formats\n * - RudderStack: Caches events when offline or not initialized\n * - PostHog: Sends immediately if initialized (has built-in caching)\n * - Respects feature flag configurations\n * - Each provider works independently - one failing won't affect the other\n *\n * @template T - The event name type from TAllEvents\n * @param {T} event - The event name to track\n * @param {TAllEvents[T]} analytics_data - The event data payload\n *\n * @example\n * ```typescript\n * // Simple event\n * analytics.trackEvent('button_clicked', { button_name: 'signup' });\n *\n * // V2 event with metadata\n * analytics.trackEvent('form_submitted', {\n * event_metadata: { form_name: 'registration' },\n * cta_information: { button_text: 'Create Account' }\n * });\n * ```\n */\n const trackEvent = <T extends keyof TAllEvents>(event: T, analytics_data: TAllEvents[T]) => {\n const userId = getId()\n let final_payload: any = {}\n\n if (isV2Payload(analytics_data)) {\n const v2_data = analytics_data as TV2EventPayload\n final_payload = {\n ...v2_data,\n event_metadata: {\n ...core_data,\n ...(userId && !core_data.user_id && { user_id: userId }),\n ...v2_data.event_metadata,\n },\n }\n } else {\n final_payload = {\n ...core_data,\n ...analytics_data,\n ...(userId && !core_data.user_id && { user_id: userId }),\n }\n }\n\n const shouldTrack = !(event in tracking_config) || tracking_config[event as string]\n if (!shouldTrack) return\n\n // Handle RudderStack independently\n const hasRudderstackInitialized = _rudderstack?.has_initialized\n if (!navigator.onLine || !hasRudderstackInitialized) {\n if (!hasRudderstackInitialized) {\n cacheEventToCookie(event as string, final_payload)\n } else {\n offline_event_cache.push({ event, payload: final_payload })\n }\n } else {\n // Send cached events to RudderStack\n if (offline_event_cache.length > 0) {\n offline_event_cache.forEach(cache => {\n const cleaned_cache_payload = cleanObject(cache.payload)\n _rudderstack?.track(cache.event, cleaned_cache_payload)\n })\n offline_event_cache = []\n }\n\n // Send current event to RudderStack\n const cleaned_payload = cleanObject(final_payload)\n _rudderstack?.track(event, cleaned_payload)\n }\n\n // Handle PostHog independently - send immediately if initialized\n if (_posthog?.has_initialized) {\n const flattened_payload = flattenObject(final_payload)\n const cleaned_posthog_payload = cleanObject(flattened_payload)\n _posthog.capture(event as string, cleaned_posthog_payload)\n }\n }\n\n const getInstances = () => ({ ab: _growthbook, tracking: _rudderstack, posthog: _posthog })\n\n const AnalyticsInstance = {\n initialise,\n setAttributes,\n identifyEvent,\n getFeatureState,\n getFeatureValue,\n getGrowthbookStatus,\n isFeatureOn,\n setUrl,\n getId,\n getAnonymousId,\n trackEvent,\n getInstances,\n pageView,\n reset,\n }\n\n if (typeof window !== 'undefined') {\n window.AnalyticsInstance = AnalyticsInstance\n }\n\n return AnalyticsInstance\n}\n\nexport const Analytics = createAnalyticsInstance()\n","/**\n * Analytics Cache Manager - Version 1.1.0\n * Enhanced TypeScript implementation with better type safety and SSR support\n */\n\ntype CachedEvent = {\n name: string\n properties: Record<string, any>\n}\n\ntype ResponseData = {\n url: string\n method: string\n status: number\n headers: string\n data: string\n payload: any\n}\n\ntype EventListenerConfig = {\n element: Element | NodeList | string\n event: {\n name: string\n properties: Record<string, any>\n }\n cache?: boolean\n callback?: (e: Event) => { name: string; properties: Record<string, any>; cache?: boolean }\n}\n\ntype LoadEventConfig = {\n event: {\n name: string\n properties: Record<string, any>\n }\n}\n\ntype PageLoadEventConfig = {\n pages?: string[]\n excludedPages?: string[]\n event: {\n name: string\n properties: Record<string, any>\n }\n callback?: () => { name: string; properties: Record<string, any> }\n}\n\nclass AnalyticsCacheManager {\n private interval: NodeJS.Timeout | null = null\n private responses: ResponseData[] = []\n private isTrackingResponses = false\n private delegatedSelectors: Set<string> = new Set()\n\n /**\n * FNV-1a hash algorithm for creating consistent hashes\n */\n private hash(inputString: string, desiredLength = 32): string {\n const fnv1aHash = (string: string): string => {\n let hash = 0x811c9dc5\n for (let i = 0; i < string.length; i++) {\n hash ^= string.charCodeAt(i)\n hash = (hash * 0x01000193) >>> 0\n }\n return hash.toString(16)\n }\n\n const base64Encode = (string: string): string => btoa(string)\n\n let hash = fnv1aHash(inputString)\n let combined = base64Encode(hash)\n\n while (combined.length < desiredLength) {\n combined += base64Encode(fnv1aHash(combined))\n }\n\n return combined.substring(0, desiredLength)\n }\n\n /**\n * Get cookie value by name\n */\n getCookies(name: string): any {\n const value = `; ${document.cookie}`\n const parts = value.split(`; ${name}=`)\n if (parts.length === 2) {\n const cookieValue = decodeURIComponent(parts.pop()!.split(';').shift()!)\n\n try {\n return JSON.parse(cookieValue)\n } catch (e) {\n return cookieValue\n }\n }\n return null\n }\n\n /**\n * Track page unload events to cache pageviews before leaving\n */\n trackPageUnload(): void {\n if (typeof window === 'undefined') return\n\n window.addEventListener('beforeunload', () => {\n if (!this.isPageViewSent()) {\n this.push('cached_analytics_page_views', {\n name: window.location.href,\n properties: {\n url: window.location.href,\n },\n })\n }\n })\n }\n\n /**\n * Track XMLHttpRequest responses to monitor analytics calls\n */\n trackResponses(): void {\n if (typeof window === 'undefined' || typeof XMLHttpRequest === 'undefined') return\n\n const originalXhrOpen = XMLHttpRequest.prototype.open\n const originalXhrSend = XMLHttpRequest.prototype.send\n\n XMLHttpRequest.prototype.open = function (method: string, url: string | URL) {\n ;(this as any)._url = url\n ;(this as any)._method = method\n return originalXhrOpen.apply(this, arguments as any)\n }\n\n XMLHttpRequest.prototype.send = function (body?: Document | XMLHttpRequestBodyInit | null) {\n this.addEventListener('load', function () {\n let parsedPayload = null\n\n if (typeof body === 'string') {\n try {\n parsedPayload = JSON.parse(body)\n } catch (e) {\n parsedPayload = body\n }\n }\n\n const responseData: ResponseData = {\n url: (this as any)._url,\n method: (this as any)._method,\n status: this.status,\n headers: this.getAllResponseHeaders(),\n data: this.responseText,\n payload: parsedPayload,\n }\n cacheTrackEvents.responses.push(responseData)\n })\n\n return originalXhrSend.apply(this, arguments as any)\n }\n\n this.isTrackingResponses = true\n }\n\n /**\n * Check if Analytics instance is ready\n */\n isReady(): boolean {\n if (typeof window === 'undefined') return false\n\n const Analytics = (window as any).Analytics\n if (typeof Analytics === 'undefined' || Analytics === null) {\n return false\n }\n\n const instances = Analytics.Analytics?.getInstances?.()\n return !!instances?.tracking\n }\n\n /**\n * Parse cookies into an object\n */\n private parseCookies(cookieName: string): any {\n if (typeof document === 'undefined') return null\n\n const cookies = document.cookie.split('; ').reduce((acc: Record<string, string>, cookie) => {\n const [key, value] = cookie.split('=')\n if (key && value) {\n acc[decodeURIComponent(key)] = decodeURIComponent(value)\n }\n return acc\n }, {})\n\n try {\n return cookies[cookieName] ? JSON.parse(cookies[cookieName]) : null\n } catch (error) {\n return null\n }\n }\n\n /**\n * Check if pageview has been sent\n */\n isPageViewSent(): boolean {\n return !!this.responses.find(e => e.payload?.type === 'page' && e.payload?.anonymousId)\n }\n\n /**\n * Set a cached event\n */\n set(event: CachedEvent): void {\n this.push('cached_analytics_events', event)\n }\n\n /**\n * Push data to cookie cache\n */\n push(cookieName: string, data: any): void {\n if (typeof document === 'undefined') return\n\n let storedCookies: any[] = []\n const cacheCookie = this.parseCookies(cookieName)\n if (cacheCookie && Array.isArray(cacheCookie)) {\n storedCookies = cacheCookie\n }\n\n storedCookies.push(data)\n\n const domain = this.getAllowedDomain()\n const maxAge = 365 * 24 * 60 * 60 // 1 year\n const cookieString = `${cookieName}=${encodeURIComponent(JSON.stringify(storedCookies))}; path=/; Domain=${domain}; max-age=${maxAge}; SameSite=Lax`\n\n document.cookie = cookieString\n }\n\n /**\n * Get the allowed domain for cookies\n * For localhost/single-part domains: use as-is\n * For multi-part domains: use top-level domain (e.g., .deriv.com)\n */\n private getAllowedDomain(): string {\n if (typeof window === 'undefined') return ''\n\n const hostname = window.location.hostname\n\n // Handle IP addresses and localhost\n if (hostname === 'localhost' || /^\\d+\\.\\d+\\.\\d+\\.\\d+$/.test(hostname)) {\n return hostname\n }\n\n const parts = hostname.split('.')\n\n // Single part domain (e.g., \"localhost\")\n if (parts.length === 1) {\n return hostname\n }\n\n // Use top-level domain for proper subdomain sharing\n return `.${parts.slice(-2).join('.')}`\n }\n\n /**\n * Process event to hash email and add client info\n */\n processEvent(event: CachedEvent): CachedEvent {\n const clientInfo = this.getCookies('client_information')\n\n if (clientInfo) {\n const { email = null } = clientInfo\n\n if (email) {\n event.properties.email_hash = this.hash(email)\n }\n }\n\n if (event?.properties?.email) {\n const email = event.properties.email\n delete event.properties.email\n event.properties.email_hash = this.hash(email)\n }\n\n return event\n }\n\n /**\n * Track an event (either immediately or cache it)\n */\n track(originalEvent: CachedEvent, cache = false): void {\n if (typeof window === 'undefined') return\n\n const event = this.processEvent(originalEvent)\n const Analytics = (window as any).Analytics\n\n if (this.isReady() && !cache) {\n Analytics.Analytics.trackEvent(event.name, event.properties)\n } else {\n this.set(event)\n }\n }\n\n /**\n * Track pageview with auto-retry until sent\n */\n pageView(): void {\n if (typeof window === 'undefined') return\n\n if (!this.isTrackingResponses) {\n this.trackResponses()\n this.trackPageUnload()\n }\n\n this.interval = setInterval(() => {\n const Analytics = (window as any).Analytics\n\n if (\n typeof Analytics !== 'undefined' &&\n typeof Analytics.Analytics?.pageView === 'function' &&\n this.isReady()\n ) {\n Analytics.Analytics.pageView(window.location.href, \"Trader's hub\")\n }\n\n if (this.isPageViewSent()) {\n if (this.interval) clearInterval(this.interval)\n }\n }, 1000)\n }\n\n /**\n * Add click event listener to element(s)\n */\n listen(\n element: Element | NodeList,\n { name = '', properties = {} }: { name: string; properties: Record<string, any> },\n cache = false,\n callback: ((e: Event) => { name: string; properties: Record<string, any>; cache?: boolean }) | null = null\n ): void {\n const addClickListener = (el: Element) => {\n if (!(el as any).dataset?.clickEventTracking) {\n el.addEventListener('click', (e: Event) => {\n let event = {\n name,\n properties,\n cache,\n }\n\n if (typeof callback === 'function') {\n const callbackResult = callback(e)\n event = {\n ...callbackResult,\n cache: callbackResult.cache ?? cache,\n }\n }\n\n this.track(event, event.cache ?? false)\n })\n ;(el as any).dataset.clickEventTracking = 'true'\n }\n }\n\n const elements = element instanceof NodeList ? Array.from(element as NodeListOf<Element>) : [element]\n elements.forEach(addClickListener)\n }\n\n /**\n * Add event handlers to multiple elements with auto-retry\n * Alias for backward compatibility with typo\n */\n addEventHandler(items: EventListenerConfig[]): this {\n if (typeof window === 'undefined') return this\n\n items.forEach(({ element, event = { name: '', properties: {} }, cache = false, callback = null }) => {\n // If a selector string is provided, use event delegation on document\n if (typeof element === 'string') {\n const selector = element\n\n if (!this.delegatedSelectors.has(selector)) {\n const delegatedHandler = (e: Event) => {\n const target = e.target as Element | null\n if (!target) return\n\n const matched = target.closest(selector)\n if (matched && !(matched as any).dataset?.clickEventTracking) {\n let evt: any = {\n name: event.name,\n properties: event.properties,\n cache,\n }\n\n if (typeof callback === 'function') {\n const callbackResult = callback(e)\n evt = {\n ...callbackResult,\n cache: callbackResult.cache ?? cache,\n }\n }\n\n ;(matched as any).dataset.clickEventTracking = 'true'\n this.track(evt, evt.cache ?? false)\n }\n }\n\n document.addEventListener('click', delegatedHandler)\n this.delegatedSelectors.add(selector)\n }\n } else {\n // Element or NodeList: attach directly to existing elements\n const elements = element instanceof NodeList ? Array.from(element) : [element]\n\n elements.forEach(el => {\n if (el && !(el as any).dataset?.clickEventTracking) {\n this.listen(el as Element, event, cache, callback)\n }\n })\n }\n })\n\n return this\n }\n\n /**\n * Backward compatibility alias (with typo from original)\n */\n addEventhandler = this.addEventHandler.bind(this)\n\n /**\n * Load events immediately\n */\n loadEvent(items: LoadEventConfig[]): this {\n items.forEach(({ event }) => {\n const { name, properties } = event\n this.track({\n name,\n properties,\n })\n })\n\n return this\n }\n\n /**\n * Load events on specific pages\n */\n pageLoadEvent(items: PageLoadEventConfig[]): this {\n if (typeof window === 'undefined') return this\n\n const pathname = window.location.pathname.slice(1)\n items.forEach(({ pages = [], excludedPages = [], event, callback = null }) => {\n let dispatch = false\n if (pages.length) {\n if (pages.includes(pathname)) {\n dispatch = true\n }\n } else if (excludedPages.length) {\n if (!excludedPages.includes(pathname)) {\n dispatch = true\n }\n } else {\n dispatch = true\n }\n\n if (dispatch) {\n const eventData = callback ? callback() : event\n this.loadEvent([{ event: eventData }])\n }\n })\n\n return this\n }\n\n /**\n * Clear the interval and cleanup\n */\n clearInterval(): void {\n if (this.interval) {\n clearInterval(this.interval)\n this.interval = null\n }\n }\n\n /**\n * Cleanup method for removing event listeners and intervals\n */\n cleanup(): void {\n this.clearInterval()\n this.responses = []\n this.isTrackingResponses = false\n }\n}\n\n// Create singleton instance\nexport const cacheTrackEvents = new AnalyticsCacheManager()\n\n// Export to global scope for backward compatibility\nif (typeof window !== 'undefined') {\n ;(window as any).cacheTrackEvents = cacheTrackEvents\n}\n"],"mappings":"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","names":["COOKIE_MAX_AGE_SECONDS","_RudderStack","RUDDERSTACK_KEY","onLoaded","RudderAnalytics","hostname","domain_name","domain","uuid","bytes","hex","byte","secureFlag","error","rudderstackDataplane","user_id","payload","currentUserId","current_page","platform","properties","pageProperties","event","err","RudderStack","assign","target","i","source","key","defaultConverter","value","init","converter","defaultAttributes","set","name","attributes","stringifiedAttributes","attributeName","get","cookies","jar","parts","found","api","CACHE_COOKIE_EVENTS","CACHE_COOKIE_PAGES","cacheEventToCookie","eventName","properties","domain","getAllowedDomain","existingCache","api","events","cookieOptions","err","cachePageViewToCookie","pageName","pages","getCachedEvents","storedEventsString","getCachedPageViews","storedPagesString","clearCachedEvents","clearCachedPageViews","isUUID","str","countryPromise","getCountry","cookieCountry","api","response","cloudflareTrace","text","v","cleanObject","obj","cleanedArr","cleaned","key","value","flattenObject","flattened","createAnalyticsInstance","_options","_growthbook","_rudderstack","_posthog","core_data","tracking_config","offline_event_cache","_pending_identify_calls","_cookie_cache_processed","processCookieCache","storedEvents","getCachedEvents","event","clearCachedEvents","storedPages","getCachedPageViews","page","getId","clearCachedPageViews","err","onSdkLoaded","userId","traits","initialise","growthbookKey","growthbookDecryptionKey","rudderstackKey","growthbookOptions","posthogOptions","_a","_b","country","getCountry","RudderStack","attrs","anonymousId","isUUID","anonId","Growthbook","interval","getFeatureValue","Posthog","setAttributes","user_language","device_language","device_type","account_type","user_id","anonymous_id","app_id","utm_source","utm_medium","utm_campaign","is_authorised","residence_country","url","domain","geo_location","loggedIn","network_downlink","network_rtt","network_type","account_currency","account_mode","user_identity","config","getFeatureState","id","defaultValue","getGrowthbookStatus","isFeatureOn","key","setUrl","href","getAnonymousId","pageView","current_page","platform","properties","cachePageViewToCookie","identifyEvent","stored_user_id","hasProviderStructure","rudderstackTraits","posthogTraits","call","reset","isV2Payload","payload","AnalyticsInstance","analytics_data","final_payload","v2_data","hasRudderstackInitialized","cacheEventToCookie","cache","cleaned_cache_payload","cleanObject","cleaned_payload","flattened_payload","flattenObject","cleaned_posthog_payload","Analytics","AnalyticsCacheManager","inputString","desiredLength","fnv1aHash","string","hash","i","base64Encode","combined","name","parts","cookieValue","originalXhrOpen","originalXhrSend","method","url","body","parsedPayload","responseData","cacheTrackEvents","Analytics","cookieName","cookies","acc","cookie","key","value","event","data","storedCookies","cacheCookie","domain","maxAge","cookieString","hostname","clientInfo","email","originalEvent","cache","element","properties","callback","addClickListener","el","e","callbackResult","items","selector","delegatedHandler","target","matched","evt","pathname","pages","excludedPages","dispatch","eventData"]}
1
+ {"version":3,"sources":["../../src/providers/rudderstack.ts","../../src/utils/cookie.ts","../../src/utils/analytics-cache.ts","../../src/analytics.ts"],"sourcesContent":["import { RudderAnalytics } from '@rudderstack/analytics-js'\nimport type { TCoreAttributes, TAllEvents } from '../types'\nimport { rudderstackDataplane } from '../utils/urls'\nimport { createLogger } from '../utils/helpers'\n\n// Constants\nconst COOKIE_MAX_AGE_SECONDS = 2 * 365 * 24 * 60 * 60 // 2 years\n\n/**\n * RudderStack analytics wrapper with singleton pattern.\n * Handles user tracking, page views, and event analytics.\n */\nexport class RudderStack {\n analytics = new RudderAnalytics()\n has_identified = false\n has_initialized = false\n current_page = ''\n rudderstack_anonymous_cookie_key = 'rudder_anonymous_id'\n private static _instance: RudderStack\n private onLoadedCallback?: () => void\n private debug = false\n private log = createLogger('[RudderStack]', () => this.debug)\n\n constructor(RUDDERSTACK_KEY: string, onLoaded?: () => void, debug = false) {\n this.onLoadedCallback = onLoaded\n this.debug = debug\n this.init(RUDDERSTACK_KEY)\n }\n\n /**\n * Get or create the singleton instance of RudderStack\n * @param RUDDERSTACK_KEY - RudderStack write key\n * @param onLoaded - Optional callback when RudderStack is loaded\n * @param debug - Enable debug logging\n * @returns The RudderStack singleton instance\n */\n public static getRudderStackInstance = (\n RUDDERSTACK_KEY: string,\n onLoaded?: () => void,\n debug = false\n ): RudderStack => {\n if (!RudderStack._instance) {\n RudderStack._instance = new RudderStack(RUDDERSTACK_KEY, onLoaded, debug)\n }\n return RudderStack._instance\n }\n\n /**\n * Get the anonymous ID from cookies\n * @returns The anonymous ID or undefined if not found\n */\n getAnonymousId = (): string | undefined => {\n return document.cookie.match('(^|;)\\\\s*' + this.rudderstack_anonymous_cookie_key + '\\\\s*=\\\\s*([^;]+)')?.pop()\n }\n\n /**\n * Set anonymous ID cookie if it doesn't exist\n * Creates a secure cookie with proper domain and security attributes\n */\n setCookieIfNotExists = (): void => {\n const anonymous_id = this.getAnonymousId()\n\n if (!anonymous_id) {\n try {\n const hostname = window.location.hostname\n const external_domains = ['webflow.io']\n const is_external_domain = external_domains.some(domain => hostname.endsWith(domain))\n const domain_name = is_external_domain ? hostname : hostname.split('.').slice(-2).join('.')\n\n // Generate cryptographically secure UUID\n let uuid: string\n if (crypto?.randomUUID) {\n uuid = crypto.randomUUID()\n } else if (crypto?.getRandomValues) {\n // Fallback: Generate UUID v4 using crypto.getRandomValues\n const bytes = new Uint8Array(16)\n crypto.getRandomValues(bytes)\n // Set version (4) and variant bits\n bytes[6] = (bytes[6]! & 0x0f) | 0x40\n bytes[8] = (bytes[8]! & 0x3f) | 0x80\n // Convert to UUID string format\n const hex = Array.from(bytes, byte => byte.toString(16).padStart(2, '0')).join('')\n uuid = `${hex.slice(0, 8)}-${hex.slice(8, 12)}-${hex.slice(12, 16)}-${hex.slice(16, 20)}-${hex.slice(20)}`\n } else {\n // Crypto API not available - this should not happen in modern browsers\n throw new Error('Crypto API not available for secure random UUID generation')\n }\n\n const isSecure = window.location.protocol === 'https:'\n const secureFlag = isSecure ? '; Secure' : ''\n\n document.cookie = `${this.rudderstack_anonymous_cookie_key}=${uuid}; path=/; Domain=${domain_name}; max-age=${COOKIE_MAX_AGE_SECONDS}; SameSite=Lax${secureFlag}`\n } catch (error) {\n console.warn('RudderStack: Failed to set anonymous ID cookie', error)\n }\n }\n }\n\n /**\n * Get the current user ID\n * @returns The user ID, null, or undefined if not identified\n */\n getUserId = () => this.analytics.getUserId()\n\n /**\n * Initialize RudderStack with the provided key\n * @param RUDDERSTACK_KEY - RudderStack write key\n */\n init = (RUDDERSTACK_KEY: string): void => {\n if (!RUDDERSTACK_KEY) {\n console.warn('RudderStack: Initialization skipped - no key provided')\n return\n }\n\n this.log('init | loading SDK', { dataplane: rudderstackDataplane })\n\n try {\n this.setCookieIfNotExists()\n\n this.analytics.load(RUDDERSTACK_KEY, rudderstackDataplane, {\n externalAnonymousIdCookieName: this.rudderstack_anonymous_cookie_key,\n // Performance optimizations\n lockIntegrationsVersion: true,\n onLoaded: () => {\n this.has_initialized = true\n this.has_identified = !!this.getUserId()\n this.log('init | SDK loaded successfully', {\n userId: this.getUserId(),\n anonymousId: this.getAnonymousId(),\n })\n this.onLoadedCallback?.()\n },\n })\n } catch (error) {\n console.error('RudderStack: Failed to initialize', error)\n }\n }\n\n /**\n * Identify a user with RudderStack\n * Only identifies if user hasn't been identified yet\n * @param user_id - The user ID to identify\n * @param payload - Optional user traits (e.g., language, custom properties)\n */\n identifyEvent = (user_id: string, payload?: Record<string, any>): void => {\n if (!this.has_initialized) {\n console.warn('RudderStack: Cannot identify - not initialized')\n return\n }\n\n const currentUserId = this.getUserId()\n if (!currentUserId || currentUserId !== user_id) {\n try {\n this.log('identifyEvent | identifying user', { user_id, traits: payload })\n this.analytics.identify(user_id, payload || {})\n this.has_identified = true\n } catch (error) {\n console.error('RudderStack: Failed to identify user', error)\n }\n } else {\n this.log('identifyEvent | user already identified', { user_id })\n this.has_identified = true\n }\n }\n\n /**\n * Track a page view event\n * @param current_page - The page name/path\n * @param platform - The platform name (default: 'Deriv App')\n * @param user_id - The user ID\n * @param properties - Additional page properties\n */\n pageView = (\n current_page: string,\n platform = 'Deriv App',\n user_id: string,\n properties?: Record<string, unknown>\n ): void => {\n if (!this.has_initialized || current_page === this.current_page) return\n\n try {\n const pageProperties = {\n ...(user_id && { user_id }),\n ...properties,\n }\n\n this.log('pageView | tracking page view', { platform, current_page, properties: pageProperties })\n // Type assertion needed due to RudderStack's type definitions\n this.analytics.page(platform, current_page, pageProperties as any)\n this.current_page = current_page\n } catch (error) {\n console.error('RudderStack: Failed to track page view', error)\n }\n }\n\n /**\n * Reset the RudderStack instance\n * Clears user identification and resets tracking state\n */\n reset = (): void => {\n if (!this.has_initialized) return\n\n try {\n this.log('reset | resetting RudderStack session')\n this.analytics.reset()\n this.has_identified = false\n } catch (error) {\n console.error('RudderStack: Failed to reset', error)\n }\n }\n\n /**\n * Track a custom event with payload\n * Payload is pre-cleaned by analytics.ts using cleanObject before being passed here\n * @param event - The event name\n * @param payload - The event payload with core attributes\n */\n track = <T extends keyof TAllEvents>(event: T, payload: TAllEvents[T] & Partial<TCoreAttributes>): void => {\n if (!this.has_initialized) return\n\n try {\n this.log('track | sending event to RudderStack', { event, payload })\n // Type assertion needed to match RudderStack's ApiObject type\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n this.analytics.track(event as string, payload as any)\n } catch (err) {\n console.warn('RudderStack: Failed to track event', err)\n }\n }\n}\n","import Cookies from 'js-cookie'\nimport { getAllowedDomain } from './urls'\n\nexport const CACHE_COOKIE_EVENTS = 'cached_analytics_events'\nexport const CACHE_COOKIE_PAGES = 'cached_analytics_page_views'\n\nexport type CachedEvent = {\n name: string\n properties: Record<string, unknown>\n timestamp: number\n}\n\nexport type CachedPageView = {\n name: string\n properties?: Record<string, unknown>\n timestamp: number\n}\n\nexport const cacheEventToCookie = (eventName: string, properties: Record<string, unknown>): void => {\n try {\n const domain = getAllowedDomain()\n const existingCache = Cookies.get(CACHE_COOKIE_EVENTS)\n const events: CachedEvent[] = existingCache ? JSON.parse(existingCache) : []\n events.push({ name: eventName, properties, timestamp: Date.now() })\n const cookieOptions: Cookies.CookieAttributes = { expires: 1 }\n if (domain) cookieOptions.domain = domain\n Cookies.set(CACHE_COOKIE_EVENTS, JSON.stringify(events), cookieOptions)\n } catch (err) {\n console.warn('Analytics: Failed to cache event', err)\n }\n}\n\nexport const cachePageViewToCookie = (pageName: string, properties?: Record<string, unknown>): void => {\n try {\n const domain = getAllowedDomain()\n const existingCache = Cookies.get(CACHE_COOKIE_PAGES)\n const pages: CachedPageView[] = existingCache ? JSON.parse(existingCache) : []\n pages.push({ name: pageName, properties, timestamp: Date.now() })\n const cookieOptions: Cookies.CookieAttributes = { expires: 1 }\n if (domain) cookieOptions.domain = domain\n Cookies.set(CACHE_COOKIE_PAGES, JSON.stringify(pages), cookieOptions)\n } catch (err) {\n console.warn('Analytics: Failed to cache page view', err)\n }\n}\n\nexport const getCachedEvents = (): CachedEvent[] => {\n try {\n const storedEventsString = Cookies.get(CACHE_COOKIE_EVENTS)\n if (storedEventsString) {\n const events = JSON.parse(storedEventsString)\n return Array.isArray(events) ? events : []\n }\n } catch (err) {\n console.warn('Analytics: Failed to get cached events', err)\n }\n return []\n}\n\nexport const getCachedPageViews = (): CachedPageView[] => {\n try {\n const storedPagesString = Cookies.get(CACHE_COOKIE_PAGES)\n if (storedPagesString) {\n const pages = JSON.parse(storedPagesString)\n return Array.isArray(pages) ? pages : []\n }\n } catch (err) {\n console.warn('Analytics: Failed to get cached pages', err)\n }\n return []\n}\n\nexport const clearCachedEvents = (): void => {\n const domain = getAllowedDomain()\n const cookieOptions = domain ? { domain } : {}\n Cookies.remove(CACHE_COOKIE_EVENTS, cookieOptions)\n}\n\nexport const clearCachedPageViews = (): void => {\n const domain = getAllowedDomain()\n const cookieOptions = domain ? { domain } : {}\n Cookies.remove(CACHE_COOKIE_PAGES, cookieOptions)\n}\n","/**\n * Analytics Cache Manager - Version 1.1.0\n * Enhanced TypeScript implementation with better type safety and SSR support\n */\nimport { createLogger } from './helpers'\n\ntype CachedEvent = {\n name: string\n properties: Record<string, any>\n}\n\ntype ResponseData = {\n url: string\n method: string\n status: number\n headers: string\n data: string\n payload: any\n}\n\ntype EventListenerConfig = {\n element: Element | NodeList | string\n event: {\n name: string\n properties: Record<string, any>\n }\n cache?: boolean\n callback?: (e: Event) => { name: string; properties: Record<string, any>; cache?: boolean }\n}\n\ntype LoadEventConfig = {\n event: {\n name: string\n properties: Record<string, any>\n }\n}\n\ntype PageLoadEventConfig = {\n pages?: string[]\n excludedPages?: string[]\n event: {\n name: string\n properties: Record<string, any>\n }\n callback?: () => { name: string; properties: Record<string, any> }\n}\n\nclass AnalyticsCacheManager {\n private interval: NodeJS.Timeout | null = null\n private responses: ResponseData[] = []\n private isTrackingResponses = false\n private delegatedSelectors: Set<string> = new Set()\n private debug = false\n private log = createLogger('[CacheManager]', () => this.debug)\n\n setDebug(debug: boolean): void {\n this.debug = debug\n }\n\n /**\n * FNV-1a hash algorithm for creating consistent hashes\n */\n private hash(inputString: string, desiredLength = 32): string {\n const fnv1aHash = (string: string): string => {\n let hash = 0x811c9dc5\n for (let i = 0; i < string.length; i++) {\n hash ^= string.charCodeAt(i)\n hash = (hash * 0x01000193) >>> 0\n }\n return hash.toString(16)\n }\n\n const base64Encode = (string: string): string => btoa(string)\n\n let hash = fnv1aHash(inputString)\n let combined = base64Encode(hash)\n\n while (combined.length < desiredLength) {\n combined += base64Encode(fnv1aHash(combined))\n }\n\n return combined.substring(0, desiredLength)\n }\n\n /**\n * Get cookie value by name\n */\n getCookies(name: string): any {\n const value = `; ${document.cookie}`\n const parts = value.split(`; ${name}=`)\n if (parts.length === 2) {\n const cookieValue = decodeURIComponent(parts.pop()!.split(';').shift()!)\n\n try {\n return JSON.parse(cookieValue)\n } catch (e) {\n return cookieValue\n }\n }\n return null\n }\n\n /**\n * Track page unload events to cache pageviews before leaving\n */\n trackPageUnload(): void {\n if (typeof window === 'undefined') return\n\n window.addEventListener('beforeunload', () => {\n if (!this.isPageViewSent()) {\n this.push('cached_analytics_page_views', {\n name: window.location.href,\n properties: {\n url: window.location.href,\n },\n })\n }\n })\n }\n\n /**\n * Track XMLHttpRequest responses to monitor analytics calls\n */\n trackResponses(): void {\n if (typeof window === 'undefined' || typeof XMLHttpRequest === 'undefined') return\n\n const originalXhrOpen = XMLHttpRequest.prototype.open\n const originalXhrSend = XMLHttpRequest.prototype.send\n\n XMLHttpRequest.prototype.open = function (method: string, url: string | URL) {\n ;(this as any)._url = url\n ;(this as any)._method = method\n return originalXhrOpen.apply(this, arguments as any)\n }\n\n XMLHttpRequest.prototype.send = function (body?: Document | XMLHttpRequestBodyInit | null) {\n this.addEventListener('load', function () {\n let parsedPayload = null\n\n if (typeof body === 'string') {\n try {\n parsedPayload = JSON.parse(body)\n } catch (e) {\n parsedPayload = body\n }\n }\n\n const responseData: ResponseData = {\n url: (this as any)._url,\n method: (this as any)._method,\n status: this.status,\n headers: this.getAllResponseHeaders(),\n data: this.responseText,\n payload: parsedPayload,\n }\n cacheTrackEvents.responses.push(responseData)\n })\n\n return originalXhrSend.apply(this, arguments as any)\n }\n\n this.isTrackingResponses = true\n }\n\n /**\n * Resolve the analytics instance from the window object.\n * Supports three usage patterns:\n * - NPM import: analytics.ts sets window.AnalyticsInstance directly\n * - IIFE bundle: tsup sets window.DerivAnalytics = { Analytics, cacheTrackEvents }\n * - Legacy: consumers that explicitly set window.Analytics = window.DerivAnalytics\n */\n private getAnalyticsInstance(): any {\n if (typeof window === 'undefined') return null\n return (\n (window as any).AnalyticsInstance ??\n (window as any).DerivAnalytics?.Analytics ??\n (window as any).Analytics?.Analytics\n )\n }\n\n /**\n * Check if Analytics instance is ready\n */\n isReady(): boolean {\n if (typeof window === 'undefined') return false\n const instance = this.getAnalyticsInstance()\n if (!instance) return false\n return !!instance.getInstances?.()?.tracking\n }\n\n /**\n * Parse cookies to find a specific cookie by name.\n * Uses an early-exit linear scan instead of building a full map,\n * which is significantly faster when there are many cookies.\n */\n private parseCookies(cookieName: string): any {\n if (typeof document === 'undefined') return null\n\n const cookies = document.cookie.split('; ')\n for (const cookie of cookies) {\n const eqIdx = cookie.indexOf('=')\n if (eqIdx === -1) continue\n if (decodeURIComponent(cookie.slice(0, eqIdx)) === cookieName) {\n const raw = cookie.slice(eqIdx + 1)\n try {\n return JSON.parse(decodeURIComponent(raw))\n } catch {\n return decodeURIComponent(raw)\n }\n }\n }\n return null\n }\n\n /**\n * Check if pageview has been sent\n */\n isPageViewSent(): boolean {\n return !!this.responses.find(e => e.payload?.type === 'page' && e.payload?.anonymousId)\n }\n\n /**\n * Set a cached event\n */\n set(event: CachedEvent): void {\n this.log('set | caching event to cookie', event)\n this.push('cached_analytics_events', event)\n }\n\n /**\n * Push data to cookie cache\n */\n push(cookieName: string, data: any): void {\n if (typeof document === 'undefined') return\n\n let storedCookies: any[] = []\n const cacheCookie = this.parseCookies(cookieName)\n if (cacheCookie && Array.isArray(cacheCookie)) {\n storedCookies = cacheCookie\n }\n\n storedCookies.push(data)\n\n const domain = this.getAllowedDomain()\n const maxAge = 365 * 24 * 60 * 60 // 1 year\n const cookieString = `${cookieName}=${encodeURIComponent(JSON.stringify(storedCookies))}; path=/; Domain=${domain}; max-age=${maxAge}; SameSite=Lax`\n\n document.cookie = cookieString\n }\n\n /**\n * Get the allowed domain for cookies\n * For localhost/single-part domains: use as-is\n * For multi-part domains: use top-level domain (e.g., .deriv.com)\n */\n private getAllowedDomain(): string {\n if (typeof window === 'undefined') return ''\n\n const hostname = window.location.hostname\n\n // Handle IP addresses and localhost\n if (hostname === 'localhost' || /^\\d+\\.\\d+\\.\\d+\\.\\d+$/.test(hostname)) {\n return hostname\n }\n\n const parts = hostname.split('.')\n\n // Single part domain (e.g., \"localhost\")\n if (parts.length === 1) {\n return hostname\n }\n\n // Use top-level domain for proper subdomain sharing\n return `.${parts.slice(-2).join('.')}`\n }\n\n /**\n * Process event to hash email and add client info\n */\n processEvent(event: CachedEvent): CachedEvent {\n const clientInfo = this.getCookies('client_information')\n\n if (clientInfo) {\n const { email = null } = clientInfo\n\n if (email) {\n event.properties.email_hash = this.hash(email)\n }\n }\n\n if (event?.properties?.email) {\n const email = event.properties.email\n delete event.properties.email\n event.properties.email_hash = this.hash(email)\n }\n\n return event\n }\n\n /**\n * Track an event (either immediately or cache it)\n */\n track(originalEvent: CachedEvent, cache = false): void {\n if (typeof window === 'undefined') return\n\n const event = this.processEvent(originalEvent)\n const instance = this.getAnalyticsInstance()\n\n if (this.isReady() && !cache) {\n this.log('track | analytics ready — calling trackEvent', {\n event: event.name,\n properties: event.properties,\n })\n instance.trackEvent(event.name, event.properties)\n } else {\n this.log('track | analytics not ready or cache=true — storing event', { event: event.name, cache })\n this.set(event)\n }\n }\n\n /**\n * Track pageview with auto-retry until sent\n */\n pageView(): void {\n if (typeof window === 'undefined') return\n\n this.log('pageView | starting page view polling')\n\n if (!this.isTrackingResponses) {\n this.trackResponses()\n this.trackPageUnload()\n }\n\n this.interval = setInterval(() => {\n const Analytics = (window as any).Analytics\n\n if (\n typeof Analytics !== 'undefined' &&\n typeof Analytics.Analytics?.pageView === 'function' &&\n this.isReady()\n ) {\n this.log('pageView | analytics ready — sending page view', { href: window.location.href })\n Analytics.Analytics.pageView(window.location.href, \"Trader's hub\")\n }\n\n if (this.isPageViewSent()) {\n this.log('pageView | page view confirmed sent — clearing interval')\n if (this.interval) clearInterval(this.interval)\n }\n }, 1000)\n }\n\n /**\n * Add click event listener to element(s)\n */\n listen(\n element: Element | NodeList,\n { name = '', properties = {} }: { name: string; properties: Record<string, any> },\n cache = false,\n callback: ((e: Event) => { name: string; properties: Record<string, any>; cache?: boolean }) | null = null\n ): void {\n const addClickListener = (el: Element) => {\n if (!(el as any).dataset?.clickEventTracking) {\n el.addEventListener('click', (e: Event) => {\n let event = {\n name,\n properties,\n cache,\n }\n\n if (typeof callback === 'function') {\n const callbackResult = callback(e)\n event = {\n ...callbackResult,\n cache: callbackResult.cache ?? cache,\n }\n }\n\n this.track(event, event.cache ?? false)\n })\n ;(el as any).dataset.clickEventTracking = 'true'\n }\n }\n\n const elements = element instanceof NodeList ? Array.from(element as NodeListOf<Element>) : [element]\n elements.forEach(addClickListener)\n }\n\n /**\n * Add event handlers to multiple elements with auto-retry\n * Alias for backward compatibility with typo\n */\n addEventHandler(items: EventListenerConfig[]): this {\n if (typeof window === 'undefined') return this\n\n items.forEach(({ element, event = { name: '', properties: {} }, cache = false, callback = null }) => {\n // If a selector string is provided, use event delegation on document\n if (typeof element === 'string') {\n const selector = element\n\n if (!this.delegatedSelectors.has(selector)) {\n const delegatedHandler = (e: Event) => {\n const target = e.target as Element | null\n if (!target) return\n\n const matched = target.closest(selector)\n if (matched && !(matched as any).dataset?.clickEventTracking) {\n let evt: any = {\n name: event.name,\n properties: event.properties,\n cache,\n }\n\n if (typeof callback === 'function') {\n const callbackResult = callback(e)\n evt = {\n ...callbackResult,\n cache: callbackResult.cache ?? cache,\n }\n }\n\n ;(matched as any).dataset.clickEventTracking = 'true'\n this.track(evt, evt.cache ?? false)\n }\n }\n\n document.addEventListener('click', delegatedHandler)\n this.delegatedSelectors.add(selector)\n }\n } else {\n // Element or NodeList: attach directly to existing elements\n const elements = element instanceof NodeList ? Array.from(element) : [element]\n\n elements.forEach(el => {\n if (el && !(el as any).dataset?.clickEventTracking) {\n this.listen(el as Element, event, cache, callback)\n }\n })\n }\n })\n\n return this\n }\n\n /**\n * Backward compatibility alias (with typo from original)\n */\n addEventhandler = this.addEventHandler.bind(this)\n\n /**\n * Load events immediately\n */\n loadEvent(items: LoadEventConfig[]): this {\n this.log(\n 'loadEvent | firing load events',\n items.map(i => i.event.name)\n )\n items.forEach(({ event }) => {\n const { name, properties } = event\n this.track({\n name,\n properties,\n })\n })\n\n return this\n }\n\n /**\n * Load events on specific pages\n */\n pageLoadEvent(items: PageLoadEventConfig[]): this {\n if (typeof window === 'undefined') return this\n\n const pathname = window.location.pathname.slice(1)\n this.log('pageLoadEvent | checking page load events', { pathname })\n\n items.forEach(({ pages = [], excludedPages = [], event, callback = null }) => {\n let dispatch = false\n if (pages.length) {\n if (pages.includes(pathname)) {\n dispatch = true\n }\n } else if (excludedPages.length) {\n if (!excludedPages.includes(pathname)) {\n dispatch = true\n }\n } else {\n dispatch = true\n }\n\n if (dispatch) {\n const eventData = callback ? callback() : event\n this.log('pageLoadEvent | dispatching event for page', { pathname, event: eventData.name })\n this.loadEvent([{ event: eventData }])\n } else {\n this.log('pageLoadEvent | skipped event for page', {\n pathname,\n event: event.name,\n pages,\n excludedPages,\n })\n }\n })\n\n return this\n }\n\n /**\n * Clear the interval and cleanup\n */\n clearInterval(): void {\n if (this.interval) {\n clearInterval(this.interval)\n this.interval = null\n }\n }\n\n /**\n * Cleanup method for removing event listeners and intervals\n */\n cleanup(): void {\n this.clearInterval()\n this.responses = []\n this.isTrackingResponses = false\n }\n}\n\n// Create singleton instance\nexport const cacheTrackEvents = new AnalyticsCacheManager()\n\n// Export to global scope for backward compatibility\nif (typeof window !== 'undefined') {\n ;(window as any).cacheTrackEvents = cacheTrackEvents\n}\n","import { RudderStack } from './providers/rudderstack'\nimport type { TCoreAttributes, TAllEvents, TV2EventPayload } from './types'\nimport {\n cacheEventToCookie,\n cachePageViewToCookie,\n getCachedEvents,\n getCachedPageViews,\n clearCachedEvents,\n clearCachedPageViews,\n} from './utils/cookie'\nimport { isUUID, getCountry, cleanObject, flattenObject, createLogger } from './utils/helpers'\nimport { cacheTrackEvents } from './utils/analytics-cache'\n\n// Optional Growthbook types - only import if using Growthbook\nimport type { Growthbook, GrowthbookConfigs } from './providers/growthbook'\nimport type { TGrowthbookAttributes, TGrowthbookOptions } from './providers/growthbookTypes'\n\n// Optional Posthog types - only import if using Posthog\nimport type { Posthog } from './providers/posthog'\nimport type { TPosthogIdentifyTraits, TPosthogOptions } from './providers/posthogTypes'\n\ndeclare global {\n interface Window {\n AnalyticsInstance: ReturnType<typeof createAnalyticsInstance>\n }\n}\n\n/**\n * Configuration options for initializing the analytics instance\n */\ntype Options = {\n /** GrowthBook client API key for A/B testing and feature flags */\n growthbookKey?: string\n /** GrowthBook decryption key for encrypted feature payloads */\n growthbookDecryptionKey?: string\n /** RudderStack write key for event tracking */\n rudderstackKey?: string\n /** Additional configuration options for GrowthBook */\n growthbookOptions?: TGrowthbookOptions\n /** PostHog configuration options including API keys and settings */\n posthogOptions?: TPosthogOptions\n /** Enable debug logging — logs all analytics calls prefixed with [ANALYTIC] */\n debug?: boolean\n}\n\n/**\n * Creates a unified analytics instance that integrates RudderStack and GrowthBook.\n *\n * This function provides a centralized interface for:\n * - Event tracking across multiple analytics platforms\n * - A/B testing and feature flag management via GrowthBook\n * - Offline event caching with automatic replay\n *\n * @param {Options} _options - Optional initialization configuration\n * @returns {Object} Analytics instance with methods for tracking, identification, and feature management\n *\n * @example\n * ```typescript\n * const analytics = createAnalyticsInstance();\n *\n * // Initialize with providers\n * await analytics.initialise({\n * rudderstackKey: 'YOUR_RS_KEY',\n * growthbookKey: 'YOUR_GB_KEY',\n * growthbookDecryptionKey: 'YOUR_GB_DECRYPT_KEY'\n * });\n *\n * // Set user attributes\n * analytics.setAttributes({\n * user_id: 'user123',\n * country: 'US',\n * user_language: 'en'\n * });\n *\n * // Track events\n * analytics.trackEvent('button_clicked', { button_name: 'signup' });\n *\n * // Track page views\n * analytics.pageView('/dashboard', 'Deriv App');\n * ```\n */\nexport function createAnalyticsInstance(_options?: Options) {\n let _debug = _options?.debug ?? false\n\n const log = createLogger('', () => _debug)\n\n let _growthbook: Growthbook | undefined,\n _rudderstack: RudderStack,\n _posthog: Posthog | undefined,\n core_data: Partial<TCoreAttributes> = {},\n tracking_config: { [key: string]: boolean } = {},\n offline_event_cache: Array<{ event: keyof TAllEvents; payload: TAllEvents[keyof TAllEvents] }> = [],\n _pending_identify_calls: Array<{ userId: string; traits?: Record<string, any> }> = [],\n _cookie_cache_processed = false\n\n const processCookieCache = () => {\n if (_cookie_cache_processed) return\n if (!_rudderstack?.has_initialized) return\n\n _cookie_cache_processed = true\n\n try {\n const storedEvents = getCachedEvents()\n if (storedEvents.length > 0) {\n log(`processCookieCache | replaying ${storedEvents.length} cached event(s)`, storedEvents)\n storedEvents.forEach(event => {\n _rudderstack?.track(event.name as keyof TAllEvents, event.properties as any)\n })\n clearCachedEvents()\n }\n\n const storedPages = getCachedPageViews()\n if (storedPages.length > 0) {\n log(`processCookieCache | replaying ${storedPages.length} cached page view(s)`, storedPages)\n storedPages.forEach(page => {\n _rudderstack?.pageView(page.name, 'Deriv App', getId(), page.properties)\n })\n clearCachedPageViews()\n }\n } catch (err) {\n console.warn('Analytics: Failed to process cookie cache', err)\n }\n }\n\n const onSdkLoaded = () => {\n log('onSdkLoaded | RudderStack SDK loaded')\n processCookieCache()\n\n if (_pending_identify_calls.length > 0) {\n log(`onSdkLoaded | flushing ${_pending_identify_calls.length} pending identify call(s)`)\n }\n _pending_identify_calls.forEach(({ userId, traits }) => {\n if (userId) {\n _rudderstack?.identifyEvent(userId, traits)\n }\n })\n _pending_identify_calls = []\n }\n\n /**\n * Initializes the analytics instance with specified provider configurations.\n * This method should be called before tracking any events.\n *\n * Features:\n * - Lazy-loads providers (GrowthBook, PostHog) only when configured\n * - Automatically fetches user's country for GrowthBook targeting\n * - Processes any cached events from previous sessions\n * - Sets up event tracking callback for GrowthBook experiments\n *\n * @param {Options} options - Configuration options for analytics providers\n * @returns {Promise<void>} Resolves when initialization is complete\n *\n * @example\n * ```typescript\n * await analytics.initialise({\n * rudderstackKey: 'YOUR_RS_KEY',\n * growthbookKey: 'YOUR_GB_KEY',\n * growthbookDecryptionKey: 'YOUR_GB_DECRYPT_KEY',\n * posthogOptions: {\n * apiKey: 'YOUR_POSTHOG_API_KEY',\n * config: {\n * session_recording: {\n * recordCrossOriginIframes: true,\n * minimumDurationMilliseconds: 30000\n * }\n * }\n * }\n * });\n * ```\n */\n const initialise = async ({\n growthbookKey,\n growthbookDecryptionKey,\n rudderstackKey,\n growthbookOptions,\n posthogOptions,\n debug,\n }: Options) => {\n if (debug !== undefined) _debug = debug\n cacheTrackEvents.setDebug(_debug)\n\n log('initialise | starting analytics initialization', {\n rudderstack: !!rudderstackKey,\n growthbook: !!growthbookKey,\n posthog: !!posthogOptions,\n })\n\n try {\n // Only fetch country if GrowthBook is enabled and country not provided\n const country = growthbookOptions?.attributes?.country || (growthbookKey ? await getCountry() : undefined)\n\n if (rudderstackKey) {\n log('initialise | initializing RudderStack')\n _rudderstack = RudderStack.getRudderStackInstance(rudderstackKey, onSdkLoaded, _debug)\n }\n\n if (growthbookOptions?.attributes && Object.keys(growthbookOptions.attributes).length > 0) {\n const attrs = growthbookOptions.attributes\n const anonymousId = _rudderstack?.getAnonymousId()\n\n core_data = {\n ...core_data,\n country,\n ...(attrs.user_language && { user_language: attrs.user_language }),\n ...(attrs.account_type && { account_type: attrs.account_type }),\n ...(attrs.app_id && { app_id: attrs.app_id }),\n ...(attrs.residence_country && { residence_country: attrs.residence_country }),\n ...(attrs.device_type && { device_type: attrs.device_type }),\n ...(attrs.url && { url: attrs.url }),\n ...(attrs.email_hash && { email_hash: attrs.email_hash }),\n ...(attrs.network_type && { network_type: attrs.network_type }),\n ...(attrs.network_rtt && { network_rtt: attrs.network_rtt }),\n ...(attrs.network_downlink && { network_downlink: attrs.network_downlink }),\n ...(attrs.account_currency && { account_currency: attrs.account_currency }),\n ...(attrs.account_mode && { account_mode: attrs.account_mode }),\n loggedIn: !!attrs.loggedIn,\n ...(attrs.user_id && !isUUID(attrs.user_id) && { user_id: attrs.user_id }),\n ...(anonymousId && { anonymous_id: anonymousId }),\n }\n }\n\n growthbookOptions ??= {}\n growthbookOptions.attributes ??= {}\n const anonId = _rudderstack?.getAnonymousId()\n growthbookOptions.attributes.id ??= anonId\n growthbookOptions.attributes.country ??= country\n\n if (growthbookKey) {\n log('initialise | initializing GrowthBook')\n // Dynamically import Growthbook only when needed\n const { Growthbook } = await import('./providers/growthbook')\n _growthbook = Growthbook.getGrowthBookInstance(\n growthbookKey,\n growthbookDecryptionKey,\n growthbookOptions,\n _debug\n )\n log('initialise | GrowthBook initialized')\n\n const interval = setInterval(() => {\n if (Object.keys(tracking_config).length > 0) clearInterval(interval)\n else tracking_config = getFeatureValue('tracking-buttons-config', {}) as { [key: string]: boolean }\n }, 1000)\n }\n\n if (posthogOptions) {\n log('initialise | initializing PostHog')\n // Dynamically import Posthog only when needed\n const { Posthog } = await import('./providers/posthog')\n _posthog = Posthog.getPosthogInstance(posthogOptions, _debug)\n log('initialise | PostHog initialized')\n }\n\n log('initialise | analytics initialization complete')\n } catch (err) {\n console.warn('Analytics: Failed to initialize', err)\n }\n }\n\n /**\n * Sets user and context attributes for analytics tracking and targeting.\n * These attributes are automatically included in all subsequent events.\n *\n * Attributes are used for:\n * - Event enrichment (added to all tracked events)\n * - GrowthBook targeting (feature flags and A/B tests)\n * - User segmentation across analytics platforms\n *\n * @param {TCoreAttributes} attributes - User and context attributes\n *\n * @example\n * ```typescript\n * analytics.setAttributes({\n * user_id: 'CR123456',\n * country: 'US',\n * user_language: 'en',\n * device_type: 'desktop',\n * account_type: 'real',\n * loggedIn: true\n * });\n * ```\n */\n const setAttributes = ({\n country,\n user_language,\n device_language,\n device_type,\n account_type,\n user_id,\n anonymous_id,\n app_id,\n utm_source,\n utm_medium,\n utm_campaign,\n is_authorised,\n residence_country,\n url,\n domain,\n geo_location,\n loggedIn,\n network_downlink,\n network_rtt,\n network_type,\n account_currency,\n account_mode,\n }: TCoreAttributes) => {\n const user_identity = user_id ?? getId()\n\n log('setAttributes | received attributes', {\n country,\n user_language,\n device_language,\n device_type,\n account_type,\n user_id,\n anonymous_id,\n app_id,\n utm_source,\n utm_medium,\n utm_campaign,\n is_authorised,\n residence_country,\n url,\n domain,\n geo_location,\n loggedIn,\n network_downlink,\n network_rtt,\n network_type,\n account_currency,\n account_mode,\n })\n\n if (_growthbook) {\n const config: TGrowthbookAttributes = {\n country,\n residence_country,\n user_language,\n device_language,\n device_type,\n utm_source,\n utm_medium,\n utm_campaign,\n is_authorised,\n url,\n domain,\n loggedIn,\n ...(user_id && !isUUID(user_id) && { user_id }),\n anonymous_id,\n }\n if (user_identity) {\n config.id = user_identity\n config.user_id = user_identity\n }\n log('setAttributes | called GrowthBook setAttributes', config)\n _growthbook.setAttributes(config)\n }\n\n core_data = {\n ...core_data,\n ...(country !== undefined && { country }),\n ...(geo_location !== undefined && { geo_location }),\n ...(user_language !== undefined && { user_language }),\n ...(account_type !== undefined && { account_type }),\n ...(app_id !== undefined && { app_id }),\n ...(residence_country !== undefined && { residence_country }),\n ...(device_type !== undefined && { device_type }),\n ...(url !== undefined && { url }),\n ...(loggedIn !== undefined && { loggedIn }),\n ...(network_downlink !== undefined && { network_downlink }),\n ...(network_rtt !== undefined && { network_rtt }),\n ...(network_type !== undefined && { network_type }),\n ...(user_id !== undefined && !isUUID(user_id) && { user_id }),\n ...(anonymous_id !== undefined && { anonymous_id }),\n ...(account_currency !== undefined && { account_currency }),\n ...(account_mode !== undefined && { account_mode }),\n }\n\n log('setAttributes | updated core_data', core_data)\n }\n\n const getFeatureState = (id: string) => _growthbook?.getFeatureState(id)?.experimentResult?.name\n\n const getFeatureValue = <K extends keyof GrowthbookConfigs, V extends GrowthbookConfigs[K]>(\n id: K,\n defaultValue: V\n ) => _growthbook?.getFeatureValue(id as string, defaultValue)\n\n const getGrowthbookStatus = async () => await _growthbook?.getStatus()\n const isFeatureOn = (key: string) => _growthbook?.isOn(key)\n const setUrl = (href: string) => _growthbook?.setUrl(href)\n\n const getId = () => {\n const userId = _rudderstack?.getUserId() || ''\n return userId && !isUUID(userId) ? userId : ''\n }\n\n const getAnonymousId = () => _rudderstack?.getAnonymousId() || ''\n\n /**\n * Tracks a page view event.\n *\n * Features:\n * - Automatically includes user ID if available\n * - Caches page views when offline or provider not initialized\n *\n * @param {string} current_page - The current page URL or path\n * @param {string} [platform='Deriv App'] - The platform name\n * @param {Record<string, unknown>} [properties] - Additional page properties\n *\n * @example\n * ```typescript\n * analytics.pageView('/dashboard');\n * analytics.pageView('/trade', 'Deriv Trader', { section: 'multipliers' });\n * ```\n */\n const pageView = (current_page: string, platform = 'Deriv App', properties?: Record<string, unknown>) => {\n const userId = getId()\n\n log('pageView | called', { current_page, platform, properties, userId })\n\n // Handle RudderStack pageView independently\n if (_rudderstack) {\n if (_rudderstack.has_initialized) {\n log('pageView | sending page view to RudderStack', { current_page, platform })\n _rudderstack.pageView(current_page, platform, userId, properties)\n } else {\n log('pageView | RudderStack not initialized — caching page view to cookie', { current_page })\n cachePageViewToCookie(current_page, { platform, ...properties })\n }\n }\n\n // PostHog handles page views automatically via autocapture\n // No need to manually send page views to PostHog\n }\n\n /**\n * Identifies a user across analytics platforms.\n * This method should be called after user login or when user identity is known.\n *\n * Features:\n * - Queues identify calls if provider not yet initialized\n * - Allows custom traits for each provider or shared traits for both\n * - Identifies user in PostHog if configured\n *\n * @param {string} [user_id] - The user ID to identify. If not provided, uses stored user ID\n * @param {Record<string, any>} [traits] - Optional traits to send to both providers, or provider-specific traits\n *\n * @example\n * ```typescript\n * // Simple identify\n * analytics.identifyEvent('CR123456');\n *\n * // Identify with same traits for both providers\n * analytics.identifyEvent('CR123456', {\n * language: 'en',\n * country_of_residence: 'US'\n * });\n *\n * // Identify with provider-specific traits\n * analytics.identifyEvent('CR123456', {\n * rudderstack: { language: 'en', custom_field: 'value' },\n * posthog: { language: 'en', country_of_residence: 'US' }\n * });\n * ```\n */\n const identifyEvent = (user_id?: string, traits?: Record<string, any>) => {\n const stored_user_id = user_id || getId()\n if (!stored_user_id) {\n log('identifyEvent | skipped — no user_id available')\n return\n }\n\n log('identifyEvent | called', { user_id: stored_user_id, traits })\n\n // Check if traits has provider-specific structure\n const hasProviderStructure = traits?.rudderstack !== undefined || traits?.posthog !== undefined\n const rudderstackTraits = hasProviderStructure ? traits?.rudderstack : traits\n const posthogTraits = hasProviderStructure ? traits?.posthog : traits\n\n // Handle RudderStack identification independently\n if (_rudderstack) {\n if (_rudderstack.has_initialized) {\n log('identifyEvent | calling RudderStack identify', {\n user_id: stored_user_id,\n traits: rudderstackTraits,\n })\n _rudderstack.identifyEvent(stored_user_id, rudderstackTraits)\n } else {\n if (!_pending_identify_calls.some(call => call.userId === stored_user_id)) {\n log('identifyEvent | RudderStack not initialized — queuing identify call', {\n user_id: stored_user_id,\n })\n _pending_identify_calls.push({ userId: stored_user_id, traits: rudderstackTraits })\n }\n }\n }\n\n // Handle PostHog identification independently\n if (_posthog?.has_initialized && posthogTraits) {\n log('identifyEvent | calling PostHog identify', { user_id: stored_user_id, traits: posthogTraits })\n _posthog.identifyEvent(stored_user_id, posthogTraits as TPosthogIdentifyTraits)\n }\n }\n\n const reset = () => {\n log('reset | resetting all providers')\n // Reset each provider independently\n if (_rudderstack?.has_initialized) {\n log('reset | resetting RudderStack')\n _rudderstack.reset()\n }\n if (_posthog?.has_initialized) {\n log('reset | resetting PostHog')\n _posthog.reset()\n }\n }\n\n const isV2Payload = (payload: any): payload is TV2EventPayload => {\n return 'event_metadata' in payload || 'cta_information' in payload || 'error' in payload\n }\n\n /**\n * Tracks a custom event with associated data.\n *\n * Features:\n * - Automatically enriches events with core attributes\n * - Supports both V1 and V2 event payload formats\n * - RudderStack: Caches events when offline or not initialized\n * - PostHog: Sends immediately if initialized (has built-in caching)\n * - Respects feature flag configurations\n * - Each provider works independently - one failing won't affect the other\n *\n * @template T - The event name type from TAllEvents\n * @param {T} event - The event name to track\n * @param {TAllEvents[T]} analytics_data - The event data payload\n *\n * @example\n * ```typescript\n * // Simple event\n * analytics.trackEvent('button_clicked', { button_name: 'signup' });\n *\n * // V2 event with metadata\n * analytics.trackEvent('form_submitted', {\n * event_metadata: { form_name: 'registration' },\n * cta_information: { button_text: 'Create Account' }\n * });\n * ```\n */\n const trackEvent = <T extends keyof TAllEvents>(event: T, analytics_data: TAllEvents[T]) => {\n const userId = getId()\n let final_payload: any = {}\n\n log('trackEvent | called', { event, analytics_data, userId, core_data })\n\n if (isV2Payload(analytics_data)) {\n const v2_data = analytics_data as TV2EventPayload\n final_payload = {\n ...v2_data,\n event_metadata: {\n ...core_data,\n ...(userId && !core_data.user_id && { user_id: userId }),\n ...v2_data.event_metadata,\n },\n }\n log('trackEvent | built V2 payload', { event, final_payload })\n } else {\n final_payload = {\n ...core_data,\n ...analytics_data,\n ...(userId && !core_data.user_id && { user_id: userId }),\n }\n log('trackEvent | built V1 payload', { event, final_payload })\n }\n\n const shouldTrack = !(event in tracking_config) || tracking_config[event as string]\n if (!shouldTrack) {\n log('trackEvent | skipped — event disabled by tracking_config', { event })\n return\n }\n\n // Handle RudderStack independently\n const hasRudderstackInitialized = _rudderstack?.has_initialized\n if (!navigator.onLine || !hasRudderstackInitialized) {\n if (!hasRudderstackInitialized) {\n log('trackEvent | RudderStack not initialized — caching event to cookie', { event })\n cacheEventToCookie(event as string, final_payload)\n } else {\n log('trackEvent | offline — caching event to memory', { event })\n offline_event_cache.push({ event, payload: final_payload })\n }\n } else {\n // Send cached events to RudderStack\n if (offline_event_cache.length > 0) {\n log(`trackEvent | flushing ${offline_event_cache.length} offline cached event(s) to RudderStack`)\n offline_event_cache.forEach(cache => {\n const cleaned_cache_payload = cleanObject(cache.payload)\n _rudderstack?.track(cache.event, cleaned_cache_payload)\n })\n offline_event_cache = []\n }\n\n // Send current event to RudderStack\n const cleaned_payload = cleanObject(final_payload)\n log('trackEvent | sending event to RudderStack', { event, payload: cleaned_payload })\n _rudderstack?.track(event, cleaned_payload)\n }\n\n // Handle PostHog independently - send immediately if initialized\n if (_posthog?.has_initialized) {\n const flattened_payload = flattenObject(final_payload)\n const cleaned_posthog_payload = cleanObject(flattened_payload)\n log('trackEvent | sending event to PostHog', { event, payload: cleaned_posthog_payload })\n _posthog.capture(event as string, cleaned_posthog_payload)\n }\n }\n\n /**\n * Ensures client_id is set in PostHog stored person properties.\n * Call this when the user ID is available and PostHog is loaded.\n * Useful for backfilling client_id for users identified in previous sessions.\n * No-op if client_id is already present or PostHog is not initialized.\n *\n * @param user_id - The user ID to use as client_id\n *\n * @example\n * ```typescript\n * if (window.posthog?.__loaded && userId) {\n * analytics.backfillPersonProperties(userId, email)\n * }\n * ```\n */\n const backfillPersonProperties = (user_id: string, email?: string): void => {\n log('backfillPersonProperties | called', { user_id })\n if (_posthog?.has_initialized) {\n log('backfillPersonProperties | backfilling person properties in PostHog', { user_id })\n _posthog.backfillPersonProperties(user_id, email)\n } else {\n log('backfillPersonProperties | skipped — PostHog not initialized')\n }\n }\n\n const getInstances = () => ({ ab: _growthbook, tracking: _rudderstack, posthog: _posthog })\n\n const AnalyticsInstance = {\n initialise,\n setAttributes,\n identifyEvent,\n backfillPersonProperties,\n getFeatureState,\n getFeatureValue,\n getGrowthbookStatus,\n isFeatureOn,\n setUrl,\n getId,\n getAnonymousId,\n trackEvent,\n getInstances,\n pageView,\n reset,\n }\n\n if (typeof window !== 'undefined') {\n window.AnalyticsInstance = AnalyticsInstance\n }\n\n return AnalyticsInstance\n}\n\nexport const Analytics = createAnalyticsInstance()\n"],"mappings":"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","names":["COOKIE_MAX_AGE_SECONDS","_RudderStack","RUDDERSTACK_KEY","onLoaded","debug","RudderAnalytics","createLogger","hostname","domain_name","domain","uuid","bytes","hex","byte","secureFlag","error","rudderstackDataplane","user_id","payload","currentUserId","current_page","platform","properties","pageProperties","event","err","RudderStack","CACHE_COOKIE_EVENTS","CACHE_COOKIE_PAGES","cacheEventToCookie","eventName","properties","domain","getAllowedDomain","existingCache","api","events","cookieOptions","err","cachePageViewToCookie","pageName","pages","getCachedEvents","storedEventsString","getCachedPageViews","storedPagesString","clearCachedEvents","clearCachedPageViews","AnalyticsCacheManager","createLogger","debug","inputString","desiredLength","fnv1aHash","string","hash","i","base64Encode","combined","name","parts","cookieValue","originalXhrOpen","originalXhrSend","method","url","body","parsedPayload","responseData","cacheTrackEvents","instance","cookieName","cookies","cookie","eqIdx","raw","e","event","data","storedCookies","cacheCookie","domain","maxAge","cookieString","hostname","clientInfo","email","originalEvent","cache","Analytics","element","properties","callback","addClickListener","el","callbackResult","items","selector","delegatedHandler","target","matched","evt","pathname","pages","excludedPages","dispatch","eventData","createAnalyticsInstance","_options","_debug","log","createLogger","_growthbook","_rudderstack","_posthog","core_data","tracking_config","offline_event_cache","_pending_identify_calls","_cookie_cache_processed","processCookieCache","storedEvents","getCachedEvents","event","clearCachedEvents","storedPages","getCachedPageViews","page","getId","clearCachedPageViews","err","onSdkLoaded","userId","traits","initialise","growthbookKey","growthbookDecryptionKey","rudderstackKey","growthbookOptions","posthogOptions","debug","_a","_b","cacheTrackEvents","country","getCountry","RudderStack","attrs","anonymousId","isUUID","anonId","Growthbook","interval","getFeatureValue","Posthog","setAttributes","user_language","device_language","device_type","account_type","user_id","anonymous_id","app_id","utm_source","utm_medium","utm_campaign","is_authorised","residence_country","url","domain","geo_location","loggedIn","network_downlink","network_rtt","network_type","account_currency","account_mode","user_identity","config","getFeatureState","id","defaultValue","getGrowthbookStatus","isFeatureOn","key","setUrl","href","getAnonymousId","pageView","current_page","platform","properties","cachePageViewToCookie","identifyEvent","stored_user_id","hasProviderStructure","rudderstackTraits","posthogTraits","call","reset","isV2Payload","payload","AnalyticsInstance","email","analytics_data","final_payload","v2_data","hasRudderstackInitialized","cacheEventToCookie","cache","cleaned_cache_payload","cleanObject","cleaned_payload","flattened_payload","flattenObject","cleaned_posthog_payload","Analytics"]}
@@ -0,0 +1,3 @@
1
+ import {g}from'./chunk-ID2GN2PS.mjs';/* @deriv-com/analytics - NPM Package - Built with tsup */
2
+ var p=class{constructor(){this.interval=null;this.responses=[];this.isTrackingResponses=false;this.delegatedSelectors=new Set;this.debug=false;this.log=g("[CacheManager]",()=>this.debug);this.addEventhandler=this.addEventHandler.bind(this);}setDebug(e){this.debug=e;}hash(e,n=32){let t=a=>{let r=2166136261;for(let c=0;c<a.length;c++)r^=a.charCodeAt(c),r=r*16777619>>>0;return r.toString(16)},i=a=>btoa(a),s=t(e),o=i(s);for(;o.length<n;)o+=i(t(o));return o.substring(0,n)}getCookies(e){let t=`; ${document.cookie}`.split(`; ${e}=`);if(t.length===2){let i=decodeURIComponent(t.pop().split(";").shift());try{return JSON.parse(i)}catch{return i}}return null}trackPageUnload(){typeof window>"u"||window.addEventListener("beforeunload",()=>{this.isPageViewSent()||this.push("cached_analytics_page_views",{name:window.location.href,properties:{url:window.location.href}});});}trackResponses(){if(typeof window>"u"||typeof XMLHttpRequest>"u")return;let e=XMLHttpRequest.prototype.open,n=XMLHttpRequest.prototype.send;XMLHttpRequest.prototype.open=function(t,i){return this._url=i,this._method=t,e.apply(this,arguments)},XMLHttpRequest.prototype.send=function(t){return this.addEventListener("load",function(){let i=null;if(typeof t=="string")try{i=JSON.parse(t);}catch{i=t;}let s={url:this._url,method:this._method,status:this.status,headers:this.getAllResponseHeaders(),data:this.responseText,payload:i};f.responses.push(s);}),n.apply(this,arguments)},this.isTrackingResponses=true;}getAnalyticsInstance(){return typeof window>"u"?null:window.AnalyticsInstance??window.DerivAnalytics?.Analytics??window.Analytics?.Analytics}isReady(){if(typeof window>"u")return false;let e=this.getAnalyticsInstance();return e?!!e.getInstances?.()?.tracking:false}parseCookies(e){if(typeof document>"u")return null;let n=document.cookie.split("; ");for(let t of n){let i=t.indexOf("=");if(i!==-1&&decodeURIComponent(t.slice(0,i))===e){let s=t.slice(i+1);try{return JSON.parse(decodeURIComponent(s))}catch{return decodeURIComponent(s)}}}return null}isPageViewSent(){return !!this.responses.find(e=>e.payload?.type==="page"&&e.payload?.anonymousId)}set(e){this.log("set | caching event to cookie",e),this.push("cached_analytics_events",e);}push(e,n){if(typeof document>"u")return;let t=[],i=this.parseCookies(e);i&&Array.isArray(i)&&(t=i),t.push(n);let s=this.getAllowedDomain(),o=365*24*60*60,a=`${e}=${encodeURIComponent(JSON.stringify(t))}; path=/; Domain=${s}; max-age=${o}; SameSite=Lax`;document.cookie=a;}getAllowedDomain(){if(typeof window>"u")return "";let e=window.location.hostname;if(e==="localhost"||/^\d+\.\d+\.\d+\.\d+$/.test(e))return e;let n=e.split(".");return n.length===1?e:`.${n.slice(-2).join(".")}`}processEvent(e){let n=this.getCookies("client_information");if(n){let{email:t=null}=n;t&&(e.properties.email_hash=this.hash(t));}if(e?.properties?.email){let t=e.properties.email;delete e.properties.email,e.properties.email_hash=this.hash(t);}return e}track(e,n=false){if(typeof window>"u")return;let t=this.processEvent(e),i=this.getAnalyticsInstance();this.isReady()&&!n?(this.log("track | analytics ready \u2014 calling trackEvent",{event:t.name,properties:t.properties}),i.trackEvent(t.name,t.properties)):(this.log("track | analytics not ready or cache=true \u2014 storing event",{event:t.name,cache:n}),this.set(t));}pageView(){typeof window>"u"||(this.log("pageView | starting page view polling"),this.isTrackingResponses||(this.trackResponses(),this.trackPageUnload()),this.interval=setInterval(()=>{let e=window.Analytics;typeof e<"u"&&typeof e.Analytics?.pageView=="function"&&this.isReady()&&(this.log("pageView | analytics ready \u2014 sending page view",{href:window.location.href}),e.Analytics.pageView(window.location.href,"Trader's hub")),this.isPageViewSent()&&(this.log("pageView | page view confirmed sent \u2014 clearing interval"),this.interval&&clearInterval(this.interval));},1e3));}listen(e,{name:n="",properties:t={}},i=false,s=null){let o=r=>{r.dataset?.clickEventTracking||(r.addEventListener("click",c=>{let l={name:n,properties:t,cache:i};if(typeof s=="function"){let d=s(c);l={...d,cache:d.cache??i};}this.track(l,l.cache??false);}),r.dataset.clickEventTracking="true");};(e instanceof NodeList?Array.from(e):[e]).forEach(o);}addEventHandler(e){return typeof window>"u"?this:(e.forEach(({element:n,event:t={name:"",properties:{}},cache:i=false,callback:s=null})=>{if(typeof n=="string"){let o=n;if(!this.delegatedSelectors.has(o)){let a=r=>{let c=r.target;if(!c)return;let l=c.closest(o);if(l&&!l.dataset?.clickEventTracking){let d={name:t.name,properties:t.properties,cache:i};if(typeof s=="function"){let h=s(r);d={...h,cache:h.cache??i};}l.dataset.clickEventTracking="true",this.track(d,d.cache??false);}};document.addEventListener("click",a),this.delegatedSelectors.add(o);}}else (n instanceof NodeList?Array.from(n):[n]).forEach(a=>{a&&!a.dataset?.clickEventTracking&&this.listen(a,t,i,s);});}),this)}loadEvent(e){return this.log("loadEvent | firing load events",e.map(n=>n.event.name)),e.forEach(({event:n})=>{let{name:t,properties:i}=n;this.track({name:t,properties:i});}),this}pageLoadEvent(e){if(typeof window>"u")return this;let n=window.location.pathname.slice(1);return this.log("pageLoadEvent | checking page load events",{pathname:n}),e.forEach(({pages:t=[],excludedPages:i=[],event:s,callback:o=null})=>{let a=false;if(t.length?t.includes(n)&&(a=true):i.length&&i.includes(n)||(a=true),a){let r=o?o():s;this.log("pageLoadEvent | dispatching event for page",{pathname:n,event:r.name}),this.loadEvent([{event:r}]);}else this.log("pageLoadEvent | skipped event for page",{pathname:n,event:s.name,pages:t,excludedPages:i});}),this}clearInterval(){this.interval&&(clearInterval(this.interval),this.interval=null);}cleanup(){this.clearInterval(),this.responses=[],this.isTrackingResponses=false;}},f=new p;typeof window<"u"&&(window.cacheTrackEvents=f);export{f as a};//# sourceMappingURL=chunk-7CT4XR74.mjs.map
3
+ //# sourceMappingURL=chunk-7CT4XR74.mjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/utils/analytics-cache.ts"],"names":["AnalyticsCacheManager","createLogger","debug","inputString","desiredLength","fnv1aHash","string","hash","i","base64Encode","combined","name","parts","cookieValue","originalXhrOpen","originalXhrSend","method","url","body","parsedPayload","responseData","cacheTrackEvents","instance","cookieName","cookies","cookie","eqIdx","raw","event","data","storedCookies","cacheCookie","domain","maxAge","cookieString","hostname","clientInfo","email","originalEvent","cache","Analytics","element","properties","callback","addClickListener","el","e","callbackResult","items","selector","delegatedHandler","target","matched","evt","pathname","pages","excludedPages","dispatch","eventData"],"mappings":"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","file":"chunk-7CT4XR74.mjs","sourcesContent":["/**\n * Analytics Cache Manager - Version 1.1.0\n * Enhanced TypeScript implementation with better type safety and SSR support\n */\nimport { createLogger } from './helpers'\n\ntype CachedEvent = {\n name: string\n properties: Record<string, any>\n}\n\ntype ResponseData = {\n url: string\n method: string\n status: number\n headers: string\n data: string\n payload: any\n}\n\ntype EventListenerConfig = {\n element: Element | NodeList | string\n event: {\n name: string\n properties: Record<string, any>\n }\n cache?: boolean\n callback?: (e: Event) => { name: string; properties: Record<string, any>; cache?: boolean }\n}\n\ntype LoadEventConfig = {\n event: {\n name: string\n properties: Record<string, any>\n }\n}\n\ntype PageLoadEventConfig = {\n pages?: string[]\n excludedPages?: string[]\n event: {\n name: string\n properties: Record<string, any>\n }\n callback?: () => { name: string; properties: Record<string, any> }\n}\n\nclass AnalyticsCacheManager {\n private interval: NodeJS.Timeout | null = null\n private responses: ResponseData[] = []\n private isTrackingResponses = false\n private delegatedSelectors: Set<string> = new Set()\n private debug = false\n private log = createLogger('[CacheManager]', () => this.debug)\n\n setDebug(debug: boolean): void {\n this.debug = debug\n }\n\n /**\n * FNV-1a hash algorithm for creating consistent hashes\n */\n private hash(inputString: string, desiredLength = 32): string {\n const fnv1aHash = (string: string): string => {\n let hash = 0x811c9dc5\n for (let i = 0; i < string.length; i++) {\n hash ^= string.charCodeAt(i)\n hash = (hash * 0x01000193) >>> 0\n }\n return hash.toString(16)\n }\n\n const base64Encode = (string: string): string => btoa(string)\n\n let hash = fnv1aHash(inputString)\n let combined = base64Encode(hash)\n\n while (combined.length < desiredLength) {\n combined += base64Encode(fnv1aHash(combined))\n }\n\n return combined.substring(0, desiredLength)\n }\n\n /**\n * Get cookie value by name\n */\n getCookies(name: string): any {\n const value = `; ${document.cookie}`\n const parts = value.split(`; ${name}=`)\n if (parts.length === 2) {\n const cookieValue = decodeURIComponent(parts.pop()!.split(';').shift()!)\n\n try {\n return JSON.parse(cookieValue)\n } catch (e) {\n return cookieValue\n }\n }\n return null\n }\n\n /**\n * Track page unload events to cache pageviews before leaving\n */\n trackPageUnload(): void {\n if (typeof window === 'undefined') return\n\n window.addEventListener('beforeunload', () => {\n if (!this.isPageViewSent()) {\n this.push('cached_analytics_page_views', {\n name: window.location.href,\n properties: {\n url: window.location.href,\n },\n })\n }\n })\n }\n\n /**\n * Track XMLHttpRequest responses to monitor analytics calls\n */\n trackResponses(): void {\n if (typeof window === 'undefined' || typeof XMLHttpRequest === 'undefined') return\n\n const originalXhrOpen = XMLHttpRequest.prototype.open\n const originalXhrSend = XMLHttpRequest.prototype.send\n\n XMLHttpRequest.prototype.open = function (method: string, url: string | URL) {\n ;(this as any)._url = url\n ;(this as any)._method = method\n return originalXhrOpen.apply(this, arguments as any)\n }\n\n XMLHttpRequest.prototype.send = function (body?: Document | XMLHttpRequestBodyInit | null) {\n this.addEventListener('load', function () {\n let parsedPayload = null\n\n if (typeof body === 'string') {\n try {\n parsedPayload = JSON.parse(body)\n } catch (e) {\n parsedPayload = body\n }\n }\n\n const responseData: ResponseData = {\n url: (this as any)._url,\n method: (this as any)._method,\n status: this.status,\n headers: this.getAllResponseHeaders(),\n data: this.responseText,\n payload: parsedPayload,\n }\n cacheTrackEvents.responses.push(responseData)\n })\n\n return originalXhrSend.apply(this, arguments as any)\n }\n\n this.isTrackingResponses = true\n }\n\n /**\n * Resolve the analytics instance from the window object.\n * Supports three usage patterns:\n * - NPM import: analytics.ts sets window.AnalyticsInstance directly\n * - IIFE bundle: tsup sets window.DerivAnalytics = { Analytics, cacheTrackEvents }\n * - Legacy: consumers that explicitly set window.Analytics = window.DerivAnalytics\n */\n private getAnalyticsInstance(): any {\n if (typeof window === 'undefined') return null\n return (\n (window as any).AnalyticsInstance ??\n (window as any).DerivAnalytics?.Analytics ??\n (window as any).Analytics?.Analytics\n )\n }\n\n /**\n * Check if Analytics instance is ready\n */\n isReady(): boolean {\n if (typeof window === 'undefined') return false\n const instance = this.getAnalyticsInstance()\n if (!instance) return false\n return !!instance.getInstances?.()?.tracking\n }\n\n /**\n * Parse cookies to find a specific cookie by name.\n * Uses an early-exit linear scan instead of building a full map,\n * which is significantly faster when there are many cookies.\n */\n private parseCookies(cookieName: string): any {\n if (typeof document === 'undefined') return null\n\n const cookies = document.cookie.split('; ')\n for (const cookie of cookies) {\n const eqIdx = cookie.indexOf('=')\n if (eqIdx === -1) continue\n if (decodeURIComponent(cookie.slice(0, eqIdx)) === cookieName) {\n const raw = cookie.slice(eqIdx + 1)\n try {\n return JSON.parse(decodeURIComponent(raw))\n } catch {\n return decodeURIComponent(raw)\n }\n }\n }\n return null\n }\n\n /**\n * Check if pageview has been sent\n */\n isPageViewSent(): boolean {\n return !!this.responses.find(e => e.payload?.type === 'page' && e.payload?.anonymousId)\n }\n\n /**\n * Set a cached event\n */\n set(event: CachedEvent): void {\n this.log('set | caching event to cookie', event)\n this.push('cached_analytics_events', event)\n }\n\n /**\n * Push data to cookie cache\n */\n push(cookieName: string, data: any): void {\n if (typeof document === 'undefined') return\n\n let storedCookies: any[] = []\n const cacheCookie = this.parseCookies(cookieName)\n if (cacheCookie && Array.isArray(cacheCookie)) {\n storedCookies = cacheCookie\n }\n\n storedCookies.push(data)\n\n const domain = this.getAllowedDomain()\n const maxAge = 365 * 24 * 60 * 60 // 1 year\n const cookieString = `${cookieName}=${encodeURIComponent(JSON.stringify(storedCookies))}; path=/; Domain=${domain}; max-age=${maxAge}; SameSite=Lax`\n\n document.cookie = cookieString\n }\n\n /**\n * Get the allowed domain for cookies\n * For localhost/single-part domains: use as-is\n * For multi-part domains: use top-level domain (e.g., .deriv.com)\n */\n private getAllowedDomain(): string {\n if (typeof window === 'undefined') return ''\n\n const hostname = window.location.hostname\n\n // Handle IP addresses and localhost\n if (hostname === 'localhost' || /^\\d+\\.\\d+\\.\\d+\\.\\d+$/.test(hostname)) {\n return hostname\n }\n\n const parts = hostname.split('.')\n\n // Single part domain (e.g., \"localhost\")\n if (parts.length === 1) {\n return hostname\n }\n\n // Use top-level domain for proper subdomain sharing\n return `.${parts.slice(-2).join('.')}`\n }\n\n /**\n * Process event to hash email and add client info\n */\n processEvent(event: CachedEvent): CachedEvent {\n const clientInfo = this.getCookies('client_information')\n\n if (clientInfo) {\n const { email = null } = clientInfo\n\n if (email) {\n event.properties.email_hash = this.hash(email)\n }\n }\n\n if (event?.properties?.email) {\n const email = event.properties.email\n delete event.properties.email\n event.properties.email_hash = this.hash(email)\n }\n\n return event\n }\n\n /**\n * Track an event (either immediately or cache it)\n */\n track(originalEvent: CachedEvent, cache = false): void {\n if (typeof window === 'undefined') return\n\n const event = this.processEvent(originalEvent)\n const instance = this.getAnalyticsInstance()\n\n if (this.isReady() && !cache) {\n this.log('track | analytics ready — calling trackEvent', {\n event: event.name,\n properties: event.properties,\n })\n instance.trackEvent(event.name, event.properties)\n } else {\n this.log('track | analytics not ready or cache=true — storing event', { event: event.name, cache })\n this.set(event)\n }\n }\n\n /**\n * Track pageview with auto-retry until sent\n */\n pageView(): void {\n if (typeof window === 'undefined') return\n\n this.log('pageView | starting page view polling')\n\n if (!this.isTrackingResponses) {\n this.trackResponses()\n this.trackPageUnload()\n }\n\n this.interval = setInterval(() => {\n const Analytics = (window as any).Analytics\n\n if (\n typeof Analytics !== 'undefined' &&\n typeof Analytics.Analytics?.pageView === 'function' &&\n this.isReady()\n ) {\n this.log('pageView | analytics ready — sending page view', { href: window.location.href })\n Analytics.Analytics.pageView(window.location.href, \"Trader's hub\")\n }\n\n if (this.isPageViewSent()) {\n this.log('pageView | page view confirmed sent — clearing interval')\n if (this.interval) clearInterval(this.interval)\n }\n }, 1000)\n }\n\n /**\n * Add click event listener to element(s)\n */\n listen(\n element: Element | NodeList,\n { name = '', properties = {} }: { name: string; properties: Record<string, any> },\n cache = false,\n callback: ((e: Event) => { name: string; properties: Record<string, any>; cache?: boolean }) | null = null\n ): void {\n const addClickListener = (el: Element) => {\n if (!(el as any).dataset?.clickEventTracking) {\n el.addEventListener('click', (e: Event) => {\n let event = {\n name,\n properties,\n cache,\n }\n\n if (typeof callback === 'function') {\n const callbackResult = callback(e)\n event = {\n ...callbackResult,\n cache: callbackResult.cache ?? cache,\n }\n }\n\n this.track(event, event.cache ?? false)\n })\n ;(el as any).dataset.clickEventTracking = 'true'\n }\n }\n\n const elements = element instanceof NodeList ? Array.from(element as NodeListOf<Element>) : [element]\n elements.forEach(addClickListener)\n }\n\n /**\n * Add event handlers to multiple elements with auto-retry\n * Alias for backward compatibility with typo\n */\n addEventHandler(items: EventListenerConfig[]): this {\n if (typeof window === 'undefined') return this\n\n items.forEach(({ element, event = { name: '', properties: {} }, cache = false, callback = null }) => {\n // If a selector string is provided, use event delegation on document\n if (typeof element === 'string') {\n const selector = element\n\n if (!this.delegatedSelectors.has(selector)) {\n const delegatedHandler = (e: Event) => {\n const target = e.target as Element | null\n if (!target) return\n\n const matched = target.closest(selector)\n if (matched && !(matched as any).dataset?.clickEventTracking) {\n let evt: any = {\n name: event.name,\n properties: event.properties,\n cache,\n }\n\n if (typeof callback === 'function') {\n const callbackResult = callback(e)\n evt = {\n ...callbackResult,\n cache: callbackResult.cache ?? cache,\n }\n }\n\n ;(matched as any).dataset.clickEventTracking = 'true'\n this.track(evt, evt.cache ?? false)\n }\n }\n\n document.addEventListener('click', delegatedHandler)\n this.delegatedSelectors.add(selector)\n }\n } else {\n // Element or NodeList: attach directly to existing elements\n const elements = element instanceof NodeList ? Array.from(element) : [element]\n\n elements.forEach(el => {\n if (el && !(el as any).dataset?.clickEventTracking) {\n this.listen(el as Element, event, cache, callback)\n }\n })\n }\n })\n\n return this\n }\n\n /**\n * Backward compatibility alias (with typo from original)\n */\n addEventhandler = this.addEventHandler.bind(this)\n\n /**\n * Load events immediately\n */\n loadEvent(items: LoadEventConfig[]): this {\n this.log(\n 'loadEvent | firing load events',\n items.map(i => i.event.name)\n )\n items.forEach(({ event }) => {\n const { name, properties } = event\n this.track({\n name,\n properties,\n })\n })\n\n return this\n }\n\n /**\n * Load events on specific pages\n */\n pageLoadEvent(items: PageLoadEventConfig[]): this {\n if (typeof window === 'undefined') return this\n\n const pathname = window.location.pathname.slice(1)\n this.log('pageLoadEvent | checking page load events', { pathname })\n\n items.forEach(({ pages = [], excludedPages = [], event, callback = null }) => {\n let dispatch = false\n if (pages.length) {\n if (pages.includes(pathname)) {\n dispatch = true\n }\n } else if (excludedPages.length) {\n if (!excludedPages.includes(pathname)) {\n dispatch = true\n }\n } else {\n dispatch = true\n }\n\n if (dispatch) {\n const eventData = callback ? callback() : event\n this.log('pageLoadEvent | dispatching event for page', { pathname, event: eventData.name })\n this.loadEvent([{ event: eventData }])\n } else {\n this.log('pageLoadEvent | skipped event for page', {\n pathname,\n event: event.name,\n pages,\n excludedPages,\n })\n }\n })\n\n return this\n }\n\n /**\n * Clear the interval and cleanup\n */\n clearInterval(): void {\n if (this.interval) {\n clearInterval(this.interval)\n this.interval = null\n }\n }\n\n /**\n * Cleanup method for removing event listeners and intervals\n */\n cleanup(): void {\n this.clearInterval()\n this.responses = []\n this.isTrackingResponses = false\n }\n}\n\n// Create singleton instance\nexport const cacheTrackEvents = new AnalyticsCacheManager()\n\n// Export to global scope for backward compatibility\nif (typeof window !== 'undefined') {\n ;(window as any).cacheTrackEvents = cacheTrackEvents\n}\n"]}
@@ -0,0 +1,4 @@
1
+ 'use strict';var m=require('js-cookie');function _interopDefault(e){return e&&e.__esModule?e:{default:e}}var m__default=/*#__PURE__*/_interopDefault(m);/* @deriv-com/analytics - NPM Package - Built with tsup */
2
+ var i="https://deriv.com/cdn-cgi/trace",y="https://cdn.growthbook.io",f="https://deriv-dataplane.rudderstack.com",g="https://ph.deriv.com",h="https://us.posthog.com",p=["deriv.com","deriv.be","deriv.me","deriv.team","deriv.ae"],c=["deriv.com","derivcrypto.com","besquare.my","besquare.com.my","ewallet.exchange","champion-fx.com","opalstraits.com","binary.com","binary.marketing","championgbs.com","4x.my","re-work.dev","regentmarkets.com","4x.com","binary.me","deriv.team","firstsource.io","firstsource.tech","deriv.hr","vmgbpo.net","mailisk.net","mailosaur.net","mobileapps.mailisk.net","w3e180zd.mailosaur.net"],x=()=>{if(typeof window>"u")return ".deriv.com";let t=window.location.hostname;if(t==="localhost")return "";let e=p.find(o=>t.includes(o));return e?`.${e}`:".deriv.com"};var b=(t,e)=>(...o)=>{e()&&console.log(`[ANALYTIC]${t}`,...o);},O=t=>{let e=t.split("@")[1]?.toLowerCase();return e?c.includes(e):false},D=t=>/^[0-9a-f]{8}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{12}$/i.test(t),a=null,R=async()=>{if(a)return a;let t=JSON.parse(m__default.default.get("website_status")||"{}")?.clients_country;return a=(async()=>{try{let e=await fetch(i).catch(()=>null);if(!e)return t||"";let o=await e.text().catch(()=>"");return o?Object.fromEntries(o.split(`
3
+ `).map(r=>r.split("=",2))).loc?.toLowerCase()||t||"":t||""}catch{return t||""}})(),a},l=t=>{if(t==null||typeof t!="object")return t;if(Array.isArray(t)){let n=[];for(let r=0;r<t.length;r++){let s=l(t[r]);s!=null&&n.push(s);}return n.length?n:void 0}let e={},o=false;for(let n in t){if(!Object.prototype.hasOwnProperty.call(t,n))continue;let r=l(t[n]);r==null||r===""||typeof r=="object"&&!Array.isArray(r)&&Object.keys(r).length===0||Array.isArray(r)&&r.length===0||(e[n]=r,o=true);}return o?e:void 0},u=(t,e)=>{for(let o in t){if(!Object.prototype.hasOwnProperty.call(t,o))continue;let n=t[o];n&&typeof n=="object"&&!Array.isArray(n)?u(n,e):e[o]=n;}},C=t=>{if(t==null||typeof t!="object"||Array.isArray(t))return t;let e={};return u(t,e),e};exports.a=y;exports.b=f;exports.c=g;exports.d=h;exports.e=p;exports.f=x;exports.g=b;exports.h=O;exports.i=D;exports.j=R;exports.k=l;exports.l=C;//# sourceMappingURL=chunk-BVGYK54Z.js.map
4
+ //# sourceMappingURL=chunk-BVGYK54Z.js.map