@live-state/sync 0.0.2 → 0.0.3

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.
@@ -0,0 +1 @@
1
+ var L="0123456789ABCDEFGHJKMNPQRSTVWXYZ";var l;(function(e){e.Base32IncorrectEncoding="B32_ENC_INVALID",e.DecodeTimeInvalidCharacter="DEC_TIME_CHAR",e.DecodeTimeValueMalformed="DEC_TIME_MALFORMED",e.EncodeTimeNegative="ENC_TIME_NEG",e.EncodeTimeSizeExceeded="ENC_TIME_SIZE_EXCEED",e.EncodeTimeValueMalformed="ENC_TIME_MALFORMED",e.PRNGDetectFailure="PRNG_DETECT",e.ULIDInvalid="ULID_INVALID",e.Unexpected="UNEXPECTED",e.UUIDInvalid="UUID_INVALID";})(l||(l={}));var u=class extends Error{constructor(t,n){super(`${n} (${t})`),this.name="ULIDError",this.code=t;}};function I(e){let t=Math.floor(e()*32);return t===32&&(t=31),L.charAt(t)}function M(e){let t=E(),n=t&&(t.crypto||t.msCrypto)||null;if(typeof(n==null?void 0:n.getRandomValues)=="function")return ()=>{let i=new Uint8Array(1);return n.getRandomValues(i),i[0]/255};if(typeof(n==null?void 0:n.randomBytes)=="function")return ()=>n.randomBytes(1).readUInt8()/255;throw new u(l.PRNGDetectFailure,"Failed to find a reliable PRNG")}function E(){return O()?self:typeof window<"u"?window:typeof global<"u"?global:typeof globalThis<"u"?globalThis:null}function A(e,t){let n="";for(;e>0;e--)n=I(t)+n;return n}function _(e,t=10){if(isNaN(e))throw new u(l.EncodeTimeValueMalformed,`Time must be a number: ${e}`);if(e>0xffffffffffff)throw new u(l.EncodeTimeSizeExceeded,`Cannot encode a time larger than ${0xffffffffffff}: ${e}`);if(e<0)throw new u(l.EncodeTimeNegative,`Time must be positive: ${e}`);if(Number.isInteger(e)===false)throw new u(l.EncodeTimeValueMalformed,`Time must be an integer: ${e}`);let n,i="";for(let r=t;r>0;r--)n=e%32,i=L.charAt(n)+i,e=(e-n)/32;return i}function O(){return typeof WorkerGlobalScope<"u"&&self instanceof WorkerGlobalScope}function b(e,t){let n=M(),i=Date.now();return _(i,10)+A(16,n)}var j=()=>b().toLowerCase(),V=(e,t)=>typeof e=="function"?e(t):e;var w=(e,t,n=[])=>new Proxy(e,{get:(i,r)=>{var p,T;if(r==="__isProxy__")return true;let o=(p=t.get)==null?void 0:p.call(t,i,[...n,r]);if(o!==void 0)return o;let a=i,s=r;return (T=a[s])!=null&&T.__isProxy__||(a[s]=w(typeof a[s]=="object"?a[s]:()=>{},t,[...n,r])),a[s]},apply:(i,r,o)=>{var a;return (a=t.apply)==null?void 0:a.call(t,i,n,o)}}),f=(e,t,n=false)=>Object.entries(t).every(([i,r])=>{if(i==="$and")return r.every(a=>f(e,a,n));if(i==="$or")return r.some(a=>f(e,a,n));let o=(r==null?void 0:r.$eq)!==void 0?r==null?void 0:r.$eq:r;if(typeof r=="object"&&r!==null&&(r==null?void 0:r.$eq)===void 0){if(r.$in!==void 0){let a=e[i];return a===void 0?false:n?!r.$in.includes(a):r.$in.includes(a)}if(r.$not!==void 0&&!n)return f(e,{[i]:r.$not},true);if(r.$gt!==void 0){let a=e[i];return typeof a!="number"?false:n?a<=r.$gt:a>r.$gt}if(r.$gte!==void 0){let a=e[i];return typeof a!="number"?false:n?a<r.$gte:a>=r.$gte}if(r.$lt!==void 0){let a=e[i];return typeof a!="number"?false:n?a>=r.$lt:a<r.$lt}if(r.$lte!==void 0){let a=e[i];return typeof a!="number"?false:n?a>r.$lte:a<=r.$lte}return !e[i]||typeof e[i]!="object"?false:f(e[i],r)}return n?e[i]!==o:e[i]===o}),D=(e,t,n)=>{let i=[],r=0;for(let o=0;o<e.length&&(n===void 0||r<n);o++)t(e[o],o)&&(i.push(e[o]),r++);return i};var R=e=>{if(e)return Array.isArray(e.value)?e.value.map(t=>R(t)):typeof e.value!="object"||e.value===null||e.value instanceof Date?e.value:Object.fromEntries(Object.entries(e.value).map(([t,n])=>[t,R(n)]))};export{j as a,V as b,w as c,f as d,D as e,R as f};
@@ -1 +1 @@
1
- var g=Object.create;var h=Object.defineProperty;var L=Object.getOwnPropertyDescriptor;var b=Object.getOwnPropertyNames;var M=Object.getPrototypeOf,O=Object.prototype.hasOwnProperty;var S=(t,e)=>()=>(e||t((e={exports:{}}).exports,e),e.exports);var j=(t,e,n,a)=>{if(e&&typeof e=="object"||typeof e=="function")for(let i of b(e))!O.call(t,i)&&i!==n&&h(t,i,{get:()=>e[i],enumerable:!(a=L(e,i))||a.enumerable});return t};var I=(t,e,n)=>(n=t!=null?g(M(t)):{},j(e||!t||!t.__esModule?h(n,"default",{value:t,enumerable:true}):n,t));var o=class{_value;_meta;_encodeInput;_decodeInput};var T=class extends o{inner;constructor(e){super(),this.inner=e;}encodeMutation(e,n,a){return this.inner.encodeMutation(e,n,a)}mergeMutation(e,n,a){return this.inner.mergeMutation(e,n,a)}getStorageFieldType(){return {...this.inner.getStorageFieldType(),nullable:true}}},s=class t extends o{storageType;convertFunc;isIndex;isUnique;defaultValue;foreignReference;isPrimary;constructor(e,n,a,i,r,u,y){super(),this.storageType=e,this.convertFunc=n,this.isIndex=a??false,this.isUnique=i??false,this.defaultValue=r,this.foreignReference=u,this.isPrimary=y??false;}encodeMutation(e,n,a){return {value:n,_meta:{timestamp:a}}}mergeMutation(e,n,a){return a&&a._meta.timestamp&&n._meta.timestamp&&a._meta.timestamp.localeCompare(n._meta.timestamp)>=0?[a,null]:[{value:this.convertFunc?this.convertFunc(n.value):n.value,_meta:n._meta},n]}getStorageFieldType(){return {type:this.storageType,nullable:false,index:this.isIndex,unique:this.isUnique,default:this.defaultValue,references:this.foreignReference,primary:this.isPrimary}}unique(){return new t(this.storageType,this.convertFunc,this.isIndex,true,this.defaultValue,this.foreignReference,this.isPrimary)}index(){return new t(this.storageType,this.convertFunc,true,this.isUnique,this.defaultValue,this.foreignReference,this.isPrimary)}default(e){return new t(this.storageType,this.convertFunc,this.isIndex,this.isUnique,e,this.foreignReference,this.isPrimary)}primary(){return new t(this.storageType,this.convertFunc,this.isIndex,this.isUnique,this.defaultValue,this.foreignReference,true)}optional(){return new T(this)}},d=class t extends s{constructor(){super("integer",e=>Number(e));}static create(){return new t}},C=d.create,l=class t extends s{constructor(e){super("varchar",void 0,void 0,void 0,void 0,e);}static create(){return new t}static createId(){return new t().index().unique().primary()}static createReference(e){return new t(e)}},q=l.create,N=l.createId,P=l.createReference,p=class t extends s{constructor(){super("boolean",e=>typeof e=="string"?e.toLowerCase()==="true":!!e);}static create(){return new t}},D=p.create,m=class t extends s{constructor(){super("timestamp",e=>typeof e=="string"?new Date(e):e);}static create(){return new t}},k=m.create;var v=class t extends o{name;fields;relations;constructor(e,n,a){super(),this.name=e,this.fields=n,this.relations=a??{};}encodeMutation(e,n,a){return Object.fromEntries(Object.entries(n).map(([i,r])=>[i,(this.fields[i]??this.relations[i]).encodeMutation("set",r,a)]))}mergeMutation(e,n,a){let i={};return [{value:{...(a==null?void 0:a.value)??{},...Object.fromEntries(Object.entries(n).map(([r,u])=>{let y=this.fields[r]??this.relations[r];if(!y)return [r,u];let[x,f]=y.mergeMutation(e,u,a==null?void 0:a.value[r]);return f&&(i[r]=f),[r,x]}))}},i]}setRelations(e){return new t(this.name,this.fields,e)}getStorageFieldType(){throw new Error("Method not implemented.")}static create(e,n){return new t(e,n)}},X=v.create,c=class t extends o{entity;type;required;relationalColumn;foreignColumn;sourceEntity;constructor(e,n,a,i,r){super(),this.entity=e,this.type=n,this.required=r??false,this.relationalColumn=a,this.foreignColumn=i;}encodeMutation(e,n,a){if(e!=="set")throw new Error("Mutation type not implemented.");if(this.type==="many")throw new Error("Many not implemented.");return {value:n,_meta:{timestamp:a}}}mergeMutation(e,n,a){if(this.type==="many")throw new Error("Many not implemented.");return a&&a._meta.timestamp&&n._meta.timestamp&&a._meta.timestamp.localeCompare(n._meta.timestamp)>=0?[a,null]:[n,n]}getStorageFieldType(){return {type:"varchar",nullable:!this.required,references:`${this.entity.name}.${String(this.foreignColumn??this.relationalColumn??"id")}`}}toJSON(){return {entityName:this.entity.name,type:this.type,required:this.required,relationalColumn:this.relationalColumn,foreignColumn:this.foreignColumn}}static createOneFactory(){return (e,n,a)=>new t(e,"one",n,void 0,a??false)}static createManyFactory(){return (e,n,a)=>new t(e,"many",void 0,n,a??false)}},Y=(t,e)=>({$type:"relations",objectName:t.name,relations:e({one:c.createOneFactory(),many:c.createManyFactory()})}),R=t=>{if(t)return Array.isArray(t.value)?t.value.map(e=>R(e)):typeof t.value!="object"||t.value===null||t.value instanceof Date?t.value:Object.fromEntries(Object.entries(t.value).map(([e,n])=>[e,R(n)]))},Z=t=>Object.fromEntries(Object.entries(t).flatMap(([e,n])=>{if(n.$type==="relations")return [];let a=n,i=Object.values(t).find(r=>r.$type==="relations"&&r.objectName===n.name);return i&&(a=a.setRelations(i.relations)),[[a.name,a]]}));export{S as a,I as b,o as c,d,C as e,l as f,q as g,N as h,P as i,p as j,D as k,m as l,k as m,v as n,X as o,c as p,Y as q,R as r,Z as s};
1
+ var L=Object.create;var h=Object.defineProperty;var g=Object.getOwnPropertyDescriptor;var b=Object.getOwnPropertyNames;var M=Object.getPrototypeOf,I=Object.prototype.hasOwnProperty;var j=(t,e)=>()=>(e||t((e={exports:{}}).exports,e),e.exports);var O=(t,e,n,a)=>{if(e&&typeof e=="object"||typeof e=="function")for(let i of b(e))!I.call(t,i)&&i!==n&&h(t,i,{get:()=>e[i],enumerable:!(a=g(e,i))||a.enumerable});return t};var S=(t,e,n)=>(n=t!=null?L(M(t)):{},O(e||!t||!t.__esModule?h(n,"default",{value:t,enumerable:true}):n,t));var s=class{_value;_meta;_encodeInput;_decodeInput};var c=class extends s{inner;constructor(e){super(),this.inner=e;}encodeMutation(e,n,a){return this.inner.encodeMutation(e,n,a)}mergeMutation(e,n,a){return this.inner.mergeMutation(e,n,a)}getStorageFieldType(){return {...this.inner.getStorageFieldType(),nullable:true}}},o=class t extends s{storageType;convertFunc;isIndex;isUnique;defaultValue;foreignReference;isPrimary;constructor(e,n,a,i,r,u,y){super(),this.storageType=e,this.convertFunc=n,this.isIndex=a??false,this.isUnique=i??false,this.defaultValue=r,this.foreignReference=u,this.isPrimary=y??false;}encodeMutation(e,n,a){return {value:n,_meta:{timestamp:a}}}mergeMutation(e,n,a){return a&&a._meta.timestamp&&n._meta.timestamp&&a._meta.timestamp.localeCompare(n._meta.timestamp)>=0?[a,null]:[{value:this.convertFunc?this.convertFunc(n.value):n.value,_meta:n._meta},n]}getStorageFieldType(){return {type:this.storageType,nullable:false,index:this.isIndex,unique:this.isUnique,default:this.defaultValue,references:this.foreignReference,primary:this.isPrimary}}unique(){return new t(this.storageType,this.convertFunc,this.isIndex,true,this.defaultValue,this.foreignReference,this.isPrimary)}index(){return new t(this.storageType,this.convertFunc,true,this.isUnique,this.defaultValue,this.foreignReference,this.isPrimary)}default(e){return new t(this.storageType,this.convertFunc,this.isIndex,this.isUnique,e,this.foreignReference,this.isPrimary)}primary(){return new t(this.storageType,this.convertFunc,this.isIndex,this.isUnique,this.defaultValue,this.foreignReference,true)}nullable(){return new c(this)}},p=class t extends o{constructor(){super("integer",e=>Number(e));}static create(){return new t}},_=p.create,l=class t extends o{constructor(e){super("varchar",void 0,void 0,void 0,void 0,e);}static create(){return new t}static createId(){return new t().index().unique().primary()}static createReference(e){return new t(e)}},V=l.create,E=l.createId,C=l.createReference,d=class t extends o{constructor(){super("boolean",e=>typeof e=="string"?e.toLowerCase()==="true":!!e);}static create(){return new t}},F=d.create,m=class t extends o{constructor(){super("timestamp",e=>typeof e=="string"?new Date(e):e);}static create(){return new t}},q=m.create;var v=class t extends s{name;fields;relations;constructor(e,n,a){super(),this.name=e,this.fields=n,this.relations=a??{};}encodeMutation(e,n,a){return Object.fromEntries(Object.entries(n).map(([i,r])=>[i,(this.fields[i]??this.relations[i]).encodeMutation("set",r,a)]))}mergeMutation(e,n,a){let i={};return [{value:{...(a==null?void 0:a.value)??{},...Object.fromEntries(Object.entries(n).map(([r,u])=>{let y=this.fields[r]??this.relations[r];if(!y)return [r,u];let[R,f]=y.mergeMutation(e,u,a==null?void 0:a.value[r]);return f&&(i[r]=f),[r,R]}))}},i]}setRelations(e){return new t(this.name,this.fields,e)}getStorageFieldType(){throw new Error("Method not implemented.")}static create(e,n){return new t(e,n)}},$=v.create,T=class t extends s{entity;type;required;relationalColumn;foreignColumn;sourceEntity;constructor(e,n,a,i,r){super(),this.entity=e,this.type=n,this.required=r??false,this.relationalColumn=a,this.foreignColumn=i;}encodeMutation(e,n,a){if(e!=="set")throw new Error("Mutation type not implemented.");if(this.type==="many")throw new Error("Many not implemented.");return {value:n,_meta:{timestamp:a}}}mergeMutation(e,n,a){if(this.type==="many")throw new Error("Many not implemented.");return a&&a._meta.timestamp&&n._meta.timestamp&&a._meta.timestamp.localeCompare(n._meta.timestamp)>=0?[a,null]:[n,n]}getStorageFieldType(){return {type:"varchar",nullable:!this.required,references:`${this.entity.name}.${String(this.foreignColumn??this.relationalColumn??"id")}`}}toJSON(){return {entityName:this.entity.name,type:this.type,required:this.required,relationalColumn:this.relationalColumn,foreignColumn:this.foreignColumn}}static createOneFactory(){return (e,n,a)=>new t(e,"one",n,void 0,a??false)}static createManyFactory(){return (e,n,a)=>new t(e,"many",void 0,n,a??false)}},D=(t,e)=>({$type:"relations",objectName:t.name,relations:e({one:T.createOneFactory(),many:T.createManyFactory()})}),x=t=>{if(t)return Array.isArray(t.value)?t.value.map(e=>x(e)):typeof t.value!="object"||t.value===null||t.value instanceof Date?t.value:Object.fromEntries(Object.entries(t.value).map(([e,n])=>[e,x(n)]))},k=t=>Object.fromEntries(Object.entries(t).flatMap(([e,n])=>{if(n.$type==="relations")return [];let a=n,i=Object.values(t).find(r=>r.$type==="relations"&&r.objectName===n.name);return i&&(a=a.setRelations(i.relations)),[[a.name,a]]}));export{j as a,S as b,s as c,p as d,_ as e,l as f,V as g,E as h,C as i,d as j,F as k,m as l,q as m,v as n,$ as o,T as p,D as q,x as r,k as s};
package/dist/client.d.ts CHANGED
@@ -1,3 +1,3 @@
1
- export { d as Client, C as ClientOptions, c as SubscriptionProvider, e as createClient, u as useLiveQuery } from './index-Cri5dZ2i.js';
1
+ export { f as Client, e as ClientEvents, C as ClientOptions, d as ConnectionStateChangeEvent, M as MessageReceivedEvent, c as SubscriptionProvider, g as createClient, u as useLiveQuery } from './index-4Y_YcJTJ.js';
2
2
  import 'react/jsx-runtime';
3
3
  import 'zod';
