@obelism/improve-sdk 1.1.0 → 2.0.0
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.
- package/dist/client.cjs +1 -1
- package/dist/client.cjs.map +1 -1
- package/dist/client.d.ts +34 -7
- package/dist/client.d.ts.map +1 -1
- package/dist/client.mjs +1 -1
- package/dist/client.mjs.map +1 -1
- package/dist/server.cjs +1 -1
- package/dist/server.cjs.map +1 -1
- package/dist/server.d.ts +7 -1
- package/dist/server.d.ts.map +1 -1
- package/dist/server.mjs +1 -1
- package/dist/server.mjs.map +1 -1
- package/dist/types.d.ts +63 -4
- package/dist/types.d.ts.map +1 -1
- package/package.json +1 -1
package/dist/client.cjs
CHANGED
|
@@ -1,2 +1,2 @@
|
|
|
1
|
-
Object.defineProperty(exports,"__esModule",{value:!0});var t=require("ua-parser-js");let e=["wearable","mobile","tablet","console","smarttv","embedded","desktop"],i=["chrome","safari","firefox","edge","ie","samsung internet","social","other"],r=["mac os","ios","android","windows","unix"],s=["tiktok","wechat","weibo","snapchat","klarna","Line","instagram","facebook","alipay","Baidu"],n=["wearable","mobile","tablet"],o=(t,e)=>!t||Object.entries(t).every(([t,i])=>e[t]===i),a=t=>{if(0===t.length)return null;if(1===t.length)return t[0].slug;let e=Math.random()*t.reduce((t,{split:e})=>t+e,0);return(t.find(({split:t})=>(e-=t)<=0)||t[0]).slug},l="visi",
|
|
1
|
+
Object.defineProperty(exports,"__esModule",{value:!0});var t=require("ua-parser-js");let e=["wearable","mobile","tablet","console","smarttv","embedded","desktop"],i=["chrome","safari","firefox","edge","ie","samsung internet","social","other"],r=["mac os","ios","android","windows","unix"],s=["tiktok","wechat","weibo","snapchat","klarna","Line","instagram","facebook","alipay","Baidu"],n=["wearable","mobile","tablet"],o=(t,e)=>!t||Object.entries(t).every(([t,i])=>e[t]===i),a=t=>{if(0===t.length)return null;if(1===t.length)return t[0].slug;let e=Math.random()*t.reduce((t,{split:e})=>t+e,0);return(t.find(({split:t})=>(e-=t)<=0)||t[0]).slug},l="visi",h="https://improve.obelism.studio",u="/api/log",d=t=>new Promise(e=>setTimeout(e,t)),c={timeout:"Configuration request timed out",network:"Configuration request could not reach the server",unauthorized:"Configuration request was rejected (403): verify the organizationId, the server token, and that the request origin is whitelisted for this environment","rate-limited":"Configuration request was rate limited (429): too many requests, or the organization usage limit was reached",server:"Configuration request failed with a server error",client:"Configuration request was rejected by the server","invalid-response":"Configuration response could not be parsed as JSON"};class f extends Error{constructor(t,e){super(c[t]),this.name="ImproveFetchError",this.reason=t,this.status=e?.status,this.retryAfterMs=e?.retryAfterMs,e&&"cause"in e&&(this.cause=e.cause)}get isRetryable(){return"rate-limited"===this.reason||"server"===this.reason||"timeout"===this.reason||"network"===this.reason}}let g=t=>429===t?"rate-limited":401===t||403===t?"unauthorized":t>=500?"server":"client",v="ABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789",m=t=>{if("object"!=typeof t||null===t||Array.isArray(t))throw new f("invalid-response");return{name:"string"==typeof t.name?t.name:"",version:"number"==typeof t.version?t.version:0,tests:t.tests??{},flags:t.flags??{},audience:t.audience??{}}},p=(t,e=Date.now())=>{if(!t)return;let i=Number(t);if(Number.isFinite(i))return Math.max(0,Math.round(1e3*i));let r=Date.parse(t);if(!Number.isNaN(r))return Math.max(0,r-e)},w=(t,e,i)=>Math.round(Math.random()*Math.min(i,e*2**t)),y=async(t=1e3,e)=>(await d(t),e?.abort(),null),b=(t=3e3,e,i)=>{let r=new AbortController;return Promise.race([fetch(e,{...i,signal:r.signal}),y(t,r)])};class C{#t;#e;constructor({organizationId:t,environment:e,state:i,config:r,fetchTimeout:s,baseUrl:n,configRetries:o,configRevalidate:a,disableWarnings:u}){this.environment="develop",this.#t=null,this.#e=!1,this.config=null,this._fetchConfig=async t=>{if(this.config)return;if(!this.#t)throw Error("No config fetch setup provided");let{url:e,timeout:i,retries:r,revalidate:s}=this.#t,n={...t};"number"==typeof s&&(n.next={...t?.next,revalidate:s});let o=new f("network");for(let t=0;t<=r;t++){let r;t>0&&await d(o.retryAfterMs??w(t-1,300,3e3));try{r=await b(i,e,n)}catch(t){o=new f(t instanceof Error&&"AbortError"===t.name?"timeout":"network",{cause:t});continue}if(!r){o=new f("timeout");continue}if(r.ok){try{this.config=m(await r.json())}catch(t){o=t instanceof f?t:new f("invalid-response",{cause:t});continue}return this.config}if(!(o=new f(g(r.status),{status:r.status,retryAfterMs:p(r.headers.get("Retry-After"))})).isRetryable)break}throw o},this.loadConfig=t=>{this.config=t},this.generateVisitorId=()=>[l,(function(t=5){return t&&"number"==typeof t?Array(t).fill("").reduce(t=>t+=v.charAt(Math.floor(Math.random()*v.length)),""):""})(26).toUpperCase()].join("_"),this.getVisitorCookieName=()=>"visitorId",this._warn=t=>{this.#e||"u">typeof console&&console.warn(`[Improve] ${t}`)},this.validateTestValue=(t,e)=>{if(!this.config)throw Error("Config is required before validating, either use `.fetchConfig()`, .loadConfig(config) or provide it during setup");let i=this.config.tests[t];if(!i)throw Error(`No config found for ${t}`);return!!i.options.find(t=>t.slug===e)},this.validateVisitorId=t=>{let e=t.split("_");if(2!==e.length)return!1;let[i,r]=e;return i===l&&26===r.length},this.organizationId=t,this.environment=e,this.state=i,this._baseUrl=n||h,this.#e=u??!1,r?this.config=r:this.#t={url:[`${this._baseUrl}/config`,this.organizationId,this.environment,this.state||"active"].join("/"),timeout:s||3e3,retries:o??2,revalidate:a}}}let x=t=>{if(!t)return!1;let e=document.cookie.split("; ").find(e=>{let[i]=e.split("=");return t===i});return!!e&&e.split("=")[1]},I=(t,e)=>{let i=new Date;i.setDate(i.getDate()+30),document.cookie=`${t}=${e};path=/;expires=${i.toUTCString()};SameSite=Lax;Secure`},E=/^[a-z][a-z0-9]*(_[a-z0-9]+)*$/,k=(t,e)=>"string"==typeof t&&t.length>e?t.slice(0,e):t;exports.ImproveClientSDK=class extends C{#i;#r;#s;#n;#o;#a;#l;#h;#u;constructor(l){super(l),this.#r=!1,this.#s="",this.#n={},this.#o={},this.#a=0,this.#l=new Set,this.#h=`${h}${u}`,this.#u=!0,this.fetchConfig=this._fetchConfig,this.setupVisitor=(o=window.navigator.userAgent)=>{let a=x(this.getVisitorCookieName()),l=a&&this.validateVisitorId(a);this.#r=l,this.#s=l?a:this.generateVisitorId();let h=(o=>{var a;if(!o||"string"!=typeof o)return null;let l=new t.UAParser(o).getResult(),h=(a=l.device.type)&&e.includes(a)?a:"desktop";return{pointer:n.includes(h)?"coarse":"fine",device:h,browser:((t="")=>{if(!t)return"other";let e=t.toLowerCase(),r=i.find(t=>e.includes(t));return r||(s.includes(e)?"social":"other")})(l.browser.name),os:((t="")=>{if(!t)return"unix";let e=t.toLowerCase(),i=r.find(t=>e.includes(t));return i||"unix"})(l.os.name)}})(o);return h?(this.#i=h,I(this.getVisitorCookieName(),this.#s),this.#s):null},this.getFlagValue=t=>{if(!this.config)return null;let e=this.config.flags[t];if(!e||!e.options[0])return null;if(this.#i||this.setupVisitor(),!this.#s||!this.#i)return e.options[0].slug;if(this.#i?.[t])return this.#i[t];if(!o(this.config.audience[e.audience],this.#i))return e.options[0].slug;let i=x(t)||a(e.options);return i?(this.#i[t]=i,I(t,i),this.#d("flag",e.id,i),i):null},this.getTestValue=t=>{if(!this.config)return null;let e=this.config.tests[t];if(!e)return null;if(this.#i||this.setupVisitor(),!this.#s||!this.#i)return e.defaultValue;if(this.#i?.[t])return this.#i[t];if(!o(this.config.audience[e.audience],this.#i))return e.defaultValue;if(e.allocation<100&&100*Math.random()>e.allocation)return this.#i[t]=e.defaultValue,this.#i?.[t];let i=x(t),r=i&&this.validateTestValue(t,i)?i:a(e.options);return r?(this.#i[t]=r,I(t,r),this.#d("test",e.id,r),r):null},this.setAnalyticsUrls=t=>{this.#h=t},this.#c=t=>{let e;return{type:t,organizationId:this.organizationId,environment:this.environment,visitorId:this.#s,pointer:this.#i.pointer,device:this.#i.device,screen:(e=window.innerWidth)<=768?"small":e<=1024?"medium":e<=1200?"large":"huge",browser:this.#i.browser,os:this.#i.os,visitor:this.#r?"recurring":"new"}},this.#f=(t,e)=>{let i=JSON.stringify(t);"u">typeof TextEncoder&&new TextEncoder().encode(i).length>8192&&this._warn(`Beacon for "${e}" exceeds the 8192-byte limit and will be rejected by the server — trim the \`params\` payload.`);let r=fetch(this.#h,{method:"POST",headers:{"Content-Type":"application/json"},body:i,keepalive:!0});return r.then(t=>{if(t?.status===429){let e=p(t.headers.get("Retry-After"));this.#a=Date.now()+(e??6e4)}}).catch(()=>{}),r},this.#d=(t,e,i)=>{if(!this.#i||!this.#s||Date.now()<this.#a)return;let r=`${t}:${e}`;if(this.#o[r])return;this.#o[r]=!0;let s={...this.#c("exposure"),subjectKind:t,subjectId:e,variant:i};return this.#f(s,r)},this.postAnalytic=(t,e)=>{if(!this.config||t.startsWith("gtm."))return null;if(E.test(t)||this.#l.has(t)||(this.#l.add(t),this._warn(`Event name "${t}" isn't snake_case. Prefer GA4-style names like "add_to_cart" or "purchase" so your events line up with GA4 / Google Tag Manager. Pass \`disableWarnings: true\` at setup to silence this.`)),Date.now()<this.#a)return null;let{value:i,currency:r,message:s,params:n}="string"==typeof e?{message:e}:e??{},o="number"==typeof i&&Number.isFinite(i),a=n&&"object"==typeof n&&!Array.isArray(n)?n:void 0;if(this.#i||this.setupVisitor(),!this.#i||this.#n[t])return null;this.#n[t]=!0;let l={...this.#c("event"),event:k(t,256),message:k(s||"",256),...o?{value:i}:{},...r?{currency:k(r,8)}:{},...a?{params:a}:{}};if(this.#u){var h;a&&"ecommerce"in a&&"u">typeof window&&(window.dataLayer=window.dataLayer||[],window.dataLayer.push({ecommerce:null})),h={event:t,improve:{visitorId:this.#s},...o?{value:i}:{},...r?{currency:r}:{},...a??{},_improve:!0},"u">typeof window&&(window.dataLayer=window.dataLayer||[],window.dataLayer.push(h))}return this.#f(l,t)},this.#h=`${this._baseUrl}${u}`,this.#u=l.dataLayer??!0}#c;#f;#d};
|
|
2
2
|
//# sourceMappingURL=client.cjs.map
|
package/dist/client.cjs.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"client.cjs","sources":["../src/config/audiences.ts","../src/utils/parseUserAgent.ts","../src/utils/getVisitorMatchesAudience.ts","../src/utils/getRandomTestValue.ts","../src/config/constants.ts","../src/config/urls.ts","../src/utils/delay.ts","../src/utils/errors.ts","../src/utils/getRandomString.ts","../src/utils/normalizeConfig.ts","../src/utils/retry.ts","../src/utils/getTimeoutError.ts","../src/utils/timeoutFetch.ts","../src/base.ts","../src/utils/clientCookie.ts","../src/utils/truncate.ts","../src/client.ts","../src/utils/pushDataLayer.ts","../src/utils/getScreenSize.ts"],"sourcesContent":["/**\n * @constant AUDIENCE_PARAMS\n * @description All posibile audience tracking options\n */\nexport const AUDIENCE_PARAMS = {\n\t//? Technical\n\tpointer: ['coarse', 'fine'],\n\tdevice: [\n\t\t'wearable',\n\t\t'mobile',\n\t\t'tablet',\n\t\t'console',\n\t\t'smarttv',\n\t\t'embedded',\n\t\t'desktop',\n\t],\n\tbrowser: [\n\t\t'chrome',\n\t\t'safari',\n\t\t'firefox',\n\t\t'edge',\n\t\t'ie',\n\t\t'samsung internet',\n\t\t'social',\n\t\t'other',\n\t],\n\tos: ['mac os', 'ios', 'android', 'windows', 'unix'],\n} as const\n\n//? Generated types\nexport type AudienceParamKey = keyof typeof AUDIENCE_PARAMS\n\nexport type AudienceParamPointer = (typeof AUDIENCE_PARAMS.pointer)[number]\n\nexport type AudienceParamDevice = (typeof AUDIENCE_PARAMS.device)[number]\n\nexport type AudienceParamBrowser = (typeof AUDIENCE_PARAMS.browser)[number]\n\nexport type AudienceParamOs = (typeof AUDIENCE_PARAMS.os)[number]\n\nexport const AUDIENCE_PARAM_KEYS = Object.keys(\n\tAUDIENCE_PARAMS,\n) as ReadonlyArray<AudienceParamKey>\n","import { UAParser } from 'ua-parser-js'\n\nimport {\n\tAUDIENCE_PARAMS,\n\tAudienceParamBrowser,\n\tAudienceParamDevice,\n\tAudienceParamOs,\n\tAudienceParamPointer,\n} from '../config/audiences'\n\nexport type ParsedUserAgent = {\n\tpointer: AudienceParamPointer\n\tdevice: AudienceParamDevice\n\tbrowser: AudienceParamBrowser\n\tos: AudienceParamOs\n}\n\nconst formatDeviceType = (deviceType?: string): AudienceParamDevice => {\n\tif (\n\t\tdeviceType &&\n\t\tAUDIENCE_PARAMS.device.includes(deviceType as AudienceParamDevice)\n\t) {\n\t\treturn deviceType as AudienceParamDevice\n\t}\n\treturn 'desktop'\n}\n\nconst SOCIAL_BROWSERS: ReadonlyArray<string> = [\n\t'tiktok',\n\t'wechat',\n\t'weibo',\n\t'snapchat',\n\t'klarna',\n\t'Line',\n\t'instagram',\n\t'facebook',\n\t'alipay',\n\t'Baidu',\n]\n\nconst formatBrowser = (browserName: string = ''): AudienceParamBrowser => {\n\tif (!browserName) return 'other'\n\n\tconst name = browserName.toLowerCase()\n\n\tconst browserTypeMatch = AUDIENCE_PARAMS.browser.find((browserType) => {\n\t\treturn name.includes(browserType)\n\t})\n\n\tif (browserTypeMatch) return browserTypeMatch\n\n\tif (SOCIAL_BROWSERS.includes(name)) return 'social'\n\treturn 'other'\n}\n\nconst PRIMARY_TOUCH_DEVICES: ReadonlyArray<AudienceParamDevice> = [\n\t'wearable',\n\t'mobile',\n\t'tablet',\n]\n\nconst formatPointer = (device: AudienceParamDevice) => {\n\treturn PRIMARY_TOUCH_DEVICES.includes(device) ? 'coarse' : 'fine'\n}\n\nconst formatOs = (osName: string = ''): AudienceParamOs => {\n\tif (!osName) return 'unix'\n\n\tconst os = osName.toLowerCase()\n\n\tconst osTypeMatch = AUDIENCE_PARAMS.os.find((osType) => {\n\t\treturn os.includes(osType)\n\t})\n\n\tif (osTypeMatch) return osTypeMatch\n\treturn 'unix'\n}\n\nexport const parseUserAgent = (userAgent: string): ParsedUserAgent | null => {\n\tif (!userAgent || typeof userAgent !== 'string') return null\n\n\tconst parser = new UAParser(userAgent)\n\tconst results = parser.getResult()\n\n\tconst device = formatDeviceType(results.device.type)\n\n\treturn {\n\t\tpointer: formatPointer(device),\n\t\tdevice: device,\n\t\tbrowser: formatBrowser(results.browser.name),\n\t\tos: formatOs(results.os.name),\n\t}\n}\n","import { ImproveAudienceValue } from '../types'\nimport { AudienceParamKey } from '../config/audiences'\nimport { ParsedUserAgent } from './parseUserAgent'\n\nexport const getVisitorMatchesAudience = (\n\taudience: ImproveAudienceValue | undefined,\n\tvisitorParams: ParsedUserAgent,\n) => {\n\tif (!audience) return true\n\treturn Object.entries(audience).every(([paramKey, paramValue]) => {\n\t\treturn visitorParams[paramKey as AudienceParamKey] === paramValue\n\t})\n}\n","import { ImproveTestOption } from '../types'\n\n// function cryptoRandom() {\n// \treturn crypto.getRandomValues(new Uint32Array(1))[0] / (0xffffffff + 1)\n// }\n\nexport const getRandomTestValue = (options: ImproveTestOption[]) => {\n\tif (options.length === 0) return null\n\tif (options.length === 1) return options[0].slug\n\n\t// Get a random number between 0 and 1\n\tconst sum = options.reduce((acc, { split }) => acc + split, 0)\n\tlet value = Math.random() * sum\n\n\tconst match =\n\t\toptions.find(({ split }) => {\n\t\t\tvalue -= split\n\t\t\treturn value <= 0\n\t\t}) || options[0]\n\n\treturn match.slug\n}\n","export const COOKIE_NAME_VISITOR = 'visitorId'\n\nexport const VISITOR_ID_PREFIX = 'visi'\n\nexport const VISITOR_ID_LENGTH = 26\n\nexport const VISITOR_ID_SEPARATOR = '_'\n\n/** Default number of retries for a failed config fetch (network / 5xx / 429). */\nexport const CONFIG_RETRY_COUNT = 2\n\n/** Base delay for config-fetch backoff, in ms. */\nexport const CONFIG_RETRY_BASE_DELAY_MS = 300\n\n/** Upper bound for a single config-fetch backoff wait, in ms. */\nexport const CONFIG_RETRY_MAX_DELAY_MS = 3000\n\n/**\n * Max length of a single analytic string field. The backend rejects any\n * field longer than this (varchar(256)) with a 400, so the client caps\n * developer-controlled values (event, message) to avoid silently dropping\n * the event.\n */\nexport const MAX_ANALYTIC_FIELD_LENGTH = 256\n\n/**\n * How long to stop sending analytics after the backend returns 429 (usage\n * limit / rate limit) without a `Retry-After`. Avoids hammering an org that is\n * already over its quota.\n */\nexport const ANALYTIC_RATE_LIMIT_COOLDOWN_MS = 60_000\n","export const BASE_URL = 'https://improve.obelism.studio'\nexport const CONFIG_PATH = `/config`\nexport const ANALYTICS_PATH = `/api/log`\n","export const delay = (ms: number) =>\n\tnew Promise((resolve) => setTimeout(resolve, ms))\n","export type ImproveFetchErrorReason =\n\t| 'timeout'\n\t| 'network'\n\t| 'unauthorized'\n\t| 'rate-limited'\n\t| 'server'\n\t| 'client'\n\t| 'invalid-response'\n\nconst REASON_MESSAGES: Record<ImproveFetchErrorReason, string> = {\n\ttimeout: 'Configuration request timed out',\n\tnetwork: 'Configuration request could not reach the server',\n\tunauthorized:\n\t\t'Configuration request was rejected (403): verify the organizationId, the server token, and that the request origin is whitelisted for this environment',\n\t'rate-limited':\n\t\t'Configuration request was rate limited (429): too many requests, or the organization usage limit was reached',\n\tserver: 'Configuration request failed with a server error',\n\tclient: 'Configuration request was rejected by the server',\n\t'invalid-response': 'Configuration response could not be parsed as JSON',\n}\n\n/**\n * Error thrown by the config fetch layer. Carries the HTTP `status`, a coarse\n * `reason`, and (when the server sent one) the `Retry-After` delay in ms, so\n * callers can distinguish a misconfiguration (unauthorized) from a transient\n * failure (rate-limited/server/timeout) instead of guessing from a string.\n */\nexport class ImproveFetchError extends Error {\n\treason: ImproveFetchErrorReason\n\tstatus?: number\n\tretryAfterMs?: number\n\n\tconstructor(\n\t\treason: ImproveFetchErrorReason,\n\t\toptions?: { status?: number; retryAfterMs?: number; cause?: unknown },\n\t) {\n\t\tsuper(REASON_MESSAGES[reason])\n\t\tthis.name = 'ImproveFetchError'\n\t\tthis.reason = reason\n\t\tthis.status = options?.status\n\t\tthis.retryAfterMs = options?.retryAfterMs\n\t\tif (options && 'cause' in options) {\n\t\t\t;(this as { cause?: unknown }).cause = options.cause\n\t\t}\n\t}\n\n\t/** Whether retrying the request could plausibly succeed. */\n\tget isRetryable() {\n\t\treturn (\n\t\t\tthis.reason === 'rate-limited' ||\n\t\t\tthis.reason === 'server' ||\n\t\t\tthis.reason === 'timeout' ||\n\t\t\tthis.reason === 'network'\n\t\t)\n\t}\n}\n\n/** Map an HTTP status onto a coarse {@link ImproveFetchErrorReason}. */\nexport const getReasonFromStatus = (\n\tstatus: number,\n): ImproveFetchErrorReason => {\n\tif (status === 429) return 'rate-limited'\n\tif (status === 401 || status === 403) return 'unauthorized'\n\tif (status >= 500) return 'server'\n\treturn 'client'\n}\n","/**\n * @constant CHAR_SET\n * @description Key/value set of characters to be used for generation strings\n */\nconst CHAR_SET = 'ABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789'\n\n/**\n * @function getRandomString\n * @description Generate a random string\n */\nexport function getRandomString(characters: number = 5): string {\n\tif (!characters || typeof characters !== 'number') return ''\n\treturn Array(characters)\n\t\t.fill('')\n\t\t.reduce((acc) => {\n\t\t\tacc += CHAR_SET.charAt(Math.floor(Math.random() * CHAR_SET.length))\n\t\t\treturn acc\n\t\t}, '')\n}\n","import { ImproveConfiguration } from '../types'\nimport { ImproveFetchError } from './errors'\n\n/**\n * @function normalizeConfig\n * @description Coerce a parsed config response into a well-formed\n * {@link ImproveConfiguration}. The backend returns `200 {}` for an unknown\n * org/environment (and on its own internal error). An empty object is truthy,\n * so it slips past `if (!this.config)` guards and then crashes downstream on\n * `config.tests[slug]` (`Cannot read properties of undefined`). Guaranteeing\n * `tests`/`flags`/`audience` are always objects lets the SDK fail open to\n * default values instead of throwing during render/middleware.\n *\n * Throws {@link ImproveFetchError} `invalid-response` when the body isn't an\n * object at all (e.g. an array, string, or null).\n */\nexport const normalizeConfig = (body: unknown): ImproveConfiguration => {\n\tif (typeof body !== 'object' || body === null || Array.isArray(body)) {\n\t\tthrow new ImproveFetchError('invalid-response')\n\t}\n\n\tconst raw = body as Partial<ImproveConfiguration>\n\n\treturn {\n\t\tname: typeof raw.name === 'string' ? raw.name : '',\n\t\tversion: typeof raw.version === 'number' ? raw.version : 0,\n\t\ttests: raw.tests ?? {},\n\t\tflags: raw.flags ?? {},\n\t\taudience: raw.audience ?? {},\n\t}\n}\n","/**\n * @function parseRetryAfterMs\n * @description Parse an HTTP `Retry-After` header into milliseconds. Supports\n * both the delta-seconds form (`\"120\"`) and the HTTP-date form\n * (`\"Wed, 21 Oct 2026 07:28:00 GMT\"`). Returns `undefined` when the header is\n * absent or unparseable so the caller can fall back to computed backoff.\n *\n * @param {string | null} [header] - the raw `Retry-After` header value\n * @param {number} [now] - current epoch ms (injectable for testing)\n */\nexport const parseRetryAfterMs = (\n\theader: string | null | undefined,\n\tnow: number = Date.now(),\n): number | undefined => {\n\tif (!header) return undefined\n\n\tconst seconds = Number(header)\n\tif (Number.isFinite(seconds)) return Math.max(0, Math.round(seconds * 1000))\n\n\tconst date = Date.parse(header)\n\tif (Number.isNaN(date)) return undefined\n\n\treturn Math.max(0, date - now)\n}\n\n/**\n * @function getBackoffDelayMs\n * @description Exponential backoff with full jitter, capped at `maxDelay`.\n * Jitter spreads retries so concurrent clients don't stampede the origin in\n * lockstep after a shared failure.\n *\n * @param {number} attempt - zero-based retry attempt index\n * @param {number} baseDelay - base delay in ms\n * @param {number} maxDelay - upper bound in ms\n */\nexport const getBackoffDelayMs = (\n\tattempt: number,\n\tbaseDelay: number,\n\tmaxDelay: number,\n): number => {\n\tconst exponential = Math.min(maxDelay, baseDelay * 2 ** attempt)\n\treturn Math.round(Math.random() * exponential)\n}\n","import { delay } from './delay'\n\n/**\n * @async @function getTimeoutError\n * @description Throw an error after a delay\n *\n * @param {number} [timeout] - time in ms after to reject default: 1000\n * @param {AbortController} [controller] - (optional) AbortController to abort the request\n */\nexport const getTimeoutError = async (\n\ttimeout: number = 1000,\n\tcontroller: AbortController,\n): Promise<null> => {\n\tawait delay(timeout)\n\tcontroller?.abort()\n\treturn null\n}\n","import { getTimeoutError } from './getTimeoutError'\n\n/**\n * @async @function timeoutFetch\n * @description Fetch with a timeout\n *\n * @param {number} timeout - time in ms after to reject default: 3000\n * @param {string} url - url to fetch\n * @param {Object} [options] - (optional) options to pass to fetch\n */\nexport const timeoutFetch = (\n\ttimeout: number = 3000,\n\turl: string,\n\toptions?: object,\n) => {\n\tconst controller = new AbortController()\n\treturn Promise.race([\n\t\tfetch(url, {\n\t\t\t...options,\n\t\t\tsignal: controller.signal,\n\t\t}),\n\t\tgetTimeoutError(timeout, controller),\n\t])\n}\n","import {\n\tCONFIG_RETRY_BASE_DELAY_MS,\n\tCONFIG_RETRY_COUNT,\n\tCONFIG_RETRY_MAX_DELAY_MS,\n\tCOOKIE_NAME_VISITOR,\n\tVISITOR_ID_LENGTH,\n\tVISITOR_ID_PREFIX,\n\tVISITOR_ID_SEPARATOR,\n} from './config/constants'\nimport { CONFIG_PATH, BASE_URL } from './config/urls'\nimport {\n\tImproveConfiguration,\n\tImproveEnvironmentOption,\n\tImproveSetupArgs,\n\tImproveTestState,\n} from './types'\nimport { delay } from './utils/delay'\nimport { getReasonFromStatus, ImproveFetchError } from './utils/errors'\nimport { getRandomString } from './utils/getRandomString'\nimport { normalizeConfig } from './utils/normalizeConfig'\nimport { getBackoffDelayMs, parseRetryAfterMs } from './utils/retry'\nimport { timeoutFetch } from './utils/timeoutFetch'\n\ntype ConfigFetch = {\n\turl: string\n\ttimeout: number\n\tretries: number\n\trevalidate?: number\n}\n\n// Next.js augments RequestInit with a `next` field for its fetch cache.\ntype RequestInitWithNext = RequestInit & {\n\tnext?: { revalidate?: number; tags?: string[] }\n}\n\nexport class BaseImproveSDK {\n\torganizationId: string\n\tenvironment: ImproveEnvironmentOption = 'develop'\n\tstate: ImproveTestState\n\n\t#configFetch: ConfigFetch | null = null\n\n\tconfig: ImproveConfiguration | null = null\n\n\t_baseUrl: undefined | string\n\n\tconstructor({\n\t\torganizationId,\n\t\tenvironment,\n\t\tstate,\n\t\tconfig,\n\t\tfetchTimeout,\n\t\tbaseUrl,\n\t\tconfigRetries,\n\t\tconfigRevalidate,\n\t}: ImproveSetupArgs) {\n\t\tthis.organizationId = organizationId\n\t\tthis.environment = environment\n\t\tthis.state = state\n\t\tthis._baseUrl = baseUrl || BASE_URL\n\n\t\tif (config) {\n\t\t\tthis.config = config\n\t\t} else {\n\t\t\tthis.#configFetch = {\n\t\t\t\turl: [\n\t\t\t\t\t`${this._baseUrl}${CONFIG_PATH}`,\n\t\t\t\t\tthis.organizationId,\n\t\t\t\t\tthis.environment,\n\t\t\t\t\tthis.state || 'active',\n\t\t\t\t].join('/'),\n\t\t\t\ttimeout: fetchTimeout || 3000,\n\t\t\t\tretries: configRetries ?? CONFIG_RETRY_COUNT,\n\t\t\t\trevalidate: configRevalidate,\n\t\t\t}\n\t\t}\n\t}\n\n\t_fetchConfig = async (config?: RequestInit) => {\n\t\tif (this.config) return\n\n\t\tif (!this.#configFetch) throw new Error('No config fetch setup provided')\n\n\t\tconst { url, timeout, retries, revalidate } = this.#configFetch\n\n\t\t// Opt into the Next.js fetch cache when a revalidate window is configured,\n\t\t// so the datafile is reused across requests instead of re-fetched.\n\t\tconst requestInit: RequestInitWithNext = { ...config }\n\t\tif (typeof revalidate === 'number') {\n\t\t\trequestInit.next = {\n\t\t\t\t...(config as RequestInitWithNext | undefined)?.next,\n\t\t\t\trevalidate,\n\t\t\t}\n\t\t}\n\n\t\t// Seed with a generic network error so an exhausted loop always has a\n\t\t// meaningful, typed error to throw.\n\t\tlet lastError = new ImproveFetchError('network')\n\n\t\tfor (let attempt = 0; attempt <= retries; attempt++) {\n\t\t\tif (attempt > 0) {\n\t\t\t\t// Prefer a server-provided Retry-After; otherwise back off with jitter.\n\t\t\t\tawait delay(\n\t\t\t\t\tlastError.retryAfterMs ??\n\t\t\t\t\t\tgetBackoffDelayMs(\n\t\t\t\t\t\t\tattempt - 1,\n\t\t\t\t\t\t\tCONFIG_RETRY_BASE_DELAY_MS,\n\t\t\t\t\t\t\tCONFIG_RETRY_MAX_DELAY_MS,\n\t\t\t\t\t\t),\n\t\t\t\t)\n\t\t\t}\n\n\t\t\tlet res: Response | null\n\t\t\ttry {\n\t\t\t\tres = await timeoutFetch(timeout, url, requestInit)\n\t\t\t} catch (cause) {\n\t\t\t\t// timeoutFetch aborts on timeout (AbortError) and rejects on\n\t\t\t\t// network-level failures — both are transient and retryable.\n\t\t\t\tconst aborted = cause instanceof Error && cause.name === 'AbortError'\n\t\t\t\tlastError = new ImproveFetchError(aborted ? 'timeout' : 'network', {\n\t\t\t\t\tcause,\n\t\t\t\t})\n\t\t\t\tcontinue\n\t\t\t}\n\n\t\t\t// timeoutFetch resolves null when the timeout wins the race.\n\t\t\tif (!res) {\n\t\t\t\tlastError = new ImproveFetchError('timeout')\n\t\t\t\tcontinue\n\t\t\t}\n\n\t\t\tif (res.ok) {\n\t\t\t\ttry {\n\t\t\t\t\tthis.config = normalizeConfig(await res.json())\n\t\t\t\t} catch (cause) {\n\t\t\t\t\tlastError =\n\t\t\t\t\t\tcause instanceof ImproveFetchError\n\t\t\t\t\t\t\t? cause\n\t\t\t\t\t\t\t: new ImproveFetchError('invalid-response', { cause })\n\t\t\t\t\tcontinue\n\t\t\t\t}\n\t\t\t\treturn this.config\n\t\t\t}\n\n\t\t\tlastError = new ImproveFetchError(getReasonFromStatus(res.status), {\n\t\t\t\tstatus: res.status,\n\t\t\t\tretryAfterMs: parseRetryAfterMs(res.headers.get('Retry-After')),\n\t\t\t})\n\n\t\t\t// Auth/validation rejections won't change on retry — fail fast.\n\t\t\tif (!lastError.isRetryable) break\n\t\t}\n\n\t\tthrow lastError\n\t}\n\n\tloadConfig = (config: ImproveConfiguration) => {\n\t\tthis.config = config\n\t}\n\n\tgenerateVisitorId = () => {\n\t\treturn [\n\t\t\tVISITOR_ID_PREFIX,\n\t\t\tgetRandomString(VISITOR_ID_LENGTH).toUpperCase(),\n\t\t].join(VISITOR_ID_SEPARATOR)\n\t}\n\n\tgetVisitorCookieName = () => COOKIE_NAME_VISITOR\n\n\tvalidateTestValue = (testName: string, testValue: string) => {\n\t\tif (!this.config)\n\t\t\tthrow new Error(\n\t\t\t\t'Config is required before validating, either use `.fetchConfig()`, .loadConfig(config) or provide it during setup',\n\t\t\t)\n\n\t\tconst testConfig = this.config.tests[testName]\n\n\t\tif (!testConfig) throw new Error(`No config found for ${testName}`)\n\n\t\treturn Boolean(\n\t\t\ttestConfig.options.find((option) => option.slug === testValue),\n\t\t)\n\t}\n\n\tvalidateVisitorId = (possibleVisitorId: string) => {\n\t\tconst visitorIdParts = possibleVisitorId.split(VISITOR_ID_SEPARATOR)\n\t\tif (visitorIdParts.length !== 2) return false\n\t\tconst [key, value] = visitorIdParts as [string, string]\n\t\treturn key === VISITOR_ID_PREFIX && value.length === VISITOR_ID_LENGTH\n\t}\n}\n","const ROW_DELIMITER = '; '\nconst ENTRY_DELIMITER = '='\n\nexport const getCookie = (name: string) => {\n\tif (!name) return false\n\tconst cookie = document.cookie.split(ROW_DELIMITER).find((row) => {\n\t\tconst [key] = row.split(ENTRY_DELIMITER)\n\t\treturn name === key\n\t})\n\treturn cookie ? cookie.split(ENTRY_DELIMITER)[1] : false\n}\n\nexport const setCookie = (name: string, value: string) => {\n\tconst now = new Date()\n\tnow.setDate(now.getDate() + 30)\n\tdocument.cookie = `${name}=${value};path=/;expires=${now.toUTCString()};SameSite=Lax;Secure`\n}\n","/**\n * @function truncate\n * @description Cap a string to `maxLength` characters. Returns non-strings\n * unchanged so it is safe to apply to loosely-typed payload values.\n *\n * @param {string} value - the value to truncate\n * @param {number} maxLength - maximum length to keep\n */\nexport const truncate = (value: string, maxLength: number): string =>\n\ttypeof value === 'string' && value.length > maxLength\n\t\t? value.slice(0, maxLength)\n\t\t: value\n","import { ParsedUserAgent, parseUserAgent } from './utils/parseUserAgent'\nimport { getVisitorMatchesAudience } from './utils/getVisitorMatchesAudience'\nimport { getRandomTestValue } from './utils/getRandomTestValue'\nimport { BaseImproveSDK } from './base'\nimport { getCookie, setCookie } from './utils/clientCookie'\nimport {\n\tANALYTIC_RATE_LIMIT_COOLDOWN_MS,\n\tMAX_ANALYTIC_FIELD_LENGTH,\n} from './config/constants'\nimport { ANALYTICS_PATH, BASE_URL } from './config/urls'\nimport { getScreenSize } from './utils/getScreenSize'\nimport { pushDataLayer } from './utils/pushDataLayer'\nimport { parseRetryAfterMs } from './utils/retry'\nimport { truncate } from './utils/truncate'\nimport { ImproveEnvironmentOption, ImproveSetupArgs } from './types'\n\ntype Visitor = ParsedUserAgent & {\n\t[testSlug: string]: string\n}\n\nexport type CreateAnalytic = {\n\torganizationId: string\n\tenvironment: ImproveEnvironmentOption\n\n\ttestId: string\n\ttestValue: string\n\tvisitorId: string\n\n\tpointer: string\n\tdevice: string\n\tscreen: string\n\tbrowser: string\n\tos: string\n\tvisitor: string\n\n\tevent: string\n\tmessage: string\n}\n\ntype TrackedAnalytics = {\n\t[TestSlug: string]: {\n\t\t[Event: string]: boolean\n\t}\n}\n\nexport class ImproveClientSDK extends BaseImproveSDK {\n\t#visitor: Visitor | undefined\n\t#visitorRecurring: boolean = false\n\n\t#visitorId: string = ''\n\n\t#analytics: TrackedAnalytics = {}\n\n\t// Epoch ms until which analytics posting is suspended after a 429.\n\t#rateLimitedUntil: number = 0\n\n\t#analyticsUrl = `${BASE_URL}${ANALYTICS_PATH}`\n\n\t#dataLayerEnabled: boolean = true\n\n\tfetchConfig = this._fetchConfig\n\n\tconstructor(args: ImproveSetupArgs) {\n\t\tsuper(args)\n\t\tthis.#analyticsUrl = `${this._baseUrl}${ANALYTICS_PATH}`\n\t\tthis.#dataLayerEnabled = args.dataLayer ?? true\n\t}\n\n\tsetupVisitor = (userAgent: string = window.navigator.userAgent) => {\n\t\tconst cookieVisitorId = getCookie(this.getVisitorCookieName())\n\t\tconst validCookieVisitorId =\n\t\t\tcookieVisitorId && this.validateVisitorId(cookieVisitorId)\n\n\t\tthis.#visitorRecurring = validCookieVisitorId\n\t\tthis.#visitorId = validCookieVisitorId\n\t\t\t? cookieVisitorId\n\t\t\t: this.generateVisitorId()\n\n\t\tconst parsedUserAgent = parseUserAgent(userAgent)\n\n\t\tif (!parsedUserAgent) return null\n\n\t\tthis.#visitor = parsedUserAgent\n\n\t\tsetCookie(this.getVisitorCookieName(), this.#visitorId)\n\n\t\treturn this.#visitorId\n\t}\n\n\tgetFlagValue = (flagSlug: string) => {\n\t\tif (!this.config) return null\n\n\t\tconst flagConfig = this.config.flags[flagSlug]\n\n\t\tif (!flagConfig || !flagConfig.options[0]) return null\n\n\t\tif (!this.#visitor) this.setupVisitor()\n\t\tif (!this.#visitorId || !this.#visitor) return flagConfig.options[0].slug\n\t\tif (this.#visitor?.[flagSlug]) return this.#visitor[flagSlug]\n\n\t\tconst visitorMatchesAudience = getVisitorMatchesAudience(\n\t\t\tthis.config.audience[flagConfig.audience],\n\t\t\tthis.#visitor,\n\t\t)\n\n\t\tif (!visitorMatchesAudience) return flagConfig.options[0].slug\n\n\t\tconst flagValue =\n\t\t\tgetCookie(flagSlug) || getRandomTestValue(flagConfig.options)\n\n\t\tif (!flagValue) return null\n\n\t\tthis.#visitor[flagSlug] = flagValue\n\n\t\tsetCookie(flagSlug, flagValue)\n\n\t\treturn flagValue\n\t}\n\n\tgetTestValue = (testSlug: string) => {\n\t\tif (!this.config) return null\n\n\t\tconst testConfig = this.config.tests[testSlug]\n\n\t\tif (!testConfig) return null\n\n\t\tif (!this.#visitor) this.setupVisitor()\n\n\t\tif (!this.#visitorId || !this.#visitor) return testConfig.defaultValue\n\t\tif (this.#visitor?.[testSlug]) return this.#visitor[testSlug]\n\n\t\tconst visitorMatchesAudience = getVisitorMatchesAudience(\n\t\t\tthis.config.audience[testConfig.audience],\n\t\t\tthis.#visitor,\n\t\t)\n\n\t\tif (!visitorMatchesAudience) return testConfig.defaultValue\n\n\t\tif (\n\t\t\ttestConfig.allocation < 100 &&\n\t\t\tMath.random() * 100 > testConfig.allocation\n\t\t) {\n\t\t\tthis.#visitor[testSlug] = testConfig.defaultValue\n\t\t\treturn this.#visitor?.[testSlug]\n\t\t}\n\n\t\tconst cookieTestValue = getCookie(testSlug)\n\t\tconst validCookieTestValue =\n\t\t\tcookieTestValue && this.validateTestValue(testSlug, cookieTestValue)\n\t\tconst testValue = validCookieTestValue\n\t\t\t? cookieTestValue\n\t\t\t: getRandomTestValue(testConfig.options)\n\n\t\tif (!testValue) return null\n\n\t\tthis.#visitor[testSlug] = testValue\n\n\t\tsetCookie(testSlug, testValue)\n\n\t\treturn testValue\n\t}\n\n\tsetAnalyticsUrls = (url: string) => {\n\t\tthis.#analyticsUrl = url\n\t}\n\n\tpostAnalytic = (testSlug: string, event: string, message?: string) => {\n\t\tif (!this.config) return null\n\n\t\t// Suspended after a 429: skip sending, but don't mark the event as\n\t\t// tracked, so it can still fire once the cooldown elapses.\n\t\tif (Date.now() < this.#rateLimitedUntil) return null\n\n\t\tconst testConfig = this.config.tests[testSlug]\n\n\t\tif (!this.#visitor) this.setupVisitor()\n\t\tif (!testConfig || !this.#visitor || this.#analytics?.[testSlug]?.[event]) {\n\t\t\treturn null\n\t\t}\n\n\t\tconst testSlugAnalytics = this.#analytics[testSlug] || {}\n\t\ttestSlugAnalytics[event] = true\n\t\tthis.#analytics[testSlug] = testSlugAnalytics\n\n\t\tlet testValue = this.#visitor?.[testSlug] || null\n\t\tif (!testValue) {\n\t\t\ttestValue = this.getTestValue(testSlug) || null\n\t\t}\n\n\t\tif (!testValue) return\n\n\t\tconst body: CreateAnalytic = {\n\t\t\torganizationId: this.organizationId,\n\t\t\tenvironment: this.environment,\n\n\t\t\ttestId: testConfig.id,\n\t\t\ttestValue: testValue,\n\t\t\tvisitorId: this.#visitorId,\n\t\t\tpointer: this.#visitor.pointer,\n\t\t\tdevice: this.#visitor.device,\n\t\t\tscreen: getScreenSize(),\n\t\t\tbrowser: this.#visitor.browser,\n\t\t\tos: this.#visitor.os,\n\t\t\tvisitor: this.#visitorRecurring ? 'recurring' : 'new',\n\t\t\t// Cap developer-controlled fields to the backend's varchar(256) limit;\n\t\t\t// an over-length value is otherwise rejected with a 400 and the event\n\t\t\t// is lost.\n\t\t\tevent: truncate(event, MAX_ANALYTIC_FIELD_LENGTH),\n\t\t\tmessage: truncate(message || '', MAX_ANALYTIC_FIELD_LENGTH),\n\t\t}\n\n\t\tif (this.#dataLayerEnabled) {\n\t\t\tpushDataLayer({\n\t\t\t\tevent,\n\t\t\t\timprove: {\n\t\t\t\t\ttest: testSlug,\n\t\t\t\t\tvariant: testValue,\n\t\t\t\t\tvisitorId: this.#visitorId,\n\t\t\t\t},\n\t\t\t\t_improve: true,\n\t\t\t})\n\t\t}\n\n\t\t// `keepalive` lets the beacon outlive the page: conversion events are\n\t\t// often fired immediately before a navigation/unload, and a normal fetch\n\t\t// would be cancelled with the document. The payload is well under the\n\t\t// 64KB keepalive budget.\n\t\tconst request = fetch(this.#analyticsUrl, {\n\t\t\tmethod: 'POST',\n\t\t\theaders: { 'Content-Type': 'application/json' },\n\t\t\tbody: JSON.stringify(body),\n\t\t\tkeepalive: true,\n\t\t})\n\n\t\t// Back off when the org hits its usage/rate limit. Honor a server\n\t\t// Retry-After when present, otherwise use a fixed cooldown.\n\t\trequest\n\t\t\t.then((res) => {\n\t\t\t\tif (res?.status === 429) {\n\t\t\t\t\tconst retryAfterMs = parseRetryAfterMs(res.headers.get('Retry-After'))\n\t\t\t\t\tthis.#rateLimitedUntil =\n\t\t\t\t\t\tDate.now() + (retryAfterMs ?? ANALYTIC_RATE_LIMIT_COOLDOWN_MS)\n\t\t\t\t}\n\t\t\t})\n\t\t\t.catch(() => {})\n\n\t\treturn request\n\t}\n}\n","export type ImproveDataLayerEntry = {\n\tevent: string\n\timprove: {\n\t\ttest: string\n\t\tvariant: string\n\t\tvisitorId: string\n\t}\n\t/**\n\t * Marks the entry as originating from Improve so platform-side dataLayer\n\t * importers can ignore it (loop prevention).\n\t */\n\t_improve: true\n}\n\ndeclare global {\n\tinterface Window {\n\t\tdataLayer?: Record<string, unknown>[]\n\t}\n}\n\n/**\n * Mirror an analytic onto the GTM dataLayer (with experiment dimensions) so it\n * can drive Google Tag Manager / Google Ads conversions. No-op outside the\n * browser. Initializes window.dataLayer if GTM hasn't yet.\n */\nexport const pushDataLayer = (entry: ImproveDataLayerEntry) => {\n\tif (typeof window === 'undefined') return\n\n\twindow.dataLayer = window.dataLayer || []\n\twindow.dataLayer.push(entry)\n}\n","type SizeOptions = 'small' | 'medium' | 'large' | 'huge'\n\nexport const getScreenSize = (): SizeOptions => {\n\tconst size = window.innerWidth\n\tif (size <= 768) return 'small'\n\tif (size <= 1024) return 'medium'\n\tif (size <= 1200) return 'large'\n\treturn 'huge'\n}\n"],"names":["SOCIAL_BROWSERS","PRIMARY_TOUCH_DEVICES","getVisitorMatchesAudience","audience","visitorParams","Object","entries","every","paramKey","paramValue","getRandomTestValue","options","length","slug","value","Math","random","reduce","acc","split","match","find","VISITOR_ID_PREFIX","BASE_URL","ANALYTICS_PATH","delay","ms","Promise","resolve","setTimeout","REASON_MESSAGES","timeout","network","unauthorized","server","client","ImproveFetchError","Error","reason","name","status","retryAfterMs","cause","isRetryable","getReasonFromStatus","CHAR_SET","normalizeConfig","body","Array","isArray","raw","version","tests","flags","parseRetryAfterMs","header","now","Date","seconds","Number","isFinite","max","round","date","parse","isNaN","getBackoffDelayMs","attempt","baseDelay","maxDelay","min","getTimeoutError","controller","abort","timeoutFetch","url","AbortController","race","fetch","signal","BaseImproveSDK","organizationId","environment","state","config","fetchTimeout","baseUrl","configRetries","configRevalidate","_fetchConfig","retries","revalidate","requestInit","next","lastError","res","aborted","ok","json","headers","get","loadConfig","generateVisitorId","getRandomString","characters","fill","charAt","floor","toUpperCase","join","getVisitorCookieName","validateTestValue","testName","testValue","testConfig","Boolean","option","validateVisitorId","possibleVisitorId","visitorIdParts","key","_baseUrl","getCookie","cookie","document","row","setCookie","setDate","getDate","toUTCString","truncate","maxLength","slice","args","fetchConfig","setupVisitor","userAgent","window","navigator","cookieVisitorId","validCookieVisitorId","parsedUserAgent","parseUserAgent","deviceType","results","parser","UAParser","getResult","device","type","AUDIENCE_PARAMS","includes","pointer","browser","formatBrowser","browserName","toLowerCase","browserTypeMatch","browserType","os","formatOs","osName","osTypeMatch","osType","getFlagValue","flagSlug","flagConfig","flagValue","getTestValue","testSlug","defaultValue","allocation","cookieTestValue","validCookieTestValue","setAnalyticsUrls","postAnalytic","event","message","entry","size","testSlugAnalytics","testId","id","visitorId","screen","innerWidth","visitor","improve","test","variant","_improve","dataLayer","push","request","method","JSON","stringify","keepalive","then","ANALYTIC_RATE_LIMIT_COOLDOWN_MS","catch"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"client.cjs","sources":["../src/config/audiences.ts","../src/utils/parseUserAgent.ts","../src/utils/getVisitorMatchesAudience.ts","../src/utils/getRandomTestValue.ts","../src/config/constants.ts","../src/config/urls.ts","../src/utils/delay.ts","../src/utils/errors.ts","../src/utils/getRandomString.ts","../src/utils/normalizeConfig.ts","../src/utils/retry.ts","../src/utils/getTimeoutError.ts","../src/utils/timeoutFetch.ts","../src/base.ts","../src/utils/clientCookie.ts","../src/utils/isSnakeCaseEventName.ts","../src/utils/truncate.ts","../src/client.ts","../src/utils/getScreenSize.ts","../src/utils/pushDataLayer.ts"],"sourcesContent":["/**\n * @constant AUDIENCE_PARAMS\n * @description All posibile audience tracking options\n */\nexport const AUDIENCE_PARAMS = {\n\t//? Technical\n\tpointer: ['coarse', 'fine'],\n\tdevice: [\n\t\t'wearable',\n\t\t'mobile',\n\t\t'tablet',\n\t\t'console',\n\t\t'smarttv',\n\t\t'embedded',\n\t\t'desktop',\n\t],\n\tbrowser: [\n\t\t'chrome',\n\t\t'safari',\n\t\t'firefox',\n\t\t'edge',\n\t\t'ie',\n\t\t'samsung internet',\n\t\t'social',\n\t\t'other',\n\t],\n\tos: ['mac os', 'ios', 'android', 'windows', 'unix'],\n} as const\n\n//? Generated types\nexport type AudienceParamKey = keyof typeof AUDIENCE_PARAMS\n\nexport type AudienceParamPointer = (typeof AUDIENCE_PARAMS.pointer)[number]\n\nexport type AudienceParamDevice = (typeof AUDIENCE_PARAMS.device)[number]\n\nexport type AudienceParamBrowser = (typeof AUDIENCE_PARAMS.browser)[number]\n\nexport type AudienceParamOs = (typeof AUDIENCE_PARAMS.os)[number]\n\nexport const AUDIENCE_PARAM_KEYS = Object.keys(\n\tAUDIENCE_PARAMS,\n) as ReadonlyArray<AudienceParamKey>\n","import { UAParser } from 'ua-parser-js'\n\nimport {\n\tAUDIENCE_PARAMS,\n\tAudienceParamBrowser,\n\tAudienceParamDevice,\n\tAudienceParamOs,\n\tAudienceParamPointer,\n} from '../config/audiences'\n\nexport type ParsedUserAgent = {\n\tpointer: AudienceParamPointer\n\tdevice: AudienceParamDevice\n\tbrowser: AudienceParamBrowser\n\tos: AudienceParamOs\n}\n\nconst formatDeviceType = (deviceType?: string): AudienceParamDevice => {\n\tif (\n\t\tdeviceType &&\n\t\tAUDIENCE_PARAMS.device.includes(deviceType as AudienceParamDevice)\n\t) {\n\t\treturn deviceType as AudienceParamDevice\n\t}\n\treturn 'desktop'\n}\n\nconst SOCIAL_BROWSERS: ReadonlyArray<string> = [\n\t'tiktok',\n\t'wechat',\n\t'weibo',\n\t'snapchat',\n\t'klarna',\n\t'Line',\n\t'instagram',\n\t'facebook',\n\t'alipay',\n\t'Baidu',\n]\n\nconst formatBrowser = (browserName: string = ''): AudienceParamBrowser => {\n\tif (!browserName) return 'other'\n\n\tconst name = browserName.toLowerCase()\n\n\tconst browserTypeMatch = AUDIENCE_PARAMS.browser.find((browserType) => {\n\t\treturn name.includes(browserType)\n\t})\n\n\tif (browserTypeMatch) return browserTypeMatch\n\n\tif (SOCIAL_BROWSERS.includes(name)) return 'social'\n\treturn 'other'\n}\n\nconst PRIMARY_TOUCH_DEVICES: ReadonlyArray<AudienceParamDevice> = [\n\t'wearable',\n\t'mobile',\n\t'tablet',\n]\n\nconst formatPointer = (device: AudienceParamDevice) => {\n\treturn PRIMARY_TOUCH_DEVICES.includes(device) ? 'coarse' : 'fine'\n}\n\nconst formatOs = (osName: string = ''): AudienceParamOs => {\n\tif (!osName) return 'unix'\n\n\tconst os = osName.toLowerCase()\n\n\tconst osTypeMatch = AUDIENCE_PARAMS.os.find((osType) => {\n\t\treturn os.includes(osType)\n\t})\n\n\tif (osTypeMatch) return osTypeMatch\n\treturn 'unix'\n}\n\nexport const parseUserAgent = (userAgent: string): ParsedUserAgent | null => {\n\tif (!userAgent || typeof userAgent !== 'string') return null\n\n\tconst parser = new UAParser(userAgent)\n\tconst results = parser.getResult()\n\n\tconst device = formatDeviceType(results.device.type)\n\n\treturn {\n\t\tpointer: formatPointer(device),\n\t\tdevice: device,\n\t\tbrowser: formatBrowser(results.browser.name),\n\t\tos: formatOs(results.os.name),\n\t}\n}\n","import { ImproveAudienceValue } from '../types'\nimport { AudienceParamKey } from '../config/audiences'\nimport { ParsedUserAgent } from './parseUserAgent'\n\nexport const getVisitorMatchesAudience = (\n\taudience: ImproveAudienceValue | undefined,\n\tvisitorParams: ParsedUserAgent,\n) => {\n\tif (!audience) return true\n\treturn Object.entries(audience).every(([paramKey, paramValue]) => {\n\t\treturn visitorParams[paramKey as AudienceParamKey] === paramValue\n\t})\n}\n","import { ImproveTestOption } from '../types'\n\n// function cryptoRandom() {\n// \treturn crypto.getRandomValues(new Uint32Array(1))[0] / (0xffffffff + 1)\n// }\n\nexport const getRandomTestValue = (options: ImproveTestOption[]) => {\n\tif (options.length === 0) return null\n\tif (options.length === 1) return options[0].slug\n\n\t// Get a random number between 0 and 1\n\tconst sum = options.reduce((acc, { split }) => acc + split, 0)\n\tlet value = Math.random() * sum\n\n\tconst match =\n\t\toptions.find(({ split }) => {\n\t\t\tvalue -= split\n\t\t\treturn value <= 0\n\t\t}) || options[0]\n\n\treturn match.slug\n}\n","export const COOKIE_NAME_VISITOR = 'visitorId'\n\nexport const VISITOR_ID_PREFIX = 'visi'\n\nexport const VISITOR_ID_LENGTH = 26\n\nexport const VISITOR_ID_SEPARATOR = '_'\n\n/** Default number of retries for a failed config fetch (network / 5xx / 429). */\nexport const CONFIG_RETRY_COUNT = 2\n\n/** Base delay for config-fetch backoff, in ms. */\nexport const CONFIG_RETRY_BASE_DELAY_MS = 300\n\n/** Upper bound for a single config-fetch backoff wait, in ms. */\nexport const CONFIG_RETRY_MAX_DELAY_MS = 3000\n\n/**\n * Max length of a single analytic string field. The backend rejects any\n * field longer than this (varchar(256)) with a 400, so the client caps\n * developer-controlled values (event, message) to avoid silently dropping\n * the event.\n */\nexport const MAX_ANALYTIC_FIELD_LENGTH = 256\n\n/**\n * Max length of the analytic `currency` field. The backend column is\n * `varchar(8)` (ISO 4217 is 3 chars), and a longer value throws at insert —\n * failing the *entire* event, not just the currency — so the client caps it\n * here rather than at the generic 256-char limit.\n */\nexport const MAX_CURRENCY_FIELD_LENGTH = 8\n\n/**\n * Max size of the analytic POST body. The backend rejects anything larger with\n * a 413 (`MAX_BODY_BYTES = 8 * 1024`), so an oversized `params` payload silently\n * loses the event. Kept in sync with the backend limit.\n */\nexport const MAX_ANALYTIC_BODY_BYTES = 8 * 1024\n\n/**\n * How long to stop sending analytics after the backend returns 429 (usage\n * limit / rate limit) without a `Retry-After`. Avoids hammering an org that is\n * already over its quota.\n */\nexport const ANALYTIC_RATE_LIMIT_COOLDOWN_MS = 60_000\n","export const BASE_URL = 'https://improve.obelism.studio'\nexport const CONFIG_PATH = `/config`\nexport const ANALYTICS_PATH = `/api/log`\n","export const delay = (ms: number) =>\n\tnew Promise((resolve) => setTimeout(resolve, ms))\n","export type ImproveFetchErrorReason =\n\t| 'timeout'\n\t| 'network'\n\t| 'unauthorized'\n\t| 'rate-limited'\n\t| 'server'\n\t| 'client'\n\t| 'invalid-response'\n\nconst REASON_MESSAGES: Record<ImproveFetchErrorReason, string> = {\n\ttimeout: 'Configuration request timed out',\n\tnetwork: 'Configuration request could not reach the server',\n\tunauthorized:\n\t\t'Configuration request was rejected (403): verify the organizationId, the server token, and that the request origin is whitelisted for this environment',\n\t'rate-limited':\n\t\t'Configuration request was rate limited (429): too many requests, or the organization usage limit was reached',\n\tserver: 'Configuration request failed with a server error',\n\tclient: 'Configuration request was rejected by the server',\n\t'invalid-response': 'Configuration response could not be parsed as JSON',\n}\n\n/**\n * Error thrown by the config fetch layer. Carries the HTTP `status`, a coarse\n * `reason`, and (when the server sent one) the `Retry-After` delay in ms, so\n * callers can distinguish a misconfiguration (unauthorized) from a transient\n * failure (rate-limited/server/timeout) instead of guessing from a string.\n */\nexport class ImproveFetchError extends Error {\n\treason: ImproveFetchErrorReason\n\tstatus?: number\n\tretryAfterMs?: number\n\n\tconstructor(\n\t\treason: ImproveFetchErrorReason,\n\t\toptions?: { status?: number; retryAfterMs?: number; cause?: unknown },\n\t) {\n\t\tsuper(REASON_MESSAGES[reason])\n\t\tthis.name = 'ImproveFetchError'\n\t\tthis.reason = reason\n\t\tthis.status = options?.status\n\t\tthis.retryAfterMs = options?.retryAfterMs\n\t\tif (options && 'cause' in options) {\n\t\t\t;(this as { cause?: unknown }).cause = options.cause\n\t\t}\n\t}\n\n\t/** Whether retrying the request could plausibly succeed. */\n\tget isRetryable() {\n\t\treturn (\n\t\t\tthis.reason === 'rate-limited' ||\n\t\t\tthis.reason === 'server' ||\n\t\t\tthis.reason === 'timeout' ||\n\t\t\tthis.reason === 'network'\n\t\t)\n\t}\n}\n\n/** Map an HTTP status onto a coarse {@link ImproveFetchErrorReason}. */\nexport const getReasonFromStatus = (\n\tstatus: number,\n): ImproveFetchErrorReason => {\n\tif (status === 429) return 'rate-limited'\n\tif (status === 401 || status === 403) return 'unauthorized'\n\tif (status >= 500) return 'server'\n\treturn 'client'\n}\n","/**\n * @constant CHAR_SET\n * @description Key/value set of characters to be used for generation strings\n */\nconst CHAR_SET = 'ABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789'\n\n/**\n * @function getRandomString\n * @description Generate a random string\n */\nexport function getRandomString(characters: number = 5): string {\n\tif (!characters || typeof characters !== 'number') return ''\n\treturn Array(characters)\n\t\t.fill('')\n\t\t.reduce((acc) => {\n\t\t\tacc += CHAR_SET.charAt(Math.floor(Math.random() * CHAR_SET.length))\n\t\t\treturn acc\n\t\t}, '')\n}\n","import { ImproveConfiguration } from '../types'\nimport { ImproveFetchError } from './errors'\n\n/**\n * @function normalizeConfig\n * @description Coerce a parsed config response into a well-formed\n * {@link ImproveConfiguration}. The backend returns `200 {}` for an unknown\n * org/environment (and on its own internal error). An empty object is truthy,\n * so it slips past `if (!this.config)` guards and then crashes downstream on\n * `config.tests[slug]` (`Cannot read properties of undefined`). Guaranteeing\n * `tests`/`flags`/`audience` are always objects lets the SDK fail open to\n * default values instead of throwing during render/middleware.\n *\n * Throws {@link ImproveFetchError} `invalid-response` when the body isn't an\n * object at all (e.g. an array, string, or null).\n */\nexport const normalizeConfig = (body: unknown): ImproveConfiguration => {\n\tif (typeof body !== 'object' || body === null || Array.isArray(body)) {\n\t\tthrow new ImproveFetchError('invalid-response')\n\t}\n\n\tconst raw = body as Partial<ImproveConfiguration>\n\n\treturn {\n\t\tname: typeof raw.name === 'string' ? raw.name : '',\n\t\tversion: typeof raw.version === 'number' ? raw.version : 0,\n\t\ttests: raw.tests ?? {},\n\t\tflags: raw.flags ?? {},\n\t\taudience: raw.audience ?? {},\n\t}\n}\n","/**\n * @function parseRetryAfterMs\n * @description Parse an HTTP `Retry-After` header into milliseconds. Supports\n * both the delta-seconds form (`\"120\"`) and the HTTP-date form\n * (`\"Wed, 21 Oct 2026 07:28:00 GMT\"`). Returns `undefined` when the header is\n * absent or unparseable so the caller can fall back to computed backoff.\n *\n * @param {string | null} [header] - the raw `Retry-After` header value\n * @param {number} [now] - current epoch ms (injectable for testing)\n */\nexport const parseRetryAfterMs = (\n\theader: string | null | undefined,\n\tnow: number = Date.now(),\n): number | undefined => {\n\tif (!header) return undefined\n\n\tconst seconds = Number(header)\n\tif (Number.isFinite(seconds)) return Math.max(0, Math.round(seconds * 1000))\n\n\tconst date = Date.parse(header)\n\tif (Number.isNaN(date)) return undefined\n\n\treturn Math.max(0, date - now)\n}\n\n/**\n * @function getBackoffDelayMs\n * @description Exponential backoff with full jitter, capped at `maxDelay`.\n * Jitter spreads retries so concurrent clients don't stampede the origin in\n * lockstep after a shared failure.\n *\n * @param {number} attempt - zero-based retry attempt index\n * @param {number} baseDelay - base delay in ms\n * @param {number} maxDelay - upper bound in ms\n */\nexport const getBackoffDelayMs = (\n\tattempt: number,\n\tbaseDelay: number,\n\tmaxDelay: number,\n): number => {\n\tconst exponential = Math.min(maxDelay, baseDelay * 2 ** attempt)\n\treturn Math.round(Math.random() * exponential)\n}\n","import { delay } from './delay'\n\n/**\n * @async @function getTimeoutError\n * @description Throw an error after a delay\n *\n * @param {number} [timeout] - time in ms after to reject default: 1000\n * @param {AbortController} [controller] - (optional) AbortController to abort the request\n */\nexport const getTimeoutError = async (\n\ttimeout: number = 1000,\n\tcontroller: AbortController,\n): Promise<null> => {\n\tawait delay(timeout)\n\tcontroller?.abort()\n\treturn null\n}\n","import { getTimeoutError } from './getTimeoutError'\n\n/**\n * @async @function timeoutFetch\n * @description Fetch with a timeout\n *\n * @param {number} timeout - time in ms after to reject default: 3000\n * @param {string} url - url to fetch\n * @param {Object} [options] - (optional) options to pass to fetch\n */\nexport const timeoutFetch = (\n\ttimeout: number = 3000,\n\turl: string,\n\toptions?: object,\n) => {\n\tconst controller = new AbortController()\n\treturn Promise.race([\n\t\tfetch(url, {\n\t\t\t...options,\n\t\t\tsignal: controller.signal,\n\t\t}),\n\t\tgetTimeoutError(timeout, controller),\n\t])\n}\n","import {\n\tCONFIG_RETRY_BASE_DELAY_MS,\n\tCONFIG_RETRY_COUNT,\n\tCONFIG_RETRY_MAX_DELAY_MS,\n\tCOOKIE_NAME_VISITOR,\n\tVISITOR_ID_LENGTH,\n\tVISITOR_ID_PREFIX,\n\tVISITOR_ID_SEPARATOR,\n} from './config/constants'\nimport { CONFIG_PATH, BASE_URL } from './config/urls'\nimport {\n\tImproveConfiguration,\n\tImproveEnvironmentOption,\n\tImproveSetupArgs,\n\tImproveTestState,\n} from './types'\nimport { delay } from './utils/delay'\nimport { getReasonFromStatus, ImproveFetchError } from './utils/errors'\nimport { getRandomString } from './utils/getRandomString'\nimport { normalizeConfig } from './utils/normalizeConfig'\nimport { getBackoffDelayMs, parseRetryAfterMs } from './utils/retry'\nimport { timeoutFetch } from './utils/timeoutFetch'\n\ntype ConfigFetch = {\n\turl: string\n\ttimeout: number\n\tretries: number\n\trevalidate?: number\n}\n\n// Next.js augments RequestInit with a `next` field for its fetch cache.\ntype RequestInitWithNext = RequestInit & {\n\tnext?: { revalidate?: number; tags?: string[] }\n}\n\nexport class BaseImproveSDK {\n\torganizationId: string\n\tenvironment: ImproveEnvironmentOption = 'develop'\n\tstate: ImproveTestState\n\n\t#configFetch: ConfigFetch | null = null\n\n\t#warningsDisabled: boolean = false\n\n\tconfig: ImproveConfiguration | null = null\n\n\t_baseUrl: undefined | string\n\n\tconstructor({\n\t\torganizationId,\n\t\tenvironment,\n\t\tstate,\n\t\tconfig,\n\t\tfetchTimeout,\n\t\tbaseUrl,\n\t\tconfigRetries,\n\t\tconfigRevalidate,\n\t\tdisableWarnings,\n\t}: ImproveSetupArgs) {\n\t\tthis.organizationId = organizationId\n\t\tthis.environment = environment\n\t\tthis.state = state\n\t\tthis._baseUrl = baseUrl || BASE_URL\n\t\tthis.#warningsDisabled = disableWarnings ?? false\n\n\t\tif (config) {\n\t\t\tthis.config = config\n\t\t} else {\n\t\t\tthis.#configFetch = {\n\t\t\t\turl: [\n\t\t\t\t\t`${this._baseUrl}${CONFIG_PATH}`,\n\t\t\t\t\tthis.organizationId,\n\t\t\t\t\tthis.environment,\n\t\t\t\t\tthis.state || 'active',\n\t\t\t\t].join('/'),\n\t\t\t\ttimeout: fetchTimeout || 3000,\n\t\t\t\tretries: configRetries ?? CONFIG_RETRY_COUNT,\n\t\t\t\trevalidate: configRevalidate,\n\t\t\t}\n\t\t}\n\t}\n\n\t_fetchConfig = async (config?: RequestInit) => {\n\t\tif (this.config) return\n\n\t\tif (!this.#configFetch) throw new Error('No config fetch setup provided')\n\n\t\tconst { url, timeout, retries, revalidate } = this.#configFetch\n\n\t\t// Opt into the Next.js fetch cache when a revalidate window is configured,\n\t\t// so the datafile is reused across requests instead of re-fetched.\n\t\tconst requestInit: RequestInitWithNext = { ...config }\n\t\tif (typeof revalidate === 'number') {\n\t\t\trequestInit.next = {\n\t\t\t\t...(config as RequestInitWithNext | undefined)?.next,\n\t\t\t\trevalidate,\n\t\t\t}\n\t\t}\n\n\t\t// Seed with a generic network error so an exhausted loop always has a\n\t\t// meaningful, typed error to throw.\n\t\tlet lastError = new ImproveFetchError('network')\n\n\t\tfor (let attempt = 0; attempt <= retries; attempt++) {\n\t\t\tif (attempt > 0) {\n\t\t\t\t// Prefer a server-provided Retry-After; otherwise back off with jitter.\n\t\t\t\tawait delay(\n\t\t\t\t\tlastError.retryAfterMs ??\n\t\t\t\t\t\tgetBackoffDelayMs(\n\t\t\t\t\t\t\tattempt - 1,\n\t\t\t\t\t\t\tCONFIG_RETRY_BASE_DELAY_MS,\n\t\t\t\t\t\t\tCONFIG_RETRY_MAX_DELAY_MS,\n\t\t\t\t\t\t),\n\t\t\t\t)\n\t\t\t}\n\n\t\t\tlet res: Response | null\n\t\t\ttry {\n\t\t\t\tres = await timeoutFetch(timeout, url, requestInit)\n\t\t\t} catch (cause) {\n\t\t\t\t// timeoutFetch aborts on timeout (AbortError) and rejects on\n\t\t\t\t// network-level failures — both are transient and retryable.\n\t\t\t\tconst aborted = cause instanceof Error && cause.name === 'AbortError'\n\t\t\t\tlastError = new ImproveFetchError(aborted ? 'timeout' : 'network', {\n\t\t\t\t\tcause,\n\t\t\t\t})\n\t\t\t\tcontinue\n\t\t\t}\n\n\t\t\t// timeoutFetch resolves null when the timeout wins the race.\n\t\t\tif (!res) {\n\t\t\t\tlastError = new ImproveFetchError('timeout')\n\t\t\t\tcontinue\n\t\t\t}\n\n\t\t\tif (res.ok) {\n\t\t\t\ttry {\n\t\t\t\t\tthis.config = normalizeConfig(await res.json())\n\t\t\t\t} catch (cause) {\n\t\t\t\t\tlastError =\n\t\t\t\t\t\tcause instanceof ImproveFetchError\n\t\t\t\t\t\t\t? cause\n\t\t\t\t\t\t\t: new ImproveFetchError('invalid-response', { cause })\n\t\t\t\t\tcontinue\n\t\t\t\t}\n\t\t\t\treturn this.config\n\t\t\t}\n\n\t\t\tlastError = new ImproveFetchError(getReasonFromStatus(res.status), {\n\t\t\t\tstatus: res.status,\n\t\t\t\tretryAfterMs: parseRetryAfterMs(res.headers.get('Retry-After')),\n\t\t\t})\n\n\t\t\t// Auth/validation rejections won't change on retry — fail fast.\n\t\t\tif (!lastError.isRetryable) break\n\t\t}\n\n\t\tthrow lastError\n\t}\n\n\tloadConfig = (config: ImproveConfiguration) => {\n\t\tthis.config = config\n\t}\n\n\tgenerateVisitorId = () => {\n\t\treturn [\n\t\t\tVISITOR_ID_PREFIX,\n\t\t\tgetRandomString(VISITOR_ID_LENGTH).toUpperCase(),\n\t\t].join(VISITOR_ID_SEPARATOR)\n\t}\n\n\tgetVisitorCookieName = () => COOKIE_NAME_VISITOR\n\n\t/**\n\t * Emit a development warning, unless warnings were disabled via the\n\t * `disableWarnings` setup option. Centralized so every SDK warning honors the\n\t * same switch. No-op where `console` is unavailable.\n\t */\n\tprotected _warn = (message: string) => {\n\t\tif (this.#warningsDisabled) return\n\t\tif (typeof console === 'undefined') return\n\t\tconsole.warn(`[Improve] ${message}`)\n\t}\n\n\tvalidateTestValue = (testName: string, testValue: string) => {\n\t\tif (!this.config)\n\t\t\tthrow new Error(\n\t\t\t\t'Config is required before validating, either use `.fetchConfig()`, .loadConfig(config) or provide it during setup',\n\t\t\t)\n\n\t\tconst testConfig = this.config.tests[testName]\n\n\t\tif (!testConfig) throw new Error(`No config found for ${testName}`)\n\n\t\treturn Boolean(\n\t\t\ttestConfig.options.find((option) => option.slug === testValue),\n\t\t)\n\t}\n\n\tvalidateVisitorId = (possibleVisitorId: string) => {\n\t\tconst visitorIdParts = possibleVisitorId.split(VISITOR_ID_SEPARATOR)\n\t\tif (visitorIdParts.length !== 2) return false\n\t\tconst [key, value] = visitorIdParts as [string, string]\n\t\treturn key === VISITOR_ID_PREFIX && value.length === VISITOR_ID_LENGTH\n\t}\n}\n","const ROW_DELIMITER = '; '\nconst ENTRY_DELIMITER = '='\n\nexport const getCookie = (name: string) => {\n\tif (!name) return false\n\tconst cookie = document.cookie.split(ROW_DELIMITER).find((row) => {\n\t\tconst [key] = row.split(ENTRY_DELIMITER)\n\t\treturn name === key\n\t})\n\treturn cookie ? cookie.split(ENTRY_DELIMITER)[1] : false\n}\n\nexport const setCookie = (name: string, value: string) => {\n\tconst now = new Date()\n\tnow.setDate(now.getDate() + 30)\n\tdocument.cookie = `${name}=${value};path=/;expires=${now.toUTCString()};SameSite=Lax;Secure`\n}\n","/**\n * A well-formed `snake_case` event name: starts with a lowercase letter and\n * contains only lowercase letters, digits, and single underscores between\n * segments. Matches the GA4 / GTM event-name convention Improve recommends\n * (`add_to_cart`, `sign_up`, `purchase`).\n */\nconst SNAKE_CASE_EVENT_NAME = /^[a-z][a-z0-9]*(_[a-z0-9]+)*$/\n\n/**\n * Whether `event` follows the recommended `snake_case` naming convention.\n * Flags camelCase (`pageView`), PascalCase (`AddToCart`), kebab-case\n * (`add-to-cart`), and names with spaces or trailing/double underscores.\n */\nexport const isSnakeCaseEventName = (event: string): boolean =>\n\tSNAKE_CASE_EVENT_NAME.test(event)\n","/**\n * @function truncate\n * @description Cap a string to `maxLength` characters. Returns non-strings\n * unchanged so it is safe to apply to loosely-typed payload values.\n *\n * @param {string} value - the value to truncate\n * @param {number} maxLength - maximum length to keep\n */\nexport const truncate = (value: string, maxLength: number): string =>\n\ttypeof value === 'string' && value.length > maxLength\n\t\t? value.slice(0, maxLength)\n\t\t: value\n","import { ParsedUserAgent, parseUserAgent } from './utils/parseUserAgent'\nimport { getVisitorMatchesAudience } from './utils/getVisitorMatchesAudience'\nimport { getRandomTestValue } from './utils/getRandomTestValue'\nimport { BaseImproveSDK } from './base'\nimport { getCookie, setCookie } from './utils/clientCookie'\nimport {\n\tANALYTIC_RATE_LIMIT_COOLDOWN_MS,\n\tMAX_ANALYTIC_BODY_BYTES,\n\tMAX_ANALYTIC_FIELD_LENGTH,\n\tMAX_CURRENCY_FIELD_LENGTH,\n} from './config/constants'\nimport { ANALYTICS_PATH, BASE_URL } from './config/urls'\nimport { getScreenSize } from './utils/getScreenSize'\nimport { isSnakeCaseEventName } from './utils/isSnakeCaseEventName'\nimport { pushDataLayer, resetDataLayerEcommerce } from './utils/pushDataLayer'\nimport { parseRetryAfterMs } from './utils/retry'\nimport { truncate } from './utils/truncate'\nimport {\n\tImproveAnalyticPayload,\n\tImproveEnvironmentOption,\n\tImproveEventName,\n\tImproveSetupArgs,\n} from './types'\n\ntype Visitor = ParsedUserAgent & {\n\t[testSlug: string]: string\n}\n\n// Fields common to every beacon, regardless of type.\ntype BeaconCommon<T extends 'event' | 'exposure'> = {\n\ttype: T\n\torganizationId: string\n\tenvironment: ImproveEnvironmentOption\n\n\tvisitorId: string\n\n\tpointer: string\n\tdevice: string\n\tscreen: string\n\tbrowser: string\n\tos: string\n\tvisitor: string\n}\n\n/**\n * A real-world tracked event. Test-independent — the server attributes it to a\n * test/flag at read time by joining the visitor's exposures.\n */\nexport type CreateEvent = BeaconCommon<'event'> & {\n\tevent: string\n\tmessage: string\n\n\t/** GA4-aligned numeric value aggregated into revenue / AOV per variant. */\n\tvalue?: number\n\t/** ISO 4217 currency for `value`. */\n\tcurrency?: string\n\t/** Extra event params stored as JSON (e.g. a GA4 `ecommerce` object). */\n\tparams?: Record<string, unknown>\n}\n\n/**\n * A visitor's assignment to a test/flag variant, emitted the first time the SDK\n * resolves a non-holdout value for them. The per-variant exposure count is the\n * denominator for results.\n */\nexport type CreateExposure = BeaconCommon<'exposure'> & {\n\tsubjectKind: 'test' | 'flag'\n\tsubjectId: string\n\tvariant: string\n}\n\n// Events already sent this instance, so each fires at most once per page/session.\ntype TrackedEvents = {\n\t[event: string]: boolean\n}\n\n// Exposures already sent this instance, keyed by `${subjectKind}:${subjectId}`.\ntype TrackedExposures = {\n\t[subject: string]: boolean\n}\n\nexport class ImproveClientSDK extends BaseImproveSDK {\n\t#visitor: Visitor | undefined\n\t#visitorRecurring: boolean = false\n\n\t#visitorId: string = ''\n\n\t#analytics: TrackedEvents = {}\n\n\t#exposures: TrackedExposures = {}\n\n\t// Epoch ms until which analytics posting is suspended after a 429.\n\t#rateLimitedUntil: number = 0\n\n\t// Event names already warned about, so the snake_case nudge fires once per\n\t// offending name instead of on every call.\n\t#warnedEventNames = new Set<string>()\n\n\t#analyticsUrl = `${BASE_URL}${ANALYTICS_PATH}`\n\n\t#dataLayerEnabled: boolean = true\n\n\tfetchConfig = this._fetchConfig\n\n\tconstructor(args: ImproveSetupArgs) {\n\t\tsuper(args)\n\t\tthis.#analyticsUrl = `${this._baseUrl}${ANALYTICS_PATH}`\n\t\tthis.#dataLayerEnabled = args.dataLayer ?? true\n\t}\n\n\tsetupVisitor = (userAgent: string = window.navigator.userAgent) => {\n\t\tconst cookieVisitorId = getCookie(this.getVisitorCookieName())\n\t\tconst validCookieVisitorId =\n\t\t\tcookieVisitorId && this.validateVisitorId(cookieVisitorId)\n\n\t\tthis.#visitorRecurring = validCookieVisitorId\n\t\tthis.#visitorId = validCookieVisitorId\n\t\t\t? cookieVisitorId\n\t\t\t: this.generateVisitorId()\n\n\t\tconst parsedUserAgent = parseUserAgent(userAgent)\n\n\t\tif (!parsedUserAgent) return null\n\n\t\tthis.#visitor = parsedUserAgent\n\n\t\tsetCookie(this.getVisitorCookieName(), this.#visitorId)\n\n\t\treturn this.#visitorId\n\t}\n\n\tgetFlagValue = (flagSlug: string) => {\n\t\tif (!this.config) return null\n\n\t\tconst flagConfig = this.config.flags[flagSlug]\n\n\t\tif (!flagConfig || !flagConfig.options[0]) return null\n\n\t\tif (!this.#visitor) this.setupVisitor()\n\t\tif (!this.#visitorId || !this.#visitor) return flagConfig.options[0].slug\n\t\tif (this.#visitor?.[flagSlug]) return this.#visitor[flagSlug]\n\n\t\tconst visitorMatchesAudience = getVisitorMatchesAudience(\n\t\t\tthis.config.audience[flagConfig.audience],\n\t\t\tthis.#visitor,\n\t\t)\n\n\t\tif (!visitorMatchesAudience) return flagConfig.options[0].slug\n\n\t\tconst flagValue =\n\t\t\tgetCookie(flagSlug) || getRandomTestValue(flagConfig.options)\n\n\t\tif (!flagValue) return null\n\n\t\tthis.#visitor[flagSlug] = flagValue\n\n\t\tsetCookie(flagSlug, flagValue)\n\n\t\t// Record the assignment so the visitor's later events can be attributed to\n\t\t// this flag/variant server-side.\n\t\tthis.#postExposure('flag', flagConfig.id, flagValue)\n\n\t\treturn flagValue\n\t}\n\n\tgetTestValue = (testSlug: string) => {\n\t\tif (!this.config) return null\n\n\t\tconst testConfig = this.config.tests[testSlug]\n\n\t\tif (!testConfig) return null\n\n\t\tif (!this.#visitor) this.setupVisitor()\n\n\t\tif (!this.#visitorId || !this.#visitor) return testConfig.defaultValue\n\t\tif (this.#visitor?.[testSlug]) return this.#visitor[testSlug]\n\n\t\tconst visitorMatchesAudience = getVisitorMatchesAudience(\n\t\t\tthis.config.audience[testConfig.audience],\n\t\t\tthis.#visitor,\n\t\t)\n\n\t\tif (!visitorMatchesAudience) return testConfig.defaultValue\n\n\t\tif (\n\t\t\ttestConfig.allocation < 100 &&\n\t\t\tMath.random() * 100 > testConfig.allocation\n\t\t) {\n\t\t\tthis.#visitor[testSlug] = testConfig.defaultValue\n\t\t\treturn this.#visitor?.[testSlug]\n\t\t}\n\n\t\tconst cookieTestValue = getCookie(testSlug)\n\t\tconst validCookieTestValue =\n\t\t\tcookieTestValue && this.validateTestValue(testSlug, cookieTestValue)\n\t\tconst testValue = validCookieTestValue\n\t\t\t? cookieTestValue\n\t\t\t: getRandomTestValue(testConfig.options)\n\n\t\tif (!testValue) return null\n\n\t\tthis.#visitor[testSlug] = testValue\n\n\t\tsetCookie(testSlug, testValue)\n\n\t\t// Record the assignment so the visitor's later events can be attributed to\n\t\t// this test/variant server-side. Holdout (allocation) and audience-excluded\n\t\t// visitors return above without an exposure — they aren't in the test.\n\t\tthis.#postExposure('test', testConfig.id, testValue)\n\n\t\treturn testValue\n\t}\n\n\tsetAnalyticsUrls = (url: string) => {\n\t\tthis.#analyticsUrl = url\n\t}\n\n\t// Fields shared by every beacon. Callers must ensure the visitor is set up.\n\t#beaconCommon = <T extends 'event' | 'exposure'>(\n\t\ttype: T,\n\t): BeaconCommon<T> => ({\n\t\ttype,\n\t\torganizationId: this.organizationId,\n\t\tenvironment: this.environment,\n\t\tvisitorId: this.#visitorId,\n\t\tpointer: this.#visitor!.pointer,\n\t\tdevice: this.#visitor!.device,\n\t\tscreen: getScreenSize(),\n\t\tbrowser: this.#visitor!.browser,\n\t\tos: this.#visitor!.os,\n\t\tvisitor: this.#visitorRecurring ? 'recurring' : 'new',\n\t})\n\n\t// Sends a beacon with a `keepalive` fetch so it outlives a navigation/unload,\n\t// warns on oversized bodies, and backs off on a 429. Shared by events and\n\t// exposures.\n\t#send = (body: CreateEvent | CreateExposure, label: string) => {\n\t\t// Serialize once so we can both measure and send the same bytes.\n\t\tconst serializedBody = JSON.stringify(body)\n\n\t\t// The backend rejects bodies over 8KB with a 413 — realistically only an\n\t\t// oversized `params` (e.g. a large GA4 `ecommerce.items` array) gets there.\n\t\tif (\n\t\t\ttypeof TextEncoder !== 'undefined' &&\n\t\t\tnew TextEncoder().encode(serializedBody).length > MAX_ANALYTIC_BODY_BYTES\n\t\t) {\n\t\t\tthis._warn(\n\t\t\t\t`Beacon for \"${label}\" exceeds the ${MAX_ANALYTIC_BODY_BYTES}-byte ` +\n\t\t\t\t\t`limit and will be rejected by the server — trim the \\`params\\` payload.`,\n\t\t\t)\n\t\t}\n\n\t\tconst request = fetch(this.#analyticsUrl, {\n\t\t\tmethod: 'POST',\n\t\t\theaders: { 'Content-Type': 'application/json' },\n\t\t\tbody: serializedBody,\n\t\t\tkeepalive: true,\n\t\t})\n\n\t\t// Back off when the org hits its usage/rate limit. Honor a server\n\t\t// Retry-After when present, otherwise use a fixed cooldown.\n\t\trequest\n\t\t\t.then((res) => {\n\t\t\t\tif (res?.status === 429) {\n\t\t\t\t\tconst retryAfterMs = parseRetryAfterMs(res.headers.get('Retry-After'))\n\t\t\t\t\tthis.#rateLimitedUntil =\n\t\t\t\t\t\tDate.now() + (retryAfterMs ?? ANALYTIC_RATE_LIMIT_COOLDOWN_MS)\n\t\t\t\t}\n\t\t\t})\n\t\t\t.catch(() => {})\n\n\t\treturn request\n\t}\n\n\t// Emits an exposure beacon the first time a visitor is assigned to a subject\n\t// this instance. Idempotent server-side (unique per visitor+subject).\n\t#postExposure = (\n\t\tsubjectKind: 'test' | 'flag',\n\t\tsubjectId: string,\n\t\tvariant: string,\n\t) => {\n\t\tif (!this.#visitor || !this.#visitorId) return\n\n\t\t// Suspended after a 429: skip, but don't mark tracked so it can retry later.\n\t\tif (Date.now() < this.#rateLimitedUntil) return\n\n\t\tconst key = `${subjectKind}:${subjectId}`\n\t\tif (this.#exposures[key]) return\n\t\tthis.#exposures[key] = true\n\n\t\tconst body: CreateExposure = {\n\t\t\t...this.#beaconCommon('exposure'),\n\t\t\tsubjectKind,\n\t\t\tsubjectId,\n\t\t\tvariant,\n\t\t}\n\n\t\treturn this.#send(body, key)\n\t}\n\n\tpostAnalytic = (\n\t\tevent: ImproveEventName,\n\t\t// Accepts a structured payload, or a plain string as shorthand for the\n\t\t// `message`.\n\t\tpayload?: ImproveAnalyticPayload | string,\n\t) => {\n\t\tif (!this.config) return null\n\n\t\t// GTM reserves the `gtm.*` namespace for its own events; never emit into\n\t\t// it, or the mirrored dataLayer entry would collide with GTM internals.\n\t\tif (event.startsWith('gtm.')) return null\n\n\t\t// Nudge developers toward the snake_case / GA4 naming convention. Warn\n\t\t// once per offending name so it's noticeable without spamming the console;\n\t\t// silence entirely with the `disableWarnings` setup option.\n\t\tif (!isSnakeCaseEventName(event) && !this.#warnedEventNames.has(event)) {\n\t\t\tthis.#warnedEventNames.add(event)\n\t\t\tthis._warn(\n\t\t\t\t`Event name \"${event}\" isn't snake_case. Prefer GA4-style names ` +\n\t\t\t\t\t`like \"add_to_cart\" or \"purchase\" so your events line up with ` +\n\t\t\t\t\t`GA4 / Google Tag Manager. Pass \\`disableWarnings: true\\` at setup ` +\n\t\t\t\t\t`to silence this.`,\n\t\t\t)\n\t\t}\n\n\t\t// Suspended after a 429: skip sending, but don't mark the event as\n\t\t// tracked, so it can still fire once the cooldown elapses.\n\t\tif (Date.now() < this.#rateLimitedUntil) return null\n\n\t\tconst { value, currency, message, params }: ImproveAnalyticPayload =\n\t\t\ttypeof payload === 'string' ? { message: payload } : (payload ?? {})\n\t\tconst hasValue = typeof value === 'number' && Number.isFinite(value)\n\t\t// The backend requires `params` to be a plain (non-array) object and 400s\n\t\t// the whole event otherwise — drop an invalid value rather than lose the\n\t\t// event.\n\t\tconst validParams =\n\t\t\tparams && typeof params === 'object' && !Array.isArray(params)\n\t\t\t\t? params\n\t\t\t\t: undefined\n\n\t\tif (!this.#visitor) this.setupVisitor()\n\n\t\t// The event is test-independent: it's recorded once per visitor and later\n\t\t// attributed to whichever tests/flags the visitor was exposed to. Dedup per\n\t\t// event name so it fires at most once per page/session.\n\t\tif (!this.#visitor || this.#analytics[event]) return null\n\n\t\tthis.#analytics[event] = true\n\n\t\tconst body: CreateEvent = {\n\t\t\t...this.#beaconCommon('event'),\n\t\t\t// Cap developer-controlled fields to the backend's varchar(256) limit;\n\t\t\t// an over-length value is otherwise rejected with a 400 and the event\n\t\t\t// is lost.\n\t\t\tevent: truncate(event, MAX_ANALYTIC_FIELD_LENGTH),\n\t\t\tmessage: truncate(message || '', MAX_ANALYTIC_FIELD_LENGTH),\n\t\t\t...(hasValue ? { value } : {}),\n\t\t\t...(currency\n\t\t\t\t? { currency: truncate(currency, MAX_CURRENCY_FIELD_LENGTH) }\n\t\t\t\t: {}),\n\t\t\t...(validParams ? { params: validParams } : {}),\n\t\t}\n\n\t\tif (this.#dataLayerEnabled) {\n\t\t\t// Clear any prior ecommerce object first so its keys don't bleed into\n\t\t\t// this event (GTM/GA4 best practice).\n\t\t\tif (validParams && 'ecommerce' in validParams) resetDataLayerEcommerce()\n\n\t\t\tpushDataLayer({\n\t\t\t\tevent,\n\t\t\t\timprove: {\n\t\t\t\t\tvisitorId: this.#visitorId,\n\t\t\t\t},\n\t\t\t\t...(hasValue ? { value } : {}),\n\t\t\t\t...(currency ? { currency } : {}),\n\t\t\t\t// Spread custom params (incl. a GA4 `ecommerce` object) to the top\n\t\t\t\t// level so GA4/GTM tags can read them directly.\n\t\t\t\t...(validParams ?? {}),\n\t\t\t\t_improve: true,\n\t\t\t})\n\t\t}\n\n\t\treturn this.#send(body, event)\n\t}\n}\n","type SizeOptions = 'small' | 'medium' | 'large' | 'huge'\n\nexport const getScreenSize = (): SizeOptions => {\n\tconst size = window.innerWidth\n\tif (size <= 768) return 'small'\n\tif (size <= 1024) return 'medium'\n\tif (size <= 1200) return 'large'\n\treturn 'huge'\n}\n","import { ImproveEventName } from '../types'\n\nexport type ImproveDataLayerEntry = {\n\tevent: ImproveEventName\n\timprove: {\n\t\t// Events are test-independent; attribution to a test/variant happens\n\t\t// server-side by joining the visitor's exposures, so only the visitor is\n\t\t// carried here.\n\t\tvisitorId: string\n\t}\n\t/** GA4 numeric value / currency, when provided on the analytic. */\n\tvalue?: number\n\tcurrency?: string\n\t/**\n\t * Marks the entry as originating from Improve so platform-side dataLayer\n\t * importers can ignore it (loop prevention).\n\t */\n\t_improve: true\n\t/** Any extra event params (e.g. a GA4 `ecommerce` object) spread onto the entry. */\n\t[param: string]: unknown\n}\n\ndeclare global {\n\tinterface Window {\n\t\tdataLayer?: Record<string, unknown>[]\n\t}\n}\n\n/**\n * Mirror an analytic onto the GTM dataLayer (with experiment dimensions) so it\n * can drive Google Tag Manager / Google Ads conversions. No-op outside the\n * browser. Initializes window.dataLayer if GTM hasn't yet.\n */\nexport const pushDataLayer = (entry: ImproveDataLayerEntry) => {\n\tif (typeof window === 'undefined') return\n\n\twindow.dataLayer = window.dataLayer || []\n\twindow.dataLayer.push(entry)\n}\n\n/**\n * Clear the dataLayer `ecommerce` object before pushing an ecommerce event, so\n * leftover keys (old `items`, a stale `coupon`…) from a previous event don't\n * bleed into the next one — the GTM/GA4 recommended reset. No-op outside the\n * browser.\n */\nexport const resetDataLayerEcommerce = () => {\n\tif (typeof window === 'undefined') return\n\n\twindow.dataLayer = window.dataLayer || []\n\twindow.dataLayer.push({ ecommerce: null })\n}\n"],"names":["SOCIAL_BROWSERS","PRIMARY_TOUCH_DEVICES","getVisitorMatchesAudience","audience","visitorParams","Object","entries","every","paramKey","paramValue","getRandomTestValue","options","length","slug","value","Math","random","reduce","acc","split","match","find","VISITOR_ID_PREFIX","BASE_URL","ANALYTICS_PATH","delay","ms","Promise","resolve","setTimeout","REASON_MESSAGES","timeout","network","unauthorized","server","client","ImproveFetchError","Error","reason","name","status","retryAfterMs","cause","isRetryable","getReasonFromStatus","CHAR_SET","normalizeConfig","body","Array","isArray","raw","version","tests","flags","parseRetryAfterMs","header","now","Date","seconds","Number","isFinite","max","round","date","parse","isNaN","getBackoffDelayMs","attempt","baseDelay","maxDelay","min","getTimeoutError","controller","abort","timeoutFetch","url","AbortController","race","fetch","signal","BaseImproveSDK","organizationId","environment","state","config","fetchTimeout","baseUrl","configRetries","configRevalidate","disableWarnings","_fetchConfig","retries","revalidate","requestInit","next","lastError","res","aborted","ok","json","headers","get","loadConfig","generateVisitorId","getRandomString","characters","fill","charAt","floor","toUpperCase","join","getVisitorCookieName","_warn","message","console","warn","validateTestValue","testName","testValue","testConfig","Boolean","option","validateVisitorId","possibleVisitorId","visitorIdParts","key","_baseUrl","getCookie","cookie","document","row","setCookie","setDate","getDate","toUTCString","SNAKE_CASE_EVENT_NAME","truncate","maxLength","slice","args","Set","fetchConfig","setupVisitor","userAgent","window","navigator","cookieVisitorId","validCookieVisitorId","parsedUserAgent","parseUserAgent","deviceType","results","parser","UAParser","getResult","device","type","AUDIENCE_PARAMS","includes","pointer","browser","formatBrowser","browserName","toLowerCase","browserTypeMatch","browserType","os","formatOs","osName","osTypeMatch","osType","getFlagValue","flagSlug","flagConfig","flagValue","id","getTestValue","testSlug","defaultValue","allocation","cookieTestValue","validCookieTestValue","setAnalyticsUrls","size","visitorId","screen","innerWidth","visitor","label","serializedBody","JSON","stringify","TextEncoder","encode","request","method","keepalive","then","ANALYTIC_RATE_LIMIT_COOLDOWN_MS","catch","subjectKind","subjectId","variant","postAnalytic","event","payload","startsWith","test","has","add","currency","params","hasValue","validParams","undefined","entry","dataLayer","push","ecommerce","improve","_improve"],"mappings":"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"}
|
package/dist/client.d.ts
CHANGED
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
import * as __types from './types.js';
|
|
2
|
-
import { ImproveEnvironmentOption, ImproveTestState, ImproveConfiguration, ImproveSetupArgs } from './types.js';
|
|
2
|
+
import { ImproveEnvironmentOption, ImproveTestState, ImproveConfiguration, ImproveSetupArgs, ImproveEventName, ImproveAnalyticPayload } from './types.js';
|
|
3
3
|
|
|
4
4
|
declare class BaseImproveSDK {
|
|
5
5
|
#private;
|
|
@@ -8,20 +8,25 @@ declare class BaseImproveSDK {
|
|
|
8
8
|
state: ImproveTestState;
|
|
9
9
|
config: ImproveConfiguration | null;
|
|
10
10
|
_baseUrl: undefined | string;
|
|
11
|
-
constructor({ organizationId, environment, state, config, fetchTimeout, baseUrl, configRetries, configRevalidate, }: ImproveSetupArgs);
|
|
11
|
+
constructor({ organizationId, environment, state, config, fetchTimeout, baseUrl, configRetries, configRevalidate, disableWarnings, }: ImproveSetupArgs);
|
|
12
12
|
_fetchConfig: (config?: RequestInit) => Promise<ImproveConfiguration>;
|
|
13
13
|
loadConfig: (config: ImproveConfiguration) => void;
|
|
14
14
|
generateVisitorId: () => string;
|
|
15
15
|
getVisitorCookieName: () => string;
|
|
16
|
+
/**
|
|
17
|
+
* Emit a development warning, unless warnings were disabled via the
|
|
18
|
+
* `disableWarnings` setup option. Centralized so every SDK warning honors the
|
|
19
|
+
* same switch. No-op where `console` is unavailable.
|
|
20
|
+
*/
|
|
21
|
+
protected _warn: (message: string) => void;
|
|
16
22
|
validateTestValue: (testName: string, testValue: string) => boolean;
|
|
17
23
|
validateVisitorId: (possibleVisitorId: string) => boolean;
|
|
18
24
|
}
|
|
19
25
|
|
|
20
|
-
type
|
|
26
|
+
type BeaconCommon<T extends 'event' | 'exposure'> = {
|
|
27
|
+
type: T;
|
|
21
28
|
organizationId: string;
|
|
22
29
|
environment: ImproveEnvironmentOption;
|
|
23
|
-
testId: string;
|
|
24
|
-
testValue: string;
|
|
25
30
|
visitorId: string;
|
|
26
31
|
pointer: string;
|
|
27
32
|
device: string;
|
|
@@ -29,8 +34,30 @@ type CreateAnalytic = {
|
|
|
29
34
|
browser: string;
|
|
30
35
|
os: string;
|
|
31
36
|
visitor: string;
|
|
37
|
+
};
|
|
38
|
+
/**
|
|
39
|
+
* A real-world tracked event. Test-independent — the server attributes it to a
|
|
40
|
+
* test/flag at read time by joining the visitor's exposures.
|
|
41
|
+
*/
|
|
42
|
+
type CreateEvent = BeaconCommon<'event'> & {
|
|
32
43
|
event: string;
|
|
33
44
|
message: string;
|
|
45
|
+
/** GA4-aligned numeric value aggregated into revenue / AOV per variant. */
|
|
46
|
+
value?: number;
|
|
47
|
+
/** ISO 4217 currency for `value`. */
|
|
48
|
+
currency?: string;
|
|
49
|
+
/** Extra event params stored as JSON (e.g. a GA4 `ecommerce` object). */
|
|
50
|
+
params?: Record<string, unknown>;
|
|
51
|
+
};
|
|
52
|
+
/**
|
|
53
|
+
* A visitor's assignment to a test/flag variant, emitted the first time the SDK
|
|
54
|
+
* resolves a non-holdout value for them. The per-variant exposure count is the
|
|
55
|
+
* denominator for results.
|
|
56
|
+
*/
|
|
57
|
+
type CreateExposure = BeaconCommon<'exposure'> & {
|
|
58
|
+
subjectKind: 'test' | 'flag';
|
|
59
|
+
subjectId: string;
|
|
60
|
+
variant: string;
|
|
34
61
|
};
|
|
35
62
|
declare class ImproveClientSDK extends BaseImproveSDK {
|
|
36
63
|
#private;
|
|
@@ -40,9 +67,9 @@ declare class ImproveClientSDK extends BaseImproveSDK {
|
|
|
40
67
|
getFlagValue: (flagSlug: string) => string;
|
|
41
68
|
getTestValue: (testSlug: string) => string;
|
|
42
69
|
setAnalyticsUrls: (url: string) => void;
|
|
43
|
-
postAnalytic: (
|
|
70
|
+
postAnalytic: (event: ImproveEventName, payload?: ImproveAnalyticPayload | string) => Promise<Response>;
|
|
44
71
|
}
|
|
45
72
|
|
|
46
73
|
export { ImproveClientSDK };
|
|
47
|
-
export type {
|
|
74
|
+
export type { CreateEvent, CreateExposure };
|
|
48
75
|
//# sourceMappingURL=client.d.ts.map
|
package/dist/client.d.ts.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"client.d.ts","sources":["../src/base.ts","../src/client.ts"],"mappings":";;;AACA,cAAc,cAAc;AAC5B;AACA;AACA,iBAAiB,wBAAwB;AACzC,WAAW,gBAAgB;AAC3B,YAAY,oBAAoB;AAChC;AACA,
|
|
1
|
+
{"version":3,"file":"client.d.ts","sources":["../src/base.ts","../src/client.ts"],"mappings":";;;AACA,cAAc,cAAc;AAC5B;AACA;AACA,iBAAiB,wBAAwB;AACzC,WAAW,gBAAgB;AAC3B,YAAY,oBAAoB;AAChC;AACA,0IAA0I,gBAAgB;AAC1J,4BAA4B,WAAW,KAAK,OAAO,CAAC,oBAAoB;AACxE,yBAAyB,oBAAoB;AAC7C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClBA,KAAK,YAAY;AACjB;AACA;AACA,iBAAiB,wBAAwB;AACzC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,KAAK,WAAW,GAAG,YAAY;AAC/B;AACA;AACA;AACA;AACA;AACA;AACA;AACA,aAAa,MAAM;AACnB;AACA;AACA;AACA;AACA;AACA;AACA,KAAK,cAAc,GAAG,YAAY;AAClC;AACA;AACA;AACA;AACA,cAAc,gBAAgB,SAAS,cAAc;AACrD;AACA,2BAA2B,WAAW,KAAK,OAAO,CAAC,OAAO,CAAC,oBAAoB;AAC/E,sBAAsB,gBAAgB;AACtC;AACA;AACA;AACA;AACA,0BAA0B,gBAAgB,YAAY,sBAAsB,cAAc,OAAO,CAAC,QAAQ;AAC1G;;;;","names":[]}
|
package/dist/client.mjs
CHANGED
|
@@ -1,2 +1,2 @@
|
|
|
1
|
-
import{UAParser as t}from"ua-parser-js";let e=["wearable","mobile","tablet","console","smarttv","embedded","desktop"],i=["chrome","safari","firefox","edge","ie","samsung internet","social","other"],r=["mac os","ios","android","windows","unix"],s=["tiktok","wechat","weibo","snapchat","klarna","Line","instagram","facebook","alipay","Baidu"],n=["wearable","mobile","tablet"],o=(t,e)=>!t||Object.entries(t).every(([t,i])=>e[t]===i),a=t=>{if(0===t.length)return null;if(1===t.length)return t[0].slug;let e=Math.random()*t.reduce((t,{split:e})=>t+e,0);return(t.find(({split:t})=>(e-=t)<=0)||t[0]).slug},l="visi",h="https://improve.obelism.studio",u="/api/log",
|
|
1
|
+
import{UAParser as t}from"ua-parser-js";let e=["wearable","mobile","tablet","console","smarttv","embedded","desktop"],i=["chrome","safari","firefox","edge","ie","samsung internet","social","other"],r=["mac os","ios","android","windows","unix"],s=["tiktok","wechat","weibo","snapchat","klarna","Line","instagram","facebook","alipay","Baidu"],n=["wearable","mobile","tablet"],o=(t,e)=>!t||Object.entries(t).every(([t,i])=>e[t]===i),a=t=>{if(0===t.length)return null;if(1===t.length)return t[0].slug;let e=Math.random()*t.reduce((t,{split:e})=>t+e,0);return(t.find(({split:t})=>(e-=t)<=0)||t[0]).slug},l="visi",h="https://improve.obelism.studio",u="/api/log",d=t=>new Promise(e=>setTimeout(e,t)),c={timeout:"Configuration request timed out",network:"Configuration request could not reach the server",unauthorized:"Configuration request was rejected (403): verify the organizationId, the server token, and that the request origin is whitelisted for this environment","rate-limited":"Configuration request was rate limited (429): too many requests, or the organization usage limit was reached",server:"Configuration request failed with a server error",client:"Configuration request was rejected by the server","invalid-response":"Configuration response could not be parsed as JSON"};class f extends Error{constructor(t,e){super(c[t]),this.name="ImproveFetchError",this.reason=t,this.status=e?.status,this.retryAfterMs=e?.retryAfterMs,e&&"cause"in e&&(this.cause=e.cause)}get isRetryable(){return"rate-limited"===this.reason||"server"===this.reason||"timeout"===this.reason||"network"===this.reason}}let g=t=>429===t?"rate-limited":401===t||403===t?"unauthorized":t>=500?"server":"client",m="ABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789",v=t=>{if("object"!=typeof t||null===t||Array.isArray(t))throw new f("invalid-response");return{name:"string"==typeof t.name?t.name:"",version:"number"==typeof t.version?t.version:0,tests:t.tests??{},flags:t.flags??{},audience:t.audience??{}}},p=(t,e=Date.now())=>{if(!t)return;let i=Number(t);if(Number.isFinite(i))return Math.max(0,Math.round(1e3*i));let r=Date.parse(t);if(!Number.isNaN(r))return Math.max(0,r-e)},w=(t,e,i)=>Math.round(Math.random()*Math.min(i,e*2**t)),y=async(t=1e3,e)=>(await d(t),e?.abort(),null),b=(t=3e3,e,i)=>{let r=new AbortController;return Promise.race([fetch(e,{...i,signal:r.signal}),y(t,r)])};class C{#t;#e;constructor({organizationId:t,environment:e,state:i,config:r,fetchTimeout:s,baseUrl:n,configRetries:o,configRevalidate:a,disableWarnings:u}){this.environment="develop",this.#t=null,this.#e=!1,this.config=null,this._fetchConfig=async t=>{if(this.config)return;if(!this.#t)throw Error("No config fetch setup provided");let{url:e,timeout:i,retries:r,revalidate:s}=this.#t,n={...t};"number"==typeof s&&(n.next={...t?.next,revalidate:s});let o=new f("network");for(let t=0;t<=r;t++){let r;t>0&&await d(o.retryAfterMs??w(t-1,300,3e3));try{r=await b(i,e,n)}catch(t){o=new f(t instanceof Error&&"AbortError"===t.name?"timeout":"network",{cause:t});continue}if(!r){o=new f("timeout");continue}if(r.ok){try{this.config=v(await r.json())}catch(t){o=t instanceof f?t:new f("invalid-response",{cause:t});continue}return this.config}if(!(o=new f(g(r.status),{status:r.status,retryAfterMs:p(r.headers.get("Retry-After"))})).isRetryable)break}throw o},this.loadConfig=t=>{this.config=t},this.generateVisitorId=()=>[l,(function(t=5){return t&&"number"==typeof t?Array(t).fill("").reduce(t=>t+=m.charAt(Math.floor(Math.random()*m.length)),""):""})(26).toUpperCase()].join("_"),this.getVisitorCookieName=()=>"visitorId",this._warn=t=>{this.#e||"u">typeof console&&console.warn(`[Improve] ${t}`)},this.validateTestValue=(t,e)=>{if(!this.config)throw Error("Config is required before validating, either use `.fetchConfig()`, .loadConfig(config) or provide it during setup");let i=this.config.tests[t];if(!i)throw Error(`No config found for ${t}`);return!!i.options.find(t=>t.slug===e)},this.validateVisitorId=t=>{let e=t.split("_");if(2!==e.length)return!1;let[i,r]=e;return i===l&&26===r.length},this.organizationId=t,this.environment=e,this.state=i,this._baseUrl=n||h,this.#e=u??!1,r?this.config=r:this.#t={url:[`${this._baseUrl}/config`,this.organizationId,this.environment,this.state||"active"].join("/"),timeout:s||3e3,retries:o??2,revalidate:a}}}let I=t=>{if(!t)return!1;let e=document.cookie.split("; ").find(e=>{let[i]=e.split("=");return t===i});return!!e&&e.split("=")[1]},x=(t,e)=>{let i=new Date;i.setDate(i.getDate()+30),document.cookie=`${t}=${e};path=/;expires=${i.toUTCString()};SameSite=Lax;Secure`},E=/^[a-z][a-z0-9]*(_[a-z0-9]+)*$/,k=(t,e)=>"string"==typeof t&&t.length>e?t.slice(0,e):t;class L extends C{#i;#r;#s;#n;#o;#a;#l;#h;#u;constructor(l){super(l),this.#r=!1,this.#s="",this.#n={},this.#o={},this.#a=0,this.#l=new Set,this.#h=`${h}${u}`,this.#u=!0,this.fetchConfig=this._fetchConfig,this.setupVisitor=(o=window.navigator.userAgent)=>{let a=I(this.getVisitorCookieName()),l=a&&this.validateVisitorId(a);this.#r=l,this.#s=l?a:this.generateVisitorId();let h=(o=>{var a;if(!o||"string"!=typeof o)return null;let l=new t(o).getResult(),h=(a=l.device.type)&&e.includes(a)?a:"desktop";return{pointer:n.includes(h)?"coarse":"fine",device:h,browser:((t="")=>{if(!t)return"other";let e=t.toLowerCase(),r=i.find(t=>e.includes(t));return r||(s.includes(e)?"social":"other")})(l.browser.name),os:((t="")=>{if(!t)return"unix";let e=t.toLowerCase(),i=r.find(t=>e.includes(t));return i||"unix"})(l.os.name)}})(o);return h?(this.#i=h,x(this.getVisitorCookieName(),this.#s),this.#s):null},this.getFlagValue=t=>{if(!this.config)return null;let e=this.config.flags[t];if(!e||!e.options[0])return null;if(this.#i||this.setupVisitor(),!this.#s||!this.#i)return e.options[0].slug;if(this.#i?.[t])return this.#i[t];if(!o(this.config.audience[e.audience],this.#i))return e.options[0].slug;let i=I(t)||a(e.options);return i?(this.#i[t]=i,x(t,i),this.#d("flag",e.id,i),i):null},this.getTestValue=t=>{if(!this.config)return null;let e=this.config.tests[t];if(!e)return null;if(this.#i||this.setupVisitor(),!this.#s||!this.#i)return e.defaultValue;if(this.#i?.[t])return this.#i[t];if(!o(this.config.audience[e.audience],this.#i))return e.defaultValue;if(e.allocation<100&&100*Math.random()>e.allocation)return this.#i[t]=e.defaultValue,this.#i?.[t];let i=I(t),r=i&&this.validateTestValue(t,i)?i:a(e.options);return r?(this.#i[t]=r,x(t,r),this.#d("test",e.id,r),r):null},this.setAnalyticsUrls=t=>{this.#h=t},this.#c=t=>{let e;return{type:t,organizationId:this.organizationId,environment:this.environment,visitorId:this.#s,pointer:this.#i.pointer,device:this.#i.device,screen:(e=window.innerWidth)<=768?"small":e<=1024?"medium":e<=1200?"large":"huge",browser:this.#i.browser,os:this.#i.os,visitor:this.#r?"recurring":"new"}},this.#f=(t,e)=>{let i=JSON.stringify(t);"u">typeof TextEncoder&&new TextEncoder().encode(i).length>8192&&this._warn(`Beacon for "${e}" exceeds the 8192-byte limit and will be rejected by the server — trim the \`params\` payload.`);let r=fetch(this.#h,{method:"POST",headers:{"Content-Type":"application/json"},body:i,keepalive:!0});return r.then(t=>{if(t?.status===429){let e=p(t.headers.get("Retry-After"));this.#a=Date.now()+(e??6e4)}}).catch(()=>{}),r},this.#d=(t,e,i)=>{if(!this.#i||!this.#s||Date.now()<this.#a)return;let r=`${t}:${e}`;if(this.#o[r])return;this.#o[r]=!0;let s={...this.#c("exposure"),subjectKind:t,subjectId:e,variant:i};return this.#f(s,r)},this.postAnalytic=(t,e)=>{if(!this.config||t.startsWith("gtm."))return null;if(E.test(t)||this.#l.has(t)||(this.#l.add(t),this._warn(`Event name "${t}" isn't snake_case. Prefer GA4-style names like "add_to_cart" or "purchase" so your events line up with GA4 / Google Tag Manager. Pass \`disableWarnings: true\` at setup to silence this.`)),Date.now()<this.#a)return null;let{value:i,currency:r,message:s,params:n}="string"==typeof e?{message:e}:e??{},o="number"==typeof i&&Number.isFinite(i),a=n&&"object"==typeof n&&!Array.isArray(n)?n:void 0;if(this.#i||this.setupVisitor(),!this.#i||this.#n[t])return null;this.#n[t]=!0;let l={...this.#c("event"),event:k(t,256),message:k(s||"",256),...o?{value:i}:{},...r?{currency:k(r,8)}:{},...a?{params:a}:{}};if(this.#u){var h;a&&"ecommerce"in a&&"u">typeof window&&(window.dataLayer=window.dataLayer||[],window.dataLayer.push({ecommerce:null})),h={event:t,improve:{visitorId:this.#s},...o?{value:i}:{},...r?{currency:r}:{},...a??{},_improve:!0},"u">typeof window&&(window.dataLayer=window.dataLayer||[],window.dataLayer.push(h))}return this.#f(l,t)},this.#h=`${this._baseUrl}${u}`,this.#u=l.dataLayer??!0}#c;#f;#d}export{L as ImproveClientSDK};
|
|
2
2
|
//# sourceMappingURL=client.mjs.map
|