@live-state/sync 0.0.4-beta.5 → 0.0.4-beta.7

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1 +1 @@
1
- var f="0123456789ABCDEFGHJKMNPQRSTVWXYZ";var c;(function(t){t.Base32IncorrectEncoding="B32_ENC_INVALID",t.DecodeTimeInvalidCharacter="DEC_TIME_CHAR",t.DecodeTimeValueMalformed="DEC_TIME_MALFORMED",t.EncodeTimeNegative="ENC_TIME_NEG",t.EncodeTimeSizeExceeded="ENC_TIME_SIZE_EXCEED",t.EncodeTimeValueMalformed="ENC_TIME_MALFORMED",t.PRNGDetectFailure="PRNG_DETECT",t.ULIDInvalid="ULID_INVALID",t.Unexpected="UNEXPECTED",t.UUIDInvalid="UUID_INVALID";})(c||(c={}));var u=class extends Error{constructor(e,n){super(`${n} (${e})`),this.name="ULIDError",this.code=e;}};function _(t){let e=Math.floor(t()*32);return e===32&&(e=31),f.charAt(e)}function m(t){let e=p(),n=e&&(e.crypto||e.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(c.PRNGDetectFailure,"Failed to find a reliable PRNG")}function p(){return g()?self:typeof window<"u"?window:typeof global<"u"?global:typeof globalThis<"u"?globalThis:null}function I(t,e){let n="";for(;t>0;t--)n=_(e)+n;return n}function w(t,e=10){if(isNaN(t))throw new u(c.EncodeTimeValueMalformed,`Time must be a number: ${t}`);if(t>0xffffffffffff)throw new u(c.EncodeTimeSizeExceeded,`Cannot encode a time larger than ${0xffffffffffff}: ${t}`);if(t<0)throw new u(c.EncodeTimeNegative,`Time must be positive: ${t}`);if(Number.isInteger(t)===false)throw new u(c.EncodeTimeValueMalformed,`Time must be an integer: ${t}`);let n,i="";for(let r=e;r>0;r--)n=t%32,i=f.charAt(n)+i,t=(t-n)/32;return i}function g(){return typeof WorkerGlobalScope<"u"&&self instanceof WorkerGlobalScope}function h(t,e){let n=m(),i=Date.now();return w(i,10)+I(16,n)}var b=()=>h().toLowerCase(),A=(t,e)=>typeof t=="function"?t(e):t;var T=class t{_collection;_client;_where;_include;_limit;_single;_sort;_shouldAwait;constructor(e,n,i,r,o,s,l,a){this._collection=e,this._client=n,this._where=i??{},this._include=r??{},this._limit=o,this._single=s,this._sort=l,this._shouldAwait=a,this.get=this.get.bind(this),this.subscribe=this.subscribe.bind(this);}where(e){return new t(this._collection,this._client,{...this._where,...e},this._include,this._limit,this._single,this._sort,this._shouldAwait)}include(e){return new t(this._collection,this._client,this._where,{...this._include,...e},this._limit,this._single,this._sort,this._shouldAwait)}get(){let e=this._client.get({resource:this._collection.name,where:this._where,include:this._include,limit:this._limit,sort:this._sort});return this._shouldAwait?Promise.resolve(e).then(n=>this._single?n[0]:n):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},n=>{if(this._single)return e(n[0]);e(n);})}limit(e){return new t(this._collection,this._client,this._where,this._include,e,this._single,this._sort,this._shouldAwait)}one(e){return this.first({id:e})}first(e){return new t(this._collection,this._client,e??this._where,this._include,1,true,this._sort,this._shouldAwait)}orderBy(e,n="asc"){let i=[...this._sort??[],{key:e,direction:n}];return new t(this._collection,this._client,this._where,this._include,this._limit,this._single,i,this._shouldAwait)}toJSON(){return {resource:this._collection.name,where:this._where,include:this._include,limit:this._limit,sort:this._sort}}static _init(e,n,i){return new t(e,n,void 0,void 0,void 0,void 0,void 0,i??false)}};var E=(t,e,n=[])=>new Proxy(t,{get:(i,r)=>{var a,d;if(r==="__isProxy__")return true;let o=(a=e.get)==null?void 0:a.call(e,i,[...n,r]);if(o!==void 0)return o;let s=i,l=r;return (d=s[l])!=null&&d.__isProxy__||(s[l]=E(typeof s[l]=="object"?s[l]:()=>{},e,[...n,r])),s[l]},apply:(i,r,o)=>{var s;return (s=e.apply)==null?void 0:s.call(e,i,n,o)}}),D=(t,e,n)=>{let i=[],r=0;for(let o=0;o<t.length&&(n===void 0||r<n);o++)e(t[o],o)&&(i.push(t[o]),r++);return i};export{b as a,A as b,T as c,E as d,D as e};
1
+ var x="0123456789ABCDEFGHJKMNPQRSTVWXYZ";var l;(function(t){t.Base32IncorrectEncoding="B32_ENC_INVALID",t.DecodeTimeInvalidCharacter="DEC_TIME_CHAR",t.DecodeTimeValueMalformed="DEC_TIME_MALFORMED",t.EncodeTimeNegative="ENC_TIME_NEG",t.EncodeTimeSizeExceeded="ENC_TIME_SIZE_EXCEED",t.EncodeTimeValueMalformed="ENC_TIME_MALFORMED",t.PRNGDetectFailure="PRNG_DETECT",t.ULIDInvalid="ULID_INVALID",t.Unexpected="UNEXPECTED",t.UUIDInvalid="UUID_INVALID";})(l||(l={}));var u=class extends Error{constructor(e,n){super(`${n} (${e})`),this.name="ULIDError",this.code=e;}};function L(t){let e=Math.floor(t()*32);return e===32&&(e=31),x.charAt(e)}function R(t){let e=w(),n=e&&(e.crypto||e.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 w(){return E()?self:typeof window<"u"?window:typeof global<"u"?global:typeof globalThis<"u"?globalThis:null}function A(t,e){let n="";for(;t>0;t--)n=L(e)+n;return n}function M(t,e=10){if(isNaN(t))throw new u(l.EncodeTimeValueMalformed,`Time must be a number: ${t}`);if(t>0xffffffffffff)throw new u(l.EncodeTimeSizeExceeded,`Cannot encode a time larger than ${0xffffffffffff}: ${t}`);if(t<0)throw new u(l.EncodeTimeNegative,`Time must be positive: ${t}`);if(Number.isInteger(t)===false)throw new u(l.EncodeTimeValueMalformed,`Time must be an integer: ${t}`);let n,i="";for(let a=e;a>0;a--)n=t%32,i=x.charAt(n)+i,t=(t-n)/32;return i}function E(){return typeof WorkerGlobalScope<"u"&&self instanceof WorkerGlobalScope}function b(t,e){let n=R(),i=Date.now();return M(i,10)+A(16,n)}var N=()=>b().toLowerCase(),j=(t,e)=>typeof t=="function"?t(e):t;var I=class t{_collection;_client;_where;_include;_limit;_single;_sort;_shouldAwait;constructor(e,n,i,a,r,s,o,T){this._collection=e,this._client=n,this._where=i??{},this._include=a??{},this._limit=r,this._single=s,this._sort=o,this._shouldAwait=T,this.get=this.get.bind(this),this.subscribe=this.subscribe.bind(this);}where(e){return new t(this._collection,this._client,{...this._where,...e},this._include,this._limit,this._single,this._sort,this._shouldAwait)}include(e){return new t(this._collection,this._client,this._where,{...this._include,...e},this._limit,this._single,this._sort,this._shouldAwait)}get(){let e=this._client.get({resource:this._collection.name,where:this._where,include:this._include,limit:this._limit,sort:this._sort});return this._shouldAwait?Promise.resolve(e).then(n=>this._single?n[0]:n):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},n=>{if(this._single)return e(n[0]);e(n);})}limit(e){return new t(this._collection,this._client,this._where,this._include,e,this._single,this._sort,this._shouldAwait)}one(e){return this.first({id:e})}first(e){return new t(this._collection,this._client,e??this._where,this._include,1,true,this._sort,this._shouldAwait)}orderBy(e,n="asc"){let i=[...this._sort??[],{key:e,direction:n}];return new t(this._collection,this._client,this._where,this._include,this._limit,this._single,i,this._shouldAwait)}toJSON(){return {resource:this._collection.name,where:this._where,include:this._include,limit:this._limit,sort:this._sort}}static _init(e,n,i){return new t(e,n,void 0,void 0,void 0,void 0,void 0,i??false)}};var O=(t,e,n=[])=>new Proxy(t,{get:(i,a)=>{var T,y;if(a==="__isProxy__")return true;let r=(T=e.get)==null?void 0:T.call(e,i,[...n,a]);if(r!==void 0)return r;let s=i,o=a;return (y=s[o])!=null&&y.__isProxy__||(s[o]=O(typeof s[o]=="object"?s[o]:()=>{},e,[...n,a])),s[o]},apply:(i,a,r)=>{var s;return (s=e.apply)==null?void 0:s.call(e,i,n,r)}}),D=(t,e,n)=>{let i=[],a=0;for(let r=0;r<t.length&&(n===void 0||a<n);r++)e(t[r],r)&&(i.push(t[r]),a++);return i};var _=t=>{if(t)return Array.isArray(t.value)?t.value.map(e=>_(e)):typeof t.value!="object"||t.value===null||t.value instanceof Date?t.value:Object.fromEntries(Object.entries(t.value).map(([e,n])=>[e,_(n)]))};export{N as a,j as b,I as c,O as d,D as e,_ as f};
@@ -1 +1 @@
1
- var R=Object.create;var x=Object.defineProperty;var b=Object.getOwnPropertyDescriptor;var g=Object.getOwnPropertyNames;var I=Object.getPrototypeOf,M=Object.prototype.hasOwnProperty;var j=(t,e)=>()=>(e||t((e={exports:{}}).exports,e),e.exports);var O=(t,e,n,i)=>{if(e&&typeof e=="object"||typeof e=="function")for(let a of g(e))!M.call(t,a)&&a!==n&&x(t,a,{get:()=>e[a],enumerable:!(i=b(e,a))||i.enumerable});return t};var K=(t,e,n)=>(n=t!=null?R(I(t)):{},O(e||!t||!t.__esModule?x(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,i){return this.inner.encodeMutation(e,n,i)}mergeMutation(e,n,i){return this.inner.mergeMutation(e,n,i)}getStorageFieldType(){return {...this.inner.getStorageFieldType(),nullable:true}}},l=class t extends s{storageType;convertFunc;isIndex;isUnique;defaultValue;foreignReference;isPrimary;constructor(e,n,i,a,r,u,y){super(),this.storageType=e,this.convertFunc=n,this.isIndex=i??false,this.isUnique=a??false,this.defaultValue=r,this.foreignReference=u,this.isPrimary=y??false;}encodeMutation(e,n,i){return {value:n,_meta:{timestamp:i}}}mergeMutation(e,n,i){return i&&i._meta.timestamp&&n._meta.timestamp&&i._meta.timestamp.localeCompare(n._meta.timestamp)>=0?[i,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 l{constructor(){super("integer",e=>Number(e));}static create(){return new t}},V=p.create,o=class t extends l{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)}},_=o.create,E=o.createId,F=o.createReference,d=class t extends l{constructor(){super("boolean",e=>typeof e=="string"?e.toLowerCase()==="true":!!e);}static create(){return new t}},C=d.create,m=class t extends l{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,i){super(),this.name=e,this.fields=n,this.relations=i??{};}encodeMutation(e,n,i){return Object.fromEntries(Object.entries(n).map(([a,r])=>[a,(this.fields[a]??this.relations[a]).encodeMutation("set",r,i)]))}mergeMutation(e,n,i){let a={};return [{value:{...(i==null?void 0:i.value)??{},...Object.fromEntries(Object.entries(n).map(([r,u])=>{let y=this.fields[r]??this.relations[r];if(!y)return [r,u];let[L,f]=y.mergeMutation(e,u,i==null?void 0:i.value[r]);return f&&(a[r]=f),[r,L]}))}},a]}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)}},P=v.create,T=class t extends s{entity;type;required;relationalColumn;foreignColumn;sourceEntity;constructor(e,n,i,a,r){super(),this.entity=e,this.type=n,this.required=r??false,this.relationalColumn=i,this.foreignColumn=a;}encodeMutation(e,n,i){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:i}}}mergeMutation(e,n,i){if(this.type==="many")throw new Error("Many not implemented.");return i&&i._meta.timestamp&&n._meta.timestamp&&i._meta.timestamp.localeCompare(n._meta.timestamp)>=0?[i,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,i)=>new t(e,"one",n,void 0,i??false)}static createManyFactory(){return (e,n,i)=>new t(e,"many",void 0,n,i??false)}},$=(t,e)=>({$type:"relations",objectName:t.name,relations:e({one:T.createOneFactory(),many:T.createManyFactory()})}),h=t=>{if(t)return Array.isArray(t.value)?t.value.map(e=>h(e)):typeof t.value!="object"||t.value===null||t.value instanceof Date?t.value:Object.fromEntries(Object.entries(t.value).map(([e,n])=>[e,h(n)]))},k=t=>Object.fromEntries(Object.entries(t).flatMap(([e,n])=>{if(n.$type==="relations")return [];let i=n,a=Object.values(t).find(r=>r.$type==="relations"&&r.objectName===n.name);return a&&(i=i.setRelations(a.relations)),[[i.name,i]]}));export{j as a,K as b,s as c,c as d,l as e,p as f,V as g,o as h,_ as i,E as j,F as k,d as l,C as m,m as n,q as o,v as p,P as q,T as r,$ as s,h as t,k as u};
1
+ var R=Object.create;var x=Object.defineProperty;var b=Object.getOwnPropertyDescriptor;var g=Object.getOwnPropertyNames;var I=Object.getPrototypeOf,M=Object.prototype.hasOwnProperty;var j=(t,e)=>()=>(e||t((e={exports:{}}).exports,e),e.exports);var O=(t,e,n,i)=>{if(e&&typeof e=="object"||typeof e=="function")for(let a of g(e))!M.call(t,a)&&a!==n&&x(t,a,{get:()=>e[a],enumerable:!(i=b(e,a))||i.enumerable});return t};var K=(t,e,n)=>(n=t!=null?R(I(t)):{},O(e||!t||!t.__esModule?x(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,i){return this.inner.encodeMutation(e,n,i)}mergeMutation(e,n,i){return this.inner.mergeMutation(e,n,i)}getStorageFieldType(){return {...this.inner.getStorageFieldType(),nullable:true}}},l=class t extends s{storageType;convertFunc;isIndex;isUnique;defaultValue;foreignReference;isPrimary;constructor(e,n,i,a,r,u,y){super(),this.storageType=e,this.convertFunc=n,this.isIndex=i??false,this.isUnique=a??false,this.defaultValue=r,this.foreignReference=u,this.isPrimary=y??false;}encodeMutation(e,n,i){return {value:n,_meta:{timestamp:i}}}mergeMutation(e,n,i){return i&&i._meta.timestamp&&n._meta.timestamp&&i._meta.timestamp.localeCompare(n._meta.timestamp)>=0?[i,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)}},d=class t extends l{constructor(){super("integer",e=>Number(e));}static create(){return new t}},V=d.create,o=class t extends l{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)}},_=o.create,E=o.createId,C=o.createReference,p=class t extends l{constructor(){super("boolean",e=>typeof e=="string"?e.toLowerCase()==="true":!!e);}static create(){return new t}},F=p.create,m=class t extends l{constructor(){super("timestamp",e=>typeof e=="string"?new Date(e):e);}static create(){return new t}},q=m.create;var f=class t extends s{name;fields;relations;constructor(e,n,i){super(),this.name=e,this.fields=n,this.relations=i??{};}encodeMutation(e,n,i){return Object.fromEntries(Object.entries(n).map(([a,r])=>[a,(this.fields[a]??this.relations[a]).encodeMutation("set",r,i)]))}mergeMutation(e,n,i){let a={};return [{value:{...(i==null?void 0:i.value)??{},...Object.fromEntries(Object.entries(n).map(([r,u])=>{let y=this.fields[r]??this.relations[r];if(!y)return [r,u];let[L,v]=y.mergeMutation(e,u,i==null?void 0:i.value[r]);return v&&(a[r]=v),[r,L]}))}},a]}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)}},P=f.create,T=class t extends s{entity;type;required;relationalColumn;foreignColumn;sourceEntity;constructor(e,n,i,a,r){super(),this.entity=e,this.type=n,this.required=r??false,this.relationalColumn=i,this.foreignColumn=a;}encodeMutation(e,n,i){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:i}}}mergeMutation(e,n,i){if(this.type==="many")throw new Error("Many not implemented.");return i&&i._meta.timestamp&&n._meta.timestamp&&i._meta.timestamp.localeCompare(n._meta.timestamp)>=0?[i,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,i)=>new t(e,"one",n,void 0,i??false)}static createManyFactory(){return (e,n,i)=>new t(e,"many",void 0,n,i??false)}},$=(t,e)=>({$type:"relations",objectName:t.name,relations:e({one:T.createOneFactory(),many:T.createManyFactory()})}),h=t=>{if(t)return Array.isArray(t.value)?t.value.map(e=>h(e)):typeof t.value!="object"||t.value===null||t.value instanceof Date?t.value:Object.fromEntries(Object.entries(t.value).map(([e,n])=>[e,h(n)]))},k=t=>Object.fromEntries(Object.entries(t).flatMap(([e,n])=>{if(n.$type==="relations")return [];let i=n,a=Object.values(t).find(r=>r.$type==="relations"&&r.objectName===n.name);return a&&(i=i.setRelations(a.relations)),[[i.name,i]]}));export{j as a,K as b,s as c,c as d,l as e,d as f,V as g,o as h,_ as i,E as j,C as k,p as l,F as m,m as n,q as o,f as p,P as q,T as r,$ as s,h as t,k as u};
package/dist/client.d.ts CHANGED
@@ -1,4 +1,4 @@
1
- export { d as Client, c as ClientEvents, C as ClientOptions, b as ConnectionStateChangeEvent, M as MessageReceivedEvent, S as SubscriptionProvider, e as createClient, u as useLiveQuery } from './index-Y5wSNhd8.js';
1
+ export { d as Client, c as ClientEvents, C as ClientOptions, b as ConnectionStateChangeEvent, M as MessageReceivedEvent, S as SubscriptionProvider, e as createClient, u as useLiveQuery } from './index-BZDuLLmZ.js';
2
2
  import 'react/jsx-runtime';
3
3
  import 'zod';
4
4
  import 'zod/v3';