package/dist/client.js CHANGED
@@ -1 +1 @@
1
- import {c,a,d,b}from'./chunk-7PTISKJ7.js';import {useState,useRef,useEffect}from'react';import {jsx,Fragment}from'react/jsx-runtime';import {z as z$1}from'zod';import {stringify}from'qs';import {openDB}from'idb';import {sha256}from'crypto-hash';var se=f=>{let[e,t]=useState(()=>f.get()),i=useRef(false);return useEffect(()=>(i.current?t(f.get()):i.current=true,f.subscribe(()=>{let r=f.get();t(r);})),[f]),e},re=({children:f,client:e})=>(useEffect(()=>{e.subscribe();},[]),jsx(Fragment,{children:f}));z$1.object({type:z$1.literal("QUERY"),resource:z$1.string(),where:z$1.record(z$1.any()).optional(),include:z$1.record(z$1.any()).optional()});var x=z$1.record(z$1.object({value:z$1.string().or(z$1.number()).or(z$1.boolean()).or(z$1.date()).nullable(),_meta:z$1.object({timestamp:z$1.string().optional().nullable()}).optional()})).superRefine((f,e)=>{f.id&&e.addIssue({code:z$1.ZodIssueCode.custom,message:"Payload cannot have an id"});}),I=z$1.object({id:z$1.string().optional(),type:z$1.literal("MUTATE"),resource:z$1.string()}),L=I.extend({procedure:z$1.string(),payload:z$1.any().optional()}),C=I.extend({resourceId:z$1.string(),payload:x});z$1.union([L,C]);var S=z$1.string(),$=z$1.object({id:S,type:z$1.literal("SUBSCRIBE"),resource:z$1.string()}),J=z$1.object({id:S,type:z$1.literal("SYNC"),lastSyncedAt:z$1.string().optional(),resources:z$1.string().array().optional(),where:z$1.record(z$1.any()).optional()}),B=C.extend({id:S}),q=L.extend({id:S}),F=z$1.union([q,B]);z$1.union([$,J,F]);var Q=z$1.object({id:S,type:z$1.literal("SYNC"),resource:z$1.string(),data:z$1.record(x)}),Z=z$1.object({id:S,type:z$1.literal("REJECT"),resource:z$1.string(),message:z$1.string().optional()}),X=z$1.object({id:S,type:z$1.literal("REPLY"),data:z$1.any()}),z=z$1.union([Q,Z,X,B]);var w=class{ws=null;url;autoConnect;autoReconnect;reconnectTimeout;reconnectLimit;reconnectAttempts=0;eventListeners=new Map;reconnectTimer=null;intentionallyDisconnected=false;credentials;constructor(e){this.url=e.url,this.autoConnect=e.autoConnect??false,this.autoReconnect=e.autoReconnect??false,this.reconnectTimeout=e.reconnectTimeout??5e3,this.reconnectLimit=e.reconnectLimit,this.credentials=e.credentials,this.autoConnect&&this.connect();}connected(){var e;return ((e=this.ws)==null?void 0:e.readyState)===WebSocket.OPEN}async connect(){if(this.ws&&(this.ws.readyState===WebSocket.OPEN||this.ws.readyState===WebSocket.CONNECTING))return;this.intentionallyDisconnected=false;let e=await b(this.credentials);this.ws=new WebSocket(this.url+(e?`?${stringify(e)}`:"")),this.ws.addEventListener("open",this.handleOpen.bind(this)),this.ws.addEventListener("close",this.handleClose.bind(this)),this.ws.addEventListener("error",this.handleError.bind(this)),this.ws.addEventListener("message",this.handleMessage.bind(this));}disconnect(){this.reconnectTimer&&(clearTimeout(this.reconnectTimer),this.reconnectTimer=null),this.intentionallyDisconnected=true,this.ws&&(this.ws.close(),this.ws=null);}addEventListener(e,t){this.eventListeners.has(e)||this.eventListeners.set(e,new Set),this.eventListeners.get(e).add(t);}removeEventListener(e,t){this.eventListeners.has(e)&&this.eventListeners.get(e).delete(t);}send(e){if(this.ws&&this.ws.readyState===WebSocket.OPEN)this.ws.send(e);else throw new Error("WebSocket is not open")}handleOpen(e){this.reconnectAttempts=0,this.dispatchEvent("open",e),this.dispatchEvent("connectionChange",{open:true});}handleClose(e){this.dispatchEvent("close",e),this.dispatchEvent("connectionChange",{open:false}),this.autoReconnect&&!this.intentionallyDisconnected&&this.scheduleReconnect();}handleError(e){this.dispatchEvent("error",e);}handleMessage(e){this.dispatchEvent("message",e);}scheduleReconnect(){this.reconnectLimit&&this.reconnectAttempts>=this.reconnectLimit||(this.reconnectAttempts++,this.reconnectTimer=setTimeout(()=>{this.connect();},this.reconnectTimeout));}dispatchEvent(e,t){this.eventListeners.has(e)&&this.eventListeners.get(e).forEach(i=>{i(t);});}};var O=class{nodes;constructor(){this.nodes=new Map;}createNode(e,t,i){if(this.nodes.has(e))throw new Error(`Node with id ${e} already exists`);let s={id:e,type:t,referencedBy:new Map(i.map(r=>[r,new Set])),references:new Map,subscriptions:new Set};return this.nodes.set(e,s),s}getNode(e){return this.nodes.get(e)}hasNode(e){return this.nodes.has(e)}createLink(e,t,i){let s=this.nodes.get(e),r=this.nodes.get(t);if(!s)throw new Error(`Source node with id ${e} does not exist`);if(!r)throw new Error(`Target node with id ${t} does not exist`);s.references.set(i,t);let n=r.referencedBy.get(i);n&&n instanceof Set?n.add(e):r.referencedBy.set(i,e),this.notifySubscribers(t);}removeLink(e,t){let i=this.nodes.get(e);if(!i)throw new Error(`Node with id ${e} does not exist`);let s=i.references.get(t);if(!s)return;i.references.delete(t);let r=this.nodes.get(s);if(!r)return;let n=r.referencedBy.get(t);n&&(n instanceof Set?n.delete(e):r.referencedBy.delete(t),this.notifySubscribers(s)),this.notifySubscribers(e);}subscribe(e,t){let i=this.nodes.get(e);if(!i)throw new Error(`Node with id ${e} does not exist`);return i.subscriptions.add(t),()=>{i.subscriptions.delete(t);}}removeNode(e){let t=this.nodes.get(e);t&&(Array.from(t.referencedBy.entries()).forEach(([i,s])=>{(s instanceof Set?Array.from(s.values()):[s]).forEach(n=>{let o=this.nodes.get(n);!o||!o.references.get(i)||(o.references.delete(i),this.notifySubscribers(n));});}),this.nodes.delete(e));}updateNode(e,t){let i=this.nodes.get(e);if(!i)throw new Error(`Node with id ${e} does not exist`);t(i),this.notifySubscribers(e);}notifySubscribers(e){let t=this.nodes.get(e);t&&Array.from(t.subscriptions).forEach(i=>{try{i(e);}catch(s){console.error(`Error in node subscription for node ${e}:`,s);}});}getAllNodes(){return Array.from(this.nodes.values())}};var A=f=>sha256(JSON.stringify(f));var P="__meta",T="databases",R=class{db;async init(e,t){var b,m;if(typeof window>"u")return;let s=((b=(await window.indexedDB.databases()).find(d=>d.name===t))==null?void 0:b.version)??1,r=await A(e),n=Object.fromEntries(await Promise.all(Object.entries(e).map(async([d,p])=>[d,await A(p)]))),o=await openDB("live-state-databases",1,{upgrade(d){d.objectStoreNames.contains(T)||d.createObjectStore(T);}}),l=(m=await this.getAll(o,T))==null?void 0:m[t];(l==null?void 0:l.schemaHash)!==r&&s++,this.db=await openDB(t,s,{async upgrade(d){[...Object.keys(e),P].forEach(p=>{(l==null?void 0:l.objectHashes[p])!==n[p]&&d.objectStoreNames.contains(p)&&d.deleteObjectStore(p),d.objectStoreNames.contains(p)||d.createObjectStore(p);}),await o.put(T,{schemaHash:r,objectHashes:n},t);},blocking(){window.location.reload();},blocked(){window.location.reload();}});}async get(e){return await this.getAll(this.db,e)??{}}getOne(e,t){return this.db?this.db.get(e,t):new Promise(i=>i(void 0))}set(e,t,i){var s;return (s=this.db)==null?void 0:s.put(e,i,t)}delete(e,t){var i;return (i=this.db)==null?void 0:i.delete(e,t)}getMeta(e){return this.db?this.db.get(P,e):new Promise(t=>t(void 0))}setMeta(e,t){var i;return (i=this.db)==null?void 0:i.put(P,t,e)}async getAll(e,t){if(!e)return;if(e.getAllRecords)return e.getAllRecords(t);let[i,s]=await Promise.all([e.getAll(t),e.getAllKeys(t)]);return Object.fromEntries(i.map((r,n)=>[s[n],r]))}};var j=class{constructor(e,t,i){this.schema=e;this.kvStorage=new R,this.kvStorage.init(this.schema,t).then(()=>{this.kvStorage.getMeta("mutationStack").then(s=>{!s||Object.keys(s).length===0||(this.optimisticMutationStack=s,i==null||i(this.optimisticMutationStack));}).then(()=>{Object.entries(this.schema).forEach(([s,r])=>{this.kvStorage.get(s).then(n=>{!n||Object.keys(n).length===0||this.loadConsolidatedState(s,n);});});}).catch(s=>{console.error("Failed to load state from storage",s);});});}rawObjPool={};optimisticMutationStack={};optimisticObjGraph=new O;optimisticRawObjPool={};resourceTypeSubscriptions={};kvStorage;get(e){return Object.fromEntries(Object.entries(this.optimisticRawObjPool[e]??{}).map(([t,i])=>[t,d(i)]))}getOne(e,t){var n;let i=this.optimisticObjGraph.getNode(t);if(!i)return;let s=(n=this.optimisticRawObjPool[e])==null?void 0:n[t];if(!s)return;let r={value:{...s.value,...Object.fromEntries(Array.from(i.references.entries()).map(([o,l])=>{var u;let b=this.optimisticObjGraph.getNode(l);if(!b)return [o,void 0];let[m,d]=Object.entries(this.schema[e].relations).find(h=>h[1].relationalColumn===o||h[1].foreignColumn===o)??[],p=d==null?void 0:d.entity.name;return !p||!d?[o,void 0]:[m,(u=this.optimisticRawObjPool[p])==null?void 0:u[b.id]]})),...Object.fromEntries(Array.from(i.referencedBy.entries()).map(([o,l])=>{var h;let b=l instanceof Set,m=b?Array.from(l.values()).flatMap(g=>{let y=this.optimisticObjGraph.getNode(g);return y?[y]:[]}):this.optimisticObjGraph.getNode(l);if(!m)return [o,void 0];let[d,p]=Object.entries(this.schema[e].relations).find(g=>g[1].entity.name===o)??[],u=p==null?void 0:p.entity.name;return !u||!p?[o,b?[]:void 0]:[d,b?{value:m.map(g=>{var y;return (y=this.optimisticRawObjPool[u])==null?void 0:y[g.id]})}:(h=this.optimisticRawObjPool[u])==null?void 0:h[m.id]]}))}};return d(r)}subscribe(e,t){if(e.length===1)return this.resourceTypeSubscriptions[e[0]]||(this.resourceTypeSubscriptions[e[0]]=new Set),this.resourceTypeSubscriptions[e[0]].add(t),()=>{this.resourceTypeSubscriptions[e[0]].delete(t);};if(e.length===2){if(!this.optimisticObjGraph.getNode(e[1]))throw new Error("Node not found");return this.optimisticObjGraph.subscribe(e[1],t)}throw new Error("Not implemented")}addMutation(e,t,i=false){var l,b,m,d,p;let s=this.schema[e];if(console.log("Adding mutation",t),!s)throw new Error("Schema not found");let r=(l=this.optimisticRawObjPool[e])==null?void 0:l[t.resourceId];if(i)(this.optimisticMutationStack[e]??=[]).push(t);else {this.optimisticMutationStack[e]=((m=(b=this.optimisticMutationStack)==null?void 0:b[e])==null?void 0:m.filter(g=>g.id!==t.id))??[],this.rawObjPool[e]??={};let u={value:{...this.schema[e].mergeMutation("set",t.payload,this.rawObjPool[e][t.resourceId])[0].value,id:{value:t.resourceId}}};this.rawObjPool[e][t.resourceId]=u;let h=u.value;delete h.id,this.kvStorage.set(e,t.resourceId,h);}this.kvStorage.setMeta("mutationStack",this.optimisticMutationStack);let n=(d=this.rawObjPool[e])==null?void 0:d[t.resourceId],o=(this.optimisticMutationStack[e]??[]).reduce((u,h)=>h.resourceId!==t.resourceId?u:this.schema[e].mergeMutation("set",h.payload,u)[0],n);if(o&&((this.optimisticRawObjPool[e]??={})[t.resourceId]={value:{...o.value,id:{value:t.resourceId}}}),this.optimisticObjGraph.hasNode(t.resourceId)||this.optimisticObjGraph.createNode(t.resourceId,e,Object.values(s.relations).flatMap(u=>u.type==="many"?[u.entity.name]:[])),Object.keys(s.relations).length>0){let u=Object.fromEntries(Object.entries(s.relations).flatMap(([h,g])=>g.type==="one"?[[g.relationalColumn,h]]:[]));Object.entries(t.payload).forEach(([h,g])=>{if(!g||!u[h])return;let y=r==null?void 0:r.value[h],[,M]=s.relations[u[h]].mergeMutation("set",g,y);if(M){if(!this.optimisticObjGraph.hasNode(M.value)){let D=s.relations[u[h]].entity.name;this.optimisticObjGraph.createNode(M.value,D,Object.values(this.schema[D].relations).flatMap(G=>G.type==="many"?[G.entity.name]:[]));}y!=null&&y.value&&this.optimisticObjGraph.removeLink(t.resourceId,h),this.optimisticObjGraph.createLink(t.resourceId,M.value,e);}});}(p=this.resourceTypeSubscriptions[e])==null||p.forEach(u=>u()),this.optimisticObjGraph.notifySubscribers(t.resourceId);}loadConsolidatedState(e,t){Object.entries(t).forEach(([i,s])=>{this.addMutation(e,{id:i,type:"MUTATE",resource:e,resourceId:i,payload:s});});}};var k=class{url;ws;store;routeSubscriptions={};replyHandlers={};constructor(e){this.url=e.url,this.store=new j(e.schema,e.storage.name,t=>{var i,s;(s=(i=Object.values(t))==null?void 0:i.flat())==null||s.forEach(r=>this.sendWsMessage(r));}),this.ws=new w({url:e.url,autoConnect:true,autoReconnect:true,reconnectTimeout:5e3,credentials:e.credentials}),this.ws.addEventListener("message",t=>{this.handleServerMessage(t.data);}),this.ws.addEventListener("connectionChange",t=>{t.open&&(this.sendWsMessage({id:a(),type:"SYNC"}),Object.entries(this.routeSubscriptions).forEach(([i,s])=>{s>0&&this.sendWsMessage({id:a(),type:"SUBSCRIBE",resource:i});}),Object.values(this.store.optimisticMutationStack).forEach(i=>{i&&i.forEach(s=>this.sendWsMessage(s));}));});}get(e){if(e.length===0)throw new Error("Path must not be empty");return e.length===1?this.store.get(e[0]):this.store.getOne(e[0],e[1])}handleServerMessage(e){try{console.log("Message received from the server:",e);let t=z.parse(JSON.parse(e));if(console.log("Parsed message:",t),t.type==="MUTATE"){let{resource:i}=t;try{this.store.addMutation(i,t);}catch(s){console.error("Error parsing mutation from the server:",s);}}else if(t.type==="SYNC"){let{resource:i,data:s}=t;console.log("Syncing resource:",s,t),this.store.loadConsolidatedState(i,s);}else if(t.type!=="REJECT"){if(t.type==="REPLY"){let{id:i,data:s}=t;if(!this.replyHandlers[i])return;clearTimeout(this.replyHandlers[i].timeoutHandle),this.replyHandlers[i].handler(s);}}}catch(t){console.error("Error parsing message from the server:",t);}}subscribeToRemote(e){return this.routeSubscriptions[e]=(this.routeSubscriptions[e]??0)+1,this.sendWsMessage({id:a(),type:"SUBSCRIBE",resource:e}),()=>{this.routeSubscriptions[e]-=1,this.routeSubscriptions[e];}}subscribeToStore(e,t){this.store.subscribe(e,t);}mutate(e,t,i){let s={id:a(),type:"MUTATE",resource:e,payload:this.store.schema[e].encodeMutation("set",i,new Date().toISOString()),resourceId:t};this.store.addMutation(e,s,true),this.sendWsMessage(s);}genericMutate(e,t,i){if(!this.ws||!this.ws.connected())throw new Error("WebSocket not connected");let s={id:a(),type:"MUTATE",resource:e,procedure:t,payload:i};return this.sendWsMessage(s),new Promise((r,n)=>{this.replyHandlers[s.id]={timeoutHandle:setTimeout(()=>{delete this.replyHandlers[s.id],n(new Error("Reply timeout"));},5e3),handler:o=>{delete this.replyHandlers[s.id],r(o);}};})}sendWsMessage(e){this.ws&&this.ws.connected()&&this.ws.send(JSON.stringify(e));}},Ve=f=>{let e=new k(f);return {client:{ws:e.ws,subscribe:t=>{let i=[];for(let s of t??Object.keys(e.store.schema))i.push(e.subscribeToRemote(s));return ()=>{console.log("Removing listeners",i),i.forEach(s=>s());}}},store:c(()=>{},{apply:(t,i,s)=>{let r=i.slice(0,-1),n=i[i.length-1];if(n==="get")return e.get(r);if(n==="subscribe")return e.subscribeToStore(r,s[0]);if(n==="subscribeToRemote")return e.subscribeToRemote(r[0]);if(n==="insert"){let{id:o,...l}=s[0];return e.mutate(r[0],o,l)}if(n==="update"){let[o,l]=s;return e.mutate(r[0],o,l)}return e.genericMutate(r[0],n,s[0])}})}};export{re as SubscriptionProvider,Ve as createClient,se as useLiveQuery};
1
+ import {c,a,f,d,e,b}from'./chunk-G5SVBG4B.js';import {useSyncExternalStore,useEffect}from'react';import {xxHash32}from'js-xxhash';import {jsx,Fragment}from'react/jsx-runtime';import {z}from'zod';import {stringify}from'qs';import ae from'fast-deep-equal';import {openDB}from'idb';var m=d=>xxHash32(JSON.stringify(d)).toString(32);var k=class{subscriptions=new Map;getOrStoreSubscription(e){let t=m(e);return this.subscriptions.has(t)?this.subscriptions.get(t).subscribe:(this.subscriptions.set(t,{subscribe:i=>{var n;(n=this.subscriptions.get(t))==null||n.callbacks.add(i);let s=e.subscribe(()=>{var r;(r=this.subscriptions.get(t))==null||r.callbacks.forEach(o=>{o();});});return ()=>{var r;(r=this.subscriptions.get(t))==null||r.callbacks.delete(i),setTimeout(()=>{var o;((o=this.subscriptions.get(t))==null?void 0:o.callbacks.size)===0&&(this.subscriptions.delete(t),s());},10);}},callbacks:new Set}),this.subscriptions.get(t).subscribe)}},F=new k,he=d=>useSyncExternalStore(F.getOrStoreSubscription(d),d.get),pe=({children:d,client:e})=>(useEffect(()=>{e.subscribe();},[e.subscribe]),jsx(Fragment,{children:d}));var N=z.object({resource:z.string(),where:z.record(z.any()).optional(),include:z.record(z.any()).optional(),lastSyncedAt:z.string().optional(),limit:z.number().optional(),sort:z.array(z.object({key:z.string(),direction:z.enum(["asc","desc"])})).optional()}),x=z.record(z.object({value:z.string().or(z.number()).or(z.boolean()).or(z.date()).nullable(),_meta:z.object({timestamp:z.string().optional().nullable()}).optional()})).superRefine((d,e)=>{d.id&&e.addIssue({code:z.ZodIssueCode.custom,message:"Payload cannot have an id"});}),H=z.object({id:z.string().optional(),type:z.literal("MUTATE"),resource:z.string()}),_=H.extend({procedure:z.string(),payload:z.any().optional()}),L=H.extend({resourceId:z.string(),payload:x});z.union([_,L]);var v=z.string(),ee=z.object({id:v,type:z.literal("SUBSCRIBE"),resource:z.string()}),te=N.extend({id:v,type:z.literal("QUERY")}),q=L.extend({id:v}),ie=_.extend({id:v}),se=z.union([ie,q]);z.union([ee,te,se]);var ne=z.object({id:v,type:z.literal("REJECT"),resource:z.string(),message:z.string().optional()}),re=z.object({id:v,type:z.literal("REPLY"),data:z.any()}),K=z.union([ne,re,q]),U=z.object({resource:z.string(),data:z.record(x)});var M=class d{_collection;_client;_where;_include;_limit;_single;_sort;constructor(e,t,i,s,n,r,o){this._collection=e,this._client=t,this._where=i??{},this._include=s??{},this._limit=n,this._single=r,this._sort=o,this.get=this.get.bind(this),this.subscribe=this.subscribe.bind(this);}where(e){return new d(this._collection,this._client,{...this._where,...e},this._include,this._limit,this._single,this._sort)}include(e){return new d(this._collection,this._client,this._where,{...this._include,...e},this._limit,this._single,this._sort)}get(){let e=this._client.get({resource:this._collection.name,where:this._where,include:this._include,limit:this._limit,sort:this._sort});return this._single?e[0]:e}subscribe(e){return this._client.subscribe({resource:this._collection.name,where:this._where,include:this._include,limit:this._limit,sort:this._sort},t=>{if(this._single)return e(t[0]);e(t);})}limit(e){return new d(this._collection,this._client,this._where,this._include,e,this._single,this._sort)}one(e){return this.first({id:e})}first(e){return new d(this._collection,this._client,e??this._where,this._include,1,true,this._sort)}orderBy(e,t="asc"){let i=[...this._sort??[],{key:e,direction:t}];return new d(this._collection,this._client,this._where,this._include,this._limit,this._single,i)}toJSON(){return {resource:this._collection.name,where:this._where,include:this._include,limit:this._limit,sort:this._sort}}static _init(e,t){return new d(e,t)}};var E=class{ws=null;url;autoConnect;autoReconnect;reconnectTimeout;reconnectLimit;reconnectAttempts=0;eventListeners=new Map;reconnectTimer=null;intentionallyDisconnected=false;credentials;constructor(e){this.url=e.url,this.autoConnect=e.autoConnect??false,this.autoReconnect=e.autoReconnect??false,this.reconnectTimeout=e.reconnectTimeout??5e3,this.reconnectLimit=e.reconnectLimit,this.credentials=e.credentials,this.autoConnect&&this.connect();}connected(){var e;return ((e=this.ws)==null?void 0:e.readyState)===WebSocket.OPEN}async connect(){if(this.ws&&(this.ws.readyState===WebSocket.OPEN||this.ws.readyState===WebSocket.CONNECTING))return;this.intentionallyDisconnected=false;let e=await b(this.credentials);this.ws=new WebSocket(this.url+(e?`?${stringify(e)}`:"")),this.ws.addEventListener("open",this.handleOpen.bind(this)),this.ws.addEventListener("close",this.handleClose.bind(this)),this.ws.addEventListener("error",this.handleError.bind(this)),this.ws.addEventListener("message",this.handleMessage.bind(this));}disconnect(){this.reconnectTimer&&(clearTimeout(this.reconnectTimer),this.reconnectTimer=null),this.intentionallyDisconnected=true,this.ws&&(this.ws.close(),this.ws=null);}addEventListener(e,t){var i;this.eventListeners.has(e)||this.eventListeners.set(e,new Set),(i=this.eventListeners.get(e))==null||i.add(t);}removeEventListener(e,t){var i;this.eventListeners.has(e)&&((i=this.eventListeners.get(e))==null||i.delete(t));}send(e){if(this.ws&&this.ws.readyState===WebSocket.OPEN)this.ws.send(e);else throw new Error("WebSocket is not open")}handleOpen(e){this.reconnectAttempts=0,this.dispatchEvent("open",e),this.dispatchEvent("connectionChange",{open:true});}handleClose(e){this.dispatchEvent("close",e),this.dispatchEvent("connectionChange",{open:false}),this.autoReconnect&&!this.intentionallyDisconnected&&this.scheduleReconnect();}handleError(e){var t,i;this.dispatchEvent("error",e),this.dispatchEvent("connectionChange",{open:false}),(i=(t=e.error)==null?void 0:t.message)!=null&&i.includes("non-101")&&(this.ws&&(this.ws.close(),this.ws=null),this.autoReconnect&&!this.intentionallyDisconnected&&this.scheduleReconnect());}handleMessage(e){this.dispatchEvent("message",e);}scheduleReconnect(){this.reconnectLimit&&this.reconnectAttempts>=this.reconnectLimit||(this.reconnectAttempts++,this.reconnectTimer=setTimeout(()=>{this.connect();},this.reconnectTimeout));}dispatchEvent(e,t){var i;(i=this.eventListeners.get(e))==null||i.forEach(s=>{s(t);});}};var T=class{nodes;constructor(){this.nodes=new Map;}createNode(e,t,i){if(this.nodes.has(e))throw new Error(`Node with id ${e} already exists`);let s={id:e,type:t,referencedBy:new Map(i.map(n=>[n,new Set])),references:new Map,subscriptions:new Set};return this.nodes.set(e,s),s}getNode(e){return this.nodes.get(e)}hasNode(e){return this.nodes.has(e)}createLink(e,t){let i=this.nodes.get(e),s=this.nodes.get(t);if(!i)throw new Error(`Source node with id ${e} does not exist`);if(!s)throw new Error(`Target node with id ${t} does not exist`);i.references.set(s.type,t);let n=s.referencedBy.get(i.type);n&&n instanceof Set?n.add(e):s.referencedBy.set(i.type,e),this.notifySubscribers(t);}removeLink(e,t){let i=this.nodes.get(e);if(!i)throw new Error(`Node with id ${e} does not exist`);let s=i.references.get(t);if(!s)return;i.references.delete(t);let n=this.nodes.get(s);if(!n)return;let r=n.referencedBy.get(i.type);r&&(r instanceof Set?r.delete(e):n.referencedBy.delete(i.type),this.notifySubscribers(s)),this.notifySubscribers(e);}subscribe(e,t){let i=this.nodes.get(e);if(!i)throw new Error(`Node with id ${e} does not exist`);return i.subscriptions.add(t),()=>{i.subscriptions.delete(t);}}removeNode(e){let t=this.nodes.get(e);t&&(Array.from(t.referencedBy.entries()).forEach(([i,s])=>{(s instanceof Set?Array.from(s.values()):[s]).forEach(r=>{let o=this.nodes.get(r);!o||!o.references.get(i)||(o.references.delete(i),this.notifySubscribers(r));});}),this.nodes.delete(e));}updateNode(e,t){let i=this.nodes.get(e);if(!i)throw new Error(`Node with id ${e} does not exist`);t(i),this.notifySubscribers(e);}notifySubscribers(e){let t=this.nodes.get(e);t&&Array.from(t.subscriptions).forEach(i=>{try{i(e);}catch(s){console.error(`Error in node subscription for node ${e}:`,s);}});}getAllNodes(){return Array.from(this.nodes.values())}};var A="__meta",C="databases",R=class{db;async init(e,t){var y,h;if(typeof window>"u")return;let s=((y=(await window.indexedDB.databases()).find(l=>l.name===t))==null?void 0:y.version)??1,n=await m(e),r=Object.fromEntries(await Promise.all(Object.entries(e).map(async([l,u])=>[l,await m(u)]))),o=await openDB("live-state-databases",1,{upgrade(l){l.objectStoreNames.contains(C)||l.createObjectStore(C);}}),c=(h=await this.getAll(o,C))==null?void 0:h[t];(c==null?void 0:c.schemaHash)!==n&&s++,this.db=await openDB(t,s,{async upgrade(l){[...Object.keys(e),A].forEach(u=>{(c==null?void 0:c.objectHashes[u])!==r[u]&&l.objectStoreNames.contains(u)&&l.deleteObjectStore(u),l.objectStoreNames.contains(u)||l.createObjectStore(u);}),await o.put(C,{schemaHash:n,objectHashes:r},t);},blocking(){window.location.reload();},blocked(){window.location.reload();}});}async get(e){return await this.getAll(this.db,e)??{}}getOne(e,t){return this.db?this.db.get(e,t):new Promise(i=>i(void 0))}set(e,t,i){var s;return (s=this.db)==null?void 0:s.put(e,i,t)}delete(e,t){var i;return (i=this.db)==null?void 0:i.delete(e,t)}getMeta(e){return this.db?this.db.get(A,e):new Promise(t=>t(void 0))}setMeta(e,t){var i;return (i=this.db)==null?void 0:i.put(A,t,e)}async getAll(e,t){if(!e)return;if(e.getAllRecords)return e.getAllRecords(t);let[i,s]=await Promise.all([e.getAll(t),e.getAllKeys(t)]);return Object.fromEntries(i.map((n,r)=>[s[r],n]))}};var O=class{constructor(e,t,i){this.schema=e;this.kvStorage=new R,t!==false&&this.kvStorage.init(this.schema,t.name).then(()=>{this.kvStorage.getMeta("mutationStack").then(s=>{!s||Object.keys(s).length===0||(this.optimisticMutationStack=s,i==null||i(this.optimisticMutationStack));}).then(()=>{Object.entries(this.schema).forEach(([s,n])=>{this.kvStorage.get(s).then(r=>{!r||Object.keys(r).length===0||this.loadConsolidatedState(s,r);});});}).catch(s=>{console.error("Failed to load state from storage",s);});});}rawObjPool={};optimisticMutationStack={};optimisticObjGraph=new T;optimisticRawObjPool={};collectionSubscriptions=new Map;querySnapshots={};kvStorage;get(e$1,t,i=false){var r;let s=t??m(e$1);if(this.querySnapshots[s]&&!i){let o=this.querySnapshots[s];if(o)return o}let n=((r=e$1.where)!=null&&r.id?[e$1.where.id]:Object.keys(this.optimisticRawObjPool[e$1.resource]??{})).flatMap(o=>{let c=f(this.materializeOneWithInclude(o,e$1.include));return c?[c]:[]});if(e$1.sort&&e$1.sort.length>0){let o=(c,y)=>{for(let h of e$1.sort){let l=c[h.key],u=y[h.key];if(l<u)return h.direction==="asc"?-1:1;if(l>u)return h.direction==="asc"?1:-1}return 0};n.sort(o);}if(e$1.where||e$1.limit){let o=e$1.where?c=>d(c,e$1.where):()=>true;n=e(n,o,e$1.limit);}return i||(this.querySnapshots[s]=n),n}subscribe(e,t){var r;let i=m(e),s=this.collectionSubscriptions.get(i),n=this.schema[e.resource];return s||this.collectionSubscriptions.set(i,{callbacks:new Set,query:e,flatInclude:e.include?Object.keys(e.include).map(o=>n.relations[o].entity.name):void 0}),(r=this.collectionSubscriptions.get(i))==null||r.callbacks.add(t),()=>{var o,c;(o=this.collectionSubscriptions.get(i))==null||o.callbacks.delete(t),((c=this.collectionSubscriptions.get(i))==null?void 0:c.callbacks.size)===0&&(this.collectionSubscriptions.delete(i),delete this.querySnapshots[i]);}}addMutation(e,t,i=false){var c,y,h,l;let s=this.schema[e];if(console.log("Adding mutation",t),!s)throw new Error("Schema not found");let n=(c=this.optimisticRawObjPool[e])==null?void 0:c[t.resourceId];if(i)this.optimisticMutationStack[e]??=[],this.optimisticMutationStack[e].push(t);else {this.optimisticMutationStack[e]=((h=(y=this.optimisticMutationStack)==null?void 0:y[e])==null?void 0:h.filter(b=>b.id!==t.id))??[],this.rawObjPool[e]??={};let u={value:{...this.schema[e].mergeMutation("set",t.payload,this.rawObjPool[e][t.resourceId])[0].value,id:{value:t.resourceId}}};this.rawObjPool[e][t.resourceId]=u;let f=u.value;delete f.id,this.kvStorage.set(e,t.resourceId,f);}this.kvStorage.setMeta("mutationStack",this.optimisticMutationStack);let r=(l=this.rawObjPool[e])==null?void 0:l[t.resourceId],o=(this.optimisticMutationStack[e]??[]).reduce((u,f)=>f.resourceId!==t.resourceId?u:this.schema[e].mergeMutation("set",f.payload,u)[0],r);if(o&&(this.optimisticRawObjPool[e]??={},this.optimisticRawObjPool[e][t.resourceId]={value:{...o.value,id:{value:t.resourceId}}}),this.optimisticObjGraph.hasNode(t.resourceId)||this.optimisticObjGraph.createNode(t.resourceId,e,Object.values(s.relations).flatMap(u=>u.type==="many"?[u.entity.name]:[])),Object.keys(s.relations).length>0){let u=Object.fromEntries(Object.entries(s.relations).flatMap(([f,b])=>b.type==="one"?[[b.relationalColumn,f]]:[]));Object.entries(t.payload).forEach(([f,b])=>{let j=s.relations[u[f]];if(!b||!u[f])return;let S=n==null?void 0:n.value[f],[,w]=j.mergeMutation("set",b,S);if(w){if(!this.optimisticObjGraph.hasNode(w.value)){let W=j.entity.name;this.optimisticObjGraph.createNode(w.value,W,Object.values(this.schema[W].relations).flatMap(P=>P.type==="many"?[P.entity.name]:[]));}S!=null&&S.value&&this.optimisticObjGraph.removeLink(t.resourceId,j.entity.name),this.optimisticObjGraph.createLink(t.resourceId,w.value);}});}this.notifyCollectionSubscribers(e),this.optimisticObjGraph.notifySubscribers(t.resourceId);}loadConsolidatedState(e,t){Object.entries(t).forEach(([i,s])=>{this.addMutation(e,{id:i,type:"MUTATE",resource:e,resourceId:i,payload:s});});}materializeOneWithInclude(e,t={}){var c;if(!e)return;let i=this.optimisticObjGraph.getNode(e);if(!i)return;let s=i.type,n=(c=this.optimisticRawObjPool[s])==null?void 0:c[e];if(!n)return;let[r,o]=Object.keys(t).reduce((y,h)=>{let l=this.schema[s].relations[h];return l.type==="one"?y[0].push([h,l.entity.name]):l.type==="many"&&y[1].push([h,l.entity.name]),y},[[],[]]);return {value:{...n.value,...Object.fromEntries(r.map(([y,h])=>[y,this.materializeOneWithInclude(i.references.get(h))])),...Object.fromEntries(o.map(([y,h])=>{let l=i.referencedBy.get(h),u=l instanceof Set;return [y,u?{value:Array.from(l.values()).map(f=>this.materializeOneWithInclude(f))}:this.materializeOneWithInclude(l)]}))}}}notifyCollectionSubscribers(e){this.collectionSubscriptions.forEach(t=>{if(t.query.resource===e||t.flatInclude&&t.flatInclude.includes(e)){let i=m(t.query),s=this.querySnapshots[i],n=this.get(t.query,void 0,true);if(ae(n,s))return;this.querySnapshots[i]=n,t.callbacks.forEach(r=>{r(n);});}});}};var I=class{url;ws;store;remoteSubCounters={};eventListeners=new Set;replyHandlers={};constructor(e){this.url=e.url,this.store=new O(e.schema,e.storage,t=>{var i,s;(s=(i=Object.values(t))==null?void 0:i.flat())==null||s.forEach(n=>{this.sendWsMessage(n);});}),this.ws=new E({url:e.url,autoConnect:true,autoReconnect:true,reconnectTimeout:5e3,credentials:e.credentials}),this.ws.addEventListener("message",t=>{this.handleServerMessage(t.data);}),this.ws.addEventListener("connectionChange",t=>{this.emitEvent({type:"CONNECTION_STATE_CHANGE",open:t.open}),t.open&&(Object.keys(this.store.schema).forEach(i=>{this.sendWsMessage({id:a(),type:"QUERY",resource:i});}),Object.entries(this.remoteSubCounters).forEach(([i,s])=>{s>0&&this.sendWsMessage({id:a(),type:"SUBSCRIBE",resource:i});}),Object.values(this.store.optimisticMutationStack).forEach(i=>{i&&i.forEach(s=>{this.sendWsMessage(s);});}));});}get(e){return this.store.get(e)}handleServerMessage(e){try{console.log("Message received from the server:",e);let t=K.parse(JSON.parse(e));if(console.log("Parsed message:",t),this.emitEvent({type:"MESSAGE_RECEIVED",message:t}),t.type==="MUTATE"){let{resource:i}=t;try{this.store.addMutation(i,t);}catch(s){console.error("Error merging mutation from the server:",s);}}else if(t.type!=="REJECT"){if(t.type==="REPLY"){let{id:i,data:s}=t;if(this.replyHandlers[i]){clearTimeout(this.replyHandlers[i].timeoutHandle),this.replyHandlers[i].handler(s);return}let n=U.parse(s);this.store.loadConsolidatedState(n.resource,n.data);}}}catch(t){console.error("Error parsing message from the server:",t);}}subscribeToRemote(e){return this.remoteSubCounters[e]=(this.remoteSubCounters[e]??0)+1,this.sendWsMessage({id:a(),type:"SUBSCRIBE",resource:e}),()=>{this.remoteSubCounters[e]-=1,this.remoteSubCounters[e];}}subscribe(e,t){return this.store.subscribe(e,t)}mutate(e,t,i){var n;let s={id:a(),type:"MUTATE",resource:e,payload:this.store.schema[e].encodeMutation("set",i,new Date().toISOString()),resourceId:t};(n=this.store)==null||n.addMutation(e,s,true),this.sendWsMessage(s);}genericMutate(e,t,i){if(!this.ws||!this.ws.connected())throw new Error("WebSocket not connected");let s={id:a(),type:"MUTATE",resource:e,procedure:t,payload:i};return this.sendWsMessage(s),new Promise((n,r)=>{this.replyHandlers[s.id]={timeoutHandle:setTimeout(()=>{delete this.replyHandlers[s.id],r(new Error("Reply timeout"));},5e3),handler:o=>{delete this.replyHandlers[s.id],n(o);}};})}addEventListener(e){return this.eventListeners.add(e),()=>{this.eventListeners.delete(e);}}sendWsMessage(e){var t;(t=this.ws)!=null&&t.connected()&&this.ws.send(JSON.stringify(e));}emitEvent(e){this.eventListeners.forEach(t=>{t(e);});}},qe=d=>{let e=new I(d);return {client:{ws:e.ws,subscribe:t=>{let i=[];for(let s of t??Object.keys(e.store.schema))i.push(e.subscribeToRemote(s));return ()=>{console.log("Removing listeners",i),i.forEach(s=>{s();});}},addEventListener:t=>e.addEventListener(t)},store:{query:Object.entries(d.schema).reduce((t,[i,s])=>(t[i]=M._init(s,e),t),{}),mutate:c(()=>{},{apply:(t,i,s)=>{if(i.length<2)return;if(i.length>2)throw new Error("Trying to access an invalid path");let[n,r]=i;if(r==="insert"){let{id:o,...c}=s[0];return e.mutate(n,o,c)}if(r==="update"){let[o,c]=s;return e.mutate(n,o,c)}return e.genericMutate(n,r,s[0])}})}}};export{pe as SubscriptionProvider,qe as createClient,he as useLiveQuery};
@@ -1,4 +1,4 @@
1
- import { A as AnyRouter, C as ClientOptions, L as LiveObjectAny, W as WhereClause, I as IncludeClause, S as Simplify, a as InferLiveObject, b as LiveObjectMutationInput } from './index-Cri5dZ2i.js';
1
+ import { A as AnyRouter, C as ClientOptions, L as LiveObjectAny, W as WhereClause, I as IncludeClause, S as Simplify, a as InferLiveObject, b as LiveObjectMutationInput } from './index-4Y_YcJTJ.js';
2
2
  import 'react/jsx-runtime';
