@live-state/sync 0.0.4-beta.1 → 0.0.4-beta.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/client.d.ts CHANGED
@@ -1,3 +1,5 @@
1
- export { f as Client, e as ClientEvents, C as ClientOptions, d as ConnectionStateChangeEvent, M as MessageReceivedEvent, c as SubscriptionProvider, g as createClient, u as useLiveQuery } from './index-C1FzxdnB.js';
1
+ export { f as Client, e as ClientEvents, C as ClientOptions, d as ConnectionStateChangeEvent, M as MessageReceivedEvent, c as SubscriptionProvider, g as createClient, u as useLiveQuery } from './index-C2ffIqoq.js';
2
2
  import 'react/jsx-runtime';
3
3
  import 'zod';
4
+ import 'zod/v3';
5
+ import 'zod/v4/core';
@@ -1,6 +1,8 @@
1
- import { A as AnyRouter, C as ClientOptions, L as LiveObjectAny, W as WhereClause, I as IncludeClause, S as Simplify, a as InferLiveObject, b as LiveObjectMutationInput } from './index-C1FzxdnB.js';
1
+ import { A as AnyRouter, C as ClientOptions, L as LiveObjectAny, W as WhereClause, I as IncludeClause, S as Simplify, a as InferLiveObject, b as LiveObjectMutationInput } from './index-C2ffIqoq.js';
2
2
  import 'react/jsx-runtime';
3
3
  import 'zod';
4
+ import 'zod/v3';
5
+ import 'zod/v4/core';
4
6
 
5
7
  type GetOptions<T extends LiveObjectAny> = {
6
8
  headers?: Record<string, string>;
@@ -1,5 +1,7 @@
1
1
  import * as react_jsx_runtime from 'react/jsx-runtime';
2
- import { ZodTypeAny, z } from 'zod';
2
+ import { z } from 'zod';
3
+ import * as z3 from 'zod/v3';
4
+ import * as z4 from 'zod/v4/core';
3
5
 
4
6
  type Promisify<T> = T extends Promise<any> ? T : Promise<T>;
5
7
  type Awaitable<T> = T | Promise<T>;
@@ -229,12 +231,12 @@ type RequestHandler<TInput, TResult, TSchema extends Schema<any> = Schema<any>>
229
231
  db: Storage;
230
232
  schema: TSchema;
231
233
  }) => Promise<TResult>;
232
- type Mutation<TInputValidator extends ZodTypeAny, // TODO use StandardSchema instead
234
+ type Mutation<TInputValidator extends z3.ZodTypeAny | z4.$ZodType, // TODO use StandardSchema instead
233
235
  THandler extends RequestHandler<z.infer<TInputValidator>, any, any>> = {
234
236
  inputValidator: TInputValidator;
235
237
  handler: THandler;
236
238
  };