package/dist/client.js CHANGED
@@ -1 +1 @@
1
- import {d,c,a,e,b as b$1}from'./chunk-VYWZ6RGP.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 pe from'fast-deep-equal';import {openDB}from'idb';var m=a=>xxHash32(JSON.stringify(a)).toString(32),b=(a,e,t=false)=>Object.entries(e).every(([i,n])=>{if(i==="$and")return n.every(r=>b(a,r,t));if(i==="$or")return n.some(r=>b(a,r,t));let s=(n==null?void 0:n.$eq)!==void 0?n==null?void 0:n.$eq:n;if(typeof n=="object"&&n!==null&&(n==null?void 0:n.$eq)===void 0){if(n.$in!==void 0){let r=a[i];return r===void 0?false:t?!n.$in.includes(r):n.$in.includes(r)}if(n.$not!==void 0&&!t)return b(a,{[i]:n.$not},true);if(n.$gt!==void 0){let r=a[i];return typeof r!="number"?false:t?r<=n.$gt:r>n.$gt}if(n.$gte!==void 0){let r=a[i];return typeof r!="number"?false:t?r<n.$gte:r>=n.$gte}if(n.$lt!==void 0){let r=a[i];return typeof r!="number"?false:t?r>=n.$lt:r<n.$lt}if(n.$lte!==void 0){let r=a[i];return typeof r!="number"?false:t?r>n.$lte:r<=n.$lte}return !a[i]||typeof a[i]!="object"?false:b(a[i],n,t)}return t?a[i]!==s:a[i]===s});var j=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 s;(s=this.subscriptions.get(t))==null||s.callbacks.add(i);let n=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),n());},10);}},callbacks:new Set}),this.subscriptions.get(t).subscribe)}},te=new j,fe=a=>useSyncExternalStore(te.getOrStoreSubscription(a),a.get),Te=({children:a,client:e})=>(useEffect(()=>{e.subscribe();},[e.subscribe]),jsx(Fragment,{children:a}));var B=z.object({resource:z.string(),where:z.record(z.string(),z.any()).optional(),include:z.record(z.string(),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()}),C=z.record(z.string(),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((a,e)=>{a.id&&e.addIssue({code:z.ZodIssueCode.custom,message:"Payload cannot have an id"});}),U=z.object({id:z.string().optional(),type:z.literal("MUTATE"),resource:z.string(),resourceId:z.string().optional()}),A=U.extend({procedure:z.string(),payload:z.any().optional()}),I=U.extend({procedure:z.enum(["INSERT","UPDATE"]),payload:C});z.union([I,A]);var g=z.string(),re=z.object({id:g,type:z.literal("SUBSCRIBE"),resource:z.string()}),se=B.extend({id:g,type:z.literal("QUERY")}),H=I.extend({id:g}),ae=A.extend({id:g}),oe=z.union([ae,H]);z.union([re,se,oe]);var ce=z.object({id:g,type:z.literal("REJECT"),resource:z.string(),message:z.string().optional()}),le=z.object({id:g,type:z.literal("REPLY"),data:z.any()}),Q=z.union([ce,le,H]),J=z.object({resource:z.string(),data:z.record(z.string(),C)});var R=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$1(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(n=>{n(t);});}};var w=a=>{if(a)return Array.isArray(a.value)?a.value.map(e=>w(e)):typeof a.value!="object"||a.value===null||a.value instanceof Date?a.value:Object.fromEntries(Object.entries(a.value).map(([e,t])=>[e,w(t)]))};var x=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 n={id:e,type:t,referencedBy:new Map(i.map(s=>[s,new Set])),references:new Map,subscriptions:new Set};return this.nodes.set(e,n),n}getNode(e){return this.nodes.get(e)}hasNode(e){return this.nodes.has(e)}createLink(e,t){let i=this.nodes.get(e),n=this.nodes.get(t);if(!i)throw new Error(`Source node with id ${e} does not exist`);if(!n)throw new Error(`Target node with id ${t} does not exist`);i.references.set(n.type,t);let s=n.referencedBy.get(i.type);s&&s instanceof Set?s.add(e):n.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 n=i.references.get(t);if(!n)return;i.references.delete(t);let s=this.nodes.get(n);if(!s)return;let r=s.referencedBy.get(i.type);r&&(r instanceof Set?r.delete(e):s.referencedBy.delete(i.type),this.notifySubscribers(n)),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,n])=>{(n instanceof Set?Array.from(n.values()):[n]).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(n){console.error(`Error in node subscription for node ${e}:`,n);}});}getAllNodes(){return Array.from(this.nodes.values())}};var D="__meta",L="databases",O=class{db;async init(e,t){var p,d;if(typeof window>"u")return;let n=((p=(await window.indexedDB.databases()).find(u=>u.name===t))==null?void 0:p.version)??1,s=await m(e),r=Object.fromEntries(await Promise.all(Object.entries(e).map(async([u,h])=>[u,await m(h)]))),o=await openDB("live-state-databases",1,{upgrade(u){u.objectStoreNames.contains(L)||u.createObjectStore(L);}}),c=(d=await this.getAll(o,L))==null?void 0:d[t];(c==null?void 0:c.schemaHash)!==s&&n++,this.db=await openDB(t,n,{async upgrade(u){[...Object.keys(e),D].forEach(h=>{(c==null?void 0:c.objectHashes[h])!==r[h]&&u.objectStoreNames.contains(h)&&u.deleteObjectStore(h),u.objectStoreNames.contains(h)||u.createObjectStore(h);}),await o.put(L,{schemaHash:s,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 n;return (n=this.db)==null?void 0:n.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(D,e):new Promise(t=>t(void 0))}setMeta(e,t){var i;return (i=this.db)==null?void 0:i.put(D,t,e)}async getAll(e,t){if(!e)return;if(e.getAllRecords)return e.getAllRecords(t);let[i,n]=await Promise.all([e.getAll(t),e.getAllKeys(t)]);return Object.fromEntries(i.map((s,r)=>[n[r],s]))}};var E=class{constructor(e,t,i){this.schema=e;this.kvStorage=new O,t!==false&&this.kvStorage.init(this.schema,t.name).then(()=>{this.kvStorage.getMeta("mutationStack").then(n=>{!n||Object.keys(n).length===0||(this.optimisticMutationStack=n,i==null||i(this.optimisticMutationStack));}).then(()=>{Object.entries(this.schema).forEach(([n,s])=>{this.kvStorage.get(n).then(r=>{!r||Object.keys(r).length===0||this.loadConsolidatedState(n,r);});});}).catch(n=>{console.error("Failed to load state from storage",n);});});}rawObjPool={};optimisticMutationStack={};optimisticObjGraph=new x;optimisticRawObjPool={};collectionSubscriptions=new Map;querySnapshots={};kvStorage;get(e$1,t,i=false){var r;let n=t??m(e$1);if(this.querySnapshots[n]&&!i){let o=this.querySnapshots[n];if(o)return o}let s=((r=e$1.where)!=null&&r.id?[e$1.where.id]:Object.keys(this.optimisticRawObjPool[e$1.resource]??{})).flatMap(o=>{let c=w(this.materializeOneWithInclude(o,e$1.include));return c?[c]:[]});if(e$1.sort&&e$1.sort.length>0){let o=(c,p)=>{for(let d of e$1.sort){let u=c[d.key],h=p[d.key];if(u<h)return d.direction==="asc"?-1:1;if(u>h)return d.direction==="asc"?1:-1}return 0};s.sort(o);}if(e$1.where||e$1.limit){let o=e$1.where?c=>b(c,e$1.where):()=>true;s=e(s,o,e$1.limit);}return i||(this.querySnapshots[n]=s),s}subscribe(e,t){var r;let i=m(e),n=this.collectionSubscriptions.get(i),s=this.schema[e.resource];return n||this.collectionSubscriptions.set(i,{callbacks:new Set,query:e,flatInclude:e.include?Object.keys(e.include).map(o=>s.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 r,o,c;let n=this.schema[e];if(console.log("Adding mutation",t),!n)throw new Error("Schema not found");let s=(r=this.optimisticRawObjPool[e])==null?void 0:r[t.resourceId];if(i)this.optimisticMutationStack[e]??=[],this.optimisticMutationStack[e].push(t);else {this.optimisticMutationStack[e]=((c=(o=this.optimisticMutationStack)==null?void 0:o[e])==null?void 0:c.filter(u=>u.id!==t.id))??[],this.rawObjPool[e]??={};let p={value:{...this.schema[e].mergeMutation("set",t.payload,this.rawObjPool[e][t.resourceId])[0].value,id:{value:t.resourceId}}};this.rawObjPool[e][t.resourceId]=p;let d=p.value;delete d.id,this.kvStorage.set(e,t.resourceId,d);}this.kvStorage.setMeta("mutationStack",this.optimisticMutationStack),this.updateRawObjPool(e,t.resourceId,t.payload,s);}undoMutation(e,t){var r,o;if(!this.optimisticMutationStack[e])return;let i=(r=this.optimisticMutationStack[e])==null?void 0:r.findIndex(c=>c.id===t);if(i===-1)return;let n=this.optimisticMutationStack[e][i];console.log("Removing mutation",n);let s=(o=this.optimisticRawObjPool[e])==null?void 0:o[n.resourceId];this.optimisticMutationStack[e].splice(i,1),this.kvStorage.setMeta("mutationStack",this.optimisticMutationStack),this.updateRawObjPool(e,n.resourceId,Object.fromEntries(Object.entries(n.payload).map(([c])=>[c,{value:null,_meta:{}}])),s);}loadConsolidatedState(e,t){Object.entries(t).forEach(([i,n])=>{this.addMutation(e,{id:i,type:"MUTATE",resource:e,resourceId:i,procedure:"INSERT",payload:n});});}updateRawObjPool(e,t,i,n){var o;if(!this.schema[e])return;let s=(o=this.rawObjPool[e])==null?void 0:o[t],r=(this.optimisticMutationStack[e]??[]).reduce((c,p)=>p.resourceId!==t?c:this.schema[e].mergeMutation("set",p.payload,c)[0],s);if(this.optimisticRawObjPool[e]??={},r?this.optimisticRawObjPool[e][t]={value:{...r.value,id:{value:t}}}:delete this.optimisticRawObjPool[e][t],!(!this.optimisticObjGraph.hasNode(t)&&!r)){if(this.optimisticObjGraph.hasNode(t)||this.optimisticObjGraph.createNode(t,e,Object.values(this.schema[e].relations).flatMap(c=>c.type==="many"?[c.entity.name]:[])),Object.keys(this.schema[e].relations).length>0){let c=Object.fromEntries(Object.entries(this.schema[e].relations).flatMap(([p,d])=>d.type==="one"?[[d.relationalColumn,p]]:[]));Object.entries(i).forEach(([p,d])=>{let u=this.schema[e].relations[c[p]];if(!c[p])return;let h=n==null?void 0:n.value[p],[,T]=u.mergeMutation("set",d,h);if(T){if(!this.optimisticObjGraph.hasNode(T.value)){let $=u.entity.name;this.optimisticObjGraph.createNode(T.value,$,Object.values(this.schema[$].relations).flatMap(q=>q.type==="many"?[q.entity.name]:[]));}h!=null&&h.value&&this.optimisticObjGraph.removeLink(t,u.entity.name),this.optimisticObjGraph.createLink(t,T.value);}});}this.notifyCollectionSubscribers(e),this.optimisticObjGraph.notifySubscribers(t);}}materializeOneWithInclude(e,t={}){var c;if(!e)return;let i=this.optimisticObjGraph.getNode(e);if(!i)return;let n=i.type,s=(c=this.optimisticRawObjPool[n])==null?void 0:c[e];if(!s)return;let[r,o]=Object.keys(t).reduce((p,d)=>{let u=this.schema[n].relations[d];return u.type==="one"?p[0].push([d,u.entity.name]):u.type==="many"&&p[1].push([d,u.entity.name]),p},[[],[]]);return {value:{...s.value,...Object.fromEntries(r.map(([p,d])=>[p,this.materializeOneWithInclude(i.references.get(d))])),...Object.fromEntries(o.map(([p,d])=>{let u=i.referencedBy.get(d),h=u instanceof Set;return [p,h?{value:Array.from(u.values()).map(T=>this.materializeOneWithInclude(T))}:this.materializeOneWithInclude(u)]}))}}}notifyCollectionSubscribers(e){this.collectionSubscriptions.forEach(t=>{if(t.query.resource===e||t.flatInclude&&t.flatInclude.includes(e)){let i=m(t.query),n=this.querySnapshots[i],s=this.get(t.query,void 0,true);if(pe(s,n))return;this.querySnapshots[i]=s,t.callbacks.forEach(r=>{r(s);});}});}};var W=class{url;ws;store;remoteSubCounters={};eventListeners=new Set;replyHandlers={};constructor(e){this.url=e.url,this.store=new E(e.schema,e.storage,t=>{var i,n;(n=(i=Object.values(t))==null?void 0:i.flat())==null||n.forEach(s=>{this.sendWsMessage(s);});}),this.ws=new R({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,n])=>{n>0&&this.sendWsMessage({id:a(),type:"SUBSCRIBE",resource:i});}),Object.values(this.store.optimisticMutationStack).forEach(i=>{i&&i.forEach(n=>{this.sendWsMessage(n);});}));});}get(e){return this.store.get(e)}handleServerMessage(e){try{console.log("Message received from the server:",e);let t=Q.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(n){console.error("Error merging mutation from the server:",n);}}else if(t.type==="REJECT")this.store.undoMutation(t.resource,t.id);else if(t.type==="REPLY"){let{id:i,data:n}=t;if(this.replyHandlers[i]){clearTimeout(this.replyHandlers[i].timeoutHandle),this.replyHandlers[i].handler(n);return}let s=J.parse(n);this.store.loadConsolidatedState(s.resource,s.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,n){var r;let s={id:a(),type:"MUTATE",resource:e,payload:this.store.schema[e].encodeMutation("set",n,new Date().toISOString()),resourceId:t,procedure:i};(r=this.store)==null||r.addMutation(e,s,true),this.sendWsMessage(s);}genericMutate(e,t,i){if(!this.ws||!this.ws.connected())throw new Error("WebSocket not connected");let n={id:a(),type:"MUTATE",resource:e,procedure:t,payload:i};return this.sendWsMessage(n),new Promise((s,r)=>{this.replyHandlers[n.id]={timeoutHandle:setTimeout(()=>{delete this.replyHandlers[n.id],r(new Error("Reply timeout"));},5e3),handler:o=>{delete this.replyHandlers[n.id],s(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);});}},ot=a=>{let e=new W(a);return {client:{ws:e.ws,subscribe:t=>{let i=[];for(let n of t??Object.keys(e.store.schema))i.push(e.subscribeToRemote(n));return ()=>{console.log("Removing listeners",i),i.forEach(n=>{n();});}},addEventListener:t=>e.addEventListener(t)},store:{query:Object.entries(a.schema).reduce((t,[i,n])=>(t[i]=c._init(n,e),t),{}),mutate:d(()=>{},{apply:(t,i,n)=>{if(i.length<2)return;if(i.length>2)throw new Error("Trying to access an invalid path");let[s,r]=i;if(r==="insert"){let{id:o,...c}=n[0];return e.mutate(s,o,"INSERT",c)}if(r==="update"){let[o,c]=n;return e.mutate(s,o,"UPDATE",c)}return e.genericMutate(s,r,n[0])}})}}};export{Te as SubscriptionProvider,ot as createClient,fe as useLiveQuery};
1
+ import {d,c,a,f,e,b}from'./chunk-WZ7JXHZL.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 ne from'fast-deep-equal';import {openDB}from'idb';var y=u=>xxHash32(JSON.stringify(u)).toString(32),g=(u,e,t=false)=>Object.entries(e).every(([i,s])=>{if(i==="$and")return s.every(n=>g(u,n,t));if(i==="$or")return s.some(n=>g(u,n,t));let r=(s==null?void 0:s.$eq)!==void 0?s==null?void 0:s.$eq:s;if(typeof s=="object"&&s!==null&&(s==null?void 0:s.$eq)===void 0){if(s.$in!==void 0){let n=u[i];return n===void 0?false:t?!s.$in.includes(n):s.$in.includes(n)}if(s.$not!==void 0&&!t)return g(u,{[i]:s.$not},true);if(s.$gt!==void 0){let n=u[i];return typeof n!="number"?false:t?n<=s.$gt:n>s.$gt}if(s.$gte!==void 0){let n=u[i];return typeof n!="number"?false:t?n<s.$gte:n>=s.$gte}if(s.$lt!==void 0){let n=u[i];return typeof n!="number"?false:t?n>=s.$lt:n<s.$lt}if(s.$lte!==void 0){let n=u[i];return typeof n!="number"?false:t?n>s.$lte:n<=s.$lte}return !u[i]||typeof u[i]!="object"?false:g(u[i],s,t)}return t?u[i]!==r:u[i]===r});var R=class{subscriptions=new Map;getOrStoreSubscription(e){let t=y(e);return this.subscriptions.has(t)?this.subscriptions.get(t).subscribe:(this.subscriptions.set(t,{subscribe:i=>{var r;(r=this.subscriptions.get(t))==null||r.callbacks.add(i);let s=e.subscribe(()=>{var n;(n=this.subscriptions.get(t))==null||n.callbacks.forEach(o=>{o();});});return ()=>{var n;(n=this.subscriptions.get(t))==null||n.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)}},N=new R,le=u=>useSyncExternalStore(N.getOrStoreSubscription(u),u.get),ue=({children:u,client:e})=>(useEffect(()=>{e.subscribe();},[e.subscribe]),jsx(Fragment,{children:u}));var $=z.object({resource:z.string(),where:z.record(z.string(),z.any()).optional(),include:z.record(z.string(),z.any()).optional(),lastSyncedAt:z.string().optional(),limit:z.coerce.number().optional(),sort:z.array(z.object({key:z.string(),direction:z.enum(["asc","desc"])})).optional()}),C=z.record(z.string(),z.object({value:z.string().or(z.number()).or(z.boolean()).or(z.date()).nullable(),_meta:z.object({timestamp:z.string().optional().nullable()}).optional()})),Y=C.superRefine((u,e)=>{u.id&&e.addIssue({code:z.ZodIssueCode.custom,message:"Payload cannot have an id"});}),G=z.object({id:z.string().optional(),type:z.literal("MUTATE"),resource:z.string(),resourceId:z.string().optional()}),j=G.extend({procedure:z.string(),payload:z.any().optional()}),k=G.extend({procedure:z.enum(["INSERT","UPDATE"]),payload:Y});z.union([k,j]);var v=z.string(),F=z.object({id:v,type:z.literal("SUBSCRIBE"),resource:z.string()}),Z=$.extend({id:v,type:z.literal("QUERY")}),I=k.extend({id:v}),X=j.extend({id:v}),ee=z.union([X,I]);z.union([F,Z,ee]);var te=z.object({id:v,type:z.literal("REJECT"),resource:z.string(),message:z.string().optional()}),ie=z.object({id:v,type:z.literal("REPLY"),data:z.any()}),H=z.union([te,ie,I]),Q=z.object({resource:z.string(),data:z.record(z.string(),C)});var M=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 w=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){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 r=s.referencedBy.get(i.type);r&&r instanceof Set?r.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 r=this.nodes.get(s);if(!r)return;let n=r.referencedBy.get(i.type);n&&(n instanceof Set?n.delete(e):r.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(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="__meta",E="databases",O=class{db;async init(e,t){var p,h;if(typeof window>"u")return;let s=((p=(await window.indexedDB.databases()).find(c=>c.name===t))==null?void 0:p.version)??1,r=await y(e),n=Object.fromEntries(await Promise.all(Object.entries(e).map(async([c,d])=>[c,await y(d)]))),o=await openDB("live-state-databases",1,{upgrade(c){c.objectStoreNames.contains(E)||c.createObjectStore(E);}}),a=(h=await this.getAll(o,E))==null?void 0:h[t];(a==null?void 0:a.schemaHash)!==r&&s++,this.db=await openDB(t,s,{async upgrade(c){[...Object.keys(e),A].forEach(d=>{(a==null?void 0:a.objectHashes[d])!==n[d]&&c.objectStoreNames.contains(d)&&c.deleteObjectStore(d),c.objectStoreNames.contains(d)||c.createObjectStore(d);}),await o.put(E,{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(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((r,n)=>[s[n],r]))}};var T=class{constructor(e,t,i){this.schema=e;this.kvStorage=new O,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,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 w;optimisticRawObjPool={};collectionSubscriptions=new Map;querySnapshots={};kvStorage;get(e$1,t,i=false){var n;let s=t??y(e$1);if(this.querySnapshots[s]&&!i){let o=this.querySnapshots[s];if(o)return o}let r=((n=e$1.where)!=null&&n.id?[e$1.where.id]:Object.keys(this.optimisticRawObjPool[e$1.resource]??{})).flatMap(o=>{let a=f(this.materializeOneWithInclude(o,e$1.include));return a?[a]:[]});if(e$1.sort&&e$1.sort.length>0){let o=(a,p)=>{for(let h of e$1.sort){let c=a[h.key],d=p[h.key];if(c<d)return h.direction==="asc"?-1:1;if(c>d)return h.direction==="asc"?1:-1}return 0};r.sort(o);}if(e$1.where||e$1.limit){let o=e$1.where?a=>g(a,e$1.where):()=>true;r=e(r,o,e$1.limit);}return i||(this.querySnapshots[s]=r),r}subscribe(e,t){var r;let i=y(e);return this.collectionSubscriptions.get(i)||this.collectionSubscriptions.set(i,{callbacks:new Set,query:e,flatInclude:e.include?this.flattenIncludes(e.include,e.resource):void 0}),(r=this.collectionSubscriptions.get(i))==null||r.callbacks.add(t),()=>{var n,o;(n=this.collectionSubscriptions.get(i))==null||n.callbacks.delete(t),((o=this.collectionSubscriptions.get(i))==null?void 0:o.callbacks.size)===0&&(this.collectionSubscriptions.delete(i),delete this.querySnapshots[i]);}}addMutation(e,t,i=false){var n,o,a;let s=this.schema[e];if(console.log("Adding mutation",t),!s)throw new Error("Schema not found");let r=(n=this.optimisticRawObjPool[e])==null?void 0:n[t.resourceId];if(i)this.optimisticMutationStack[e]??=[],this.optimisticMutationStack[e].push(t);else {this.optimisticMutationStack[e]=((a=(o=this.optimisticMutationStack)==null?void 0:o[e])==null?void 0:a.filter(c=>c.id!==t.id))??[],this.rawObjPool[e]??={};let p={value:{...this.schema[e].mergeMutation("set",t.payload,this.rawObjPool[e][t.resourceId])[0].value,id:{value:t.resourceId}}};this.rawObjPool[e][t.resourceId]=p;let h=p.value;delete h.id,this.kvStorage.set(e,t.resourceId,h);}this.kvStorage.setMeta("mutationStack",this.optimisticMutationStack),this.updateRawObjPool(e,t.resourceId,t.payload,r);}undoMutation(e,t){var n,o;if(!this.optimisticMutationStack[e])return;let i=(n=this.optimisticMutationStack[e])==null?void 0:n.findIndex(a=>a.id===t);if(i===-1)return;let s=this.optimisticMutationStack[e][i];console.log("Removing mutation",s);let r=(o=this.optimisticRawObjPool[e])==null?void 0:o[s.resourceId];this.optimisticMutationStack[e].splice(i,1),this.kvStorage.setMeta("mutationStack",this.optimisticMutationStack),this.updateRawObjPool(e,s.resourceId,Object.fromEntries(Object.entries(s.payload).map(([a])=>[a,{value:null,_meta:{}}])),r);}loadConsolidatedState(e,t){Object.entries(t).forEach(([i,s])=>{this.addMutation(e,{id:i,type:"MUTATE",resource:e,resourceId:i,procedure:"INSERT",payload:s});});}updateRawObjPool(e,t,i,s){var o;if(!this.schema[e])return;let r=(o=this.rawObjPool[e])==null?void 0:o[t],n=(this.optimisticMutationStack[e]??[]).reduce((a,p)=>p.resourceId!==t?a:this.schema[e].mergeMutation("set",p.payload,a)[0],r);if(this.optimisticRawObjPool[e]??={},n?this.optimisticRawObjPool[e][t]={value:{...n.value,id:{value:t}}}:delete this.optimisticRawObjPool[e][t],!(!this.optimisticObjGraph.hasNode(t)&&!n)){if(this.optimisticObjGraph.hasNode(t)||this.optimisticObjGraph.createNode(t,e,Object.values(this.schema[e].relations).flatMap(a=>a.type==="many"?[a.entity.name]:[])),Object.keys(this.schema[e].relations).length>0){let a=Object.fromEntries(Object.entries(this.schema[e].relations).flatMap(([p,h])=>h.type==="one"?[[h.relationalColumn,p]]:[]));Object.entries(i).forEach(([p,h])=>{let c=this.schema[e].relations[a[p]];if(!a[p])return;let d=s==null?void 0:s.value[p],[,m]=c.mergeMutation("set",h,d);if(m){if(!this.optimisticObjGraph.hasNode(m.value)){let S=c.entity.name;this.optimisticObjGraph.createNode(m.value,S,Object.values(this.schema[S].relations).flatMap(x=>x.type==="many"?[x.entity.name]:[]));}d!=null&&d.value&&this.optimisticObjGraph.removeLink(t,c.entity.name),this.optimisticObjGraph.createLink(t,m.value);}});}this.notifyCollectionSubscribers(e),this.optimisticObjGraph.notifySubscribers(t);}}materializeOneWithInclude(e,t={}){var a;if(!e)return;let i=this.optimisticObjGraph.getNode(e);if(!i)return;let s=i.type,r=(a=this.optimisticRawObjPool[s])==null?void 0:a[e];if(!r)return;let[n,o]=Object.entries(t).reduce((p,[h,c])=>{let d=this.schema[s].relations[h];return d&&(d.type==="one"?p[0].push([h,d.entity.name,c??true]):d.type==="many"&&p[1].push([h,d.entity.name,c??true])),p},[[],[]]);return {value:{...r.value,...Object.fromEntries(n.map(([p,h,c])=>[p,this.materializeOneWithInclude(i.references.get(h),typeof c=="object"&&c!==null?c:{})])),...Object.fromEntries(o.map(([p,h,c])=>{let d=i.referencedBy.get(h),m=d instanceof Set;return [p,m?{value:Array.from(d.values()).map(S=>this.materializeOneWithInclude(S,typeof c=="object"&&c!==null?c:{}))}:this.materializeOneWithInclude(d,typeof c=="object"&&c!==null?c:{})]}))}}}notifyCollectionSubscribers(e){this.collectionSubscriptions.forEach(t=>{if(t.query.resource===e||t.flatInclude&&t.flatInclude.includes(e)){let i=y(t.query),s=this.querySnapshots[i],r=this.get(t.query,void 0,true);if(ne(r,s))return;this.querySnapshots[i]=r,t.callbacks.forEach(n=>{n(r);});}});}flattenIncludes(e,t){let i=[];return Object.entries(e).forEach(([s,r])=>{var a;let n=(a=this.schema[t])==null?void 0:a.relations[s];if(!n)return;let o=n.entity.name;i.push(o),typeof r=="object"&&r!==null&&i.push(...this.flattenIncludes(r,o));}),Array.from(new Set(i))}};var L=class{url;ws;store;remoteSubCounters={};eventListeners=new Set;replyHandlers={};constructor(e){var t,i,s,r;this.url=e.url,this.store=new T(e.schema,e.storage,n=>{var o,a;(a=(o=Object.values(n))==null?void 0:o.flat())==null||a.forEach(p=>{this.sendWsMessage(p);});}),this.ws=new M({url:e.url,autoConnect:((t=e.connection)==null?void 0:t.autoConnect)??true,autoReconnect:((i=e.connection)==null?void 0:i.autoReconnect)??true,reconnectTimeout:((s=e.connection)==null?void 0:s.reconnectTimeout)??5e3,reconnectLimit:(r=e.connection)==null?void 0:r.maxReconnectAttempts,credentials:e.credentials}),this.ws.addEventListener("message",n=>{this.handleServerMessage(n.data);}),this.ws.addEventListener("connectionChange",n=>{this.emitEvent({type:"CONNECTION_STATE_CHANGE",open:n.open}),n.open&&(Object.keys(this.store.schema).forEach(o=>{this.sendWsMessage({id:a(),type:"QUERY",resource:o});}),Object.entries(this.remoteSubCounters).forEach(([o,a$1])=>{a$1>0&&this.sendWsMessage({id:a(),type:"SUBSCRIBE",resource:o});}),Object.values(this.store.optimisticMutationStack).forEach(o=>{o&&o.forEach(a=>{this.sendWsMessage(a);});}));});}get(e){return this.store.get(e)}handleServerMessage(e){try{console.log("Message received from the server:",e);let t=H.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")this.store.undoMutation(t.resource,t.id);else 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 r=Q.parse(s);this.store.loadConsolidatedState(r.resource,r.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,s){var n;let r={id:a(),type:"MUTATE",resource:e,payload:this.store.schema[e].encodeMutation("set",s,new Date().toISOString()),resourceId:t,procedure:i};(n=this.store)==null||n.addMutation(e,r,true),this.sendWsMessage(r);}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);}};})}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);});}},Ie=u=>{let e=new L(u);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(u.schema).reduce((t,[i,s])=>(t[i]=c._init(s,e),t),{}),mutate:d(()=>{},{apply:(t,i,s)=>{if(i.length<2)return;if(i.length>2)throw new Error("Trying to access an invalid path");let[r,n]=i;if(n==="insert"){let{id:o,...a}=s[0];return e.mutate(r,o,"INSERT",a)}if(n==="update"){let[o,a]=s;return e.mutate(r,o,"UPDATE",a)}return e.genericMutate(r,n,s[0])}})}}};export{ue as SubscriptionProvider,Ie as createClient,le as useLiveQuery};
@@ -1,4 +1,4 @@
1
- import { A as AnyRouter, C as ClientOptions, a as Client } from './index-Y5wSNhd8.js';
1
+ import { A as AnyRouter, C as ClientOptions, a as Client } from './index-BZDuLLmZ.js';
2
2
  import 'react/jsx-runtime';
3
3
  import 'zod';
4
4
  import 'zod/v3';
@@ -1 +1 @@
1
- import {d,b,c}from'./chunk-VYWZ6RGP.js';import {stringify}from'qs';var i=async(...e)=>{let n=await fetch(...e),t;try{t=await n.json();}catch{t=await n.text().catch(()=>{});}if(!n.ok)throw new Error(`Failed to fetch: ${n.status} ${n.statusText}`,{cause:t});return t},$=e=>{let n={get:async t=>{let r=stringify(t),o=await b(e.credentials)??{};return await i(`${e.url}/${t.resource}${r?`?${r}`:""}`,{headers:{...o,"Content-Type":"application/json"}})},subscribe:()=>{throw new Error("Fetch client does not support subscriptions")}};return {query:Object.entries(e.schema).reduce((t,[r,o])=>(t[r]=c._init(o,n,true),t),{}),mutate:d(()=>{},{apply:async(t,r,o)=>{if(r.length<2)return;if(r.length>2)throw new Error("Trying to access an invalid path");let[a,s]=r,u=await b(e.credentials)??{};if(s==="insert"){let{id:c,...y}=o[0];return i(`${e.url}/${a}/insert`,{method:"POST",headers:{...u,"Content-Type":"application/json"},body:JSON.stringify({resourceId:c,payload:e.schema[a].encodeMutation("set",y,new Date().toISOString())})})}if(s==="update"){let[c,y]=o,{id:h,...m}=y;return i(`${e.url}/${a}/update`,{method:"POST",headers:{...u,"Content-Type":"application/json"},body:JSON.stringify({resourceId:c,payload:e.schema[a].encodeMutation("set",m,new Date().toISOString())})})}return i(`${e.url}/${a}/${s}`,{method:"POST",headers:{...u,"Content-Type":"application/json"},body:JSON.stringify({payload:o[0]})})}})}};export{$ as createClient};
1
+ import {d,b,c,f}from'./chunk-WZ7JXHZL.js';import {stringify}from'qs';var u=async(...e)=>{let o=await fetch(...e),t;try{t=await o.json();}catch{t=await o.text().catch(()=>{});}if(!o.ok)throw new Error(`Failed to fetch: ${o.status} ${o.statusText}`,{cause:t});return t},S=e=>{let o={get:async t=>{let n=stringify(t),a=await b(e.credentials)??{},r=await u(`${e.url}/${t.resource}${n?`?${n}`:""}`,{headers:{...a,"Content-Type":"application/json"}});return !r||typeof r!="object"?[]:Object.entries(r).map(([i,s])=>({...f(s),id:i}))},subscribe:()=>{throw new Error("Fetch client does not support subscriptions")}};return {query:Object.entries(e.schema).reduce((t,[n,a])=>(t[n]=c._init(a,o,true),t),{}),mutate:d(()=>{},{apply:async(t,n,a)=>{if(n.length<2)return;if(n.length>2)throw new Error("Trying to access an invalid path");let[r,i]=n,s=await b(e.credentials)??{};if(i==="insert"){let{id:c,...y}=a[0];await u(`${e.url}/${r}/insert`,{method:"POST",headers:{...s,"Content-Type":"application/json"},body:JSON.stringify({resourceId:c,payload:e.schema[r].encodeMutation("set",y,new Date().toISOString())})});return}if(i==="update"){let[c,y]=a,{id:O,...m}=y;await u(`${e.url}/${r}/update`,{method:"POST",headers:{...s,"Content-Type":"application/json"},body:JSON.stringify({resourceId:c,payload:e.schema[r].encodeMutation("set",m,new Date().toISOString())})});return}await u(`${e.url}/${r}/${i}`,{method:"POST",headers:{...s,"Content-Type":"application/json"},body:JSON.stringify({payload:a[0]})});}})}};export{S as createClient};
@@ -3,11 +3,6 @@ import { z } from 'zod';
3
3
  import * as z3 from 'zod/v3';
4
4
  import * as z4 from 'zod/v4/core';
5
5
 
6
- type Promisify<T> = T extends Promise<any> ? T : Promise<T>;
7
- type ConditionalPromise<T, P extends boolean> = P extends true ? Promise<T> : T;
8
- type Awaitable<T> = T | Promise<T>;
9
- type Generatable<T, Arg = never> = T | ((arg: Arg) => T);
10
-
11
6
  type LiveTypeMeta = {};
12
7
  type MutationType = "set";