3
3
  import 'zod';
4
4
 
@@ -13,6 +13,6 @@ type FetchClient<TRouter extends AnyRouter> = {
13
13
  upsert: (input: Simplify<LiveObjectMutationInput<TRouter["routes"][K]["_resourceSchema"]>>) => Promise<void>;
14
14
  };
15
15
  };
16
- declare const createClient: <TRouter extends AnyRouter>(opts: ClientOptions) => FetchClient<TRouter>;
16
+ declare const createClient: <TRouter extends AnyRouter>(opts: Omit<ClientOptions, "storage">) => FetchClient<TRouter>;
17
17
 
18
18
  export { createClient };
@@ -1 +1 @@
1
- import {c,b,d}from'./chunk-7PTISKJ7.js';import {stringify}from'qs';var v=r=>c(()=>{},{apply:async(f,c,i)=>{if(c.length>2)throw new Error("Trying to access invalid property");let[n,o]=c,s=await b(r.credentials)??{};if(o==="get"){let e=i[0],t={};return e!=null&&e.where&&(t.where=e.where),e!=null&&e.include&&(t.include=e.include),fetch(`${r.url}/${n}${Object.keys(t).length>0?`?${stringify(t)}`:""}`,{headers:s}).then(async y=>Object.fromEntries(Object.entries(await y.json()??{}).map(([l,d$1])=>[l,d(d$1)])))}if(o==="upsert"){let{id:e,...t}=i[0];return fetch(`${r.url}/${n}/set`,{method:"POST",headers:{...s,"Content-Type":"application/json"},body:JSON.stringify({resourceId:e,payload:r.schema[n].encodeMutation("set",t,new Date().toISOString())})})}return fetch(`${r.url}/${n}/${o}`,{method:"POST",headers:{...s,"Content-Type":"application/json"},body:JSON.stringify(i[0])})}});export{v as createClient};
1
+ import {c,b,f}from'./chunk-G5SVBG4B.js';import {stringify}from'qs';var v=r=>c(()=>{},{apply:async(f$1,c,i)=>{if(c.length>2)throw new Error("Trying to access invalid property");let[n,o]=c,s=await b(r.credentials)??{};if(o==="get"){let e=i[0],t={};return e!=null&&e.where&&(t.where=e.where),e!=null&&e.include&&(t.include=e.include),fetch(`${r.url}/${n}${Object.keys(t).length>0?`?${stringify(t)}`:""}`,{headers:s}).then(async y=>Object.fromEntries(Object.entries(await y.json()??{}).map(([l,d])=>[l,f(d)])))}if(o==="upsert"){let{id:e,...t}=i[0];return fetch(`${r.url}/${n}/set`,{method:"POST",headers:{...s,"Content-Type":"application/json"},body:JSON.stringify({resourceId:e,payload:r.schema[n].encodeMutation("set",t,new Date().toISOString())})})}return fetch(`${r.url}/${n}/${o}`,{method:"POST",headers:{...s,"Content-Type":"application/json"},body:JSON.stringify(i[0])})}});export{v as createClient};
@@ -41,12 +41,12 @@ type InferLiveType<T extends LiveTypeAny> = T["_value"] extends Record<string, L
41
41
  } : T["_value"];
42
42
  type InferIndex<T extends LiveTypeAny> = string;
43
43
 
