@schematichq/schematic-js 0.1.13 → 1.0.0-rc.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.
@@ -1,3 +1,3 @@
1
- "use strict";(()=>{var G=Object.create;var I=Object.defineProperty;var X=Object.getOwnPropertyDescriptor;var z=Object.getOwnPropertyNames;var Y=Object.getPrototypeOf,Z=Object.prototype.hasOwnProperty;var ee=(i,r)=>()=>(r||i((r={exports:{}}).exports,r),r.exports);var te=(i,r,n,o)=>{if(r&&typeof r=="object"||typeof r=="function")for(let s of z(r))!Z.call(i,s)&&s!==n&&I(i,s,{get:()=>r[s],enumerable:!(o=X(r,s))||o.enumerable});return i};var re=(i,r,n)=>(n=i!=null?G(Y(i)):{},te(r||!i||!i.__esModule?I(n,"default",{value:i,enumerable:!0}):n,i));var L=ee(F=>{(function(i){var r=function(n){var o=typeof globalThis<"u"&&globalThis||typeof i<"u"&&i||typeof o<"u"&&o,s={searchParams:"URLSearchParams"in o,iterable:"Symbol"in o&&"iterator"in Symbol,blob:"FileReader"in o&&"Blob"in o&&function(){try{return new Blob,!0}catch{return!1}}(),formData:"FormData"in o,arrayBuffer:"ArrayBuffer"in o};function y(e){return e&&DataView.prototype.isPrototypeOf(e)}if(s.arrayBuffer)var h=["[object Int8Array]","[object Uint8Array]","[object Uint8ClampedArray]","[object Int16Array]","[object Uint16Array]","[object Int32Array]","[object Uint32Array]","[object Float32Array]","[object Float64Array]"],b=ArrayBuffer.isView||function(e){return e&&h.indexOf(Object.prototype.toString.call(e))>-1};function g(e){if(typeof e!="string"&&(e=String(e)),/[^a-z0-9\-#$%&'*+.^_`|~!]/i.test(e)||e==="")throw new TypeError('Invalid character in header field name: "'+e+'"');return e.toLowerCase()}function S(e){return typeof e!="string"&&(e=String(e)),e}function E(e){var t={next:function(){var a=e.shift();return{done:a===void 0,value:a}}};return s.iterable&&(t[Symbol.iterator]=function(){return t}),t}function d(e){this.map={},e instanceof d?e.forEach(function(t,a){this.append(a,t)},this):Array.isArray(e)?e.forEach(function(t){this.append(t[0],t[1])},this):e&&Object.getOwnPropertyNames(e).forEach(function(t){this.append(t,e[t])},this)}d.prototype.append=function(e,t){e=g(e),t=S(t);var a=this.map[e];this.map[e]=a?a+", "+t:t},d.prototype.delete=function(e){delete this.map[g(e)]},d.prototype.get=function(e){return e=g(e),this.has(e)?this.map[e]:null},d.prototype.has=function(e){return this.map.hasOwnProperty(g(e))},d.prototype.set=function(e,t){this.map[g(e)]=S(t)},d.prototype.forEach=function(e,t){for(var a in this.map)this.map.hasOwnProperty(a)&&e.call(t,this.map[a],a,this)},d.prototype.keys=function(){var e=[];return this.forEach(function(t,a){e.push(a)}),E(e)},d.prototype.values=function(){var e=[];return this.forEach(function(t){e.push(t)}),E(e)},d.prototype.entries=function(){var e=[];return this.forEach(function(t,a){e.push([a,t])}),E(e)},s.iterable&&(d.prototype[Symbol.iterator]=d.prototype.entries);function _(e){if(e.bodyUsed)return Promise.reject(new TypeError("Already read"));e.bodyUsed=!0}function R(e){return new Promise(function(t,a){e.onload=function(){t(e.result)},e.onerror=function(){a(e.error)}})}function N(e){var t=new FileReader,a=R(t);return t.readAsArrayBuffer(e),a}function q(e){var t=new FileReader,a=R(t);return t.readAsText(e),a}function M(e){for(var t=new Uint8Array(e),a=new Array(t.length),u=0;u<t.length;u++)a[u]=String.fromCharCode(t[u]);return a.join("")}function C(e){if(e.slice)return e.slice(0);var t=new Uint8Array(e.byteLength);return t.set(new Uint8Array(e)),t.buffer}function D(){return this.bodyUsed=!1,this._initBody=function(e){this.bodyUsed=this.bodyUsed,this._bodyInit=e,e?typeof e=="string"?this._bodyText=e:s.blob&&Blob.prototype.isPrototypeOf(e)?this._bodyBlob=e:s.formData&&FormData.prototype.isPrototypeOf(e)?this._bodyFormData=e:s.searchParams&&URLSearchParams.prototype.isPrototypeOf(e)?this._bodyText=e.toString():s.arrayBuffer&&s.blob&&y(e)?(this._bodyArrayBuffer=C(e.buffer),this._bodyInit=new Blob([this._bodyArrayBuffer])):s.arrayBuffer&&(ArrayBuffer.prototype.isPrototypeOf(e)||b(e))?this._bodyArrayBuffer=C(e):this._bodyText=e=Object.prototype.toString.call(e):this._bodyText="",this.headers.get("content-type")||(typeof e=="string"?this.headers.set("content-type","text/plain;charset=UTF-8"):this._bodyBlob&&this._bodyBlob.type?this.headers.set("content-type",this._bodyBlob.type):s.searchParams&&URLSearchParams.prototype.isPrototypeOf(e)&&this.headers.set("content-type","application/x-www-form-urlencoded;charset=UTF-8"))},s.blob&&(this.blob=function(){var e=_(this);if(e)return e;if(this._bodyBlob)return Promise.resolve(this._bodyBlob);if(this._bodyArrayBuffer)return Promise.resolve(new Blob([this._bodyArrayBuffer]));if(this._bodyFormData)throw new Error("could not read FormData body as blob");return Promise.resolve(new Blob([this._bodyText]))},this.arrayBuffer=function(){if(this._bodyArrayBuffer){var e=_(this);return e||(ArrayBuffer.isView(this._bodyArrayBuffer)?Promise.resolve(this._bodyArrayBuffer.buffer.slice(this._bodyArrayBuffer.byteOffset,this._bodyArrayBuffer.byteOffset+this._bodyArrayBuffer.byteLength)):Promise.resolve(this._bodyArrayBuffer))}else return this.blob().then(N)}),this.text=function(){var e=_(this);if(e)return e;if(this._bodyBlob)return q(this._bodyBlob);if(this._bodyArrayBuffer)return Promise.resolve(M(this._bodyArrayBuffer));if(this._bodyFormData)throw new Error("could not read FormData body as text");return Promise.resolve(this._bodyText)},s.formData&&(this.formData=function(){return this.text().then(V)}),this.json=function(){return this.text().then(JSON.parse)},this}var W=["DELETE","GET","HEAD","OPTIONS","POST","PUT"];function H(e){var t=e.toUpperCase();return W.indexOf(t)>-1?t:e}function v(e,t){if(!(this instanceof v))throw new TypeError('Please use the "new" operator, this DOM object constructor cannot be called as a function.');t=t||{};var a=t.body;if(e instanceof v){if(e.bodyUsed)throw new TypeError("Already read");this.url=e.url,this.credentials=e.credentials,t.headers||(this.headers=new d(e.headers)),this.method=e.method,this.mode=e.mode,this.signal=e.signal,!a&&e._bodyInit!=null&&(a=e._bodyInit,e.bodyUsed=!0)}else this.url=String(e);if(this.credentials=t.credentials||this.credentials||"same-origin",(t.headers||!this.headers)&&(this.headers=new d(t.headers)),this.method=H(t.method||this.method||"GET"),this.mode=t.mode||this.mode||null,this.signal=t.signal||this.signal,this.referrer=null,(this.method==="GET"||this.method==="HEAD")&&a)throw new TypeError("Body not allowed for GET or HEAD requests");if(this._initBody(a),(this.method==="GET"||this.method==="HEAD")&&(t.cache==="no-store"||t.cache==="no-cache")){var u=/([?&])_=[^&]*/;if(u.test(this.url))this.url=this.url.replace(u,"$1_="+new Date().getTime());else{var f=/\?/;this.url+=(f.test(this.url)?"&":"?")+"_="+new Date().getTime()}}}v.prototype.clone=function(){return new v(this,{body:this._bodyInit})};function V(e){var t=new FormData;return e.trim().split("&").forEach(function(a){if(a){var u=a.split("="),f=u.shift().replace(/\+/g," "),c=u.join("=").replace(/\+/g," ");t.append(decodeURIComponent(f),decodeURIComponent(c))}}),t}function J(e){var t=new d,a=e.replace(/\r?\n[\t ]+/g," ");return a.split("\r").map(function(u){return u.indexOf(`
2
- `)===0?u.substr(1,u.length):u}).forEach(function(u){var f=u.split(":"),c=f.shift().trim();if(c){var T=f.join(":").trim();t.append(c,T)}}),t}D.call(v.prototype);function m(e,t){if(!(this instanceof m))throw new TypeError('Please use the "new" operator, this DOM object constructor cannot be called as a function.');t||(t={}),this.type="default",this.status=t.status===void 0?200:t.status,this.ok=this.status>=200&&this.status<300,this.statusText=t.statusText===void 0?"":""+t.statusText,this.headers=new d(t.headers),this.url=t.url||"",this._initBody(e)}D.call(m.prototype),m.prototype.clone=function(){return new m(this._bodyInit,{status:this.status,statusText:this.statusText,headers:new d(this.headers),url:this.url})},m.error=function(){var e=new m(null,{status:0,statusText:""});return e.type="error",e};var Q=[301,302,303,307,308];m.redirect=function(e,t){if(Q.indexOf(t)===-1)throw new RangeError("Invalid status code");return new m(null,{status:t,headers:{location:e}})},n.DOMException=o.DOMException;try{new n.DOMException}catch{n.DOMException=function(t,a){this.message=t,this.name=a;var u=Error(t);this.stack=u.stack},n.DOMException.prototype=Object.create(Error.prototype),n.DOMException.prototype.constructor=n.DOMException}function A(e,t){return new Promise(function(a,u){var f=new v(e,t);if(f.signal&&f.signal.aborted)return u(new n.DOMException("Aborted","AbortError"));var c=new XMLHttpRequest;function T(){c.abort()}c.onload=function(){var p={status:c.status,statusText:c.statusText,headers:J(c.getAllResponseHeaders()||"")};p.url="responseURL"in c?c.responseURL:p.headers.get("X-Request-URL");var k="response"in c?c.response:c.responseText;setTimeout(function(){a(new m(k,p))},0)},c.onerror=function(){setTimeout(function(){u(new TypeError("Network request failed"))},0)},c.ontimeout=function(){setTimeout(function(){u(new TypeError("Network request failed"))},0)},c.onabort=function(){setTimeout(function(){u(new n.DOMException("Aborted","AbortError"))},0)};function $(p){try{return p===""&&o.location.href?o.location.href:p}catch{return p}}c.open(f.method,$(f.url),!0),f.credentials==="include"?c.withCredentials=!0:f.credentials==="omit"&&(c.withCredentials=!1),"responseType"in c&&(s.blob?c.responseType="blob":s.arrayBuffer&&f.headers.get("Content-Type")&&f.headers.get("Content-Type").indexOf("application/octet-stream")!==-1&&(c.responseType="arraybuffer")),t&&typeof t.headers=="object"&&!(t.headers instanceof d)?Object.getOwnPropertyNames(t.headers).forEach(function(p){c.setRequestHeader(p,S(t.headers[p]))}):f.headers.forEach(function(p,k){c.setRequestHeader(k,p)}),f.signal&&(f.signal.addEventListener("abort",T),c.onreadystatechange=function(){c.readyState===4&&f.signal.removeEventListener("abort",T)}),c.send(typeof f._bodyInit>"u"?null:f._bodyInit)})}return A.polyfill=!0,o.fetch||(o.fetch=A,o.Headers=d,o.Request=v,o.Response=m),n.Headers=d,n.Request=v,n.Response=m,n.fetch=A,n}({})})(typeof self<"u"?self:F)});var U,ne=new Uint8Array(16);function O(){if(!U&&(U=typeof crypto<"u"&&crypto.getRandomValues&&crypto.getRandomValues.bind(crypto),!U))throw new Error("crypto.getRandomValues() not supported. See https://github.com/uuidjs/uuid#getrandomvalues-not-supported");return U(ne)}var l=[];for(let i=0;i<256;++i)l.push((i+256).toString(16).slice(1));function j(i,r=0){return l[i[r+0]]+l[i[r+1]]+l[i[r+2]]+l[i[r+3]]+"-"+l[i[r+4]]+l[i[r+5]]+"-"+l[i[r+6]]+l[i[r+7]]+"-"+l[i[r+8]]+l[i[r+9]]+"-"+l[i[r+10]]+l[i[r+11]]+l[i[r+12]]+l[i[r+13]]+l[i[r+14]]+l[i[r+15]]}var oe=typeof crypto<"u"&&crypto.randomUUID&&crypto.randomUUID.bind(crypto),P={randomUUID:oe};function ie(i,r,n){if(P.randomUUID&&!r&&!i)return P.randomUUID();i=i||{};let o=i.random||(i.rng||O)();if(o[6]=o[6]&15|64,o[8]=o[8]&63|128,r){n=n||0;for(let s=0;s<16;++s)r[n+s]=o[s];return r}return j(o)}var x=ie;var ve=re(L()),K="schematicId";var B=class{apiKey;apiUrl="https://api.schematichq.com";webSocketUrl="wss://api.schematichq.com";eventUrl="https://c.schematichq.com";conn=null;context={};eventQueue;storage;useWebSocket=!1;values={};flagListener;constructor(r,n){this.apiKey=r,this.eventQueue=[],this.useWebSocket=n?.useWebSocket??!1,this.flagListener=n?.flagListener,n?.storage?this.storage=n.storage:typeof localStorage<"u"&&(this.storage=localStorage),n?.apiUrl!==void 0&&(this.apiUrl=n.apiUrl),n?.eventUrl!==void 0&&(this.eventUrl=n.eventUrl),n?.webSocketUrl!==void 0&&(this.webSocketUrl=n.webSocketUrl),window?.addEventListener&&window.addEventListener("beforeunload",()=>{this.flushEventQueue()})}async checkFlag(r){let{fallback:n=!1,key:o}=r,s=r.context||this.context;if(this.useWebSocket){let h=this.values[w(s)]??{};return typeof h[o]>"u"?n:h[o]}let y=`${this.apiUrl}/flags/${o}/check`;return fetch(y,{method:"POST",headers:{"X-Schematic-Api-Key":this.apiKey,"Content-Type":"application/json;charset=UTF-8"},body:JSON.stringify(s)}).then(h=>{if(!h.ok)throw new Error("Network response was not ok");return h.json()}).then(h=>h.data.value).catch(h=>(console.error("There was a problem with the fetch operation:",h),n))}checkFlags=async r=>{r=r||this.context;let n=`${this.apiUrl}/flags/check`,o=JSON.stringify(r);return fetch(n,{method:"POST",headers:{"Content-Type":"application/json;charset=UTF-8","X-Schematic-Api-Key":this.apiKey},body:o}).then(s=>{if(!s.ok)throw new Error("Network response was not ok");return s.json()}).then(s=>(s?.data?.flags??[]).reduce((y,h)=>(y[h.flag]=h.value,y),{})).catch(s=>(console.error("There was a problem with the fetch operation:",s),!1))};cleanup=async()=>{if(this.conn)try{(await this.conn).close()}catch(r){console.error("Error during cleanup:",r)}finally{this.conn=null}};identify=r=>(this.setContext({company:r.company?.keys,user:r.keys}),this.handleEvent("identify",r));setContext=async r=>this.useWebSocket?(this.conn||(this.conn=this.wsConnect()),this.conn.then(n=>this.wsSendMessage(n,r))):(this.context=r,Promise.resolve());track=r=>{let{company:n,user:o,event:s,traits:y}=r;return this.handleEvent("track",{company:n??this.context.company,event:s,traits:y??{},user:o??this.context.user})};flushEventQueue=()=>{for(;this.eventQueue.length>0;){let r=this.eventQueue.shift();r&&this.sendEvent(r)}};getAnonymousId=()=>{if(!this.storage)return x();let r=this.storage.getItem(K);if(typeof r<"u")return r;let n=x();return this.storage.setItem(K,n),n};handleEvent=(r,n)=>{let o={api_key:this.apiKey,body:n,sent_at:new Date().toISOString(),tracker_event_id:x(),tracker_user_id:this.getAnonymousId(),type:r};return document?.hidden?this.storeEvent(o):this.sendEvent(o)};sendEvent=async r=>{let n=`${this.eventUrl}/e`,o=JSON.stringify(r);try{await fetch(n,{method:"POST",headers:{"Content-Type":"application/json;charset=UTF-8"},body:o})}catch(s){console.error("Error sending Schematic event: ",s)}return Promise.resolve()};storeEvent=r=>(this.eventQueue.push(r),Promise.resolve());wsConnect=()=>new Promise((r,n)=>{let o=`${this.webSocketUrl}/flags/bootstrap`,s=new WebSocket(o);s.onopen=()=>{r(s)},s.onerror=y=>{n(y)},s.onclose=()=>{this.conn=null}});wsSendMessage=(r,n)=>new Promise((o,s)=>{if(w(n)==w(this.context)){o();return}this.context=n;let y=()=>{let h=!1,b=g=>{let S=JSON.parse(g.data);w(n)in this.values||(this.values[w(n)]={}),(S.flags??[]).forEach(E=>{this.values[w(n)][E.flag]=E.value}),this.flagListener&&this.flagListener(this.values[w(n)]),h||(h=!0,o()),r.removeEventListener("message",b)};r.addEventListener("message",b),r.send(JSON.stringify({apiKey:this.apiKey,data:n}))};r.readyState===WebSocket.OPEN?y():r.readyState===WebSocket.CONNECTING?r.addEventListener("open",y):s("WebSocket is not open or connecting")})};function w(i){let r=Object.keys(i).reduce((n,o)=>{let y=Object.keys(i[o]||{}).sort().reduce((h,b)=>(h[b]=i[o][b],h),{});return n[o]=y,n},{});return JSON.stringify(r)}window.Schematic=B;})();
1
+ "use strict";(()=>{var X=Object.create;var R=Object.defineProperty;var z=Object.getOwnPropertyDescriptor;var Y=Object.getOwnPropertyNames;var Z=Object.getPrototypeOf,ee=Object.prototype.hasOwnProperty;var te=(s,t)=>()=>(t||s((t={exports:{}}).exports,t),t.exports);var re=(s,t,n,i)=>{if(t&&typeof t=="object"||typeof t=="function")for(let o of Y(t))!ee.call(s,o)&&o!==n&&R(s,o,{get:()=>t[o],enumerable:!(i=z(t,o))||i.enumerable});return s};var ne=(s,t,n)=>(n=s!=null?X(Z(s)):{},re(t||!s||!s.__esModule?R(n,"default",{value:s,enumerable:!0}):n,s));var V=te(j=>{(function(s){var t=function(n){var i=typeof globalThis<"u"&&globalThis||typeof s<"u"&&s||typeof i<"u"&&i,o={searchParams:"URLSearchParams"in i,iterable:"Symbol"in i&&"iterator"in Symbol,blob:"FileReader"in i&&"Blob"in i&&function(){try{return new Blob,!0}catch{return!1}}(),formData:"FormData"in i,arrayBuffer:"ArrayBuffer"in i};function y(e){return e&&DataView.prototype.isPrototypeOf(e)}if(o.arrayBuffer)var h=["[object Int8Array]","[object Uint8Array]","[object Uint8ClampedArray]","[object Int16Array]","[object Uint16Array]","[object Int32Array]","[object Uint32Array]","[object Float32Array]","[object Float64Array]"],E=ArrayBuffer.isView||function(e){return e&&h.indexOf(Object.prototype.toString.call(e))>-1};function b(e){if(typeof e!="string"&&(e=String(e)),/[^a-z0-9\-#$%&'*+.^_`|~!]/i.test(e)||e==="")throw new TypeError('Invalid character in header field name: "'+e+'"');return e.toLowerCase()}function S(e){return typeof e!="string"&&(e=String(e)),e}function w(e){var r={next:function(){var a=e.shift();return{done:a===void 0,value:a}}};return o.iterable&&(r[Symbol.iterator]=function(){return r}),r}function d(e){this.map={},e instanceof d?e.forEach(function(r,a){this.append(a,r)},this):Array.isArray(e)?e.forEach(function(r){this.append(r[0],r[1])},this):e&&Object.getOwnPropertyNames(e).forEach(function(r){this.append(r,e[r])},this)}d.prototype.append=function(e,r){e=b(e),r=S(r);var a=this.map[e];this.map[e]=a?a+", "+r:r},d.prototype.delete=function(e){delete this.map[b(e)]},d.prototype.get=function(e){return e=b(e),this.has(e)?this.map[e]:null},d.prototype.has=function(e){return this.map.hasOwnProperty(b(e))},d.prototype.set=function(e,r){this.map[b(e)]=S(r)},d.prototype.forEach=function(e,r){for(var a in this.map)this.map.hasOwnProperty(a)&&e.call(r,this.map[a],a,this)},d.prototype.keys=function(){var e=[];return this.forEach(function(r,a){e.push(a)}),w(e)},d.prototype.values=function(){var e=[];return this.forEach(function(r){e.push(r)}),w(e)},d.prototype.entries=function(){var e=[];return this.forEach(function(r,a){e.push([a,r])}),w(e)},o.iterable&&(d.prototype[Symbol.iterator]=d.prototype.entries);function O(e){if(e.bodyUsed)return Promise.reject(new TypeError("Already read"));e.bodyUsed=!0}function D(e){return new Promise(function(r,a){e.onload=function(){r(e.result)},e.onerror=function(){a(e.error)}})}function M(e){var r=new FileReader,a=D(r);return r.readAsArrayBuffer(e),a}function q(e){var r=new FileReader,a=D(r);return r.readAsText(e),a}function H(e){for(var r=new Uint8Array(e),a=new Array(r.length),u=0;u<r.length;u++)a[u]=String.fromCharCode(r[u]);return a.join("")}function C(e){if(e.slice)return e.slice(0);var r=new Uint8Array(e.byteLength);return r.set(new Uint8Array(e)),r.buffer}function L(){return this.bodyUsed=!1,this._initBody=function(e){this.bodyUsed=this.bodyUsed,this._bodyInit=e,e?typeof e=="string"?this._bodyText=e:o.blob&&Blob.prototype.isPrototypeOf(e)?this._bodyBlob=e:o.formData&&FormData.prototype.isPrototypeOf(e)?this._bodyFormData=e:o.searchParams&&URLSearchParams.prototype.isPrototypeOf(e)?this._bodyText=e.toString():o.arrayBuffer&&o.blob&&y(e)?(this._bodyArrayBuffer=C(e.buffer),this._bodyInit=new Blob([this._bodyArrayBuffer])):o.arrayBuffer&&(ArrayBuffer.prototype.isPrototypeOf(e)||E(e))?this._bodyArrayBuffer=C(e):this._bodyText=e=Object.prototype.toString.call(e):this._bodyText="",this.headers.get("content-type")||(typeof e=="string"?this.headers.set("content-type","text/plain;charset=UTF-8"):this._bodyBlob&&this._bodyBlob.type?this.headers.set("content-type",this._bodyBlob.type):o.searchParams&&URLSearchParams.prototype.isPrototypeOf(e)&&this.headers.set("content-type","application/x-www-form-urlencoded;charset=UTF-8"))},o.blob&&(this.blob=function(){var e=O(this);if(e)return e;if(this._bodyBlob)return Promise.resolve(this._bodyBlob);if(this._bodyArrayBuffer)return Promise.resolve(new Blob([this._bodyArrayBuffer]));if(this._bodyFormData)throw new Error("could not read FormData body as blob");return Promise.resolve(new Blob([this._bodyText]))},this.arrayBuffer=function(){if(this._bodyArrayBuffer){var e=O(this);return e||(ArrayBuffer.isView(this._bodyArrayBuffer)?Promise.resolve(this._bodyArrayBuffer.buffer.slice(this._bodyArrayBuffer.byteOffset,this._bodyArrayBuffer.byteOffset+this._bodyArrayBuffer.byteLength)):Promise.resolve(this._bodyArrayBuffer))}else return this.blob().then(M)}),this.text=function(){var e=O(this);if(e)return e;if(this._bodyBlob)return q(this._bodyBlob);if(this._bodyArrayBuffer)return Promise.resolve(H(this._bodyArrayBuffer));if(this._bodyFormData)throw new Error("could not read FormData body as text");return Promise.resolve(this._bodyText)},o.formData&&(this.formData=function(){return this.text().then(J)}),this.json=function(){return this.text().then(JSON.parse)},this}var W=["DELETE","GET","HEAD","OPTIONS","POST","PUT"];function K(e){var r=e.toUpperCase();return W.indexOf(r)>-1?r:e}function v(e,r){if(!(this instanceof v))throw new TypeError('Please use the "new" operator, this DOM object constructor cannot be called as a function.');r=r||{};var a=r.body;if(e instanceof v){if(e.bodyUsed)throw new TypeError("Already read");this.url=e.url,this.credentials=e.credentials,r.headers||(this.headers=new d(e.headers)),this.method=e.method,this.mode=e.mode,this.signal=e.signal,!a&&e._bodyInit!=null&&(a=e._bodyInit,e.bodyUsed=!0)}else this.url=String(e);if(this.credentials=r.credentials||this.credentials||"same-origin",(r.headers||!this.headers)&&(this.headers=new d(r.headers)),this.method=K(r.method||this.method||"GET"),this.mode=r.mode||this.mode||null,this.signal=r.signal||this.signal,this.referrer=null,(this.method==="GET"||this.method==="HEAD")&&a)throw new TypeError("Body not allowed for GET or HEAD requests");if(this._initBody(a),(this.method==="GET"||this.method==="HEAD")&&(r.cache==="no-store"||r.cache==="no-cache")){var u=/([?&])_=[^&]*/;if(u.test(this.url))this.url=this.url.replace(u,"$1_="+new Date().getTime());else{var f=/\?/;this.url+=(f.test(this.url)?"&":"?")+"_="+new Date().getTime()}}}v.prototype.clone=function(){return new v(this,{body:this._bodyInit})};function J(e){var r=new FormData;return e.trim().split("&").forEach(function(a){if(a){var u=a.split("="),f=u.shift().replace(/\+/g," "),c=u.join("=").replace(/\+/g," ");r.append(decodeURIComponent(f),decodeURIComponent(c))}}),r}function Q(e){var r=new d,a=e.replace(/\r?\n[\t ]+/g," ");return a.split("\r").map(function(u){return u.indexOf(`
2
+ `)===0?u.substr(1,u.length):u}).forEach(function(u){var f=u.split(":"),c=f.shift().trim();if(c){var T=f.join(":").trim();r.append(c,T)}}),r}L.call(v.prototype);function m(e,r){if(!(this instanceof m))throw new TypeError('Please use the "new" operator, this DOM object constructor cannot be called as a function.');r||(r={}),this.type="default",this.status=r.status===void 0?200:r.status,this.ok=this.status>=200&&this.status<300,this.statusText=r.statusText===void 0?"":""+r.statusText,this.headers=new d(r.headers),this.url=r.url||"",this._initBody(e)}L.call(m.prototype),m.prototype.clone=function(){return new m(this._bodyInit,{status:this.status,statusText:this.statusText,headers:new d(this.headers),url:this.url})},m.error=function(){var e=new m(null,{status:0,statusText:""});return e.type="error",e};var $=[301,302,303,307,308];m.redirect=function(e,r){if($.indexOf(r)===-1)throw new RangeError("Invalid status code");return new m(null,{status:r,headers:{location:e}})},n.DOMException=i.DOMException;try{new n.DOMException}catch{n.DOMException=function(r,a){this.message=r,this.name=a;var u=Error(r);this.stack=u.stack},n.DOMException.prototype=Object.create(Error.prototype),n.DOMException.prototype.constructor=n.DOMException}function k(e,r){return new Promise(function(a,u){var f=new v(e,r);if(f.signal&&f.signal.aborted)return u(new n.DOMException("Aborted","AbortError"));var c=new XMLHttpRequest;function T(){c.abort()}c.onload=function(){var p={status:c.status,statusText:c.statusText,headers:Q(c.getAllResponseHeaders()||"")};p.url="responseURL"in c?c.responseURL:p.headers.get("X-Request-URL");var P="response"in c?c.response:c.responseText;setTimeout(function(){a(new m(P,p))},0)},c.onerror=function(){setTimeout(function(){u(new TypeError("Network request failed"))},0)},c.ontimeout=function(){setTimeout(function(){u(new TypeError("Network request failed"))},0)},c.onabort=function(){setTimeout(function(){u(new n.DOMException("Aborted","AbortError"))},0)};function G(p){try{return p===""&&i.location.href?i.location.href:p}catch{return p}}c.open(f.method,G(f.url),!0),f.credentials==="include"?c.withCredentials=!0:f.credentials==="omit"&&(c.withCredentials=!1),"responseType"in c&&(o.blob?c.responseType="blob":o.arrayBuffer&&f.headers.get("Content-Type")&&f.headers.get("Content-Type").indexOf("application/octet-stream")!==-1&&(c.responseType="arraybuffer")),r&&typeof r.headers=="object"&&!(r.headers instanceof d)?Object.getOwnPropertyNames(r.headers).forEach(function(p){c.setRequestHeader(p,S(r.headers[p]))}):f.headers.forEach(function(p,P){c.setRequestHeader(P,p)}),f.signal&&(f.signal.addEventListener("abort",T),c.onreadystatechange=function(){c.readyState===4&&f.signal.removeEventListener("abort",T)}),c.send(typeof f._bodyInit>"u"?null:f._bodyInit)})}return k.polyfill=!0,i.fetch||(i.fetch=k,i.Headers=d,i.Request=v,i.Response=m),n.Headers=d,n.Request=v,n.Response=m,n.fetch=k,n}({})})(typeof self<"u"?self:j)});var l=[];for(U=0;U<256;++U)l.push((U+256).toString(16).slice(1));var U;function F(s,t=0){return(l[s[t+0]]+l[s[t+1]]+l[s[t+2]]+l[s[t+3]]+"-"+l[s[t+4]]+l[s[t+5]]+"-"+l[s[t+6]]+l[s[t+7]]+"-"+l[s[t+8]]+l[s[t+9]]+"-"+l[s[t+10]]+l[s[t+11]]+l[s[t+12]]+l[s[t+13]]+l[s[t+14]]+l[s[t+15]]).toLowerCase()}var A,ie=new Uint8Array(16);function _(){if(!A&&(A=typeof crypto<"u"&&crypto.getRandomValues&&crypto.getRandomValues.bind(crypto),!A))throw new Error("crypto.getRandomValues() not supported. See https://github.com/uuidjs/uuid#getrandomvalues-not-supported");return A(ie)}var oe=typeof crypto<"u"&&crypto.randomUUID&&crypto.randomUUID.bind(crypto),I={randomUUID:oe};function se(s,t,n){if(I.randomUUID&&!t&&!s)return I.randomUUID();s=s||{};var i=s.random||(s.rng||_)();if(i[6]=i[6]&15|64,i[8]=i[8]&63|128,t){n=n||0;for(var o=0;o<16;++o)t[n+o]=i[o];return t}return F(i)}var x=se;var be=ne(V());function g(s){let t=Object.keys(s).reduce((n,i)=>{let y=Object.keys(s[i]||{}).sort().reduce((h,E)=>(h[E]=s[i][E],h),{});return n[i]=y,n},{});return JSON.stringify(t)}var N="schematicId";var B=class{apiKey;apiUrl="https://api.schematichq.com";conn=null;context={};eventQueue;eventUrl="https://c.schematichq.com";flagListener;flagValueListeners={};isPending=!0;isPendingListeners=new Set;storage;useWebSocket=!1;values={};webSocketUrl="wss://api.schematichq.com";constructor(t,n){this.apiKey=t,this.eventQueue=[],this.useWebSocket=n?.useWebSocket??!1,this.flagListener=n?.flagListener,n?.storage?this.storage=n.storage:typeof localStorage<"u"&&(this.storage=localStorage),n?.apiUrl!==void 0&&(this.apiUrl=n.apiUrl),n?.eventUrl!==void 0&&(this.eventUrl=n.eventUrl),n?.webSocketUrl!==void 0&&(this.webSocketUrl=n.webSocketUrl),typeof window<"u"&&window?.addEventListener&&window.addEventListener("beforeunload",()=>{this.flushEventQueue()})}async checkFlag(t){let{fallback:n=!1,key:i}=t,o=t.context||this.context;if(this.useWebSocket){let h=this.values[g(o)]??{};return typeof h[i]>"u"?n:h[i]}let y=`${this.apiUrl}/flags/${i}/check`;return fetch(y,{method:"POST",headers:{"X-Schematic-Api-Key":this.apiKey,"Content-Type":"application/json;charset=UTF-8"},body:JSON.stringify(o)}).then(h=>{if(!h.ok)throw new Error("Network response was not ok");return h.json()}).then(h=>h.data.value).catch(h=>(console.error("There was a problem with the fetch operation:",h),n))}checkFlags=async t=>{t=t||this.context;let n=`${this.apiUrl}/flags/check`,i=JSON.stringify(t);return fetch(n,{method:"POST",headers:{"Content-Type":"application/json;charset=UTF-8","X-Schematic-Api-Key":this.apiKey},body:i}).then(o=>{if(!o.ok)throw new Error("Network response was not ok");return o.json()}).then(o=>(o?.data?.flags??[]).reduce((y,h)=>(y[h.flag]=h.value,y),{})).catch(o=>(console.error("There was a problem with the fetch operation:",o),!1))};identify=t=>(this.setContext({company:t.company?.keys,user:t.keys}),this.handleEvent("identify",t));setContext=async t=>{if(!this.useWebSocket)return this.context=t,Promise.resolve();try{this.setIsPending(!0),this.conn||(this.conn=this.wsConnect());let n=await this.conn;await this.wsSendMessage(n,t)}catch(n){console.error("Error setting Schematic context:",n)}};track=t=>{let{company:n,user:i,event:o,traits:y}=t;return this.handleEvent("track",{company:n??this.context.company,event:o,traits:y??{},user:i??this.context.user})};flushEventQueue=()=>{for(;this.eventQueue.length>0;){let t=this.eventQueue.shift();t&&this.sendEvent(t)}};getAnonymousId=()=>{if(!this.storage)return x();let t=this.storage.getItem(N);if(typeof t<"u")return t;let n=x();return this.storage.setItem(N,n),n};handleEvent=(t,n)=>{let i={api_key:this.apiKey,body:n,sent_at:new Date().toISOString(),tracker_event_id:x(),tracker_user_id:this.getAnonymousId(),type:t};return document?.hidden?this.storeEvent(i):this.sendEvent(i)};sendEvent=async t=>{let n=`${this.eventUrl}/e`,i=JSON.stringify(t);try{await fetch(n,{method:"POST",headers:{"Content-Type":"application/json;charset=UTF-8"},body:i})}catch(o){console.error("Error sending Schematic event: ",o)}return Promise.resolve()};storeEvent=t=>(this.eventQueue.push(t),Promise.resolve());cleanup=async()=>{if(this.conn)try{(await this.conn).close()}catch(t){console.error("Error during cleanup:",t)}finally{this.conn=null}};wsConnect=()=>new Promise((t,n)=>{let i=`${this.webSocketUrl}/flags/bootstrap`,o=new WebSocket(i);o.onopen=()=>{t(o)},o.onerror=y=>{n(y)},o.onclose=()=>{this.conn=null}});wsSendMessage=(t,n)=>new Promise((i,o)=>{if(g(n)==g(this.context)){i();return}this.context=n;let y=()=>{let h=!1,E=b=>{let S=JSON.parse(b.data);g(n)in this.values||(this.values[g(n)]={}),(S.flags??[]).forEach(w=>{this.values[g(n)][w.flag]=w.value,this.notifyFlagValueListeners(w.flag)}),this.flagListener&&this.flagListener(this.getFlagValues()),this.setIsPending(!1),h||(h=!0,i())};t.addEventListener("message",E),t.send(JSON.stringify({apiKey:this.apiKey,data:n}))};t.readyState===WebSocket.OPEN?y():t.readyState===WebSocket.CONNECTING?t.addEventListener("open",y):o("WebSocket is not open or connecting")});getIsPending=()=>this.isPending;addIsPendingListener=t=>(this.isPendingListeners.add(t),()=>{this.isPendingListeners.delete(t)});setIsPending=t=>{this.isPending=t,this.isPendingListeners.forEach(n=>n())};getFlagValue=t=>this.getFlagValues()[t];getFlagValues=()=>{let t=g(this.context);return this.values[t]??{}};addFlagValueListener=(t,n)=>(t in this.flagValueListeners||(this.flagValueListeners[t]=new Set),this.flagValueListeners[t].add(n),()=>{this.flagValueListeners[t].delete(n)});notifyFlagValueListeners=t=>{(this.flagValueListeners?.[t]??[]).forEach(i=>i())}};window.Schematic=B;})();
3
3
  /* @preserve */