237
- declare const mutationCreator: <TInputValidator extends ZodTypeAny>(validator?: TInputValidator) => {
239
+ declare const mutationCreator: <TInputValidator extends z3.ZodTypeAny | z4.$ZodType>(validator?: TInputValidator) => {
238
240
  handler: <THandler extends RequestHandler<z.infer<TInputValidator>, any, any>>(handler: THandler) => Mutation<TInputValidator, THandler>;
239
241
  };
240
242
  declare class Route<TResourceSchema extends LiveObjectAny, TMiddleware extends Middleware<any>, TCustomMutations extends Record<string, Mutation<any, RequestHandler<any, any>>>> {
package/dist/server.cjs CHANGED
@@ -1,2 +1,2 @@
1
- 'use strict';var _e=require('qs'),zod=require('zod'),$=require('crypto'),kysely=require('kysely'),postgres=require('kysely/helpers/postgres');function _interopDefault(e){return e&&e.__esModule?e:{default:e}}var _e__default=/*#__PURE__*/_interopDefault(_e);var $__default=/*#__PURE__*/_interopDefault($);var fe=Object.create;var Y=Object.defineProperty;var he=Object.getOwnPropertyDescriptor;var ge=Object.getOwnPropertyNames;var Re=Object.getPrototypeOf,xe=Object.prototype.hasOwnProperty;var be=(n,e)=>()=>(e||n((e={exports:{}}).exports,e),e.exports);var ve=(n,e,t,r)=>{if(e&&typeof e=="object"||typeof e=="function")for(let a of ge(e))!xe.call(n,a)&&a!==t&&Y(n,a,{get:()=>e[a],enumerable:!(r=he(e,a))||r.enumerable});return n};var J=(n,e,t)=>(t=n!=null?fe(Re(n)):{},ve(Y(t,"default",{value:n,enumerable:true}),n));var _=be(O=>{Object.defineProperty(O,"__esModule",{value:true});O.parse=Ee;O.serialize=Oe;var we=/^[\u0021-\u003A\u003C\u003E-\u007E]+$/,Me=/^[\u0021-\u003A\u003C-\u007E]*$/,Se=/^([.]?[a-z0-9]([a-z0-9-]{0,61}[a-z0-9])?)([.][a-z0-9]([a-z0-9-]{0,61}[a-z0-9])?)*$/i,Ie=/^[\u0020-\u003A\u003D-\u007E]*$/,Le=Object.prototype.toString,Ae=(()=>{let n=function(){};return n.prototype=Object.create(null),n})();function Ee(n,e){let t=new Ae,r=n.length;if(r<2)return t;let a=(e==null?void 0:e.decode)||je,o=0;do{let s=n.indexOf("=",o);if(s===-1)break;let c=n.indexOf(";",o),i=c===-1?r:c;if(s>i){o=n.lastIndexOf(";",s-1)+1;continue}let T=X(n,o,s),m=ee(n,s,T),u=n.slice(T,m);if(t[u]===void 0){let p=X(n,s+1,i),y=ee(n,i,p),l=a(n.slice(p,y));t[u]=l;}o=i+1;}while(o<r);return t}function X(n,e,t){do{let r=n.charCodeAt(e);if(r!==32&&r!==9)return e}while(++e<t);return t}function ee(n,e,t){for(;e>t;){let r=n.charCodeAt(--e);if(r!==32&&r!==9)return e+1}return t}function Oe(n,e,t){let r=(t==null?void 0:t.encode)||encodeURIComponent;if(!we.test(n))throw new TypeError(`argument name is invalid: ${n}`);let a=r(e);if(!Me.test(a))throw new TypeError(`argument val is invalid: ${e}`);let o=n+"="+a;if(!t)return o;if(t.maxAge!==void 0){if(!Number.isInteger(t.maxAge))throw new TypeError(`option maxAge is invalid: ${t.maxAge}`);o+="; Max-Age="+t.maxAge;}if(t.domain){if(!Se.test(t.domain))throw new TypeError(`option domain is invalid: ${t.domain}`);o+="; Domain="+t.domain;}if(t.path){if(!Ie.test(t.path))throw new TypeError(`option path is invalid: ${t.path}`);o+="; Path="+t.path;}if(t.expires){if(!$e(t.expires)||!Number.isFinite(t.expires.valueOf()))throw new TypeError(`option expires is invalid: ${t.expires}`);o+="; Expires="+t.expires.toUTCString();}if(t.httpOnly&&(o+="; HttpOnly"),t.secure&&(o+="; Secure"),t.partitioned&&(o+="; Partitioned"),t.priority)switch(typeof t.priority=="string"?t.priority.toLowerCase():void 0){case "low":o+="; Priority=Low";break;case "medium":o+="; Priority=Medium";break;case "high":o+="; 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":o+="; SameSite=Strict";break;case "lax":o+="; SameSite=Lax";break;case "none":o+="; SameSite=None";break;default:throw new TypeError(`option sameSite is invalid: ${t.sameSite}`)}return o}function je(n){if(n.indexOf("%")===-1)return n;try{return decodeURIComponent(n)}catch{return n}}function $e(n){return Le.call(n)==="[object Date]"}});var re=J(_());var j=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()}),P=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((n,e)=>{n.id&&e.addIssue({code:zod.z.ZodIssueCode.custom,message:"Payload cannot have an id"});}),te=zod.z.object({id:zod.z.string().optional(),type:zod.z.literal("MUTATE"),resource:zod.z.string()}),S=te.extend({procedure:zod.z.string(),payload:zod.z.any().optional()}),I=te.extend({resourceId:zod.z.string(),payload:P});zod.z.union([S,I]);var ne=j.omit({resource:true}),V=S.omit({id:true,type:true,resource:true,procedure:true}),F=I.omit({id:true,type:true,resource:true});zod.z.union([F,V]);var ae=n=>async e=>{var t;try{let r=typeof e.headers.getSetCookie=="function"?Object.fromEntries(e.headers):e.headers,a={headers:r,cookies:r.cookie?re.default.parse(r.cookie):{}},o=new URL(e.url),s=o.pathname.split("/"),c=o.searchParams,i=_e__default.default.parse(c.toString()),T=await((t=n.contextProvider)==null?void 0:t.call(n,{transport:"HTTP",headers:a.headers,cookies:a.cookies,query:i}))??{};if(e.method==="GET"){let m=s[s.length-1],{success:u,data:p,error:y}=ne.safeParse(i);if(!u)return Response.json({message:"Invalid query",code:"INVALID_QUERY",details:y},{status:400});let l=await n.handleRequest({req:{...a,type:"QUERY",resourceName:m,context:T,where:p.where,include:p.include,query:i}});return !l||!l.data?Response.json({message:"Invalid resource",code:"INVALID_RESOURCE"},{status:400}):Response.json(l.data)}if(e.method==="POST")try{let m=s[s.length-1],u=s[s.length-2],p=e.body?await e.json():{},y;if(m==="set"){let{success:w,data:M,error:C}=F.safeParse(p);if(!w)return Response.json({message:"Invalid mutation",code:"INVALID_REQUEST",details:C},{status:400});y=M;}else {let{success:w,data:M,error:C}=V.safeParse(p);if(!w)return Response.json({message:"Invalid mutation",code:"INVALID_REQUEST",details:C},{status:400});y=M;}let l=await n.handleRequest({req:{...a,type:"MUTATE",resourceName:u,input:y.payload,context:T,resourceId:y.resourceId,procedure:m!=="set"?m:void 0,query:{}}});return Response.json(l)}catch(m){return console.error("Error parsing mutation from the client:",m),Response.json({message:"Internal server error",code:"INTERNAL_SERVER_ERROR"},{status:500})}return Response.json({message:"Not found",code:"NOT_FOUND"},{status:404})}catch(r){return console.error("Unexpected error:",r),Response.json({message:"Internal server error",code:"INTERNAL_SERVER_ERROR"},{status:500})}};var de=J(_());var b=zod.z.string(),Pe=zod.z.object({id:b,type:zod.z.literal("SUBSCRIBE"),resource:zod.z.string()}),Ve=j.extend({id:b,type:zod.z.literal("QUERY")}),ie=I.extend({id:b}),Fe=S.extend({id:b}),Ne=zod.z.union([Fe,ie]),oe=zod.z.union([Pe,Ve,Ne]),De=zod.z.object({id:b,type:zod.z.literal("REJECT"),resource:zod.z.string(),message:zod.z.string().optional()}),Ke=zod.z.object({id:b,type:zod.z.literal("REPLY"),data:zod.z.any()});zod.z.union([De,Ke,ie]);zod.z.object({resource:zod.z.string(),data:zod.z.record(zod.z.string(),P)});var ce="0123456789ABCDEFGHJKMNPQRSTVWXYZ",L=32;var Ue=16,ue=10,se=0xffffffffffff;var g;(function(n){n.Base32IncorrectEncoding="B32_ENC_INVALID",n.DecodeTimeInvalidCharacter="DEC_TIME_CHAR",n.DecodeTimeValueMalformed="DEC_TIME_MALFORMED",n.EncodeTimeNegative="ENC_TIME_NEG",n.EncodeTimeSizeExceeded="ENC_TIME_SIZE_EXCEED",n.EncodeTimeValueMalformed="ENC_TIME_MALFORMED",n.PRNGDetectFailure="PRNG_DETECT",n.ULIDInvalid="ULID_INVALID",n.Unexpected="UNEXPECTED",n.UUIDInvalid="UUID_INVALID";})(g||(g={}));var R=class extends Error{constructor(e,t){super(`${t} (${e})`),this.name="ULIDError",this.code=e;}};function ze(n){let e=Math.floor(n()*L);return e===L&&(e=L-1),ce.charAt(e)}function ke(n){var r;let e=qe(),t=e&&(e.crypto||e.msCrypto)||(typeof $__default.default<"u"?$__default.default:null);if(typeof(t==null?void 0:t.getRandomValues)=="function")return ()=>{let a=new Uint8Array(1);return t.getRandomValues(a),a[0]/255};if(typeof(t==null?void 0:t.randomBytes)=="function")return ()=>t.randomBytes(1).readUInt8()/255;if((r=$__default.default)!=null&&r.randomBytes)return ()=>$__default.default.randomBytes(1).readUInt8()/255;throw new R(g.PRNGDetectFailure,"Failed to find a reliable PRNG")}function qe(){return He()?self:typeof window<"u"?window:typeof global<"u"?global:typeof globalThis<"u"?globalThis:null}function Ze(n,e){let t="";for(;n>0;n--)t=ze(e)+t;return t}function We(n,e=ue){if(isNaN(n))throw new R(g.EncodeTimeValueMalformed,`Time must be a number: ${n}`);if(n>se)throw new R(g.EncodeTimeSizeExceeded,`Cannot encode a time larger than ${se}: ${n}`);if(n<0)throw new R(g.EncodeTimeNegative,`Time must be positive: ${n}`);if(Number.isInteger(n)===false)throw new R(g.EncodeTimeValueMalformed,`Time must be an integer: ${n}`);let t,r="";for(let a=e;a>0;a--)t=n%L,r=ce.charAt(t)+r,n=(n-t)/L;return r}function He(){return typeof WorkerGlobalScope<"u"&&self instanceof WorkerGlobalScope}function le(n,e){let t=ke(),r=Date.now();return We(r,ue)+Ze(Ue,t)}var N=()=>le().toLowerCase();var pe=n=>{let e={},t={};return n.subscribeToMutations(r=>{let a=r;!a.resourceId||!a.payload||(console.log("Mutation propagated:",a),Object.entries(t[a.resource]??{}).forEach(([o,s])=>{var c;(c=e[o])==null||c.send(JSON.stringify({...a,id:a.id??N()}));}));}),(r,a)=>{var m;let o=u=>{r.send(JSON.stringify(u));},s=N(),c={headers:a.headers,cookies:typeof a.headers.cookie=="string"?de.default.parse(a.headers.cookie):{}},i=_e.parse(a.url.split("?")[1]),T=(m=n.contextProvider)==null?void 0:m.call(n,{transport:"WEBSOCKET",headers:c.headers,cookies:c.cookies,query:i});e[s]=r,console.log("Client connected:",s),r.on("message",async u=>{try{console.log("Message received from the client:",u);let p=oe.parse(JSON.parse(u.toString()));if(p.type==="SUBSCRIBE"){let{resource:y}=p;t[y]||(t[y]={}),t[y][s]={};}else if(p.type==="QUERY"){let{resource:y}=p,l=await n.handleRequest({req:{...c,type:"QUERY",resourceName:y,context:await T??{},query:i}});if(!l||!l.data)throw new Error("Invalid resource");o({id:p.id,type:"REPLY",data:{resource:y,data:Object.fromEntries(Object.entries(l.data??{}).map(([w,M])=>[w,M.value]))}});}else if(p.type==="MUTATE"){let{resource:y}=p;console.log("Received mutation from client:",p);try{let l=await n.handleRequest({req:{...c,type:"MUTATE",resourceName:y,input:p.payload,context:{messageId:p.id,...await T??{}},resourceId:p.resourceId,procedure:p.procedure,query:i}});p.procedure&&o({id:p.id,type:"REPLY",data:l});}catch(l){o({id:p.id,type:"REJECT",resource:y,message:l.message}),console.error("Error parsing mutation from the client:",l);}}}catch(p){console.error("Error handling message from the client:",p);}}),r.on("close",()=>{console.log("Connection closed",s),delete e[s];for(let u of Object.values(t))delete u[s];});}};function ye(n){let e=`${n.protocol}://${n.hostname}${n.url}`,t=new Headers;return Object.entries(n.headers).forEach(([r,a])=>{a&&t.set(r,Array.isArray(a)?a.join(","):a);}),new Request(e,{method:n.method,headers:t,body:n.body&&n.method!=="GET"?JSON.stringify(n.body):void 0})}var Et=(n,e,t)=>{n.ws(`${(t==null?void 0:t.basePath)??""}/ws`,pe(e)),n.use(`${(t==null?void 0:t.basePath)??""}/`,(r,a)=>{ae(e)(ye(r)).then(s=>s.json().then(c=>a.status(s.status).send(c)));});};var D=class n{routes;constructor(e){this.routes=e.routes;}static create(e){return new n(e)}},$t=n=>D.create({...n}),Qe=n=>({handler:e=>({inputValidator:n??zod.z.undefined(),handler:e})}),K=class n{_resourceSchema;resourceName;middlewares;customMutations;constructor(e,t){this.resourceName=e,this.middlewares=new Set,this.customMutations=t??{};}handleFind=async({req:e,db:t})=>({data:await t.rawFind(e.resourceName,e.where,e.include),acceptedValues:null});handleSet=async({req:e,db:t,schema:r})=>{if(!e.input)throw new Error("Payload is required");if(!e.resourceId)throw new Error("ResourceId is required");let a=await t.rawFindById(e.resourceName,e.resourceId),[o,s]=r[this.resourceName].mergeMutation("set",e.input,a);if(!s)throw new Error("Mutation rejected");return {data:await t.rawUpsert(e.resourceName,e.resourceId,o),acceptedValues:s}};async handleRequest(e){let t=r=>(()=>{if(r.type==="QUERY")return this.handleFind({req:r,db:e.db,schema:e.schema});if(r.type==="MUTATE")if(r.procedure){if(this.customMutations[r.procedure]){let a=this.customMutations[r.procedure].inputValidator.parse(r.input);return r.input=a,this.customMutations[r.procedure].handler({req:r,db:e.db,schema:e.schema})}}else return this.handleSet({req:r,db:e.db,schema:e.schema});throw new Error("Invalid request")})();return await Array.from(this.middlewares.values()).reduceRight((r,a)=>o=>a({req:o,next:r}),async r=>t(r))(e.req)}use(...e){for(let t of e)this.middlewares.add(t);return this}withMutations(e){return new n(this.resourceName,e({mutation:Qe}))}},U=class n{middlewares;constructor(e=[]){this.middlewares=e;}createBasicRoute(e){return new K(e.name).use(...this.middlewares)}use(...e){return new n([...this.middlewares,...e])}static create(){return new n}},Ct=U.create;var h=n=>{if(n)return Array.isArray(n.value)?n.value.map(e=>h(e)):typeof n.value!="object"||n.value===null||n.value instanceof Date?n.value:Object.fromEntries(Object.entries(n.value).map(([e,t])=>[e,h(t)]))};var E=class{async insert(e,t){let r=new Date().toISOString();return h(await this.rawUpsert(e.name,t.id,{value:Object.fromEntries(Object.entries(t).map(([a,o])=>[a,{value:o,_meta:{timestamp:r}}]))}))}async update(e,t,r){let a=new Date().toISOString(),{id:o,...s}=r;return h(await this.rawUpsert(e.name,t,{value:Object.fromEntries(Object.entries(s).map(([c,i])=>[c,{value:i,_meta:{timestamp:a}}]))}))}};function H(n,e,t,r){var o,s;if(!r)return t;if(!n)throw new Error("Schema not initialized");let a=n[e];if(!a)throw new Error("Resource not found");for(let[c,i]of Object.entries(r))if(a.fields[c])(i==null?void 0:i.$eq)!==void 0?t=t.where(`${e}.${c}`,i.$eq===null?"is":"=",i.$eq):(i==null?void 0:i.$in)!==void 0?t=t.where(`${e}.${c}`,"in",i.$in):(i==null?void 0:i.$not)!==void 0?((o=i==null?void 0:i.$not)==null?void 0:o.$in)!==void 0?t=t.where(`${e}.${c}`,"not in",i.$not.$in):((s=i==null?void 0:i.$not)==null?void 0:s.$eq)!==void 0?t=t.where(`${e}.${c}`,i.$not.$eq===null?"is not":"!=",i.$not.$eq):t=t.where(`${e}.${c}`,i.$not===null?"is not":"!=",i.$not):(i==null?void 0:i.$gt)!==void 0?t=t.where(`${e}.${c}`,">",i.$gt):(i==null?void 0:i.$gte)!==void 0?t=t.where(`${e}.${c}`,">=",i.$gte):(i==null?void 0:i.$lt)!==void 0?t=t.where(`${e}.${c}`,"<",i.$lt):(i==null?void 0:i.$lte)!==void 0?t=t.where(`${e}.${c}`,"<=",i.$lte):t=t.where(`${e}.${c}`,i===null?"is":"=",i);else if(a.relations[c]){let T=a.relations[c],m=T.entity.name,u=T.type==="one"?"id":T.foreignColumn,p=T.type==="one"?T.relationalColumn:"id";t=t.leftJoin(m,`${m}.${u}`,`${e}.${p}`),t=H(n,m,t,i);}return t}function B(n,e,t,r){if(!r)return t;if(!n)throw new Error("Schema not initialized");let a=n[e];if(!a)throw new Error(`Resource not found: ${e}`);for(let o of Object.keys(r)){if(!a.relations[o])throw new Error(`Relation ${o} not found in resource ${e}`);let s=a.relations[o],c=s.entity.name,i=s.type==="one"?"id":s.foreignColumn,T=s.type==="one"?s.relationalColumn:"id",m=s.type==="one"?postgres.jsonObjectFrom:postgres.jsonArrayFrom;t=t.select(u=>m(u.selectFrom(c).selectAll(c).whereRef(`${c}.${i}`,"=",`${e}.${T}`).select(p=>postgres.jsonObjectFrom(p.selectFrom(`${c}_meta`).selectAll(`${c}_meta`).whereRef(`${c}_meta.id`,"=",`${c}.id`)).as("_meta"))).as(o));}return t}var G=class extends E{db;schema;constructor(e){super(),this.db=new kysely.Kysely({dialect:new kysely.PostgresDialect({pool:e})});}async updateSchema(e){this.schema=e;let t=await this.db.introspection.getTables();for(let[r,a]of Object.entries(e)){let o=t.find(i=>i.name===r);o||await this.db.schema.createTable(r).ifNotExists().execute();let s=`${r}_meta`,c=t.find(i=>i.name===s);c||await this.db.schema.createTable(s).ifNotExists().execute();for(let[i,T]of Object.entries(a.fields)){let m=o==null?void 0:o.columns.find(y=>y.name===i),u=T.getStorageFieldType();m?m.dataType!==u.type&&console.error("Column type mismatch:",i,"expected to have type:",u.type,"but has type:",m.dataType):(await this.db.schema.alterTable(r).addColumn(i,u.type,y=>{let l=y;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(y=>{throw console.error("Error adding column",i,y),y}),u.index&&await this.db.schema.createIndex(`${r}_${i}_index`).on(r).column(i).execute().catch(y=>{})),(c==null?void 0:c.columns.find(y=>y.name===i))||await this.db.schema.alterTable(s).addColumn(i,"varchar",y=>{let l=y;return u.primary&&(l=l.primaryKey().references(`${r}.${i}`)),l}).execute();}}}async rawFindById(e,t,r){if(!this.schema)throw new Error("Schema not initialized");let a=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"));a=B(this.schema,e,a,r);let o=await a.executeTakeFirst();if(o)return this.convertToMaterializedLiveType(o)}async findOne(e,t,r){let a=await this.rawFindById(e.name,t,r==null?void 0:r.include);if(a)return h(a)}async rawFind(e,t,r){if(!this.schema)throw new Error("Schema not initialized");let a=this.db.selectFrom(e).selectAll(e).select(i=>postgres.jsonObjectFrom(i.selectFrom(`${e}_meta`).selectAll(`${e}_meta`).whereRef(`${e}_meta.id`,"=",`${e}.id`)).as("_meta"));a=H(this.schema,e,a,t),a=B(this.schema,e,a,r);let o=await a.execute(),s=Object.fromEntries(o.map(i=>{let{id:T,...m}=i;return [T,m]}));return Object.keys(s).length===0?{}:Object.entries(s).reduce((i,[T,m])=>(i[T]=this.convertToMaterializedLiveType(m),i),{})}async find(e,t){let r=await this.rawFind(e.name,t==null?void 0:t.where,t==null?void 0:t.include);return Object.fromEntries(Object.entries(r).map(([a,o])=>[a,h(o)]))}async rawUpsert(e,t,r){return await this.db.transaction().execute(async a=>{var i;let o=!!await a.selectFrom(e).select("id").where("id","=",t).executeTakeFirst(),s={},c={};for(let[T,m]of Object.entries(r.value)){let u=(i=m._meta)==null?void 0:i.timestamp;u&&(s[T]=m.value,c[T]=u);}o?await Promise.all([a.updateTable(e).set(s).where("id","=",t).execute(),a.updateTable(`${e}_meta`).set(c).where("id","=",t).execute()]):await Promise.all([a.insertInto(e).values({...s,id:t}).execute(),a.insertInto(`${e}_meta`).values({...c,id:t}).execute()]);}),r}convertToMaterializedLiveType(e){if(!e._meta)throw new Error("Missing _meta");return {value:Object.entries(e).reduce((t,[r,a])=>{var o,s,c;return r==="_meta"||(r==="id"?t[r]={value:a}:Array.isArray(a)?t[r]={value:a.map(i=>this.convertToMaterializedLiveType(i)),_meta:{timestamp:(o=e==null?void 0:e._meta)==null?void 0:o[r]}}:typeof a=="object"&&a!==null&&!(a instanceof Date)?t[r]={...this.convertToMaterializedLiveType(a),_meta:{timestamp:(s=e==null?void 0:e._meta)==null?void 0:s[r]}}:t[r]={value:a,_meta:{timestamp:(c=e==null?void 0:e._meta)==null?void 0:c[r]}}),t},{})}}};var Q=class n{router;storage;schema;middlewares=new Set;contextProvider;mutationSubscriptions=new Set;constructor(e){var t;this.router=e.router,this.storage=e.storage,this.schema=e.schema,(t=e.middlewares)==null||t.forEach(r=>{this.middlewares.add(r);}),this.storage.updateSchema(this.schema),this.contextProvider=e.contextProvider;}static create(e){return new n(e)}subscribeToMutations(e){return this.mutationSubscriptions.add(e),()=>{this.mutationSubscriptions.delete(e);}}async handleRequest(e){if(!this.router.routes[e.req.resourceName])throw new Error("Invalid resource");let t=await Array.from(this.middlewares.values()).reduceRight((r,a)=>o=>a({req:o,next:r}),async r=>this.router.routes[e.req.resourceName].handleRequest({req:r,db:this.storage,schema:this.schema}))(e.req);return t&&e.req.type==="MUTATE"&&t.acceptedValues&&Object.keys(t.acceptedValues).length>0&&this.mutationSubscriptions.forEach(r=>{r({id:e.req.context.messageId,type:"MUTATE",resource:e.req.resourceName,payload:t.acceptedValues??{},resourceId:e.req.resourceId});}),t}use(e){return this.middlewares.add(e),this}context(e){return this.contextProvider=e,this}},un=Q.create;
2
- exports.Route=K;exports.RouteFactory=U;exports.Router=D;exports.SQLStorage=G;exports.Server=Q;exports.Storage=E;exports.expressAdapter=Et;exports.routeFactory=Ct;exports.router=$t;exports.server=un;
1
+ 'use strict';var _e=require('qs'),zod=require('zod'),$=require('crypto'),kysely=require('kysely'),postgres=require('kysely/helpers/postgres');function _interopDefault(e){return e&&e.__esModule?e:{default:e}}var _e__default=/*#__PURE__*/_interopDefault(_e);var $__default=/*#__PURE__*/_interopDefault($);var fe=Object.create;var Y=Object.defineProperty;var he=Object.getOwnPropertyDescriptor;var ge=Object.getOwnPropertyNames;var Re=Object.getPrototypeOf,xe=Object.prototype.hasOwnProperty;var be=(n,e)=>()=>(e||n((e={exports:{}}).exports,e),e.exports);var ve=(n,e,t,r)=>{if(e&&typeof e=="object"||typeof e=="function")for(let a of ge(e))!xe.call(n,a)&&a!==t&&Y(n,a,{get:()=>e[a],enumerable:!(r=he(e,a))||r.enumerable});return n};var J=(n,e,t)=>(t=n!=null?fe(Re(n)):{},ve(Y(t,"default",{value:n,enumerable:true}),n));var _=be(O=>{Object.defineProperty(O,"__esModule",{value:true});O.parse=Ee;O.serialize=Oe;var we=/^[\u0021-\u003A\u003C\u003E-\u007E]+$/,Me=/^[\u0021-\u003A\u003C-\u007E]*$/,Se=/^([.]?[a-z0-9]([a-z0-9-]{0,61}[a-z0-9])?)([.][a-z0-9]([a-z0-9-]{0,61}[a-z0-9])?)*$/i,Ie=/^[\u0020-\u003A\u003D-\u007E]*$/,Le=Object.prototype.toString,Ae=(()=>{let n=function(){};return n.prototype=Object.create(null),n})();function Ee(n,e){let t=new Ae,r=n.length;if(r<2)return t;let a=(e==null?void 0:e.decode)||je,o=0;do{let s=n.indexOf("=",o);if(s===-1)break;let c=n.indexOf(";",o),i=c===-1?r:c;if(s>i){o=n.lastIndexOf(";",s-1)+1;continue}let T=X(n,o,s),m=ee(n,s,T),u=n.slice(T,m);if(t[u]===void 0){let p=X(n,s+1,i),y=ee(n,i,p),l=a(n.slice(p,y));t[u]=l;}o=i+1;}while(o<r);return t}function X(n,e,t){do{let r=n.charCodeAt(e);if(r!==32&&r!==9)return e}while(++e<t);return t}function ee(n,e,t){for(;e>t;){let r=n.charCodeAt(--e);if(r!==32&&r!==9)return e+1}return t}function Oe(n,e,t){let r=(t==null?void 0:t.encode)||encodeURIComponent;if(!we.test(n))throw new TypeError(`argument name is invalid: ${n}`);let a=r(e);if(!Me.test(a))throw new TypeError(`argument val is invalid: ${e}`);let o=n+"="+a;if(!t)return o;if(t.maxAge!==void 0){if(!Number.isInteger(t.maxAge))throw new TypeError(`option maxAge is invalid: ${t.maxAge}`);o+="; Max-Age="+t.maxAge;}if(t.domain){if(!Se.test(t.domain))throw new TypeError(`option domain is invalid: ${t.domain}`);o+="; Domain="+t.domain;}if(t.path){if(!Ie.test(t.path))throw new TypeError(`option path is invalid: ${t.path}`);o+="; Path="+t.path;}if(t.expires){if(!$e(t.expires)||!Number.isFinite(t.expires.valueOf()))throw new TypeError(`option expires is invalid: ${t.expires}`);o+="; Expires="+t.expires.toUTCString();}if(t.httpOnly&&(o+="; HttpOnly"),t.secure&&(o+="; Secure"),t.partitioned&&(o+="; Partitioned"),t.priority)switch(typeof t.priority=="string"?t.priority.toLowerCase():void 0){case "low":o+="; Priority=Low";break;case "medium":o+="; Priority=Medium";break;case "high":o+="; 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":o+="; SameSite=Strict";break;case "lax":o+="; SameSite=Lax";break;case "none":o+="; SameSite=None";break;default:throw new TypeError(`option sameSite is invalid: ${t.sameSite}`)}return o}function je(n){if(n.indexOf("%")===-1)return n;try{return decodeURIComponent(n)}catch{return n}}function $e(n){return Le.call(n)==="[object Date]"}});var re=J(_());var j=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()}),P=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((n,e)=>{n.id&&e.addIssue({code:zod.z.ZodIssueCode.custom,message:"Payload cannot have an id"});}),te=zod.z.object({id:zod.z.string().optional(),type:zod.z.literal("MUTATE"),resource:zod.z.string()}),S=te.extend({procedure:zod.z.string(),payload:zod.z.any().optional()}),I=te.extend({resourceId:zod.z.string(),payload:P});zod.z.union([S,I]);var ne=j.omit({resource:true}),V=S.omit({id:true,type:true,resource:true,procedure:true}),F=I.omit({id:true,type:true,resource:true});zod.z.union([F,V]);var ae=n=>async e=>{var t;try{let r=typeof e.headers.getSetCookie=="function"?Object.fromEntries(e.headers):e.headers,a={headers:r,cookies:r.cookie?re.default.parse(r.cookie):{}},o=new URL(e.url),s=o.pathname.split("/"),c=o.searchParams,i=_e__default.default.parse(c.toString()),T=await((t=n.contextProvider)==null?void 0:t.call(n,{transport:"HTTP",headers:a.headers,cookies:a.cookies,query:i}))??{};if(e.method==="GET"){let m=s[s.length-1],{success:u,data:p,error:y}=ne.safeParse(i);if(!u)return Response.json({message:"Invalid query",code:"INVALID_QUERY",details:y},{status:400});let l=await n.handleRequest({req:{...a,type:"QUERY",resourceName:m,context:T,where:p.where,include:p.include,query:i}});return !l||!l.data?Response.json({message:"Invalid resource",code:"INVALID_RESOURCE"},{status:400}):Response.json(l.data)}if(e.method==="POST")try{let m=s[s.length-1],u=s[s.length-2],p=e.body?await e.json():{},y;if(m==="set"){let{success:w,data:M,error:C}=F.safeParse(p);if(!w)return Response.json({message:"Invalid mutation",code:"INVALID_REQUEST",details:C},{status:400});y=M;}else {let{success:w,data:M,error:C}=V.safeParse(p);if(!w)return Response.json({message:"Invalid mutation",code:"INVALID_REQUEST",details:C},{status:400});y=M;}let l=await n.handleRequest({req:{...a,type:"MUTATE",resourceName:u,input:y.payload,context:T,resourceId:y.resourceId,procedure:m!=="set"?m:void 0,query:{}}});return Response.json(l)}catch(m){return console.error("Error parsing mutation from the client:",m),Response.json({message:"Internal server error",code:"INTERNAL_SERVER_ERROR"},{status:500})}return Response.json({message:"Not found",code:"NOT_FOUND"},{status:404})}catch(r){return console.error("Unexpected error:",r),Response.json({message:"Internal server error",code:"INTERNAL_SERVER_ERROR"},{status:500})}};var de=J(_());var b=zod.z.string(),Pe=zod.z.object({id:b,type:zod.z.literal("SUBSCRIBE"),resource:zod.z.string()}),Ve=j.extend({id:b,type:zod.z.literal("QUERY")}),ie=I.extend({id:b}),Fe=S.extend({id:b}),Ne=zod.z.union([Fe,ie]),oe=zod.z.union([Pe,Ve,Ne]),De=zod.z.object({id:b,type:zod.z.literal("REJECT"),resource:zod.z.string(),message:zod.z.string().optional()}),ze=zod.z.object({id:b,type:zod.z.literal("REPLY"),data:zod.z.any()});zod.z.union([De,ze,ie]);zod.z.object({resource:zod.z.string(),data:zod.z.record(zod.z.string(),P)});var ce="0123456789ABCDEFGHJKMNPQRSTVWXYZ",L=32;var Ke=16,ue=10,se=0xffffffffffff;var g;(function(n){n.Base32IncorrectEncoding="B32_ENC_INVALID",n.DecodeTimeInvalidCharacter="DEC_TIME_CHAR",n.DecodeTimeValueMalformed="DEC_TIME_MALFORMED",n.EncodeTimeNegative="ENC_TIME_NEG",n.EncodeTimeSizeExceeded="ENC_TIME_SIZE_EXCEED",n.EncodeTimeValueMalformed="ENC_TIME_MALFORMED",n.PRNGDetectFailure="PRNG_DETECT",n.ULIDInvalid="ULID_INVALID",n.Unexpected="UNEXPECTED",n.UUIDInvalid="UUID_INVALID";})(g||(g={}));var R=class extends Error{constructor(e,t){super(`${t} (${e})`),this.name="ULIDError",this.code=e;}};function Ue(n){let e=Math.floor(n()*L);return e===L&&(e=L-1),ce.charAt(e)}function ke(n){var r;let e=qe(),t=e&&(e.crypto||e.msCrypto)||(typeof $__default.default<"u"?$__default.default:null);if(typeof(t==null?void 0:t.getRandomValues)=="function")return ()=>{let a=new Uint8Array(1);return t.getRandomValues(a),a[0]/255};if(typeof(t==null?void 0:t.randomBytes)=="function")return ()=>t.randomBytes(1).readUInt8()/255;if((r=$__default.default)!=null&&r.randomBytes)return ()=>$__default.default.randomBytes(1).readUInt8()/255;throw new R(g.PRNGDetectFailure,"Failed to find a reliable PRNG")}function qe(){return He()?self:typeof window<"u"?window:typeof global<"u"?global:typeof globalThis<"u"?globalThis:null}function Ze(n,e){let t="";for(;n>0;n--)t=Ue(e)+t;return t}function We(n,e=ue){if(isNaN(n))throw new R(g.EncodeTimeValueMalformed,`Time must be a number: ${n}`);if(n>se)throw new R(g.EncodeTimeSizeExceeded,`Cannot encode a time larger than ${se}: ${n}`);if(n<0)throw new R(g.EncodeTimeNegative,`Time must be positive: ${n}`);if(Number.isInteger(n)===false)throw new R(g.EncodeTimeValueMalformed,`Time must be an integer: ${n}`);let t,r="";for(let a=e;a>0;a--)t=n%L,r=ce.charAt(t)+r,n=(n-t)/L;return r}function He(){return typeof WorkerGlobalScope<"u"&&self instanceof WorkerGlobalScope}function le(n,e){let t=ke(),r=Date.now();return We(r,ue)+Ze(Ke,t)}var N=()=>le().toLowerCase();var pe=n=>{let e={},t={};return n.subscribeToMutations(r=>{let a=r;!a.resourceId||!a.payload||(console.log("Mutation propagated:",a),Object.entries(t[a.resource]??{}).forEach(([o,s])=>{var c;(c=e[o])==null||c.send(JSON.stringify({...a,id:a.id??N()}));}));}),(r,a)=>{var m;let o=u=>{r.send(JSON.stringify(u));},s=N(),c={headers:a.headers,cookies:typeof a.headers.cookie=="string"?de.default.parse(a.headers.cookie):{}},i=_e.parse(a.url.split("?")[1]),T=(m=n.contextProvider)==null?void 0:m.call(n,{transport:"WEBSOCKET",headers:c.headers,cookies:c.cookies,query:i});e[s]=r,console.log("Client connected:",s),r.on("message",async u=>{try{console.log("Message received from the client:",u);let p=oe.parse(JSON.parse(u.toString()));if(p.type==="SUBSCRIBE"){let{resource:y}=p;t[y]||(t[y]={}),t[y][s]={};}else if(p.type==="QUERY"){let{resource:y}=p,l=await n.handleRequest({req:{...c,type:"QUERY",resourceName:y,context:await T??{},query:i}});if(!l||!l.data)throw new Error("Invalid resource");o({id:p.id,type:"REPLY",data:{resource:y,data:Object.fromEntries(Object.entries(l.data??{}).map(([w,M])=>[w,M.value]))}});}else if(p.type==="MUTATE"){let{resource:y}=p;console.log("Received mutation from client:",p);try{let l=await n.handleRequest({req:{...c,type:"MUTATE",resourceName:y,input:p.payload,context:{messageId:p.id,...await T??{}},resourceId:p.resourceId,procedure:p.procedure,query:i}});p.procedure&&o({id:p.id,type:"REPLY",data:l});}catch(l){o({id:p.id,type:"REJECT",resource:y,message:l.message}),console.error("Error parsing mutation from the client:",l);}}}catch(p){console.error("Error handling message from the client:",p);}}),r.on("close",()=>{console.log("Connection closed",s),delete e[s];for(let u of Object.values(t))delete u[s];});}};function ye(n){let e=`${n.protocol}://${n.hostname}${n.url}`,t=new Headers;return Object.entries(n.headers).forEach(([r,a])=>{a&&t.set(r,Array.isArray(a)?a.join(","):a);}),new Request(e,{method:n.method,headers:t,body:n.body&&n.method!=="GET"?JSON.stringify(n.body):void 0})}var Et=(n,e,t)=>{n.ws(`${(t==null?void 0:t.basePath)??""}/ws`,pe(e)),n.use(`${(t==null?void 0:t.basePath)??""}/`,(r,a)=>{ae(e)(ye(r)).then(s=>s.json().then(c=>a.status(s.status).send(c)));});};var D=class n{routes;constructor(e){this.routes=e.routes;}static create(e){return new n(e)}},$t=n=>D.create({...n}),Qe=n=>({handler:e=>({inputValidator:n??zod.z.undefined(),handler:e})}),z=class n{_resourceSchema;resourceName;middlewares;customMutations;constructor(e,t){this.resourceName=e,this.middlewares=new Set,this.customMutations=t??{};}handleFind=async({req:e,db:t})=>({data:await t.rawFind(e.resourceName,e.where,e.include),acceptedValues:null});handleSet=async({req:e,db:t,schema:r})=>{if(!e.input)throw new Error("Payload is required");if(!e.resourceId)throw new Error("ResourceId is required");let a=await t.rawFindById(e.resourceName,e.resourceId),[o,s]=r[this.resourceName].mergeMutation("set",e.input,a);if(!s)throw new Error("Mutation rejected");return {data:await t.rawUpsert(e.resourceName,e.resourceId,o),acceptedValues:s}};async handleRequest(e){let t=r=>(()=>{if(r.type==="QUERY")return this.handleFind({req:r,db:e.db,schema:e.schema});if(r.type==="MUTATE")if(r.procedure){if(this.customMutations[r.procedure]){let a=this.customMutations[r.procedure].inputValidator.parse(r.input);return r.input=a,this.customMutations[r.procedure].handler({req:r,db:e.db,schema:e.schema})}}else return this.handleSet({req:r,db:e.db,schema:e.schema});throw new Error("Invalid request")})();return await Array.from(this.middlewares.values()).reduceRight((r,a)=>o=>a({req:o,next:r}),async r=>t(r))(e.req)}use(...e){for(let t of e)this.middlewares.add(t);return this}withMutations(e){return new n(this.resourceName,e({mutation:Qe}))}},K=class n{middlewares;constructor(e=[]){this.middlewares=e;}createBasicRoute(e){return new z(e.name).use(...this.middlewares)}use(...e){return new n([...this.middlewares,...e])}static create(){return new n}},Ct=K.create;var h=n=>{if(n)return Array.isArray(n.value)?n.value.map(e=>h(e)):typeof n.value!="object"||n.value===null||n.value instanceof Date?n.value:Object.fromEntries(Object.entries(n.value).map(([e,t])=>[e,h(t)]))};var E=class{async insert(e,t){let r=new Date().toISOString();return h(await this.rawUpsert(e.name,t.id,{value:Object.fromEntries(Object.entries(t).map(([a,o])=>[a,{value:o,_meta:{timestamp:r}}]))}))}async update(e,t,r){let a=new Date().toISOString(),{id:o,...s}=r;return h(await this.rawUpsert(e.name,t,{value:Object.fromEntries(Object.entries(s).map(([c,i])=>[c,{value:i,_meta:{timestamp:a}}]))}))}};function H(n,e,t,r){var o,s;if(!r)return t;if(!n)throw new Error("Schema not initialized");let a=n[e];if(!a)throw new Error("Resource not found");for(let[c,i]of Object.entries(r))if(a.fields[c])(i==null?void 0:i.$eq)!==void 0?t=t.where(`${e}.${c}`,i.$eq===null?"is":"=",i.$eq):(i==null?void 0:i.$in)!==void 0?t=t.where(`${e}.${c}`,"in",i.$in):(i==null?void 0:i.$not)!==void 0?((o=i==null?void 0:i.$not)==null?void 0:o.$in)!==void 0?t=t.where(`${e}.${c}`,"not in",i.$not.$in):((s=i==null?void 0:i.$not)==null?void 0:s.$eq)!==void 0?t=t.where(`${e}.${c}`,i.$not.$eq===null?"is not":"!=",i.$not.$eq):t=t.where(`${e}.${c}`,i.$not===null?"is not":"!=",i.$not):(i==null?void 0:i.$gt)!==void 0?t=t.where(`${e}.${c}`,">",i.$gt):(i==null?void 0:i.$gte)!==void 0?t=t.where(`${e}.${c}`,">=",i.$gte):(i==null?void 0:i.$lt)!==void 0?t=t.where(`${e}.${c}`,"<",i.$lt):(i==null?void 0:i.$lte)!==void 0?t=t.where(`${e}.${c}`,"<=",i.$lte):t=t.where(`${e}.${c}`,i===null?"is":"=",i);else if(a.relations[c]){let T=a.relations[c],m=T.entity.name,u=T.type==="one"?"id":T.foreignColumn,p=T.type==="one"?T.relationalColumn:"id";t=t.leftJoin(m,`${m}.${u}`,`${e}.${p}`),t=H(n,m,t,i);}return t}function B(n,e,t,r){if(!r)return t;if(!n)throw new Error("Schema not initialized");let a=n[e];if(!a)throw new Error(`Resource not found: ${e}`);for(let o of Object.keys(r)){if(!a.relations[o])throw new Error(`Relation ${o} not found in resource ${e}`);let s=a.relations[o],c=s.entity.name,i=s.type==="one"?"id":s.foreignColumn,T=s.type==="one"?s.relationalColumn:"id",m=s.type==="one"?postgres.jsonObjectFrom:postgres.jsonArrayFrom;t=t.select(u=>m(u.selectFrom(c).selectAll(c).whereRef(`${c}.${i}`,"=",`${e}.${T}`).select(p=>postgres.jsonObjectFrom(p.selectFrom(`${c}_meta`).selectAll(`${c}_meta`).whereRef(`${c}_meta.id`,"=",`${c}.id`)).as("_meta"))).as(o));}return t}var G=class extends E{db;schema;constructor(e){super(),this.db=new kysely.Kysely({dialect:new kysely.PostgresDialect({pool:e})});}async updateSchema(e){this.schema=e;let t=await this.db.introspection.getTables();for(let[r,a]of Object.entries(e)){let o=t.find(i=>i.name===r);o||await this.db.schema.createTable(r).ifNotExists().execute();let s=`${r}_meta`,c=t.find(i=>i.name===s);c||await this.db.schema.createTable(s).ifNotExists().execute();for(let[i,T]of Object.entries(a.fields)){let m=o==null?void 0:o.columns.find(y=>y.name===i),u=T.getStorageFieldType();m?m.dataType!==u.type&&console.error("Column type mismatch:",i,"expected to have type:",u.type,"but has type:",m.dataType):(await this.db.schema.alterTable(r).addColumn(i,u.type,y=>{let l=y;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(y=>{throw console.error("Error adding column",i,y),y}),u.index&&await this.db.schema.createIndex(`${r}_${i}_index`).on(r).column(i).execute().catch(y=>{})),(c==null?void 0:c.columns.find(y=>y.name===i))||await this.db.schema.alterTable(s).addColumn(i,"varchar",y=>{let l=y;return u.primary&&(l=l.primaryKey().references(`${r}.${i}`)),l}).execute();}}}async rawFindById(e,t,r){if(!this.schema)throw new Error("Schema not initialized");let a=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"));a=B(this.schema,e,a,r);let o=await a.executeTakeFirst();if(o)return this.convertToMaterializedLiveType(o)}async findOne(e,t,r){let a=await this.rawFindById(e.name,t,r==null?void 0:r.include);if(a)return h(a)}async rawFind(e,t,r){if(!this.schema)throw new Error("Schema not initialized");let a=this.db.selectFrom(e).selectAll(e).select(i=>postgres.jsonObjectFrom(i.selectFrom(`${e}_meta`).selectAll(`${e}_meta`).whereRef(`${e}_meta.id`,"=",`${e}.id`)).as("_meta"));a=H(this.schema,e,a,t),a=B(this.schema,e,a,r);let o=await a.execute(),s=Object.fromEntries(o.map(i=>{let{id:T,...m}=i;return [T,m]}));return Object.keys(s).length===0?{}:Object.entries(s).reduce((i,[T,m])=>(i[T]=this.convertToMaterializedLiveType(m),i),{})}async find(e,t){let r=await this.rawFind(e.name,t==null?void 0:t.where,t==null?void 0:t.include);return Object.fromEntries(Object.entries(r).map(([a,o])=>[a,h(o)]))}async rawUpsert(e,t,r){return await this.db.transaction().execute(async a=>{var i;let o=!!await a.selectFrom(e).select("id").where("id","=",t).executeTakeFirst(),s={},c={};for(let[T,m]of Object.entries(r.value)){let u=(i=m._meta)==null?void 0:i.timestamp;u&&(s[T]=m.value,c[T]=u);}o?await Promise.all([a.updateTable(e).set(s).where("id","=",t).execute(),a.updateTable(`${e}_meta`).set(c).where("id","=",t).execute()]):await Promise.all([a.insertInto(e).values({...s,id:t}).execute(),a.insertInto(`${e}_meta`).values({...c,id:t}).execute()]);}),r}convertToMaterializedLiveType(e){if(!e._meta)throw new Error("Missing _meta");return {value:Object.entries(e).reduce((t,[r,a])=>{var o,s,c;return r==="_meta"||(r==="id"?t[r]={value:a}:Array.isArray(a)?t[r]={value:a.map(i=>this.convertToMaterializedLiveType(i)),_meta:{timestamp:(o=e==null?void 0:e._meta)==null?void 0:o[r]}}:typeof a=="object"&&a!==null&&!(a instanceof Date)?t[r]={...this.convertToMaterializedLiveType(a),_meta:{timestamp:(s=e==null?void 0:e._meta)==null?void 0:s[r]}}:t[r]={value:a,_meta:{timestamp:(c=e==null?void 0:e._meta)==null?void 0:c[r]}}),t},{})}}};var Q=class n{router;storage;schema;middlewares=new Set;contextProvider;mutationSubscriptions=new Set;constructor(e){var t;this.router=e.router,this.storage=e.storage,this.schema=e.schema,(t=e.middlewares)==null||t.forEach(r=>{this.middlewares.add(r);}),this.storage.updateSchema(this.schema),this.contextProvider=e.contextProvider;}static create(e){return new n(e)}subscribeToMutations(e){return this.mutationSubscriptions.add(e),()=>{this.mutationSubscriptions.delete(e);}}async handleRequest(e){if(!this.router.routes[e.req.resourceName])throw new Error("Invalid resource");let t=await Array.from(this.middlewares.values()).reduceRight((r,a)=>o=>a({req:o,next:r}),async r=>this.router.routes[e.req.resourceName].handleRequest({req:r,db:this.storage,schema:this.schema}))(e.req);return t&&e.req.type==="MUTATE"&&t.acceptedValues&&Object.keys(t.acceptedValues).length>0&&this.mutationSubscriptions.forEach(r=>{r({id:e.req.context.messageId,type:"MUTATE",resource:e.req.resourceName,payload:t.acceptedValues??{},resourceId:e.req.resourceId});}),t}use(e){return this.middlewares.add(e),this}context(e){return this.contextProvider=e,this}},un=Q.create;
2
+ exports.Route=z;exports.RouteFactory=K;exports.Router=D;exports.SQLStorage=G;exports.Server=Q;exports.Storage=E;exports.expressAdapter=Et;exports.routeFactory=Ct;exports.router=$t;exports.server=un;
package/dist/server.d.cts CHANGED
@@ -1,5 +1,7 @@
1
- import { z, ZodTypeAny } from 'zod';
1
+ import { z } from 'zod';
2
2
  import { LiveObjectAny, Schema, MaterializedLiveType, IncludeClause, InferLiveObject, WhereClause, LiveObjectMutationInput } from './index.cjs';
3
+ import * as z3 from 'zod/v3';
4
+ import * as z4 from 'zod/v4/core';
3
5
  import { PostgresPool } from 'kysely';
4
6
  import { Application } from 'express-ws';
5
7
 
@@ -48,12 +50,12 @@ type RequestHandler<TInput, TResult, TSchema extends Schema<any> = Schema<any>>
48
50
  db: Storage;
49
51
  schema: TSchema;
50
52
  }) => Promise<TResult>;
51
- type Mutation<TInputValidator extends ZodTypeAny, // TODO use StandardSchema instead
53
+ type Mutation<TInputValidator extends z3.ZodTypeAny | z4.$ZodType, // TODO use StandardSchema instead
52
54
  THandler extends RequestHandler<z.infer<TInputValidator>, any, any>> = {
53
55
  inputValidator: TInputValidator;
54
56
  handler: THandler;
55
57
  };
56
- declare const mutationCreator: <TInputValidator extends ZodTypeAny>(validator?: TInputValidator) => {
58
+ declare const mutationCreator: <TInputValidator extends z3.ZodTypeAny | z4.$ZodType>(validator?: TInputValidator) => {
57
59
  handler: <THandler extends RequestHandler<z.infer<TInputValidator>, any, any>>(handler: THandler) => Mutation<TInputValidator, THandler>;
58
60
  };
59
61
  declare class Route<TResourceSchema extends LiveObjectAny, TMiddleware extends Middleware<any>, TCustomMutations extends Record<string, Mutation<any, RequestHandler<any, any>>>> {
package/dist/server.d.ts CHANGED
@@ -1,5 +1,7 @@
1
- import { z, ZodTypeAny } from 'zod';
1
+ import { z } from 'zod';
2
2
  import { LiveObjectAny, Schema, MaterializedLiveType, IncludeClause, InferLiveObject, WhereClause, LiveObjectMutationInput } from './index.js';
3
+ import * as z3 from 'zod/v3';
4
+ import * as z4 from 'zod/v4/core';
3
5
  import { PostgresPool } from 'kysely';
4
6
  import { Application } from 'express-ws';
5
7
 
@@ -48,12 +50,12 @@ type RequestHandler<TInput, TResult, TSchema extends Schema<any> = Schema<any>>
48
50
  db: Storage;
49
51
  schema: TSchema;
50
52
  }) => Promise<TResult>;
51
- type Mutation<TInputValidator extends ZodTypeAny, // TODO use StandardSchema instead
53
+ type Mutation<TInputValidator extends z3.ZodTypeAny | z4.$ZodType, // TODO use StandardSchema instead
52
54
  THandler extends RequestHandler<z.infer<TInputValidator>, any, any>> = {
53
55
  inputValidator: TInputValidator;
54
56
  handler: THandler;
55
57
  };
56
- declare const mutationCreator: <TInputValidator extends ZodTypeAny>(validator?: TInputValidator) => {
58
+ declare const mutationCreator: <TInputValidator extends z3.ZodTypeAny | z4.$ZodType>(validator?: TInputValidator) => {
57
59
  handler: <THandler extends RequestHandler<z.infer<TInputValidator>, any, any>>(handler: THandler) => Mutation<TInputValidator, THandler>;
58
60
  };
59
61
  declare class Route<TResourceSchema extends LiveObjectAny, TMiddleware extends Middleware<any>, TCustomMutations extends Record<string, Mutation<any, RequestHandler<any, any>>>> {
package/dist/server.js CHANGED
@@ -1,2 +1,2 @@
1
- import {a,b,r}from'./chunk-LLHCJUB6.js';import Re,{parse}from'qs';import {z as z$1}from'zod';import $ from'node:crypto';import {Kysely,PostgresDialect}from'kysely';import {jsonObjectFrom,jsonArrayFrom}from'kysely/helpers/postgres';var O=a(v=>{Object.defineProperty(v,"__esModule",{value:true});v.parse=fe;v.serialize=ye;var ce=/^[\u0021-\u003A\u003C\u003E-\u007E]+$/,de=/^[\u0021-\u003A\u003C-\u007E]*$/,ue=/^([.]?[a-z0-9]([a-z0-9-]{0,61}[a-z0-9])?)([.][a-z0-9]([a-z0-9-]{0,61}[a-z0-9])?)*$/i,le=/^[\u0020-\u003A\u003D-\u007E]*$/,me=Object.prototype.toString,pe=(()=>{let r=function(){};return r.prototype=Object.create(null),r})();function fe(r,e){let t=new pe,n=r.length;if(n<2)return t;let a=(e==null?void 0:e.decode)||he,o=0;do{let s=r.indexOf("=",o);if(s===-1)break;let c=r.indexOf(";",o),i=c===-1?n:c;if(s>i){o=r.lastIndexOf(";",s-1)+1;continue}let y=q(r,o,s),f=B(r,s,y),d=r.slice(y,f);if(t[d]===void 0){let m=q(r,s+1,i),p=B(r,i,m),u=a(r.slice(m,p));t[d]=u;}o=i+1;}while(o<n);return t}function q(r,e,t){do{let n=r.charCodeAt(e);if(n!==32&&n!==9)return e}while(++e<t);return t}function B(r,e,t){for(;e>t;){let n=r.charCodeAt(--e);if(n!==32&&n!==9)return e+1}return t}function ye(r,e,t){let n=(t==null?void 0:t.encode)||encodeURIComponent;if(!ce.test(r))throw new TypeError(`argument name is invalid: ${r}`);let a=n(e);if(!de.test(a))throw new TypeError(`argument val is invalid: ${e}`);let o=r+"="+a;if(!t)return o;if(t.maxAge!==void 0){if(!Number.isInteger(t.maxAge))throw new TypeError(`option maxAge is invalid: ${t.maxAge}`);o+="; Max-Age="+t.maxAge;}if(t.domain){if(!ue.test(t.domain))throw new TypeError(`option domain is invalid: ${t.domain}`);o+="; Domain="+t.domain;}if(t.path){if(!le.test(t.path))throw new TypeError(`option path is invalid: ${t.path}`);o+="; Path="+t.path;}if(t.expires){if(!Te(t.expires)||!Number.isFinite(t.expires.valueOf()))throw new TypeError(`option expires is invalid: ${t.expires}`);o+="; Expires="+t.expires.toUTCString();}if(t.httpOnly&&(o+="; HttpOnly"),t.secure&&(o+="; Secure"),t.partitioned&&(o+="; Partitioned"),t.priority)switch(typeof t.priority=="string"?t.priority.toLowerCase():void 0){case "low":o+="; Priority=Low";break;case "medium":o+="; Priority=Medium";break;case "high":o+="; 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":o+="; SameSite=Strict";break;case "lax":o+="; SameSite=Lax";break;case "none":o+="; SameSite=None";break;default:throw new TypeError(`option sameSite is invalid: ${t.sameSite}`)}return o}function he(r){if(r.indexOf("%")===-1)return r;try{return decodeURIComponent(r)}catch{return r}}function Te(r){return me.call(r)==="[object Date]"}});var Q=b(O(),1);var A=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()}),L=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((r,e)=>{r.id&&e.addIssue({code:z$1.ZodIssueCode.custom,message:"Payload cannot have an id"});}),H=z$1.object({id:z$1.string().optional(),type:z$1.literal("MUTATE"),resource:z$1.string()}),S=H.extend({procedure:z$1.string(),payload:z$1.any().optional()}),M=H.extend({resourceId:z$1.string(),payload:L});z$1.union([S,M]);var G=A.omit({resource:true}),P=S.omit({id:true,type:true,resource:true,procedure:true}),C=M.omit({id:true,type:true,resource:true});z$1.union([C,P]);var W=r=>async e=>{var t;try{let n=typeof e.headers.getSetCookie=="function"?Object.fromEntries(e.headers):e.headers,a={headers:n,cookies:n.cookie?Q.default.parse(n.cookie):{}},o=new URL(e.url),s=o.pathname.split("/"),c=o.searchParams,i=Re.parse(c.toString()),y=await((t=r.contextProvider)==null?void 0:t.call(r,{transport:"HTTP",headers:a.headers,cookies:a.cookies,query:i}))??{};if(e.method==="GET"){let f=s[s.length-1],{success:d,data:m,error:p}=G.safeParse(i);if(!d)return Response.json({message:"Invalid query",code:"INVALID_QUERY",details:p},{status:400});let u=await r.handleRequest({req:{...a,type:"QUERY",resourceName:f,context:y,where:m.where,include:m.include,query:i}});return !u||!u.data?Response.json({message:"Invalid resource",code:"INVALID_RESOURCE"},{status:400}):Response.json(u.data)}if(e.method==="POST")try{let f=s[s.length-1],d=s[s.length-2],m=e.body?await e.json():{},p;if(f==="set"){let{success:b,data:x,error:j}=C.safeParse(m);if(!b)return Response.json({message:"Invalid mutation",code:"INVALID_REQUEST",details:j},{status:400});p=x;}else {let{success:b,data:x,error:j}=P.safeParse(m);if(!b)return Response.json({message:"Invalid mutation",code:"INVALID_REQUEST",details:j},{status:400});p=x;}let u=await r.handleRequest({req:{...a,type:"MUTATE",resourceName:d,input:p.payload,context:y,resourceId:p.resourceId,procedure:f!=="set"?f:void 0,query:{}}});return Response.json(u)}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 re=b(O(),1);var w=z$1.string(),we=z$1.object({id:w,type:z$1.literal("SUBSCRIBE"),resource:z$1.string()}),be=A.extend({id:w,type:z$1.literal("QUERY")}),Y=M.extend({id:w}),xe=S.extend({id:w}),Se=z$1.union([xe,Y]),J=z$1.union([we,be,Se]),Me=z$1.object({id:w,type:z$1.literal("REJECT"),resource:z$1.string(),message:z$1.string().optional()}),Ie=z$1.object({id:w,type:z$1.literal("REPLY"),data:z$1.any()});z$1.union([Me,Ie,Y]);z$1.object({resource:z$1.string(),data:z$1.record(z$1.string(),L)});var X="0123456789ABCDEFGHJKMNPQRSTVWXYZ",I=32;var Ee=16,ee=10,K=0xffffffffffff;var T;(function(r){r.Base32IncorrectEncoding="B32_ENC_INVALID",r.DecodeTimeInvalidCharacter="DEC_TIME_CHAR",r.DecodeTimeValueMalformed="DEC_TIME_MALFORMED",r.EncodeTimeNegative="ENC_TIME_NEG",r.EncodeTimeSizeExceeded="ENC_TIME_SIZE_EXCEED",r.EncodeTimeValueMalformed="ENC_TIME_MALFORMED",r.PRNGDetectFailure="PRNG_DETECT",r.ULIDInvalid="ULID_INVALID",r.Unexpected="UNEXPECTED",r.UUIDInvalid="UUID_INVALID";})(T||(T={}));var g=class extends Error{constructor(e,t){super(`${t} (${e})`),this.name="ULIDError",this.code=e;}};function ve(r){let e=Math.floor(r()*I);return e===I&&(e=I-1),X.charAt(e)}function Ae(r){var n;let e=$e(),t=e&&(e.crypto||e.msCrypto)||(typeof $<"u"?$:null);if(typeof(t==null?void 0:t.getRandomValues)=="function")return ()=>{let a=new Uint8Array(1);return t.getRandomValues(a),a[0]/255};if(typeof(t==null?void 0:t.randomBytes)=="function")return ()=>t.randomBytes(1).readUInt8()/255;if((n=$)!=null&&n.randomBytes)return ()=>$.randomBytes(1).readUInt8()/255;throw new g(T.PRNGDetectFailure,"Failed to find a reliable PRNG")}function $e(){return Le()?self:typeof window<"u"?window:typeof global<"u"?global:typeof globalThis<"u"?globalThis:null}function je(r,e){let t="";for(;r>0;r--)t=ve(e)+t;return t}function Oe(r,e=ee){if(isNaN(r))throw new g(T.EncodeTimeValueMalformed,`Time must be a number: ${r}`);if(r>K)throw new g(T.EncodeTimeSizeExceeded,`Cannot encode a time larger than ${K}: ${r}`);if(r<0)throw new g(T.EncodeTimeNegative,`Time must be positive: ${r}`);if(Number.isInteger(r)===false)throw new g(T.EncodeTimeValueMalformed,`Time must be an integer: ${r}`);let t,n="";for(let a=e;a>0;a--)t=r%I,n=X.charAt(t)+n,r=(r-t)/I;return n}function Le(){return typeof WorkerGlobalScope<"u"&&self instanceof WorkerGlobalScope}function te(r,e){let t=Ae(),n=Date.now();return Oe(n,ee)+je(Ee,t)}var _=()=>te().toLowerCase();var ne=r=>{let e={},t={};return r.subscribeToMutations(n=>{let a=n;!a.resourceId||!a.payload||(console.log("Mutation propagated:",a),Object.entries(t[a.resource]??{}).forEach(([o,s])=>{var c;(c=e[o])==null||c.send(JSON.stringify({...a,id:a.id??_()}));}));}),(n,a)=>{var f;let o=d=>{n.send(JSON.stringify(d));},s=_(),c={headers:a.headers,cookies:typeof a.headers.cookie=="string"?re.default.parse(a.headers.cookie):{}},i=parse(a.url.split("?")[1]),y=(f=r.contextProvider)==null?void 0:f.call(r,{transport:"WEBSOCKET",headers:c.headers,cookies:c.cookies,query:i});e[s]=n,console.log("Client connected:",s),n.on("message",async d=>{try{console.log("Message received from the client:",d);let m=J.parse(JSON.parse(d.toString()));if(m.type==="SUBSCRIBE"){let{resource:p}=m;t[p]||(t[p]={}),t[p][s]={};}else if(m.type==="QUERY"){let{resource:p}=m,u=await r.handleRequest({req:{...c,type:"QUERY",resourceName:p,context:await y??{},query:i}});if(!u||!u.data)throw new Error("Invalid resource");o({id:m.id,type:"REPLY",data:{resource:p,data:Object.fromEntries(Object.entries(u.data??{}).map(([b,x])=>[b,x.value]))}});}else if(m.type==="MUTATE"){let{resource:p}=m;console.log("Received mutation from client:",m);try{let u=await r.handleRequest({req:{...c,type:"MUTATE",resourceName:p,input:m.payload,context:{messageId:m.id,...await y??{}},resourceId:m.resourceId,procedure:m.procedure,query:i}});m.procedure&&o({id:m.id,type:"REPLY",data:u});}catch(u){o({id:m.id,type:"REJECT",resource:p,message:u.message}),console.error("Error parsing mutation from the client:",u);}}}catch(m){console.error("Error handling message from the client:",m);}}),n.on("close",()=>{console.log("Connection closed",s),delete e[s];for(let d of Object.values(t))delete d[s];});}};function ae(r){let e=`${r.protocol}://${r.hostname}${r.url}`,t=new Headers;return Object.entries(r.headers).forEach(([n,a])=>{a&&t.set(n,Array.isArray(a)?a.join(","):a);}),new Request(e,{method:r.method,headers:t,body:r.body&&r.method!=="GET"?JSON.stringify(r.body):void 0})}var pt=(r,e,t)=>{r.ws(`${(t==null?void 0:t.basePath)??""}/ws`,ne(e)),r.use(`${(t==null?void 0:t.basePath)??""}/`,(n,a)=>{W(e)(ae(n)).then(s=>s.json().then(c=>a.status(s.status).send(c)));});};var U=class r{routes;constructor(e){this.routes=e.routes;}static create(e){return new r(e)}},ht=r=>U.create({...r}),_e=r=>({handler:e=>({inputValidator:r??z$1.undefined(),handler:e})}),N=class r{_resourceSchema;resourceName;middlewares;customMutations;constructor(e,t){this.resourceName=e,this.middlewares=new Set,this.customMutations=t??{};}handleFind=async({req:e,db:t})=>({data:await t.rawFind(e.resourceName,e.where,e.include),acceptedValues:null});handleSet=async({req:e,db:t,schema:n})=>{if(!e.input)throw new Error("Payload is required");if(!e.resourceId)throw new Error("ResourceId is required");let a=await t.rawFindById(e.resourceName,e.resourceId),[o,s]=n[this.resourceName].mergeMutation("set",e.input,a);if(!s)throw new Error("Mutation rejected");return {data:await t.rawUpsert(e.resourceName,e.resourceId,o),acceptedValues:s}};async handleRequest(e){let t=n=>(()=>{if(n.type==="QUERY")return this.handleFind({req:n,db:e.db,schema:e.schema});if(n.type==="MUTATE")if(n.procedure){if(this.customMutations[n.procedure]){let a=this.customMutations[n.procedure].inputValidator.parse(n.input);return n.input=a,this.customMutations[n.procedure].handler({req:n,db:e.db,schema:e.schema})}}else return this.handleSet({req:n,db:e.db,schema:e.schema});throw new Error("Invalid request")})();return await Array.from(this.middlewares.values()).reduceRight((n,a)=>o=>a({req:o,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:_e}))}},D=class r{middlewares;constructor(e=[]){this.middlewares=e;}createBasicRoute(e){return new N(e.name).use(...this.middlewares)}use(...e){return new r([...this.middlewares,...e])}static create(){return new r}},Tt=D.create;var E=class{async insert(e,t){let n=new Date().toISOString();return r(await this.rawUpsert(e.name,t.id,{value:Object.fromEntries(Object.entries(t).map(([a,o])=>[a,{value:o,_meta:{timestamp:n}}]))}))}async update(e,t,n){let a=new Date().toISOString(),{id:o,...s}=n;return r(await this.rawUpsert(e.name,t,{value:Object.fromEntries(Object.entries(s).map(([c,i])=>[c,{value:i,_meta:{timestamp:a}}]))}))}};function z(r,e,t,n){var o,s;if(!n)return t;if(!r)throw new Error("Schema not initialized");let a=r[e];if(!a)throw new Error("Resource not found");for(let[c,i]of Object.entries(n))if(a.fields[c])(i==null?void 0:i.$eq)!==void 0?t=t.where(`${e}.${c}`,i.$eq===null?"is":"=",i.$eq):(i==null?void 0:i.$in)!==void 0?t=t.where(`${e}.${c}`,"in",i.$in):(i==null?void 0:i.$not)!==void 0?((o=i==null?void 0:i.$not)==null?void 0:o.$in)!==void 0?t=t.where(`${e}.${c}`,"not in",i.$not.$in):((s=i==null?void 0:i.$not)==null?void 0:s.$eq)!==void 0?t=t.where(`${e}.${c}`,i.$not.$eq===null?"is not":"!=",i.$not.$eq):t=t.where(`${e}.${c}`,i.$not===null?"is not":"!=",i.$not):(i==null?void 0:i.$gt)!==void 0?t=t.where(`${e}.${c}`,">",i.$gt):(i==null?void 0:i.$gte)!==void 0?t=t.where(`${e}.${c}`,">=",i.$gte):(i==null?void 0:i.$lt)!==void 0?t=t.where(`${e}.${c}`,"<",i.$lt):(i==null?void 0:i.$lte)!==void 0?t=t.where(`${e}.${c}`,"<=",i.$lte):t=t.where(`${e}.${c}`,i===null?"is":"=",i);else if(a.relations[c]){let y=a.relations[c],f=y.entity.name,d=y.type==="one"?"id":y.foreignColumn,m=y.type==="one"?y.relationalColumn:"id";t=t.leftJoin(f,`${f}.${d}`,`${e}.${m}`),t=z(r,f,t,i);}return t}function V(r,e,t,n){if(!n)return t;if(!r)throw new Error("Schema not initialized");let a=r[e];if(!a)throw new Error(`Resource not found: ${e}`);for(let o of Object.keys(n)){if(!a.relations[o])throw new Error(`Relation ${o} not found in resource ${e}`);let s=a.relations[o],c=s.entity.name,i=s.type==="one"?"id":s.foreignColumn,y=s.type==="one"?s.relationalColumn:"id",f=s.type==="one"?jsonObjectFrom:jsonArrayFrom;t=t.select(d=>f(d.selectFrom(c).selectAll(c).whereRef(`${c}.${i}`,"=",`${e}.${y}`).select(m=>jsonObjectFrom(m.selectFrom(`${c}_meta`).selectAll(`${c}_meta`).whereRef(`${c}_meta.id`,"=",`${c}.id`)).as("_meta"))).as(o));}return t}var k=class extends E{db;schema;constructor(e){super(),this.db=new Kysely({dialect:new PostgresDialect({pool:e})});}async updateSchema(e){this.schema=e;let t=await this.db.introspection.getTables();for(let[n,a]of Object.entries(e)){let o=t.find(i=>i.name===n);o||await this.db.schema.createTable(n).ifNotExists().execute();let s=`${n}_meta`,c=t.find(i=>i.name===s);c||await this.db.schema.createTable(s).ifNotExists().execute();for(let[i,y]of Object.entries(a.fields)){let f=o==null?void 0:o.columns.find(p=>p.name===i),d=y.getStorageFieldType();f?f.dataType!==d.type&&console.error("Column type mismatch:",i,"expected to have type:",d.type,"but has type:",f.dataType):(await this.db.schema.alterTable(n).addColumn(i,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",i,p),p}),d.index&&await this.db.schema.createIndex(`${n}_${i}_index`).on(n).column(i).execute().catch(p=>{})),(c==null?void 0:c.columns.find(p=>p.name===i))||await this.db.schema.alterTable(s).addColumn(i,"varchar",p=>{let u=p;return d.primary&&(u=u.primaryKey().references(`${n}.${i}`)),u}).execute();}}}async rawFindById(e,t,n){if(!this.schema)throw new Error("Schema not initialized");let a=await this.db.selectFrom(e).where("id","=",t).selectAll(e).select(s=>jsonObjectFrom(s.selectFrom(`${e}_meta`).selectAll(`${e}_meta`).whereRef(`${e}_meta.id`,"=",`${e}.id`)).as("_meta"));a=V(this.schema,e,a,n);let o=await a.executeTakeFirst();if(o)return this.convertToMaterializedLiveType(o)}async findOne(e,t,n){let a=await this.rawFindById(e.name,t,n==null?void 0:n.include);if(a)return r(a)}async rawFind(e,t,n){if(!this.schema)throw new Error("Schema not initialized");let a=this.db.selectFrom(e).selectAll(e).select(i=>jsonObjectFrom(i.selectFrom(`${e}_meta`).selectAll(`${e}_meta`).whereRef(`${e}_meta.id`,"=",`${e}.id`)).as("_meta"));a=z(this.schema,e,a,t),a=V(this.schema,e,a,n);let o=await a.execute(),s=Object.fromEntries(o.map(i=>{let{id:y,...f}=i;return [y,f]}));return Object.keys(s).length===0?{}:Object.entries(s).reduce((i,[y,f])=>(i[y]=this.convertToMaterializedLiveType(f),i),{})}async find(e,t){let n=await this.rawFind(e.name,t==null?void 0:t.where,t==null?void 0:t.include);return Object.fromEntries(Object.entries(n).map(([a,o])=>[a,r(o)]))}async rawUpsert(e,t,n){return await this.db.transaction().execute(async a=>{var i;let o=!!await a.selectFrom(e).select("id").where("id","=",t).executeTakeFirst(),s={},c={};for(let[y,f]of Object.entries(n.value)){let d=(i=f._meta)==null?void 0:i.timestamp;d&&(s[y]=f.value,c[y]=d);}o?await Promise.all([a.updateTable(e).set(s).where("id","=",t).execute(),a.updateTable(`${e}_meta`).set(c).where("id","=",t).execute()]):await Promise.all([a.insertInto(e).values({...s,id:t}).execute(),a.insertInto(`${e}_meta`).values({...c,id:t}).execute()]);}),n}convertToMaterializedLiveType(e){if(!e._meta)throw new Error("Missing _meta");return {value:Object.entries(e).reduce((t,[n,a])=>{var o,s,c;return n==="_meta"||(n==="id"?t[n]={value:a}:Array.isArray(a)?t[n]={value:a.map(i=>this.convertToMaterializedLiveType(i)),_meta:{timestamp:(o=e==null?void 0:e._meta)==null?void 0:o[n]}}:typeof a=="object"&&a!==null&&!(a instanceof Date)?t[n]={...this.convertToMaterializedLiveType(a),_meta:{timestamp:(s=e==null?void 0:e._meta)==null?void 0:s[n]}}:t[n]={value:a,_meta:{timestamp:(c=e==null?void 0:e._meta)==null?void 0:c[n]}}),t},{})}}};var F=class r{router;storage;schema;middlewares=new Set;contextProvider;mutationSubscriptions=new Set;constructor(e){var t;this.router=e.router,this.storage=e.storage,this.schema=e.schema,(t=e.middlewares)==null||t.forEach(n=>{this.middlewares.add(n);}),this.storage.updateSchema(this.schema),this.contextProvider=e.contextProvider;}static create(e){return new r(e)}subscribeToMutations(e){return this.mutationSubscriptions.add(e),()=>{this.mutationSubscriptions.delete(e);}}async handleRequest(e){if(!this.router.routes[e.req.resourceName])throw new Error("Invalid resource");let t=await Array.from(this.middlewares.values()).reduceRight((n,a)=>o=>a({req:o,next:n}),async n=>this.router.routes[e.req.resourceName].handleRequest({req:n,db:this.storage,schema:this.schema}))(e.req);return t&&e.req.type==="MUTATE"&&t.acceptedValues&&Object.keys(t.acceptedValues).length>0&&this.mutationSubscriptions.forEach(n=>{n({id:e.req.context.messageId,type:"MUTATE",resource:e.req.resourceName,payload:t.acceptedValues??{},resourceId:e.req.resourceId});}),t}use(e){return this.middlewares.add(e),this}context(e){return this.contextProvider=e,this}},Lt=F.create;
2
- export{N as Route,D as RouteFactory,U as Router,k as SQLStorage,F as Server,E as Storage,pt as expressAdapter,Tt as routeFactory,ht as router,Lt as server};
1
+ import {a,b,r}from'./chunk-LLHCJUB6.js';import Re,{parse}from'qs';import {z as z$1}from'zod';import $ from'node:crypto';import {Kysely,PostgresDialect}from'kysely';import {jsonObjectFrom,jsonArrayFrom}from'kysely/helpers/postgres';var O=a(v=>{Object.defineProperty(v,"__esModule",{value:true});v.parse=fe;v.serialize=ye;var ce=/^[\u0021-\u003A\u003C\u003E-\u007E]+$/,de=/^[\u0021-\u003A\u003C-\u007E]*$/,ue=/^([.]?[a-z0-9]([a-z0-9-]{0,61}[a-z0-9])?)([.][a-z0-9]([a-z0-9-]{0,61}[a-z0-9])?)*$/i,le=/^[\u0020-\u003A\u003D-\u007E]*$/,me=Object.prototype.toString,pe=(()=>{let r=function(){};return r.prototype=Object.create(null),r})();function fe(r,e){let t=new pe,n=r.length;if(n<2)return t;let a=(e==null?void 0:e.decode)||he,o=0;do{let s=r.indexOf("=",o);if(s===-1)break;let c=r.indexOf(";",o),i=c===-1?n:c;if(s>i){o=r.lastIndexOf(";",s-1)+1;continue}let y=q(r,o,s),f=B(r,s,y),d=r.slice(y,f);if(t[d]===void 0){let m=q(r,s+1,i),p=B(r,i,m),u=a(r.slice(m,p));t[d]=u;}o=i+1;}while(o<n);return t}function q(r,e,t){do{let n=r.charCodeAt(e);if(n!==32&&n!==9)return e}while(++e<t);return t}function B(r,e,t){for(;e>t;){let n=r.charCodeAt(--e);if(n!==32&&n!==9)return e+1}return t}function ye(r,e,t){let n=(t==null?void 0:t.encode)||encodeURIComponent;if(!ce.test(r))throw new TypeError(`argument name is invalid: ${r}`);let a=n(e);if(!de.test(a))throw new TypeError(`argument val is invalid: ${e}`);let o=r+"="+a;if(!t)return o;if(t.maxAge!==void 0){if(!Number.isInteger(t.maxAge))throw new TypeError(`option maxAge is invalid: ${t.maxAge}`);o+="; Max-Age="+t.maxAge;}if(t.domain){if(!ue.test(t.domain))throw new TypeError(`option domain is invalid: ${t.domain}`);o+="; Domain="+t.domain;}if(t.path){if(!le.test(t.path))throw new TypeError(`option path is invalid: ${t.path}`);o+="; Path="+t.path;}if(t.expires){if(!Te(t.expires)||!Number.isFinite(t.expires.valueOf()))throw new TypeError(`option expires is invalid: ${t.expires}`);o+="; Expires="+t.expires.toUTCString();}if(t.httpOnly&&(o+="; HttpOnly"),t.secure&&(o+="; Secure"),t.partitioned&&(o+="; Partitioned"),t.priority)switch(typeof t.priority=="string"?t.priority.toLowerCase():void 0){case "low":o+="; Priority=Low";break;case "medium":o+="; Priority=Medium";break;case "high":o+="; 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":o+="; SameSite=Strict";break;case "lax":o+="; SameSite=Lax";break;case "none":o+="; SameSite=None";break;default:throw new TypeError(`option sameSite is invalid: ${t.sameSite}`)}return o}function he(r){if(r.indexOf("%")===-1)return r;try{return decodeURIComponent(r)}catch{return r}}function Te(r){return me.call(r)==="[object Date]"}});var Q=b(O(),1);var A=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()}),L=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((r,e)=>{r.id&&e.addIssue({code:z$1.ZodIssueCode.custom,message:"Payload cannot have an id"});}),H=z$1.object({id:z$1.string().optional(),type:z$1.literal("MUTATE"),resource:z$1.string()}),S=H.extend({procedure:z$1.string(),payload:z$1.any().optional()}),M=H.extend({resourceId:z$1.string(),payload:L});z$1.union([S,M]);var G=A.omit({resource:true}),P=S.omit({id:true,type:true,resource:true,procedure:true}),C=M.omit({id:true,type:true,resource:true});z$1.union([C,P]);var W=r=>async e=>{var t;try{let n=typeof e.headers.getSetCookie=="function"?Object.fromEntries(e.headers):e.headers,a={headers:n,cookies:n.cookie?Q.default.parse(n.cookie):{}},o=new URL(e.url),s=o.pathname.split("/"),c=o.searchParams,i=Re.parse(c.toString()),y=await((t=r.contextProvider)==null?void 0:t.call(r,{transport:"HTTP",headers:a.headers,cookies:a.cookies,query:i}))??{};if(e.method==="GET"){let f=s[s.length-1],{success:d,data:m,error:p}=G.safeParse(i);if(!d)return Response.json({message:"Invalid query",code:"INVALID_QUERY",details:p},{status:400});let u=await r.handleRequest({req:{...a,type:"QUERY",resourceName:f,context:y,where:m.where,include:m.include,query:i}});return !u||!u.data?Response.json({message:"Invalid resource",code:"INVALID_RESOURCE"},{status:400}):Response.json(u.data)}if(e.method==="POST")try{let f=s[s.length-1],d=s[s.length-2],m=e.body?await e.json():{},p;if(f==="set"){let{success:b,data:x,error:j}=C.safeParse(m);if(!b)return Response.json({message:"Invalid mutation",code:"INVALID_REQUEST",details:j},{status:400});p=x;}else {let{success:b,data:x,error:j}=P.safeParse(m);if(!b)return Response.json({message:"Invalid mutation",code:"INVALID_REQUEST",details:j},{status:400});p=x;}let u=await r.handleRequest({req:{...a,type:"MUTATE",resourceName:d,input:p.payload,context:y,resourceId:p.resourceId,procedure:f!=="set"?f:void 0,query:{}}});return Response.json(u)}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 re=b(O(),1);var w=z$1.string(),we=z$1.object({id:w,type:z$1.literal("SUBSCRIBE"),resource:z$1.string()}),be=A.extend({id:w,type:z$1.literal("QUERY")}),Y=M.extend({id:w}),xe=S.extend({id:w}),Se=z$1.union([xe,Y]),J=z$1.union([we,be,Se]),Me=z$1.object({id:w,type:z$1.literal("REJECT"),resource:z$1.string(),message:z$1.string().optional()}),Ie=z$1.object({id:w,type:z$1.literal("REPLY"),data:z$1.any()});z$1.union([Me,Ie,Y]);z$1.object({resource:z$1.string(),data:z$1.record(z$1.string(),L)});var X="0123456789ABCDEFGHJKMNPQRSTVWXYZ",I=32;var Ee=16,ee=10,K=0xffffffffffff;var T;(function(r){r.Base32IncorrectEncoding="B32_ENC_INVALID",r.DecodeTimeInvalidCharacter="DEC_TIME_CHAR",r.DecodeTimeValueMalformed="DEC_TIME_MALFORMED",r.EncodeTimeNegative="ENC_TIME_NEG",r.EncodeTimeSizeExceeded="ENC_TIME_SIZE_EXCEED",r.EncodeTimeValueMalformed="ENC_TIME_MALFORMED",r.PRNGDetectFailure="PRNG_DETECT",r.ULIDInvalid="ULID_INVALID",r.Unexpected="UNEXPECTED",r.UUIDInvalid="UUID_INVALID";})(T||(T={}));var g=class extends Error{constructor(e,t){super(`${t} (${e})`),this.name="ULIDError",this.code=e;}};function ve(r){let e=Math.floor(r()*I);return e===I&&(e=I-1),X.charAt(e)}function Ae(r){var n;let e=$e(),t=e&&(e.crypto||e.msCrypto)||(typeof $<"u"?$:null);if(typeof(t==null?void 0:t.getRandomValues)=="function")return ()=>{let a=new Uint8Array(1);return t.getRandomValues(a),a[0]/255};if(typeof(t==null?void 0:t.randomBytes)=="function")return ()=>t.randomBytes(1).readUInt8()/255;if((n=$)!=null&&n.randomBytes)return ()=>$.randomBytes(1).readUInt8()/255;throw new g(T.PRNGDetectFailure,"Failed to find a reliable PRNG")}function $e(){return Le()?self:typeof window<"u"?window:typeof global<"u"?global:typeof globalThis<"u"?globalThis:null}function je(r,e){let t="";for(;r>0;r--)t=ve(e)+t;return t}function Oe(r,e=ee){if(isNaN(r))throw new g(T.EncodeTimeValueMalformed,`Time must be a number: ${r}`);if(r>K)throw new g(T.EncodeTimeSizeExceeded,`Cannot encode a time larger than ${K}: ${r}`);if(r<0)throw new g(T.EncodeTimeNegative,`Time must be positive: ${r}`);if(Number.isInteger(r)===false)throw new g(T.EncodeTimeValueMalformed,`Time must be an integer: ${r}`);let t,n="";for(let a=e;a>0;a--)t=r%I,n=X.charAt(t)+n,r=(r-t)/I;return n}function Le(){return typeof WorkerGlobalScope<"u"&&self instanceof WorkerGlobalScope}function te(r,e){let t=Ae(),n=Date.now();return Oe(n,ee)+je(Ee,t)}var _=()=>te().toLowerCase();var ne=r=>{let e={},t={};return r.subscribeToMutations(n=>{let a=n;!a.resourceId||!a.payload||(console.log("Mutation propagated:",a),Object.entries(t[a.resource]??{}).forEach(([o,s])=>{var c;(c=e[o])==null||c.send(JSON.stringify({...a,id:a.id??_()}));}));}),(n,a)=>{var f;let o=d=>{n.send(JSON.stringify(d));},s=_(),c={headers:a.headers,cookies:typeof a.headers.cookie=="string"?re.default.parse(a.headers.cookie):{}},i=parse(a.url.split("?")[1]),y=(f=r.contextProvider)==null?void 0:f.call(r,{transport:"WEBSOCKET",headers:c.headers,cookies:c.cookies,query:i});e[s]=n,console.log("Client connected:",s),n.on("message",async d=>{try{console.log("Message received from the client:",d);let m=J.parse(JSON.parse(d.toString()));if(m.type==="SUBSCRIBE"){let{resource:p}=m;t[p]||(t[p]={}),t[p][s]={};}else if(m.type==="QUERY"){let{resource:p}=m,u=await r.handleRequest({req:{...c,type:"QUERY",resourceName:p,context:await y??{},query:i}});if(!u||!u.data)throw new Error("Invalid resource");o({id:m.id,type:"REPLY",data:{resource:p,data:Object.fromEntries(Object.entries(u.data??{}).map(([b,x])=>[b,x.value]))}});}else if(m.type==="MUTATE"){let{resource:p}=m;console.log("Received mutation from client:",m);try{let u=await r.handleRequest({req:{...c,type:"MUTATE",resourceName:p,input:m.payload,context:{messageId:m.id,...await y??{}},resourceId:m.resourceId,procedure:m.procedure,query:i}});m.procedure&&o({id:m.id,type:"REPLY",data:u});}catch(u){o({id:m.id,type:"REJECT",resource:p,message:u.message}),console.error("Error parsing mutation from the client:",u);}}}catch(m){console.error("Error handling message from the client:",m);}}),n.on("close",()=>{console.log("Connection closed",s),delete e[s];for(let d of Object.values(t))delete d[s];});}};function ae(r){let e=`${r.protocol}://${r.hostname}${r.url}`,t=new Headers;return Object.entries(r.headers).forEach(([n,a])=>{a&&t.set(n,Array.isArray(a)?a.join(","):a);}),new Request(e,{method:r.method,headers:t,body:r.body&&r.method!=="GET"?JSON.stringify(r.body):void 0})}var pt=(r,e,t)=>{r.ws(`${(t==null?void 0:t.basePath)??""}/ws`,ne(e)),r.use(`${(t==null?void 0:t.basePath)??""}/`,(n,a)=>{W(e)(ae(n)).then(s=>s.json().then(c=>a.status(s.status).send(c)));});};var z=class r{routes;constructor(e){this.routes=e.routes;}static create(e){return new r(e)}},ht=r=>z.create({...r}),_e=r=>({handler:e=>({inputValidator:r??z$1.undefined(),handler:e})}),U=class r{_resourceSchema;resourceName;middlewares;customMutations;constructor(e,t){this.resourceName=e,this.middlewares=new Set,this.customMutations=t??{};}handleFind=async({req:e,db:t})=>({data:await t.rawFind(e.resourceName,e.where,e.include),acceptedValues:null});handleSet=async({req:e,db:t,schema:n})=>{if(!e.input)throw new Error("Payload is required");if(!e.resourceId)throw new Error("ResourceId is required");let a=await t.rawFindById(e.resourceName,e.resourceId),[o,s]=n[this.resourceName].mergeMutation("set",e.input,a);if(!s)throw new Error("Mutation rejected");return {data:await t.rawUpsert(e.resourceName,e.resourceId,o),acceptedValues:s}};async handleRequest(e){let t=n=>(()=>{if(n.type==="QUERY")return this.handleFind({req:n,db:e.db,schema:e.schema});if(n.type==="MUTATE")if(n.procedure){if(this.customMutations[n.procedure]){let a=this.customMutations[n.procedure].inputValidator.parse(n.input);return n.input=a,this.customMutations[n.procedure].handler({req:n,db:e.db,schema:e.schema})}}else return this.handleSet({req:n,db:e.db,schema:e.schema});throw new Error("Invalid request")})();return await Array.from(this.middlewares.values()).reduceRight((n,a)=>o=>a({req:o,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:_e}))}},N=class r{middlewares;constructor(e=[]){this.middlewares=e;}createBasicRoute(e){return new U(e.name).use(...this.middlewares)}use(...e){return new r([...this.middlewares,...e])}static create(){return new r}},Tt=N.create;var E=class{async insert(e,t){let n=new Date().toISOString();return r(await this.rawUpsert(e.name,t.id,{value:Object.fromEntries(Object.entries(t).map(([a,o])=>[a,{value:o,_meta:{timestamp:n}}]))}))}async update(e,t,n){let a=new Date().toISOString(),{id:o,...s}=n;return r(await this.rawUpsert(e.name,t,{value:Object.fromEntries(Object.entries(s).map(([c,i])=>[c,{value:i,_meta:{timestamp:a}}]))}))}};function D(r,e,t,n){var o,s;if(!n)return t;if(!r)throw new Error("Schema not initialized");let a=r[e];if(!a)throw new Error("Resource not found");for(let[c,i]of Object.entries(n))if(a.fields[c])(i==null?void 0:i.$eq)!==void 0?t=t.where(`${e}.${c}`,i.$eq===null?"is":"=",i.$eq):(i==null?void 0:i.$in)!==void 0?t=t.where(`${e}.${c}`,"in",i.$in):(i==null?void 0:i.$not)!==void 0?((o=i==null?void 0:i.$not)==null?void 0:o.$in)!==void 0?t=t.where(`${e}.${c}`,"not in",i.$not.$in):((s=i==null?void 0:i.$not)==null?void 0:s.$eq)!==void 0?t=t.where(`${e}.${c}`,i.$not.$eq===null?"is not":"!=",i.$not.$eq):t=t.where(`${e}.${c}`,i.$not===null?"is not":"!=",i.$not):(i==null?void 0:i.$gt)!==void 0?t=t.where(`${e}.${c}`,">",i.$gt):(i==null?void 0:i.$gte)!==void 0?t=t.where(`${e}.${c}`,">=",i.$gte):(i==null?void 0:i.$lt)!==void 0?t=t.where(`${e}.${c}`,"<",i.$lt):(i==null?void 0:i.$lte)!==void 0?t=t.where(`${e}.${c}`,"<=",i.$lte):t=t.where(`${e}.${c}`,i===null?"is":"=",i);else if(a.relations[c]){let y=a.relations[c],f=y.entity.name,d=y.type==="one"?"id":y.foreignColumn,m=y.type==="one"?y.relationalColumn:"id";t=t.leftJoin(f,`${f}.${d}`,`${e}.${m}`),t=D(r,f,t,i);}return t}function V(r,e,t,n){if(!n)return t;if(!r)throw new Error("Schema not initialized");let a=r[e];if(!a)throw new Error(`Resource not found: ${e}`);for(let o of Object.keys(n)){if(!a.relations[o])throw new Error(`Relation ${o} not found in resource ${e}`);let s=a.relations[o],c=s.entity.name,i=s.type==="one"?"id":s.foreignColumn,y=s.type==="one"?s.relationalColumn:"id",f=s.type==="one"?jsonObjectFrom:jsonArrayFrom;t=t.select(d=>f(d.selectFrom(c).selectAll(c).whereRef(`${c}.${i}`,"=",`${e}.${y}`).select(m=>jsonObjectFrom(m.selectFrom(`${c}_meta`).selectAll(`${c}_meta`).whereRef(`${c}_meta.id`,"=",`${c}.id`)).as("_meta"))).as(o));}return t}var k=class extends E{db;schema;constructor(e){super(),this.db=new Kysely({dialect:new PostgresDialect({pool:e})});}async updateSchema(e){this.schema=e;let t=await this.db.introspection.getTables();for(let[n,a]of Object.entries(e)){let o=t.find(i=>i.name===n);o||await this.db.schema.createTable(n).ifNotExists().execute();let s=`${n}_meta`,c=t.find(i=>i.name===s);c||await this.db.schema.createTable(s).ifNotExists().execute();for(let[i,y]of Object.entries(a.fields)){let f=o==null?void 0:o.columns.find(p=>p.name===i),d=y.getStorageFieldType();f?f.dataType!==d.type&&console.error("Column type mismatch:",i,"expected to have type:",d.type,"but has type:",f.dataType):(await this.db.schema.alterTable(n).addColumn(i,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",i,p),p}),d.index&&await this.db.schema.createIndex(`${n}_${i}_index`).on(n).column(i).execute().catch(p=>{})),(c==null?void 0:c.columns.find(p=>p.name===i))||await this.db.schema.alterTable(s).addColumn(i,"varchar",p=>{let u=p;return d.primary&&(u=u.primaryKey().references(`${n}.${i}`)),u}).execute();}}}async rawFindById(e,t,n){if(!this.schema)throw new Error("Schema not initialized");let a=await this.db.selectFrom(e).where("id","=",t).selectAll(e).select(s=>jsonObjectFrom(s.selectFrom(`${e}_meta`).selectAll(`${e}_meta`).whereRef(`${e}_meta.id`,"=",`${e}.id`)).as("_meta"));a=V(this.schema,e,a,n);let o=await a.executeTakeFirst();if(o)return this.convertToMaterializedLiveType(o)}async findOne(e,t,n){let a=await this.rawFindById(e.name,t,n==null?void 0:n.include);if(a)return r(a)}async rawFind(e,t,n){if(!this.schema)throw new Error("Schema not initialized");let a=this.db.selectFrom(e).selectAll(e).select(i=>jsonObjectFrom(i.selectFrom(`${e}_meta`).selectAll(`${e}_meta`).whereRef(`${e}_meta.id`,"=",`${e}.id`)).as("_meta"));a=D(this.schema,e,a,t),a=V(this.schema,e,a,n);let o=await a.execute(),s=Object.fromEntries(o.map(i=>{let{id:y,...f}=i;return [y,f]}));return Object.keys(s).length===0?{}:Object.entries(s).reduce((i,[y,f])=>(i[y]=this.convertToMaterializedLiveType(f),i),{})}async find(e,t){let n=await this.rawFind(e.name,t==null?void 0:t.where,t==null?void 0:t.include);return Object.fromEntries(Object.entries(n).map(([a,o])=>[a,r(o)]))}async rawUpsert(e,t,n){return await this.db.transaction().execute(async a=>{var i;let o=!!await a.selectFrom(e).select("id").where("id","=",t).executeTakeFirst(),s={},c={};for(let[y,f]of Object.entries(n.value)){let d=(i=f._meta)==null?void 0:i.timestamp;d&&(s[y]=f.value,c[y]=d);}o?await Promise.all([a.updateTable(e).set(s).where("id","=",t).execute(),a.updateTable(`${e}_meta`).set(c).where("id","=",t).execute()]):await Promise.all([a.insertInto(e).values({...s,id:t}).execute(),a.insertInto(`${e}_meta`).values({...c,id:t}).execute()]);}),n}convertToMaterializedLiveType(e){if(!e._meta)throw new Error("Missing _meta");return {value:Object.entries(e).reduce((t,[n,a])=>{var o,s,c;return n==="_meta"||(n==="id"?t[n]={value:a}:Array.isArray(a)?t[n]={value:a.map(i=>this.convertToMaterializedLiveType(i)),_meta:{timestamp:(o=e==null?void 0:e._meta)==null?void 0:o[n]}}:typeof a=="object"&&a!==null&&!(a instanceof Date)?t[n]={...this.convertToMaterializedLiveType(a),_meta:{timestamp:(s=e==null?void 0:e._meta)==null?void 0:s[n]}}:t[n]={value:a,_meta:{timestamp:(c=e==null?void 0:e._meta)==null?void 0:c[n]}}),t},{})}}};var F=class r{router;storage;schema;middlewares=new Set;contextProvider;mutationSubscriptions=new Set;constructor(e){var t;this.router=e.router,this.storage=e.storage,this.schema=e.schema,(t=e.middlewares)==null||t.forEach(n=>{this.middlewares.add(n);}),this.storage.updateSchema(this.schema),this.contextProvider=e.contextProvider;}static create(e){return new r(e)}subscribeToMutations(e){return this.mutationSubscriptions.add(e),()=>{this.mutationSubscriptions.delete(e);}}async handleRequest(e){if(!this.router.routes[e.req.resourceName])throw new Error("Invalid resource");let t=await Array.from(this.middlewares.values()).reduceRight((n,a)=>o=>a({req:o,next:n}),async n=>this.router.routes[e.req.resourceName].handleRequest({req:n,db:this.storage,schema:this.schema}))(e.req);return t&&e.req.type==="MUTATE"&&t.acceptedValues&&Object.keys(t.acceptedValues).length>0&&this.mutationSubscriptions.forEach(n=>{n({id:e.req.context.messageId,type:"MUTATE",resource:e.req.resourceName,payload:t.acceptedValues??{},resourceId:e.req.resourceId});}),t}use(e){return this.middlewares.add(e),this}context(e){return this.contextProvider=e,this}},Lt=F.create;
2
+ export{U as Route,N as RouteFactory,z as Router,k as SQLStorage,F as Server,E as Storage,pt as expressAdapter,Tt as routeFactory,ht as router,Lt as server};
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@live-state/sync",
3
- "version": "0.0.4-beta.1",
3
+ "version": "0.0.4-beta.2",
4
4
  "private": false,
5
5
  "publishConfig": {
6
6
  "access": "public"