@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.
- package/dist/schematic.browser.js +2 -2
- package/dist/schematic.cjs.js +112 -49
- package/dist/schematic.d.ts +23 -4
- package/dist/schematic.esm.js +112 -49
- package/package.json +24 -24
|
@@ -1,3 +1,3 @@
|
|
|
1
|
-
"use strict";(()=>{var
|
|
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();
|
|
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 */
|
package/dist/schematic.cjs.js
CHANGED
|
@@ -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
|
-
|
|
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 (
|
|
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
|
-
|
|
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
|
|
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
|
-
|
|
717
|
-
this.
|
|
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.
|
|
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 */
|
package/dist/schematic.d.ts
CHANGED
|
@@ -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
|
|
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 = {
|
package/dist/schematic.esm.js
CHANGED
|
@@ -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
|
-
|
|
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 (
|
|
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
|
-
|
|
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
|
|
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
|
-
|
|
704
|
-
this.
|
|
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.
|
|
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
|
-
"
|
|
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
|
-
"
|
|
50
|
-
|
|
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
|
}
|