@@ -539,6 +539,16 @@ __export(src_exports, {
539
539
  });
540
540
  module.exports = __toCommonJS(src_exports);
541
541
 
542
+ // node_modules/uuid/dist/esm-browser/stringify.js
543
+ var byteToHex = [];
544
+ for (i = 0; i < 256; ++i) {
545
+ byteToHex.push((i + 256).toString(16).slice(1));
546
+ }
547
+ var i;
548
+ function unsafeStringify(arr, offset = 0) {
549
+ return (byteToHex[arr[offset + 0]] + byteToHex[arr[offset + 1]] + byteToHex[arr[offset + 2]] + byteToHex[arr[offset + 3]] + "-" + byteToHex[arr[offset + 4]] + byteToHex[arr[offset + 5]] + "-" + byteToHex[arr[offset + 6]] + byteToHex[arr[offset + 7]] + "-" + byteToHex[arr[offset + 8]] + byteToHex[arr[offset + 9]] + "-" + byteToHex[arr[offset + 10]] + byteToHex[arr[offset + 11]] + byteToHex[arr[offset + 12]] + byteToHex[arr[offset + 13]] + byteToHex[arr[offset + 14]] + byteToHex[arr[offset + 15]]).toLowerCase();
550
+ }
551
+
542
552
  // node_modules/uuid/dist/esm-browser/rng.js