13
8
  type StorageFieldType = {
@@ -202,7 +197,7 @@ type WhereClause<T extends LiveObjectAny> = ({
202
197
  $in?: InferLiveType<T["fields"][K]>[];
203
198
  $eq?: InferLiveType<T["fields"][K]>;
204
199
  };
205
- } & (InferLiveType<T["fields"][K]> extends number ? {
200
+ } & (Exclude<InferLiveType<T["fields"][K]>, null | undefined> extends number | Date ? {
206
201
  $gt?: InferLiveType<T["fields"][K]>;
207
202
  $gte?: InferLiveType<T["fields"][K]>;
208
203
  $lt?: InferLiveType<T["fields"][K]>;
@@ -215,7 +210,7 @@ type WhereClause<T extends LiveObjectAny> = ({
215
210
  $or?: WhereClause<T>[];
216
211
  };
217
212
  type IncludeClause<T extends LiveObjectAny> = {
218
- [K in keyof T["relations"]]?: boolean;
213
+ [K in keyof T["relations"]]?: boolean | IncludeClause<T["relations"][K]["entity"]>;
219
214
  };
220
215
  type GetFieldType<T> = T extends NullableLiveType<any> ? T["inner"] : T;
221
216
  type HasDefaultValue<T> = T extends LiveAtomicType<any, undefined> ? false : true;
@@ -226,6 +221,27 @@ type InferInsert<T extends LiveObjectAny> = {
226
221
  };
227
222
  type InferUpdate<T extends LiveObjectAny> = Omit<LiveObjectMutationInput<T>, "id">;
228
223
 
224
+ type Promisify<T> = T extends Promise<any> ? T : Promise<T>;
225
+ type ConditionalPromise<T, P extends boolean> = P extends true ? Promise<T> : T;
226
+ type Awaitable<T> = T | Promise<T>;
227
+ type Generatable<T, Arg = never> = T | ((arg: Arg) => T);
228
+
229
+ declare const querySchema: z.ZodObject<{
230
+ resource: z.ZodString;
231
+ where: z.ZodOptional<z.ZodRecord<z.ZodString, z.ZodAny>>;
232
+ include: z.ZodOptional<z.ZodRecord<z.ZodString, z.ZodAny>>;
233
+ lastSyncedAt: z.ZodOptional<z.ZodString>;
234
+ limit: z.ZodOptional<z.ZodCoercedNumber<unknown>>;
235
+ sort: z.ZodOptional<z.ZodArray<z.ZodObject<{
236
+ key: z.ZodString;
237
+ direction: z.ZodEnum<{
238
+ asc: "asc";
239
+ desc: "desc";
240
+ }>;
241
+ }, z.core.$strip>>>;
242
+ }, z.core.$strip>;
243
+ type RawQueryRequest = z.infer<typeof querySchema>;
244
+
229
245
  type Simplify<T> = T extends Record<string, unknown> ? {
230
246
  [K in keyof T]: Simplify<T[K]>;
231
247
  } : T extends Array<infer U> ? Array<Simplify<U>> : T;
@@ -242,24 +258,22 @@ declare class Router<TRoutes extends RouteRecord> {
242
258
  }): Router<TRoutes>;
243
259
  }
244
260
  type AnyRouter = Router<any>;
245
- type RequestHandler<TInput, TResult, TSchema extends Schema<any> = Schema<any>> = (opts: {
246
- req: ParsedRequest<TInput>;
247
- db: Storage;
248
- schema: TSchema;
249
- }) => Promise<TResult>;
250
261
  type Mutation<TInputValidator extends z3.ZodTypeAny | z4.$ZodType, // TODO use StandardSchema instead
251
- THandler extends RequestHandler<z.infer<TInputValidator>, any, any>> = {
262
+ TOutput> = {
252
263
  inputValidator: TInputValidator;
253
- handler: THandler;
264
+ handler: (opts: {
265
+ req: MutationRequest<z.infer<TInputValidator>>;
266
+ db: Storage;
267
+ }) => TOutput;
254
268
  };
255
269
  declare const mutationCreator: <TInputValidator extends z3.ZodTypeAny | z4.$ZodType>(validator?: TInputValidator) => {
256
- handler: <THandler extends RequestHandler<z.infer<TInputValidator>, any, any>>(handler: THandler) => Mutation<TInputValidator, THandler>;
270
+ handler: <THandler extends Mutation<TInputValidator, any>["handler"]>(handler: THandler) => Mutation<TInputValidator, ReturnType<THandler>>;
257
271
  };
258
272
  type ReadAuthorizationHandler<TShape extends LiveObjectAny> = (opts: {
259
- ctx: ParsedRequest["context"];
273
+ ctx: BaseRequest["context"];
260
274
  }) => WhereClause<TShape> | boolean;
261
275
  type MutationAuthorizationHandler<TShape extends LiveObjectAny> = (opts: {
262
- ctx: ParsedRequest["context"];
276
+ ctx: BaseRequest["context"];
263
277
  value: Simplify<InferLiveObjectWithRelationalIds<TShape>>;
264
278
  }) => WhereClause<TShape> | boolean;
265
279
  type Authorization<TShape extends LiveObjectAny> = {
@@ -270,24 +284,18 @@ type Authorization<TShape extends LiveObjectAny> = {
270
284
  postMutation?: MutationAuthorizationHandler<TShape>;
271
285
  };
272
286
  };
273
- declare class Route<TResourceSchema extends LiveObjectAny, TMiddleware extends Middleware<any>, TCustomMutations extends Record<string, Mutation<any, RequestHandler<any, any>>>> {
274
- readonly _resourceSchema: TResourceSchema;
275
- readonly resourceName: TResourceSchema["name"];
287
+ declare class Route<TResourceSchema extends LiveObjectAny, TMiddleware extends Middleware<any>, TCustomMutations extends Record<string, Mutation<any, any>>> {
288
+ readonly resourceSchema: TResourceSchema;
276
289
  readonly middlewares: Set<TMiddleware>;
277
290
  readonly customMutations: TCustomMutations;
278
291
  readonly authorization?: Authorization<TResourceSchema>;
279
- constructor(resourceName: TResourceSchema["name"], customMutations?: TCustomMutations, authorization?: Authorization<TResourceSchema>);
280
- handleRequest(opts: {
281
- req: ParsedRequest;
282
- db: Storage;
283
- schema: Schema<any>;
284
- }): Promise<any>;
292
+ constructor(resourceSchema: TResourceSchema, customMutations?: TCustomMutations, authorization?: Authorization<TResourceSchema>);
285
293
  use(...middlewares: TMiddleware[]): this;
286
- withMutations<T extends Record<string, Mutation<any, RequestHandler<any, any>>>>(mutationFactory: (opts: {
294
+ withMutations<T extends Record<string, Mutation<any, any>>>(mutationFactory: (opts: {
287
295
  mutation: typeof mutationCreator;
288
296
  }) => T): Route<TResourceSchema, TMiddleware, T>;
289
- private handleFind;
290
297
  private handleSet;
298
+ private wrapInMiddlewares;
291
299
  }
292
300
  type AnyRoute = Route<LiveObjectAny, Middleware<any>, Record<string, any>>;
293
301
 
@@ -310,22 +318,28 @@ declare abstract class Storage {
310
318
  }) => Promise<T>): Promise<T>;
311
319
  }
312
320
 
313
- type ParsedRequest<TInput = any> = {
321
+ /** biome-ignore-all lint/suspicious/noExplicitAny: any's are actually used correctly */
322
+
323
+ interface BaseRequest {
314
324
  headers: Record<string, string>;
315
325
  cookies: Record<string, string>;
316
- query: Record<string, string>;
317
- resourceName: string;
318
- procedure?: string;
326
+ queryParams: Record<string, string>;
319
327
  context: Record<string, any>;
320
- where?: Record<string, any>;
321
- include?: Record<string, any>;
322
- type: "QUERY" | "MUTATE";
328
+ }
329
+ interface QueryRequest extends BaseRequest, RawQueryRequest {
330
+ type: "QUERY";
331
+ }
332
+ interface MutationRequest<TInput = any> extends BaseRequest {
333
+ type: "MUTATE";
334
+ input: TInput;
335
+ resource: string;
323
336
  resourceId?: string;
324
- input?: TInput;
325
- };
326
- type NextFunction<T> = (req: ParsedRequest) => Promise<T> | T;
337
+ procedure: string;
338
+ }
339
+ type Request = QueryRequest | MutationRequest;
340
+ type NextFunction<O, R = Request> = (req: R) => Awaitable<O>;
327
341
  type Middleware<T = any> = (opts: {
328
- req: ParsedRequest;
342
+ req: BaseRequest;
329
343
  next: NextFunction<T>;
330
344
  }) => ReturnType<NextFunction<T>>;
331
345
 
@@ -400,12 +414,12 @@ declare class QueryBuilder<TCollection extends LiveObjectAny, TInclude extends I
400
414
 
401
415
  type Client$1<TRouter extends AnyRouter, TShouldAwait extends boolean = false> = {
402
416
  query: {
403
- [K in keyof TRouter["routes"]]: QueryBuilder<TRouter["routes"][K]["_resourceSchema"], {}, false, TShouldAwait>;
417
+ [K in keyof TRouter["routes"]]: QueryBuilder<TRouter["routes"][K]["resourceSchema"], {}, false, TShouldAwait>;
404
418
  };
405
419
  mutate: {
406
420
  [K in keyof TRouter["routes"]]: {
407
- insert: (input: Simplify<InferInsert<TRouter["routes"][K]["_resourceSchema"]>>) => ConditionalPromise<void, TShouldAwait>;
408
- update: (id: string, value: Simplify<InferUpdate<TRouter["routes"][K]["_resourceSchema"]>>) => ConditionalPromise<void, TShouldAwait>;
421
+ insert: (input: Simplify<InferInsert<TRouter["routes"][K]["resourceSchema"]>>) => ConditionalPromise<void, TShouldAwait>;
422
+ update: (id: string, value: Simplify<InferUpdate<TRouter["routes"][K]["resourceSchema"]>>) => ConditionalPromise<void, TShouldAwait>;
409
423
  } & {
410
424
  [K2 in keyof TRouter["routes"][K]["customMutations"]]: (input: z.infer<TRouter["routes"][K]["customMutations"][K2]["inputValidator"]>) => Promisify<ReturnType<TRouter["routes"][K]["customMutations"][K2]["handler"]>>;
411
425
  };
@@ -451,6 +465,14 @@ declare class WebSocketClient {
451
465
  private dispatchEvent;
452
466
  }
453
467
 
468
+ interface WebSocketClientOptions extends ClientOptions {
469
+ connection?: {
470
+ autoConnect?: boolean;
471
+ autoReconnect?: boolean;
472
+ reconnectTimeout?: number;
473
+ maxReconnectAttempts?: number;
474
+ };
475
+ }
454
476
  type ConnectionStateChangeEvent = {
455
477
  type: "CONNECTION_STATE_CHANGE";
456
478
  open: boolean;
@@ -468,7 +490,7 @@ type Client<TRouter extends AnyRouter> = {
468
490
  };
469
491
  store: Client$1<TRouter>;
470
492
  };
471
- declare const createClient: <TRouter extends AnyRouter>(opts: ClientOptions) => Client<TRouter>;
493
+ declare const createClient: <TRouter extends AnyRouter>(opts: WebSocketClientOptions) => Client<TRouter>;
472
494
 
473
495
  type ClientOptions = {
474
496
  url: string;
package/dist/index.cjs CHANGED
@@ -1 +1 @@
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}}},l=class i extends s{storageType;convertFunc;isIndex;isUnique;defaultValue;foreignReference;isPrimary;constructor(e,t,n,r,a,u,y){super(),this.storageType=e,this.convertFunc=t,this.isIndex=n??false,this.isUnique=r??false,this.defaultValue=a,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 i(this.storageType,this.convertFunc,this.isIndex,true,this.defaultValue,this.foreignReference,this.isPrimary)}index(){return new i(this.storageType,this.convertFunc,true,this.isUnique,this.defaultValue,this.foreignReference,this.isPrimary)}default(e){return new i(this.storageType,this.convertFunc,this.isIndex,this.isUnique,e,this.foreignReference,this.isPrimary)}primary(){return new i(this.storageType,this.convertFunc,this.isIndex,this.isUnique,this.defaultValue,this.foreignReference,true)}nullable(){return new c(this)}},p=class i extends l{constructor(){super("integer",e=>Number(e));}static create(){return new i}},b=p.create,o=class i extends l{constructor(e){super("varchar",void 0,void 0,void 0,void 0,e);}static create(){return new i}static createId(){return new i().index().unique().primary()}static createReference(e){return new i(e)}},g=o.create,I=o.createId,M=o.createReference,d=class i extends l{constructor(){super("boolean",e=>typeof e=="string"?e.toLowerCase()==="true":!!e);}static create(){return new i}},O=d.create,m=class i extends l{constructor(){super("timestamp",e=>typeof e=="string"?new Date(e):e);}static create(){return new i}},j=m.create;var f=class i 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,a])=>[r,(this.fields[r]??this.relations[r]).encodeMutation("set",a,n)]))}mergeMutation(e,t,n){let r={};return [{value:{...(n==null?void 0:n.value)??{},...Object.fromEntries(Object.entries(t).map(([a,u])=>{let y=this.fields[a]??this.relations[a];if(!y)return [a,u];let[h,v]=y.mergeMutation(e,u,n==null?void 0:n.value[a]);return v&&(r[a]=v),[a,h]}))}},r]}setRelations(e){return new i(this.name,this.fields,e)}getStorageFieldType(){throw new Error("Method not implemented.")}static create(e,t){return new i(e,t)}},A=f.create,T=class i extends s{entity;type;required;relationalColumn;foreignColumn;sourceEntity;constructor(e,t,n,r,a){super(),this.entity=e,this.type=t,this.required=a??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 i(e,"one",t,void 0,n??false)}static createManyFactory(){return (e,t,n)=>new i(e,"many",void 0,t,n??false)}},w=(i,e)=>({$type:"relations",objectName:i.name,relations:e({one:T.createOneFactory(),many:T.createManyFactory()})}),x=i=>{if(i)return Array.isArray(i.value)?i.value.map(e=>x(e)):typeof i.value!="object"||i.value===null||i.value instanceof Date?i.value:Object.fromEntries(Object.entries(i.value).map(([e,t])=>[e,x(t)]))},V=i=>Object.fromEntries(Object.entries(i).flatMap(([e,t])=>{if(t.$type==="relations")return [];let n=t,r=Object.values(i).find(a=>a.$type==="relations"&&a.objectName===t.name);return r&&(n=n.setRelations(r.relations)),[[n.name,n]]}));exports.LiveAtomicType=l;exports.LiveBoolean=d;exports.LiveNumber=p;exports.LiveObject=f;exports.LiveString=o;exports.LiveTimestamp=m;exports.LiveType=s;exports.NullableLiveType=c;exports.Relation=T;exports.boolean=O;exports.createRelations=w;exports.createSchema=V;exports.id=I;exports.inferValue=x;exports.number=b;exports.object=A;exports.reference=M;exports.string=g;exports.timestamp=j;
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}}},l=class i extends s{storageType;convertFunc;isIndex;isUnique;defaultValue;foreignReference;isPrimary;constructor(e,t,n,r,a,u,y){super(),this.storageType=e,this.convertFunc=t,this.isIndex=n??false,this.isUnique=r??false,this.defaultValue=a,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 i(this.storageType,this.convertFunc,this.isIndex,true,this.defaultValue,this.foreignReference,this.isPrimary)}index(){return new i(this.storageType,this.convertFunc,true,this.isUnique,this.defaultValue,this.foreignReference,this.isPrimary)}default(e){return new i(this.storageType,this.convertFunc,this.isIndex,this.isUnique,e,this.foreignReference,this.isPrimary)}primary(){return new i(this.storageType,this.convertFunc,this.isIndex,this.isUnique,this.defaultValue,this.foreignReference,true)}nullable(){return new c(this)}},d=class i extends l{constructor(){super("integer",e=>Number(e));}static create(){return new i}},b=d.create,o=class i extends l{constructor(e){super("varchar",void 0,void 0,void 0,void 0,e);}static create(){return new i}static createId(){return new i().index().unique().primary()}static createReference(e){return new i(e)}},g=o.create,I=o.createId,M=o.createReference,p=class i extends l{constructor(){super("boolean",e=>typeof e=="string"?e.toLowerCase()==="true":!!e);}static create(){return new i}},O=p.create,m=class i extends l{constructor(){super("timestamp",e=>typeof e=="string"?new Date(e):e);}static create(){return new i}},j=m.create;var f=class i 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,a])=>[r,(this.fields[r]??this.relations[r]).encodeMutation("set",a,n)]))}mergeMutation(e,t,n){let r={};return [{value:{...(n==null?void 0:n.value)??{},...Object.fromEntries(Object.entries(t).map(([a,u])=>{let y=this.fields[a]??this.relations[a];if(!y)return [a,u];let[h,v]=y.mergeMutation(e,u,n==null?void 0:n.value[a]);return v&&(r[a]=v),[a,h]}))}},r]}setRelations(e){return new i(this.name,this.fields,e)}getStorageFieldType(){throw new Error("Method not implemented.")}static create(e,t){return new i(e,t)}},A=f.create,T=class i extends s{entity;type;required;relationalColumn;foreignColumn;sourceEntity;constructor(e,t,n,r,a){super(),this.entity=e,this.type=t,this.required=a??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 i(e,"one",t,void 0,n??false)}static createManyFactory(){return (e,t,n)=>new i(e,"many",void 0,t,n??false)}},w=(i,e)=>({$type:"relations",objectName:i.name,relations:e({one:T.createOneFactory(),many:T.createManyFactory()})}),x=i=>{if(i)return Array.isArray(i.value)?i.value.map(e=>x(e)):typeof i.value!="object"||i.value===null||i.value instanceof Date?i.value:Object.fromEntries(Object.entries(i.value).map(([e,t])=>[e,x(t)]))},V=i=>Object.fromEntries(Object.entries(i).flatMap(([e,t])=>{if(t.$type==="relations")return [];let n=t,r=Object.values(i).find(a=>a.$type==="relations"&&a.objectName===t.name);return r&&(n=n.setRelations(r.relations)),[[n.name,n]]}));exports.LiveAtomicType=l;exports.LiveBoolean=p;exports.LiveNumber=d;exports.LiveObject=f;exports.LiveString=o;exports.LiveTimestamp=m;exports.LiveType=s;exports.NullableLiveType=c;exports.Relation=T;exports.boolean=O;exports.createRelations=w;exports.createSchema=V;exports.id=I;exports.inferValue=x;exports.number=b;exports.object=A;exports.reference=M;exports.string=g;exports.timestamp=j;
package/dist/index.d.cts CHANGED
@@ -217,7 +217,7 @@ type WhereClause<T extends LiveObjectAny> = ({
217
217
  $in?: InferLiveType<T["fields"][K]>[];
218
218
  $eq?: InferLiveType<T["fields"][K]>;
219
219
  };
220
- } & (InferLiveType<T["fields"][K]> extends number ? {
220
+ } & (Exclude<InferLiveType<T["fields"][K]>, null | undefined> extends number | Date ? {
221
221
  $gt?: InferLiveType<T["fields"][K]>;
222
222
  $gte?: InferLiveType<T["fields"][K]>;
223
223
  $lt?: InferLiveType<T["fields"][K]>;
@@ -230,7 +230,7 @@ type WhereClause<T extends LiveObjectAny> = ({
230
230
  $or?: WhereClause<T>[];
231
231
  };
232
232
  type IncludeClause<T extends LiveObjectAny> = {
233
- [K in keyof T["relations"]]?: boolean;
233
+ [K in keyof T["relations"]]?: boolean | IncludeClause<T["relations"][K]["entity"]>;
234
234
  };
235
235
  type GetFieldType<T> = T extends NullableLiveType<any> ? T["inner"] : T;
236
236
  type HasDefaultValue<T> = T extends LiveAtomicType<any, undefined> ? false : true;
package/dist/index.d.ts CHANGED
@@ -217,7 +217,7 @@ type WhereClause<T extends LiveObjectAny> = ({
217
217
  $in?: InferLiveType<T["fields"][K]>[];
218
218
  $eq?: InferLiveType<T["fields"][K]>;
219
219
  };
220
- } & (InferLiveType<T["fields"][K]> extends number ? {
220
+ } & (Exclude<InferLiveType<T["fields"][K]>, null | undefined> extends number | Date ? {
221
221
  $gt?: InferLiveType<T["fields"][K]>;
222
222
  $gte?: InferLiveType<T["fields"][K]>;
223
223
  $lt?: InferLiveType<T["fields"][K]>;
@@ -230,7 +230,7 @@ type WhereClause<T extends LiveObjectAny> = ({
230
230
  $or?: WhereClause<T>[];
231
231
  };
232
232
  type IncludeClause<T extends LiveObjectAny> = {
233
- [K in keyof T["relations"]]?: boolean;
233
+ [K in keyof T["relations"]]?: boolean | IncludeClause<T["relations"][K]["entity"]>;
234
234
  };
235
235
  type GetFieldType<T> = T extends NullableLiveType<any> ? T["inner"] : T;
236
236
  type HasDefaultValue<T> = T extends LiveAtomicType<any, undefined> ? false : true;
package/dist/index.js CHANGED
@@ -1 +1 @@
1
- export{e as LiveAtomicType,l as LiveBoolean,f as LiveNumber,p as LiveObject,h as LiveString,n as LiveTimestamp,c as LiveType,d as NullableLiveType,r as Relation,m as boolean,s as createRelations,u as createSchema,j as id,t as inferValue,g as number,q as object,k as reference,i as string,o as timestamp}from'./chunk-7C33XQ3U.js';
1
+ export{e as LiveAtomicType,l as LiveBoolean,f as LiveNumber,p as LiveObject,h as LiveString,n as LiveTimestamp,c as LiveType,d as NullableLiveType,r as Relation,m as boolean,s as createRelations,u as createSchema,j as id,t as inferValue,g as number,q as object,k as reference,i as string,o as timestamp}from'./chunk-XNKBHWSC.js';
package/dist/server.cjs CHANGED
@@ -1,2 +1,2 @@
1
- 'use strict';var ze=require('qs'),zod=require('zod'),P=require('crypto');require('js-xxhash');var kysely=require('kysely'),postgres=require('kysely/helpers/postgres');function _interopDefault(e){return e&&e.__esModule?e:{default:e}}var ze__default=/*#__PURE__*/_interopDefault(ze);var P__default=/*#__PURE__*/_interopDefault(P);var ge=Object.create;var Y=Object.defineProperty;var Re=Object.getOwnPropertyDescriptor;var be=Object.getOwnPropertyNames;var Se=Object.getPrototypeOf,we=Object.prototype.hasOwnProperty;var ve=(r,e)=>()=>(e||r((e={exports:{}}).exports,e),e.exports);var Ie=(r,e,t,n)=>{if(e&&typeof e=="object"||typeof e=="function")for(let i of be(e))!we.call(r,i)&&i!==t&&Y(r,i,{get:()=>e[i],enumerable:!(n=Re(e,i))||n.enumerable});return r};var X=(r,e,t)=>(t=r!=null?ge(Se(r)):{},Ie(Y(t,"default",{value:r,enumerable:true}),r));var V=ve($=>{Object.defineProperty($,"__esModule",{value:true});$.parse=$e;$.serialize=Ce;var Me=/^[\u0021-\u003A\u003C\u003E-\u007E]+$/,Le=/^[\u0021-\u003A\u003C-\u007E]*$/,Ae=/^([.]?[a-z0-9]([a-z0-9-]{0,61}[a-z0-9])?)([.][a-z0-9]([a-z0-9-]{0,61}[a-z0-9])?)*$/i,Ee=/^[\u0020-\u003A\u003D-\u007E]*$/,Oe=Object.prototype.toString,je=(()=>{let r=function(){};return r.prototype=Object.create(null),r})();function $e(r,e){let t=new je,n=r.length;if(n<2)return t;let i=(e==null?void 0:e.decode)||Pe,a=0;do{let s=r.indexOf("=",a);if(s===-1)break;let c=r.indexOf(";",a),o=c===-1?n:c;if(s>o){a=r.lastIndexOf(";",s-1)+1;continue}let y=ee(r,a,s),f=te(r,s,y),u=r.slice(y,f);if(t[u]===void 0){let l=ee(r,s+1,o),p=te(r,o,l),d=i(r.slice(l,p));t[u]=d;}a=o+1;}while(a<n);return t}function ee(r,e,t){do{let n=r.charCodeAt(e);if(n!==32&&n!==9)return e}while(++e<t);return t}function te(r,e,t){for(;e>t;){let n=r.charCodeAt(--e);if(n!==32&&n!==9)return e+1}return t}function Ce(r,e,t){let n=(t==null?void 0:t.encode)||encodeURIComponent;if(!Me.test(r))throw new TypeError(`argument name is invalid: ${r}`);let i=n(e);if(!Le.test(i))throw new TypeError(`argument val is invalid: ${e}`);let a=r+"="+i;if(!t)return a;if(t.maxAge!==void 0){if(!Number.isInteger(t.maxAge))throw new TypeError(`option maxAge is invalid: ${t.maxAge}`);a+="; Max-Age="+t.maxAge;}if(t.domain){if(!Ae.test(t.domain))throw new TypeError(`option domain is invalid: ${t.domain}`);a+="; Domain="+t.domain;}if(t.path){if(!Ee.test(t.path))throw new TypeError(`option path is invalid: ${t.path}`);a+="; Path="+t.path;}if(t.expires){if(!_e(t.expires)||!Number.isFinite(t.expires.valueOf()))throw new TypeError(`option expires is invalid: ${t.expires}`);a+="; Expires="+t.expires.toUTCString();}if(t.httpOnly&&(a+="; HttpOnly"),t.secure&&(a+="; Secure"),t.partitioned&&(a+="; Partitioned"),t.priority)switch(typeof t.priority=="string"?t.priority.toLowerCase():void 0){case "low":a+="; Priority=Low";break;case "medium":a+="; Priority=Medium";break;case "high":a+="; 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":a+="; SameSite=Strict";break;case "lax":a+="; SameSite=Lax";break;case "none":a+="; SameSite=None";break;default:throw new TypeError(`option sameSite is invalid: ${t.sameSite}`)}return a}function Pe(r){if(r.indexOf("%")===-1)return r;try{return decodeURIComponent(r)}catch{return r}}function _e(r){return Oe.call(r)==="[object Date]"}});var ie=X(V());var C=zod.z.object({resource:zod.z.string(),where:zod.z.record(zod.z.string(),zod.z.any()).optional(),include:zod.z.record(zod.z.string(),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()}),z=zod.z.record(zod.z.string(),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((r,e)=>{r.id&&e.addIssue({code:zod.z.ZodIssueCode.custom,message:"Payload cannot have an id"});}),ne=zod.z.object({id:zod.z.string().optional(),type:zod.z.literal("MUTATE"),resource:zod.z.string(),resourceId:zod.z.string().optional()}),L=ne.extend({procedure:zod.z.string(),payload:zod.z.any().optional()}),A=ne.extend({procedure:zod.z.enum(["INSERT","UPDATE"]),payload:z});zod.z.union([A,L]);var re=C.omit({resource:true}),N=L.omit({id:true,type:true,resource:true,procedure:true}),D=A.omit({id:true,type:true,resource:true,procedure:true});zod.z.union([D,N]);var ae=r=>async e=>{var t;try{let n=typeof e.headers.getSetCookie=="function"?Object.fromEntries(e.headers):e.headers,i={headers:n,cookies:n.cookie?ie.default.parse(n.cookie):{}},a=new URL(e.url),s=a.pathname.split("/"),c=a.searchParams,o=ze__default.default.parse(c.toString()),y=await((t=r.contextProvider)==null?void 0:t.call(r,{transport:"HTTP",headers:i.headers,cookies:i.cookies,query:o}))??{};if(e.method==="GET"){let f=s[s.length-1],{success:u,data:l,error:p}=re.safeParse(o);if(!u)return Response.json({message:"Invalid query",code:"INVALID_QUERY",details:p},{status:400});let d=await r.handleRequest({req:{...i,type:"QUERY",resourceName:f,context:y,where:l.where,include:l.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 f=s[s.length-1],u=s[s.length-2],l=e.body?await e.json():{},p;if(f==="insert"||f==="update"){let{success:h,data:x,error:R}=D.safeParse(l);if(!h)return Response.json({message:"Invalid mutation",code:"INVALID_REQUEST",details:R},{status:400});p=x;}else {let{success:h,data:x,error:R}=N.safeParse(l);if(!h)return Response.json({message:"Invalid mutation",code:"INVALID_REQUEST",details:R},{status:400});p=x;}let d=await r.handleRequest({req:{...i,type:"MUTATE",resourceName:u,input:p.payload,context:y,resourceId:p.resourceId,procedure:f==="insert"||f==="update"?f.toUpperCase():f,query:{}}});return Response.json(d)}catch(f){return console.error("Error parsing mutation from the client:",f),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 pe=X(V());var I=zod.z.string(),Ne=zod.z.object({id:I,type:zod.z.literal("SUBSCRIBE"),resource:zod.z.string()}),De=C.extend({id:I,type:zod.z.literal("QUERY")}),oe=A.extend({id:I}),Ke=L.extend({id:I}),Fe=zod.z.union([Ke,oe]),se=zod.z.union([Ne,De,Fe]),Ue=zod.z.object({id:I,type:zod.z.literal("REJECT"),resource:zod.z.string(),message:zod.z.string().optional()}),qe=zod.z.object({id:I,type:zod.z.literal("REPLY"),data:zod.z.any()});zod.z.union([Ue,qe,oe]);zod.z.object({resource:zod.z.string(),data:zod.z.record(zod.z.string(),z)});var ue="0123456789ABCDEFGHJKMNPQRSTVWXYZ",E=32;var ke=16,le=10,ce=0xffffffffffff;var S;(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";})(S||(S={}));var w=class extends Error{constructor(e,t){super(`${t} (${e})`),this.name="ULIDError",this.code=e;}};function We(r){let e=Math.floor(r()*E);return e===E&&(e=E-1),ue.charAt(e)}function He(r){var n;let e=Ze(),t=e&&(e.crypto||e.msCrypto)||(typeof P__default.default<"u"?P__default.default:null);if(typeof(t==null?void 0:t.getRandomValues)=="function")return ()=>{let i=new Uint8Array(1);return t.getRandomValues(i),i[0]/255};if(typeof(t==null?void 0:t.randomBytes)=="function")return ()=>t.randomBytes(1).readUInt8()/255;if((n=P__default.default)!=null&&n.randomBytes)return ()=>P__default.default.randomBytes(1).readUInt8()/255;throw new w(S.PRNGDetectFailure,"Failed to find a reliable PRNG")}function Ze(){return Qe()?self:typeof window<"u"?window:typeof global<"u"?global:typeof globalThis<"u"?globalThis:null}function Be(r,e){let t="";for(;r>0;r--)t=We(e)+t;return t}function Ge(r,e=le){if(isNaN(r))throw new w(S.EncodeTimeValueMalformed,`Time must be a number: ${r}`);if(r>ce)throw new w(S.EncodeTimeSizeExceeded,`Cannot encode a time larger than ${ce}: ${r}`);if(r<0)throw new w(S.EncodeTimeNegative,`Time must be positive: ${r}`);if(Number.isInteger(r)===false)throw new w(S.EncodeTimeValueMalformed,`Time must be an integer: ${r}`);let t,n="";for(let i=e;i>0;i--)t=r%E,n=ue.charAt(t)+n,r=(r-t)/E;return n}function Qe(){return typeof WorkerGlobalScope<"u"&&self instanceof WorkerGlobalScope}function de(r,e){let t=He(),n=Date.now();return Ge(n,le)+Be(ke,t)}var K=()=>de().toLowerCase();var ye=r=>{let e={},t={};return r.subscribeToMutations(n=>{let i=n;!i.resourceId||!i.payload||(console.log("Mutation propagated:",i),Object.entries(t[i.resource]??{}).forEach(([a,s])=>{var c;(c=e[a])==null||c.send(JSON.stringify({...i,id:i.id??K()}));}));}),(n,i)=>{var f;let a=u=>{n.send(JSON.stringify(u));},s=K(),c={headers:i.headers,cookies:typeof i.headers.cookie=="string"?pe.default.parse(i.headers.cookie):{}},o=ze.parse(i.url.split("?")[1]),y=(f=r.contextProvider)==null?void 0:f.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 l=se.parse(JSON.parse(u.toString()));if(l.type==="SUBSCRIBE"){let{resource:p}=l;t[p]||(t[p]={}),t[p][s]={};}else if(l.type==="QUERY"){let{resource:p}=l,d=await r.handleRequest({req:{...c,type:"QUERY",resourceName:p,context:await y??{},query:o}});if(!d||!d.data)throw new Error("Invalid resource");a({id:l.id,type:"REPLY",data:{resource:p,data:Object.fromEntries(Object.entries(d.data??{}).map(([h,x])=>[h,x.value]))}});}else if(l.type==="MUTATE"){let{resource:p}=l;console.log("Received mutation from client:",l);try{let d=await r.handleRequest({req:{...c,type:"MUTATE",resourceName:p,input:l.payload,context:{messageId:l.id,...await y??{}},resourceId:l.resourceId,procedure:l.procedure,query:o}});l.procedure&&l.procedure!=="INSERT"&&l.procedure!=="UPDATE"&&a({id:l.id,type:"REPLY",data:d});}catch(d){a({id:l.id,type:"REJECT",resource:p,message:d.message}),console.error("Error parsing mutation from the client:",d);}}}catch(l){console.error("Error handling message from the client:",l);}}),n.on("close",()=>{console.log("Connection closed",s),delete e[s];for(let u of Object.values(t))delete u[s];});}};function me(r){let e=`${r.protocol}://${r.hostname}${r.url}`,t=new Headers;return Object.entries(r.headers).forEach(([n,i])=>{i&&t.set(n,Array.isArray(i)?i.join(","):i);}),new Request(e,{method:r.method,headers:t,body:r.body&&r.method!=="GET"?JSON.stringify(r.body):void 0})}var Ct=(r,e,t)=>{r.ws(`${(t==null?void 0:t.basePath)??""}/ws`,ye(e)),r.use(`${(t==null?void 0:t.basePath)??""}/`,(n,i)=>{ae(e)(me(n)).then(s=>s.json().then(c=>i.status(s.status).send(c)));});};var g=r=>{if(r)return Array.isArray(r.value)?r.value.map(e=>g(e)):typeof r.value!="object"||r.value===null||r.value instanceof Date?r.value:Object.fromEntries(Object.entries(r.value).map(([e,t])=>[e,g(t)]))};var b=(r,e,t=false)=>Object.entries(e).every(([n,i])=>{if(n==="$and")return i.every(s=>b(r,s,t));if(n==="$or")return i.some(s=>b(r,s,t));let a=(i==null?void 0:i.$eq)!==void 0?i==null?void 0:i.$eq:i;if(typeof i=="object"&&i!==null&&(i==null?void 0:i.$eq)===void 0){if(i.$in!==void 0){let s=r[n];return s===void 0?false:t?!i.$in.includes(s):i.$in.includes(s)}if(i.$not!==void 0&&!t)return b(r,{[n]:i.$not},true);if(i.$gt!==void 0){let s=r[n];return typeof s!="number"?false:t?s<=i.$gt:s>i.$gt}if(i.$gte!==void 0){let s=r[n];return typeof s!="number"?false:t?s<i.$gte:s>=i.$gte}if(i.$lt!==void 0){let s=r[n];return typeof s!="number"?false:t?s>=i.$lt:s<i.$lt}if(i.$lte!==void 0){let s=r[n];return typeof s!="number"?false:t?s>i.$lte:s<=i.$lte}return !r[n]||typeof r[n]!="object"?false:b(r[n],i,t)}return t?r[n]!==a:r[n]===a});var H=class r{routes;constructor(e){this.routes=e.routes;}static create(e){return new r(e)}},tn=r=>H.create({...r}),Xe=r=>({handler:e=>({inputValidator:r??zod.z.undefined(),handler:e})}),Z=class r{_resourceSchema;resourceName;middlewares;customMutations;authorization;constructor(e,t,n){this.resourceName=e,this.middlewares=new Set,this.customMutations=t??{},this.authorization=n;}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)throw new Error("Procedure is required for mutations");let i=this.customMutations[n.procedure];if(i){let a=i.inputValidator.parse(n.input);return n.input=a,i.handler({req:n,db:e.db,schema:e.schema})}else if(n.procedure==="INSERT"||n.procedure==="UPDATE")return this.handleSet({req:n,db:e.db,schema:e.schema,operation:n.procedure})}throw new Error("Invalid request")})();return await Array.from(this.middlewares.values()).reduceRight((n,i)=>a=>i({req:a,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:Xe}))}handleFind=async({req:e,db:t})=>{var i,a;let n=(a=(i=this.authorization)==null?void 0:i.read)==null?void 0:a.call(i,{ctx:e.context});if(typeof n=="boolean"&&!n)throw new Error("Not authorized");return {data:await t.rawFind(e.resourceName,e.where&&n&&n!==true?{$and:[e.where,n]}:n&&n!==true?n:e.where,e.include),acceptedValues:null}};handleSet=async({req:e,db:t,schema:n,operation:i})=>{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);if(i==="INSERT"&&a)throw new Error("Resource already exists");if(i==="UPDATE"&&!a)throw new Error("Resource not found");return t.transaction(async({trx:s})=>{var f,u,l,p,d;let[c,o]=n[this.resourceName].mergeMutation("set",e.input,a);if(!o)throw new Error("Mutation rejected");if(i==="INSERT"){let h=await s.rawInsert(e.resourceName,e.resourceId,c);if((f=this.authorization)!=null&&f.insert){let x=g(h);x.id=x.id??e.resourceId;let R=this.authorization.insert({ctx:e.context,value:x});if(!(typeof R=="boolean"?R:b(x,R)))throw new Error("Not authorized")}return {data:h,acceptedValues:o}}if((l=(u=this.authorization)==null?void 0:u.update)!=null&&l.preMutation){let h=g(a);h.id=h.id??e.resourceId;let x=this.authorization.update.preMutation({ctx:e.context,value:h});if(!(typeof x=="boolean"?x:b(h,x)))throw new Error("Not authorized")}let y=await s.rawUpdate(e.resourceName,e.resourceId,c);if((d=(p=this.authorization)==null?void 0:p.update)!=null&&d.postMutation){let h=g(y);h.id=h.id??e.resourceId;let x=this.authorization.update.postMutation({ctx:e.context,value:h});if(!(typeof x=="boolean"?x:b(h,x)))throw new Error("Not authorized")}return {data:y,acceptedValues:o}})}},B=class r{middlewares;constructor(e=[]){this.middlewares=e;}collectionRoute(e,t){return new Z(e.name,void 0,t).use(...this.middlewares)}use(...e){return new r([...this.middlewares,...e])}static create(){return new r}},nn=B.create;var j=class{async insert(e,t){let n=new Date().toISOString();return g(await this.rawInsert(e.name,t.id,{value:Object.fromEntries(Object.entries(t).map(([i,a])=>[i,{value:a,_meta:{timestamp:n}}]))}))}async update(e,t,n){let i=new Date().toISOString(),{id:a,...s}=n;return g(await this.rawUpdate(e.name,t,{value:Object.fromEntries(Object.entries(s).map(([c,o])=>[c,{value:o,_meta:{timestamp:i}}]))}))}};function _(r,e,t,n){if(!r)throw new Error("Schema not initialized");let i=r[e];if(!i)throw new Error("Resource not found");let a=n.$or,s=n.$and;return (a?t.or:t.and)(a?n.$or.map(c=>_(r,e,t,c)):s?n.$and.map(c=>_(r,e,t,c)):Object.entries(n).map(([c,o])=>{var y,f;if(i.fields[c])return (o==null?void 0:o.$eq)!==void 0?t(`${e}.${c}`,o.$eq===null?"is":"=",o.$eq):(o==null?void 0:o.$in)!==void 0?t(`${e}.${c}`,"in",o.$in):(o==null?void 0:o.$not)!==void 0?((y=o==null?void 0:o.$not)==null?void 0:y.$in)!==void 0?t(`${e}.${c}`,"not in",o.$not.$in):((f=o==null?void 0:o.$not)==null?void 0:f.$eq)!==void 0?t(`${e}.${c}`,o.$not.$eq===null?"is not":"!=",o.$not.$eq):t(`${e}.${c}`,o.$not===null?"is not":"!=",o.$not):(o==null?void 0:o.$gt)!==void 0?t(`${e}.${c}`,">",o.$gt):(o==null?void 0:o.$gte)!==void 0?t(`${e}.${c}`,">=",o.$gte):(o==null?void 0:o.$lt)!==void 0?t(`${e}.${c}`,"<",o.$lt):(o==null?void 0:o.$lte)!==void 0?t(`${e}.${c}`,"<=",o.$lte):t(`${e}.${c}`,o===null?"is":"=",o);if(i.relations[c]){let u=i.relations[c],l=u.entity.name;u.type==="one"?"id":u.foreignColumn;u.type==="one"?u.relationalColumn:"id";return _(r,l,t,o)}return null}).filter(Boolean))}function tt(r,e,t,n){let i=r[e];if(!i)throw new Error("Resource not found");if(!n)return t;for(let a of Object.keys(n)){if(!i.relations[a])continue;let s=i.relations[a],c=s.entity.name,o=s.type==="one"?"id":s.foreignColumn,y=s.type==="one"?s.relationalColumn:"id";t=t.leftJoin(c,`${c}.${o}`,`${e}.${y}`);}return t}function Te(r,e,t,n){return !n||Object.keys(n).length===0?t:(t=tt(r,e,t,n),t.where(i=>_(r,e,i,n)))}function G(r,e,t,n){if(!n)return t;if(!r)throw new Error("Schema not initialized");let i=r[e];if(!i)throw new Error(`Resource not found: ${e}`);for(let a of Object.keys(n)){if(!i.relations[a])throw new Error(`Relation ${a} not found in resource ${e}`);let s=i.relations[a],c=s.entity.name,o=s.type==="one"?"id":s.foreignColumn,y=s.type==="one"?s.relationalColumn:"id",f=s.type==="one"?postgres.jsonObjectFrom:postgres.jsonArrayFrom;t=t.select(u=>f(u.selectFrom(c).selectAll(c).whereRef(`${c}.${o}`,"=",`${e}.${y}`).select(l=>postgres.jsonObjectFrom(l.selectFrom(`${c}_meta`).selectAll(`${c}_meta`).whereRef(`${c}_meta.id`,"=",`${c}.id`)).as("_meta"))).as(a));}return t}var Q=class r extends j{db;schema;constructor(e,t){super(),this.isKyselyLike(e)?this.db=e:this.db=new kysely.Kysely({dialect:new kysely.PostgresDialect({pool:e})}),this.schema=t,this.rawInsert=this.rawInsert.bind(this),this.rawUpdate=this.rawUpdate.bind(this);}async updateSchema(e){this.schema=e;let t=await this.db.introspection.getTables();for(let[n,i]of Object.entries(e)){let a=t.find(o=>o.name===n);a||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,y]of Object.entries(i.fields)){let f=a==null?void 0:a.columns.find(p=>p.name===o),u=y.getStorageFieldType();f?f.dataType!==u.type&&console.error("Column type mismatch:",o,"expected to have type:",u.type,"but has type:",f.dataType):(await this.db.schema.alterTable(n).addColumn(o,u.type,p=>{let d=p;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(p=>{throw console.error("Error adding column",o,p),p}),u.index&&await this.db.schema.createIndex(`${n}_${o}_index`).on(n).column(o).execute().catch(p=>{})),(c==null?void 0:c.columns.find(p=>p.name===o))||await this.db.schema.alterTable(s).addColumn(o,"varchar",p=>{let d=p;return u.primary&&(d=d.primaryKey().references(`${n}.${o}`)),d}).execute();}}}async rawFindById(e,t,n){if(!this.schema)throw new Error("Schema not initialized");let i=await this.db.selectFrom(e).where("id","=",t).selectAll(e).select(s=>postgres.jsonObjectFrom(s.selectFrom(`${e}_meta`).selectAll(`${e}_meta`).whereRef(`${e}_meta.id`,"=",`${e}.id`)).as("_meta"));i=G(this.schema,e,i,n);let a=await i.executeTakeFirst();if(a)return this.convertToMaterializedLiveType(a)}async findOne(e,t,n){let i=await this.rawFindById(e.name,t,n==null?void 0:n.include);if(i)return g(i)}async rawFind(e,t,n){if(!this.schema)throw new Error("Schema not initialized");let i=this.db.selectFrom(e).selectAll(e).select(o=>postgres.jsonObjectFrom(o.selectFrom(`${e}_meta`).selectAll(`${e}_meta`).whereRef(`${e}_meta.id`,"=",`${e}.id`)).as("_meta"));i=Te(this.schema,e,i,t),i=G(this.schema,e,i,n);let a=await i.execute(),s=Object.fromEntries(a.map(o=>{let{id:y,...f}=o;return [y,f]}));return Object.keys(s).length===0?{}:Object.entries(s).reduce((o,[y,f])=>(o[y]=this.convertToMaterializedLiveType(f),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(([i,a])=>[i,g(a)]))}async rawInsert(e,t,n){var s;let i={},a={};for(let[c,o]of Object.entries(n.value)){let y=(s=o._meta)==null?void 0:s.timestamp;y&&(i[c]=o.value,a[c]=y);}return await this.db.insertInto(e).values({...i,id:t}).execute().then(()=>{this.db.insertInto(`${e}_meta`).values({...a,id:t}).execute();}),n}async rawUpdate(e,t,n){var s;let i={},a={};for(let[c,o]of Object.entries(n.value)){let y=(s=o._meta)==null?void 0:s.timestamp;y&&(i[c]=o.value,a[c]=y);}return await Promise.all([this.db.updateTable(e).set(i).where("id","=",t).execute(),this.db.insertInto(`${e}_meta`).values({...a,id:t}).onConflict(c=>c.column("id").doUpdateSet(a)).execute()]),n}async transaction(e){if(!this.schema)throw new Error("Schema not initialized");if(this.db.isTransaction){let n=Math.random().toString(36).substring(2,15),i=await this.db.savepoint(n).execute();try{return await e({trx:this,commit:()=>i.releaseSavepoint(n).execute().then(()=>{}),rollback:()=>i.rollbackToSavepoint(n).execute().then(()=>{})}).then(a=>i.isCommitted||i.isRolledBack?a:i.releaseSavepoint(n).execute().then(()=>a))}catch(a){throw await i.rollbackToSavepoint(n).execute().catch(()=>{}),a}}let t=await this.db.startTransaction().execute();try{return await e({trx:new r(t,this.schema),commit:()=>t.commit().execute(),rollback:()=>t.rollback().execute()}).then(n=>t.isCommitted||t.isRolledBack?n:t.commit().execute().then(()=>n))}catch(n){throw await t.rollback().execute(),n}}convertToMaterializedLiveType(e){return {value:Object.entries(e).reduce((t,[n,i])=>{var a,s,c;return n==="_meta"||(n==="id"?t[n]={value:i}:Array.isArray(i)?t[n]={value:i.map(o=>this.convertToMaterializedLiveType(o)),_meta:{timestamp:(a=e==null?void 0:e._meta)==null?void 0:a[n]}}:typeof i=="object"&&i!==null&&!(i instanceof Date)?t[n]={...this.convertToMaterializedLiveType(i),_meta:{timestamp:(s=e==null?void 0:e._meta)==null?void 0:s[n]}}:t[n]={value:i,_meta:{timestamp:(c=e==null?void 0:e._meta)==null?void 0:c[n]}}),t},{})}}isKyselyLike(e){if(e instanceof kysely.Kysely)return true;if(!e||typeof e!="object")return false;let t=e,n=typeof t.selectFrom=="function",i=typeof t.startTransaction=="function",a=typeof t.savepoint=="function",s=typeof t.isTransaction=="boolean"||typeof t.isTransaction=="function";return n&&i||a&&s}};var J=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,i)=>a=>i({req:a,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&&(e.req.procedure==="INSERT"||e.req.procedure==="UPDATE")&&this.mutationSubscriptions.forEach(n=>{n({id:e.req.context.messageId,type:"MUTATE",resource:e.req.resourceName,payload:t.acceptedValues??{},resourceId:e.req.resourceId,procedure:e.req.procedure});}),t}use(e){return this.middlewares.add(e),this}context(e){return this.contextProvider=e,this}},xn=J.create;
2
- exports.Route=Z;exports.RouteFactory=B;exports.Router=H;exports.SQLStorage=Q;exports.Server=J;exports.Storage=j;exports.expressAdapter=Ct;exports.routeFactory=nn;exports.router=tn;exports.server=xn;
1
+ 'use strict';var P=require('crypto'),Ye=require('qs'),zod=require('zod');require('js-xxhash');var kysely=require('kysely'),postgres=require('kysely/helpers/postgres');function _interopDefault(e){return e&&e.__esModule?e:{default:e}}var P__default=/*#__PURE__*/_interopDefault(P);var Ye__default=/*#__PURE__*/_interopDefault(Ye);var Ie=Object.create;var ie=Object.defineProperty;var Le=Object.getOwnPropertyDescriptor;var Oe=Object.getOwnPropertyNames;var Ae=Object.getPrototypeOf,je=Object.prototype.hasOwnProperty;var Ee=(r,e)=>()=>(e||r((e={exports:{}}).exports,e),e.exports);var $e=(r,e,t,n)=>{if(e&&typeof e=="object"||typeof e=="function")for(let i of Oe(e))!je.call(r,i)&&i!==t&&ie(r,i,{get:()=>e[i],enumerable:!(n=Le(e,i))||n.enumerable});return r};var ae=(r,e,t)=>(t=r!=null?Ie(Ae(r)):{},$e(ie(t,"default",{value:r,enumerable:true}),r));var U=Ee(V=>{Object.defineProperty(V,"__esModule",{value:true});V.parse=Be;V.serialize=Qe;var Fe=/^[\u0021-\u003A\u003C\u003E-\u007E]+$/,Ke=/^[\u0021-\u003A\u003C-\u007E]*$/,De=/^([.]?[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]*$/,We=Object.prototype.toString,ke=(()=>{let r=function(){};return r.prototype=Object.create(null),r})();function Be(r,e){let t=new ke,n=r.length;if(n<2)return t;let i=(e==null?void 0:e.decode)||Ge,a=0;do{let o=r.indexOf("=",a);if(o===-1)break;let c=r.indexOf(";",a),s=c===-1?n:c;if(o>s){a=r.lastIndexOf(";",o-1)+1;continue}let m=le(r,a,o),y=de(r,o,m),d=r.slice(m,y);if(t[d]===void 0){let l=le(r,o+1,s),p=de(r,s,l),u=i(r.slice(l,p));t[d]=u;}a=s+1;}while(a<n);return t}function le(r,e,t){do{let n=r.charCodeAt(e);if(n!==32&&n!==9)return e}while(++e<t);return t}function de(r,e,t){for(;e>t;){let n=r.charCodeAt(--e);if(n!==32&&n!==9)return e+1}return t}function Qe(r,e,t){let n=(t==null?void 0:t.encode)||encodeURIComponent;if(!Fe.test(r))throw new TypeError(`argument name is invalid: ${r}`);let i=n(e);if(!Ke.test(i))throw new TypeError(`argument val is invalid: ${e}`);let a=r+"="+i;if(!t)return a;if(t.maxAge!==void 0){if(!Number.isInteger(t.maxAge))throw new TypeError(`option maxAge is invalid: ${t.maxAge}`);a+="; Max-Age="+t.maxAge;}if(t.domain){if(!De.test(t.domain))throw new TypeError(`option domain is invalid: ${t.domain}`);a+="; Domain="+t.domain;}if(t.path){if(!Ue.test(t.path))throw new TypeError(`option path is invalid: ${t.path}`);a+="; Path="+t.path;}if(t.expires){if(!Ze(t.expires)||!Number.isFinite(t.expires.valueOf()))throw new TypeError(`option expires is invalid: ${t.expires}`);a+="; Expires="+t.expires.toUTCString();}if(t.httpOnly&&(a+="; HttpOnly"),t.secure&&(a+="; Secure"),t.partitioned&&(a+="; Partitioned"),t.priority)switch(typeof t.priority=="string"?t.priority.toLowerCase():void 0){case "low":a+="; Priority=Low";break;case "medium":a+="; Priority=Medium";break;case "high":a+="; 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":a+="; SameSite=Strict";break;case "lax":a+="; SameSite=Lax";break;case "none":a+="; SameSite=None";break;default:throw new TypeError(`option sameSite is invalid: ${t.sameSite}`)}return a}function Ge(r){if(r.indexOf("%")===-1)return r;try{return decodeURIComponent(r)}catch{return r}}function Ze(r){return We.call(r)==="[object Date]"}});var se="0123456789ABCDEFGHJKMNPQRSTVWXYZ",O=32;var Ce=16,ce=10,oe=0xffffffffffff;var w;(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";})(w||(w={}));var v=class extends Error{constructor(e,t){super(`${t} (${e})`),this.name="ULIDError",this.code=e;}};function Pe(r){let e=Math.floor(r()*O);return e===O&&(e=O-1),se.charAt(e)}function Ne(r){var n;let e=ze(),t=e&&(e.crypto||e.msCrypto)||(typeof P__default.default<"u"?P__default.default:null);if(typeof(t==null?void 0:t.getRandomValues)=="function")return ()=>{let i=new Uint8Array(1);return t.getRandomValues(i),i[0]/255};if(typeof(t==null?void 0:t.randomBytes)=="function")return ()=>t.randomBytes(1).readUInt8()/255;if((n=P__default.default)!=null&&n.randomBytes)return ()=>P__default.default.randomBytes(1).readUInt8()/255;throw new v(w.PRNGDetectFailure,"Failed to find a reliable PRNG")}function ze(){return qe()?self:typeof window<"u"?window:typeof global<"u"?global:typeof globalThis<"u"?globalThis:null}function Ve(r,e){let t="";for(;r>0;r--)t=Pe(e)+t;return t}function _e(r,e=ce){if(isNaN(r))throw new v(w.EncodeTimeValueMalformed,`Time must be a number: ${r}`);if(r>oe)throw new v(w.EncodeTimeSizeExceeded,`Cannot encode a time larger than ${oe}: ${r}`);if(r<0)throw new v(w.EncodeTimeNegative,`Time must be positive: ${r}`);if(Number.isInteger(r)===false)throw new v(w.EncodeTimeValueMalformed,`Time must be an integer: ${r}`);let t,n="";for(let i=e;i>0;i--)t=r%O,n=se.charAt(t)+n,r=(r-t)/O;return n}function qe(){return typeof WorkerGlobalScope<"u"&&self instanceof WorkerGlobalScope}function ue(r,e){let t=Ne(),n=Date.now();return _e(n,ce)+Ve(Ce,t)}var D=()=>ue().toLowerCase();var N=(...r)=>{let e=r.filter(t=>!!t);return e.length===0?{}:e.length===1?e[0]:{$and:e}};var z=class{storage;queue=new Map;scheduled=false;constructor(e){this.storage=e;}async rawFind({resource:e,commonWhere:t,uniqueWhere:n,...i}){return new Promise((a,o)=>{let c=this.getBatchKey({resource:e,commonWhere:t,...i}),s={resource:e,commonWhere:t,uniqueWhere:n,...i,resolve:a,reject:o};this.queue.has(c)||this.queue.set(c,[]);let m=this.queue.get(c);m&&m.push(s),this.scheduled||(this.scheduled=true,setImmediate(()=>{this.processBatch();}));})}getBatchKey(e){let{resource:t,commonWhere:n,...i}=e;return `${t}:${JSON.stringify(n??{})}:${JSON.stringify(i??{})}`}async processBatch(){this.scheduled=false;let e=Array.from(this.queue.entries());this.queue.clear();for(let[,t]of e)try{await this.executeBatchedRequests(t);}catch(n){t.forEach(i=>{i.reject(n);});}}async executeBatchedRequests(e){var l,p;if(e.length===0)return;let t=e[0],{resource:n,commonWhere:i,include:a,sort:o}=t,c=e.length===1?t.limit:void 0,s=e.map(u=>u.uniqueWhere).filter(u=>u!==void 0),m=i,y=(l=Object.entries(s[0]??{})[0])==null?void 0:l[0];if(s.length>0){let u=s.map(f=>f[y]).filter(f=>f!=null);u.length>0&&(m=N(i,{[y]:{$in:u}}));}let d=await this.storage.rawFind({resource:n,where:m,include:a,sort:o,limit:c});for(let u of e){let f={};if(u.uniqueWhere){let[h,R]=Object.entries(u.uniqueWhere)[0];for(let[M,C]of Object.entries(d))((p=C.value[h])==null?void 0:p.value)===R&&(f[M]=C);}else Object.assign(f,d);u.resolve(f);}}};var me=ae(U());var _=zod.z.object({resource:zod.z.string(),where:zod.z.record(zod.z.string(),zod.z.any()).optional(),include:zod.z.record(zod.z.string(),zod.z.any()).optional(),lastSyncedAt:zod.z.string().optional(),limit:zod.z.coerce.number().optional(),sort:zod.z.array(zod.z.object({key:zod.z.string(),direction:zod.z.enum(["asc","desc"])})).optional()}),W=zod.z.record(zod.z.string(),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()})),He=W.superRefine((r,e)=>{r.id&&e.addIssue({code:zod.z.ZodIssueCode.custom,message:"Payload cannot have an id"});}),ye=zod.z.object({id:zod.z.string().optional(),type:zod.z.literal("MUTATE"),resource:zod.z.string(),resourceId:zod.z.string().optional()}),A=ye.extend({procedure:zod.z.string(),payload:zod.z.any().optional()}),j=ye.extend({procedure:zod.z.enum(["INSERT","UPDATE"]),payload:He});zod.z.union([j,A]);var pe=_.omit({resource:true}),k=A.omit({id:true,type:true,resource:true,procedure:true}),B=j.omit({id:true,type:true,resource:true,procedure:true});zod.z.union([B,k]);var fe=r=>async e=>{var t;try{let n=typeof e.headers.getSetCookie=="function"?Object.fromEntries(e.headers):e.headers,i={headers:n,cookies:n.cookie?me.default.parse(n.cookie):{}},a=new URL(e.url),o=a.pathname.split("/"),c=a.searchParams,s=Ye__default.default.parse(c.toString()),m=await((t=r.contextProvider)==null?void 0:t.call(r,{transport:"HTTP",headers:i.headers,cookies:i.cookies,queryParams:s}))??{};if(e.method==="GET"){let y=o[o.length-1],{success:d,data:l,error:p}=pe.safeParse(s);if(!d)return Response.json({message:"Invalid query",code:"INVALID_QUERY",details:p},{status:400});let u=await r.handleQuery({req:{...i,...l,type:"QUERY",resource:y,context:m,queryParams:s}});return !u||!u.data?Response.json({message:"Invalid resource",code:"INVALID_RESOURCE"},{status:400}):Response.json(u.data)}if(e.method==="POST")try{let y=o[o.length-1],d=o[o.length-2],l=e.body?await e.json():{},p;if(y==="insert"||y==="update"){let{success:f,data:h,error:R}=B.safeParse(l);if(!f)return Response.json({message:"Invalid mutation",code:"INVALID_REQUEST",details:R},{status:400});p=h;}else {let{success:f,data:h,error:R}=k.safeParse(l);if(!f)return Response.json({message:"Invalid mutation",code:"INVALID_REQUEST",details:R},{status:400});p=h;}let u=await r.handleMutation({req:{...i,type:"MUTATE",resource:d,input:p.payload,context:m,resourceId:p.resourceId,procedure:y==="insert"||y==="update"?y.toUpperCase():y,queryParams:{}}});return Response.json(u)}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 ge=ae(U());var I=zod.z.string(),Xe=zod.z.object({id:I,type:zod.z.literal("SUBSCRIBE"),resource:zod.z.string()}),et=_.extend({id:I,type:zod.z.literal("QUERY")}),Te=j.extend({id:I}),tt=A.extend({id:I}),nt=zod.z.union([tt,Te]),he=zod.z.union([Xe,et,nt]),rt=zod.z.object({id:I,type:zod.z.literal("REJECT"),resource:zod.z.string(),message:zod.z.string().optional()}),it=zod.z.object({id:I,type:zod.z.literal("REPLY"),data:zod.z.any()});zod.z.union([rt,it,Te]);zod.z.object({resource:zod.z.string(),data:zod.z.record(zod.z.string(),W)});var Re=r=>{let e={},t={};return r.subscribeToMutations(n=>{let i=n;!i.resourceId||!i.payload||(console.log("Mutation propagated:",i),Object.entries(t[i.resource]??{}).forEach(([a,o])=>{var c;(c=e[a])==null||c.send(JSON.stringify({...i,id:i.id??D()}));}));}),(n,i)=>{var y;let a=d=>{n.send(JSON.stringify(d));},o=D(),c={headers:i.headers,cookies:typeof i.headers.cookie=="string"?ge.default.parse(i.headers.cookie):{}},s=Ye.parse(i.url.split("?")[1]),m=(y=r.contextProvider)==null?void 0:y.call(r,{transport:"WEBSOCKET",headers:c.headers,cookies:c.cookies,queryParams:s});e[o]=n,console.log("Client connected:",o),n.on("message",async d=>{try{console.log("Message received from the client:",d);let l=he.parse(JSON.parse(d.toString()));if(l.type==="SUBSCRIBE"){let{resource:p}=l;t[p]||(t[p]={}),t[p][o]={};}else if(l.type==="QUERY"){let{resource:p}=l,u=await r.handleQuery({req:{...c,type:"QUERY",resource:p,context:await m??{},queryParams:s}});if(!u||!u.data)throw new Error("Invalid resource");a({id:l.id,type:"REPLY",data:{resource:p,data:Object.fromEntries(Object.entries(u.data??{}).map(([f,h])=>[f,h.value]))}});}else if(l.type==="MUTATE"){let{resource:p}=l;console.log("Received mutation from client:",l);try{let u=await r.handleMutation({req:{...c,type:"MUTATE",resource:p,input:l.payload,context:{messageId:l.id,...await m??{}},resourceId:l.resourceId,procedure:l.procedure,queryParams:s}});l.procedure&&l.procedure!=="INSERT"&&l.procedure!=="UPDATE"&&a({id:l.id,type:"REPLY",data:u});}catch(u){a({id:l.id,type:"REJECT",resource:p,message:u.message}),console.error("Error parsing mutation from the client:",u);}}}catch(l){console.error("Error handling message from the client:",l);}}),n.on("close",()=>{console.log("Connection closed",o),delete e[o];for(let d of Object.values(t))delete d[o];});}};function xe(r){let e=`${r.protocol}://${r.hostname}${r.url}`,t=new Headers;return Object.entries(r.headers).forEach(([n,i])=>{i&&t.set(n,Array.isArray(i)?i.join(","):i);}),new Request(e,{method:r.method,headers:t,body:r.body&&r.method!=="GET"?JSON.stringify(r.body):void 0})}var Kt=(r,e,t)=>{r.ws(`${(t==null?void 0:t.basePath)??""}/ws`,Re(e)),r.use(`${(t==null?void 0:t.basePath)??""}/`,(n,i)=>{fe(e)(xe(n)).then(o=>o.json().then(c=>i.status(o.status).send(c)));});};var x=r=>{if(r)return Array.isArray(r.value)?r.value.map(e=>x(e)):typeof r.value!="object"||r.value===null||r.value instanceof Date?r.value:Object.fromEntries(Object.entries(r.value).map(([e,t])=>[e,x(t)]))};var b=(r,e,t=false)=>Object.entries(e).every(([n,i])=>{if(n==="$and")return i.every(o=>b(r,o,t));if(n==="$or")return i.some(o=>b(r,o,t));let a=(i==null?void 0:i.$eq)!==void 0?i==null?void 0:i.$eq:i;if(typeof i=="object"&&i!==null&&(i==null?void 0:i.$eq)===void 0){if(i.$in!==void 0){let o=r[n];return o===void 0?false:t?!i.$in.includes(o):i.$in.includes(o)}if(i.$not!==void 0&&!t)return b(r,{[n]:i.$not},true);if(i.$gt!==void 0){let o=r[n];return typeof o!="number"?false:t?o<=i.$gt:o>i.$gt}if(i.$gte!==void 0){let o=r[n];return typeof o!="number"?false:t?o<i.$gte:o>=i.$gte}if(i.$lt!==void 0){let o=r[n];return typeof o!="number"?false:t?o>=i.$lt:o<i.$lt}if(i.$lte!==void 0){let o=r[n];return typeof o!="number"?false:t?o>i.$lte:o<=i.$lte}return !r[n]||typeof r[n]!="object"?false:b(r[n],i,t)}return t?r[n]!==a:r[n]===a});var Y=class r{routes;constructor(e){this.routes=e.routes;}static create(e){return new r(e)}},dn=r=>Y.create({...r}),st=r=>({handler:e=>({inputValidator:r??zod.z.undefined(),handler:e})}),X=class r{resourceSchema;middlewares;customMutations;authorization;constructor(e,t,n){this.resourceSchema=e,this.middlewares=new Set,this.customMutations=t??{},this.authorization=n;}use(...e){for(let t of e)this.middlewares.add(t);return this}withMutations(e){return new r(this.resourceSchema,e({mutation:st}))}handleQuery=async({req:e,batcher:t})=>await this.wrapInMiddlewares(async n=>{var a,o;let i=(o=(a=this.authorization)==null?void 0:a.read)==null?void 0:o.call(a,{ctx:n.context});if(typeof i=="boolean"&&!i)throw new Error("Not authorized");return {data:await t.rawFind({resource:n.resource,commonWhere:N(n.where,typeof i=="object"?i:void 0),uniqueWhere:n.relationalWhere,include:n.include,limit:n.limit,sort:n.sort})}})(e);handleMutation=async({req:e,db:t})=>await this.wrapInMiddlewares(async n=>{if(!n.procedure)throw new Error("Procedure is required for mutations");let i=this.customMutations[n.procedure];if(i){let a=i.inputValidator.parse(n.input);return n.input=a,i.handler({req:n,db:t})}else {if(n.procedure==="INSERT"||n.procedure==="UPDATE")return this.handleSet({req:n,db:t,operation:n.procedure});throw new Error(`Unknown procedure: ${n.procedure}`)}})(e);handleSet=async({req:e,db:t,operation:n})=>{if(!e.input)throw new Error("Payload is required");if(!e.resourceId)throw new Error("ResourceId is required");let i=await t.rawFindById(e.resource,e.resourceId);if(n==="INSERT"&&i)throw new Error("Resource already exists");if(n==="UPDATE"&&!i)throw new Error("Resource not found");return t.transaction(async({trx:a})=>{var m,y,d,l,p;let[o,c]=this.resourceSchema.mergeMutation("set",e.input,i);if(!c)throw new Error("Mutation rejected");if(n==="INSERT"){let u=await a.rawInsert(e.resource,e.resourceId,o);if((m=this.authorization)!=null&&m.insert){let f=x(u);f.id=f.id??e.resourceId;let h=this.authorization.insert({ctx:e.context,value:f});if(!(typeof h=="boolean"?h:b(f,h)))throw new Error("Not authorized")}return {data:u,acceptedValues:c}}if((d=(y=this.authorization)==null?void 0:y.update)!=null&&d.preMutation){let u=x(i);u.id=u.id??e.resourceId;let f=this.authorization.update.preMutation({ctx:e.context,value:u});if(!(typeof f=="boolean"?f:b(u,f)))throw new Error("Not authorized")}let s=await a.rawUpdate(e.resource,e.resourceId,o);if((p=(l=this.authorization)==null?void 0:l.update)!=null&&p.postMutation){let u=x(s);u.id=u.id??e.resourceId;let f=this.authorization.update.postMutation({ctx:e.context,value:u});if(!(typeof f=="boolean"?f:b(u,f)))throw new Error("Not authorized")}return {data:s,acceptedValues:c}})};wrapInMiddlewares(e){return t=>Array.from(this.middlewares.values()).reduceRight((n,i)=>a=>i({req:a,next:n}),e)(t)}},ee=class r{middlewares;constructor(e=[]){this.middlewares=e;}collectionRoute(e,t){return new X(e,void 0,t).use(...this.middlewares)}use(...e){return new r([...this.middlewares,...e])}static create(){return new r}},yn=ee.create;var $=class{async insert(e,t){let n=new Date().toISOString();return x(await this.rawInsert(e.name,t.id,{value:Object.fromEntries(Object.entries(t).map(([i,a])=>[i,{value:a,_meta:{timestamp:n}}]))}))}async update(e,t,n){let i=new Date().toISOString(),{id:a,...o}=n;return x(await this.rawUpdate(e.name,t,{value:Object.fromEntries(Object.entries(o).map(([c,s])=>[c,{value:s,_meta:{timestamp:i}}]))}))}};function q(r,e,t,n){if(!r)throw new Error("Schema not initialized");let i=r[e];if(!i)throw new Error("Resource not found");let a=n.$or,o=n.$and;return (a?t.or:t.and)(a?n.$or.map(c=>q(r,e,t,c)):o?n.$and.map(c=>q(r,e,t,c)):Object.entries(n).map(([c,s])=>{var m,y;if(i.fields[c])return (s==null?void 0:s.$eq)!==void 0?t(`${e}.${c}`,s.$eq===null?"is":"=",s.$eq):(s==null?void 0:s.$in)!==void 0?t(`${e}.${c}`,"in",s.$in):(s==null?void 0:s.$not)!==void 0?((m=s==null?void 0:s.$not)==null?void 0:m.$in)!==void 0?t(`${e}.${c}`,"not in",s.$not.$in):((y=s==null?void 0:s.$not)==null?void 0:y.$eq)!==void 0?t(`${e}.${c}`,s.$not.$eq===null?"is not":"!=",s.$not.$eq):t(`${e}.${c}`,s.$not===null?"is not":"!=",s.$not):(s==null?void 0:s.$gt)!==void 0?t(`${e}.${c}`,">",s.$gt):(s==null?void 0:s.$gte)!==void 0?t(`${e}.${c}`,">=",s.$gte):(s==null?void 0:s.$lt)!==void 0?t(`${e}.${c}`,"<",s.$lt):(s==null?void 0:s.$lte)!==void 0?t(`${e}.${c}`,"<=",s.$lte):t(`${e}.${c}`,s===null?"is":"=",s);if(i.relations[c]){let d=i.relations[c],l=d.entity.name;return d.type==="many"?t.exists(te(r,l,t.selectFrom(l).select("id").whereRef(d.foreignColumn,"=",`${e}.id`),s)):q(r,l,t,s)}return null}).filter(Boolean))}function F(r,e,t,n){let i=r[e];if(!i)throw new Error("Resource not found");if(!n)return t;if(n.$and){for(let a of n.$and)t=F(r,e,t,a);return t}else if(n.$or){for(let a of n.$or)t=F(r,e,t,a);return t}for(let[a,o]of Object.entries(n)){if(!i.relations[a])continue;let c=i.relations[a],s=c.entity.name,m=c.type==="one"?"id":c.foreignColumn,y=c.type==="one"?c.relationalColumn:"id";t=t.leftJoin(s,`${s}.${m}`,`${e}.${y}`),o instanceof Object&&!Array.isArray(o)&&o!==null&&(t=F(r,s,t,o));}return t}function te(r,e,t,n){return !n||Object.keys(n).length===0?t:(t=F(r,e,t,n),t.where(i=>q(r,e,i,n)))}function K(r,e,t,n){if(!n)return t;if(!r)throw new Error("Schema not initialized");let i=r[e];if(!i)throw new Error(`Resource not found: ${e}`);for(let a of Object.keys(n)){if(!i.relations[a])throw new Error(`Relation ${a} not found in resource ${e}`);let o=i.relations[a],c=o.entity.name,s=n[a],m=o.type==="one"?"id":o.foreignColumn,y=o.type==="one"?o.relationalColumn:"id",d=o.type==="one"?postgres.jsonObjectFrom:postgres.jsonArrayFrom,l=typeof s=="object"&&s!==null;t=t.select(p=>{let u=p.selectFrom(c).selectAll(c).whereRef(`${c}.${m}`,"=",`${e}.${y}`).select(f=>postgres.jsonObjectFrom(f.selectFrom(`${c}_meta`).selectAll(`${c}_meta`).whereRef(`${c}_meta.id`,"=",`${c}.id`)).as("_meta"));return l&&(u=K(r,c,u,s)),d(u).as(a)});}return t}var ne=class r extends ${db;schema;constructor(e,t){super(),this.isKyselyLike(e)?this.db=e:this.db=new kysely.Kysely({dialect:new kysely.PostgresDialect({pool:e})}),this.schema=t,this.rawInsert=this.rawInsert.bind(this),this.rawUpdate=this.rawUpdate.bind(this);}async updateSchema(e){this.schema=e;let t=await this.db.introspection.getTables();for(let[n,i]of Object.entries(e)){let a=t.find(s=>s.name===n);a||await this.db.schema.createTable(n).ifNotExists().execute();let o=`${n}_meta`,c=t.find(s=>s.name===o);c||await this.db.schema.createTable(o).ifNotExists().execute();for(let[s,m]of Object.entries(i.fields)){let y=a==null?void 0:a.columns.find(p=>p.name===s),d=m.getStorageFieldType();y?y.dataType!==d.type&&console.error("Column type mismatch:",s,"expected to have type:",d.type,"but has type:",y.dataType):(await this.db.schema.alterTable(n).addColumn(s,d.type,p=>{let u=p;return d.unique&&(u=u.unique()),d.nullable||(u=u.notNull()),d.references&&(u=u.references(d.references)),d.primary&&(u=u.primaryKey()),d.default!==void 0&&(u=u.defaultTo(d.default)),u}).execute().catch(p=>{throw console.error("Error adding column",s,p),p}),d.index&&await this.db.schema.createIndex(`${n}_${s}_index`).on(n).column(s).execute().catch(()=>{})),(c==null?void 0:c.columns.find(p=>p.name===s))||await this.db.schema.alterTable(o).addColumn(s,"varchar",p=>{let u=p;return d.primary&&(u=u.primaryKey().references(`${n}.${s}`)),u}).execute();}}}async rawFindById(e,t,n){if(!this.schema)throw new Error("Schema not initialized");let i=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"));i=K(this.schema,e,i,n);let a=await i.executeTakeFirst();if(a)return this.convertToMaterializedLiveType(a)}async findOne(e,t,n){let i=await this.rawFindById(e.name,t,n==null?void 0:n.include);if(i)return x(i)}async rawFind(e){if(!this.schema)throw new Error("Schema not initialized");let{resource:t,where:n,include:i,limit:a,sort:o}=e,c=this.db.selectFrom(t).selectAll(t).select(d=>postgres.jsonObjectFrom(d.selectFrom(`${t}_meta`).selectAll(`${t}_meta`).whereRef(`${t}_meta.id`,"=",`${t}.id`)).as("_meta"));c=te(this.schema,t,c,n),c=K(this.schema,t,c,i),a!==void 0&&(c=c.limit(a)),o!==void 0&&o.forEach(d=>{c=c.orderBy(d.key,d.direction);});let s=await c.execute(),m=Object.fromEntries(s.map(d=>{let{id:l}=d;return [l,d]}));return Object.keys(m).length===0?{}:Object.entries(m).reduce((d,[l,p])=>(d[l]=this.convertToMaterializedLiveType(p),d),{})}async find(e,t){let n=await this.rawFind({resource:e.name,where:t==null?void 0:t.where,include:t==null?void 0:t.include,limit:t==null?void 0:t.limit,sort:t==null?void 0:t.sort});return Object.fromEntries(Object.entries(n).map(([i,a])=>[i,x(a)]))}async rawInsert(e,t,n){var o;let i={},a={};for(let[c,s]of Object.entries(n.value)){let m=(o=s._meta)==null?void 0:o.timestamp;m&&(i[c]=s.value,a[c]=m);}return await this.db.insertInto(e).values({...i,id:t}).execute().then(()=>{this.db.insertInto(`${e}_meta`).values({...a,id:t}).execute();}),n}async rawUpdate(e,t,n){var o;let i={},a={};for(let[c,s]of Object.entries(n.value)){let m=(o=s._meta)==null?void 0:o.timestamp;m&&(i[c]=s.value,a[c]=m);}return await Promise.all([this.db.updateTable(e).set(i).where("id","=",t).execute(),this.db.insertInto(`${e}_meta`).values({...a,id:t}).onConflict(c=>c.column("id").doUpdateSet(a)).execute()]),n}async transaction(e){if(!this.schema)throw new Error("Schema not initialized");if(this.db.isTransaction){let n=Math.random().toString(36).substring(2,15),i=await this.db.savepoint(n).execute();try{return await e({trx:this,commit:()=>i.releaseSavepoint(n).execute().then(()=>{}),rollback:()=>i.rollbackToSavepoint(n).execute().then(()=>{})}).then(a=>i.isCommitted||i.isRolledBack?a:i.releaseSavepoint(n).execute().then(()=>a))}catch(a){throw await i.rollbackToSavepoint(n).execute().catch(()=>{}),a}}let t=await this.db.startTransaction().execute();try{return await e({trx:new r(t,this.schema),commit:()=>t.commit().execute(),rollback:()=>t.rollback().execute()}).then(n=>t.isCommitted||t.isRolledBack?n:t.commit().execute().then(()=>n))}catch(n){throw await t.rollback().execute(),n}}convertToMaterializedLiveType(e){return {value:Object.entries(e).reduce((t,[n,i])=>{var a,o,c;return n==="_meta"||(n==="id"?t[n]={value:i}:Array.isArray(i)?t[n]={value:i.map(s=>this.convertToMaterializedLiveType(s)),_meta:{timestamp:(a=e==null?void 0:e._meta)==null?void 0:a[n]}}:typeof i=="object"&&i!==null&&!(i instanceof Date)?t[n]={...this.convertToMaterializedLiveType(i),_meta:{timestamp:(o=e==null?void 0:e._meta)==null?void 0:o[n]}}:t[n]={value:i,_meta:{timestamp:(c=e==null?void 0:e._meta)==null?void 0:c[n]}}),t},{})}}isKyselyLike(e){if(e instanceof kysely.Kysely)return true;if(!e||typeof e!="object")return false;let t=e,n=typeof t.selectFrom=="function",i=typeof t.startTransaction=="function",a=typeof t.savepoint=="function",o=typeof t.isTransaction=="boolean"||typeof t.isTransaction=="function";return n&&i||a&&o}};var re=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);}}handleQuery(e){let t=new z(this.storage);return this.wrapInMiddlewares(async n=>{var m;let i=Se(n,this.schema,{stepId:"query",collectionName:n.resource,included:Object.keys(n.include??{})}),a={headers:n.headers,cookies:n.cookies,queryParams:n.queryParams,context:n.context},o={};for(let y=0;y<i.length;y++){let d=i[y],l=this.router.routes[d.resource];if(!l)throw new Error("Invalid resource");let p=d.getWhere&&d.referenceGetter?d.referenceGetter(o).map(d.getWhere):[void 0],u=(m=o[d.prevStepId??""])==null?void 0:m.flatMap(R=>{var M;return Object.keys(((M=R==null?void 0:R.result)==null?void 0:M.data)??{})}),h=(await Promise.allSettled(p.map(async(R,M)=>{let C=u==null?void 0:u[M],Me=await l.handleQuery({req:{type:"QUERY",...d,...a,where:d.where,relationalWhere:R},batcher:t});return {includedBy:C,result:Me}}))).flatMap(R=>R.status==="fulfilled"?[R.value]:[]);o[d.stepId]=h;}let c=Object.fromEntries(Object.entries(o).flatMap(([y,d],l)=>d.flatMap(p=>Object.entries(p.result.data).map(([u,f])=>[`${y}.${u}`,{data:f,includedBy:y!=="query"&&p.includedBy?`${y.split(".").slice(0,-1).join(".")}.${p.includedBy}`:void 0,path:y.split(".").slice(-1)[0],isMany:i[l].isMany,collectionName:i[l].collectionName,included:i[l].included}]))));return Object.keys(c).reduceRight((y,d)=>{var u,f;let l=c[d],p=l.path;if(p==="query"&&(y.data[d]=l.data),l.included.length)for(let h of l.included)l.data.value[h]??=((f=(u=this.schema[l.collectionName])==null?void 0:u.relations[h])==null?void 0:f.type)==="many"?{value:[]}:{value:null};if(l.includedBy){let h=c[l.includedBy];if(!h)return y;l.isMany?(h.data.value[p]??={value:[]},h.data.value[p].value.push(l.data)):h.data.value[p]=l.data;}return y},{data:{}})})(e.req)}async handleMutation(e){let t=await this.wrapInMiddlewares(async n=>{let i=this.router.routes[n.resource];if(!i)throw new Error("Invalid resource");return i.handleMutation({req:n,db:this.storage})})(e.req);if(t&&e.req.type==="MUTATE"&&t.acceptedValues&&(e.req.procedure==="INSERT"||e.req.procedure==="UPDATE")&&e.req.resourceId){let i=t.acceptedValues??{},a=e.req,o=a.resourceId;Object.keys(i).length&&o&&this.mutationSubscriptions.forEach(c=>{c({id:e.req.context.messageId,type:"MUTATE",resource:a.resource,payload:i,resourceId:o,procedure:a.procedure});});}return t}use(e){return this.middlewares.add(e),this}context(e){return this.contextProvider=e,this}wrapInMiddlewares(e){return t=>Array.from(this.middlewares.values()).reduceRight((n,i)=>a=>i({req:a,next:n}),e)(t)}},On=re.create;function Se(r,e,t){let{include:n,...i}=r,{stepId:a}=t,o=[{...i,...t}];if(n&&typeof n=="object"&&Object.keys(n).length>0){let c=e[i.resource];if(!c)throw new Error(`Resource ${i.resource} not found`);o.push(...Object.entries(n).flatMap(([s,m])=>{let y=c.relations[s];if(!y)throw new Error(`Relation ${s} not found for resource ${i.resource}`);let d=y.entity.name;return Se({...i,resource:d,include:m},e,{getWhere:y.type==="one"?l=>({id:l}):l=>({[y.foreignColumn]:l}),referenceGetter:l=>l[a].flatMap(p=>p.result.data?y.type==="one"?Object.values(p.result.data).map(u=>{var f,h;return (h=(f=u.value)==null?void 0:f[y.relationalColumn])==null?void 0:h.value}):Object.keys(p.result.data):[]),stepId:`${a}.${s}`,prevStepId:a,isMany:y.type==="many",collectionName:d,included:typeof m=="object"?Object.keys(m):[]})}));}return o}
2
+ exports.Route=X;exports.RouteFactory=ee;exports.Router=Y;exports.SQLStorage=ne;exports.Server=re;exports.Storage=$;exports.expressAdapter=Kt;exports.routeFactory=yn;exports.router=dn;exports.server=On;
package/dist/server.d.cts CHANGED
@@ -1,10 +1,25 @@
1
1
  import { z } from 'zod';
2
- import { LiveObjectAny, Schema, WhereClause, InferLiveObjectWithRelationalIds, MaterializedLiveType, IncludeClause, InferLiveObject, InferInsert, InferUpdate } from './index.cjs';
2
+ import { LiveObjectAny, WhereClause, InferLiveObjectWithRelationalIds, MaterializedLiveType, Schema, IncludeClause, InferLiveObject, InferInsert, InferUpdate } from './index.cjs';
3
3
  import * as z3 from 'zod/v3';
4
4
  import * as z4 from 'zod/v4/core';
5
5
  import { PostgresPool } from 'kysely';
6
6
  import { Application } from 'express-ws';
7
7
 
8
+ declare const querySchema: z.ZodObject<{
9
+ resource: z.ZodString;
10
+ where: z.ZodOptional<z.ZodRecord<z.ZodString, z.ZodAny>>;
11
+ include: z.ZodOptional<z.ZodRecord<z.ZodString, z.ZodAny>>;
12
+ lastSyncedAt: z.ZodOptional<z.ZodString>;
13
+ limit: z.ZodOptional<z.ZodCoercedNumber<unknown>>;
14
+ sort: z.ZodOptional<z.ZodArray<z.ZodObject<{
15
+ key: z.ZodString;
16
+ direction: z.ZodEnum<{
17
+ asc: "asc";
18
+ desc: "desc";
19
+ }>;
20
+ }, z.core.$strip>>>;
21
+ }, z.core.$strip>;
22
+ type RawQueryRequest = z.infer<typeof querySchema>;
8
23
  declare const defaultMutationSchema: z.ZodObject<{
9
24
  id: z.ZodOptional<z.ZodString>;
10
25
  type: z.ZodLiteral<"MUTATE">;
@@ -25,6 +40,8 @@ type DefaultMutation = Omit<z.infer<typeof defaultMutationSchema>, "resourceId">
25
40
  resourceId: string;
26
41
  };
27
42
 
43
+ type Awaitable<T> = T | Promise<T>;
44
+
28
45
  type Simplify<T> = T extends Record<string, unknown> ? {
29
46
  [K in keyof T]: Simplify<T[K]>;
30
47
  } : T extends Array<infer U> ? Array<Simplify<U>> : T;
@@ -52,24 +69,22 @@ type MutationResult<TShape extends LiveObjectAny> = {
52
69
  data: MaterializedLiveType<TShape>;
53
70
  acceptedValues: Record<string, any> | null;
54
71
  };
55
- type RequestHandler<TInput, TResult, TSchema extends Schema<any> = Schema<any>> = (opts: {
56
- req: ParsedRequest<TInput>;
57
- db: Storage;
58
- schema: TSchema;
59
- }) => Promise<TResult>;
60
72
  type Mutation<TInputValidator extends z3.ZodTypeAny | z4.$ZodType, // TODO use StandardSchema instead
61
- THandler extends RequestHandler<z.infer<TInputValidator>, any, any>> = {
73
+ TOutput> = {
62
74
  inputValidator: TInputValidator;
63
- handler: THandler;
75
+ handler: (opts: {
76
+ req: MutationRequest<z.infer<TInputValidator>>;
77
+ db: Storage;
78
+ }) => TOutput;
64
79
  };
65
80
  declare const mutationCreator: <TInputValidator extends z3.ZodTypeAny | z4.$ZodType>(validator?: TInputValidator) => {
66
- handler: <THandler extends RequestHandler<z.infer<TInputValidator>, any, any>>(handler: THandler) => Mutation<TInputValidator, THandler>;
81
+ handler: <THandler extends Mutation<TInputValidator, any>["handler"]>(handler: THandler) => Mutation<TInputValidator, ReturnType<THandler>>;
67
82
  };
68
83
  type ReadAuthorizationHandler<TShape extends LiveObjectAny> = (opts: {
69
- ctx: ParsedRequest["context"];
84
+ ctx: BaseRequest["context"];
70
85
  }) => WhereClause<TShape> | boolean;
71
86
  type MutationAuthorizationHandler<TShape extends LiveObjectAny> = (opts: {
72
- ctx: ParsedRequest["context"];
87
+ ctx: BaseRequest["context"];
73
88
  value: Simplify<InferLiveObjectWithRelationalIds<TShape>>;
74
89
  }) => WhereClause<TShape> | boolean;
75
90
  type Authorization<TShape extends LiveObjectAny> = {
@@ -80,24 +95,18 @@ type Authorization<TShape extends LiveObjectAny> = {
80
95
  postMutation?: MutationAuthorizationHandler<TShape>;
81
96
  };
82
97
  };
83
- declare class Route<TResourceSchema extends LiveObjectAny, TMiddleware extends Middleware<any>, TCustomMutations extends Record<string, Mutation<any, RequestHandler<any, any>>>> {
84
- readonly _resourceSchema: TResourceSchema;
85
- readonly resourceName: TResourceSchema["name"];
98
+ declare class Route<TResourceSchema extends LiveObjectAny, TMiddleware extends Middleware<any>, TCustomMutations extends Record<string, Mutation<any, any>>> {
99
+ readonly resourceSchema: TResourceSchema;
86
100
  readonly middlewares: Set<TMiddleware>;
87
101
  readonly customMutations: TCustomMutations;
88
102
  readonly authorization?: Authorization<TResourceSchema>;
89
- constructor(resourceName: TResourceSchema["name"], customMutations?: TCustomMutations, authorization?: Authorization<TResourceSchema>);
90
- handleRequest(opts: {
91
- req: ParsedRequest;
92
- db: Storage;
93
- schema: Schema<any>;
94
- }): Promise<any>;
103
+ constructor(resourceSchema: TResourceSchema, customMutations?: TCustomMutations, authorization?: Authorization<TResourceSchema>);
95
104
  use(...middlewares: TMiddleware[]): this;
96
- withMutations<T extends Record<string, Mutation<any, RequestHandler<any, any>>>>(mutationFactory: (opts: {
105
+ withMutations<T extends Record<string, Mutation<any, any>>>(mutationFactory: (opts: {
97
106
  mutation: typeof mutationCreator;
98
107
  }) => T): Route<TResourceSchema, TMiddleware, T>;
99
- private handleFind;
100
108
  private handleSet;
109
+ private wrapInMiddlewares;
101
110
  }
102
111
  declare class RouteFactory {
103
112
  private middlewares;
@@ -140,6 +149,11 @@ declare class SQLStorage extends Storage {
140
149
  find<T extends LiveObjectAny>(resource: T, options?: {
141
150
  where?: WhereClause<T>;
142
151
  include?: IncludeClause<T>;
152
+ limit?: number;
153
+ sort?: {
154
+ key: string;
155
+ direction: "asc" | "desc";
156
+ }[];
143
157
  }): Promise<Record<string, InferLiveObject<T>>>;
144
158
  transaction<T>(fn: (opts: {
145
159
  trx: Storage;
@@ -154,27 +168,32 @@ declare const expressAdapter: (app: Application, server: Server<AnyRouter>, opti
154
168
  basePath?: string;
155
169
  }) => void;
156
170
 
157
- type ParsedRequest<TInput = any> = {
171
+ /** biome-ignore-all lint/suspicious/noExplicitAny: any's are actually used correctly */
172
+
173
+ interface BaseRequest {
158
174
  headers: Record<string, string>;
159
175
  cookies: Record<string, string>;
160
- query: Record<string, string>;
161
- resourceName: string;
162
- procedure?: string;
176
+ queryParams: Record<string, string>;
163
177
  context: Record<string, any>;
164
- where?: Record<string, any>;
165
- include?: Record<string, any>;
166
- type: "QUERY" | "MUTATE";
178
+ }
179
+ interface QueryRequest extends BaseRequest, RawQueryRequest {
180
+ type: "QUERY";
181
+ }
182
+ interface MutationRequest<TInput = any> extends BaseRequest {
183
+ type: "MUTATE";
184
+ input: TInput;
185
+ resource: string;
167
186
  resourceId?: string;
168
- input?: TInput;
169
- };
170
- type ContextProvider = (req: Pick<ParsedRequest, "headers" | "cookies" | "query"> & {
187
+ procedure: string;
188
+ }
189
+ type Request = QueryRequest | MutationRequest;
190
+ type ContextProvider = (req: Omit<BaseRequest, "context"> & {
171
191
  transport: "HTTP" | "WEBSOCKET";
172
192
  }) => Record<string, any>;
173
- type RequestType = ParsedRequest["type"];
174
193
  type MutationHandler = (mutation: DefaultMutation) => void;
175
- type NextFunction<T> = (req: ParsedRequest) => Promise<T> | T;
194
+ type NextFunction<O, R = Request> = (req: R) => Awaitable<O>;
176
195
  type Middleware<T = any> = (opts: {
177
- req: ParsedRequest;
196
+ req: BaseRequest;
178
197
  next: NextFunction<T>;
179
198
  }) => ReturnType<NextFunction<T>>;
180
199
  declare class Server<TRouter extends AnyRouter> {
@@ -193,12 +212,16 @@ declare class Server<TRouter extends AnyRouter> {
193
212
  contextProvider?: ContextProvider;
194
213
  }): Server<TRouter>;
195
214
  subscribeToMutations(handler: MutationHandler): () => void;
196
- handleRequest(opts: {
197
- req: ParsedRequest;
215
+ handleQuery(opts: {
216
+ req: QueryRequest;
217
+ }): Promise<QueryResult<any>>;
218
+ handleMutation(opts: {
219
+ req: MutationRequest;
198
220
  }): Promise<any>;
199
221
  use(middleware: Middleware<any>): this;
200
222
  context(contextProvider: ContextProvider): this;
223
+ private wrapInMiddlewares;
201
224
  }
202
225
  declare const server: typeof Server.create;
203
226
 
204
- export { type AnyRoute, type AnyRouter, type Authorization, type ContextProvider, type Middleware, type Mutation, type MutationAuthorizationHandler, type MutationHandler, type MutationResult, type NextFunction, type ParsedRequest, type QueryResult, type ReadAuthorizationHandler, type RequestHandler, type RequestType, Route, RouteFactory, type RouteRecord, Router, SQLStorage, Server, Storage, expressAdapter, routeFactory, router, server };
227
+ export { type AnyRoute, type AnyRouter, type Authorization, type BaseRequest, type ContextProvider, type Middleware, type Mutation, type MutationAuthorizationHandler, type MutationHandler, type MutationRequest, type MutationResult, type NextFunction, type QueryRequest, type QueryResult, type ReadAuthorizationHandler, type Request, Route, RouteFactory, type RouteRecord, Router, SQLStorage, Server, Storage, expressAdapter, routeFactory, router, server };
package/dist/server.d.ts CHANGED
@@ -1,10 +1,25 @@
1
1
  import { z } from 'zod';
2
- import { LiveObjectAny, Schema, WhereClause, InferLiveObjectWithRelationalIds, MaterializedLiveType, IncludeClause, InferLiveObject, InferInsert, InferUpdate } from './index.js';
2
+ import { LiveObjectAny, WhereClause, InferLiveObjectWithRelationalIds, MaterializedLiveType, Schema, IncludeClause, InferLiveObject, InferInsert, InferUpdate } from './index.js';
3
3
  import * as z3 from 'zod/v3';
4
4
  import * as z4 from 'zod/v4/core';
5
5
  import { PostgresPool } from 'kysely';
6
6
  import { Application } from 'express-ws';
7
7
 
8
+ declare const querySchema: z.ZodObject<{
9
+ resource: z.ZodString;
10
+ where: z.ZodOptional<z.ZodRecord<z.ZodString, z.ZodAny>>;
11
+ include: z.ZodOptional<z.ZodRecord<z.ZodString, z.ZodAny>>;
12
+ lastSyncedAt: z.ZodOptional<z.ZodString>;
13
+ limit: z.ZodOptional<z.ZodCoercedNumber<unknown>>;
14
+ sort: z.ZodOptional<z.ZodArray<z.ZodObject<{
15
+ key: z.ZodString;
16
+ direction: z.ZodEnum<{
17
+ asc: "asc";
18
+ desc: "desc";
19
+ }>;
20
+ }, z.core.$strip>>>;
21
+ }, z.core.$strip>;
22
+ type RawQueryRequest = z.infer<typeof querySchema>;
8
23
  declare const defaultMutationSchema: z.ZodObject<{
9
24
  id: z.ZodOptional<z.ZodString>;
10
25
  type: z.ZodLiteral<"MUTATE">;
@@ -25,6 +40,8 @@ type DefaultMutation = Omit<z.infer<typeof defaultMutationSchema>, "resourceId">
25
40
  resourceId: string;
26
41
  };
27
42
 
43
+ type Awaitable<T> = T | Promise<T>;
44
+
28
45
  type Simplify<T> = T extends Record<string, unknown> ? {
29
46
  [K in keyof T]: Simplify<T[K]>;
30
47
  } : T extends Array<infer U> ? Array<Simplify<U>> : T;
@@ -52,24 +69,22 @@ type MutationResult<TShape extends LiveObjectAny> = {
52
69
  data: MaterializedLiveType<TShape>;
53
70
  acceptedValues: Record<string, any> | null;
54
71
  };
55
- type RequestHandler<TInput, TResult, TSchema extends Schema<any> = Schema<any>> = (opts: {
56
- req: ParsedRequest<TInput>;
57
- db: Storage;
58
- schema: TSchema;
59
- }) => Promise<TResult>;
60
72
  type Mutation<TInputValidator extends z3.ZodTypeAny | z4.$ZodType, // TODO use StandardSchema instead
61
- THandler extends RequestHandler<z.infer<TInputValidator>, any, any>> = {
73
+ TOutput> = {
62
74
  inputValidator: TInputValidator;
63
- handler: THandler;
75
+ handler: (opts: {
76
+ req: MutationRequest<z.infer<TInputValidator>>;
77
+ db: Storage;
78
+ }) => TOutput;
64
79
  };
65
80
  declare const mutationCreator: <TInputValidator extends z3.ZodTypeAny | z4.$ZodType>(validator?: TInputValidator) => {
66
- handler: <THandler extends RequestHandler<z.infer<TInputValidator>, any, any>>(handler: THandler) => Mutation<TInputValidator, THandler>;
81
+ handler: <THandler extends Mutation<TInputValidator, any>["handler"]>(handler: THandler) => Mutation<TInputValidator, ReturnType<THandler>>;
67
82
  };
68
83
  type ReadAuthorizationHandler<TShape extends LiveObjectAny> = (opts: {
69
- ctx: ParsedRequest["context"];
84
+ ctx: BaseRequest["context"];
70
85
  }) => WhereClause<TShape> | boolean;
71
86
  type MutationAuthorizationHandler<TShape extends LiveObjectAny> = (opts: {
72
- ctx: ParsedRequest["context"];
87
+ ctx: BaseRequest["context"];
73
88
  value: Simplify<InferLiveObjectWithRelationalIds<TShape>>;
74
89
  }) => WhereClause<TShape> | boolean;
75
90
  type Authorization<TShape extends LiveObjectAny> = {
@@ -80,24 +95,18 @@ type Authorization<TShape extends LiveObjectAny> = {
80
95
  postMutation?: MutationAuthorizationHandler<TShape>;
81
96
  };
82
97
  };
83
- declare class Route<TResourceSchema extends LiveObjectAny, TMiddleware extends Middleware<any>, TCustomMutations extends Record<string, Mutation<any, RequestHandler<any, any>>>> {
84
- readonly _resourceSchema: TResourceSchema;
85
- readonly resourceName: TResourceSchema["name"];
98
+ declare class Route<TResourceSchema extends LiveObjectAny, TMiddleware extends Middleware<any>, TCustomMutations extends Record<string, Mutation<any, any>>> {
99
+ readonly resourceSchema: TResourceSchema;
86
100
  readonly middlewares: Set<TMiddleware>;
87
101
  readonly customMutations: TCustomMutations;
88
102
  readonly authorization?: Authorization<TResourceSchema>;
89
- constructor(resourceName: TResourceSchema["name"], customMutations?: TCustomMutations, authorization?: Authorization<TResourceSchema>);
90
- handleRequest(opts: {
91
- req: ParsedRequest;
92
- db: Storage;
93
- schema: Schema<any>;
94
- }): Promise<any>;
103
+ constructor(resourceSchema: TResourceSchema, customMutations?: TCustomMutations, authorization?: Authorization<TResourceSchema>);
95
104
  use(...middlewares: TMiddleware[]): this;
96
- withMutations<T extends Record<string, Mutation<any, RequestHandler<any, any>>>>(mutationFactory: (opts: {
105
+ withMutations<T extends Record<string, Mutation<any, any>>>(mutationFactory: (opts: {
97
106
  mutation: typeof mutationCreator;
98
107
  }) => T): Route<TResourceSchema, TMiddleware, T>;
99
- private handleFind;
100
108
  private handleSet;
109
+ private wrapInMiddlewares;
101
110
  }
102
111
  declare class RouteFactory {
103
112
  private middlewares;
@@ -140,6 +149,11 @@ declare class SQLStorage extends Storage {
140
149
  find<T extends LiveObjectAny>(resource: T, options?: {
141
150
  where?: WhereClause<T>;
142
151
  include?: IncludeClause<T>;
152
+ limit?: number;
153
+ sort?: {
154
+ key: string;
155
+ direction: "asc" | "desc";
156
+ }[];
143
157
  }): Promise<Record<string, InferLiveObject<T>>>;
144
158
  transaction<T>(fn: (opts: {
145
159
  trx: Storage;
@@ -154,27 +168,32 @@ declare const expressAdapter: (app: Application, server: Server<AnyRouter>, opti
154
168
  basePath?: string;
155
169
  }) => void;
156
170
 
157
- type ParsedRequest<TInput = any> = {
171
+ /** biome-ignore-all lint/suspicious/noExplicitAny: any's are actually used correctly */
172
+
173
+ interface BaseRequest {
158
174
  headers: Record<string, string>;
159
175
  cookies: Record<string, string>;
160
- query: Record<string, string>;
161
- resourceName: string;
162
- procedure?: string;
176
+ queryParams: Record<string, string>;
163
177
  context: Record<string, any>;
164
- where?: Record<string, any>;
165
- include?: Record<string, any>;
166
- type: "QUERY" | "MUTATE";
178
+ }
179
+ interface QueryRequest extends BaseRequest, RawQueryRequest {
180
+ type: "QUERY";
181
+ }
182
+ interface MutationRequest<TInput = any> extends BaseRequest {
183
+ type: "MUTATE";
184
+ input: TInput;
185
+ resource: string;
167
186
  resourceId?: string;
168
- input?: TInput;
169
- };
170
- type ContextProvider = (req: Pick<ParsedRequest, "headers" | "cookies" | "query"> & {
187
+ procedure: string;
188
+ }
189
+ type Request = QueryRequest | MutationRequest;
190
+ type ContextProvider = (req: Omit<BaseRequest, "context"> & {
171
191
  transport: "HTTP" | "WEBSOCKET";
172
192
  }) => Record<string, any>;
173
- type RequestType = ParsedRequest["type"];
174
193
  type MutationHandler = (mutation: DefaultMutation) => void;
175
- type NextFunction<T> = (req: ParsedRequest) => Promise<T> | T;
194
+ type NextFunction<O, R = Request> = (req: R) => Awaitable<O>;
176
195
  type Middleware<T = any> = (opts: {
177
- req: ParsedRequest;
196
+ req: BaseRequest;
178
197
  next: NextFunction<T>;
179
198
  }) => ReturnType<NextFunction<T>>;
180
199
  declare class Server<TRouter extends AnyRouter> {
@@ -193,12 +212,16 @@ declare class Server<TRouter extends AnyRouter> {
193
212
  contextProvider?: ContextProvider;
194
213
  }): Server<TRouter>;
195
214
  subscribeToMutations(handler: MutationHandler): () => void;
196
- handleRequest(opts: {
197
- req: ParsedRequest;
215
+ handleQuery(opts: {
216
+ req: QueryRequest;
217
+ }): Promise<QueryResult<any>>;
218
+ handleMutation(opts: {
219
+ req: MutationRequest;
198
220
  }): Promise<any>;
199
221
  use(middleware: Middleware<any>): this;
200
222
  context(contextProvider: ContextProvider): this;
223
+ private wrapInMiddlewares;
201
224
  }
202
225
  declare const server: typeof Server.create;
203
226
 
204
- export { type AnyRoute, type AnyRouter, type Authorization, type ContextProvider, type Middleware, type Mutation, type MutationAuthorizationHandler, type MutationHandler, type MutationResult, type NextFunction, type ParsedRequest, type QueryResult, type ReadAuthorizationHandler, type RequestHandler, type RequestType, Route, RouteFactory, type RouteRecord, Router, SQLStorage, Server, Storage, expressAdapter, routeFactory, router, server };
227
+ export { type AnyRoute, type AnyRouter, type Authorization, type BaseRequest, type ContextProvider, type Middleware, type Mutation, type MutationAuthorizationHandler, type MutationHandler, type MutationRequest, type MutationResult, type NextFunction, type QueryRequest, type QueryResult, type ReadAuthorizationHandler, type Request, Route, RouteFactory, type RouteRecord, Router, SQLStorage, Server, Storage, expressAdapter, routeFactory, router, server };
package/dist/server.js CHANGED
@@ -1,2 +1,2 @@
1
- import {a,b,t}from'./chunk-7C33XQ3U.js';import Se,{parse}from'qs';import {z as z$1}from'zod';import P from'node:crypto';import'js-xxhash';import {Kysely,PostgresDialect}from'kysely';import {jsonObjectFrom,jsonArrayFrom}from'kysely/helpers/postgres';var C=a(O=>{Object.defineProperty(O,"__esModule",{value:true});O.parse=Te;O.serialize=ge;var le=/^[\u0021-\u003A\u003C\u003E-\u007E]+$/,pe=/^[\u0021-\u003A\u003C-\u007E]*$/,me=/^([.]?[a-z0-9]([a-z0-9-]{0,61}[a-z0-9])?)([.][a-z0-9]([a-z0-9-]{0,61}[a-z0-9])?)*$/i,fe=/^[\u0020-\u003A\u003D-\u007E]*$/,ye=Object.prototype.toString,he=(()=>{let i=function(){};return i.prototype=Object.create(null),i})();function Te(i,e){let t=new he,r=i.length;if(r<2)return t;let n=(e==null?void 0:e.decode)||xe,a=0;do{let s=i.indexOf("=",a);if(s===-1)break;let c=i.indexOf(";",a),o=c===-1?r:c;if(s>o){a=i.lastIndexOf(";",s-1)+1;continue}let y=H(i,a,s),f=Z(i,s,y),u=i.slice(y,f);if(t[u]===void 0){let d=H(i,s+1,o),p=Z(i,o,d),l=n(i.slice(d,p));t[u]=l;}a=o+1;}while(a<r);return t}function H(i,e,t){do{let r=i.charCodeAt(e);if(r!==32&&r!==9)return e}while(++e<t);return t}function Z(i,e,t){for(;e>t;){let r=i.charCodeAt(--e);if(r!==32&&r!==9)return e+1}return t}function ge(i,e,t){let r=(t==null?void 0:t.encode)||encodeURIComponent;if(!le.test(i))throw new TypeError(`argument name is invalid: ${i}`);let n=r(e);if(!pe.test(n))throw new TypeError(`argument val is invalid: ${e}`);let a=i+"="+n;if(!t)return a;if(t.maxAge!==void 0){if(!Number.isInteger(t.maxAge))throw new TypeError(`option maxAge is invalid: ${t.maxAge}`);a+="; Max-Age="+t.maxAge;}if(t.domain){if(!me.test(t.domain))throw new TypeError(`option domain is invalid: ${t.domain}`);a+="; Domain="+t.domain;}if(t.path){if(!fe.test(t.path))throw new TypeError(`option path is invalid: ${t.path}`);a+="; Path="+t.path;}if(t.expires){if(!be(t.expires)||!Number.isFinite(t.expires.valueOf()))throw new TypeError(`option expires is invalid: ${t.expires}`);a+="; Expires="+t.expires.toUTCString();}if(t.httpOnly&&(a+="; HttpOnly"),t.secure&&(a+="; Secure"),t.partitioned&&(a+="; Partitioned"),t.priority)switch(typeof t.priority=="string"?t.priority.toLowerCase():void 0){case "low":a+="; Priority=Low";break;case "medium":a+="; Priority=Medium";break;case "high":a+="; 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":a+="; SameSite=Strict";break;case "lax":a+="; SameSite=Lax";break;case "none":a+="; SameSite=None";break;default:throw new TypeError(`option sameSite is invalid: ${t.sameSite}`)}return a}function xe(i){if(i.indexOf("%")===-1)return i;try{return decodeURIComponent(i)}catch{return i}}function be(i){return ye.call(i)==="[object Date]"}});var Q=b(C(),1);var L=z$1.object({resource:z$1.string(),where:z$1.record(z$1.string(),z$1.any()).optional(),include:z$1.record(z$1.string(),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()}),z=z$1.record(z$1.string(),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((i,e)=>{i.id&&e.addIssue({code:z$1.ZodIssueCode.custom,message:"Payload cannot have an id"});}),B=z$1.object({id:z$1.string().optional(),type:z$1.literal("MUTATE"),resource:z$1.string(),resourceId:z$1.string().optional()}),M=B.extend({procedure:z$1.string(),payload:z$1.any().optional()}),E=B.extend({procedure:z$1.enum(["INSERT","UPDATE"]),payload:z});z$1.union([E,M]);var G=L.omit({resource:true}),v=M.omit({id:true,type:true,resource:true,procedure:true}),N=E.omit({id:true,type:true,resource:true,procedure:true});z$1.union([N,v]);var K=i=>async e=>{var t;try{let r=typeof e.headers.getSetCookie=="function"?Object.fromEntries(e.headers):e.headers,n={headers:r,cookies:r.cookie?Q.default.parse(r.cookie):{}},a=new URL(e.url),s=a.pathname.split("/"),c=a.searchParams,o=Se.parse(c.toString()),y=await((t=i.contextProvider)==null?void 0:t.call(i,{transport:"HTTP",headers:n.headers,cookies:n.cookies,query:o}))??{};if(e.method==="GET"){let f=s[s.length-1],{success:u,data:d,error:p}=G.safeParse(o);if(!u)return Response.json({message:"Invalid query",code:"INVALID_QUERY",details:p},{status:400});let l=await i.handleRequest({req:{...n,type:"QUERY",resourceName:f,context:y,where:d.where,include:d.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 f=s[s.length-1],u=s[s.length-2],d=e.body?await e.json():{},p;if(f==="insert"||f==="update"){let{success:T,data:g,error:b}=N.safeParse(d);if(!T)return Response.json({message:"Invalid mutation",code:"INVALID_REQUEST",details:b},{status:400});p=g;}else {let{success:T,data:g,error:b}=v.safeParse(d);if(!T)return Response.json({message:"Invalid mutation",code:"INVALID_REQUEST",details:b},{status:400});p=g;}let l=await i.handleRequest({req:{...n,type:"MUTATE",resourceName:u,input:p.payload,context:y,resourceId:p.resourceId,procedure:f==="insert"||f==="update"?f.toUpperCase():f,query:{}}});return Response.json(l)}catch(f){return console.error("Error parsing mutation from the client:",f),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 ne=b(C(),1);var I=z$1.string(),we=z$1.object({id:I,type:z$1.literal("SUBSCRIBE"),resource:z$1.string()}),Ie=L.extend({id:I,type:z$1.literal("QUERY")}),J=E.extend({id:I}),Me=M.extend({id:I}),Ee=z$1.union([Me,J]),Y=z$1.union([we,Ie,Ee]),Ae=z$1.object({id:I,type:z$1.literal("REJECT"),resource:z$1.string(),message:z$1.string().optional()}),$e=z$1.object({id:I,type:z$1.literal("REPLY"),data:z$1.any()});z$1.union([Ae,$e,J]);z$1.object({resource:z$1.string(),data:z$1.record(z$1.string(),z)});var ee="0123456789ABCDEFGHJKMNPQRSTVWXYZ",A=32;var Oe=16,te=10,X=0xffffffffffff;var S;(function(i){i.Base32IncorrectEncoding="B32_ENC_INVALID",i.DecodeTimeInvalidCharacter="DEC_TIME_CHAR",i.DecodeTimeValueMalformed="DEC_TIME_MALFORMED",i.EncodeTimeNegative="ENC_TIME_NEG",i.EncodeTimeSizeExceeded="ENC_TIME_SIZE_EXCEED",i.EncodeTimeValueMalformed="ENC_TIME_MALFORMED",i.PRNGDetectFailure="PRNG_DETECT",i.ULIDInvalid="ULID_INVALID",i.Unexpected="UNEXPECTED",i.UUIDInvalid="UUID_INVALID";})(S||(S={}));var w=class extends Error{constructor(e,t){super(`${t} (${e})`),this.name="ULIDError",this.code=e;}};function Le(i){let e=Math.floor(i()*A);return e===A&&(e=A-1),ee.charAt(e)}function Pe(i){var r;let e=je(),t=e&&(e.crypto||e.msCrypto)||(typeof P<"u"?P:null);if(typeof(t==null?void 0:t.getRandomValues)=="function")return ()=>{let n=new Uint8Array(1);return t.getRandomValues(n),n[0]/255};if(typeof(t==null?void 0:t.randomBytes)=="function")return ()=>t.randomBytes(1).readUInt8()/255;if((r=P)!=null&&r.randomBytes)return ()=>P.randomBytes(1).readUInt8()/255;throw new w(S.PRNGDetectFailure,"Failed to find a reliable PRNG")}function je(){return ve()?self:typeof window<"u"?window:typeof global<"u"?global:typeof globalThis<"u"?globalThis:null}function Ce(i,e){let t="";for(;i>0;i--)t=Le(e)+t;return t}function ze(i,e=te){if(isNaN(i))throw new w(S.EncodeTimeValueMalformed,`Time must be a number: ${i}`);if(i>X)throw new w(S.EncodeTimeSizeExceeded,`Cannot encode a time larger than ${X}: ${i}`);if(i<0)throw new w(S.EncodeTimeNegative,`Time must be positive: ${i}`);if(Number.isInteger(i)===false)throw new w(S.EncodeTimeValueMalformed,`Time must be an integer: ${i}`);let t,r="";for(let n=e;n>0;n--)t=i%A,r=ee.charAt(t)+r,i=(i-t)/A;return r}function ve(){return typeof WorkerGlobalScope<"u"&&self instanceof WorkerGlobalScope}function re(i,e){let t=Pe(),r=Date.now();return ze(r,te)+Ce(Oe,t)}var U=()=>re().toLowerCase();var ie=i=>{let e={},t={};return i.subscribeToMutations(r=>{let n=r;!n.resourceId||!n.payload||(console.log("Mutation propagated:",n),Object.entries(t[n.resource]??{}).forEach(([a,s])=>{var c;(c=e[a])==null||c.send(JSON.stringify({...n,id:n.id??U()}));}));}),(r,n)=>{var f;let a=u=>{r.send(JSON.stringify(u));},s=U(),c={headers:n.headers,cookies:typeof n.headers.cookie=="string"?ne.default.parse(n.headers.cookie):{}},o=parse(n.url.split("?")[1]),y=(f=i.contextProvider)==null?void 0:f.call(i,{transport:"WEBSOCKET",headers:c.headers,cookies:c.cookies,query:o});e[s]=r,console.log("Client connected:",s),r.on("message",async u=>{try{console.log("Message received from the client:",u);let d=Y.parse(JSON.parse(u.toString()));if(d.type==="SUBSCRIBE"){let{resource:p}=d;t[p]||(t[p]={}),t[p][s]={};}else if(d.type==="QUERY"){let{resource:p}=d,l=await i.handleRequest({req:{...c,type:"QUERY",resourceName:p,context:await y??{},query:o}});if(!l||!l.data)throw new Error("Invalid resource");a({id:d.id,type:"REPLY",data:{resource:p,data:Object.fromEntries(Object.entries(l.data??{}).map(([T,g])=>[T,g.value]))}});}else if(d.type==="MUTATE"){let{resource:p}=d;console.log("Received mutation from client:",d);try{let l=await i.handleRequest({req:{...c,type:"MUTATE",resourceName:p,input:d.payload,context:{messageId:d.id,...await y??{}},resourceId:d.resourceId,procedure:d.procedure,query:o}});d.procedure&&d.procedure!=="INSERT"&&d.procedure!=="UPDATE"&&a({id:d.id,type:"REPLY",data:l});}catch(l){a({id:d.id,type:"REJECT",resource:p,message:l.message}),console.error("Error parsing mutation from the client:",l);}}}catch(d){console.error("Error handling message from the client:",d);}}),r.on("close",()=>{console.log("Connection closed",s),delete e[s];for(let u of Object.values(t))delete u[s];});}};function oe(i){let e=`${i.protocol}://${i.hostname}${i.url}`,t=new Headers;return Object.entries(i.headers).forEach(([r,n])=>{n&&t.set(r,Array.isArray(n)?n.join(","):n);}),new Request(e,{method:i.method,headers:t,body:i.body&&i.method!=="GET"?JSON.stringify(i.body):void 0})}var Tt=(i,e,t)=>{i.ws(`${(t==null?void 0:t.basePath)??""}/ws`,ie(e)),i.use(`${(t==null?void 0:t.basePath)??""}/`,(r,n)=>{K(e)(oe(r)).then(s=>s.json().then(c=>n.status(s.status).send(c)));});};var R=(i,e,t=false)=>Object.entries(e).every(([r,n])=>{if(r==="$and")return n.every(s=>R(i,s,t));if(r==="$or")return n.some(s=>R(i,s,t));let a=(n==null?void 0:n.$eq)!==void 0?n==null?void 0:n.$eq:n;if(typeof n=="object"&&n!==null&&(n==null?void 0:n.$eq)===void 0){if(n.$in!==void 0){let s=i[r];return s===void 0?false:t?!n.$in.includes(s):n.$in.includes(s)}if(n.$not!==void 0&&!t)return R(i,{[r]:n.$not},true);if(n.$gt!==void 0){let s=i[r];return typeof s!="number"?false:t?s<=n.$gt:s>n.$gt}if(n.$gte!==void 0){let s=i[r];return typeof s!="number"?false:t?s<n.$gte:s>=n.$gte}if(n.$lt!==void 0){let s=i[r];return typeof s!="number"?false:t?s>=n.$lt:s<n.$lt}if(n.$lte!==void 0){let s=i[r];return typeof s!="number"?false:t?s>n.$lte:s<=n.$lte}return !i[r]||typeof i[r]!="object"?false:R(i[r],n,t)}return t?i[r]!==a:i[r]===a});var D=class i{routes;constructor(e){this.routes=e.routes;}static create(e){return new i(e)}},Mt=i=>D.create({...i}),De=i=>({handler:e=>({inputValidator:i??z$1.undefined(),handler:e})}),_=class i{_resourceSchema;resourceName;middlewares;customMutations;authorization;constructor(e,t,r){this.resourceName=e,this.middlewares=new Set,this.customMutations=t??{},this.authorization=r;}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)throw new Error("Procedure is required for mutations");let n=this.customMutations[r.procedure];if(n){let a=n.inputValidator.parse(r.input);return r.input=a,n.handler({req:r,db:e.db,schema:e.schema})}else if(r.procedure==="INSERT"||r.procedure==="UPDATE")return this.handleSet({req:r,db:e.db,schema:e.schema,operation:r.procedure})}throw new Error("Invalid request")})();return await Array.from(this.middlewares.values()).reduceRight((r,n)=>a=>n({req:a,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 i(this.resourceName,e({mutation:De}))}handleFind=async({req:e,db:t})=>{var n,a;let r=(a=(n=this.authorization)==null?void 0:n.read)==null?void 0:a.call(n,{ctx:e.context});if(typeof r=="boolean"&&!r)throw new Error("Not authorized");return {data:await t.rawFind(e.resourceName,e.where&&r&&r!==true?{$and:[e.where,r]}:r&&r!==true?r:e.where,e.include),acceptedValues:null}};handleSet=async({req:e,db:t$1,schema:r,operation:n})=>{if(!e.input)throw new Error("Payload is required");if(!e.resourceId)throw new Error("ResourceId is required");let a=await t$1.rawFindById(e.resourceName,e.resourceId);if(n==="INSERT"&&a)throw new Error("Resource already exists");if(n==="UPDATE"&&!a)throw new Error("Resource not found");return t$1.transaction(async({trx:s})=>{var f,u,d,p,l;let[c,o]=r[this.resourceName].mergeMutation("set",e.input,a);if(!o)throw new Error("Mutation rejected");if(n==="INSERT"){let T=await s.rawInsert(e.resourceName,e.resourceId,c);if((f=this.authorization)!=null&&f.insert){let g=t(T);g.id=g.id??e.resourceId;let b=this.authorization.insert({ctx:e.context,value:g});if(!(typeof b=="boolean"?b:R(g,b)))throw new Error("Not authorized")}return {data:T,acceptedValues:o}}if((d=(u=this.authorization)==null?void 0:u.update)!=null&&d.preMutation){let T=t(a);T.id=T.id??e.resourceId;let g=this.authorization.update.preMutation({ctx:e.context,value:T});if(!(typeof g=="boolean"?g:R(T,g)))throw new Error("Not authorized")}let y=await s.rawUpdate(e.resourceName,e.resourceId,c);if((l=(p=this.authorization)==null?void 0:p.update)!=null&&l.postMutation){let T=t(y);T.id=T.id??e.resourceId;let g=this.authorization.update.postMutation({ctx:e.context,value:T});if(!(typeof g=="boolean"?g:R(T,g)))throw new Error("Not authorized")}return {data:y,acceptedValues:o}})}},V=class i{middlewares;constructor(e=[]){this.middlewares=e;}collectionRoute(e,t){return new _(e.name,void 0,t).use(...this.middlewares)}use(...e){return new i([...this.middlewares,...e])}static create(){return new i}},Et=V.create;var $=class{async insert(e,t$1){let r=new Date().toISOString();return t(await this.rawInsert(e.name,t$1.id,{value:Object.fromEntries(Object.entries(t$1).map(([n,a])=>[n,{value:a,_meta:{timestamp:r}}]))}))}async update(e,t$1,r){let n=new Date().toISOString(),{id:a,...s}=r;return t(await this.rawUpdate(e.name,t$1,{value:Object.fromEntries(Object.entries(s).map(([c,o])=>[c,{value:o,_meta:{timestamp:n}}]))}))}};function j(i,e,t,r){if(!i)throw new Error("Schema not initialized");let n=i[e];if(!n)throw new Error("Resource not found");let a=r.$or,s=r.$and;return (a?t.or:t.and)(a?r.$or.map(c=>j(i,e,t,c)):s?r.$and.map(c=>j(i,e,t,c)):Object.entries(r).map(([c,o])=>{var y,f;if(n.fields[c])return (o==null?void 0:o.$eq)!==void 0?t(`${e}.${c}`,o.$eq===null?"is":"=",o.$eq):(o==null?void 0:o.$in)!==void 0?t(`${e}.${c}`,"in",o.$in):(o==null?void 0:o.$not)!==void 0?((y=o==null?void 0:o.$not)==null?void 0:y.$in)!==void 0?t(`${e}.${c}`,"not in",o.$not.$in):((f=o==null?void 0:o.$not)==null?void 0:f.$eq)!==void 0?t(`${e}.${c}`,o.$not.$eq===null?"is not":"!=",o.$not.$eq):t(`${e}.${c}`,o.$not===null?"is not":"!=",o.$not):(o==null?void 0:o.$gt)!==void 0?t(`${e}.${c}`,">",o.$gt):(o==null?void 0:o.$gte)!==void 0?t(`${e}.${c}`,">=",o.$gte):(o==null?void 0:o.$lt)!==void 0?t(`${e}.${c}`,"<",o.$lt):(o==null?void 0:o.$lte)!==void 0?t(`${e}.${c}`,"<=",o.$lte):t(`${e}.${c}`,o===null?"is":"=",o);if(n.relations[c]){let u=n.relations[c],d=u.entity.name;u.type==="one"?"id":u.foreignColumn;u.type==="one"?u.relationalColumn:"id";return j(i,d,t,o)}return null}).filter(Boolean))}function Ve(i,e,t,r){let n=i[e];if(!n)throw new Error("Resource not found");if(!r)return t;for(let a of Object.keys(r)){if(!n.relations[a])continue;let s=n.relations[a],c=s.entity.name,o=s.type==="one"?"id":s.foreignColumn,y=s.type==="one"?s.relationalColumn:"id";t=t.leftJoin(c,`${c}.${o}`,`${e}.${y}`);}return t}function se(i,e,t,r){return !r||Object.keys(r).length===0?t:(t=Ve(i,e,t,r),t.where(n=>j(i,e,n,r)))}function k(i,e,t,r){if(!r)return t;if(!i)throw new Error("Schema not initialized");let n=i[e];if(!n)throw new Error(`Resource not found: ${e}`);for(let a of Object.keys(r)){if(!n.relations[a])throw new Error(`Relation ${a} not found in resource ${e}`);let s=n.relations[a],c=s.entity.name,o=s.type==="one"?"id":s.foreignColumn,y=s.type==="one"?s.relationalColumn:"id",f=s.type==="one"?jsonObjectFrom:jsonArrayFrom;t=t.select(u=>f(u.selectFrom(c).selectAll(c).whereRef(`${c}.${o}`,"=",`${e}.${y}`).select(d=>jsonObjectFrom(d.selectFrom(`${c}_meta`).selectAll(`${c}_meta`).whereRef(`${c}_meta.id`,"=",`${c}.id`)).as("_meta"))).as(a));}return t}var F=class i extends ${db;schema;constructor(e,t){super(),this.isKyselyLike(e)?this.db=e:this.db=new Kysely({dialect:new PostgresDialect({pool:e})}),this.schema=t,this.rawInsert=this.rawInsert.bind(this),this.rawUpdate=this.rawUpdate.bind(this);}async updateSchema(e){this.schema=e;let t=await this.db.introspection.getTables();for(let[r,n]of Object.entries(e)){let a=t.find(o=>o.name===r);a||await this.db.schema.createTable(r).ifNotExists().execute();let s=`${r}_meta`,c=t.find(o=>o.name===s);c||await this.db.schema.createTable(s).ifNotExists().execute();for(let[o,y]of Object.entries(n.fields)){let f=a==null?void 0:a.columns.find(p=>p.name===o),u=y.getStorageFieldType();f?f.dataType!==u.type&&console.error("Column type mismatch:",o,"expected to have type:",u.type,"but has type:",f.dataType):(await this.db.schema.alterTable(r).addColumn(o,u.type,p=>{let l=p;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(p=>{throw console.error("Error adding column",o,p),p}),u.index&&await this.db.schema.createIndex(`${r}_${o}_index`).on(r).column(o).execute().catch(p=>{})),(c==null?void 0:c.columns.find(p=>p.name===o))||await this.db.schema.alterTable(s).addColumn(o,"varchar",p=>{let l=p;return u.primary&&(l=l.primaryKey().references(`${r}.${o}`)),l}).execute();}}}async rawFindById(e,t,r){if(!this.schema)throw new Error("Schema not initialized");let n=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"));n=k(this.schema,e,n,r);let a=await n.executeTakeFirst();if(a)return this.convertToMaterializedLiveType(a)}async findOne(e,t$1,r){let n=await this.rawFindById(e.name,t$1,r==null?void 0:r.include);if(n)return t(n)}async rawFind(e,t,r){if(!this.schema)throw new Error("Schema not initialized");let n=this.db.selectFrom(e).selectAll(e).select(o=>jsonObjectFrom(o.selectFrom(`${e}_meta`).selectAll(`${e}_meta`).whereRef(`${e}_meta.id`,"=",`${e}.id`)).as("_meta"));n=se(this.schema,e,n,t),n=k(this.schema,e,n,r);let a=await n.execute(),s=Object.fromEntries(a.map(o=>{let{id:y,...f}=o;return [y,f]}));return Object.keys(s).length===0?{}:Object.entries(s).reduce((o,[y,f])=>(o[y]=this.convertToMaterializedLiveType(f),o),{})}async find(e,t$1){let r=await this.rawFind(e.name,t$1==null?void 0:t$1.where,t$1==null?void 0:t$1.include);return Object.fromEntries(Object.entries(r).map(([n,a])=>[n,t(a)]))}async rawInsert(e,t,r){var s;let n={},a={};for(let[c,o]of Object.entries(r.value)){let y=(s=o._meta)==null?void 0:s.timestamp;y&&(n[c]=o.value,a[c]=y);}return await this.db.insertInto(e).values({...n,id:t}).execute().then(()=>{this.db.insertInto(`${e}_meta`).values({...a,id:t}).execute();}),r}async rawUpdate(e,t,r){var s;let n={},a={};for(let[c,o]of Object.entries(r.value)){let y=(s=o._meta)==null?void 0:s.timestamp;y&&(n[c]=o.value,a[c]=y);}return await Promise.all([this.db.updateTable(e).set(n).where("id","=",t).execute(),this.db.insertInto(`${e}_meta`).values({...a,id:t}).onConflict(c=>c.column("id").doUpdateSet(a)).execute()]),r}async transaction(e){if(!this.schema)throw new Error("Schema not initialized");if(this.db.isTransaction){let r=Math.random().toString(36).substring(2,15),n=await this.db.savepoint(r).execute();try{return await e({trx:this,commit:()=>n.releaseSavepoint(r).execute().then(()=>{}),rollback:()=>n.rollbackToSavepoint(r).execute().then(()=>{})}).then(a=>n.isCommitted||n.isRolledBack?a:n.releaseSavepoint(r).execute().then(()=>a))}catch(a){throw await n.rollbackToSavepoint(r).execute().catch(()=>{}),a}}let t=await this.db.startTransaction().execute();try{return await e({trx:new i(t,this.schema),commit:()=>t.commit().execute(),rollback:()=>t.rollback().execute()}).then(r=>t.isCommitted||t.isRolledBack?r:t.commit().execute().then(()=>r))}catch(r){throw await t.rollback().execute(),r}}convertToMaterializedLiveType(e){return {value:Object.entries(e).reduce((t,[r,n])=>{var a,s,c;return r==="_meta"||(r==="id"?t[r]={value:n}:Array.isArray(n)?t[r]={value:n.map(o=>this.convertToMaterializedLiveType(o)),_meta:{timestamp:(a=e==null?void 0:e._meta)==null?void 0:a[r]}}:typeof n=="object"&&n!==null&&!(n instanceof Date)?t[r]={...this.convertToMaterializedLiveType(n),_meta:{timestamp:(s=e==null?void 0:e._meta)==null?void 0:s[r]}}:t[r]={value:n,_meta:{timestamp:(c=e==null?void 0:e._meta)==null?void 0:c[r]}}),t},{})}}isKyselyLike(e){if(e instanceof Kysely)return true;if(!e||typeof e!="object")return false;let t=e,r=typeof t.selectFrom=="function",n=typeof t.startTransaction=="function",a=typeof t.savepoint=="function",s=typeof t.isTransaction=="boolean"||typeof t.isTransaction=="function";return r&&n||a&&s}};var q=class i{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 i(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,n)=>a=>n({req:a,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&&(e.req.procedure==="INSERT"||e.req.procedure==="UPDATE")&&this.mutationSubscriptions.forEach(r=>{r({id:e.req.context.messageId,type:"MUTATE",resource:e.req.resourceName,payload:t.acceptedValues??{},resourceId:e.req.resourceId,procedure:e.req.procedure});}),t}use(e){return this.middlewares.add(e),this}context(e){return this.contextProvider=e,this}},kt=q.create;
2
- export{_ as Route,V as RouteFactory,D as Router,F as SQLStorage,q as Server,$ as Storage,Tt as expressAdapter,Et as routeFactory,Mt as router,kt as server};
1
+ import {a,b,t}from'./chunk-XNKBHWSC.js';import v from'node:crypto';import Ue,{parse}from'qs';import {z as z$1}from'zod';import'js-xxhash';import {Kysely,PostgresDialect}from'kysely';import {jsonObjectFrom,jsonArrayFrom}from'kysely/helpers/postgres';var D=a(P=>{Object.defineProperty(P,"__esModule",{value:true});P.parse=ve;P.serialize=Le;var Ie=/^[\u0021-\u003A\u003C\u003E-\u007E]+$/,Ee=/^[\u0021-\u003A\u003C-\u007E]*$/,Ae=/^([.]?[a-z0-9]([a-z0-9-]{0,61}[a-z0-9])?)([.][a-z0-9]([a-z0-9-]{0,61}[a-z0-9])?)*$/i,Oe=/^[\u0020-\u003A\u003D-\u007E]*$/,$e=Object.prototype.toString,je=(()=>{let i=function(){};return i.prototype=Object.create(null),i})();function ve(i,t){let e=new je,r=i.length;if(r<2)return e;let n=(t==null?void 0:t.decode)||Ce,a=0;do{let o=i.indexOf("=",a);if(o===-1)break;let c=i.indexOf(";",a),s=c===-1?r:c;if(o>s){a=i.lastIndexOf(";",o-1)+1;continue}let y=te(i,a,o),p=re(i,o,y),l=i.slice(y,p);if(e[l]===void 0){let d=te(i,o+1,s),m=re(i,s,d),u=n(i.slice(d,m));e[l]=u;}a=s+1;}while(a<r);return e}function te(i,t,e){do{let r=i.charCodeAt(t);if(r!==32&&r!==9)return t}while(++t<e);return e}function re(i,t,e){for(;t>e;){let r=i.charCodeAt(--t);if(r!==32&&r!==9)return t+1}return e}function Le(i,t,e){let r=(e==null?void 0:e.encode)||encodeURIComponent;if(!Ie.test(i))throw new TypeError(`argument name is invalid: ${i}`);let n=r(t);if(!Ee.test(n))throw new TypeError(`argument val is invalid: ${t}`);let a=i+"="+n;if(!e)return a;if(e.maxAge!==void 0){if(!Number.isInteger(e.maxAge))throw new TypeError(`option maxAge is invalid: ${e.maxAge}`);a+="; Max-Age="+e.maxAge;}if(e.domain){if(!Ae.test(e.domain))throw new TypeError(`option domain is invalid: ${e.domain}`);a+="; Domain="+e.domain;}if(e.path){if(!Oe.test(e.path))throw new TypeError(`option path is invalid: ${e.path}`);a+="; Path="+e.path;}if(e.expires){if(!Pe(e.expires)||!Number.isFinite(e.expires.valueOf()))throw new TypeError(`option expires is invalid: ${e.expires}`);a+="; Expires="+e.expires.toUTCString();}if(e.httpOnly&&(a+="; HttpOnly"),e.secure&&(a+="; Secure"),e.partitioned&&(a+="; Partitioned"),e.priority)switch(typeof e.priority=="string"?e.priority.toLowerCase():void 0){case "low":a+="; Priority=Low";break;case "medium":a+="; Priority=Medium";break;case "high":a+="; Priority=High";break;default:throw new TypeError(`option priority is invalid: ${e.priority}`)}if(e.sameSite)switch(typeof e.sameSite=="string"?e.sameSite.toLowerCase():e.sameSite){case true:case "strict":a+="; SameSite=Strict";break;case "lax":a+="; SameSite=Lax";break;case "none":a+="; SameSite=None";break;default:throw new TypeError(`option sameSite is invalid: ${e.sameSite}`)}return a}function Ce(i){if(i.indexOf("%")===-1)return i;try{return decodeURIComponent(i)}catch{return i}}function Pe(i){return $e.call(i)==="[object Date]"}});var Y="0123456789ABCDEFGHJKMNPQRSTVWXYZ",E=32;var ge=16,X=10,J=0xffffffffffff;var w;(function(i){i.Base32IncorrectEncoding="B32_ENC_INVALID",i.DecodeTimeInvalidCharacter="DEC_TIME_CHAR",i.DecodeTimeValueMalformed="DEC_TIME_MALFORMED",i.EncodeTimeNegative="ENC_TIME_NEG",i.EncodeTimeSizeExceeded="ENC_TIME_SIZE_EXCEED",i.EncodeTimeValueMalformed="ENC_TIME_MALFORMED",i.PRNGDetectFailure="PRNG_DETECT",i.ULIDInvalid="ULID_INVALID",i.Unexpected="UNEXPECTED",i.UUIDInvalid="UUID_INVALID";})(w||(w={}));var S=class extends Error{constructor(t,e){super(`${e} (${t})`),this.name="ULIDError",this.code=t;}};function Re(i){let t=Math.floor(i()*E);return t===E&&(t=E-1),Y.charAt(t)}function be(i){var r;let t=xe(),e=t&&(t.crypto||t.msCrypto)||(typeof v<"u"?v:null);if(typeof(e==null?void 0:e.getRandomValues)=="function")return ()=>{let n=new Uint8Array(1);return e.getRandomValues(n),n[0]/255};if(typeof(e==null?void 0:e.randomBytes)=="function")return ()=>e.randomBytes(1).readUInt8()/255;if((r=v)!=null&&r.randomBytes)return ()=>v.randomBytes(1).readUInt8()/255;throw new S(w.PRNGDetectFailure,"Failed to find a reliable PRNG")}function xe(){return Me()?self:typeof window<"u"?window:typeof global<"u"?global:typeof globalThis<"u"?globalThis:null}function we(i,t){let e="";for(;i>0;i--)e=Re(t)+e;return e}function Se(i,t=X){if(isNaN(i))throw new S(w.EncodeTimeValueMalformed,`Time must be a number: ${i}`);if(i>J)throw new S(w.EncodeTimeSizeExceeded,`Cannot encode a time larger than ${J}: ${i}`);if(i<0)throw new S(w.EncodeTimeNegative,`Time must be positive: ${i}`);if(Number.isInteger(i)===false)throw new S(w.EncodeTimeValueMalformed,`Time must be an integer: ${i}`);let e,r="";for(let n=t;n>0;n--)e=i%E,r=Y.charAt(e)+r,i=(i-e)/E;return r}function Me(){return typeof WorkerGlobalScope<"u"&&self instanceof WorkerGlobalScope}function ee(i,t){let e=be(),r=Date.now();return Se(r,X)+we(ge,e)}var W=()=>ee().toLowerCase();var L=(...i)=>{let t=i.filter(e=>!!e);return t.length===0?{}:t.length===1?t[0]:{$and:t}};var C=class{storage;queue=new Map;scheduled=false;constructor(t){this.storage=t;}async rawFind({resource:t,commonWhere:e,uniqueWhere:r,...n}){return new Promise((a,o)=>{let c=this.getBatchKey({resource:t,commonWhere:e,...n}),s={resource:t,commonWhere:e,uniqueWhere:r,...n,resolve:a,reject:o};this.queue.has(c)||this.queue.set(c,[]);let y=this.queue.get(c);y&&y.push(s),this.scheduled||(this.scheduled=true,setImmediate(()=>{this.processBatch();}));})}getBatchKey(t){let{resource:e,commonWhere:r,...n}=t;return `${e}:${JSON.stringify(r??{})}:${JSON.stringify(n??{})}`}async processBatch(){this.scheduled=false;let t=Array.from(this.queue.entries());this.queue.clear();for(let[,e]of t)try{await this.executeBatchedRequests(e);}catch(r){e.forEach(n=>{n.reject(r);});}}async executeBatchedRequests(t){var d,m;if(t.length===0)return;let e=t[0],{resource:r,commonWhere:n,include:a,sort:o}=e,c=t.length===1?e.limit:void 0,s=t.map(u=>u.uniqueWhere).filter(u=>u!==void 0),y=n,p=(d=Object.entries(s[0]??{})[0])==null?void 0:d[0];if(s.length>0){let u=s.map(f=>f[p]).filter(f=>f!=null);u.length>0&&(y=L(n,{[p]:{$in:u}}));}let l=await this.storage.rawFind({resource:r,where:y,include:a,sort:o,limit:c});for(let u of t){let f={};if(u.uniqueWhere){let[T,R]=Object.entries(u.uniqueWhere)[0];for(let[M,j]of Object.entries(l))((m=j.value[T])==null?void 0:m.value)===R&&(f[M]=j);}else Object.assign(f,l);u.resolve(f);}}};var ae=b(D(),1);var z=z$1.object({resource:z$1.string(),where:z$1.record(z$1.string(),z$1.any()).optional(),include:z$1.record(z$1.string(),z$1.any()).optional(),lastSyncedAt:z$1.string().optional(),limit:z$1.coerce.number().optional(),sort:z$1.array(z$1.object({key:z$1.string(),direction:z$1.enum(["asc","desc"])})).optional()}),k=z$1.record(z$1.string(),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()})),ze=k.superRefine((i,t)=>{i.id&&t.addIssue({code:z$1.ZodIssueCode.custom,message:"Payload cannot have an id"});}),ne=z$1.object({id:z$1.string().optional(),type:z$1.literal("MUTATE"),resource:z$1.string(),resourceId:z$1.string().optional()}),A=ne.extend({procedure:z$1.string(),payload:z$1.any().optional()}),O=ne.extend({procedure:z$1.enum(["INSERT","UPDATE"]),payload:ze});z$1.union([O,A]);var ie=z.omit({resource:true}),B=A.omit({id:true,type:true,resource:true,procedure:true}),_=O.omit({id:true,type:true,resource:true,procedure:true});z$1.union([_,B]);var oe=i=>async t=>{var e;try{let r=typeof t.headers.getSetCookie=="function"?Object.fromEntries(t.headers):t.headers,n={headers:r,cookies:r.cookie?ae.default.parse(r.cookie):{}},a=new URL(t.url),o=a.pathname.split("/"),c=a.searchParams,s=Ue.parse(c.toString()),y=await((e=i.contextProvider)==null?void 0:e.call(i,{transport:"HTTP",headers:n.headers,cookies:n.cookies,queryParams:s}))??{};if(t.method==="GET"){let p=o[o.length-1],{success:l,data:d,error:m}=ie.safeParse(s);if(!l)return Response.json({message:"Invalid query",code:"INVALID_QUERY",details:m},{status:400});let u=await i.handleQuery({req:{...n,...d,type:"QUERY",resource:p,context:y,queryParams:s}});return !u||!u.data?Response.json({message:"Invalid resource",code:"INVALID_RESOURCE"},{status:400}):Response.json(u.data)}if(t.method==="POST")try{let p=o[o.length-1],l=o[o.length-2],d=t.body?await t.json():{},m;if(p==="insert"||p==="update"){let{success:f,data:T,error:R}=_.safeParse(d);if(!f)return Response.json({message:"Invalid mutation",code:"INVALID_REQUEST",details:R},{status:400});m=T;}else {let{success:f,data:T,error:R}=B.safeParse(d);if(!f)return Response.json({message:"Invalid mutation",code:"INVALID_REQUEST",details:R},{status:400});m=T;}let u=await i.handleMutation({req:{...n,type:"MUTATE",resource:l,input:m.payload,context:y,resourceId:m.resourceId,procedure:p==="insert"||p==="update"?p.toUpperCase():p,queryParams:{}}});return Response.json(u)}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(r){return console.error("Unexpected error:",r),Response.json({message:"Internal server error",code:"INTERNAL_SERVER_ERROR"},{status:500})}};var ue=b(D(),1);var I=z$1.string(),qe=z$1.object({id:I,type:z$1.literal("SUBSCRIBE"),resource:z$1.string()}),We=z.extend({id:I,type:z$1.literal("QUERY")}),se=O.extend({id:I}),De=A.extend({id:I}),ke=z$1.union([De,se]),ce=z$1.union([qe,We,ke]),Be=z$1.object({id:I,type:z$1.literal("REJECT"),resource:z$1.string(),message:z$1.string().optional()}),_e=z$1.object({id:I,type:z$1.literal("REPLY"),data:z$1.any()});z$1.union([Be,_e,se]);z$1.object({resource:z$1.string(),data:z$1.record(z$1.string(),k)});var de=i=>{let t={},e={};return i.subscribeToMutations(r=>{let n=r;!n.resourceId||!n.payload||(console.log("Mutation propagated:",n),Object.entries(e[n.resource]??{}).forEach(([a,o])=>{var c;(c=t[a])==null||c.send(JSON.stringify({...n,id:n.id??W()}));}));}),(r,n)=>{var p;let a=l=>{r.send(JSON.stringify(l));},o=W(),c={headers:n.headers,cookies:typeof n.headers.cookie=="string"?ue.default.parse(n.headers.cookie):{}},s=parse(n.url.split("?")[1]),y=(p=i.contextProvider)==null?void 0:p.call(i,{transport:"WEBSOCKET",headers:c.headers,cookies:c.cookies,queryParams:s});t[o]=r,console.log("Client connected:",o),r.on("message",async l=>{try{console.log("Message received from the client:",l);let d=ce.parse(JSON.parse(l.toString()));if(d.type==="SUBSCRIBE"){let{resource:m}=d;e[m]||(e[m]={}),e[m][o]={};}else if(d.type==="QUERY"){let{resource:m}=d,u=await i.handleQuery({req:{...c,type:"QUERY",resource:m,context:await y??{},queryParams:s}});if(!u||!u.data)throw new Error("Invalid resource");a({id:d.id,type:"REPLY",data:{resource:m,data:Object.fromEntries(Object.entries(u.data??{}).map(([f,T])=>[f,T.value]))}});}else if(d.type==="MUTATE"){let{resource:m}=d;console.log("Received mutation from client:",d);try{let u=await i.handleMutation({req:{...c,type:"MUTATE",resource:m,input:d.payload,context:{messageId:d.id,...await y??{}},resourceId:d.resourceId,procedure:d.procedure,queryParams:s}});d.procedure&&d.procedure!=="INSERT"&&d.procedure!=="UPDATE"&&a({id:d.id,type:"REPLY",data:u});}catch(u){a({id:d.id,type:"REJECT",resource:m,message:u.message}),console.error("Error parsing mutation from the client:",u);}}}catch(d){console.error("Error handling message from the client:",d);}}),r.on("close",()=>{console.log("Connection closed",o),delete t[o];for(let l of Object.values(e))delete l[o];});}};function le(i){let t=`${i.protocol}://${i.hostname}${i.url}`,e=new Headers;return Object.entries(i.headers).forEach(([r,n])=>{n&&e.set(r,Array.isArray(n)?n.join(","):n);}),new Request(t,{method:i.method,headers:e,body:i.body&&i.method!=="GET"?JSON.stringify(i.body):void 0})}var It=(i,t,e)=>{i.ws(`${(e==null?void 0:e.basePath)??""}/ws`,de(t)),i.use(`${(e==null?void 0:e.basePath)??""}/`,(r,n)=>{oe(t)(le(r)).then(o=>o.json().then(c=>n.status(o.status).send(c)));});};var x=(i,t,e=false)=>Object.entries(t).every(([r,n])=>{if(r==="$and")return n.every(o=>x(i,o,e));if(r==="$or")return n.some(o=>x(i,o,e));let a=(n==null?void 0:n.$eq)!==void 0?n==null?void 0:n.$eq:n;if(typeof n=="object"&&n!==null&&(n==null?void 0:n.$eq)===void 0){if(n.$in!==void 0){let o=i[r];return o===void 0?false:e?!n.$in.includes(o):n.$in.includes(o)}if(n.$not!==void 0&&!e)return x(i,{[r]:n.$not},true);if(n.$gt!==void 0){let o=i[r];return typeof o!="number"?false:e?o<=n.$gt:o>n.$gt}if(n.$gte!==void 0){let o=i[r];return typeof o!="number"?false:e?o<n.$gte:o>=n.$gte}if(n.$lt!==void 0){let o=i[r];return typeof o!="number"?false:e?o>=n.$lt:o<n.$lt}if(n.$lte!==void 0){let o=i[r];return typeof o!="number"?false:e?o>n.$lte:o<=n.$lte}return !i[r]||typeof i[r]!="object"?false:x(i[r],n,e)}return e?i[r]!==a:i[r]===a});var V=class i{routes;constructor(t){this.routes=t.routes;}static create(t){return new i(t)}},Pt=i=>V.create({...i}),Fe=i=>({handler:t=>({inputValidator:i??z$1.undefined(),handler:t})}),Q=class i{resourceSchema;middlewares;customMutations;authorization;constructor(t,e,r){this.resourceSchema=t,this.middlewares=new Set,this.customMutations=e??{},this.authorization=r;}use(...t){for(let e of t)this.middlewares.add(e);return this}withMutations(t){return new i(this.resourceSchema,t({mutation:Fe}))}handleQuery=async({req:t,batcher:e})=>await this.wrapInMiddlewares(async r=>{var a,o;let n=(o=(a=this.authorization)==null?void 0:a.read)==null?void 0:o.call(a,{ctx:r.context});if(typeof n=="boolean"&&!n)throw new Error("Not authorized");return {data:await e.rawFind({resource:r.resource,commonWhere:L(r.where,typeof n=="object"?n:void 0),uniqueWhere:r.relationalWhere,include:r.include,limit:r.limit,sort:r.sort})}})(t);handleMutation=async({req:t,db:e})=>await this.wrapInMiddlewares(async r=>{if(!r.procedure)throw new Error("Procedure is required for mutations");let n=this.customMutations[r.procedure];if(n){let a=n.inputValidator.parse(r.input);return r.input=a,n.handler({req:r,db:e})}else {if(r.procedure==="INSERT"||r.procedure==="UPDATE")return this.handleSet({req:r,db:e,operation:r.procedure});throw new Error(`Unknown procedure: ${r.procedure}`)}})(t);handleSet=async({req:t$1,db:e,operation:r})=>{if(!t$1.input)throw new Error("Payload is required");if(!t$1.resourceId)throw new Error("ResourceId is required");let n=await e.rawFindById(t$1.resource,t$1.resourceId);if(r==="INSERT"&&n)throw new Error("Resource already exists");if(r==="UPDATE"&&!n)throw new Error("Resource not found");return e.transaction(async({trx:a})=>{var y,p,l,d,m;let[o,c]=this.resourceSchema.mergeMutation("set",t$1.input,n);if(!c)throw new Error("Mutation rejected");if(r==="INSERT"){let u=await a.rawInsert(t$1.resource,t$1.resourceId,o);if((y=this.authorization)!=null&&y.insert){let f=t(u);f.id=f.id??t$1.resourceId;let T=this.authorization.insert({ctx:t$1.context,value:f});if(!(typeof T=="boolean"?T:x(f,T)))throw new Error("Not authorized")}return {data:u,acceptedValues:c}}if((l=(p=this.authorization)==null?void 0:p.update)!=null&&l.preMutation){let u=t(n);u.id=u.id??t$1.resourceId;let f=this.authorization.update.preMutation({ctx:t$1.context,value:u});if(!(typeof f=="boolean"?f:x(u,f)))throw new Error("Not authorized")}let s=await a.rawUpdate(t$1.resource,t$1.resourceId,o);if((m=(d=this.authorization)==null?void 0:d.update)!=null&&m.postMutation){let u=t(s);u.id=u.id??t$1.resourceId;let f=this.authorization.update.postMutation({ctx:t$1.context,value:u});if(!(typeof f=="boolean"?f:x(u,f)))throw new Error("Not authorized")}return {data:s,acceptedValues:c}})};wrapInMiddlewares(t){return e=>Array.from(this.middlewares.values()).reduceRight((r,n)=>a=>n({req:a,next:r}),t)(e)}},F=class i{middlewares;constructor(t=[]){this.middlewares=t;}collectionRoute(t,e){return new Q(t,void 0,e).use(...this.middlewares)}use(...t){return new i([...this.middlewares,...t])}static create(){return new i}},zt=F.create;var $=class{async insert(t$1,e){let r=new Date().toISOString();return t(await this.rawInsert(t$1.name,e.id,{value:Object.fromEntries(Object.entries(e).map(([n,a])=>[n,{value:a,_meta:{timestamp:r}}]))}))}async update(t$1,e,r){let n=new Date().toISOString(),{id:a,...o}=r;return t(await this.rawUpdate(t$1.name,e,{value:Object.fromEntries(Object.entries(o).map(([c,s])=>[c,{value:s,_meta:{timestamp:n}}]))}))}};function N(i,t,e,r){if(!i)throw new Error("Schema not initialized");let n=i[t];if(!n)throw new Error("Resource not found");let a=r.$or,o=r.$and;return (a?e.or:e.and)(a?r.$or.map(c=>N(i,t,e,c)):o?r.$and.map(c=>N(i,t,e,c)):Object.entries(r).map(([c,s])=>{var y,p;if(n.fields[c])return (s==null?void 0:s.$eq)!==void 0?e(`${t}.${c}`,s.$eq===null?"is":"=",s.$eq):(s==null?void 0:s.$in)!==void 0?e(`${t}.${c}`,"in",s.$in):(s==null?void 0:s.$not)!==void 0?((y=s==null?void 0:s.$not)==null?void 0:y.$in)!==void 0?e(`${t}.${c}`,"not in",s.$not.$in):((p=s==null?void 0:s.$not)==null?void 0:p.$eq)!==void 0?e(`${t}.${c}`,s.$not.$eq===null?"is not":"!=",s.$not.$eq):e(`${t}.${c}`,s.$not===null?"is not":"!=",s.$not):(s==null?void 0:s.$gt)!==void 0?e(`${t}.${c}`,">",s.$gt):(s==null?void 0:s.$gte)!==void 0?e(`${t}.${c}`,">=",s.$gte):(s==null?void 0:s.$lt)!==void 0?e(`${t}.${c}`,"<",s.$lt):(s==null?void 0:s.$lte)!==void 0?e(`${t}.${c}`,"<=",s.$lte):e(`${t}.${c}`,s===null?"is":"=",s);if(n.relations[c]){let l=n.relations[c],d=l.entity.name;return l.type==="many"?e.exists(Z(i,d,e.selectFrom(d).select("id").whereRef(l.foreignColumn,"=",`${t}.id`),s)):N(i,d,e,s)}return null}).filter(Boolean))}function U(i,t,e,r){let n=i[t];if(!n)throw new Error("Resource not found");if(!r)return e;if(r.$and){for(let a of r.$and)e=U(i,t,e,a);return e}else if(r.$or){for(let a of r.$or)e=U(i,t,e,a);return e}for(let[a,o]of Object.entries(r)){if(!n.relations[a])continue;let c=n.relations[a],s=c.entity.name,y=c.type==="one"?"id":c.foreignColumn,p=c.type==="one"?c.relationalColumn:"id";e=e.leftJoin(s,`${s}.${y}`,`${t}.${p}`),o instanceof Object&&!Array.isArray(o)&&o!==null&&(e=U(i,s,e,o));}return e}function Z(i,t,e,r){return !r||Object.keys(r).length===0?e:(e=U(i,t,e,r),e.where(n=>N(i,t,n,r)))}function q(i,t,e,r){if(!r)return e;if(!i)throw new Error("Schema not initialized");let n=i[t];if(!n)throw new Error(`Resource not found: ${t}`);for(let a of Object.keys(r)){if(!n.relations[a])throw new Error(`Relation ${a} not found in resource ${t}`);let o=n.relations[a],c=o.entity.name,s=r[a],y=o.type==="one"?"id":o.foreignColumn,p=o.type==="one"?o.relationalColumn:"id",l=o.type==="one"?jsonObjectFrom:jsonArrayFrom,d=typeof s=="object"&&s!==null;e=e.select(m=>{let u=m.selectFrom(c).selectAll(c).whereRef(`${c}.${y}`,"=",`${t}.${p}`).select(f=>jsonObjectFrom(f.selectFrom(`${c}_meta`).selectAll(`${c}_meta`).whereRef(`${c}_meta.id`,"=",`${c}.id`)).as("_meta"));return d&&(u=q(i,c,u,s)),l(u).as(a)});}return e}var G=class i extends ${db;schema;constructor(t,e){super(),this.isKyselyLike(t)?this.db=t:this.db=new Kysely({dialect:new PostgresDialect({pool:t})}),this.schema=e,this.rawInsert=this.rawInsert.bind(this),this.rawUpdate=this.rawUpdate.bind(this);}async updateSchema(t){this.schema=t;let e=await this.db.introspection.getTables();for(let[r,n]of Object.entries(t)){let a=e.find(s=>s.name===r);a||await this.db.schema.createTable(r).ifNotExists().execute();let o=`${r}_meta`,c=e.find(s=>s.name===o);c||await this.db.schema.createTable(o).ifNotExists().execute();for(let[s,y]of Object.entries(n.fields)){let p=a==null?void 0:a.columns.find(m=>m.name===s),l=y.getStorageFieldType();p?p.dataType!==l.type&&console.error("Column type mismatch:",s,"expected to have type:",l.type,"but has type:",p.dataType):(await this.db.schema.alterTable(r).addColumn(s,l.type,m=>{let u=m;return l.unique&&(u=u.unique()),l.nullable||(u=u.notNull()),l.references&&(u=u.references(l.references)),l.primary&&(u=u.primaryKey()),l.default!==void 0&&(u=u.defaultTo(l.default)),u}).execute().catch(m=>{throw console.error("Error adding column",s,m),m}),l.index&&await this.db.schema.createIndex(`${r}_${s}_index`).on(r).column(s).execute().catch(()=>{})),(c==null?void 0:c.columns.find(m=>m.name===s))||await this.db.schema.alterTable(o).addColumn(s,"varchar",m=>{let u=m;return l.primary&&(u=u.primaryKey().references(`${r}.${s}`)),u}).execute();}}}async rawFindById(t,e,r){if(!this.schema)throw new Error("Schema not initialized");let n=await this.db.selectFrom(t).where("id","=",e).selectAll(t).select(o=>jsonObjectFrom(o.selectFrom(`${t}_meta`).selectAll(`${t}_meta`).whereRef(`${t}_meta.id`,"=",`${t}.id`)).as("_meta"));n=q(this.schema,t,n,r);let a=await n.executeTakeFirst();if(a)return this.convertToMaterializedLiveType(a)}async findOne(t$1,e,r){let n=await this.rawFindById(t$1.name,e,r==null?void 0:r.include);if(n)return t(n)}async rawFind(t){if(!this.schema)throw new Error("Schema not initialized");let{resource:e,where:r,include:n,limit:a,sort:o}=t,c=this.db.selectFrom(e).selectAll(e).select(l=>jsonObjectFrom(l.selectFrom(`${e}_meta`).selectAll(`${e}_meta`).whereRef(`${e}_meta.id`,"=",`${e}.id`)).as("_meta"));c=Z(this.schema,e,c,r),c=q(this.schema,e,c,n),a!==void 0&&(c=c.limit(a)),o!==void 0&&o.forEach(l=>{c=c.orderBy(l.key,l.direction);});let s=await c.execute(),y=Object.fromEntries(s.map(l=>{let{id:d}=l;return [d,l]}));return Object.keys(y).length===0?{}:Object.entries(y).reduce((l,[d,m])=>(l[d]=this.convertToMaterializedLiveType(m),l),{})}async find(t$1,e){let r=await this.rawFind({resource:t$1.name,where:e==null?void 0:e.where,include:e==null?void 0:e.include,limit:e==null?void 0:e.limit,sort:e==null?void 0:e.sort});return Object.fromEntries(Object.entries(r).map(([n,a])=>[n,t(a)]))}async rawInsert(t,e,r){var o;let n={},a={};for(let[c,s]of Object.entries(r.value)){let y=(o=s._meta)==null?void 0:o.timestamp;y&&(n[c]=s.value,a[c]=y);}return await this.db.insertInto(t).values({...n,id:e}).execute().then(()=>{this.db.insertInto(`${t}_meta`).values({...a,id:e}).execute();}),r}async rawUpdate(t,e,r){var o;let n={},a={};for(let[c,s]of Object.entries(r.value)){let y=(o=s._meta)==null?void 0:o.timestamp;y&&(n[c]=s.value,a[c]=y);}return await Promise.all([this.db.updateTable(t).set(n).where("id","=",e).execute(),this.db.insertInto(`${t}_meta`).values({...a,id:e}).onConflict(c=>c.column("id").doUpdateSet(a)).execute()]),r}async transaction(t){if(!this.schema)throw new Error("Schema not initialized");if(this.db.isTransaction){let r=Math.random().toString(36).substring(2,15),n=await this.db.savepoint(r).execute();try{return await t({trx:this,commit:()=>n.releaseSavepoint(r).execute().then(()=>{}),rollback:()=>n.rollbackToSavepoint(r).execute().then(()=>{})}).then(a=>n.isCommitted||n.isRolledBack?a:n.releaseSavepoint(r).execute().then(()=>a))}catch(a){throw await n.rollbackToSavepoint(r).execute().catch(()=>{}),a}}let e=await this.db.startTransaction().execute();try{return await t({trx:new i(e,this.schema),commit:()=>e.commit().execute(),rollback:()=>e.rollback().execute()}).then(r=>e.isCommitted||e.isRolledBack?r:e.commit().execute().then(()=>r))}catch(r){throw await e.rollback().execute(),r}}convertToMaterializedLiveType(t){return {value:Object.entries(t).reduce((e,[r,n])=>{var a,o,c;return r==="_meta"||(r==="id"?e[r]={value:n}:Array.isArray(n)?e[r]={value:n.map(s=>this.convertToMaterializedLiveType(s)),_meta:{timestamp:(a=t==null?void 0:t._meta)==null?void 0:a[r]}}:typeof n=="object"&&n!==null&&!(n instanceof Date)?e[r]={...this.convertToMaterializedLiveType(n),_meta:{timestamp:(o=t==null?void 0:t._meta)==null?void 0:o[r]}}:e[r]={value:n,_meta:{timestamp:(c=t==null?void 0:t._meta)==null?void 0:c[r]}}),e},{})}}isKyselyLike(t){if(t instanceof Kysely)return true;if(!t||typeof t!="object")return false;let e=t,r=typeof e.selectFrom=="function",n=typeof e.startTransaction=="function",a=typeof e.savepoint=="function",o=typeof e.isTransaction=="boolean"||typeof e.isTransaction=="function";return r&&n||a&&o}};var H=class i{router;storage;schema;middlewares=new Set;contextProvider;mutationSubscriptions=new Set;constructor(t){var e;this.router=t.router,this.storage=t.storage,this.schema=t.schema,(e=t.middlewares)==null||e.forEach(r=>{this.middlewares.add(r);}),this.storage.updateSchema(this.schema),this.contextProvider=t.contextProvider;}static create(t){return new i(t)}subscribeToMutations(t){return this.mutationSubscriptions.add(t),()=>{this.mutationSubscriptions.delete(t);}}handleQuery(t){let e=new C(this.storage);return this.wrapInMiddlewares(async r=>{var y;let n=fe(r,this.schema,{stepId:"query",collectionName:r.resource,included:Object.keys(r.include??{})}),a={headers:r.headers,cookies:r.cookies,queryParams:r.queryParams,context:r.context},o={};for(let p=0;p<n.length;p++){let l=n[p],d=this.router.routes[l.resource];if(!d)throw new Error("Invalid resource");let m=l.getWhere&&l.referenceGetter?l.referenceGetter(o).map(l.getWhere):[void 0],u=(y=o[l.prevStepId??""])==null?void 0:y.flatMap(R=>{var M;return Object.keys(((M=R==null?void 0:R.result)==null?void 0:M.data)??{})}),T=(await Promise.allSettled(m.map(async(R,M)=>{let j=u==null?void 0:u[M],he=await d.handleQuery({req:{type:"QUERY",...l,...a,where:l.where,relationalWhere:R},batcher:e});return {includedBy:j,result:he}}))).flatMap(R=>R.status==="fulfilled"?[R.value]:[]);o[l.stepId]=T;}let c=Object.fromEntries(Object.entries(o).flatMap(([p,l],d)=>l.flatMap(m=>Object.entries(m.result.data).map(([u,f])=>[`${p}.${u}`,{data:f,includedBy:p!=="query"&&m.includedBy?`${p.split(".").slice(0,-1).join(".")}.${m.includedBy}`:void 0,path:p.split(".").slice(-1)[0],isMany:n[d].isMany,collectionName:n[d].collectionName,included:n[d].included}]))));return Object.keys(c).reduceRight((p,l)=>{var u,f;let d=c[l],m=d.path;if(m==="query"&&(p.data[l]=d.data),d.included.length)for(let T of d.included)d.data.value[T]??=((f=(u=this.schema[d.collectionName])==null?void 0:u.relations[T])==null?void 0:f.type)==="many"?{value:[]}:{value:null};if(d.includedBy){let T=c[d.includedBy];if(!T)return p;d.isMany?(T.data.value[m]??={value:[]},T.data.value[m].value.push(d.data)):T.data.value[m]=d.data;}return p},{data:{}})})(t.req)}async handleMutation(t){let e=await this.wrapInMiddlewares(async r=>{let n=this.router.routes[r.resource];if(!n)throw new Error("Invalid resource");return n.handleMutation({req:r,db:this.storage})})(t.req);if(e&&t.req.type==="MUTATE"&&e.acceptedValues&&(t.req.procedure==="INSERT"||t.req.procedure==="UPDATE")&&t.req.resourceId){let n=e.acceptedValues??{},a=t.req,o=a.resourceId;Object.keys(n).length&&o&&this.mutationSubscriptions.forEach(c=>{c({id:t.req.context.messageId,type:"MUTATE",resource:a.resource,payload:n,resourceId:o,procedure:a.procedure});});}return e}use(t){return this.middlewares.add(t),this}context(t){return this.contextProvider=t,this}wrapInMiddlewares(t){return e=>Array.from(this.middlewares.values()).reduceRight((r,n)=>a=>n({req:a,next:r}),t)(e)}},Jt=H.create;function fe(i,t,e){let{include:r,...n}=i,{stepId:a}=e,o=[{...n,...e}];if(r&&typeof r=="object"&&Object.keys(r).length>0){let c=t[n.resource];if(!c)throw new Error(`Resource ${n.resource} not found`);o.push(...Object.entries(r).flatMap(([s,y])=>{let p=c.relations[s];if(!p)throw new Error(`Relation ${s} not found for resource ${n.resource}`);let l=p.entity.name;return fe({...n,resource:l,include:y},t,{getWhere:p.type==="one"?d=>({id:d}):d=>({[p.foreignColumn]:d}),referenceGetter:d=>d[a].flatMap(m=>m.result.data?p.type==="one"?Object.values(m.result.data).map(u=>{var f,T;return (T=(f=u.value)==null?void 0:f[p.relationalColumn])==null?void 0:T.value}):Object.keys(m.result.data):[]),stepId:`${a}.${s}`,prevStepId:a,isMany:p.type==="many",collectionName:l,included:typeof y=="object"?Object.keys(y):[]})}));}return o}
2
+ export{Q as Route,F as RouteFactory,V as Router,G as SQLStorage,H as Server,$ as Storage,It as expressAdapter,zt as routeFactory,Pt as router,Jt as server};
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@live-state/sync",
3
- "version": "0.0.4-beta.5",
3
+ "version": "0.0.4-beta.7",
4
4
  "private": false,
5
5
  "publishConfig": {
6
6
  "access": "public"