@obelism/improve-sdk 1.0.0 → 1.1.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 +1 -1
- 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 +1 -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 +19 -0
- 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
|
|
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",u="https://improve.obelism.studio",h="/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;constructor({organizationId:t,environment:e,state:i,config:r,fetchTimeout:s,baseUrl:n,configRetries:o,configRevalidate:a}){this.environment="develop",this.#t=null,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.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||u,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]},V=(t,e)=>{let i=new Date;i.setDate(i.getDate()+30),document.cookie=`${t}=${e};path=/;expires=${i.toUTCString()};SameSite=Lax;Secure`},k=(t,e)=>"string"==typeof t&&t.length>e?t.slice(0,e):t;exports.ImproveClientSDK=class extends C{#e;#i;#r;#s;#n;#o;#a;constructor(l){super(l),this.#i=!1,this.#r="",this.#s={},this.#n=0,this.#o=`${u}${h}`,this.#a=!0,this.fetchConfig=this._fetchConfig,this.setupVisitor=(o=window.navigator.userAgent)=>{let a=I(this.getVisitorCookieName()),l=a&&this.validateVisitorId(a);this.#i=l,this.#r=l?a:this.generateVisitorId();let u=(o=>{var a;if(!o||"string"!=typeof o)return null;let l=new t.UAParser(o).getResult(),u=(a=l.device.type)&&e.includes(a)?a:"desktop";return{pointer:n.includes(u)?"coarse":"fine",device:u,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 u?(this.#e=u,V(this.getVisitorCookieName(),this.#r),this.#r):null},this.getFlagValue=t=>{if(!this.config)return null;let e=this.config.flags[t];if(!e||!e.options[0])return null;if(this.#e||this.setupVisitor(),!this.#r||!this.#e)return e.options[0].slug;if(this.#e?.[t])return this.#e[t];if(!o(this.config.audience[e.audience],this.#e))return e.options[0].slug;let i=I(t)||a(e.options);return i?(this.#e[t]=i,V(t,i),i):null},this.getTestValue=t=>{if(!this.config)return null;let e=this.config.tests[t];if(!e)return null;if(this.#e||this.setupVisitor(),!this.#r||!this.#e)return e.defaultValue;if(this.#e?.[t])return this.#e[t];if(!o(this.config.audience[e.audience],this.#e))return e.defaultValue;if(e.allocation<100&&100*Math.random()>e.allocation)return this.#e[t]=e.defaultValue,this.#e?.[t];let i=I(t),r=i&&this.validateTestValue(t,i)?i:a(e.options);return r?(this.#e[t]=r,V(t,r),r):null},this.setAnalyticsUrls=t=>{this.#o=t},this.postAnalytic=(t,e,i)=>{var r;let s;if(!this.config||Date.now()<this.#n)return null;let n=this.config.tests[t];if(this.#e||this.setupVisitor(),!n||!this.#e||this.#s?.[t]?.[e])return null;let o=this.#s[t]||{};o[e]=!0,this.#s[t]=o;let a=this.#e?.[t]||null;if(a||(a=this.getTestValue(t)||null),!a)return;let l={organizationId:this.organizationId,environment:this.environment,testId:n.id,testValue:a,visitorId:this.#r,pointer:this.#e.pointer,device:this.#e.device,screen:(s=window.innerWidth)<=768?"small":s<=1024?"medium":s<=1200?"large":"huge",browser:this.#e.browser,os:this.#e.os,visitor:this.#i?"recurring":"new",event:k(e,256),message:k(i||"",256)};this.#a&&(r={event:e,improve:{test:t,variant:a,visitorId:this.#r},_improve:!0},"u">typeof window&&(window.dataLayer=window.dataLayer||[],window.dataLayer.push(r)));let u=fetch(this.#o,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify(l),keepalive:!0});return u.then(t=>{if(t?.status===429){let e=p(t.headers.get("Retry-After"));this.#n=Date.now()+(e??6e4)}}).catch(()=>{}),u},this.#o=`${this._baseUrl}${h}`,this.#a=l.dataLayer??!0}};
|
|
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/getRandomString.ts","../src/utils/getTimeoutError.ts","../src/utils/delay.ts","../src/base.ts","../src/utils/timeoutFetch.ts","../src/utils/clientCookie.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","export const BASE_URL = 'https://improve.obelism.studio'\nexport const CONFIG_PATH = `/config`\nexport const ANALYTICS_PATH = `/api/log`\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 { 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","export const delay = (ms: number) =>\n\tnew Promise((resolve) => setTimeout(resolve, ms))\n","import {\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 { getRandomString } from './utils/getRandomString'\nimport { timeoutFetch } from './utils/timeoutFetch'\n\ntype ConfigFetch = {\n\turl: string\n\ttimeout: number\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}: 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}\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 res = await timeoutFetch(\n\t\t\tthis.#configFetch.timeout,\n\t\t\tthis.#configFetch.url,\n\t\t\tconfig,\n\t\t)\n\t\tif (!res || !res.ok) throw new Error('Configuration fetch timed-out')\n\n\t\tthis.config = await res.json()\n\t\treturn this.config\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","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","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","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 { ANALYTICS_PATH, BASE_URL } from './config/urls'\nimport { getScreenSize } from './utils/getScreenSize'\nimport { pushDataLayer } from './utils/pushDataLayer'\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#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\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\tevent: event,\n\t\t\tmessage: message || '',\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\treturn 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})\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","CHAR_SET","getTimeoutError","timeout","controller","Promise","resolve","setTimeout","abort","BaseImproveSDK","organizationId","environment","state","config","fetchTimeout","baseUrl","_fetchConfig","Error","res","timeoutFetch","url","AbortController","race","fetch","signal","ok","json","loadConfig","generateVisitorId","getRandomString","characters","Array","fill","charAt","floor","toUpperCase","join","getVisitorCookieName","validateTestValue","testName","testValue","testConfig","tests","Boolean","option","validateVisitorId","possibleVisitorId","visitorIdParts","key","_baseUrl","getCookie","name","cookie","document","row","setCookie","now","Date","setDate","getDate","toUTCString","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","flags","flagValue","getTestValue","testSlug","defaultValue","allocation","cookieTestValue","validCookieTestValue","setAnalyticsUrls","postAnalytic","event","message","entry","size","testSlugAnalytics","body","testId","id","visitorId","screen","innerWidth","visitor","improve","test","variant","_improve","dataLayer","push","method","headers","JSON","stringify"],"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/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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"}
|
package/dist/client.d.ts
CHANGED
|
@@ -8,7 +8,7 @@ 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, }: ImproveSetupArgs);
|
|
11
|
+
constructor({ organizationId, environment, state, config, fetchTimeout, baseUrl, configRetries, configRevalidate, }: ImproveSetupArgs);
|
|
12
12
|
_fetchConfig: (config?: RequestInit) => Promise<ImproveConfiguration>;
|
|
13
13
|
loadConfig: (config: ImproveConfiguration) => void;
|
|
14
14
|
generateVisitorId: () => string;
|
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,yHAAyH,gBAAgB;AACzI,4BAA4B,WAAW,KAAK,OAAO,CAAC,oBAAoB;AACxE,yBAAyB,oBAAoB;AAC7C;AACA;AACA;AACA;AACA;;ACZA,KAAK,cAAc;AACnB;AACA,iBAAiB,wBAAwB;AACzC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;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,yEAAyE,OAAO,CAAC,QAAQ;AACzF;;;;","names":[]}
|
package/dist/client.mjs
CHANGED
|
@@ -1,2 +1,2 @@
|
|
|
1
|
-
import{UAParser as
|
|
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",c=t=>new Promise(e=>setTimeout(e,t)),d={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(d[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 c(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;constructor({organizationId:t,environment:e,state:i,config:r,fetchTimeout:s,baseUrl:n,configRetries:o,configRevalidate:a}){this.environment="develop",this.#t=null,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 c(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.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,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]},V=(t,e)=>{let i=new Date;i.setDate(i.getDate()+30),document.cookie=`${t}=${e};path=/;expires=${i.toUTCString()};SameSite=Lax;Secure`},k=(t,e)=>"string"==typeof t&&t.length>e?t.slice(0,e):t;class L extends C{#e;#i;#r;#s;#n;#o;#a;constructor(l){super(l),this.#i=!1,this.#r="",this.#s={},this.#n=0,this.#o=`${h}${u}`,this.#a=!0,this.fetchConfig=this._fetchConfig,this.setupVisitor=(o=window.navigator.userAgent)=>{let a=I(this.getVisitorCookieName()),l=a&&this.validateVisitorId(a);this.#i=l,this.#r=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.#e=h,V(this.getVisitorCookieName(),this.#r),this.#r):null},this.getFlagValue=t=>{if(!this.config)return null;let e=this.config.flags[t];if(!e||!e.options[0])return null;if(this.#e||this.setupVisitor(),!this.#r||!this.#e)return e.options[0].slug;if(this.#e?.[t])return this.#e[t];if(!o(this.config.audience[e.audience],this.#e))return e.options[0].slug;let i=I(t)||a(e.options);return i?(this.#e[t]=i,V(t,i),i):null},this.getTestValue=t=>{if(!this.config)return null;let e=this.config.tests[t];if(!e)return null;if(this.#e||this.setupVisitor(),!this.#r||!this.#e)return e.defaultValue;if(this.#e?.[t])return this.#e[t];if(!o(this.config.audience[e.audience],this.#e))return e.defaultValue;if(e.allocation<100&&100*Math.random()>e.allocation)return this.#e[t]=e.defaultValue,this.#e?.[t];let i=I(t),r=i&&this.validateTestValue(t,i)?i:a(e.options);return r?(this.#e[t]=r,V(t,r),r):null},this.setAnalyticsUrls=t=>{this.#o=t},this.postAnalytic=(t,e,i)=>{var r;let s;if(!this.config||Date.now()<this.#n)return null;let n=this.config.tests[t];if(this.#e||this.setupVisitor(),!n||!this.#e||this.#s?.[t]?.[e])return null;let o=this.#s[t]||{};o[e]=!0,this.#s[t]=o;let a=this.#e?.[t]||null;if(a||(a=this.getTestValue(t)||null),!a)return;let l={organizationId:this.organizationId,environment:this.environment,testId:n.id,testValue:a,visitorId:this.#r,pointer:this.#e.pointer,device:this.#e.device,screen:(s=window.innerWidth)<=768?"small":s<=1024?"medium":s<=1200?"large":"huge",browser:this.#e.browser,os:this.#e.os,visitor:this.#i?"recurring":"new",event:k(e,256),message:k(i||"",256)};this.#a&&(r={event:e,improve:{test:t,variant:a,visitorId:this.#r},_improve:!0},"u">typeof window&&(window.dataLayer=window.dataLayer||[],window.dataLayer.push(r)));let h=fetch(this.#o,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify(l),keepalive:!0});return h.then(t=>{if(t?.status===429){let e=p(t.headers.get("Retry-After"));this.#n=Date.now()+(e??6e4)}}).catch(()=>{}),h},this.#o=`${this._baseUrl}${u}`,this.#a=l.dataLayer??!0}}export{L as ImproveClientSDK};
|
|
2
2
|
//# sourceMappingURL=client.mjs.map
|
package/dist/client.mjs.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"client.mjs","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/getRandomString.ts","../src/utils/getTimeoutError.ts","../src/utils/delay.ts","../src/base.ts","../src/utils/timeoutFetch.ts","../src/utils/clientCookie.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","export const BASE_URL = 'https://improve.obelism.studio'\nexport const CONFIG_PATH = `/config`\nexport const ANALYTICS_PATH = `/api/log`\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 { 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","export const delay = (ms: number) =>\n\tnew Promise((resolve) => setTimeout(resolve, ms))\n","import {\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 { getRandomString } from './utils/getRandomString'\nimport { timeoutFetch } from './utils/timeoutFetch'\n\ntype ConfigFetch = {\n\turl: string\n\ttimeout: number\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}: 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}\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 res = await timeoutFetch(\n\t\t\tthis.#configFetch.timeout,\n\t\t\tthis.#configFetch.url,\n\t\t\tconfig,\n\t\t)\n\t\tif (!res || !res.ok) throw new Error('Configuration fetch timed-out')\n\n\t\tthis.config = await res.json()\n\t\treturn this.config\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","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","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","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 { ANALYTICS_PATH, BASE_URL } from './config/urls'\nimport { getScreenSize } from './utils/getScreenSize'\nimport { pushDataLayer } from './utils/pushDataLayer'\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#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\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\tevent: event,\n\t\t\tmessage: message || '',\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\treturn 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})\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","CHAR_SET","getTimeoutError","timeout","controller","Promise","resolve","setTimeout","abort","BaseImproveSDK","organizationId","environment","state","config","fetchTimeout","baseUrl","_fetchConfig","Error","res","timeoutFetch","url","AbortController","race","fetch","signal","ok","json","loadConfig","generateVisitorId","getRandomString","characters","Array","fill","charAt","floor","toUpperCase","join","getVisitorCookieName","validateTestValue","testName","testValue","testConfig","tests","Boolean","option","validateVisitorId","possibleVisitorId","visitorIdParts","key","_baseUrl","getCookie","name","cookie","document","row","setCookie","now","Date","setDate","getDate","toUTCString","ImproveClientSDK","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","flags","flagValue","getTestValue","testSlug","defaultValue","allocation","cookieTestValue","validCookieTestValue","setAnalyticsUrls","postAnalytic","event","message","entry","size","testSlugAnalytics","body","testId","id","visitorId","screen","innerWidth","visitor","improve","test","variant","_improve","dataLayer","push","method","headers","JSON","stringify"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"client.mjs","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","ImproveClientSDK","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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"}
|
package/dist/server.cjs
CHANGED
|
@@ -1,2 +1,2 @@
|
|
|
1
|
-
Object.defineProperty(exports,"__esModule",{value:!0});var
|
|
1
|
+
Object.defineProperty(exports,"__esModule",{value:!0});var e=require("ua-parser-js");let t=["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=(e,t)=>!e||Object.entries(e).every(([e,i])=>t[e]===i),a=e=>{if(0===e.length)return null;if(1===e.length)return e[0].slug;let t=Math.random()*e.reduce((e,{split:t})=>e+t,0);return(e.find(({split:e})=>(t-=e)<=0)||e[0]).slug},u="visi",l=e=>new Promise(t=>setTimeout(t,e)),h={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(e,t){super(h[e]),this.name="ImproveFetchError",this.reason=e,this.status=t?.status,this.retryAfterMs=t?.retryAfterMs,t&&"cause"in t&&(this.cause=t.cause)}get isRetryable(){return"rate-limited"===this.reason||"server"===this.reason||"timeout"===this.reason||"network"===this.reason}}let c=e=>429===e?"rate-limited":401===e||403===e?"unauthorized":e>=500?"server":"client",g="ABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789",d=e=>{if("object"!=typeof e||null===e||Array.isArray(e))throw new f("invalid-response");return{name:"string"==typeof e.name?e.name:"",version:"number"==typeof e.version?e.version:0,tests:e.tests??{},flags:e.flags??{},audience:e.audience??{}}},m=(e,t=Date.now())=>{if(!e)return;let i=Number(e);if(Number.isFinite(i))return Math.max(0,Math.round(1e3*i));let r=Date.parse(e);if(!Number.isNaN(r))return Math.max(0,r-t)},v=(e,t,i)=>Math.round(Math.random()*Math.min(i,t*2**e)),p=async(e=1e3,t)=>(await l(e),t?.abort(),null),w=(e=3e3,t,i)=>{let r=new AbortController;return Promise.race([fetch(t,{...i,signal:r.signal}),p(e,r)])};class b{#e;constructor({organizationId:e,environment:t,state:i,config:r,fetchTimeout:s,baseUrl:n,configRetries:o,configRevalidate:a}){this.environment="develop",this.#e=null,this.config=null,this._fetchConfig=async e=>{if(this.config)return;if(!this.#e)throw Error("No config fetch setup provided");let{url:t,timeout:i,retries:r,revalidate:s}=this.#e,n={...e};"number"==typeof s&&(n.next={...e?.next,revalidate:s});let o=new f("network");for(let e=0;e<=r;e++){let r;e>0&&await l(o.retryAfterMs??v(e-1,300,3e3));try{r=await w(i,t,n)}catch(e){o=new f(e instanceof Error&&"AbortError"===e.name?"timeout":"network",{cause:e});continue}if(!r){o=new f("timeout");continue}if(r.ok){try{this.config=d(await r.json())}catch(e){o=e instanceof f?e:new f("invalid-response",{cause:e});continue}return this.config}if(!(o=new f(c(r.status),{status:r.status,retryAfterMs:m(r.headers.get("Retry-After"))})).isRetryable)break}throw o},this.loadConfig=e=>{this.config=e},this.generateVisitorId=()=>[u,(function(e=5){return e&&"number"==typeof e?Array(e).fill("").reduce(e=>e+=g.charAt(Math.floor(Math.random()*g.length)),""):""})(26).toUpperCase()].join("_"),this.getVisitorCookieName=()=>"visitorId",this.validateTestValue=(e,t)=>{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[e];if(!i)throw Error(`No config found for ${e}`);return!!i.options.find(e=>e.slug===t)},this.validateVisitorId=e=>{let t=e.split("_");if(2!==t.length)return!1;let[i,r]=t;return i===u&&26===r.length},this.organizationId=e,this.environment=t,this.state=i,this._baseUrl=n||"https://improve.obelism.studio",r?this.config=r:this.#e={url:[`${this._baseUrl}/config`,this.organizationId,this.environment,this.state||"active"].join("/"),timeout:s||3e3,retries:o??2,revalidate:a}}}exports.ImproveServerSDK=class extends b{#t;#i;#r;constructor({token:u,maxVisitors:l,...h}){super(h),this.#t=new Map,this.fetchConfig=async e=>this._fetchConfig({...e,headers:{...e?.headers,token:this.#r}}),this.getFlagConfig=e=>this.config?.flags?.[e],this.getTestConfig=e=>this.config?.tests?.[e],this.#s=(o,a)=>{let u=this.#t.get(o);if(u)this.#t.delete(o),this.#t.set(o,u);else{if(this.#t.size>=this.#i){let e=this.#t.keys().next().value;e&&this.#t.delete(e)}u={},this.#t.set(o,u)}return u[a]=u[a]||(o=>{var a;if(!o||"string"!=typeof o)return null;let u=new e.UAParser(o).getResult(),l=(a=u.device.type)&&t.includes(a)?a:"desktop";return{pointer:n.includes(l)?"coarse":"fine",device:l,browser:((e="")=>{if(!e)return"other";let t=e.toLowerCase(),r=i.find(e=>t.includes(e));return r||(s.includes(t)?"social":"other")})(u.browser.name),os:((e="")=>{if(!e)return"unix";let t=e.toLowerCase(),i=r.find(e=>t.includes(e));return i||"unix"})(u.os.name)}})(a),u},this.getFlagValue=(e,t,i)=>{let r=this.getFlagConfig(e);if(!r||!this.config)return null;if(!t)return r.options[0].slug;let s=this.#s(t,i);if(s[i]?.[e])return s[i][e];if(!o(this.config.audience[r.audience],s[i]))return r.options[0].slug;let n=a(r.options);return n?(s[i][e]=n,n):null},this.getTestValue=(e,t,i)=>{let r=this.getTestConfig(e);if(!r||!this.config)return null;if(!t||!i)return r.defaultValue;let s=this.#s(t,i);if(s[i]?.[e])return s[i][e];if(!o(this.config.audience[r.audience],s[i]))return r.defaultValue;if(r.allocation<100&&100*Math.random()>r.allocation)return s[i][e]=r.defaultValue,s[i][e];let n=a(r.options);return n?(s[i][e]=n,n):null},this.#r=u,this.#i=l??1e4}#s};
|
|
2
2
|
//# sourceMappingURL=server.cjs.map
|
package/dist/server.cjs.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"server.cjs","sources":["../src/config/audiences.ts","../src/utils/parseUserAgent.ts","../src/utils/getVisitorMatchesAudience.ts","../src/utils/getRandomTestValue.ts","../src/config/constants.ts","../src/utils/getRandomString.ts","../src/utils/getTimeoutError.ts","../src/utils/delay.ts","../src/base.ts","../src/utils/timeoutFetch.ts","../src/config/urls.ts","../src/server.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 * @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 { 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","export const delay = (ms: number) =>\n\tnew Promise((resolve) => setTimeout(resolve, ms))\n","import {\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 { getRandomString } from './utils/getRandomString'\nimport { timeoutFetch } from './utils/timeoutFetch'\n\ntype ConfigFetch = {\n\turl: string\n\ttimeout: number\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}: 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}\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 res = await timeoutFetch(\n\t\t\tthis.#configFetch.timeout,\n\t\t\tthis.#configFetch.url,\n\t\t\tconfig,\n\t\t)\n\t\tif (!res || !res.ok) throw new Error('Configuration fetch timed-out')\n\n\t\tthis.config = await res.json()\n\t\treturn this.config\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","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","export const BASE_URL = 'https://improve.obelism.studio'\nexport const CONFIG_PATH = `/config`\nexport const ANALYTICS_PATH = `/api/log`\n","import { ParsedUserAgent, parseUserAgent } from './utils/parseUserAgent'\nimport { getVisitorMatchesAudience } from './utils/getVisitorMatchesAudience'\nimport { getRandomTestValue } from './utils/getRandomTestValue'\nimport { BaseImproveSDK } from './base'\nimport { ImproveConfiguration, ImproveSetupArgs } from './types'\n\nconst DEFAULT_MAX_VISITORS = 10_000\n\ntype VisitorData = {\n\t[userAgent: string]: ParsedUserAgent & {\n\t\t[testSlug: string]: string\n\t}\n}\n\ntype ImproveServerSetupArgs =\n\t| (Omit<ImproveSetupArgs, 'config' | 'baseUrl'> & {\n\t\t\tconfig: ImproveConfiguration\n\t\t\tmaxVisitors?: number\n\t })\n\t| (Omit<ImproveSetupArgs, 'config'> & {\n\t\t\ttoken: string\n\t\t\tmaxVisitors?: number\n\t })\n\nexport class ImproveServerSDK extends BaseImproveSDK {\n\t#visitors: Map<string, VisitorData> = new Map()\n\t#maxVisitors: number\n\t#token: string\n\n\t// @ts-ignore It could be there\n\tconstructor({ token, maxVisitors, ...args }: ImproveServerSetupArgs) {\n\t\tsuper(args)\n\t\tthis.#token = token\n\t\tthis.#maxVisitors = maxVisitors ?? DEFAULT_MAX_VISITORS\n\t}\n\n\tfetchConfig = async (config?: RequestInit) => {\n\t\treturn this._fetchConfig({\n\t\t\t...config,\n\t\t\theaders: {\n\t\t\t\t...config?.headers,\n\t\t\t\ttoken: this.#token,\n\t\t\t},\n\t\t})\n\t}\n\n\tgetFlagConfig = (flagSlug: string) => this.config?.flags?.[flagSlug]\n\n\tgetTestConfig = (testSlug: string) => this.config?.tests?.[testSlug]\n\n\t#getVisitor = (visitorId: string, userAgent: string) => {\n\t\tlet visitor = this.#visitors.get(visitorId)\n\t\tif (visitor) {\n\t\t\t// Move to end for LRU freshness\n\t\t\tthis.#visitors.delete(visitorId)\n\t\t\tthis.#visitors.set(visitorId, visitor)\n\t\t} else {\n\t\t\t// Evict oldest entries when at capacity\n\t\t\tif (this.#visitors.size >= this.#maxVisitors) {\n\t\t\t\tconst oldest = this.#visitors.keys().next().value\n\t\t\t\tif (oldest) this.#visitors.delete(oldest)\n\t\t\t}\n\t\t\tvisitor = {}\n\t\t\tthis.#visitors.set(visitorId, visitor)\n\t\t}\n\t\tvisitor[userAgent] = visitor[userAgent] || parseUserAgent(userAgent)\n\t\treturn visitor\n\t}\n\n\tgetFlagValue = (flagSlug: string, visitorId: string, userAgent: string) => {\n\t\tconst flagConfig = this.getFlagConfig(flagSlug)\n\n\t\tif (!flagConfig || !this.config) return null\n\t\tif (!visitorId) return flagConfig.options[0].slug\n\n\t\tconst visitor = this.#getVisitor(visitorId, userAgent)\n\n\t\tif (visitor[userAgent]?.[flagSlug]) {\n\t\t\treturn visitor[userAgent][flagSlug]\n\t\t}\n\n\t\tconst visitorMatchesAudience = getVisitorMatchesAudience(\n\t\t\tthis.config.audience[flagConfig.audience],\n\t\t\tvisitor[userAgent],\n\t\t)\n\n\t\tif (!visitorMatchesAudience) return flagConfig.options[0].slug\n\n\t\tconst flagValue = getRandomTestValue(flagConfig.options)\n\n\t\tif (!flagValue) return null\n\n\t\tvisitor[userAgent][flagSlug] = flagValue\n\t\treturn flagValue\n\t}\n\n\tgetTestValue = (testSlug: string, visitorId: string, userAgent: string) => {\n\t\tconst testConfig = this.getTestConfig(testSlug)\n\n\t\tif (!testConfig || !this.config) return null\n\n\t\tif (!visitorId || !userAgent) return testConfig.defaultValue\n\n\t\tconst visitor = this.#getVisitor(visitorId, userAgent)\n\n\t\tif (visitor[userAgent]?.[testSlug]) {\n\t\t\treturn visitor[userAgent][testSlug]\n\t\t}\n\n\t\tconst visitorMatchesAudience = getVisitorMatchesAudience(\n\t\t\tthis.config.audience[testConfig.audience],\n\t\t\tvisitor[userAgent],\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\tvisitor[userAgent][testSlug] = testConfig.defaultValue\n\t\t\treturn visitor[userAgent][testSlug]\n\t\t}\n\n\t\tconst testValue = getRandomTestValue(testConfig.options)\n\n\t\tif (!testValue) return null\n\n\t\tvisitor[userAgent][testSlug] = testValue\n\t\treturn testValue\n\t}\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","CHAR_SET","getTimeoutError","timeout","controller","Promise","resolve","setTimeout","abort","BaseImproveSDK","organizationId","environment","state","config","fetchTimeout","baseUrl","_fetchConfig","Error","res","timeoutFetch","url","AbortController","race","fetch","signal","ok","json","loadConfig","generateVisitorId","getRandomString","characters","Array","fill","charAt","floor","toUpperCase","join","getVisitorCookieName","validateTestValue","testName","testValue","testConfig","tests","Boolean","option","validateVisitorId","possibleVisitorId","visitorIdParts","key","_baseUrl","token","maxVisitors","args","Map","fetchConfig","headers","getFlagConfig","flagSlug","flags","getTestConfig","testSlug","visitorId","userAgent","visitor","get","delete","set","size","oldest","keys","next","parseUserAgent","deviceType","results","parser","UAParser","getResult","device","type","AUDIENCE_PARAMS","includes","pointer","browser","formatBrowser","browserName","name","toLowerCase","browserTypeMatch","browserType","os","formatOs","osName","osTypeMatch","osType","getFlagValue","flagConfig","flagValue","getTestValue","defaultValue","allocation"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"server.cjs","sources":["../src/config/audiences.ts","../src/utils/parseUserAgent.ts","../src/utils/getVisitorMatchesAudience.ts","../src/utils/getRandomTestValue.ts","../src/config/constants.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/config/urls.ts","../src/server.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 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","export const BASE_URL = 'https://improve.obelism.studio'\nexport const CONFIG_PATH = `/config`\nexport const ANALYTICS_PATH = `/api/log`\n","import { ParsedUserAgent, parseUserAgent } from './utils/parseUserAgent'\nimport { getVisitorMatchesAudience } from './utils/getVisitorMatchesAudience'\nimport { getRandomTestValue } from './utils/getRandomTestValue'\nimport { BaseImproveSDK } from './base'\nimport { ImproveConfiguration, ImproveSetupArgs } from './types'\n\nconst DEFAULT_MAX_VISITORS = 10_000\n\ntype VisitorData = {\n\t[userAgent: string]: ParsedUserAgent & {\n\t\t[testSlug: string]: string\n\t}\n}\n\ntype ImproveServerSetupArgs =\n\t| (Omit<ImproveSetupArgs, 'config' | 'baseUrl'> & {\n\t\t\tconfig: ImproveConfiguration\n\t\t\tmaxVisitors?: number\n\t })\n\t| (Omit<ImproveSetupArgs, 'config'> & {\n\t\t\ttoken: string\n\t\t\tmaxVisitors?: number\n\t })\n\nexport class ImproveServerSDK extends BaseImproveSDK {\n\t#visitors: Map<string, VisitorData> = new Map()\n\t#maxVisitors: number\n\t#token: string\n\n\t// @ts-ignore It could be there\n\tconstructor({ token, maxVisitors, ...args }: ImproveServerSetupArgs) {\n\t\tsuper(args)\n\t\tthis.#token = token\n\t\tthis.#maxVisitors = maxVisitors ?? DEFAULT_MAX_VISITORS\n\t}\n\n\tfetchConfig = async (config?: RequestInit) => {\n\t\treturn this._fetchConfig({\n\t\t\t...config,\n\t\t\theaders: {\n\t\t\t\t...config?.headers,\n\t\t\t\ttoken: this.#token,\n\t\t\t},\n\t\t})\n\t}\n\n\tgetFlagConfig = (flagSlug: string) => this.config?.flags?.[flagSlug]\n\n\tgetTestConfig = (testSlug: string) => this.config?.tests?.[testSlug]\n\n\t#getVisitor = (visitorId: string, userAgent: string) => {\n\t\tlet visitor = this.#visitors.get(visitorId)\n\t\tif (visitor) {\n\t\t\t// Move to end for LRU freshness\n\t\t\tthis.#visitors.delete(visitorId)\n\t\t\tthis.#visitors.set(visitorId, visitor)\n\t\t} else {\n\t\t\t// Evict oldest entries when at capacity\n\t\t\tif (this.#visitors.size >= this.#maxVisitors) {\n\t\t\t\tconst oldest = this.#visitors.keys().next().value\n\t\t\t\tif (oldest) this.#visitors.delete(oldest)\n\t\t\t}\n\t\t\tvisitor = {}\n\t\t\tthis.#visitors.set(visitorId, visitor)\n\t\t}\n\t\tvisitor[userAgent] = visitor[userAgent] || parseUserAgent(userAgent)\n\t\treturn visitor\n\t}\n\n\tgetFlagValue = (flagSlug: string, visitorId: string, userAgent: string) => {\n\t\tconst flagConfig = this.getFlagConfig(flagSlug)\n\n\t\tif (!flagConfig || !this.config) return null\n\t\tif (!visitorId) return flagConfig.options[0].slug\n\n\t\tconst visitor = this.#getVisitor(visitorId, userAgent)\n\n\t\tif (visitor[userAgent]?.[flagSlug]) {\n\t\t\treturn visitor[userAgent][flagSlug]\n\t\t}\n\n\t\tconst visitorMatchesAudience = getVisitorMatchesAudience(\n\t\t\tthis.config.audience[flagConfig.audience],\n\t\t\tvisitor[userAgent],\n\t\t)\n\n\t\tif (!visitorMatchesAudience) return flagConfig.options[0].slug\n\n\t\tconst flagValue = getRandomTestValue(flagConfig.options)\n\n\t\tif (!flagValue) return null\n\n\t\tvisitor[userAgent][flagSlug] = flagValue\n\t\treturn flagValue\n\t}\n\n\tgetTestValue = (testSlug: string, visitorId: string, userAgent: string) => {\n\t\tconst testConfig = this.getTestConfig(testSlug)\n\n\t\tif (!testConfig || !this.config) return null\n\n\t\tif (!visitorId || !userAgent) return testConfig.defaultValue\n\n\t\tconst visitor = this.#getVisitor(visitorId, userAgent)\n\n\t\tif (visitor[userAgent]?.[testSlug]) {\n\t\t\treturn visitor[userAgent][testSlug]\n\t\t}\n\n\t\tconst visitorMatchesAudience = getVisitorMatchesAudience(\n\t\t\tthis.config.audience[testConfig.audience],\n\t\t\tvisitor[userAgent],\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\tvisitor[userAgent][testSlug] = testConfig.defaultValue\n\t\t\treturn visitor[userAgent][testSlug]\n\t\t}\n\n\t\tconst testValue = getRandomTestValue(testConfig.options)\n\n\t\tif (!testValue) return null\n\n\t\tvisitor[userAgent][testSlug] = testValue\n\t\treturn testValue\n\t}\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","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","token","maxVisitors","args","Map","fetchConfig","getFlagConfig","flagSlug","getTestConfig","testSlug","visitorId","userAgent","visitor","delete","set","size","oldest","keys","parseUserAgent","deviceType","results","parser","UAParser","getResult","device","type","AUDIENCE_PARAMS","includes","pointer","browser","formatBrowser","browserName","toLowerCase","browserTypeMatch","browserType","os","formatOs","osName","osTypeMatch","osType","getFlagValue","flagConfig","flagValue","getTestValue","defaultValue","allocation"],"mappings":"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"}
|
package/dist/server.d.ts
CHANGED
|
@@ -8,7 +8,7 @@ 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, }: ImproveSetupArgs);
|
|
11
|
+
constructor({ organizationId, environment, state, config, fetchTimeout, baseUrl, configRetries, configRevalidate, }: ImproveSetupArgs);
|
|
12
12
|
_fetchConfig: (config?: RequestInit) => Promise<ImproveConfiguration>;
|
|
13
13
|
loadConfig: (config: ImproveConfiguration) => void;
|
|
14
14
|
generateVisitorId: () => string;
|
package/dist/server.d.ts.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"server.d.ts","sources":["../src/base.ts","../src/server.ts"],"mappings":";;;AACA,cAAc,cAAc;AAC5B;AACA;AACA,iBAAiB,wBAAwB;AACzC,WAAW,gBAAgB;AAC3B,YAAY,oBAAoB;AAChC;AACA,
|
|
1
|
+
{"version":3,"file":"server.d.ts","sources":["../src/base.ts","../src/server.ts"],"mappings":";;;AACA,cAAc,cAAc;AAC5B;AACA;AACA,iBAAiB,wBAAwB;AACzC,WAAW,gBAAgB;AAC3B,YAAY,oBAAoB;AAChC;AACA,yHAAyH,gBAAgB;AACzI,4BAA4B,WAAW,KAAK,OAAO,CAAC,oBAAoB;AACxE,yBAAyB,oBAAoB;AAC7C;AACA;AACA;AACA;AACA;;ACZA,KAAK,sBAAsB,IAAI,IAAI,CAAC,gBAAgB;AACpD,YAAY,oBAAoB;AAChC;AACA,MAAM,IAAI,CAAC,gBAAgB;AAC3B;AACA;AACA;AACA,cAAc,gBAAgB,SAAS,cAAc;AACrD;AACA,iDAAiD,sBAAsB;AACvE,2BAA2B,WAAW,KAAK,OAAO,CAAC,oBAAoB;AACvE,yCAAyC,OAAO,CAAC,WAAW;AAC5D,yCAAyC,OAAO,CAAC,WAAW;AAC5D;AACA;AACA;;;;","names":[]}
|
package/dist/server.mjs
CHANGED
|
@@ -1,2 +1,2 @@
|
|
|
1
|
-
import{UAParser as
|
|
1
|
+
import{UAParser as e}from"ua-parser-js";let t=["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=(e,t)=>!e||Object.entries(e).every(([e,i])=>t[e]===i),a=e=>{if(0===e.length)return null;if(1===e.length)return e[0].slug;let t=Math.random()*e.reduce((e,{split:t})=>e+t,0);return(e.find(({split:e})=>(t-=e)<=0)||e[0]).slug},u="visi",l=e=>new Promise(t=>setTimeout(t,e)),h={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(e,t){super(h[e]),this.name="ImproveFetchError",this.reason=e,this.status=t?.status,this.retryAfterMs=t?.retryAfterMs,t&&"cause"in t&&(this.cause=t.cause)}get isRetryable(){return"rate-limited"===this.reason||"server"===this.reason||"timeout"===this.reason||"network"===this.reason}}let c=e=>429===e?"rate-limited":401===e||403===e?"unauthorized":e>=500?"server":"client",g="ABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789",d=e=>{if("object"!=typeof e||null===e||Array.isArray(e))throw new f("invalid-response");return{name:"string"==typeof e.name?e.name:"",version:"number"==typeof e.version?e.version:0,tests:e.tests??{},flags:e.flags??{},audience:e.audience??{}}},m=(e,t=Date.now())=>{if(!e)return;let i=Number(e);if(Number.isFinite(i))return Math.max(0,Math.round(1e3*i));let r=Date.parse(e);if(!Number.isNaN(r))return Math.max(0,r-t)},v=(e,t,i)=>Math.round(Math.random()*Math.min(i,t*2**e)),p=async(e=1e3,t)=>(await l(e),t?.abort(),null),w=(e=3e3,t,i)=>{let r=new AbortController;return Promise.race([fetch(t,{...i,signal:r.signal}),p(e,r)])};class b{#e;constructor({organizationId:e,environment:t,state:i,config:r,fetchTimeout:s,baseUrl:n,configRetries:o,configRevalidate:a}){this.environment="develop",this.#e=null,this.config=null,this._fetchConfig=async e=>{if(this.config)return;if(!this.#e)throw Error("No config fetch setup provided");let{url:t,timeout:i,retries:r,revalidate:s}=this.#e,n={...e};"number"==typeof s&&(n.next={...e?.next,revalidate:s});let o=new f("network");for(let e=0;e<=r;e++){let r;e>0&&await l(o.retryAfterMs??v(e-1,300,3e3));try{r=await w(i,t,n)}catch(e){o=new f(e instanceof Error&&"AbortError"===e.name?"timeout":"network",{cause:e});continue}if(!r){o=new f("timeout");continue}if(r.ok){try{this.config=d(await r.json())}catch(e){o=e instanceof f?e:new f("invalid-response",{cause:e});continue}return this.config}if(!(o=new f(c(r.status),{status:r.status,retryAfterMs:m(r.headers.get("Retry-After"))})).isRetryable)break}throw o},this.loadConfig=e=>{this.config=e},this.generateVisitorId=()=>[u,(function(e=5){return e&&"number"==typeof e?Array(e).fill("").reduce(e=>e+=g.charAt(Math.floor(Math.random()*g.length)),""):""})(26).toUpperCase()].join("_"),this.getVisitorCookieName=()=>"visitorId",this.validateTestValue=(e,t)=>{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[e];if(!i)throw Error(`No config found for ${e}`);return!!i.options.find(e=>e.slug===t)},this.validateVisitorId=e=>{let t=e.split("_");if(2!==t.length)return!1;let[i,r]=t;return i===u&&26===r.length},this.organizationId=e,this.environment=t,this.state=i,this._baseUrl=n||"https://improve.obelism.studio",r?this.config=r:this.#e={url:[`${this._baseUrl}/config`,this.organizationId,this.environment,this.state||"active"].join("/"),timeout:s||3e3,retries:o??2,revalidate:a}}}class y extends b{#t;#i;#r;constructor({token:u,maxVisitors:l,...h}){super(h),this.#t=new Map,this.fetchConfig=async e=>this._fetchConfig({...e,headers:{...e?.headers,token:this.#r}}),this.getFlagConfig=e=>this.config?.flags?.[e],this.getTestConfig=e=>this.config?.tests?.[e],this.#s=(o,a)=>{let u=this.#t.get(o);if(u)this.#t.delete(o),this.#t.set(o,u);else{if(this.#t.size>=this.#i){let e=this.#t.keys().next().value;e&&this.#t.delete(e)}u={},this.#t.set(o,u)}return u[a]=u[a]||(o=>{var a;if(!o||"string"!=typeof o)return null;let u=new e(o).getResult(),l=(a=u.device.type)&&t.includes(a)?a:"desktop";return{pointer:n.includes(l)?"coarse":"fine",device:l,browser:((e="")=>{if(!e)return"other";let t=e.toLowerCase(),r=i.find(e=>t.includes(e));return r||(s.includes(t)?"social":"other")})(u.browser.name),os:((e="")=>{if(!e)return"unix";let t=e.toLowerCase(),i=r.find(e=>t.includes(e));return i||"unix"})(u.os.name)}})(a),u},this.getFlagValue=(e,t,i)=>{let r=this.getFlagConfig(e);if(!r||!this.config)return null;if(!t)return r.options[0].slug;let s=this.#s(t,i);if(s[i]?.[e])return s[i][e];if(!o(this.config.audience[r.audience],s[i]))return r.options[0].slug;let n=a(r.options);return n?(s[i][e]=n,n):null},this.getTestValue=(e,t,i)=>{let r=this.getTestConfig(e);if(!r||!this.config)return null;if(!t||!i)return r.defaultValue;let s=this.#s(t,i);if(s[i]?.[e])return s[i][e];if(!o(this.config.audience[r.audience],s[i]))return r.defaultValue;if(r.allocation<100&&100*Math.random()>r.allocation)return s[i][e]=r.defaultValue,s[i][e];let n=a(r.options);return n?(s[i][e]=n,n):null},this.#r=u,this.#i=l??1e4}#s}export{y as ImproveServerSDK};
|
|
2
2
|
//# sourceMappingURL=server.mjs.map
|
package/dist/server.mjs.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"server.mjs","sources":["../src/config/audiences.ts","../src/utils/parseUserAgent.ts","../src/utils/getVisitorMatchesAudience.ts","../src/utils/getRandomTestValue.ts","../src/config/constants.ts","../src/utils/getRandomString.ts","../src/utils/getTimeoutError.ts","../src/utils/delay.ts","../src/base.ts","../src/utils/timeoutFetch.ts","../src/config/urls.ts","../src/server.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 * @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 { 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","export const delay = (ms: number) =>\n\tnew Promise((resolve) => setTimeout(resolve, ms))\n","import {\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 { getRandomString } from './utils/getRandomString'\nimport { timeoutFetch } from './utils/timeoutFetch'\n\ntype ConfigFetch = {\n\turl: string\n\ttimeout: number\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}: 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}\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 res = await timeoutFetch(\n\t\t\tthis.#configFetch.timeout,\n\t\t\tthis.#configFetch.url,\n\t\t\tconfig,\n\t\t)\n\t\tif (!res || !res.ok) throw new Error('Configuration fetch timed-out')\n\n\t\tthis.config = await res.json()\n\t\treturn this.config\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","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","export const BASE_URL = 'https://improve.obelism.studio'\nexport const CONFIG_PATH = `/config`\nexport const ANALYTICS_PATH = `/api/log`\n","import { ParsedUserAgent, parseUserAgent } from './utils/parseUserAgent'\nimport { getVisitorMatchesAudience } from './utils/getVisitorMatchesAudience'\nimport { getRandomTestValue } from './utils/getRandomTestValue'\nimport { BaseImproveSDK } from './base'\nimport { ImproveConfiguration, ImproveSetupArgs } from './types'\n\nconst DEFAULT_MAX_VISITORS = 10_000\n\ntype VisitorData = {\n\t[userAgent: string]: ParsedUserAgent & {\n\t\t[testSlug: string]: string\n\t}\n}\n\ntype ImproveServerSetupArgs =\n\t| (Omit<ImproveSetupArgs, 'config' | 'baseUrl'> & {\n\t\t\tconfig: ImproveConfiguration\n\t\t\tmaxVisitors?: number\n\t })\n\t| (Omit<ImproveSetupArgs, 'config'> & {\n\t\t\ttoken: string\n\t\t\tmaxVisitors?: number\n\t })\n\nexport class ImproveServerSDK extends BaseImproveSDK {\n\t#visitors: Map<string, VisitorData> = new Map()\n\t#maxVisitors: number\n\t#token: string\n\n\t// @ts-ignore It could be there\n\tconstructor({ token, maxVisitors, ...args }: ImproveServerSetupArgs) {\n\t\tsuper(args)\n\t\tthis.#token = token\n\t\tthis.#maxVisitors = maxVisitors ?? DEFAULT_MAX_VISITORS\n\t}\n\n\tfetchConfig = async (config?: RequestInit) => {\n\t\treturn this._fetchConfig({\n\t\t\t...config,\n\t\t\theaders: {\n\t\t\t\t...config?.headers,\n\t\t\t\ttoken: this.#token,\n\t\t\t},\n\t\t})\n\t}\n\n\tgetFlagConfig = (flagSlug: string) => this.config?.flags?.[flagSlug]\n\n\tgetTestConfig = (testSlug: string) => this.config?.tests?.[testSlug]\n\n\t#getVisitor = (visitorId: string, userAgent: string) => {\n\t\tlet visitor = this.#visitors.get(visitorId)\n\t\tif (visitor) {\n\t\t\t// Move to end for LRU freshness\n\t\t\tthis.#visitors.delete(visitorId)\n\t\t\tthis.#visitors.set(visitorId, visitor)\n\t\t} else {\n\t\t\t// Evict oldest entries when at capacity\n\t\t\tif (this.#visitors.size >= this.#maxVisitors) {\n\t\t\t\tconst oldest = this.#visitors.keys().next().value\n\t\t\t\tif (oldest) this.#visitors.delete(oldest)\n\t\t\t}\n\t\t\tvisitor = {}\n\t\t\tthis.#visitors.set(visitorId, visitor)\n\t\t}\n\t\tvisitor[userAgent] = visitor[userAgent] || parseUserAgent(userAgent)\n\t\treturn visitor\n\t}\n\n\tgetFlagValue = (flagSlug: string, visitorId: string, userAgent: string) => {\n\t\tconst flagConfig = this.getFlagConfig(flagSlug)\n\n\t\tif (!flagConfig || !this.config) return null\n\t\tif (!visitorId) return flagConfig.options[0].slug\n\n\t\tconst visitor = this.#getVisitor(visitorId, userAgent)\n\n\t\tif (visitor[userAgent]?.[flagSlug]) {\n\t\t\treturn visitor[userAgent][flagSlug]\n\t\t}\n\n\t\tconst visitorMatchesAudience = getVisitorMatchesAudience(\n\t\t\tthis.config.audience[flagConfig.audience],\n\t\t\tvisitor[userAgent],\n\t\t)\n\n\t\tif (!visitorMatchesAudience) return flagConfig.options[0].slug\n\n\t\tconst flagValue = getRandomTestValue(flagConfig.options)\n\n\t\tif (!flagValue) return null\n\n\t\tvisitor[userAgent][flagSlug] = flagValue\n\t\treturn flagValue\n\t}\n\n\tgetTestValue = (testSlug: string, visitorId: string, userAgent: string) => {\n\t\tconst testConfig = this.getTestConfig(testSlug)\n\n\t\tif (!testConfig || !this.config) return null\n\n\t\tif (!visitorId || !userAgent) return testConfig.defaultValue\n\n\t\tconst visitor = this.#getVisitor(visitorId, userAgent)\n\n\t\tif (visitor[userAgent]?.[testSlug]) {\n\t\t\treturn visitor[userAgent][testSlug]\n\t\t}\n\n\t\tconst visitorMatchesAudience = getVisitorMatchesAudience(\n\t\t\tthis.config.audience[testConfig.audience],\n\t\t\tvisitor[userAgent],\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\tvisitor[userAgent][testSlug] = testConfig.defaultValue\n\t\t\treturn visitor[userAgent][testSlug]\n\t\t}\n\n\t\tconst testValue = getRandomTestValue(testConfig.options)\n\n\t\tif (!testValue) return null\n\n\t\tvisitor[userAgent][testSlug] = testValue\n\t\treturn testValue\n\t}\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","CHAR_SET","getTimeoutError","timeout","controller","Promise","resolve","setTimeout","abort","BaseImproveSDK","organizationId","environment","state","config","fetchTimeout","baseUrl","_fetchConfig","Error","res","timeoutFetch","url","AbortController","race","fetch","signal","ok","json","loadConfig","generateVisitorId","getRandomString","characters","Array","fill","charAt","floor","toUpperCase","join","getVisitorCookieName","validateTestValue","testName","testValue","testConfig","tests","Boolean","option","validateVisitorId","possibleVisitorId","visitorIdParts","key","_baseUrl","ImproveServerSDK","token","maxVisitors","args","Map","fetchConfig","headers","getFlagConfig","flagSlug","flags","getTestConfig","testSlug","visitorId","userAgent","visitor","get","delete","set","size","oldest","keys","next","parseUserAgent","deviceType","results","parser","UAParser","getResult","device","type","AUDIENCE_PARAMS","includes","pointer","browser","formatBrowser","browserName","name","toLowerCase","browserTypeMatch","browserType","os","formatOs","osName","osTypeMatch","osType","getFlagValue","flagConfig","flagValue","getTestValue","defaultValue","allocation"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"server.mjs","sources":["../src/config/audiences.ts","../src/utils/parseUserAgent.ts","../src/utils/getVisitorMatchesAudience.ts","../src/utils/getRandomTestValue.ts","../src/config/constants.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/config/urls.ts","../src/server.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 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","export const BASE_URL = 'https://improve.obelism.studio'\nexport const CONFIG_PATH = `/config`\nexport const ANALYTICS_PATH = `/api/log`\n","import { ParsedUserAgent, parseUserAgent } from './utils/parseUserAgent'\nimport { getVisitorMatchesAudience } from './utils/getVisitorMatchesAudience'\nimport { getRandomTestValue } from './utils/getRandomTestValue'\nimport { BaseImproveSDK } from './base'\nimport { ImproveConfiguration, ImproveSetupArgs } from './types'\n\nconst DEFAULT_MAX_VISITORS = 10_000\n\ntype VisitorData = {\n\t[userAgent: string]: ParsedUserAgent & {\n\t\t[testSlug: string]: string\n\t}\n}\n\ntype ImproveServerSetupArgs =\n\t| (Omit<ImproveSetupArgs, 'config' | 'baseUrl'> & {\n\t\t\tconfig: ImproveConfiguration\n\t\t\tmaxVisitors?: number\n\t })\n\t| (Omit<ImproveSetupArgs, 'config'> & {\n\t\t\ttoken: string\n\t\t\tmaxVisitors?: number\n\t })\n\nexport class ImproveServerSDK extends BaseImproveSDK {\n\t#visitors: Map<string, VisitorData> = new Map()\n\t#maxVisitors: number\n\t#token: string\n\n\t// @ts-ignore It could be there\n\tconstructor({ token, maxVisitors, ...args }: ImproveServerSetupArgs) {\n\t\tsuper(args)\n\t\tthis.#token = token\n\t\tthis.#maxVisitors = maxVisitors ?? DEFAULT_MAX_VISITORS\n\t}\n\n\tfetchConfig = async (config?: RequestInit) => {\n\t\treturn this._fetchConfig({\n\t\t\t...config,\n\t\t\theaders: {\n\t\t\t\t...config?.headers,\n\t\t\t\ttoken: this.#token,\n\t\t\t},\n\t\t})\n\t}\n\n\tgetFlagConfig = (flagSlug: string) => this.config?.flags?.[flagSlug]\n\n\tgetTestConfig = (testSlug: string) => this.config?.tests?.[testSlug]\n\n\t#getVisitor = (visitorId: string, userAgent: string) => {\n\t\tlet visitor = this.#visitors.get(visitorId)\n\t\tif (visitor) {\n\t\t\t// Move to end for LRU freshness\n\t\t\tthis.#visitors.delete(visitorId)\n\t\t\tthis.#visitors.set(visitorId, visitor)\n\t\t} else {\n\t\t\t// Evict oldest entries when at capacity\n\t\t\tif (this.#visitors.size >= this.#maxVisitors) {\n\t\t\t\tconst oldest = this.#visitors.keys().next().value\n\t\t\t\tif (oldest) this.#visitors.delete(oldest)\n\t\t\t}\n\t\t\tvisitor = {}\n\t\t\tthis.#visitors.set(visitorId, visitor)\n\t\t}\n\t\tvisitor[userAgent] = visitor[userAgent] || parseUserAgent(userAgent)\n\t\treturn visitor\n\t}\n\n\tgetFlagValue = (flagSlug: string, visitorId: string, userAgent: string) => {\n\t\tconst flagConfig = this.getFlagConfig(flagSlug)\n\n\t\tif (!flagConfig || !this.config) return null\n\t\tif (!visitorId) return flagConfig.options[0].slug\n\n\t\tconst visitor = this.#getVisitor(visitorId, userAgent)\n\n\t\tif (visitor[userAgent]?.[flagSlug]) {\n\t\t\treturn visitor[userAgent][flagSlug]\n\t\t}\n\n\t\tconst visitorMatchesAudience = getVisitorMatchesAudience(\n\t\t\tthis.config.audience[flagConfig.audience],\n\t\t\tvisitor[userAgent],\n\t\t)\n\n\t\tif (!visitorMatchesAudience) return flagConfig.options[0].slug\n\n\t\tconst flagValue = getRandomTestValue(flagConfig.options)\n\n\t\tif (!flagValue) return null\n\n\t\tvisitor[userAgent][flagSlug] = flagValue\n\t\treturn flagValue\n\t}\n\n\tgetTestValue = (testSlug: string, visitorId: string, userAgent: string) => {\n\t\tconst testConfig = this.getTestConfig(testSlug)\n\n\t\tif (!testConfig || !this.config) return null\n\n\t\tif (!visitorId || !userAgent) return testConfig.defaultValue\n\n\t\tconst visitor = this.#getVisitor(visitorId, userAgent)\n\n\t\tif (visitor[userAgent]?.[testSlug]) {\n\t\t\treturn visitor[userAgent][testSlug]\n\t\t}\n\n\t\tconst visitorMatchesAudience = getVisitorMatchesAudience(\n\t\t\tthis.config.audience[testConfig.audience],\n\t\t\tvisitor[userAgent],\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\tvisitor[userAgent][testSlug] = testConfig.defaultValue\n\t\t\treturn visitor[userAgent][testSlug]\n\t\t}\n\n\t\tconst testValue = getRandomTestValue(testConfig.options)\n\n\t\tif (!testValue) return null\n\n\t\tvisitor[userAgent][testSlug] = testValue\n\t\treturn testValue\n\t}\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","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","ImproveServerSDK","token","maxVisitors","args","Map","fetchConfig","getFlagConfig","flagSlug","getTestConfig","testSlug","visitorId","userAgent","visitor","delete","set","size","oldest","keys","parseUserAgent","deviceType","results","parser","UAParser","getResult","device","type","AUDIENCE_PARAMS","includes","pointer","browser","formatBrowser","browserName","toLowerCase","browserTypeMatch","browserType","os","formatOs","osName","osTypeMatch","osType","getFlagValue","flagConfig","flagValue","getTestValue","defaultValue","allocation"],"mappings":"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"}
|
package/dist/types.d.ts
CHANGED
|
@@ -19,6 +19,25 @@ type ImproveSetupArgs = {
|
|
|
19
19
|
config?: ImproveConfiguration;
|
|
20
20
|
baseUrl?: string;
|
|
21
21
|
fetchTimeout?: number;
|
|
22
|
+
/**
|
|
23
|
+
* How many times to retry a failed config fetch before giving up. Only
|
|
24
|
+
* transient failures are retried (network errors, timeouts, HTTP 5xx, and
|
|
25
|
+
* 429 rate limits); auth/validation rejections (403/404/400) fail fast.
|
|
26
|
+
* Retries use exponential backoff with jitter and honor `Retry-After`.
|
|
27
|
+
* Defaults to 2. Set to 0 to disable retries (e.g. in latency-sensitive
|
|
28
|
+
* middleware).
|
|
29
|
+
*/
|
|
30
|
+
configRetries?: number;
|
|
31
|
+
/**
|
|
32
|
+
* When set, forwarded to the config fetch as `next: { revalidate }`
|
|
33
|
+
* (seconds), so Next.js / edge runtimes cache the config datafile between
|
|
34
|
+
* requests instead of fetching it on every request. The backend serves the
|
|
35
|
+
* datafile with a short `s-maxage` + `stale-while-revalidate`, so a small
|
|
36
|
+
* value here (e.g. 30) lets a deployment reuse it and shields the origin
|
|
37
|
+
* under load. No effect outside fetch implementations that honor
|
|
38
|
+
* `next.revalidate`.
|
|
39
|
+
*/
|
|
40
|
+
configRevalidate?: number;
|
|
22
41
|
/**
|
|
23
42
|
* Mirror analytics onto the GTM dataLayer (window.dataLayer) with experiment
|
|
24
43
|
* dimensions, so they can drive Google Tag Manager / Google Ads conversions.
|
package/dist/types.d.ts.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"types.d.ts","sources":["../src/config/audiences.ts","../src/types.ts"],"mappings":"AAAA;AACA;AACA;AACA;AACA,cAAc,eAAe;AAC7B;AACA;AACA;AACA;AACA;AACA,KAAK,gBAAgB,gBAAgB,eAAe;;ACTpD,KAAK,wBAAwB;AAC7B,KAAK,gBAAgB;AACrB,KAAK,gBAAgB;AACrB;AACA,iBAAiB,wBAAwB;AACzC,YAAY,gBAAgB;AAC5B,aAAa,oBAAoB;AACjC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,KAAK,iBAAiB;AACtB;AACA;AACA;AACA;AACA;AACA,KAAK,iBAAiB;AACtB;AACA;AACA;AACA;AACA;AACA,KAAK,WAAW;AAChB;AACA;AACA;AACA,aAAa,iBAAiB;AAC9B;AACA,KAAK,YAAY;AACjB,0BAA0B,WAAW;AACrC;AACA,KAAK,aAAa;AAClB;AACA;AACA;AACA;AACA,KAAK,aAAa;AAClB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,KAAK,WAAW;AAChB;AACA;AACA;AACA;AACA;AACA,aAAa,iBAAiB;AAC9B,YAAY,aAAa;AACzB;AACA,KAAK,YAAY;AACjB,0BAA0B,WAAW;AACrC;AACA,KAAK,oBAAoB;AACzB,YAAY,gBAAgB;AAC5B;AACA,KAAK,eAAe;AACpB,8BAA8B,oBAAoB;AAClD;AACA,KAAK,oBAAoB;AACzB;AACA;AACA,WAAW,YAAY;AACvB,WAAW,YAAY;AACvB,cAAc,eAAe;AAC7B;;;;","names":[]}
|
|
1
|
+
{"version":3,"file":"types.d.ts","sources":["../src/config/audiences.ts","../src/types.ts"],"mappings":"AAAA;AACA;AACA;AACA;AACA,cAAc,eAAe;AAC7B;AACA;AACA;AACA;AACA;AACA,KAAK,gBAAgB,gBAAgB,eAAe;;ACTpD,KAAK,wBAAwB;AAC7B,KAAK,gBAAgB;AACrB,KAAK,gBAAgB;AACrB;AACA,iBAAiB,wBAAwB;AACzC,YAAY,gBAAgB;AAC5B,aAAa,oBAAoB;AACjC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,KAAK,iBAAiB;AACtB;AACA;AACA;AACA;AACA;AACA,KAAK,iBAAiB;AACtB;AACA;AACA;AACA;AACA;AACA,KAAK,WAAW;AAChB;AACA;AACA;AACA,aAAa,iBAAiB;AAC9B;AACA,KAAK,YAAY;AACjB,0BAA0B,WAAW;AACrC;AACA,KAAK,aAAa;AAClB;AACA;AACA;AACA;AACA,KAAK,aAAa;AAClB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,KAAK,WAAW;AAChB;AACA;AACA;AACA;AACA;AACA,aAAa,iBAAiB;AAC9B,YAAY,aAAa;AACzB;AACA,KAAK,YAAY;AACjB,0BAA0B,WAAW;AACrC;AACA,KAAK,oBAAoB;AACzB,YAAY,gBAAgB;AAC5B;AACA,KAAK,eAAe;AACpB,8BAA8B,oBAAoB;AAClD;AACA,KAAK,oBAAoB;AACzB;AACA;AACA,WAAW,YAAY;AACvB,WAAW,YAAY;AACvB,cAAc,eAAe;AAC7B;;;;","names":[]}
|