543
553
  var getRandomValues;
544
554
  var rnds8 = new Uint8Array(16);
@@ -552,15 +562,6 @@ function rng() {
552
562
  return getRandomValues(rnds8);
553
563
  }
554
564
 
555
- // node_modules/uuid/dist/esm-browser/stringify.js
556
- var byteToHex = [];
557
- for (let i = 0; i < 256; ++i) {
558
- byteToHex.push((i + 256).toString(16).slice(1));
559
- }
560
- function unsafeStringify(arr, offset = 0) {
561
- return byteToHex[arr[offset + 0]] + byteToHex[arr[offset + 1]] + byteToHex[arr[offset + 2]] + byteToHex[arr[offset + 3]] + "-" + byteToHex[arr[offset + 4]] + byteToHex[arr[offset + 5]] + "-" + byteToHex[arr[offset + 6]] + byteToHex[arr[offset + 7]] + "-" + byteToHex[arr[offset + 8]] + byteToHex[arr[offset + 9]] + "-" + byteToHex[arr[offset + 10]] + byteToHex[arr[offset + 11]] + byteToHex[arr[offset + 12]] + byteToHex[arr[offset + 13]] + byteToHex[arr[offset + 14]] + byteToHex[arr[offset + 15]];
562
- }
563
-
564
565
  // node_modules/uuid/dist/esm-browser/native.js