44
- declare class OptionalLiveType<T extends LiveTypeAny> extends LiveType<T["_value"] | undefined, T["_meta"], T["_encodeInput"], T["_decodeInput"]> {
44
+ declare class NullableLiveType<T extends LiveTypeAny> extends LiveType<T["_value"] | null, T["_meta"], T["_encodeInput"], T["_decodeInput"]> {
45
45
  readonly inner: T;
46
46
  constructor(inner: T);
47
- encodeMutation(mutationType: MutationType, input: T["_value"] | undefined, timestamp: string): T["_decodeInput"];
48
- mergeMutation(mutationType: MutationType, encodedMutation: T["_decodeInput"], materializedShape?: MaterializedLiveType<LiveType<T["_value"] | undefined, T["_meta"], T["_value"] | Partial<T["_value"] | undefined>, T["_decodeInput"]>> | undefined): [
49
- MaterializedLiveType<LiveType<T["_value"] | undefined, T["_meta"], T["_value"] | Partial<T["_value"] | undefined>, T["_decodeInput"]>>,
47
+ encodeMutation(mutationType: MutationType, input: T["_value"] | null, timestamp: string): T["_decodeInput"];
48
+ mergeMutation(mutationType: MutationType, encodedMutation: T["_decodeInput"], materializedShape?: MaterializedLiveType<LiveType<T["_value"] | null, T["_meta"], T["_value"] | Partial<T["_value"] | null>, T["_decodeInput"]>> | undefined): [
49
+ MaterializedLiveType<LiveType<T["_value"] | null, T["_meta"], T["_value"] | Partial<T["_value"] | null>, T["_decodeInput"]>>,
50
50
  T["_decodeInput"] | null
51
51
  ];
52
52
  getStorageFieldType(): StorageFieldType;
@@ -91,7 +91,7 @@ declare class LiveAtomicType<Value> extends LiveType<Value, LiveAtomicTypeMeta,
91
91
  index(): LiveAtomicType<Value>;
92
92
  default(value: Value): LiveAtomicType<Value>;
93
93
  primary(): LiveAtomicType<Value>;
94
- optional(): OptionalLiveType<this>;
94
+ nullable(): NullableLiveType<this>;
95
95
  }
96
96
  declare class LiveString extends LiveAtomicType<string> {
97
97
  private constructor();
@@ -100,12 +100,14 @@ declare class LiveString extends LiveAtomicType<string> {
100
100
  static createReference(foreignField: `${string}.${string}`): LiveString;
101
101
  }
102
102
 
103
+ /** biome-ignore-all lint/complexity/noBannedTypes: false positive */
104
+
103
105
  type InferLiveObjectWithoutRelations<T extends LiveObjectAny> = {
104
106
  [K in keyof T["fields"]]: InferLiveType<T["fields"][K]>;
105
107
  };
106
- type InferLiveObject<T extends LiveObjectAny> = InferLiveObjectWithoutRelations<T> & {
107
- [K in keyof T["relations"]]: T["relations"][K]["type"] extends "one" ? InferLiveObject<T["relations"][K]["entity"]> : InferLiveObject<T["relations"][K]["entity"]>[];
108
- };
108
+ type InferLiveObject<T extends LiveObjectAny, Include extends IncludeClause<T> | undefined = undefined> = InferLiveObjectWithoutRelations<T> & (Include extends IncludeClause<T> ? {
109
+ [K in keyof T["relations"] as Include[K] extends true ? K : never]: T["relations"][K]["type"] extends "one" ? InferLiveObject<T["relations"][K]["entity"]> : InferLiveObject<T["relations"][K]["entity"]>[];
110
+ } : {});
109
111
  type InferRelationalColumns<T extends Record<string, RelationAny>> = {
110
112
  [K in keyof T as T[K] extends Relation<any, any, any, infer ColumnName, any, any> ? ColumnName extends string ? ColumnName : never : never]: T[K]["type"] extends "one" ? T[K] extends Relation<infer Entity, any, any, any, any, any> ? T[K]["required"] extends true ? InferIndex<Entity> : InferIndex<Entity> | undefined : never : never;
111
113
  };
@@ -189,10 +191,25 @@ type RawSchema = Record<string, LiveObjectAny | RelationsDecl>;
189
191
  type Schema<TRawSchema extends RawSchema> = {
190
192
  [K in keyof TRawSchema as TRawSchema[K] extends LiveObjectAny ? TRawSchema[K]["name"] : never]: TRawSchema[K] extends LiveObjectAny ? ParseObjectFromSchema<TRawSchema, TRawSchema[K]["name"]> : never;
191
193
  };
192
- type WhereClause<T extends LiveObjectAny> = {
193
- [K in keyof T["fields"]]?: InferLiveType<T["fields"][K]>;
194
+ type WhereClause<T extends LiveObjectAny> = ({
195
+ [K in keyof T["fields"]]?: InferLiveType<T["fields"][K]> | ({
196
+ $eq?: InferLiveType<T["fields"][K]>;
197
+ $in?: InferLiveType<T["fields"][K]>[];
198
+ $not?: InferLiveType<T["fields"][K]> | {
199
+ $in?: InferLiveType<T["fields"][K]>[];
200
+ $eq?: InferLiveType<T["fields"][K]>;
201
+ };
202
+ } & (InferLiveType<T["fields"][K]> extends number ? {
203
+ $gt?: InferLiveType<T["fields"][K]>;
204
+ $gte?: InferLiveType<T["fields"][K]>;
205
+ $lt?: InferLiveType<T["fields"][K]>;
206
+ $lte?: InferLiveType<T["fields"][K]>;
207
+ } : {}));
194
208
  } & {
195
209
  [K in keyof T["relations"]]?: WhereClause<T["relations"][K]["entity"]>;
210
+ }) | {
211
+ $and?: WhereClause<T>[];
212
+ $or?: WhereClause<T>[];
196
213
  };
197
214
  type IncludeClause<T extends LiveObjectAny> = {
198
215
  [K in keyof T["relations"]]?: boolean;
@@ -240,7 +257,7 @@ declare class Route<TResourceSchema extends LiveObjectAny, TMiddleware extends M
240
257
  }
241
258
  type AnyRoute = Route<LiveObjectAny, Middleware<any>, Record<string, any>>;
242
259
 
243
- type Simplify<T> = T extends Record<string, any> ? {
260
+ type Simplify<T> = T extends Record<string, unknown> ? {
244
261
  [K in keyof T]: Simplify<T[K]>;
245
262
  } : T;
246
263
 
@@ -275,19 +292,151 @@ type Middleware<T = any> = (opts: {
275
292
  next: NextFunction<T>;
276
293
  }) => ReturnType<NextFunction<T>>;
277
294
 
278
- type DeepSubscribable<T> = {
279
- [K in keyof T]: DeepSubscribable<T[K]>;
280
- } & {
281
- get: () => T;
282
- subscribe: (callback: (value: T) => void) => () => void;
283
- };
284
-
285
- declare const useLiveQuery: <T extends DeepSubscribable<U>, U>(observable: T) => Simplify<ReturnType<T["get"]>>;
295
+ declare const useLiveQuery: <T extends {
296
+ get: () => U;
297
+ subscribe: (cb: (v: U) => void) => () => void;
298
+ }, U>(observable: T) => ReturnType<T["get"]>;
286
299
  declare const SubscriptionProvider: ({ children, client, }: {
287
300
  children: React.ReactNode;
288
301
  client: Client<AnyRouter>["client"];
289
302
  }) => react_jsx_runtime.JSX.Element;
290
303
 
304
+ declare const serverMessageSchema: z.ZodUnion<[z.ZodObject<{
305
+ id: z.ZodString;
306
+ type: z.ZodLiteral<"REJECT">;
307
+ resource: z.ZodString;
308
+ message: z.ZodOptional<z.ZodString>;
309
+ }, "strip", z.ZodTypeAny, {
310
+ type: "REJECT";
311
+ id: string;
312
+ resource: string;
313
+ message?: string | undefined;
314
+ }, {
315
+ type: "REJECT";
316
+ id: string;
317
+ resource: string;
318
+ message?: string | undefined;
319
+ }>, z.ZodObject<{
320
+ id: z.ZodString;
321
+ type: z.ZodLiteral<"REPLY">;
322
+ data: z.ZodAny;
323
+ }, "strip", z.ZodTypeAny, {
324
+ type: "REPLY";
325
+ id: string;
326
+ data?: any;
327
+ }, {
328
+ type: "REPLY";
329
+ id: string;
330
+ data?: any;
331
+ }>, z.ZodObject<z.objectUtil.extendShape<z.objectUtil.extendShape<{
332
+ id: z.ZodOptional<z.ZodString>;
333
+ type: z.ZodLiteral<"MUTATE">;
334
+ resource: z.ZodString;
335
+ }, {
336
+ resourceId: z.ZodString;
337
+ payload: z.ZodEffects<z.ZodRecord<z.ZodString, z.ZodObject<{
338
+ value: z.ZodNullable<z.ZodUnion<[z.ZodUnion<[z.ZodUnion<[z.ZodString, z.ZodNumber]>, z.ZodBoolean]>, z.ZodDate]>>;
339
+ _meta: z.ZodOptional<z.ZodObject<{
340
+ timestamp: z.ZodNullable<z.ZodOptional<z.ZodString>>;
341
+ }, "strip", z.ZodTypeAny, {
342
+ timestamp?: string | null | undefined;
343
+ }, {
344
+ timestamp?: string | null | undefined;
345
+ }>>;
346
+ }, "strip", z.ZodTypeAny, {
347
+ value: string | number | boolean | Date | null;
348
+ _meta?: {
349
+ timestamp?: string | null | undefined;
350
+ } | undefined;
351
+ }, {
352
+ value: string | number | boolean | Date | null;
353
+ _meta?: {
354
+ timestamp?: string | null | undefined;
355
+ } | undefined;
356
+ }>>, Record<string, {
357
+ value: string | number | boolean | Date | null;
358
+ _meta?: {
359
+ timestamp?: string | null | undefined;
360
+ } | undefined;
361
+ }>, Record<string, {
362
+ value: string | number | boolean | Date | null;
363
+ _meta?: {
364
+ timestamp?: string | null | undefined;
365
+ } | undefined;
366
+ }>>;
367
+ }>, {
368
+ id: z.ZodString;
369
+ }>, "strip", z.ZodTypeAny, {
370
+ type: "MUTATE";
371
+ id: string;
372
+ resource: string;
373
+ payload: Record<string, {
374
+ value: string | number | boolean | Date | null;
375
+ _meta?: {
376
+ timestamp?: string | null | undefined;
377
+ } | undefined;
378
+ }>;
379
+ resourceId: string;
380
+ }, {
381
+ type: "MUTATE";
382
+ id: string;
383
+ resource: string;
384
+ payload: Record<string, {
385
+ value: string | number | boolean | Date | null;
386
+ _meta?: {
387
+ timestamp?: string | null | undefined;
388
+ } | undefined;
389
+ }>;
390
+ resourceId: string;
391
+ }>]>;
392
+ type ServerMessage = z.infer<typeof serverMessageSchema>;
393
+
394
+ /** biome-ignore-all lint/complexity/noBannedTypes: <explanation> */
395
+
396
+ type InferQueryResult<TCollection extends LiveObjectAny, TInclude extends IncludeClause<TCollection>, TSingle extends boolean = false> = TSingle extends true ? Simplify<InferLiveObject<TCollection, TInclude>> | undefined : Simplify<InferLiveObject<TCollection, TInclude>>[];
397
+ declare class QueryBuilder<TCollection extends LiveObjectAny, TInclude extends IncludeClause<TCollection> = {}, TSingle extends boolean = false> {
398
+ private _collection;
399
+ private _client;
400
+ private _where;
401
+ private _include;
402
+ private _limit?;
403
+ private _single?;
404
+ private _sort?;
405
+ private constructor();
406
+ where(where: WhereClause<TCollection>): QueryBuilder<TCollection, TInclude, TSingle>;
407
+ include<TNewInclude extends IncludeClause<TCollection>>(include: TNewInclude): QueryBuilder<TCollection, TInclude & TNewInclude, TSingle>;
408
+ get(): InferQueryResult<TCollection, TInclude, TSingle>;
409
+ subscribe(callback: (value: InferQueryResult<TCollection, TInclude, TSingle>) => void): () => void;
410
+ limit(limit: number): QueryBuilder<TCollection, TInclude, TSingle>;
411
+ one(id: string): QueryBuilder<TCollection, TInclude, true>;
412
+ first(where?: WhereClause<TCollection>): QueryBuilder<TCollection, TInclude, true>;
413
+ orderBy(key: keyof TCollection["fields"], direction?: "asc" | "desc"): QueryBuilder<TCollection, TInclude, TSingle>;
414
+ toJSON(): {
415
+ resource: string;
416
+ where: WhereClause<TCollection>;
417
+ include: TInclude;
418
+ limit: number | undefined;
419
+ sort: {
420
+ key: string;
421
+ direction: "asc" | "desc";
422
+ }[] | undefined;
423
+ };
424
+ }
425
+
426
+ type Client$1<TRouter extends AnyRouter> = {
427
+ query: {
428
+ [K in keyof TRouter["routes"]]: QueryBuilder<TRouter["routes"][K]["_resourceSchema"]>;
429
+ };
430
+ mutate: {
431
+ [K in keyof TRouter["routes"]]: {
432
+ insert: (input: Simplify<LiveObjectMutationInput<TRouter["routes"][K]["_resourceSchema"]>>) => void;
433
+ update: (id: string, value: Omit<Simplify<LiveObjectMutationInput<TRouter["routes"][K]["_resourceSchema"]>>, "id">) => void;
434
+ } & {
435
+ [K2 in keyof TRouter["routes"][K]["customMutations"]]: (input: z.infer<TRouter["routes"][K]["customMutations"][K2]["inputValidator"]>) => Promisify<ReturnType<TRouter["routes"][K]["customMutations"][K2]["handler"]>>;
436
+ };
437
+ };
438
+ };
439
+
291
440
  type WebSocketClientEventMap = WebSocketEventMap & {
292
441
  connectionChange: {
293
442
  open: boolean;
@@ -327,35 +476,32 @@ declare class WebSocketClient {
327
476
  private dispatchEvent;
328
477
  }
329
478
 
330
- type ClientState<TRouter extends AnyRouter> = {
331
- [K in keyof TRouter["routes"]]: Record<InferIndex<TRouter["routes"][K]["_resourceSchema"]>, InferLiveObject<TRouter["routes"][K]["_resourceSchema"]>> | undefined;
479
+ type ConnectionStateChangeEvent = {
480
+ type: "CONNECTION_STATE_CHANGE";
481
+ open: boolean;
482
+ };
483
+ type MessageReceivedEvent = {
484
+ type: "MESSAGE_RECEIVED";
485
+ message: ServerMessage;
332
486
  };
487
+ type ClientEvents = ConnectionStateChangeEvent | MessageReceivedEvent;
333
488
  type Client<TRouter extends AnyRouter> = {
334
489
  client: {
335
490
  ws: WebSocketClient;
336
491
  subscribe: (resourceType?: string[]) => () => void;
492
+ addEventListener: (listener: (event: ClientEvents) => void) => () => void;
337
493
  };
338
- store: DeepSubscribable<ClientState<TRouter>> & {
339
- [K in keyof TRouter["routes"]]: {
340
- insert: (input: Simplify<LiveObjectMutationInput<TRouter["routes"][K]["_resourceSchema"]>>) => void;
341
- update: (id: string, value: Omit<Simplify<LiveObjectMutationInput<TRouter["routes"][K]["_resourceSchema"]>>, "id">) => void;
342
- };
343
- } & {
344
- [K in keyof TRouter["routes"]]: {
345
- [K2 in keyof TRouter["routes"][K]["customMutations"]]: (input: z.infer<TRouter["routes"][K]["customMutations"][K2]["inputValidator"]>) => Promisify<ReturnType<TRouter["routes"][K]["customMutations"][K2]["handler"]>>;
346
- };
347
- };
494
+ store: Client$1<TRouter>;
348
495
  };
349
- declare const createClient: <TRouter extends AnyRouter>(opts: ClientOptions & {
350
- storage: {
351
- name: string;
352
- };
353
- }) => Client<TRouter>;
496
+ declare const createClient: <TRouter extends AnyRouter>(opts: ClientOptions) => Client<TRouter>;
354
497
 
355
498
  type ClientOptions = {
356
499
  url: string;
357
500
  schema: Schema<any>;
358
501
  credentials?: Generatable<Awaitable<Record<string, string>>>;
502
+ storage: {
503
+ name: string;
504
+ } | false;
359
505
  };
360
506
 
361
- export { type AnyRouter as A, type ClientOptions as C, type IncludeClause as I, type LiveObjectAny as L, type Simplify as S, type WhereClause as W, type InferLiveObject as a, type LiveObjectMutationInput as b, SubscriptionProvider as c, type Client as d, createClient as e, useLiveQuery as u };
507
+ export { type AnyRouter as A, type ClientOptions as C, type IncludeClause as I, type LiveObjectAny as L, type MessageReceivedEvent as M, type Simplify as S, type WhereClause as W, type InferLiveObject as a, type LiveObjectMutationInput as b, SubscriptionProvider as c, type ConnectionStateChangeEvent as d, type ClientEvents as e, type Client as f, createClient as g, useLiveQuery as u };
package/dist/index.cjs CHANGED
@@ -1 +1 @@
1
- 'use strict';var o=class{_value;_meta;_encodeInput;_decodeInput};var T=class extends o{inner;constructor(e){super(),this.inner=e;}encodeMutation(e,t,n){return this.inner.encodeMutation(e,t,n)}mergeMutation(e,t,n){return this.inner.mergeMutation(e,t,n)}getStorageFieldType(){return {...this.inner.getStorageFieldType(),nullable:true}}},s=class a extends o{storageType;convertFunc;isIndex;isUnique;defaultValue;foreignReference;isPrimary;constructor(e,t,n,r,i,u,y){super(),this.storageType=e,this.convertFunc=t,this.isIndex=n??false,this.isUnique=r??false,this.defaultValue=i,this.foreignReference=u,this.isPrimary=y??false;}encodeMutation(e,t,n){return {value:t,_meta:{timestamp:n}}}mergeMutation(e,t,n){return n&&n._meta.timestamp&&t._meta.timestamp&&n._meta.timestamp.localeCompare(t._meta.timestamp)>=0?[n,null]:[{value:this.convertFunc?this.convertFunc(t.value):t.value,_meta:t._meta},t]}getStorageFieldType(){return {type:this.storageType,nullable:false,index:this.isIndex,unique:this.isUnique,default:this.defaultValue,references:this.foreignReference,primary:this.isPrimary}}unique(){return new a(this.storageType,this.convertFunc,this.isIndex,true,this.defaultValue,this.foreignReference,this.isPrimary)}index(){return new a(this.storageType,this.convertFunc,true,this.isUnique,this.defaultValue,this.foreignReference,this.isPrimary)}default(e){return new a(this.storageType,this.convertFunc,this.isIndex,this.isUnique,e,this.foreignReference,this.isPrimary)}primary(){return new a(this.storageType,this.convertFunc,this.isIndex,this.isUnique,this.defaultValue,this.foreignReference,true)}optional(){return new T(this)}},d=class a extends s{constructor(){super("integer",e=>Number(e));}static create(){return new a}},j=d.create,l=class a extends s{constructor(e){super("varchar",void 0,void 0,void 0,void 0,e);}static create(){return new a}static createId(){return new a().index().unique().primary()}static createReference(e){return new a(e)}},S=l.create,I=l.createId,w=l.createReference,p=class a extends s{constructor(){super("boolean",e=>typeof e=="string"?e.toLowerCase()==="true":!!e);}static create(){return new a}},A=p.create,m=class a extends s{constructor(){super("timestamp",e=>typeof e=="string"?new Date(e):e);}static create(){return new a}},_=m.create;var v=class a extends o{name;fields;relations;constructor(e,t,n){super(),this.name=e,this.fields=t,this.relations=n??{};}encodeMutation(e,t,n){return Object.fromEntries(Object.entries(t).map(([r,i])=>[r,(this.fields[r]??this.relations[r]).encodeMutation("set",i,n)]))}mergeMutation(e,t,n){let r={};return [{value:{...(n==null?void 0:n.value)??{},...Object.fromEntries(Object.entries(t).map(([i,u])=>{let y=this.fields[i]??this.relations[i];if(!y)return [i,u];let[R,f]=y.mergeMutation(e,u,n==null?void 0:n.value[i]);return f&&(r[i]=f),[i,R]}))}},r]}setRelations(e){return new a(this.name,this.fields,e)}getStorageFieldType(){throw new Error("Method not implemented.")}static create(e,t){return new a(e,t)}},D=v.create,c=class a extends o{entity;type;required;relationalColumn;foreignColumn;sourceEntity;constructor(e,t,n,r,i){super(),this.entity=e,this.type=t,this.required=i??false,this.relationalColumn=n,this.foreignColumn=r;}encodeMutation(e,t,n){if(e!=="set")throw new Error("Mutation type not implemented.");if(this.type==="many")throw new Error("Many not implemented.");return {value:t,_meta:{timestamp:n}}}mergeMutation(e,t,n){if(this.type==="many")throw new Error("Many not implemented.");return n&&n._meta.timestamp&&t._meta.timestamp&&n._meta.timestamp.localeCompare(t._meta.timestamp)>=0?[n,null]:[t,t]}getStorageFieldType(){return {type:"varchar",nullable:!this.required,references:`${this.entity.name}.${String(this.foreignColumn??this.relationalColumn??"id")}`}}toJSON(){return {entityName:this.entity.name,type:this.type,required:this.required,relationalColumn:this.relationalColumn,foreignColumn:this.foreignColumn}}static createOneFactory(){return (e,t,n)=>new a(e,"one",t,void 0,n??false)}static createManyFactory(){return (e,t,n)=>new a(e,"many",void 0,t,n??false)}},k=(a,e)=>({$type:"relations",objectName:a.name,relations:e({one:c.createOneFactory(),many:c.createManyFactory()})}),h=a=>{if(a)return Array.isArray(a.value)?a.value.map(e=>h(e)):typeof a.value!="object"||a.value===null||a.value instanceof Date?a.value:Object.fromEntries(Object.entries(a.value).map(([e,t])=>[e,h(t)]))},W=a=>Object.fromEntries(Object.entries(a).flatMap(([e,t])=>{if(t.$type==="relations")return [];let n=t,r=Object.values(a).find(i=>i.$type==="relations"&&i.objectName===t.name);return r&&(n=n.setRelations(r.relations)),[[n.name,n]]}));exports.LiveBoolean=p;exports.LiveNumber=d;exports.LiveObject=v;exports.LiveString=l;exports.LiveTimestamp=m;exports.LiveType=o;exports.Relation=c;exports.boolean=A;exports.createRelations=k;exports.createSchema=W;exports.id=I;exports.inferValue=h;exports.number=j;exports.object=D;exports.reference=w;exports.string=S;exports.timestamp=_;
1
+ 'use strict';var s=class{_value;_meta;_encodeInput;_decodeInput};var c=class extends s{inner;constructor(e){super(),this.inner=e;}encodeMutation(e,t,n){return this.inner.encodeMutation(e,t,n)}mergeMutation(e,t,n){return this.inner.mergeMutation(e,t,n)}getStorageFieldType(){return {...this.inner.getStorageFieldType(),nullable:true}}},o=class a extends s{storageType;convertFunc;isIndex;isUnique;defaultValue;foreignReference;isPrimary;constructor(e,t,n,r,i,u,y){super(),this.storageType=e,this.convertFunc=t,this.isIndex=n??false,this.isUnique=r??false,this.defaultValue=i,this.foreignReference=u,this.isPrimary=y??false;}encodeMutation(e,t,n){return {value:t,_meta:{timestamp:n}}}mergeMutation(e,t,n){return n&&n._meta.timestamp&&t._meta.timestamp&&n._meta.timestamp.localeCompare(t._meta.timestamp)>=0?[n,null]:[{value:this.convertFunc?this.convertFunc(t.value):t.value,_meta:t._meta},t]}getStorageFieldType(){return {type:this.storageType,nullable:false,index:this.isIndex,unique:this.isUnique,default:this.defaultValue,references:this.foreignReference,primary:this.isPrimary}}unique(){return new a(this.storageType,this.convertFunc,this.isIndex,true,this.defaultValue,this.foreignReference,this.isPrimary)}index(){return new a(this.storageType,this.convertFunc,true,this.isUnique,this.defaultValue,this.foreignReference,this.isPrimary)}default(e){return new a(this.storageType,this.convertFunc,this.isIndex,this.isUnique,e,this.foreignReference,this.isPrimary)}primary(){return new a(this.storageType,this.convertFunc,this.isIndex,this.isUnique,this.defaultValue,this.foreignReference,true)}nullable(){return new c(this)}},p=class a extends o{constructor(){super("integer",e=>Number(e));}static create(){return new a}},g=p.create,l=class a extends o{constructor(e){super("varchar",void 0,void 0,void 0,void 0,e);}static create(){return new a}static createId(){return new a().index().unique().primary()}static createReference(e){return new a(e)}},b=l.create,M=l.createId,I=l.createReference,d=class a extends o{constructor(){super("boolean",e=>typeof e=="string"?e.toLowerCase()==="true":!!e);}static create(){return new a}},O=d.create,m=class a extends o{constructor(){super("timestamp",e=>typeof e=="string"?new Date(e):e);}static create(){return new a}},j=m.create;var v=class a extends s{name;fields;relations;constructor(e,t,n){super(),this.name=e,this.fields=t,this.relations=n??{};}encodeMutation(e,t,n){return Object.fromEntries(Object.entries(t).map(([r,i])=>[r,(this.fields[r]??this.relations[r]).encodeMutation("set",i,n)]))}mergeMutation(e,t,n){let r={};return [{value:{...(n==null?void 0:n.value)??{},...Object.fromEntries(Object.entries(t).map(([i,u])=>{let y=this.fields[i]??this.relations[i];if(!y)return [i,u];let[x,f]=y.mergeMutation(e,u,n==null?void 0:n.value[i]);return f&&(r[i]=f),[i,x]}))}},r]}setRelations(e){return new a(this.name,this.fields,e)}getStorageFieldType(){throw new Error("Method not implemented.")}static create(e,t){return new a(e,t)}},K=v.create,T=class a extends s{entity;type;required;relationalColumn;foreignColumn;sourceEntity;constructor(e,t,n,r,i){super(),this.entity=e,this.type=t,this.required=i??false,this.relationalColumn=n,this.foreignColumn=r;}encodeMutation(e,t,n){if(e!=="set")throw new Error("Mutation type not implemented.");if(this.type==="many")throw new Error("Many not implemented.");return {value:t,_meta:{timestamp:n}}}mergeMutation(e,t,n){if(this.type==="many")throw new Error("Many not implemented.");return n&&n._meta.timestamp&&t._meta.timestamp&&n._meta.timestamp.localeCompare(t._meta.timestamp)>=0?[n,null]:[t,t]}getStorageFieldType(){return {type:"varchar",nullable:!this.required,references:`${this.entity.name}.${String(this.foreignColumn??this.relationalColumn??"id")}`}}toJSON(){return {entityName:this.entity.name,type:this.type,required:this.required,relationalColumn:this.relationalColumn,foreignColumn:this.foreignColumn}}static createOneFactory(){return (e,t,n)=>new a(e,"one",t,void 0,n??false)}static createManyFactory(){return (e,t,n)=>new a(e,"many",void 0,t,n??false)}},A=(a,e)=>({$type:"relations",objectName:a.name,relations:e({one:T.createOneFactory(),many:T.createManyFactory()})}),h=a=>{if(a)return Array.isArray(a.value)?a.value.map(e=>h(e)):typeof a.value!="object"||a.value===null||a.value instanceof Date?a.value:Object.fromEntries(Object.entries(a.value).map(([e,t])=>[e,h(t)]))},_=a=>Object.fromEntries(Object.entries(a).flatMap(([e,t])=>{if(t.$type==="relations")return [];let n=t,r=Object.values(a).find(i=>i.$type==="relations"&&i.objectName===t.name);return r&&(n=n.setRelations(r.relations)),[[n.name,n]]}));exports.LiveBoolean=d;exports.LiveNumber=p;exports.LiveObject=v;exports.LiveString=l;exports.LiveTimestamp=m;exports.LiveType=s;exports.Relation=T;exports.boolean=O;exports.createRelations=A;exports.createSchema=_;exports.id=M;exports.inferValue=h;exports.number=g;exports.object=K;exports.reference=I;exports.string=b;exports.timestamp=j;
package/dist/index.d.cts CHANGED
@@ -34,12 +34,12 @@ type InferLiveType<T extends LiveTypeAny> = T["_value"] extends Record<string, L
34
34
  } : T["_value"];
35
35
  type InferIndex<T extends LiveTypeAny> = string;
36
36
 
37
- declare class OptionalLiveType<T extends LiveTypeAny> extends LiveType<T["_value"] | undefined, T["_meta"], T["_encodeInput"], T["_decodeInput"]> {
37
+ declare class NullableLiveType<T extends LiveTypeAny> extends LiveType<T["_value"] | null, T["_meta"], T["_encodeInput"], T["_decodeInput"]> {
38
38
  readonly inner: T;
39
39
  constructor(inner: T);
40
- encodeMutation(mutationType: MutationType, input: T["_value"] | undefined, timestamp: string): T["_decodeInput"];
41
- mergeMutation(mutationType: MutationType, encodedMutation: T["_decodeInput"], materializedShape?: MaterializedLiveType<LiveType<T["_value"] | undefined, T["_meta"], T["_value"] | Partial<T["_value"] | undefined>, T["_decodeInput"]>> | undefined): [
42
- MaterializedLiveType<LiveType<T["_value"] | undefined, T["_meta"], T["_value"] | Partial<T["_value"] | undefined>, T["_decodeInput"]>>,
40
+ encodeMutation(mutationType: MutationType, input: T["_value"] | null, timestamp: string): T["_decodeInput"];
41
+ mergeMutation(mutationType: MutationType, encodedMutation: T["_decodeInput"], materializedShape?: MaterializedLiveType<LiveType<T["_value"] | null, T["_meta"], T["_value"] | Partial<T["_value"] | null>, T["_decodeInput"]>> | undefined): [
42
+ MaterializedLiveType<LiveType<T["_value"] | null, T["_meta"], T["_value"] | Partial<T["_value"] | null>, T["_decodeInput"]>>,
43
43
  T["_decodeInput"] | null
44
44
  ];
45
45
  getStorageFieldType(): StorageFieldType;
@@ -84,7 +84,7 @@ declare class LiveAtomicType<Value> extends LiveType<Value, LiveAtomicTypeMeta,
84
84
  index(): LiveAtomicType<Value>;
85
85
  default(value: Value): LiveAtomicType<Value>;
86
86
  primary(): LiveAtomicType<Value>;
87
- optional(): OptionalLiveType<this>;
87
+ nullable(): NullableLiveType<this>;
88
88
  }
89
89
  declare class LiveNumber extends LiveAtomicType<number> {
90
90
  private constructor();
@@ -111,12 +111,14 @@ declare class LiveTimestamp extends LiveAtomicType<Date> {
111
111
  }
112
112
  declare const timestamp: typeof LiveTimestamp.create;
113
113
 
114
+ /** biome-ignore-all lint/complexity/noBannedTypes: false positive */
115
+
114
116
  type InferLiveObjectWithoutRelations<T extends LiveObjectAny> = {
115
117
  [K in keyof T["fields"]]: InferLiveType<T["fields"][K]>;
116
118
  };
117
- type InferLiveObject<T extends LiveObjectAny> = InferLiveObjectWithoutRelations<T> & {
118
- [K in keyof T["relations"]]: T["relations"][K]["type"] extends "one" ? InferLiveObject<T["relations"][K]["entity"]> : InferLiveObject<T["relations"][K]["entity"]>[];
119
- };
119
+ type InferLiveObject<T extends LiveObjectAny, Include extends IncludeClause<T> | undefined = undefined> = InferLiveObjectWithoutRelations<T> & (Include extends IncludeClause<T> ? {
120
+ [K in keyof T["relations"] as Include[K] extends true ? K : never]: T["relations"][K]["type"] extends "one" ? InferLiveObject<T["relations"][K]["entity"]> : InferLiveObject<T["relations"][K]["entity"]>[];
121
+ } : {});
120
122
  type InferRelationalColumns<T extends Record<string, RelationAny>> = {
121
123
  [K in keyof T as T[K] extends Relation<any, any, any, infer ColumnName, any, any> ? ColumnName extends string ? ColumnName : never : never]: T[K]["type"] extends "one" ? T[K] extends Relation<infer Entity, any, any, any, any, any> ? T[K]["required"] extends true ? InferIndex<Entity> : InferIndex<Entity> | undefined : never : never;
122
124
  };
@@ -207,10 +209,25 @@ type Schema<TRawSchema extends RawSchema> = {
207
209
  [K in keyof TRawSchema as TRawSchema[K] extends LiveObjectAny ? TRawSchema[K]["name"] : never]: TRawSchema[K] extends LiveObjectAny ? ParseObjectFromSchema<TRawSchema, TRawSchema[K]["name"]> : never;
208
210
  };
209
211
  declare const createSchema: <TRawSchema extends RawSchema>(schema: TRawSchema) => Schema<TRawSchema>;
210
- type WhereClause<T extends LiveObjectAny> = {
211
- [K in keyof T["fields"]]?: InferLiveType<T["fields"][K]>;
212
+ type WhereClause<T extends LiveObjectAny> = ({
213
+ [K in keyof T["fields"]]?: InferLiveType<T["fields"][K]> | ({
214
+ $eq?: InferLiveType<T["fields"][K]>;
215
+ $in?: InferLiveType<T["fields"][K]>[];
216
+ $not?: InferLiveType<T["fields"][K]> | {
217
+ $in?: InferLiveType<T["fields"][K]>[];
218
+ $eq?: InferLiveType<T["fields"][K]>;
219
+ };
220
+ } & (InferLiveType<T["fields"][K]> extends number ? {
221
+ $gt?: InferLiveType<T["fields"][K]>;
222
+ $gte?: InferLiveType<T["fields"][K]>;
223
+ $lt?: InferLiveType<T["fields"][K]>;
224
+ $lte?: InferLiveType<T["fields"][K]>;
225
+ } : {}));
212
226
  } & {
213
227
  [K in keyof T["relations"]]?: WhereClause<T["relations"][K]["entity"]>;
228
+ }) | {
229
+ $and?: WhereClause<T>[];
230
+ $or?: WhereClause<T>[];
214
231
  };
215
232
  type IncludeClause<T extends LiveObjectAny> = {
216
233
  [K in keyof T["relations"]]?: boolean;
package/dist/index.d.ts CHANGED
@@ -34,12 +34,12 @@ type InferLiveType<T extends LiveTypeAny> = T["_value"] extends Record<string, L
34
34
  } : T["_value"];
35
35
  type InferIndex<T extends LiveTypeAny> = string;
36
36
 
37
- declare class OptionalLiveType<T extends LiveTypeAny> extends LiveType<T["_value"] | undefined, T["_meta"], T["_encodeInput"], T["_decodeInput"]> {
37
+ declare class NullableLiveType<T extends LiveTypeAny> extends LiveType<T["_value"] | null, T["_meta"], T["_encodeInput"], T["_decodeInput"]> {
38
38
  readonly inner: T;
39
39
  constructor(inner: T);
40
- encodeMutation(mutationType: MutationType, input: T["_value"] | undefined, timestamp: string): T["_decodeInput"];
41
- mergeMutation(mutationType: MutationType, encodedMutation: T["_decodeInput"], materializedShape?: MaterializedLiveType<LiveType<T["_value"] | undefined, T["_meta"], T["_value"] | Partial<T["_value"] | undefined>, T["_decodeInput"]>> | undefined): [
42
- MaterializedLiveType<LiveType<T["_value"] | undefined, T["_meta"], T["_value"] | Partial<T["_value"] | undefined>, T["_decodeInput"]>>,
40
+ encodeMutation(mutationType: MutationType, input: T["_value"] | null, timestamp: string): T["_decodeInput"];
41
+ mergeMutation(mutationType: MutationType, encodedMutation: T["_decodeInput"], materializedShape?: MaterializedLiveType<LiveType<T["_value"] | null, T["_meta"], T["_value"] | Partial<T["_value"] | null>, T["_decodeInput"]>> | undefined): [
42
+ MaterializedLiveType<LiveType<T["_value"] | null, T["_meta"], T["_value"] | Partial<T["_value"] | null>, T["_decodeInput"]>>,
43
43
  T["_decodeInput"] | null
44
44
  ];
45
45
  getStorageFieldType(): StorageFieldType;
@@ -84,7 +84,7 @@ declare class LiveAtomicType<Value> extends LiveType<Value, LiveAtomicTypeMeta,
84
84
  index(): LiveAtomicType<Value>;
85
85
  default(value: Value): LiveAtomicType<Value>;
86
86
  primary(): LiveAtomicType<Value>;
87
- optional(): OptionalLiveType<this>;
87
+ nullable(): NullableLiveType<this>;
88
88
  }
89
89
  declare class LiveNumber extends LiveAtomicType<number> {
90
90
  private constructor();
@@ -111,12 +111,14 @@ declare class LiveTimestamp extends LiveAtomicType<Date> {
111
111
  }
112
112
  declare const timestamp: typeof LiveTimestamp.create;
113
113
 
114
+ /** biome-ignore-all lint/complexity/noBannedTypes: false positive */
115
+
114
116
  type InferLiveObjectWithoutRelations<T extends LiveObjectAny> = {
115
117
  [K in keyof T["fields"]]: InferLiveType<T["fields"][K]>;
116
118
  };
117
- type InferLiveObject<T extends LiveObjectAny> = InferLiveObjectWithoutRelations<T> & {
118
- [K in keyof T["relations"]]: T["relations"][K]["type"] extends "one" ? InferLiveObject<T["relations"][K]["entity"]> : InferLiveObject<T["relations"][K]["entity"]>[];
119
- };
119
+ type InferLiveObject<T extends LiveObjectAny, Include extends IncludeClause<T> | undefined = undefined> = InferLiveObjectWithoutRelations<T> & (Include extends IncludeClause<T> ? {
120
+ [K in keyof T["relations"] as Include[K] extends true ? K : never]: T["relations"][K]["type"] extends "one" ? InferLiveObject<T["relations"][K]["entity"]> : InferLiveObject<T["relations"][K]["entity"]>[];
121
+ } : {});
120
122
  type InferRelationalColumns<T extends Record<string, RelationAny>> = {
121
123
  [K in keyof T as T[K] extends Relation<any, any, any, infer ColumnName, any, any> ? ColumnName extends string ? ColumnName : never : never]: T[K]["type"] extends "one" ? T[K] extends Relation<infer Entity, any, any, any, any, any> ? T[K]["required"] extends true ? InferIndex<Entity> : InferIndex<Entity> | undefined : never : never;
122
124
  };
@@ -207,10 +209,25 @@ type Schema<TRawSchema extends RawSchema> = {
207
209
  [K in keyof TRawSchema as TRawSchema[K] extends LiveObjectAny ? TRawSchema[K]["name"] : never]: TRawSchema[K] extends LiveObjectAny ? ParseObjectFromSchema<TRawSchema, TRawSchema[K]["name"]> : never;
208
210
  };
209
211
  declare const createSchema: <TRawSchema extends RawSchema>(schema: TRawSchema) => Schema<TRawSchema>;
210
- type WhereClause<T extends LiveObjectAny> = {
211
- [K in keyof T["fields"]]?: InferLiveType<T["fields"][K]>;
212
+ type WhereClause<T extends LiveObjectAny> = ({
213
+ [K in keyof T["fields"]]?: InferLiveType<T["fields"][K]> | ({
214
+ $eq?: InferLiveType<T["fields"][K]>;
215
+ $in?: InferLiveType<T["fields"][K]>[];
216
+ $not?: InferLiveType<T["fields"][K]> | {
217
+ $in?: InferLiveType<T["fields"][K]>[];
218
+ $eq?: InferLiveType<T["fields"][K]>;
219
+ };
220
+ } & (InferLiveType<T["fields"][K]> extends number ? {
221
+ $gt?: InferLiveType<T["fields"][K]>;
222
+ $gte?: InferLiveType<T["fields"][K]>;
223
+ $lt?: InferLiveType<T["fields"][K]>;
224
+ $lte?: InferLiveType<T["fields"][K]>;
225
+ } : {}));
212
226
  } & {
213
227
  [K in keyof T["relations"]]?: WhereClause<T["relations"][K]["entity"]>;
228
+ }) | {
229
+ $and?: WhereClause<T>[];
230
+ $or?: WhereClause<T>[];
214
231
  };
215
232
  type IncludeClause<T extends LiveObjectAny> = {
216
233
  [K in keyof T["relations"]]?: boolean;
package/dist/index.js CHANGED
@@ -1 +1 @@
1
- export{j as LiveBoolean,d as LiveNumber,n as LiveObject,f as LiveString,l as LiveTimestamp,c as LiveType,p as Relation,k as boolean,q as createRelations,s as createSchema,h as id,r as inferValue,e as number,o as object,i as reference,g as string,m as timestamp}from'./chunk-ECP6AHZL.js';
1
+ export{j as LiveBoolean,d as LiveNumber,n as LiveObject,f as LiveString,l as LiveTimestamp,c as LiveType,p as Relation,k as boolean,q as createRelations,s as createSchema,h as id,r as inferValue,e as number,o as object,i as reference,g as string,m as timestamp}from'./chunk-LLHCJUB6.js';
package/dist/server.cjs CHANGED
@@ -1,2 +1,2 @@
1
- 'use strict';var Ce=require('qs'),zod=require('zod'),j=require('crypto'),kysely=require('kysely'),postgres=require('kysely/helpers/postgres');function _interopDefault(e){return e&&e.__esModule?e:{default:e}}var Ce__default=/*#__PURE__*/_interopDefault(Ce);var j__default=/*#__PURE__*/_interopDefault(j);var pe=Object.create;var W=Object.defineProperty;var Te=Object.getOwnPropertyDescriptor;var fe=Object.getOwnPropertyNames;var he=Object.getPrototypeOf,ge=Object.prototype.hasOwnProperty;var Re=(n,e)=>()=>(e||n((e={exports:{}}).exports,e),e.exports);var ve=(n,e,t,r)=>{if(e&&typeof e=="object"||typeof e=="function")for(let a of fe(e))!ge.call(n,a)&&a!==t&&W(n,a,{get:()=>e[a],enumerable:!(r=Te(e,a))||r.enumerable});return n};var Q=(n,e,t)=>(t=n!=null?pe(he(n)):{},ve(W(t,"default",{value:n,enumerable:true}),n));var _=Re(E=>{Object.defineProperty(E,"__esModule",{value:true});E.parse=Le;E.serialize=Ae;var be=/^[\u0021-\u003A\u003C\u003E-\u007E]+$/,xe=/^[\u0021-\u003A\u003C-\u007E]*$/,we=/^([.]?[a-z0-9]([a-z0-9-]{0,61}[a-z0-9])?)([.][a-z0-9]([a-z0-9-]{0,61}[a-z0-9])?)*$/i,Me=/^[\u0020-\u003A\u003D-\u007E]*$/,Se=Object.prototype.toString,Ie=(()=>{let n=function(){};return n.prototype=Object.create(null),n})();function Le(n,e){let t=new Ie,r=n.length;if(r<2)return t;let a=(e==null?void 0:e.decode)||Oe,i=0;do{let o=n.indexOf("=",i);if(o===-1)break;let c=n.indexOf(";",i),s=c===-1?r:c;if(o>s){i=n.lastIndexOf(";",o-1)+1;continue}let p=B(n,i,o),m=Y(n,o,p),u=n.slice(p,m);if(t[u]===void 0){let y=B(n,o+1,s),l=Y(n,s,y),d=a(n.slice(y,l));t[u]=d;}i=s+1;}while(i<r);return t}function B(n,e,t){do{let r=n.charCodeAt(e);if(r!==32&&r!==9)return e}while(++e<t);return t}function Y(n,e,t){for(;e>t;){let r=n.charCodeAt(--e);if(r!==32&&r!==9)return e+1}return t}function Ae(n,e,t){let r=(t==null?void 0:t.encode)||encodeURIComponent;if(!be.test(n))throw new TypeError(`argument name is invalid: ${n}`);let a=r(e);if(!xe.test(a))throw new TypeError(`argument val is invalid: ${e}`);let i=n+"="+a;if(!t)return i;if(t.maxAge!==void 0){if(!Number.isInteger(t.maxAge))throw new TypeError(`option maxAge is invalid: ${t.maxAge}`);i+="; Max-Age="+t.maxAge;}if(t.domain){if(!we.test(t.domain))throw new TypeError(`option domain is invalid: ${t.domain}`);i+="; Domain="+t.domain;}if(t.path){if(!Me.test(t.path))throw new TypeError(`option path is invalid: ${t.path}`);i+="; Path="+t.path;}if(t.expires){if(!Ee(t.expires)||!Number.isFinite(t.expires.valueOf()))throw new TypeError(`option expires is invalid: ${t.expires}`);i+="; Expires="+t.expires.toUTCString();}if(t.httpOnly&&(i+="; HttpOnly"),t.secure&&(i+="; Secure"),t.partitioned&&(i+="; Partitioned"),t.priority)switch(typeof t.priority=="string"?t.priority.toLowerCase():void 0){case "low":i+="; Priority=Low";break;case "medium":i+="; Priority=Medium";break;case "high":i+="; Priority=High";break;default:throw new TypeError(`option priority is invalid: ${t.priority}`)}if(t.sameSite)switch(typeof t.sameSite=="string"?t.sameSite.toLowerCase():t.sameSite){case true:case "strict":i+="; SameSite=Strict";break;case "lax":i+="; SameSite=Lax";break;case "none":i+="; SameSite=None";break;default:throw new TypeError(`option sameSite is invalid: ${t.sameSite}`)}return i}function Oe(n){if(n.indexOf("%")===-1)return n;try{return decodeURIComponent(n)}catch{return n}}function Ee(n){return Se.call(n)==="[object Date]"}});var te=Q(_());var J=zod.z.object({type:zod.z.literal("QUERY"),resource:zod.z.string(),where:zod.z.record(zod.z.any()).optional(),include:zod.z.record(zod.z.any()).optional()}),P=zod.z.record(zod.z.object({value:zod.z.string().or(zod.z.number()).or(zod.z.boolean()).or(zod.z.date()).nullable(),_meta:zod.z.object({timestamp:zod.z.string().optional().nullable()}).optional()})).superRefine((n,e)=>{n.id&&e.addIssue({code:zod.z.ZodIssueCode.custom,message:"Payload cannot have an id"});}),X=zod.z.object({id:zod.z.string().optional(),type:zod.z.literal("MUTATE"),resource:zod.z.string()}),I=X.extend({procedure:zod.z.string(),payload:zod.z.any().optional()}),L=X.extend({resourceId:zod.z.string(),payload:P});zod.z.union([I,L]);var ee=J.omit({type:true,resource:true}),V=I.omit({id:true,type:true,resource:true,procedure:true}),N=L.omit({id:true,type:true,resource:true});zod.z.union([N,V]);var ne=n=>async e=>{var t;try{let r=typeof e.headers.getSetCookie=="function"?Object.fromEntries(e.headers):e.headers,a={headers:r,cookies:r.cookie?te.default.parse(r.cookie):{}},i=new URL(e.url),o=i.pathname.split("/"),c=i.searchParams,s=Ce__default.default.parse(c.toString()),p=await((t=n.contextProvider)==null?void 0:t.call(n,{transport:"HTTP",headers:a.headers,cookies:a.cookies,query:s}))??{};if(e.method==="GET"){let m=o[o.length-1],{success:u,data:y,error:l}=ee.safeParse(s);if(!u)return Response.json({message:"Invalid query",code:"INVALID_QUERY",details:l},{status:400});let d=await n.handleRequest({req:{...a,type:"QUERY",resourceName:m,context:p,where:y.where,include:y.include,query:s}});return !d||!d.data?Response.json({message:"Invalid resource",code:"INVALID_RESOURCE"},{status:400}):Response.json(d.data)}if(e.method==="POST")try{let m=o[o.length-1],u=o[o.length-2],y=e.body?await e.json():{},l;if(m==="set"){let{success:w,data:h,error:S}=N.safeParse(y);if(!w)return Response.json({message:"Invalid mutation",code:"INVALID_REQUEST",details:S},{status:400});l=h;}else {let{success:w,data:h,error:S}=V.safeParse(y);if(!w)return Response.json({message:"Invalid mutation",code:"INVALID_REQUEST",details:S},{status:400});l=h;}let d=await n.handleRequest({req:{...a,type:"MUTATE",resourceName:u,input:l.payload,context:p,resourceId:l.resourceId,procedure:m!=="set"?m:void 0,query:{}}});return Response.json(d)}catch(m){return console.error("Error parsing mutation from the client:",m),Response.json({message:"Internal server error",code:"INTERNAL_SERVER_ERROR"},{status:500})}return Response.json({message:"Not found",code:"NOT_FOUND"},{status:404})}catch(r){return console.error("Unexpected error:",r),Response.json({message:"Internal server error",code:"INTERNAL_SERVER_ERROR"},{status:500})}};var ue=Q(_());var g=zod.z.string(),_e=zod.z.object({id:g,type:zod.z.literal("SUBSCRIBE"),resource:zod.z.string()}),Pe=zod.z.object({id:g,type:zod.z.literal("SYNC"),lastSyncedAt:zod.z.string().optional(),resources:zod.z.string().array().optional(),where:zod.z.record(zod.z.any()).optional()}),re=L.extend({id:g}),Ve=I.extend({id:g}),Ne=zod.z.union([Ve,re]),ae=zod.z.union([_e,Pe,Ne]),Fe=zod.z.object({id:g,type:zod.z.literal("SYNC"),resource:zod.z.string(),data:zod.z.record(P)}),De=zod.z.object({id:g,type:zod.z.literal("REJECT"),resource:zod.z.string(),message:zod.z.string().optional()}),$e=zod.z.object({id:g,type:zod.z.literal("REPLY"),data:zod.z.any()});zod.z.union([Fe,De,$e,re]);var oe="0123456789ABCDEFGHJKMNPQRSTVWXYZ",A=32;var qe=16,se=10,ie=0xffffffffffff;var R;(function(n){n.Base32IncorrectEncoding="B32_ENC_INVALID",n.DecodeTimeInvalidCharacter="DEC_TIME_CHAR",n.DecodeTimeValueMalformed="DEC_TIME_MALFORMED",n.EncodeTimeNegative="ENC_TIME_NEG",n.EncodeTimeSizeExceeded="ENC_TIME_SIZE_EXCEED",n.EncodeTimeValueMalformed="ENC_TIME_MALFORMED",n.PRNGDetectFailure="PRNG_DETECT",n.ULIDInvalid="ULID_INVALID",n.Unexpected="UNEXPECTED",n.UUIDInvalid="UUID_INVALID";})(R||(R={}));var v=class extends Error{constructor(e,t){super(`${t} (${e})`),this.name="ULIDError",this.code=e;}};function Ue(n){let e=Math.floor(n()*A);return e===A&&(e=A-1),oe.charAt(e)}function Ke(n){var r;let e=ke(),t=e&&(e.crypto||e.msCrypto)||(typeof j__default.default<"u"?j__default.default:null);if(typeof(t==null?void 0:t.getRandomValues)=="function")return ()=>{let a=new Uint8Array(1);return t.getRandomValues(a),a[0]/255};if(typeof(t==null?void 0:t.randomBytes)=="function")return ()=>t.randomBytes(1).readUInt8()/255;if((r=j__default.default)!=null&&r.randomBytes)return ()=>j__default.default.randomBytes(1).readUInt8()/255;throw new v(R.PRNGDetectFailure,"Failed to find a reliable PRNG")}function ke(){return He()?self:typeof window<"u"?window:typeof global<"u"?global:typeof globalThis<"u"?globalThis:null}function ze(n,e){let t="";for(;n>0;n--)t=Ue(e)+t;return t}function Ze(n,e=se){if(isNaN(n))throw new v(R.EncodeTimeValueMalformed,`Time must be a number: ${n}`);if(n>ie)throw new v(R.EncodeTimeSizeExceeded,`Cannot encode a time larger than ${ie}: ${n}`);if(n<0)throw new v(R.EncodeTimeNegative,`Time must be positive: ${n}`);if(Number.isInteger(n)===false)throw new v(R.EncodeTimeValueMalformed,`Time must be an integer: ${n}`);let t,r="";for(let a=e;a>0;a--)t=n%A,r=oe.charAt(t)+r,n=(n-t)/A;return r}function He(){return typeof WorkerGlobalScope<"u"&&self instanceof WorkerGlobalScope}function ce(n,e){let t=Ke(),r=Date.now();return Ze(r,se)+ze(qe,t)}var F=()=>ce().toLowerCase();var le=n=>{let e={},t={};return n.subscribeToMutations(r=>{let a=r;!a.resourceId||!a.payload||(console.log("Mutation propagated:",a),Object.entries(t[a.resource]??{}).forEach(([i,o])=>{var c;(c=e[i])==null||c.send(JSON.stringify({...a,id:a.id??F()}));}));}),(r,a)=>{var m;let i=u=>{r.send(JSON.stringify(u));},o=F(),c={headers:a.headers,cookies:typeof a.headers.cookie=="string"?ue.default.parse(a.headers.cookie):{}},s=Ce.parse(a.url.split("?")[1]),p=(m=n.contextProvider)==null?void 0:m.call(n,{transport:"WEBSOCKET",headers:c.headers,cookies:c.cookies,query:s});e[o]=r,console.log("Client connected:",o),r.on("message",async u=>{try{console.log("Message received from the client:",u);let y=ae.parse(JSON.parse(u.toString()));if(y.type==="SUBSCRIBE"){let{resource:l}=y;t[l]||(t[l]={}),t[l][o]={};}else if(y.type==="SYNC"){let{resources:l}=y,d=l??Object.keys(n.schema);console.log("Syncing resources:",d),await Promise.all(d.map(async w=>{let h=await n.handleRequest({req:{...c,type:"QUERY",resourceName:w,context:await p??{},query:s}});if(!h||!h.data)throw new Error("Invalid resource");i({id:y.id,type:"SYNC",resource:w,data:Object.fromEntries(Object.entries(h.data??{}).map(([S,me])=>[S,me.value]))});}));}else if(y.type==="MUTATE"){let{resource:l}=y;console.log("Received mutation from client:",y);try{let d=await n.handleRequest({req:{...c,type:"MUTATE",resourceName:l,input:y.payload,context:{messageId:y.id,...await p??{}},resourceId:y.resourceId,procedure:y.procedure,query:s}});y.procedure&&i({id:y.id,type:"REPLY",data:d});}catch(d){i({id:y.id,type:"REJECT",resource:l,message:d.message}),console.error("Error parsing mutation from the client:",d);}}}catch(y){console.error("Error handling message from the client:",y);}}),r.on("close",()=>{console.log("Connection closed",o),delete e[o];for(let u of Object.values(t))delete u[o];});}};function de(n){let e=`${n.protocol}://${n.hostname}${n.url}`,t=new Headers;return Object.entries(n.headers).forEach(([r,a])=>{a&&t.set(r,Array.isArray(a)?a.join(","):a);}),new Request(e,{method:n.method,headers:t,body:n.body&&n.method!=="GET"?JSON.stringify(n.body):void 0})}var Et=(n,e,t)=>{n.ws(`${(t==null?void 0:t.basePath)??""}/ws`,le(e)),n.use(`${(t==null?void 0:t.basePath)??""}/`,(r,a)=>{ne(e)(de(r)).then(o=>o.json().then(c=>a.status(o.status).send(c)));});};var D=class n{routes;constructor(e){this.routes=e.routes;}static create(e){return new n(e)}},Pt=n=>D.create({...n}),Qe=n=>({handler:e=>({inputValidator:n??zod.z.undefined(),handler:e})}),$=class n{_resourceSchema;resourceName;middlewares;customMutations;constructor(e,t){this.resourceName=e,this.middlewares=new Set,this.customMutations=t??{};}handleFind=async({req:e,db:t})=>({data:await t.rawFind(e.resourceName,e.where,e.include),acceptedValues:null});handleSet=async({req:e,db:t,schema:r})=>{if(!e.input)throw new Error("Payload is required");if(!e.resourceId)throw new Error("ResourceId is required");let a=await t.rawFindById(e.resourceName,e.resourceId),[i,o]=r[this.resourceName].mergeMutation("set",e.input,a);if(!o)throw new Error("Mutation rejected");return {data:await t.rawUpsert(e.resourceName,e.resourceId,i),acceptedValues:o}};async handleRequest(e){let t=r=>(()=>{if(r.type==="QUERY")return this.handleFind({req:r,db:e.db,schema:e.schema});if(r.type==="MUTATE")if(r.procedure){if(this.customMutations[r.procedure]){let a=this.customMutations[r.procedure].inputValidator.parse(r.input);return r.input=a,this.customMutations[r.procedure].handler({req:r,db:e.db,schema:e.schema})}}else return this.handleSet({req:r,db:e.db,schema:e.schema});throw new Error("Invalid request")})();return await Array.from(this.middlewares.values()).reduceRight((r,a)=>i=>a({req:i,next:r}),async r=>t(r))(e.req)}use(...e){for(let t of e)this.middlewares.add(t);return this}withMutations(e){return new n(this.resourceName,e({mutation:Qe}))}},q=class n{middlewares;constructor(e=[]){this.middlewares=e;}createBasicRoute(e){return new $(e.name).use(...this.middlewares)}use(...e){return new n([...this.middlewares,...e])}static create(){return new n}},Vt=q.create;var x=n=>{if(n)return Array.isArray(n.value)?n.value.map(e=>x(e)):typeof n.value!="object"||n.value===null||n.value instanceof Date?n.value:Object.fromEntries(Object.entries(n.value).map(([e,t])=>[e,x(t)]))};var H=class{async insert(e,t){let r=new Date().toISOString();return x(await this.rawUpsert(e.name,t.id,{value:Object.fromEntries(Object.entries(t).map(([a,i])=>[a,{value:i,_meta:{timestamp:r}}]))}))}async update(e,t,r){let a=new Date().toISOString(),{id:i,...o}=r;return x(await this.rawUpsert(e.name,t,{value:Object.fromEntries(Object.entries(o).map(([c,s])=>[c,{value:s,_meta:{timestamp:a}}]))}))}},ye=class extends H{db;schema;constructor(e){super(),this.db=new kysely.Kysely({dialect:new kysely.PostgresDialect({pool:e})});}async updateSchema(e){this.schema=e;let t=await this.db.introspection.getTables();for(let[r,a]of Object.entries(e)){let i=t.find(s=>s.name===r);i||await this.db.schema.createTable(r).ifNotExists().execute();let o=`${r}_meta`,c=t.find(s=>s.name===o);c||await this.db.schema.createTable(o).ifNotExists().execute();for(let[s,p]of Object.entries(a.fields)){let m=i==null?void 0:i.columns.find(l=>l.name===s),u=p.getStorageFieldType();m?m.dataType!==u.type&&console.error("Column type mismatch:",s,"expected to have type:",u.type,"but has type:",m.dataType):(await this.db.schema.alterTable(r).addColumn(s,u.type,l=>{let d=l;return u.unique&&(d=d.unique()),u.nullable||(d=d.notNull()),u.references&&(d=d.references(u.references)),u.primary&&(d=d.primaryKey()),u.default!==void 0&&(d=d.defaultTo(u.default)),d}).execute().catch(l=>{throw console.error("Error adding column",s,l),l}),u.index&&await this.db.schema.createIndex(`${r}_${s}_index`).on(r).column(s).execute().catch(l=>{})),(c==null?void 0:c.columns.find(l=>l.name===s))||await this.db.schema.alterTable(o).addColumn(s,"varchar",l=>{let d=l;return u.primary&&(d=d.primaryKey().references(`${r}.${s}`)),d}).execute();}}}async rawFindById(e,t,r){let a=await this.db.selectFrom(e).where("id","=",t).selectAll(e).select(o=>postgres.jsonObjectFrom(o.selectFrom(`${e}_meta`).selectAll(`${e}_meta`).whereRef(`${e}_meta.id`,"=",`${e}.id`)).as("_meta"));a=this.applyInclude(e,a,r);let i=await a.executeTakeFirst();if(i)return this.convertToMaterializedLiveType(i)}async findOne(e,t,r){let a=await this.rawFindById(e.name,t,r==null?void 0:r.include);if(a)return x(a)}async rawFind(e,t,r){let a=this.db.selectFrom(e).selectAll(e).select(s=>postgres.jsonObjectFrom(s.selectFrom(`${e}_meta`).selectAll(`${e}_meta`).whereRef(`${e}_meta.id`,"=",`${e}.id`)).as("_meta"));a=this.applyWhere(e,a,t),a=this.applyInclude(e,a,r);let i=await a.execute(),o=Object.fromEntries(i.map(s=>{let{id:p,...m}=s;return [p,m]}));return Object.keys(o).length===0?{}:Object.entries(o).reduce((s,[p,m])=>(s[p]=this.convertToMaterializedLiveType(m),s),{})}async find(e,t){let r=await this.rawFind(e.name,t==null?void 0:t.where,t==null?void 0:t.include);return Object.fromEntries(Object.entries(r).map(([a,i])=>[a,x(i)]))}async rawUpsert(e,t,r){return await this.db.transaction().execute(async a=>{var s;let i=!!await a.selectFrom(e).select("id").where("id","=",t).executeTakeFirst(),o={},c={};for(let[p,m]of Object.entries(r.value)){let u=(s=m._meta)==null?void 0:s.timestamp;u&&(o[p]=m.value,c[p]=u);}i?await Promise.all([a.updateTable(e).set(o).where("id","=",t).execute(),a.updateTable(`${e}_meta`).set(c).where("id","=",t).execute()]):await Promise.all([a.insertInto(e).values({...o,id:t}).execute(),a.insertInto(`${e}_meta`).values({...c,id:t}).execute()]);}),r}convertToMaterializedLiveType(e){if(!e._meta)throw new Error("Missing _meta");return {value:Object.entries(e).reduce((t,[r,a])=>{var i,o,c;return r==="_meta"||(r==="id"?t[r]={value:a}:Array.isArray(a)?t[r]={value:a.map(s=>this.convertToMaterializedLiveType(s)),_meta:{timestamp:(i=e==null?void 0:e._meta)==null?void 0:i[r]}}:typeof a=="object"&&a!==null&&!(a instanceof Date)?t[r]={...this.convertToMaterializedLiveType(a),_meta:{timestamp:(o=e==null?void 0:e._meta)==null?void 0:o[r]}}:t[r]={value:a,_meta:{timestamp:(c=e==null?void 0:e._meta)==null?void 0:c[r]}}),t},{})}}applyWhere(e,t,r){if(!r)return t;if(!this.schema)throw new Error("Schema not initialized");let a=this.schema[e];if(!a)throw new Error("Resource not found");for(let[i,o]of Object.entries(r))if(a.fields[i])t=t.where(`${e}.${i}`,"=",o);else if(a.relations[i]){let c=a.relations[i],s=c.entity.name,p=c.type==="one"?"id":c.foreignColumn,m=c.type==="one"?c.relationalColumn:"id";t=t.leftJoin(s,`${s}.${p}`,`${e}.${m}`),t=this.applyWhere(s,t,o);}return t}applyInclude(e,t,r){if(!r)return t;if(!this.schema)throw new Error("Schema not initialized");let a=this.schema[e];if(!a)throw new Error(`Resource not found: ${e}`);for(let[i,o]of Object.entries(r)){if(!a.relations[i])throw new Error(`Relation ${i} not found in resource ${e}`);let c=a.relations[i],s=c.entity.name,p=c.type==="one"?"id":c.foreignColumn,m=c.type==="one"?c.relationalColumn:"id",u=c.type==="one"?postgres.jsonObjectFrom:postgres.jsonArrayFrom;t=t.select(y=>u(y.selectFrom(s).selectAll(s).whereRef(`${s}.${p}`,"=",`${e}.${m}`).select(l=>postgres.jsonObjectFrom(l.selectFrom(`${s}_meta`).selectAll(`${s}_meta`).whereRef(`${s}_meta.id`,"=",`${s}.id`)).as("_meta"))).as(i));}return t}};var G=class n{router;storage;schema;middlewares=new Set;contextProvider;mutationSubscriptions=new Set;constructor(e){var t;this.router=e.router,this.storage=e.storage,this.schema=e.schema,(t=e.middlewares)==null||t.forEach(r=>this.middlewares.add(r)),this.storage.updateSchema(this.schema),this.contextProvider=e.contextProvider;}static create(e){return new n(e)}subscribeToMutations(e){return this.mutationSubscriptions.add(e),()=>{this.mutationSubscriptions.delete(e);}}async handleRequest(e){if(!this.router.routes[e.req.resourceName])throw new Error("Invalid resource");let t=await Array.from(this.middlewares.values()).reduceRight((r,a)=>i=>a({req:i,next:r}),async r=>this.router.routes[e.req.resourceName].handleRequest({req:r,db:this.storage,schema:this.schema}))(e.req);return t&&e.req.type==="MUTATE"&&t.acceptedValues&&Object.keys(t.acceptedValues).length>0&&this.mutationSubscriptions.forEach(r=>{r({id:e.req.context.messageId,type:"MUTATE",resource:e.req.resourceName,payload:t.acceptedValues??{},resourceId:e.req.resourceId});}),t}use(e){return this.middlewares.add(e),this}context(e){return this.contextProvider=e,this}},xn=G.create;
2
- exports.Route=$;exports.RouteFactory=q;exports.Router=D;exports.SQLStorage=ye;exports.Server=G;exports.Storage=H;exports.expressAdapter=Et;exports.routeFactory=Vt;exports.router=Pt;exports.server=xn;
1
+ 'use strict';var je=require('qs'),zod=require('zod'),j=require('crypto'),kysely=require('kysely'),postgres=require('kysely/helpers/postgres');function _interopDefault(e){return e&&e.__esModule?e:{default:e}}var je__default=/*#__PURE__*/_interopDefault(je);var j__default=/*#__PURE__*/_interopDefault(j);var pe=Object.create;var Q=Object.defineProperty;var me=Object.getOwnPropertyDescriptor;var Te=Object.getOwnPropertyNames;var fe=Object.getPrototypeOf,he=Object.prototype.hasOwnProperty;var ge=(n,e)=>()=>(e||n((e={exports:{}}).exports,e),e.exports);var Re=(n,e,t,r)=>{if(e&&typeof e=="object"||typeof e=="function")for(let a of Te(e))!he.call(n,a)&&a!==t&&Q(n,a,{get:()=>e[a],enumerable:!(r=me(e,a))||r.enumerable});return n};var B=(n,e,t)=>(t=n!=null?pe(fe(n)):{},Re(Q(t,"default",{value:n,enumerable:true}),n));var P=ge(E=>{Object.defineProperty(E,"__esModule",{value:true});E.parse=Ie;E.serialize=Le;var ve=/^[\u0021-\u003A\u003C\u003E-\u007E]+$/,xe=/^[\u0021-\u003A\u003C-\u007E]*$/,be=/^([.]?[a-z0-9]([a-z0-9-]{0,61}[a-z0-9])?)([.][a-z0-9]([a-z0-9-]{0,61}[a-z0-9])?)*$/i,we=/^[\u0020-\u003A\u003D-\u007E]*$/,Me=Object.prototype.toString,Se=(()=>{let n=function(){};return n.prototype=Object.create(null),n})();function Ie(n,e){let t=new Se,r=n.length;if(r<2)return t;let a=(e==null?void 0:e.decode)||Ae,i=0;do{let o=n.indexOf("=",i);if(o===-1)break;let c=n.indexOf(";",i),s=c===-1?r:c;if(o>s){i=n.lastIndexOf(";",o-1)+1;continue}let T=J(n,i,o),m=Y(n,o,T),d=n.slice(T,m);if(t[d]===void 0){let y=J(n,o+1,s),u=Y(n,s,y),l=a(n.slice(y,u));t[d]=l;}i=s+1;}while(i<r);return t}function J(n,e,t){do{let r=n.charCodeAt(e);if(r!==32&&r!==9)return e}while(++e<t);return t}function Y(n,e,t){for(;e>t;){let r=n.charCodeAt(--e);if(r!==32&&r!==9)return e+1}return t}function Le(n,e,t){let r=(t==null?void 0:t.encode)||encodeURIComponent;if(!ve.test(n))throw new TypeError(`argument name is invalid: ${n}`);let a=r(e);if(!xe.test(a))throw new TypeError(`argument val is invalid: ${e}`);let i=n+"="+a;if(!t)return i;if(t.maxAge!==void 0){if(!Number.isInteger(t.maxAge))throw new TypeError(`option maxAge is invalid: ${t.maxAge}`);i+="; Max-Age="+t.maxAge;}if(t.domain){if(!be.test(t.domain))throw new TypeError(`option domain is invalid: ${t.domain}`);i+="; Domain="+t.domain;}if(t.path){if(!we.test(t.path))throw new TypeError(`option path is invalid: ${t.path}`);i+="; Path="+t.path;}if(t.expires){if(!Ee(t.expires)||!Number.isFinite(t.expires.valueOf()))throw new TypeError(`option expires is invalid: ${t.expires}`);i+="; Expires="+t.expires.toUTCString();}if(t.httpOnly&&(i+="; HttpOnly"),t.secure&&(i+="; Secure"),t.partitioned&&(i+="; Partitioned"),t.priority)switch(typeof t.priority=="string"?t.priority.toLowerCase():void 0){case "low":i+="; Priority=Low";break;case "medium":i+="; Priority=Medium";break;case "high":i+="; Priority=High";break;default:throw new TypeError(`option priority is invalid: ${t.priority}`)}if(t.sameSite)switch(typeof t.sameSite=="string"?t.sameSite.toLowerCase():t.sameSite){case true:case "strict":i+="; SameSite=Strict";break;case "lax":i+="; SameSite=Lax";break;case "none":i+="; SameSite=None";break;default:throw new TypeError(`option sameSite is invalid: ${t.sameSite}`)}return i}function Ae(n){if(n.indexOf("%")===-1)return n;try{return decodeURIComponent(n)}catch{return n}}function Ee(n){return Me.call(n)==="[object Date]"}});var te=B(P());var O=zod.z.object({resource:zod.z.string(),where:zod.z.record(zod.z.any()).optional(),include:zod.z.record(zod.z.any()).optional(),lastSyncedAt:zod.z.string().optional(),limit:zod.z.number().optional(),sort:zod.z.array(zod.z.object({key:zod.z.string(),direction:zod.z.enum(["asc","desc"])})).optional()}),V=zod.z.record(zod.z.object({value:zod.z.string().or(zod.z.number()).or(zod.z.boolean()).or(zod.z.date()).nullable(),_meta:zod.z.object({timestamp:zod.z.string().optional().nullable()}).optional()})).superRefine((n,e)=>{n.id&&e.addIssue({code:zod.z.ZodIssueCode.custom,message:"Payload cannot have an id"});}),X=zod.z.object({id:zod.z.string().optional(),type:zod.z.literal("MUTATE"),resource:zod.z.string()}),S=X.extend({procedure:zod.z.string(),payload:zod.z.any().optional()}),I=X.extend({resourceId:zod.z.string(),payload:V});zod.z.union([S,I]);var ee=O.omit({resource:true}),N=S.omit({id:true,type:true,resource:true,procedure:true}),F=I.omit({id:true,type:true,resource:true});zod.z.union([F,N]);var ne=n=>async e=>{var t;try{let r=typeof e.headers.getSetCookie=="function"?Object.fromEntries(e.headers):e.headers,a={headers:r,cookies:r.cookie?te.default.parse(r.cookie):{}},i=new URL(e.url),o=i.pathname.split("/"),c=i.searchParams,s=je__default.default.parse(c.toString()),T=await((t=n.contextProvider)==null?void 0:t.call(n,{transport:"HTTP",headers:a.headers,cookies:a.cookies,query:s}))??{};if(e.method==="GET"){let m=o[o.length-1],{success:d,data:y,error:u}=ee.safeParse(s);if(!d)return Response.json({message:"Invalid query",code:"INVALID_QUERY",details:u},{status:400});let l=await n.handleRequest({req:{...a,type:"QUERY",resourceName:m,context:T,where:y.where,include:y.include,query:s}});return !l||!l.data?Response.json({message:"Invalid resource",code:"INVALID_RESOURCE"},{status:400}):Response.json(l.data)}if(e.method==="POST")try{let m=o[o.length-1],d=o[o.length-2],y=e.body?await e.json():{},u;if(m==="set"){let{success:w,data:M,error:_}=F.safeParse(y);if(!w)return Response.json({message:"Invalid mutation",code:"INVALID_REQUEST",details:_},{status:400});u=M;}else {let{success:w,data:M,error:_}=N.safeParse(y);if(!w)return Response.json({message:"Invalid mutation",code:"INVALID_REQUEST",details:_},{status:400});u=M;}let l=await n.handleRequest({req:{...a,type:"MUTATE",resourceName:d,input:u.payload,context:T,resourceId:u.resourceId,procedure:m!=="set"?m:void 0,query:{}}});return Response.json(l)}catch(m){return console.error("Error parsing mutation from the client:",m),Response.json({message:"Internal server error",code:"INTERNAL_SERVER_ERROR"},{status:500})}return Response.json({message:"Not found",code:"NOT_FOUND"},{status:404})}catch(r){return console.error("Unexpected error:",r),Response.json({message:"Internal server error",code:"INTERNAL_SERVER_ERROR"},{status:500})}};var ue=B(P());var x=zod.z.string(),Ce=zod.z.object({id:x,type:zod.z.literal("SUBSCRIBE"),resource:zod.z.string()}),_e=O.extend({id:x,type:zod.z.literal("QUERY")}),re=I.extend({id:x}),Pe=S.extend({id:x}),Ve=zod.z.union([Pe,re]),ae=zod.z.union([Ce,_e,Ve]),Ne=zod.z.object({id:x,type:zod.z.literal("REJECT"),resource:zod.z.string(),message:zod.z.string().optional()}),Fe=zod.z.object({id:x,type:zod.z.literal("REPLY"),data:zod.z.any()});zod.z.union([Ne,Fe,re]);zod.z.object({resource:zod.z.string(),data:zod.z.record(V)});var oe="0123456789ABCDEFGHJKMNPQRSTVWXYZ",L=32;var $e=16,se=10,ie=0xffffffffffff;var h;(function(n){n.Base32IncorrectEncoding="B32_ENC_INVALID",n.DecodeTimeInvalidCharacter="DEC_TIME_CHAR",n.DecodeTimeValueMalformed="DEC_TIME_MALFORMED",n.EncodeTimeNegative="ENC_TIME_NEG",n.EncodeTimeSizeExceeded="ENC_TIME_SIZE_EXCEED",n.EncodeTimeValueMalformed="ENC_TIME_MALFORMED",n.PRNGDetectFailure="PRNG_DETECT",n.ULIDInvalid="ULID_INVALID",n.Unexpected="UNEXPECTED",n.UUIDInvalid="UUID_INVALID";})(h||(h={}));var g=class extends Error{constructor(e,t){super(`${t} (${e})`),this.name="ULIDError",this.code=e;}};function Ke(n){let e=Math.floor(n()*L);return e===L&&(e=L-1),oe.charAt(e)}function De(n){var r;let e=qe(),t=e&&(e.crypto||e.msCrypto)||(typeof j__default.default<"u"?j__default.default:null);if(typeof(t==null?void 0:t.getRandomValues)=="function")return ()=>{let a=new Uint8Array(1);return t.getRandomValues(a),a[0]/255};if(typeof(t==null?void 0:t.randomBytes)=="function")return ()=>t.randomBytes(1).readUInt8()/255;if((r=j__default.default)!=null&&r.randomBytes)return ()=>j__default.default.randomBytes(1).readUInt8()/255;throw new g(h.PRNGDetectFailure,"Failed to find a reliable PRNG")}function qe(){return ze()?self:typeof window<"u"?window:typeof global<"u"?global:typeof globalThis<"u"?globalThis:null}function Ue(n,e){let t="";for(;n>0;n--)t=Ke(e)+t;return t}function ke(n,e=se){if(isNaN(n))throw new g(h.EncodeTimeValueMalformed,`Time must be a number: ${n}`);if(n>ie)throw new g(h.EncodeTimeSizeExceeded,`Cannot encode a time larger than ${ie}: ${n}`);if(n<0)throw new g(h.EncodeTimeNegative,`Time must be positive: ${n}`);if(Number.isInteger(n)===false)throw new g(h.EncodeTimeValueMalformed,`Time must be an integer: ${n}`);let t,r="";for(let a=e;a>0;a--)t=n%L,r=oe.charAt(t)+r,n=(n-t)/L;return r}function ze(){return typeof WorkerGlobalScope<"u"&&self instanceof WorkerGlobalScope}function ce(n,e){let t=De(),r=Date.now();return ke(r,se)+Ue($e,t)}var $=()=>ce().toLowerCase();var le=n=>{let e={},t={};return n.subscribeToMutations(r=>{let a=r;!a.resourceId||!a.payload||(console.log("Mutation propagated:",a),Object.entries(t[a.resource]??{}).forEach(([i,o])=>{var c;(c=e[i])==null||c.send(JSON.stringify({...a,id:a.id??$()}));}));}),(r,a)=>{var m;let i=d=>{r.send(JSON.stringify(d));},o=$(),c={headers:a.headers,cookies:typeof a.headers.cookie=="string"?ue.default.parse(a.headers.cookie):{}},s=je.parse(a.url.split("?")[1]),T=(m=n.contextProvider)==null?void 0:m.call(n,{transport:"WEBSOCKET",headers:c.headers,cookies:c.cookies,query:s});e[o]=r,console.log("Client connected:",o),r.on("message",async d=>{try{console.log("Message received from the client:",d);let y=ae.parse(JSON.parse(d.toString()));if(y.type==="SUBSCRIBE"){let{resource:u}=y;t[u]||(t[u]={}),t[u][o]={};}else if(y.type==="QUERY"){let{resource:u}=y,l=await n.handleRequest({req:{...c,type:"QUERY",resourceName:u,context:await T??{},query:s}});if(!l||!l.data)throw new Error("Invalid resource");i({id:y.id,type:"REPLY",data:{resource:u,data:Object.fromEntries(Object.entries(l.data??{}).map(([w,M])=>[w,M.value]))}});}else if(y.type==="MUTATE"){let{resource:u}=y;console.log("Received mutation from client:",y);try{let l=await n.handleRequest({req:{...c,type:"MUTATE",resourceName:u,input:y.payload,context:{messageId:y.id,...await T??{}},resourceId:y.resourceId,procedure:y.procedure,query:s}});y.procedure&&i({id:y.id,type:"REPLY",data:l});}catch(l){i({id:y.id,type:"REJECT",resource:u,message:l.message}),console.error("Error parsing mutation from the client:",l);}}}catch(y){console.error("Error handling message from the client:",y);}}),r.on("close",()=>{console.log("Connection closed",o),delete e[o];for(let d of Object.values(t))delete d[o];});}};function de(n){let e=`${n.protocol}://${n.hostname}${n.url}`,t=new Headers;return Object.entries(n.headers).forEach(([r,a])=>{a&&t.set(r,Array.isArray(a)?a.join(","):a);}),new Request(e,{method:n.method,headers:t,body:n.body&&n.method!=="GET"?JSON.stringify(n.body):void 0})}var It=(n,e,t)=>{n.ws(`${(t==null?void 0:t.basePath)??""}/ws`,le(e)),n.use(`${(t==null?void 0:t.basePath)??""}/`,(r,a)=>{ne(e)(de(r)).then(o=>o.json().then(c=>a.status(o.status).send(c)));});};var K=class n{routes;constructor(e){this.routes=e.routes;}static create(e){return new n(e)}},Et=n=>K.create({...n}),He=n=>({handler:e=>({inputValidator:n??zod.z.undefined(),handler:e})}),D=class n{_resourceSchema;resourceName;middlewares;customMutations;constructor(e,t){this.resourceName=e,this.middlewares=new Set,this.customMutations=t??{};}handleFind=async({req:e,db:t})=>({data:await t.rawFind(e.resourceName,e.where,e.include),acceptedValues:null});handleSet=async({req:e,db:t,schema:r})=>{if(!e.input)throw new Error("Payload is required");if(!e.resourceId)throw new Error("ResourceId is required");let a=await t.rawFindById(e.resourceName,e.resourceId),[i,o]=r[this.resourceName].mergeMutation("set",e.input,a);if(!o)throw new Error("Mutation rejected");return {data:await t.rawUpsert(e.resourceName,e.resourceId,i),acceptedValues:o}};async handleRequest(e){let t=r=>(()=>{if(r.type==="QUERY")return this.handleFind({req:r,db:e.db,schema:e.schema});if(r.type==="MUTATE")if(r.procedure){if(this.customMutations[r.procedure]){let a=this.customMutations[r.procedure].inputValidator.parse(r.input);return r.input=a,this.customMutations[r.procedure].handler({req:r,db:e.db,schema:e.schema})}}else return this.handleSet({req:r,db:e.db,schema:e.schema});throw new Error("Invalid request")})();return await Array.from(this.middlewares.values()).reduceRight((r,a)=>i=>a({req:i,next:r}),async r=>t(r))(e.req)}use(...e){for(let t of e)this.middlewares.add(t);return this}withMutations(e){return new n(this.resourceName,e({mutation:He}))}},q=class n{middlewares;constructor(e=[]){this.middlewares=e;}createBasicRoute(e){return new D(e.name).use(...this.middlewares)}use(...e){return new n([...this.middlewares,...e])}static create(){return new n}},Ot=q.create;var v=n=>{if(n)return Array.isArray(n.value)?n.value.map(e=>v(e)):typeof n.value!="object"||n.value===null||n.value instanceof Date?n.value:Object.fromEntries(Object.entries(n.value).map(([e,t])=>[e,v(t)]))};var H=class{async insert(e,t){let r=new Date().toISOString();return v(await this.rawUpsert(e.name,t.id,{value:Object.fromEntries(Object.entries(t).map(([a,i])=>[a,{value:i,_meta:{timestamp:r}}]))}))}async update(e,t,r){let a=new Date().toISOString(),{id:i,...o}=r;return v(await this.rawUpsert(e.name,t,{value:Object.fromEntries(Object.entries(o).map(([c,s])=>[c,{value:s,_meta:{timestamp:a}}]))}))}},ye=class extends H{db;schema;constructor(e){super(),this.db=new kysely.Kysely({dialect:new kysely.PostgresDialect({pool:e})});}async updateSchema(e){this.schema=e;let t=await this.db.introspection.getTables();for(let[r,a]of Object.entries(e)){let i=t.find(s=>s.name===r);i||await this.db.schema.createTable(r).ifNotExists().execute();let o=`${r}_meta`,c=t.find(s=>s.name===o);c||await this.db.schema.createTable(o).ifNotExists().execute();for(let[s,T]of Object.entries(a.fields)){let m=i==null?void 0:i.columns.find(u=>u.name===s),d=T.getStorageFieldType();m?m.dataType!==d.type&&console.error("Column type mismatch:",s,"expected to have type:",d.type,"but has type:",m.dataType):(await this.db.schema.alterTable(r).addColumn(s,d.type,u=>{let l=u;return d.unique&&(l=l.unique()),d.nullable||(l=l.notNull()),d.references&&(l=l.references(d.references)),d.primary&&(l=l.primaryKey()),d.default!==void 0&&(l=l.defaultTo(d.default)),l}).execute().catch(u=>{throw console.error("Error adding column",s,u),u}),d.index&&await this.db.schema.createIndex(`${r}_${s}_index`).on(r).column(s).execute().catch(u=>{})),(c==null?void 0:c.columns.find(u=>u.name===s))||await this.db.schema.alterTable(o).addColumn(s,"varchar",u=>{let l=u;return d.primary&&(l=l.primaryKey().references(`${r}.${s}`)),l}).execute();}}}async rawFindById(e,t,r){let a=await this.db.selectFrom(e).where("id","=",t).selectAll(e).select(o=>postgres.jsonObjectFrom(o.selectFrom(`${e}_meta`).selectAll(`${e}_meta`).whereRef(`${e}_meta.id`,"=",`${e}.id`)).as("_meta"));a=this.applyInclude(e,a,r);let i=await a.executeTakeFirst();if(i)return this.convertToMaterializedLiveType(i)}async findOne(e,t,r){let a=await this.rawFindById(e.name,t,r==null?void 0:r.include);if(a)return v(a)}async rawFind(e,t,r){let a=this.db.selectFrom(e).selectAll(e).select(s=>postgres.jsonObjectFrom(s.selectFrom(`${e}_meta`).selectAll(`${e}_meta`).whereRef(`${e}_meta.id`,"=",`${e}.id`)).as("_meta"));a=this.applyWhere(e,a,t),a=this.applyInclude(e,a,r);let i=await a.execute(),o=Object.fromEntries(i.map(s=>{let{id:T,...m}=s;return [T,m]}));return Object.keys(o).length===0?{}:Object.entries(o).reduce((s,[T,m])=>(s[T]=this.convertToMaterializedLiveType(m),s),{})}async find(e,t){let r=await this.rawFind(e.name,t==null?void 0:t.where,t==null?void 0:t.include);return Object.fromEntries(Object.entries(r).map(([a,i])=>[a,v(i)]))}async rawUpsert(e,t,r){return await this.db.transaction().execute(async a=>{var s;let i=!!await a.selectFrom(e).select("id").where("id","=",t).executeTakeFirst(),o={},c={};for(let[T,m]of Object.entries(r.value)){let d=(s=m._meta)==null?void 0:s.timestamp;d&&(o[T]=m.value,c[T]=d);}i?await Promise.all([a.updateTable(e).set(o).where("id","=",t).execute(),a.updateTable(`${e}_meta`).set(c).where("id","=",t).execute()]):await Promise.all([a.insertInto(e).values({...o,id:t}).execute(),a.insertInto(`${e}_meta`).values({...c,id:t}).execute()]);}),r}convertToMaterializedLiveType(e){if(!e._meta)throw new Error("Missing _meta");return {value:Object.entries(e).reduce((t,[r,a])=>{var i,o,c;return r==="_meta"||(r==="id"?t[r]={value:a}:Array.isArray(a)?t[r]={value:a.map(s=>this.convertToMaterializedLiveType(s)),_meta:{timestamp:(i=e==null?void 0:e._meta)==null?void 0:i[r]}}:typeof a=="object"&&a!==null&&!(a instanceof Date)?t[r]={...this.convertToMaterializedLiveType(a),_meta:{timestamp:(o=e==null?void 0:e._meta)==null?void 0:o[r]}}:t[r]={value:a,_meta:{timestamp:(c=e==null?void 0:e._meta)==null?void 0:c[r]}}),t},{})}}applyWhere(e,t,r){if(!r)return t;if(!this.schema)throw new Error("Schema not initialized");let a=this.schema[e];if(!a)throw new Error("Resource not found");for(let[i,o]of Object.entries(r))if(a.fields[i])t=t.where(`${e}.${i}`,"=",o);else if(a.relations[i]){let c=a.relations[i],s=c.entity.name,T=c.type==="one"?"id":c.foreignColumn,m=c.type==="one"?c.relationalColumn:"id";t=t.leftJoin(s,`${s}.${T}`,`${e}.${m}`),t=this.applyWhere(s,t,o);}return t}applyInclude(e,t,r){if(!r)return t;if(!this.schema)throw new Error("Schema not initialized");let a=this.schema[e];if(!a)throw new Error(`Resource not found: ${e}`);for(let[i,o]of Object.entries(r)){if(!a.relations[i])throw new Error(`Relation ${i} not found in resource ${e}`);let c=a.relations[i],s=c.entity.name,T=c.type==="one"?"id":c.foreignColumn,m=c.type==="one"?c.relationalColumn:"id",d=c.type==="one"?postgres.jsonObjectFrom:postgres.jsonArrayFrom;t=t.select(y=>d(y.selectFrom(s).selectAll(s).whereRef(`${s}.${T}`,"=",`${e}.${m}`).select(u=>postgres.jsonObjectFrom(u.selectFrom(`${s}_meta`).selectAll(`${s}_meta`).whereRef(`${s}_meta.id`,"=",`${s}.id`)).as("_meta"))).as(i));}return t}};var G=class n{router;storage;schema;middlewares=new Set;contextProvider;mutationSubscriptions=new Set;constructor(e){var t;this.router=e.router,this.storage=e.storage,this.schema=e.schema,(t=e.middlewares)==null||t.forEach(r=>{this.middlewares.add(r);}),this.storage.updateSchema(this.schema),this.contextProvider=e.contextProvider;}static create(e){return new n(e)}subscribeToMutations(e){return this.mutationSubscriptions.add(e),()=>{this.mutationSubscriptions.delete(e);}}async handleRequest(e){if(!this.router.routes[e.req.resourceName])throw new Error("Invalid resource");let t=await Array.from(this.middlewares.values()).reduceRight((r,a)=>i=>a({req:i,next:r}),async r=>this.router.routes[e.req.resourceName].handleRequest({req:r,db:this.storage,schema:this.schema}))(e.req);return t&&e.req.type==="MUTATE"&&t.acceptedValues&&Object.keys(t.acceptedValues).length>0&&this.mutationSubscriptions.forEach(r=>{r({id:e.req.context.messageId,type:"MUTATE",resource:e.req.resourceName,payload:t.acceptedValues??{},resourceId:e.req.resourceId});}),t}use(e){return this.middlewares.add(e),this}context(e){return this.contextProvider=e,this}},Bt=G.create;
2
+ exports.Route=D;exports.RouteFactory=q;exports.Router=K;exports.SQLStorage=ye;exports.Server=G;exports.Storage=H;exports.expressAdapter=It;exports.routeFactory=Ot;exports.router=Et;exports.server=Bt;
package/dist/server.d.cts CHANGED
@@ -144,7 +144,7 @@ declare class RouteFactory {
144
144
  declare const routeFactory: typeof RouteFactory.create;
145
145
  type AnyRoute = Route<LiveObjectAny, Middleware<any>, Record<string, any>>;
146
146
 
147
- type Simplify<T> = T extends Record<string, any> ? {
147
+ type Simplify<T> = T extends Record<string, unknown> ? {
148
148
  [K in keyof T]: Simplify<T[K]>;
149
149
  } : T;
150
150
 
package/dist/server.d.ts CHANGED
@@ -144,7 +144,7 @@ declare class RouteFactory {
144
144
  declare const routeFactory: typeof RouteFactory.create;
145
145
  type AnyRoute = Route<LiveObjectAny, Middleware<any>, Record<string, any>>;
146
146
 
147
- type Simplify<T> = T extends Record<string, any> ? {
147
+ type Simplify<T> = T extends Record<string, unknown> ? {
148
148
  [K in keyof T]: Simplify<T[K]>;
149
149
  } : T;
150
150
 
package/dist/server.js CHANGED
@@ -1,2 +1,2 @@
1
- import {a,b as b$1,r}from'./chunk-ECP6AHZL.js';import Te,{parse}from'qs';import {z as z$1}from'zod';import A from'node:crypto';import {Kysely,PostgresDialect}from'kysely';import {jsonObjectFrom,jsonArrayFrom}from'kysely/helpers/postgres';var O=a(E=>{Object.defineProperty(E,"__esModule",{value:true});E.parse=me;E.serialize=pe;var oe=/^[\u0021-\u003A\u003C\u003E-\u007E]+$/,se=/^[\u0021-\u003A\u003C-\u007E]*$/,ce=/^([.]?[a-z0-9]([a-z0-9-]{0,61}[a-z0-9])?)([.][a-z0-9]([a-z0-9-]{0,61}[a-z0-9])?)*$/i,ue=/^[\u0020-\u003A\u003D-\u007E]*$/,de=Object.prototype.toString,le=(()=>{let r=function(){};return r.prototype=Object.create(null),r})();function me(r,e){let t=new le,n=r.length;if(n<2)return t;let a=(e==null?void 0:e.decode)||ye,i=0;do{let s=r.indexOf("=",i);if(s===-1)break;let c=r.indexOf(";",i),o=c===-1?n:c;if(s>o){i=r.lastIndexOf(";",s-1)+1;continue}let h=V(r,i,s),p=F(r,s,h),u=r.slice(h,p);if(t[u]===void 0){let m=V(r,s+1,o),d=F(r,o,m),l=a(r.slice(m,d));t[u]=l;}i=o+1;}while(i<n);return t}function V(r,e,t){do{let n=r.charCodeAt(e);if(n!==32&&n!==9)return e}while(++e<t);return t}function F(r,e,t){for(;e>t;){let n=r.charCodeAt(--e);if(n!==32&&n!==9)return e+1}return t}function pe(r,e,t){let n=(t==null?void 0:t.encode)||encodeURIComponent;if(!oe.test(r))throw new TypeError(`argument name is invalid: ${r}`);let a=n(e);if(!se.test(a))throw new TypeError(`argument val is invalid: ${e}`);let i=r+"="+a;if(!t)return i;if(t.maxAge!==void 0){if(!Number.isInteger(t.maxAge))throw new TypeError(`option maxAge is invalid: ${t.maxAge}`);i+="; Max-Age="+t.maxAge;}if(t.domain){if(!ce.test(t.domain))throw new TypeError(`option domain is invalid: ${t.domain}`);i+="; Domain="+t.domain;}if(t.path){if(!ue.test(t.path))throw new TypeError(`option path is invalid: ${t.path}`);i+="; Path="+t.path;}if(t.expires){if(!fe(t.expires)||!Number.isFinite(t.expires.valueOf()))throw new TypeError(`option expires is invalid: ${t.expires}`);i+="; Expires="+t.expires.toUTCString();}if(t.httpOnly&&(i+="; HttpOnly"),t.secure&&(i+="; Secure"),t.partitioned&&(i+="; Partitioned"),t.priority)switch(typeof t.priority=="string"?t.priority.toLowerCase():void 0){case "low":i+="; Priority=Low";break;case "medium":i+="; Priority=Medium";break;case "high":i+="; Priority=High";break;default:throw new TypeError(`option priority is invalid: ${t.priority}`)}if(t.sameSite)switch(typeof t.sameSite=="string"?t.sameSite.toLowerCase():t.sameSite){case true:case "strict":i+="; SameSite=Strict";break;case "lax":i+="; SameSite=Lax";break;case "none":i+="; SameSite=None";break;default:throw new TypeError(`option sameSite is invalid: ${t.sameSite}`)}return i}function ye(r){if(r.indexOf("%")===-1)return r;try{return decodeURIComponent(r)}catch{return r}}function fe(r){return de.call(r)==="[object Date]"}});var G=b$1(O(),1);var q=z$1.object({type:z$1.literal("QUERY"),resource:z$1.string(),where:z$1.record(z$1.any()).optional(),include:z$1.record(z$1.any()).optional()}),L=z$1.record(z$1.object({value:z$1.string().or(z$1.number()).or(z$1.boolean()).or(z$1.date()).nullable(),_meta:z$1.object({timestamp:z$1.string().optional().nullable()}).optional()})).superRefine((r,e)=>{r.id&&e.addIssue({code:z$1.ZodIssueCode.custom,message:"Payload cannot have an id"});}),Z=z$1.object({id:z$1.string().optional(),type:z$1.literal("MUTATE"),resource:z$1.string()}),M=Z.extend({procedure:z$1.string(),payload:z$1.any().optional()}),I=Z.extend({resourceId:z$1.string(),payload:L});z$1.union([M,I]);var H=q.omit({type:true,resource:true}),P=M.omit({id:true,type:true,resource:true,procedure:true}),C=I.omit({id:true,type:true,resource:true});z$1.union([C,P]);var B=r=>async e=>{var t;try{let n=typeof e.headers.getSetCookie=="function"?Object.fromEntries(e.headers):e.headers,a={headers:n,cookies:n.cookie?G.default.parse(n.cookie):{}},i=new URL(e.url),s=i.pathname.split("/"),c=i.searchParams,o=Te.parse(c.toString()),h=await((t=r.contextProvider)==null?void 0:t.call(r,{transport:"HTTP",headers:a.headers,cookies:a.cookies,query:o}))??{};if(e.method==="GET"){let p=s[s.length-1],{success:u,data:m,error:d}=H.safeParse(o);if(!u)return Response.json({message:"Invalid query",code:"INVALID_QUERY",details:d},{status:400});let l=await r.handleRequest({req:{...a,type:"QUERY",resourceName:p,context:h,where:m.where,include:m.include,query:o}});return !l||!l.data?Response.json({message:"Invalid resource",code:"INVALID_RESOURCE"},{status:400}):Response.json(l.data)}if(e.method==="POST")try{let p=s[s.length-1],u=s[s.length-2],m=e.body?await e.json():{},d;if(p==="set"){let{success:w,data:T,error:x}=C.safeParse(m);if(!w)return Response.json({message:"Invalid mutation",code:"INVALID_REQUEST",details:x},{status:400});d=T;}else {let{success:w,data:T,error:x}=P.safeParse(m);if(!w)return Response.json({message:"Invalid mutation",code:"INVALID_REQUEST",details:x},{status:400});d=T;}let l=await r.handleRequest({req:{...a,type:"MUTATE",resourceName:u,input:d.payload,context:h,resourceId:d.resourceId,procedure:p!=="set"?p:void 0,query:{}}});return Response.json(l)}catch(p){return console.error("Error parsing mutation from the client:",p),Response.json({message:"Internal server error",code:"INTERNAL_SERVER_ERROR"},{status:500})}return Response.json({message:"Not found",code:"NOT_FOUND"},{status:404})}catch(n){return console.error("Unexpected error:",n),Response.json({message:"Internal server error",code:"INTERNAL_SERVER_ERROR"},{status:500})}};var ee=b$1(O(),1);var g=z$1.string(),ge=z$1.object({id:g,type:z$1.literal("SUBSCRIBE"),resource:z$1.string()}),Re=z$1.object({id:g,type:z$1.literal("SYNC"),lastSyncedAt:z$1.string().optional(),resources:z$1.string().array().optional(),where:z$1.record(z$1.any()).optional()}),Q=I.extend({id:g}),be=M.extend({id:g}),we=z$1.union([be,Q]),W=z$1.union([ge,Re,we]),xe=z$1.object({id:g,type:z$1.literal("SYNC"),resource:z$1.string(),data:z$1.record(L)}),Se=z$1.object({id:g,type:z$1.literal("REJECT"),resource:z$1.string(),message:z$1.string().optional()}),Me=z$1.object({id:g,type:z$1.literal("REPLY"),data:z$1.any()});z$1.union([xe,Se,Me,Q]);var Y="0123456789ABCDEFGHJKMNPQRSTVWXYZ",v=32;var Ie=16,J=10,K=0xffffffffffff;var R;(function(r){r.Base32IncorrectEncoding="B32_ENC_INVALID",r.DecodeTimeInvalidCharacter="DEC_TIME_CHAR",r.DecodeTimeValueMalformed="DEC_TIME_MALFORMED",r.EncodeTimeNegative="ENC_TIME_NEG",r.EncodeTimeSizeExceeded="ENC_TIME_SIZE_EXCEED",r.EncodeTimeValueMalformed="ENC_TIME_MALFORMED",r.PRNGDetectFailure="PRNG_DETECT",r.ULIDInvalid="ULID_INVALID",r.Unexpected="UNEXPECTED",r.UUIDInvalid="UUID_INVALID";})(R||(R={}));var b=class extends Error{constructor(e,t){super(`${t} (${e})`),this.name="ULIDError",this.code=e;}};function ve(r){let e=Math.floor(r()*v);return e===v&&(e=v-1),Y.charAt(e)}function Ee(r){var n;let e=Ae(),t=e&&(e.crypto||e.msCrypto)||(typeof A<"u"?A:null);if(typeof(t==null?void 0:t.getRandomValues)=="function")return ()=>{let a=new Uint8Array(1);return t.getRandomValues(a),a[0]/255};if(typeof(t==null?void 0:t.randomBytes)=="function")return ()=>t.randomBytes(1).readUInt8()/255;if((n=A)!=null&&n.randomBytes)return ()=>A.randomBytes(1).readUInt8()/255;throw new b(R.PRNGDetectFailure,"Failed to find a reliable PRNG")}function Ae(){return Le()?self:typeof window<"u"?window:typeof global<"u"?global:typeof globalThis<"u"?globalThis:null}function je(r,e){let t="";for(;r>0;r--)t=ve(e)+t;return t}function Oe(r,e=J){if(isNaN(r))throw new b(R.EncodeTimeValueMalformed,`Time must be a number: ${r}`);if(r>K)throw new b(R.EncodeTimeSizeExceeded,`Cannot encode a time larger than ${K}: ${r}`);if(r<0)throw new b(R.EncodeTimeNegative,`Time must be positive: ${r}`);if(Number.isInteger(r)===false)throw new b(R.EncodeTimeValueMalformed,`Time must be an integer: ${r}`);let t,n="";for(let a=e;a>0;a--)t=r%v,n=Y.charAt(t)+n,r=(r-t)/v;return n}function Le(){return typeof WorkerGlobalScope<"u"&&self instanceof WorkerGlobalScope}function X(r,e){let t=Ee(),n=Date.now();return Oe(n,J)+je(Ie,t)}var N=()=>X().toLowerCase();var te=r=>{let e={},t={};return r.subscribeToMutations(n=>{let a=n;!a.resourceId||!a.payload||(console.log("Mutation propagated:",a),Object.entries(t[a.resource]??{}).forEach(([i,s])=>{var c;(c=e[i])==null||c.send(JSON.stringify({...a,id:a.id??N()}));}));}),(n,a)=>{var p;let i=u=>{n.send(JSON.stringify(u));},s=N(),c={headers:a.headers,cookies:typeof a.headers.cookie=="string"?ee.default.parse(a.headers.cookie):{}},o=parse(a.url.split("?")[1]),h=(p=r.contextProvider)==null?void 0:p.call(r,{transport:"WEBSOCKET",headers:c.headers,cookies:c.cookies,query:o});e[s]=n,console.log("Client connected:",s),n.on("message",async u=>{try{console.log("Message received from the client:",u);let m=W.parse(JSON.parse(u.toString()));if(m.type==="SUBSCRIBE"){let{resource:d}=m;t[d]||(t[d]={}),t[d][s]={};}else if(m.type==="SYNC"){let{resources:d}=m,l=d??Object.keys(r.schema);console.log("Syncing resources:",l),await Promise.all(l.map(async w=>{let T=await r.handleRequest({req:{...c,type:"QUERY",resourceName:w,context:await h??{},query:o}});if(!T||!T.data)throw new Error("Invalid resource");i({id:m.id,type:"SYNC",resource:w,data:Object.fromEntries(Object.entries(T.data??{}).map(([x,ae])=>[x,ae.value]))});}));}else if(m.type==="MUTATE"){let{resource:d}=m;console.log("Received mutation from client:",m);try{let l=await r.handleRequest({req:{...c,type:"MUTATE",resourceName:d,input:m.payload,context:{messageId:m.id,...await h??{}},resourceId:m.resourceId,procedure:m.procedure,query:o}});m.procedure&&i({id:m.id,type:"REPLY",data:l});}catch(l){i({id:m.id,type:"REJECT",resource:d,message:l.message}),console.error("Error parsing mutation from the client:",l);}}}catch(m){console.error("Error handling message from the client:",m);}}),n.on("close",()=>{console.log("Connection closed",s),delete e[s];for(let u of Object.values(t))delete u[s];});}};function re(r){let e=`${r.protocol}://${r.hostname}${r.url}`,t=new Headers;return Object.entries(r.headers).forEach(([n,a])=>{a&&t.set(n,Array.isArray(a)?a.join(","):a);}),new Request(e,{method:r.method,headers:t,body:r.body&&r.method!=="GET"?JSON.stringify(r.body):void 0})}var yt=(r,e,t)=>{r.ws(`${(t==null?void 0:t.basePath)??""}/ws`,te(e)),r.use(`${(t==null?void 0:t.basePath)??""}/`,(n,a)=>{B(e)(re(n)).then(s=>s.json().then(c=>a.status(s.status).send(c)));});};var _=class r{routes;constructor(e){this.routes=e.routes;}static create(e){return new r(e)}},gt=r=>_.create({...r}),Ne=r=>({handler:e=>({inputValidator:r??z$1.undefined(),handler:e})}),$=class r{_resourceSchema;resourceName;middlewares;customMutations;constructor(e,t){this.resourceName=e,this.middlewares=new Set,this.customMutations=t??{};}handleFind=async({req:e,db:t})=>({data:await t.rawFind(e.resourceName,e.where,e.include),acceptedValues:null});handleSet=async({req:e,db:t,schema:n})=>{if(!e.input)throw new Error("Payload is required");if(!e.resourceId)throw new Error("ResourceId is required");let a=await t.rawFindById(e.resourceName,e.resourceId),[i,s]=n[this.resourceName].mergeMutation("set",e.input,a);if(!s)throw new Error("Mutation rejected");return {data:await t.rawUpsert(e.resourceName,e.resourceId,i),acceptedValues:s}};async handleRequest(e){let t=n=>(()=>{if(n.type==="QUERY")return this.handleFind({req:n,db:e.db,schema:e.schema});if(n.type==="MUTATE")if(n.procedure){if(this.customMutations[n.procedure]){let a=this.customMutations[n.procedure].inputValidator.parse(n.input);return n.input=a,this.customMutations[n.procedure].handler({req:n,db:e.db,schema:e.schema})}}else return this.handleSet({req:n,db:e.db,schema:e.schema});throw new Error("Invalid request")})();return await Array.from(this.middlewares.values()).reduceRight((n,a)=>i=>a({req:i,next:n}),async n=>t(n))(e.req)}use(...e){for(let t of e)this.middlewares.add(t);return this}withMutations(e){return new r(this.resourceName,e({mutation:Ne}))}},U=class r{middlewares;constructor(e=[]){this.middlewares=e;}createBasicRoute(e){return new $(e.name).use(...this.middlewares)}use(...e){return new r([...this.middlewares,...e])}static create(){return new r}},Rt=U.create;var D=class{async insert(e,t){let n=new Date().toISOString();return r(await this.rawUpsert(e.name,t.id,{value:Object.fromEntries(Object.entries(t).map(([a,i])=>[a,{value:i,_meta:{timestamp:n}}]))}))}async update(e,t,n){let a=new Date().toISOString(),{id:i,...s}=n;return r(await this.rawUpsert(e.name,t,{value:Object.fromEntries(Object.entries(s).map(([c,o])=>[c,{value:o,_meta:{timestamp:a}}]))}))}},ne=class extends D{db;schema;constructor(e){super(),this.db=new Kysely({dialect:new PostgresDialect({pool:e})});}async updateSchema(e){this.schema=e;let t=await this.db.introspection.getTables();for(let[n,a]of Object.entries(e)){let i=t.find(o=>o.name===n);i||await this.db.schema.createTable(n).ifNotExists().execute();let s=`${n}_meta`,c=t.find(o=>o.name===s);c||await this.db.schema.createTable(s).ifNotExists().execute();for(let[o,h]of Object.entries(a.fields)){let p=i==null?void 0:i.columns.find(d=>d.name===o),u=h.getStorageFieldType();p?p.dataType!==u.type&&console.error("Column type mismatch:",o,"expected to have type:",u.type,"but has type:",p.dataType):(await this.db.schema.alterTable(n).addColumn(o,u.type,d=>{let l=d;return u.unique&&(l=l.unique()),u.nullable||(l=l.notNull()),u.references&&(l=l.references(u.references)),u.primary&&(l=l.primaryKey()),u.default!==void 0&&(l=l.defaultTo(u.default)),l}).execute().catch(d=>{throw console.error("Error adding column",o,d),d}),u.index&&await this.db.schema.createIndex(`${n}_${o}_index`).on(n).column(o).execute().catch(d=>{})),(c==null?void 0:c.columns.find(d=>d.name===o))||await this.db.schema.alterTable(s).addColumn(o,"varchar",d=>{let l=d;return u.primary&&(l=l.primaryKey().references(`${n}.${o}`)),l}).execute();}}}async rawFindById(e,t,n){let a=await this.db.selectFrom(e).where("id","=",t).selectAll(e).select(s=>jsonObjectFrom(s.selectFrom(`${e}_meta`).selectAll(`${e}_meta`).whereRef(`${e}_meta.id`,"=",`${e}.id`)).as("_meta"));a=this.applyInclude(e,a,n);let i=await a.executeTakeFirst();if(i)return this.convertToMaterializedLiveType(i)}async findOne(e,t,n){let a=await this.rawFindById(e.name,t,n==null?void 0:n.include);if(a)return r(a)}async rawFind(e,t,n){let a=this.db.selectFrom(e).selectAll(e).select(o=>jsonObjectFrom(o.selectFrom(`${e}_meta`).selectAll(`${e}_meta`).whereRef(`${e}_meta.id`,"=",`${e}.id`)).as("_meta"));a=this.applyWhere(e,a,t),a=this.applyInclude(e,a,n);let i=await a.execute(),s=Object.fromEntries(i.map(o=>{let{id:h,...p}=o;return [h,p]}));return Object.keys(s).length===0?{}:Object.entries(s).reduce((o,[h,p])=>(o[h]=this.convertToMaterializedLiveType(p),o),{})}async find(e,t){let n=await this.rawFind(e.name,t==null?void 0:t.where,t==null?void 0:t.include);return Object.fromEntries(Object.entries(n).map(([a,i])=>[a,r(i)]))}async rawUpsert(e,t,n){return await this.db.transaction().execute(async a=>{var o;let i=!!await a.selectFrom(e).select("id").where("id","=",t).executeTakeFirst(),s={},c={};for(let[h,p]of Object.entries(n.value)){let u=(o=p._meta)==null?void 0:o.timestamp;u&&(s[h]=p.value,c[h]=u);}i?await Promise.all([a.updateTable(e).set(s).where("id","=",t).execute(),a.updateTable(`${e}_meta`).set(c).where("id","=",t).execute()]):await Promise.all([a.insertInto(e).values({...s,id:t}).execute(),a.insertInto(`${e}_meta`).values({...c,id:t}).execute()]);}),n}convertToMaterializedLiveType(e){if(!e._meta)throw new Error("Missing _meta");return {value:Object.entries(e).reduce((t,[n,a])=>{var i,s,c;return n==="_meta"||(n==="id"?t[n]={value:a}:Array.isArray(a)?t[n]={value:a.map(o=>this.convertToMaterializedLiveType(o)),_meta:{timestamp:(i=e==null?void 0:e._meta)==null?void 0:i[n]}}:typeof a=="object"&&a!==null&&!(a instanceof Date)?t[n]={...this.convertToMaterializedLiveType(a),_meta:{timestamp:(s=e==null?void 0:e._meta)==null?void 0:s[n]}}:t[n]={value:a,_meta:{timestamp:(c=e==null?void 0:e._meta)==null?void 0:c[n]}}),t},{})}}applyWhere(e,t,n){if(!n)return t;if(!this.schema)throw new Error("Schema not initialized");let a=this.schema[e];if(!a)throw new Error("Resource not found");for(let[i,s]of Object.entries(n))if(a.fields[i])t=t.where(`${e}.${i}`,"=",s);else if(a.relations[i]){let c=a.relations[i],o=c.entity.name,h=c.type==="one"?"id":c.foreignColumn,p=c.type==="one"?c.relationalColumn:"id";t=t.leftJoin(o,`${o}.${h}`,`${e}.${p}`),t=this.applyWhere(o,t,s);}return t}applyInclude(e,t,n){if(!n)return t;if(!this.schema)throw new Error("Schema not initialized");let a=this.schema[e];if(!a)throw new Error(`Resource not found: ${e}`);for(let[i,s]of Object.entries(n)){if(!a.relations[i])throw new Error(`Relation ${i} not found in resource ${e}`);let c=a.relations[i],o=c.entity.name,h=c.type==="one"?"id":c.foreignColumn,p=c.type==="one"?c.relationalColumn:"id",u=c.type==="one"?jsonObjectFrom:jsonArrayFrom;t=t.select(m=>u(m.selectFrom(o).selectAll(o).whereRef(`${o}.${h}`,"=",`${e}.${p}`).select(d=>jsonObjectFrom(d.selectFrom(`${o}_meta`).selectAll(`${o}_meta`).whereRef(`${o}_meta.id`,"=",`${o}.id`)).as("_meta"))).as(i));}return t}};var z=class r{router;storage;schema;middlewares=new Set;contextProvider;mutationSubscriptions=new Set;constructor(e){var t;this.router=e.router,this.storage=e.storage,this.schema=e.schema,(t=e.middlewares)==null||t.forEach(n=>this.middlewares.add(n)),this.storage.updateSchema(this.schema),this.contextProvider=e.contextProvider;}static create(e){return new r(e)}subscribeToMutations(e){return this.mutationSubscriptions.add(e),()=>{this.mutationSubscriptions.delete(e);}}async handleRequest(e){if(!this.router.routes[e.req.resourceName])throw new Error("Invalid resource");let t=await Array.from(this.middlewares.values()).reduceRight((n,a)=>i=>a({req:i,next:n}),async n=>this.router.routes[e.req.resourceName].handleRequest({req:n,db:this.storage,schema:this.schema}))(e.req);return t&&e.req.type==="MUTATE"&&t.acceptedValues&&Object.keys(t.acceptedValues).length>0&&this.mutationSubscriptions.forEach(n=>{n({id:e.req.context.messageId,type:"MUTATE",resource:e.req.resourceName,payload:t.acceptedValues??{},resourceId:e.req.resourceId});}),t}use(e){return this.middlewares.add(e),this}context(e){return this.contextProvider=e,this}},_t=z.create;
2
- export{$ as Route,U as RouteFactory,_ as Router,ne as SQLStorage,z as Server,D as Storage,yt as expressAdapter,Rt as routeFactory,gt as router,_t as server};
1
+ import {a,b,r}from'./chunk-LLHCJUB6.js';import he,{parse}from'qs';import {z as z$1}from'zod';import A from'node:crypto';import {Kysely,PostgresDialect}from'kysely';import {jsonObjectFrom,jsonArrayFrom}from'kysely/helpers/postgres';var L=a(v=>{Object.defineProperty(v,"__esModule",{value:true});v.parse=le;v.serialize=me;var ie=/^[\u0021-\u003A\u003C\u003E-\u007E]+$/,oe=/^[\u0021-\u003A\u003C-\u007E]*$/,se=/^([.]?[a-z0-9]([a-z0-9-]{0,61}[a-z0-9])?)([.][a-z0-9]([a-z0-9-]{0,61}[a-z0-9])?)*$/i,ce=/^[\u0020-\u003A\u003D-\u007E]*$/,ue=Object.prototype.toString,de=(()=>{let r=function(){};return r.prototype=Object.create(null),r})();function le(r,e){let t=new de,n=r.length;if(n<2)return t;let a=(e==null?void 0:e.decode)||pe,i=0;do{let s=r.indexOf("=",i);if(s===-1)break;let c=r.indexOf(";",i),o=c===-1?n:c;if(s>o){i=r.lastIndexOf(";",s-1)+1;continue}let f=F(r,i,s),y=q(r,s,f),l=r.slice(f,y);if(t[l]===void 0){let m=F(r,s+1,o),u=q(r,o,m),d=a(r.slice(m,u));t[l]=d;}i=o+1;}while(i<n);return t}function F(r,e,t){do{let n=r.charCodeAt(e);if(n!==32&&n!==9)return e}while(++e<t);return t}function q(r,e,t){for(;e>t;){let n=r.charCodeAt(--e);if(n!==32&&n!==9)return e+1}return t}function me(r,e,t){let n=(t==null?void 0:t.encode)||encodeURIComponent;if(!ie.test(r))throw new TypeError(`argument name is invalid: ${r}`);let a=n(e);if(!oe.test(a))throw new TypeError(`argument val is invalid: ${e}`);let i=r+"="+a;if(!t)return i;if(t.maxAge!==void 0){if(!Number.isInteger(t.maxAge))throw new TypeError(`option maxAge is invalid: ${t.maxAge}`);i+="; Max-Age="+t.maxAge;}if(t.domain){if(!se.test(t.domain))throw new TypeError(`option domain is invalid: ${t.domain}`);i+="; Domain="+t.domain;}if(t.path){if(!ce.test(t.path))throw new TypeError(`option path is invalid: ${t.path}`);i+="; Path="+t.path;}if(t.expires){if(!ye(t.expires)||!Number.isFinite(t.expires.valueOf()))throw new TypeError(`option expires is invalid: ${t.expires}`);i+="; Expires="+t.expires.toUTCString();}if(t.httpOnly&&(i+="; HttpOnly"),t.secure&&(i+="; Secure"),t.partitioned&&(i+="; Partitioned"),t.priority)switch(typeof t.priority=="string"?t.priority.toLowerCase():void 0){case "low":i+="; Priority=Low";break;case "medium":i+="; Priority=Medium";break;case "high":i+="; Priority=High";break;default:throw new TypeError(`option priority is invalid: ${t.priority}`)}if(t.sameSite)switch(typeof t.sameSite=="string"?t.sameSite.toLowerCase():t.sameSite){case true:case "strict":i+="; SameSite=Strict";break;case "lax":i+="; SameSite=Lax";break;case "none":i+="; SameSite=None";break;default:throw new TypeError(`option sameSite is invalid: ${t.sameSite}`)}return i}function pe(r){if(r.indexOf("%")===-1)return r;try{return decodeURIComponent(r)}catch{return r}}function ye(r){return ue.call(r)==="[object Date]"}});var G=b(L(),1);var E=z$1.object({resource:z$1.string(),where:z$1.record(z$1.any()).optional(),include:z$1.record(z$1.any()).optional(),lastSyncedAt:z$1.string().optional(),limit:z$1.number().optional(),sort:z$1.array(z$1.object({key:z$1.string(),direction:z$1.enum(["asc","desc"])})).optional()}),P=z$1.record(z$1.object({value:z$1.string().or(z$1.number()).or(z$1.boolean()).or(z$1.date()).nullable(),_meta:z$1.object({timestamp:z$1.string().optional().nullable()}).optional()})).superRefine((r,e)=>{r.id&&e.addIssue({code:z$1.ZodIssueCode.custom,message:"Payload cannot have an id"});}),Z=z$1.object({id:z$1.string().optional(),type:z$1.literal("MUTATE"),resource:z$1.string()}),S=Z.extend({procedure:z$1.string(),payload:z$1.any().optional()}),M=Z.extend({resourceId:z$1.string(),payload:P});z$1.union([S,M]);var H=E.omit({resource:true}),C=S.omit({id:true,type:true,resource:true,procedure:true}),N=M.omit({id:true,type:true,resource:true});z$1.union([N,C]);var Q=r=>async e=>{var t;try{let n=typeof e.headers.getSetCookie=="function"?Object.fromEntries(e.headers):e.headers,a={headers:n,cookies:n.cookie?G.default.parse(n.cookie):{}},i=new URL(e.url),s=i.pathname.split("/"),c=i.searchParams,o=he.parse(c.toString()),f=await((t=r.contextProvider)==null?void 0:t.call(r,{transport:"HTTP",headers:a.headers,cookies:a.cookies,query:o}))??{};if(e.method==="GET"){let y=s[s.length-1],{success:l,data:m,error:u}=H.safeParse(o);if(!l)return Response.json({message:"Invalid query",code:"INVALID_QUERY",details:u},{status:400});let d=await r.handleRequest({req:{...a,type:"QUERY",resourceName:y,context:f,where:m.where,include:m.include,query:o}});return !d||!d.data?Response.json({message:"Invalid resource",code:"INVALID_RESOURCE"},{status:400}):Response.json(d.data)}if(e.method==="POST")try{let y=s[s.length-1],l=s[s.length-2],m=e.body?await e.json():{},u;if(y==="set"){let{success:b,data:w,error:O}=N.safeParse(m);if(!b)return Response.json({message:"Invalid mutation",code:"INVALID_REQUEST",details:O},{status:400});u=w;}else {let{success:b,data:w,error:O}=C.safeParse(m);if(!b)return Response.json({message:"Invalid mutation",code:"INVALID_REQUEST",details:O},{status:400});u=w;}let d=await r.handleRequest({req:{...a,type:"MUTATE",resourceName:l,input:u.payload,context:f,resourceId:u.resourceId,procedure:y!=="set"?y:void 0,query:{}}});return Response.json(d)}catch(y){return console.error("Error parsing mutation from the client:",y),Response.json({message:"Internal server error",code:"INTERNAL_SERVER_ERROR"},{status:500})}return Response.json({message:"Not found",code:"NOT_FOUND"},{status:404})}catch(n){return console.error("Unexpected error:",n),Response.json({message:"Internal server error",code:"INTERNAL_SERVER_ERROR"},{status:500})}};var ee=b(L(),1);var R=z$1.string(),Te=z$1.object({id:R,type:z$1.literal("SUBSCRIBE"),resource:z$1.string()}),ge=E.extend({id:R,type:z$1.literal("QUERY")}),B=M.extend({id:R}),Re=S.extend({id:R}),be=z$1.union([Re,B]),W=z$1.union([Te,ge,be]),we=z$1.object({id:R,type:z$1.literal("REJECT"),resource:z$1.string(),message:z$1.string().optional()}),xe=z$1.object({id:R,type:z$1.literal("REPLY"),data:z$1.any()});z$1.union([we,xe,B]);z$1.object({resource:z$1.string(),data:z$1.record(P)});var Y="0123456789ABCDEFGHJKMNPQRSTVWXYZ",I=32;var Se=16,J=10,K=0xffffffffffff;var T;(function(r){r.Base32IncorrectEncoding="B32_ENC_INVALID",r.DecodeTimeInvalidCharacter="DEC_TIME_CHAR",r.DecodeTimeValueMalformed="DEC_TIME_MALFORMED",r.EncodeTimeNegative="ENC_TIME_NEG",r.EncodeTimeSizeExceeded="ENC_TIME_SIZE_EXCEED",r.EncodeTimeValueMalformed="ENC_TIME_MALFORMED",r.PRNGDetectFailure="PRNG_DETECT",r.ULIDInvalid="ULID_INVALID",r.Unexpected="UNEXPECTED",r.UUIDInvalid="UUID_INVALID";})(T||(T={}));var g=class extends Error{constructor(e,t){super(`${t} (${e})`),this.name="ULIDError",this.code=e;}};function Me(r){let e=Math.floor(r()*I);return e===I&&(e=I-1),Y.charAt(e)}function Ie(r){var n;let e=ve(),t=e&&(e.crypto||e.msCrypto)||(typeof A<"u"?A:null);if(typeof(t==null?void 0:t.getRandomValues)=="function")return ()=>{let a=new Uint8Array(1);return t.getRandomValues(a),a[0]/255};if(typeof(t==null?void 0:t.randomBytes)=="function")return ()=>t.randomBytes(1).readUInt8()/255;if((n=A)!=null&&n.randomBytes)return ()=>A.randomBytes(1).readUInt8()/255;throw new g(T.PRNGDetectFailure,"Failed to find a reliable PRNG")}function ve(){return je()?self:typeof window<"u"?window:typeof global<"u"?global:typeof globalThis<"u"?globalThis:null}function Ee(r,e){let t="";for(;r>0;r--)t=Me(e)+t;return t}function Ae(r,e=J){if(isNaN(r))throw new g(T.EncodeTimeValueMalformed,`Time must be a number: ${r}`);if(r>K)throw new g(T.EncodeTimeSizeExceeded,`Cannot encode a time larger than ${K}: ${r}`);if(r<0)throw new g(T.EncodeTimeNegative,`Time must be positive: ${r}`);if(Number.isInteger(r)===false)throw new g(T.EncodeTimeValueMalformed,`Time must be an integer: ${r}`);let t,n="";for(let a=e;a>0;a--)t=r%I,n=Y.charAt(t)+n,r=(r-t)/I;return n}function je(){return typeof WorkerGlobalScope<"u"&&self instanceof WorkerGlobalScope}function X(r,e){let t=Ie(),n=Date.now();return Ae(n,J)+Ee(Se,t)}var _=()=>X().toLowerCase();var te=r=>{let e={},t={};return r.subscribeToMutations(n=>{let a=n;!a.resourceId||!a.payload||(console.log("Mutation propagated:",a),Object.entries(t[a.resource]??{}).forEach(([i,s])=>{var c;(c=e[i])==null||c.send(JSON.stringify({...a,id:a.id??_()}));}));}),(n,a)=>{var y;let i=l=>{n.send(JSON.stringify(l));},s=_(),c={headers:a.headers,cookies:typeof a.headers.cookie=="string"?ee.default.parse(a.headers.cookie):{}},o=parse(a.url.split("?")[1]),f=(y=r.contextProvider)==null?void 0:y.call(r,{transport:"WEBSOCKET",headers:c.headers,cookies:c.cookies,query:o});e[s]=n,console.log("Client connected:",s),n.on("message",async l=>{try{console.log("Message received from the client:",l);let m=W.parse(JSON.parse(l.toString()));if(m.type==="SUBSCRIBE"){let{resource:u}=m;t[u]||(t[u]={}),t[u][s]={};}else if(m.type==="QUERY"){let{resource:u}=m,d=await r.handleRequest({req:{...c,type:"QUERY",resourceName:u,context:await f??{},query:o}});if(!d||!d.data)throw new Error("Invalid resource");i({id:m.id,type:"REPLY",data:{resource:u,data:Object.fromEntries(Object.entries(d.data??{}).map(([b,w])=>[b,w.value]))}});}else if(m.type==="MUTATE"){let{resource:u}=m;console.log("Received mutation from client:",m);try{let d=await r.handleRequest({req:{...c,type:"MUTATE",resourceName:u,input:m.payload,context:{messageId:m.id,...await f??{}},resourceId:m.resourceId,procedure:m.procedure,query:o}});m.procedure&&i({id:m.id,type:"REPLY",data:d});}catch(d){i({id:m.id,type:"REJECT",resource:u,message:d.message}),console.error("Error parsing mutation from the client:",d);}}}catch(m){console.error("Error handling message from the client:",m);}}),n.on("close",()=>{console.log("Connection closed",s),delete e[s];for(let l of Object.values(t))delete l[s];});}};function re(r){let e=`${r.protocol}://${r.hostname}${r.url}`,t=new Headers;return Object.entries(r.headers).forEach(([n,a])=>{a&&t.set(n,Array.isArray(a)?a.join(","):a);}),new Request(e,{method:r.method,headers:t,body:r.body&&r.method!=="GET"?JSON.stringify(r.body):void 0})}var dt=(r,e,t)=>{r.ws(`${(t==null?void 0:t.basePath)??""}/ws`,te(e)),r.use(`${(t==null?void 0:t.basePath)??""}/`,(n,a)=>{Q(e)(re(n)).then(s=>s.json().then(c=>a.status(s.status).send(c)));});};var $=class r{routes;constructor(e){this.routes=e.routes;}static create(e){return new r(e)}},pt=r=>$.create({...r}),Pe=r=>({handler:e=>({inputValidator:r??z$1.undefined(),handler:e})}),U=class r{_resourceSchema;resourceName;middlewares;customMutations;constructor(e,t){this.resourceName=e,this.middlewares=new Set,this.customMutations=t??{};}handleFind=async({req:e,db:t})=>({data:await t.rawFind(e.resourceName,e.where,e.include),acceptedValues:null});handleSet=async({req:e,db:t,schema:n})=>{if(!e.input)throw new Error("Payload is required");if(!e.resourceId)throw new Error("ResourceId is required");let a=await t.rawFindById(e.resourceName,e.resourceId),[i,s]=n[this.resourceName].mergeMutation("set",e.input,a);if(!s)throw new Error("Mutation rejected");return {data:await t.rawUpsert(e.resourceName,e.resourceId,i),acceptedValues:s}};async handleRequest(e){let t=n=>(()=>{if(n.type==="QUERY")return this.handleFind({req:n,db:e.db,schema:e.schema});if(n.type==="MUTATE")if(n.procedure){if(this.customMutations[n.procedure]){let a=this.customMutations[n.procedure].inputValidator.parse(n.input);return n.input=a,this.customMutations[n.procedure].handler({req:n,db:e.db,schema:e.schema})}}else return this.handleSet({req:n,db:e.db,schema:e.schema});throw new Error("Invalid request")})();return await Array.from(this.middlewares.values()).reduceRight((n,a)=>i=>a({req:i,next:n}),async n=>t(n))(e.req)}use(...e){for(let t of e)this.middlewares.add(t);return this}withMutations(e){return new r(this.resourceName,e({mutation:Pe}))}},D=class r{middlewares;constructor(e=[]){this.middlewares=e;}createBasicRoute(e){return new U(e.name).use(...this.middlewares)}use(...e){return new r([...this.middlewares,...e])}static create(){return new r}},yt=D.create;var z=class{async insert(e,t){let n=new Date().toISOString();return r(await this.rawUpsert(e.name,t.id,{value:Object.fromEntries(Object.entries(t).map(([a,i])=>[a,{value:i,_meta:{timestamp:n}}]))}))}async update(e,t,n){let a=new Date().toISOString(),{id:i,...s}=n;return r(await this.rawUpsert(e.name,t,{value:Object.fromEntries(Object.entries(s).map(([c,o])=>[c,{value:o,_meta:{timestamp:a}}]))}))}},ne=class extends z{db;schema;constructor(e){super(),this.db=new Kysely({dialect:new PostgresDialect({pool:e})});}async updateSchema(e){this.schema=e;let t=await this.db.introspection.getTables();for(let[n,a]of Object.entries(e)){let i=t.find(o=>o.name===n);i||await this.db.schema.createTable(n).ifNotExists().execute();let s=`${n}_meta`,c=t.find(o=>o.name===s);c||await this.db.schema.createTable(s).ifNotExists().execute();for(let[o,f]of Object.entries(a.fields)){let y=i==null?void 0:i.columns.find(u=>u.name===o),l=f.getStorageFieldType();y?y.dataType!==l.type&&console.error("Column type mismatch:",o,"expected to have type:",l.type,"but has type:",y.dataType):(await this.db.schema.alterTable(n).addColumn(o,l.type,u=>{let d=u;return l.unique&&(d=d.unique()),l.nullable||(d=d.notNull()),l.references&&(d=d.references(l.references)),l.primary&&(d=d.primaryKey()),l.default!==void 0&&(d=d.defaultTo(l.default)),d}).execute().catch(u=>{throw console.error("Error adding column",o,u),u}),l.index&&await this.db.schema.createIndex(`${n}_${o}_index`).on(n).column(o).execute().catch(u=>{})),(c==null?void 0:c.columns.find(u=>u.name===o))||await this.db.schema.alterTable(s).addColumn(o,"varchar",u=>{let d=u;return l.primary&&(d=d.primaryKey().references(`${n}.${o}`)),d}).execute();}}}async rawFindById(e,t,n){let a=await this.db.selectFrom(e).where("id","=",t).selectAll(e).select(s=>jsonObjectFrom(s.selectFrom(`${e}_meta`).selectAll(`${e}_meta`).whereRef(`${e}_meta.id`,"=",`${e}.id`)).as("_meta"));a=this.applyInclude(e,a,n);let i=await a.executeTakeFirst();if(i)return this.convertToMaterializedLiveType(i)}async findOne(e,t,n){let a=await this.rawFindById(e.name,t,n==null?void 0:n.include);if(a)return r(a)}async rawFind(e,t,n){let a=this.db.selectFrom(e).selectAll(e).select(o=>jsonObjectFrom(o.selectFrom(`${e}_meta`).selectAll(`${e}_meta`).whereRef(`${e}_meta.id`,"=",`${e}.id`)).as("_meta"));a=this.applyWhere(e,a,t),a=this.applyInclude(e,a,n);let i=await a.execute(),s=Object.fromEntries(i.map(o=>{let{id:f,...y}=o;return [f,y]}));return Object.keys(s).length===0?{}:Object.entries(s).reduce((o,[f,y])=>(o[f]=this.convertToMaterializedLiveType(y),o),{})}async find(e,t){let n=await this.rawFind(e.name,t==null?void 0:t.where,t==null?void 0:t.include);return Object.fromEntries(Object.entries(n).map(([a,i])=>[a,r(i)]))}async rawUpsert(e,t,n){return await this.db.transaction().execute(async a=>{var o;let i=!!await a.selectFrom(e).select("id").where("id","=",t).executeTakeFirst(),s={},c={};for(let[f,y]of Object.entries(n.value)){let l=(o=y._meta)==null?void 0:o.timestamp;l&&(s[f]=y.value,c[f]=l);}i?await Promise.all([a.updateTable(e).set(s).where("id","=",t).execute(),a.updateTable(`${e}_meta`).set(c).where("id","=",t).execute()]):await Promise.all([a.insertInto(e).values({...s,id:t}).execute(),a.insertInto(`${e}_meta`).values({...c,id:t}).execute()]);}),n}convertToMaterializedLiveType(e){if(!e._meta)throw new Error("Missing _meta");return {value:Object.entries(e).reduce((t,[n,a])=>{var i,s,c;return n==="_meta"||(n==="id"?t[n]={value:a}:Array.isArray(a)?t[n]={value:a.map(o=>this.convertToMaterializedLiveType(o)),_meta:{timestamp:(i=e==null?void 0:e._meta)==null?void 0:i[n]}}:typeof a=="object"&&a!==null&&!(a instanceof Date)?t[n]={...this.convertToMaterializedLiveType(a),_meta:{timestamp:(s=e==null?void 0:e._meta)==null?void 0:s[n]}}:t[n]={value:a,_meta:{timestamp:(c=e==null?void 0:e._meta)==null?void 0:c[n]}}),t},{})}}applyWhere(e,t,n){if(!n)return t;if(!this.schema)throw new Error("Schema not initialized");let a=this.schema[e];if(!a)throw new Error("Resource not found");for(let[i,s]of Object.entries(n))if(a.fields[i])t=t.where(`${e}.${i}`,"=",s);else if(a.relations[i]){let c=a.relations[i],o=c.entity.name,f=c.type==="one"?"id":c.foreignColumn,y=c.type==="one"?c.relationalColumn:"id";t=t.leftJoin(o,`${o}.${f}`,`${e}.${y}`),t=this.applyWhere(o,t,s);}return t}applyInclude(e,t,n){if(!n)return t;if(!this.schema)throw new Error("Schema not initialized");let a=this.schema[e];if(!a)throw new Error(`Resource not found: ${e}`);for(let[i,s]of Object.entries(n)){if(!a.relations[i])throw new Error(`Relation ${i} not found in resource ${e}`);let c=a.relations[i],o=c.entity.name,f=c.type==="one"?"id":c.foreignColumn,y=c.type==="one"?c.relationalColumn:"id",l=c.type==="one"?jsonObjectFrom:jsonArrayFrom;t=t.select(m=>l(m.selectFrom(o).selectAll(o).whereRef(`${o}.${f}`,"=",`${e}.${y}`).select(u=>jsonObjectFrom(u.selectFrom(`${o}_meta`).selectAll(`${o}_meta`).whereRef(`${o}_meta.id`,"=",`${o}.id`)).as("_meta"))).as(i));}return t}};var k=class r{router;storage;schema;middlewares=new Set;contextProvider;mutationSubscriptions=new Set;constructor(e){var t;this.router=e.router,this.storage=e.storage,this.schema=e.schema,(t=e.middlewares)==null||t.forEach(n=>{this.middlewares.add(n);}),this.storage.updateSchema(this.schema),this.contextProvider=e.contextProvider;}static create(e){return new r(e)}subscribeToMutations(e){return this.mutationSubscriptions.add(e),()=>{this.mutationSubscriptions.delete(e);}}async handleRequest(e){if(!this.router.routes[e.req.resourceName])throw new Error("Invalid resource");let t=await Array.from(this.middlewares.values()).reduceRight((n,a)=>i=>a({req:i,next:n}),async n=>this.router.routes[e.req.resourceName].handleRequest({req:n,db:this.storage,schema:this.schema}))(e.req);return t&&e.req.type==="MUTATE"&&t.acceptedValues&&Object.keys(t.acceptedValues).length>0&&this.mutationSubscriptions.forEach(n=>{n({id:e.req.context.messageId,type:"MUTATE",resource:e.req.resourceName,payload:t.acceptedValues??{},resourceId:e.req.resourceId});}),t}use(e){return this.middlewares.add(e),this}context(e){return this.contextProvider=e,this}},bt=k.create;
2
+ export{U as Route,D as RouteFactory,$ as Router,ne as SQLStorage,k as Server,z as Storage,dt as expressAdapter,yt as routeFactory,pt as router,bt as server};
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@live-state/sync",
3
- "version": "0.0.2",
3
+ "version": "0.0.3",
4
4
  "private": false,
5
5
  "publishConfig": {
6
6
  "access": "public"
@@ -21,22 +21,32 @@
21
21
  ],
22
22
  "type": "module",
23
23
  "devDependencies": {
24
+ "@testing-library/jest-dom": "^6.8.0",
25
+ "@testing-library/react": "^16.3.0",
26
+ "@testing-library/react-hooks": "^8.0.1",
24
27
  "@types/express": "^4.17.21",
25
28
  "@types/express-ws": "^3.0.5",
26
29
  "@types/node": "^20.11.24",
27
30
  "@types/qs": "^6.14.0",
31
+ "@types/react": "^18.2.62",
32
+ "@types/react-dom": "^18.2.19",
28
33
  "@types/ws": "^8.5.13",
34
+ "@vitest/coverage-v8": "3.2.4",
29
35
  "cookie": "^1.0.2",
30
36
  "express": "^4.21.2",
37
+ "jsdom": "^26.1.0",
31
38
  "react": "18.0.0",
39
+ "react-dom": "^18.2.0",
32
40
  "tsup": "^8.0.2",
33
41
  "typescript": "5.5.4",
34
42
  "ulid": "^3.0.0",
43
+ "vitest": "^3.2.4",
35
44
  "@repo/typescript-config": "0.0.0"
36
45
  },
37
46
  "dependencies": {
38
- "crypto-hash": "^3.1.0",
47
+ "fast-deep-equal": "^3.1.3",
39
48
  "idb": "^8.0.3",
49
+ "js-xxhash": "^4.0.0",
40
50
  "kysely": "^0.28.2",
41
51
  "qs": "^6.14.0",
42
52
  "ws": "^8.18.0",
@@ -59,7 +69,7 @@
59
69
  "require": "./dist/client.js"
60
70
  },
61
71
  "./client/fetch": {
62
- "types": "./src/client/fetch-client.ts",
72
+ "types": "./src/client/fetch/index.ts",
63
73
  "import": "./dist/fetch-client.js",
64
74
  "require": "./dist/fetch-client.js"
65
75
  }
@@ -79,7 +89,9 @@
79
89
  "scripts": {
80
90
  "build": "NODE_ENV=production tsup",
81
91
  "dev": "NODE_ENV=development tsup --watch",
82
- "lint": "eslint src/",
83
- "typecheck": "tsc --noEmit"
92
+ "lint": "biome lint",
93
+ "typecheck": "tsc --noEmit --project . && tsc --noEmit --project src/client",
94
+ "test": "vitest run",
95
+ "test:watch": "vitest"
84
96
  }
85
97
  }
@@ -1 +0,0 @@
1
- var x="0123456789ABCDEFGHJKMNPQRSTVWXYZ";var l;(function(e){e.Base32IncorrectEncoding="B32_ENC_INVALID",e.DecodeTimeInvalidCharacter="DEC_TIME_CHAR",e.DecodeTimeValueMalformed="DEC_TIME_MALFORMED",e.EncodeTimeNegative="ENC_TIME_NEG",e.EncodeTimeSizeExceeded="ENC_TIME_SIZE_EXCEED",e.EncodeTimeValueMalformed="ENC_TIME_MALFORMED",e.PRNGDetectFailure="PRNG_DETECT",e.ULIDInvalid="ULID_INVALID",e.Unexpected="UNEXPECTED",e.UUIDInvalid="UUID_INVALID";})(l||(l={}));var u=class extends Error{constructor(t,n){super(`${n} (${t})`),this.name="ULIDError",this.code=t;}};function L(e){let t=Math.floor(e()*32);return t===32&&(t=31),x.charAt(t)}function I(e){let t=M(),n=t&&(t.crypto||t.msCrypto)||null;if(typeof(n==null?void 0:n.getRandomValues)=="function")return ()=>{let i=new Uint8Array(1);return n.getRandomValues(i),i[0]/255};if(typeof(n==null?void 0:n.randomBytes)=="function")return ()=>n.randomBytes(1).readUInt8()/255;throw new u(l.PRNGDetectFailure,"Failed to find a reliable PRNG")}function M(){return A()?self:typeof window<"u"?window:typeof global<"u"?global:typeof globalThis<"u"?globalThis:null}function E(e,t){let n="";for(;e>0;e--)n=L(t)+n;return n}function _(e,t=10){if(isNaN(e))throw new u(l.EncodeTimeValueMalformed,`Time must be a number: ${e}`);if(e>0xffffffffffff)throw new u(l.EncodeTimeSizeExceeded,`Cannot encode a time larger than ${0xffffffffffff}: ${e}`);if(e<0)throw new u(l.EncodeTimeNegative,`Time must be positive: ${e}`);if(Number.isInteger(e)===false)throw new u(l.EncodeTimeValueMalformed,`Time must be an integer: ${e}`);let n,i="";for(let a=t;a>0;a--)n=e%32,i=x.charAt(n)+i,e=(e-n)/32;return i}function A(){return typeof WorkerGlobalScope<"u"&&self instanceof WorkerGlobalScope}function R(e,t){let n=I(),i=Date.now();return _(i,10)+E(16,n)}var S=()=>R().toLowerCase(),j=(e,t)=>typeof e=="function"?e(t):e;var w=(e,t,n=[])=>new Proxy(e,{get:(i,a)=>{var p,y;if(a==="__isProxy__")return true;let r=(p=t.get)==null?void 0:p.call(t,i,[...n,a]);if(r!==void 0)return r;let o=i,s=a;return (y=o[s])!=null&&y.__isProxy__||(o[s]=w(typeof o[s]=="object"?o[s]:function(){},t,[...n,a])),o[s]},apply:(i,a,r)=>{var o;return (o=t.apply)==null?void 0:o.call(t,i,n,r)}});var b=e=>{if(e)return Array.isArray(e.value)?e.value.map(t=>b(t)):typeof e.value!="object"||e.value===null||e.value instanceof Date?e.value:Object.fromEntries(Object.entries(e.value).map(([t,n])=>[t,b(n)]))};export{S as a,j as b,w as c,b as d};