565
566
  var randomUUID = typeof crypto !== "undefined" && crypto.randomUUID && crypto.randomUUID.bind(crypto);
566
567
  var native_default = {
@@ -573,12 +574,12 @@ function v4(options, buf, offset) {
573
574
  return native_default.randomUUID();
574
575
  }
575
576
  options = options || {};
576
- const rnds = options.random || (options.rng || rng)();
577
+ var rnds = options.random || (options.rng || rng)();
577
578
  rnds[6] = rnds[6] & 15 | 64;
578
579
  rnds[8] = rnds[8] & 63 | 128;
579
580
  if (buf) {
580
581
  offset = offset || 0;
581
- for (let i = 0; i < 16; ++i) {
582
+ for (var i = 0; i < 16; ++i) {
582
583
  buf[offset + i] = rnds[i];
583
584
  }
584
585
  return buf;
@@ -589,19 +590,40 @@ var v4_default = v4;
589
590
 
590
591
  // src/index.ts
591
592
  var import_polyfill = __toESM(require_browser_polyfill());
593
+
594
+ // src/utils.ts
595
+ function contextString(context) {
596
+ const sortedContext = Object.keys(context).reduce((acc, key) => {
597
+ const sortedKeys = Object.keys(
598
+ context[key] || {}
599
+ ).sort();
600
+ const sortedObj = sortedKeys.reduce((obj, sortedKey) => {
601
+ obj[sortedKey] = context[key][sortedKey];
602
+ return obj;
603
+ }, {});
604
+ acc[key] = sortedObj;
605
+ return acc;
606
+ }, {});
607
+ return JSON.stringify(sortedContext);
608
+ }
609
+
610
+ // src/index.ts
592
611
  var anonymousIdKey = "schematicId";
593
612
  var Schematic = class {
594
613
  apiKey;
595
614
  apiUrl = "https://api.schematichq.com";
596
- webSocketUrl = "wss://api.schematichq.com";
597
- eventUrl = "https://c.schematichq.com";
598
615
  conn = null;
599
616
  context = {};
600
617
  eventQueue;
618
+ eventUrl = "https://c.schematichq.com";
619
+ flagListener;
620
+ flagValueListeners = {};
621
+ isPending = true;
622
+ isPendingListeners = /* @__PURE__ */ new Set();
601
623
  storage;
602
624
  useWebSocket = false;
603
625
  values = {};
604
- flagListener;
626
+ webSocketUrl = "wss://api.schematichq.com";
605
627
  constructor(apiKey, options) {
606
628
  this.apiKey = apiKey;
607
629
  this.eventQueue = [];
@@ -621,12 +643,14 @@ var Schematic = class {
621
643
  if (options?.webSocketUrl !== void 0) {
622
644
  this.webSocketUrl = options.webSocketUrl;
623
645
  }
624
- if (window?.addEventListener) {
646
+ if (typeof window !== "undefined" && window?.addEventListener) {
625
647
  window.addEventListener("beforeunload", () => {
626
648
  this.flushEventQueue();
627
649
  });
628
650
  }
629
651
  }
652
+ // Get value for a single flag
653
+ // If in websocket mode, return the local value, otherwise make an API call
630
654
  async checkFlag(options) {
631
655
  const { fallback = false, key } = options;
632
656
  const context = options.context || this.context;
@@ -654,7 +678,7 @@ var Schematic = class {
654
678
  return fallback;
655
679
  });
656
680
  }
657
- // Make a REST API call to fetch all flag values for a given context
681
+ // Make an API call to fetch all flag values for a given context (use if not in websocket mode)
658
682
  checkFlags = async (context) => {
659
683
  context = context || this.context;
660
684
  const requestUrl = `${this.apiUrl}/flags/check`;
@@ -684,18 +708,6 @@ var Schematic = class {
684
708
  return false;
685
709
  });
686
710
  };
687
- cleanup = async () => {
688
- if (this.conn) {
689
- try {
690
- const socket = await this.conn;
691
- socket.close();
692
- } catch (error) {
693
- console.error("Error during cleanup:", error);
694
- } finally {
695
- this.conn = null;
696
- }
697
- }
698
- };
699
711
  // Send an identify event
700
712
  identify = (body) => {
701
713
  this.setContext({
@@ -713,12 +725,16 @@ var Schematic = class {
713
725
  this.context = context;
714
726
  return Promise.resolve();
715
727
  }
716
- if (!this.conn) {
717
- this.conn = this.wsConnect();
728
+ try {
729
+ this.setIsPending(true);
730
+ if (!this.conn) {
731
+ this.conn = this.wsConnect();
732
+ }
733
+ const socket = await this.conn;
734
+ await this.wsSendMessage(socket, context);
735
+ } catch (error) {
736
+ console.error("Error setting Schematic context:", error);
718
737
  }
719
- return this.conn.then((socket) => {
720
- return this.wsSendMessage(socket, context);
721
- });
722
738
  };
723
739
  // Send track event
724
740
  track = (body) => {
@@ -730,6 +746,9 @@ var Schematic = class {
730
746
  user: user ?? this.context.user
731
747
  });
732
748
  };
749
+ /**
750
+ * Event processing
751
+ */
733
752
  flushEventQueue = () => {
734
753
  while (this.eventQueue.length > 0) {
735
754
  const event = this.eventQueue.shift();
@@ -785,6 +804,22 @@ var Schematic = class {
785
804
  this.eventQueue.push(event);
786
805
  return Promise.resolve();
787
806
  };
807
+ /**
808
+ * Websocket management
809
+ */
810
+ cleanup = async () => {
811
+ if (this.conn) {
812
+ try {
813
+ const socket = await this.conn;
814
+ socket.close();
815
+ } catch (error) {
816
+ console.error("Error during cleanup:", error);
817
+ } finally {
818
+ this.conn = null;
819
+ }
820
+ }
821
+ };
822
+ // Open a websocket connection
788
823
  wsConnect = () => {
789
824
  return new Promise((resolve, reject) => {
790
825
  const wsUrl = `${this.webSocketUrl}/flags/bootstrap`;
@@ -800,6 +835,8 @@ var Schematic = class {
800
835
  };
801
836
  });
802
837
  };
838
+ // Send a message on the websocket indicating interest in a particular evaluation context
839
+ // and wait for the initial set of flag values to be returned
803
840
  wsSendMessage = (socket, context) => {
804
841
  return new Promise((resolve, reject) => {
805
842
  if (contextString(context) == contextString(this.context)) {
@@ -817,16 +854,17 @@ var Schematic = class {
817
854
  (message.flags ?? []).forEach(
818
855
  (flag) => {
819
856
  this.values[contextString(context)][flag.flag] = flag.value;
857
+ this.notifyFlagValueListeners(flag.flag);
820
858
  }
821
859
  );
822
860
  if (this.flagListener) {
823
- this.flagListener(this.values[contextString(context)]);
861
+ this.flagListener(this.getFlagValues());
824
862
  }
863
+ this.setIsPending(false);
825
864
  if (!resolved) {
826
865
  resolved = true;
827
866
  resolve();
828
867
  }
829
- socket.removeEventListener("message", messageHandler);
830
868
  };
831
869
  socket.addEventListener("message", messageHandler);
832
870
  socket.send(
@@ -845,19 +883,44 @@ var Schematic = class {
845
883
  }
846
884
  });
847
885
  };
886
+ /**
887
+ * State management
888
+ */
889
+ // isPending state
890
+ getIsPending = () => {
891
+ return this.isPending;
892
+ };
893
+ addIsPendingListener = (listener) => {
894
+ this.isPendingListeners.add(listener);
895
+ return () => {
896
+ this.isPendingListeners.delete(listener);
897
+ };
898
+ };
899
+ setIsPending = (isPending) => {
900
+ this.isPending = isPending;
901
+ this.isPendingListeners.forEach((listener) => listener());
902
+ };
903
+ // flagValues state
904
+ getFlagValue = (flagKey) => {
905
+ const values = this.getFlagValues();
906
+ return values[flagKey];
907
+ };
908
+ getFlagValues = () => {
909
+ const contextStr = contextString(this.context);
910
+ return this.values[contextStr] ?? {};
911
+ };
912
+ addFlagValueListener = (flagKey, listener) => {
913
+ if (!(flagKey in this.flagValueListeners)) {
914
+ this.flagValueListeners[flagKey] = /* @__PURE__ */ new Set();
915
+ }
916
+ this.flagValueListeners[flagKey].add(listener);
917
+ return () => {
918
+ this.flagValueListeners[flagKey].delete(listener);
919
+ };
920
+ };
921
+ notifyFlagValueListeners = (flagKey) => {
922
+ const listeners = this.flagValueListeners?.[flagKey] ?? [];
923
+ listeners.forEach((listener) => listener());
924
+ };
848
925
  };
849
- function contextString(context) {
850
- const sortedContext = Object.keys(context).reduce((acc, key) => {
851
- const sortedKeys = Object.keys(
852
- context[key] || {}
853
- ).sort();
854
- const sortedObj = sortedKeys.reduce((obj, sortedKey) => {
855
- obj[sortedKey] = context[key][sortedKey];
856
- return obj;
857
- }, {});
858
- acc[key] = sortedObj;
859
- return acc;
860
- }, {});
861
- return JSON.stringify(sortedContext);
862
- }
863
926
  /* @preserve */
@@ -52,29 +52,48 @@ export declare type Keys = Record<string, string>;
52
52
  export declare class Schematic {
53
53
  private apiKey;
54
54
  private apiUrl;
55
- private webSocketUrl;
56
- private eventUrl;
57
55
  private conn;
58
56
  private context;
59
57
  private eventQueue;
58
+ private eventUrl;
59
+ private flagListener?;
60
+ private flagValueListeners;
61
+ private isPending;
62
+ private isPendingListeners;
60
63
  private storage;
61
64
  private useWebSocket;
62
65
  private values;
63
- private flagListener?;
66
+ private webSocketUrl;
64
67
  constructor(apiKey: string, options?: SchematicOptions);
65
68
  checkFlag(options: CheckOptions): Promise<boolean>;
66
69
  checkFlags: (context?: SchematicContext) => Promise<Record<string, boolean>>;
67
- cleanup: () => Promise<void>;
68
70
  identify: (body: EventBodyIdentify) => Promise<void>;
69
71
  setContext: (context: SchematicContext) => Promise<void>;
70
72
  track: (body: EventBodyTrack) => Promise<void>;
73
+ /**
74
+ * Event processing
75
+ */
71
76
  private flushEventQueue;
72
77
  private getAnonymousId;
73
78
  private handleEvent;
74
79
  private sendEvent;
75
80
  private storeEvent;
81
+ /**
82
+ * Websocket management
83
+ */
84
+ cleanup: () => Promise<void>;
76
85
  private wsConnect;
77
86
  private wsSendMessage;
87
+ /**
88
+ * State management
89
+ */
90
+ getIsPending: () => boolean;
91
+ addIsPendingListener: (listener: () => void) => () => void;
92
+ private setIsPending;
93
+ getFlagValue: (flagKey: string) => boolean;
94
+ getFlagValues: () => Record<string, boolean>;
95
+ addFlagValueListener: (flagKey: string, listener: () => void) => () => void;
96
+ private notifyFlagValueListeners;
78
97
  }
79
98
 
80
99
  export declare type SchematicContext = {
@@ -526,6 +526,16 @@ var require_browser_polyfill = __commonJS({
526
526
  }
527
527
  });
528
528
 
529
+ // node_modules/uuid/dist/esm-browser/stringify.js
530
+ var byteToHex = [];
531
+ for (i = 0; i < 256; ++i) {
532
+ byteToHex.push((i + 256).toString(16).slice(1));
533
+ }
534
+ var i;
535
+ function unsafeStringify(arr, offset = 0) {
536
+ return (byteToHex[arr[offset + 0]] + byteToHex[arr[offset + 1]] + byteToHex[arr[offset + 2]] + byteToHex[arr[offset + 3]] + "-" + byteToHex[arr[offset + 4]] + byteToHex[arr[offset + 5]] + "-" + byteToHex[arr[offset + 6]] + byteToHex[arr[offset + 7]] + "-" + byteToHex[arr[offset + 8]] + byteToHex[arr[offset + 9]] + "-" + byteToHex[arr[offset + 10]] + byteToHex[arr[offset + 11]] + byteToHex[arr[offset + 12]] + byteToHex[arr[offset + 13]] + byteToHex[arr[offset + 14]] + byteToHex[arr[offset + 15]]).toLowerCase();
537
+ }
538
+
529
539
  // node_modules/uuid/dist/esm-browser/rng.js
530
540
  var getRandomValues;
531
541
  var rnds8 = new Uint8Array(16);
@@ -539,15 +549,6 @@ function rng() {
539
549
  return getRandomValues(rnds8);
540
550
  }
541
551
 
542
- // node_modules/uuid/dist/esm-browser/stringify.js
543
- var byteToHex = [];
544
- for (let i = 0; i < 256; ++i) {
545
- byteToHex.push((i + 256).toString(16).slice(1));
546
- }
547
- function unsafeStringify(arr, offset = 0) {
548
- return byteToHex[arr[offset + 0]] + byteToHex[arr[offset + 1]] + byteToHex[arr[offset + 2]] + byteToHex[arr[offset + 3]] + "-" + byteToHex[arr[offset + 4]] + byteToHex[arr[offset + 5]] + "-" + byteToHex[arr[offset + 6]] + byteToHex[arr[offset + 7]] + "-" + byteToHex[arr[offset + 8]] + byteToHex[arr[offset + 9]] + "-" + byteToHex[arr[offset + 10]] + byteToHex[arr[offset + 11]] + byteToHex[arr[offset + 12]] + byteToHex[arr[offset + 13]] + byteToHex[arr[offset + 14]] + byteToHex[arr[offset + 15]];
549
- }
550
-
551
552
  // node_modules/uuid/dist/esm-browser/native.js
552
553
  var randomUUID = typeof crypto !== "undefined" && crypto.randomUUID && crypto.randomUUID.bind(crypto);
553
554
  var native_default = {
@@ -560,12 +561,12 @@ function v4(options, buf, offset) {
560
561
  return native_default.randomUUID();
561
562
  }
562
563
  options = options || {};
563
- const rnds = options.random || (options.rng || rng)();
564
+ var rnds = options.random || (options.rng || rng)();
564
565
  rnds[6] = rnds[6] & 15 | 64;
565
566
  rnds[8] = rnds[8] & 63 | 128;
566
567
  if (buf) {
567
568
  offset = offset || 0;
568
- for (let i = 0; i < 16; ++i) {
569
+ for (var i = 0; i < 16; ++i) {
569
570
  buf[offset + i] = rnds[i];
570
571
  }
571
572
  return buf;
@@ -576,19 +577,40 @@ var v4_default = v4;
576
577
 
577
578
  // src/index.ts
578
579
  var import_polyfill = __toESM(require_browser_polyfill());
580
+
581
+ // src/utils.ts
582
+ function contextString(context) {
583
+ const sortedContext = Object.keys(context).reduce((acc, key) => {
584
+ const sortedKeys = Object.keys(
585
+ context[key] || {}
586
+ ).sort();
587
+ const sortedObj = sortedKeys.reduce((obj, sortedKey) => {
588
+ obj[sortedKey] = context[key][sortedKey];
589
+ return obj;
590
+ }, {});
591
+ acc[key] = sortedObj;
592
+ return acc;
593
+ }, {});
594
+ return JSON.stringify(sortedContext);
595
+ }
596
+
597
+ // src/index.ts
579
598
  var anonymousIdKey = "schematicId";
580
599
  var Schematic = class {
581
600
  apiKey;
582
601
  apiUrl = "https://api.schematichq.com";
583
- webSocketUrl = "wss://api.schematichq.com";
584
- eventUrl = "https://c.schematichq.com";
585
602
  conn = null;
586
603
  context = {};
587
604
  eventQueue;
605
+ eventUrl = "https://c.schematichq.com";
606
+ flagListener;
607
+ flagValueListeners = {};
608
+ isPending = true;
609
+ isPendingListeners = /* @__PURE__ */ new Set();
588
610
  storage;
589
611
  useWebSocket = false;
590
612
  values = {};
591
- flagListener;
613
+ webSocketUrl = "wss://api.schematichq.com";
592
614
  constructor(apiKey, options) {
593
615
  this.apiKey = apiKey;
594
616
  this.eventQueue = [];
@@ -608,12 +630,14 @@ var Schematic = class {
608
630
  if (options?.webSocketUrl !== void 0) {
609
631
  this.webSocketUrl = options.webSocketUrl;
610
632
  }
611
- if (window?.addEventListener) {
633
+ if (typeof window !== "undefined" && window?.addEventListener) {
612
634
  window.addEventListener("beforeunload", () => {
613
635
  this.flushEventQueue();
614
636
  });
615
637
  }
616
638
  }
639
+ // Get value for a single flag
640
+ // If in websocket mode, return the local value, otherwise make an API call
617
641
  async checkFlag(options) {
618
642
  const { fallback = false, key } = options;
619
643
  const context = options.context || this.context;
@@ -641,7 +665,7 @@ var Schematic = class {
641
665
  return fallback;
642
666
  });
643
667
  }
644
- // Make a REST API call to fetch all flag values for a given context
668
+ // Make an API call to fetch all flag values for a given context (use if not in websocket mode)
645
669
  checkFlags = async (context) => {
646
670
  context = context || this.context;
647
671
  const requestUrl = `${this.apiUrl}/flags/check`;
@@ -671,18 +695,6 @@ var Schematic = class {
671
695
  return false;
672
696
  });
673
697
  };
674
- cleanup = async () => {
675
- if (this.conn) {
676
- try {
677
- const socket = await this.conn;
678
- socket.close();
679
- } catch (error) {
680
- console.error("Error during cleanup:", error);
681
- } finally {
682
- this.conn = null;
683
- }
684
- }
685
- };
686
698
  // Send an identify event
687
699
  identify = (body) => {
688
700
  this.setContext({
@@ -700,12 +712,16 @@ var Schematic = class {
700
712
  this.context = context;
701
713
  return Promise.resolve();
702
714
  }
703
- if (!this.conn) {
704
- this.conn = this.wsConnect();
715
+ try {
716
+ this.setIsPending(true);
717
+ if (!this.conn) {
718
+ this.conn = this.wsConnect();
719
+ }
720
+ const socket = await this.conn;
721
+ await this.wsSendMessage(socket, context);
722
+ } catch (error) {
723
+ console.error("Error setting Schematic context:", error);
705
724
  }
706
- return this.conn.then((socket) => {
707
- return this.wsSendMessage(socket, context);
708
- });
709
725
  };
710
726
  // Send track event
711
727
  track = (body) => {
@@ -717,6 +733,9 @@ var Schematic = class {
717
733
  user: user ?? this.context.user
718
734
  });
719
735
  };
736
+ /**
737
+ * Event processing
738
+ */
720
739
  flushEventQueue = () => {
721
740
  while (this.eventQueue.length > 0) {
722
741
  const event = this.eventQueue.shift();
@@ -772,6 +791,22 @@ var Schematic = class {
772
791
  this.eventQueue.push(event);
773
792
  return Promise.resolve();
774
793
  };
794
+ /**
795
+ * Websocket management
796
+ */
797
+ cleanup = async () => {
798
+ if (this.conn) {
799
+ try {
800
+ const socket = await this.conn;
801
+ socket.close();
802
+ } catch (error) {
803
+ console.error("Error during cleanup:", error);
804
+ } finally {
805
+ this.conn = null;
806
+ }
807
+ }
808
+ };
809
+ // Open a websocket connection
775
810
  wsConnect = () => {
776
811
  return new Promise((resolve, reject) => {
777
812
  const wsUrl = `${this.webSocketUrl}/flags/bootstrap`;
@@ -787,6 +822,8 @@ var Schematic = class {
787
822
  };
788
823
  });
789
824
  };
825
+ // Send a message on the websocket indicating interest in a particular evaluation context
826
+ // and wait for the initial set of flag values to be returned
790
827
  wsSendMessage = (socket, context) => {
791
828
  return new Promise((resolve, reject) => {
792
829
  if (contextString(context) == contextString(this.context)) {
@@ -804,16 +841,17 @@ var Schematic = class {
804
841
  (message.flags ?? []).forEach(
805
842
  (flag) => {
806
843
  this.values[contextString(context)][flag.flag] = flag.value;
844
+ this.notifyFlagValueListeners(flag.flag);
807
845
  }
808
846
  );
809
847
  if (this.flagListener) {
810
- this.flagListener(this.values[contextString(context)]);
848
+ this.flagListener(this.getFlagValues());
811
849
  }
850
+ this.setIsPending(false);
812
851
  if (!resolved) {
813
852
  resolved = true;
814
853
  resolve();
815
854
  }
816
- socket.removeEventListener("message", messageHandler);
817
855
  };
818
856
  socket.addEventListener("message", messageHandler);
819
857
  socket.send(
@@ -832,21 +870,46 @@ var Schematic = class {
832
870
  }
833
871
  });
834
872
  };
873
+ /**
874
+ * State management
875
+ */
876
+ // isPending state
877
+ getIsPending = () => {
878
+ return this.isPending;
879
+ };
880
+ addIsPendingListener = (listener) => {
881
+ this.isPendingListeners.add(listener);
882
+ return () => {
883
+ this.isPendingListeners.delete(listener);
884
+ };
885
+ };
886
+ setIsPending = (isPending) => {
887
+ this.isPending = isPending;
888
+ this.isPendingListeners.forEach((listener) => listener());
889
+ };
890
+ // flagValues state
891
+ getFlagValue = (flagKey) => {
892
+ const values = this.getFlagValues();
893
+ return values[flagKey];
894
+ };
895
+ getFlagValues = () => {
896
+ const contextStr = contextString(this.context);
897
+ return this.values[contextStr] ?? {};
898
+ };
899
+ addFlagValueListener = (flagKey, listener) => {
900
+ if (!(flagKey in this.flagValueListeners)) {
901
+ this.flagValueListeners[flagKey] = /* @__PURE__ */ new Set();
902
+ }
903
+ this.flagValueListeners[flagKey].add(listener);
904
+ return () => {
905
+ this.flagValueListeners[flagKey].delete(listener);
906
+ };
907
+ };
908
+ notifyFlagValueListeners = (flagKey) => {
909
+ const listeners = this.flagValueListeners?.[flagKey] ?? [];
910
+ listeners.forEach((listener) => listener());
911
+ };
835
912
  };
836
- function contextString(context) {
837
- const sortedContext = Object.keys(context).reduce((acc, key) => {
838
- const sortedKeys = Object.keys(
839
- context[key] || {}
840
- ).sort();
841
- const sortedObj = sortedKeys.reduce((obj, sortedKey) => {
842
- obj[sortedKey] = context[key][sortedKey];
843
- return obj;
844
- }, {});
845
- acc[key] = sortedObj;
846
- return acc;
847
- }, {});
848
- return JSON.stringify(sortedContext);
849
- }
850
913
  export {
851
914
  Schematic
852
915
  };
package/package.json CHANGED
@@ -1,35 +1,16 @@
1
1
  {
2
2
  "name": "@schematichq/schematic-js",
3
+ "version": "1.0.0-rc.0",
3
4
  "main": "dist/schematic.cjs.js",
4
5
  "module": "dist/schematic.esm.js",
5
- "author": "Ben Papillon <ben@schematichq.com>",
6
- "dependencies": {
7
- "cross-fetch": "^4.0.0",
8
- "uuid": "^9.0.0"
9
- },
10
- "devDependencies": {
11
- "@microsoft/api-extractor": "^7.38.3",
12
- "@types/jest": "^29.5.11",
13
- "@types/uuid": "^9.0.2",
14
- "@typescript-eslint/eslint-plugin": "^6.13.2",
15
- "@typescript-eslint/parser": "^6.13.2",
16
- "esbuild": "^0.19.9",
17
- "esbuild-jest": "^0.5.0",
18
- "eslint": "^8.55.0",
19
- "jest": "^29.7.0",
20
- "jest-environment-jsdom": "^29.7.0",
21
- "jest-esbuild": "^0.3.0",
22
- "jest-fetch-mock": "^3.0.3",
23
- "prettier": "^3.3.1",
24
- "ts-jest": "^29.1.1",
25
- "typescript": "^5.0.2"
26
- },
6
+ "types": "dist/schematic.d.ts",
27
7
  "files": [
28
8
  "dist/schematic.cjs.js",
29
9
  "dist/schematic.esm.js",
30
10
  "dist/schematic.browser.js",
31
11
  "dist/schematic.d.ts"
32
12
  ],
13
+ "author": "Schematic <engineering@schematichq.com>",
33
14
  "license": "MIT",
34
15
  "repository": {
35
16
  "type": "git",
@@ -46,7 +27,26 @@
46
27
  "lint": "eslint src --ext ts,tsx --report-unused-disable-directives --fix",
47
28
  "test": "jest --config jest.config.js"
48
29
  },
49
- "types": "dist/schematic.d.ts",
50
- "version": "0.1.13",
30
+ "dependencies": {
31
+ "cross-fetch": "^4.0.0",
32
+ "uuid": "^10.0.0"
33
+ },
34
+ "devDependencies": {
35
+ "@microsoft/api-extractor": "^7.47.9",
36
+ "@types/jest": "^29.5.13",
37
+ "@types/uuid": "^10.0.0",
38
+ "@typescript-eslint/eslint-plugin": "^8.7.0",
39
+ "@typescript-eslint/parser": "^8.7.0",
40
+ "esbuild": "^0.24.0",
41
+ "esbuild-jest": "^0.5.0",
42
+ "eslint": "^8.57.1",
43
+ "jest": "^29.7.0",
44
+ "jest-environment-jsdom": "^29.7.0",
45
+ "jest-esbuild": "^0.3.0",
46
+ "jest-fetch-mock": "^3.0.3",
47
+ "prettier": "^3.3.3",
48
+ "ts-jest": "^29.2.5",
49
+ "typescript": "^5.6.2"
50
+ },
51
51
  "packageManager": "yarn@1.22.22+sha512.a6b2f7906b721bba3d67d4aff083df04dad64c399707841b7acf00f6b133b7ac24255f2652fa22ae3534329dc6180534e98d17432037ff6fd140556e2bb3137e"
52
52
  }