@live-state/sync 0.0.5 → 0.0.6-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/server.cjs CHANGED
@@ -1,2 +1,6 @@
1
- 'use strict';require('js-xxhash');var V=require('crypto'),rt=require('qs'),zod=require('zod'),kysely=require('kysely'),postgres=require('kysely/helpers/postgres');function _interopDefault(e){return e&&e.__esModule?e:{default:e}}var V__default=/*#__PURE__*/_interopDefault(V);var rt__default=/*#__PURE__*/_interopDefault(rt);var je=Object.create;var ce=Object.defineProperty;var Ee=Object.getOwnPropertyDescriptor;var $e=Object.getOwnPropertyNames;var Ce=Object.getPrototypeOf,Pe=Object.prototype.hasOwnProperty;var Ne=(i,e)=>()=>(e||i((e={exports:{}}).exports,e),e.exports);var ze=(i,e,t,r)=>{if(e&&typeof e=="object"||typeof e=="function")for(let n of $e(e))!Pe.call(i,n)&&n!==t&&ce(i,n,{get:()=>e[n],enumerable:!(r=Ee(e,n))||r.enumerable});return i};var ue=(i,e,t)=>(t=i!=null?je(Ce(i)):{},ze(ce(t,"default",{value:i,enumerable:true}),i));var Q=Ne(K=>{Object.defineProperty(K,"__esModule",{value:true});K.parse=He;K.serialize=Je;var Ue=/^[\u0021-\u003A\u003C\u003E-\u007E]+$/,ke=/^[\u0021-\u003A\u003C-\u007E]*$/,Be=/^([.]?[a-z0-9]([a-z0-9-]{0,61}[a-z0-9])?)([.][a-z0-9]([a-z0-9-]{0,61}[a-z0-9])?)*$/i,Qe=/^[\u0020-\u003A\u003D-\u007E]*$/,Ge=Object.prototype.toString,Ze=(()=>{let i=function(){};return i.prototype=Object.create(null),i})();function He(i,e){let t=new Ze,r=i.length;if(r<2)return t;let n=(e==null?void 0:e.decode)||Ye,a=0;do{let s=i.indexOf("=",a);if(s===-1)break;let o=i.indexOf(";",a),c=o===-1?r:o;if(s>c){a=i.lastIndexOf(";",s-1)+1;continue}let m=fe(i,a,s),y=he(i,s,m),l=i.slice(m,y);if(t[l]===void 0){let p=fe(i,s+1,c),u=he(i,c,p),h=n(i.slice(p,u));t[l]=h;}a=c+1;}while(a<r);return t}function fe(i,e,t){do{let r=i.charCodeAt(e);if(r!==32&&r!==9)return e}while(++e<t);return t}function he(i,e,t){for(;e>t;){let r=i.charCodeAt(--e);if(r!==32&&r!==9)return e+1}return t}function Je(i,e,t){let r=(t==null?void 0:t.encode)||encodeURIComponent;if(!Ue.test(i))throw new TypeError(`argument name is invalid: ${i}`);let n=r(e);if(!ke.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(!Be.test(t.domain))throw new TypeError(`option domain is invalid: ${t.domain}`);a+="; Domain="+t.domain;}if(t.path){if(!Qe.test(t.path))throw new TypeError(`option path is invalid: ${t.path}`);a+="; Path="+t.path;}if(t.expires){if(!Xe(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 Ye(i){if(i.indexOf("%")===-1)return i;try{return decodeURIComponent(i)}catch{return i}}function Xe(i){return Ge.call(i)==="[object Date]"}});var E=(i,e,t)=>{let r={},n=t[e];if(!n)return r;let a=s=>{s.$and?s.$and.forEach(a):s.$or?s.$or.forEach(a):Object.entries(s).forEach(([o,c])=>{var m;if((m=n.relations)!=null&&m[o]&&(r[o]=true,typeof c=="object"&&c!==null&&!Array.isArray(c))){let y=E(c,n.relations[o].entity.name,t);Object.keys(y).length>0&&(r[o]=y);}});};return a(i),r},w=(i,e,t=false)=>Object.entries(e).every(([r,n])=>{if(r==="$and")return n.every(s=>w(i,s,t));if(r==="$or")return n.some(s=>w(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 o=i[r];return o===void 0?false:t?!n.$in.includes(o):n.$in.includes(o)}if(n.$not!==void 0&&!t)return w(i,{[r]:n.$not},true);if(n.$gt!==void 0){let o=i[r];return typeof o!="number"?false:t?o<=n.$gt:o>n.$gt}if(n.$gte!==void 0){let o=i[r];return typeof o!="number"?false:t?o<n.$gte:o>=n.$gte}if(n.$lt!==void 0){let o=i[r];return typeof o!="number"?false:t?o>=n.$lt:o<n.$lt}if(n.$lte!==void 0){let o=i[r];return typeof o!="number"?false:t?o>n.$lte:o<=n.$lte}let s=i[r];return !s||typeof s!="object"&&!Array.isArray(s)?false:Array.isArray(s)?t?!s.some(o=>w(o,n,false)):s.some(o=>w(o,n,false)):w(s,n,t)}return t?i[r]!==a:i[r]===a}),S={CRITICAL:0,ERROR:1,WARN:2,INFO:3,DEBUG:4},k=class{level;prefix;constructor(e={}){this.level=e.level??S.INFO,this.prefix=e.prefix?`[${e.prefix}] `:"";}critical(...e){this.level>=S.CRITICAL&&console.error(`${this.prefix}[CRITICAL]`,...e);}error(...e){this.level>=S.ERROR&&console.error(`${this.prefix}[ERROR]`,...e);}warn(...e){this.level>=S.WARN&&console.warn(`${this.prefix}[WARN]`,...e);}info(...e){this.level>=S.INFO&&console.log(`${this.prefix}[INFO]`,...e);}debug(...e){this.level>=S.DEBUG&&console.log(`${this.prefix}[DEBUG]`,...e);}setLevel(e){this.level=e;}getLevel(){return this.level}},le=i=>new k(i);var ye="0123456789ABCDEFGHJKMNPQRSTVWXYZ",$=32;var Ve=16,pe=10,de=0xffffffffffff;var L;(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";})(L||(L={}));var M=class extends Error{constructor(e,t){super(`${t} (${e})`),this.name="ULIDError",this.code=e;}};function _e(i){let e=Math.floor(i()*$);return e===$&&(e=$-1),ye.charAt(e)}function Fe(i){var r;let e=Ke(),t=e&&(e.crypto||e.msCrypto)||(typeof V__default.default<"u"?V__default.default: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=V__default.default)!=null&&r.randomBytes)return ()=>V__default.default.randomBytes(1).readUInt8()/255;throw new M(L.PRNGDetectFailure,"Failed to find a reliable PRNG")}function Ke(){return De()?self:typeof window<"u"?window:typeof global<"u"?global:typeof globalThis<"u"?globalThis:null}function We(i,e){let t="";for(;i>0;i--)t=_e(e)+t;return t}function qe(i,e=pe){if(isNaN(i))throw new M(L.EncodeTimeValueMalformed,`Time must be a number: ${i}`);if(i>de)throw new M(L.EncodeTimeSizeExceeded,`Cannot encode a time larger than ${de}: ${i}`);if(i<0)throw new M(L.EncodeTimeNegative,`Time must be positive: ${i}`);if(Number.isInteger(i)===false)throw new M(L.EncodeTimeValueMalformed,`Time must be an integer: ${i}`);let t,r="";for(let n=e;n>0;n--)t=i%$,r=ye.charAt(t)+r,i=(i-t)/$;return r}function De(){return typeof WorkerGlobalScope<"u"&&self instanceof WorkerGlobalScope}function me(i,e){let t=Fe(),r=Date.now();return qe(r,pe)+We(Ve,t)}var B=()=>me().toLowerCase();var _=(...i)=>{let e=i.filter(t=>!!t);return e.length===0?{}:e.length===1?e[0]:{$and:e}};var F=class{storage;queue=new Map;scheduled=false;constructor(e){this.storage=e;}async rawFind({resource:e,commonWhere:t,uniqueWhere:r,...n}){return new Promise((a,s)=>{let o=this.getBatchKey({resource:e,commonWhere:t,...n}),c={resource:e,commonWhere:t,uniqueWhere:r,...n,resolve:a,reject:s};this.queue.has(o)||this.queue.set(o,[]);let m=this.queue.get(o);m&&m.push(c),this.scheduled||(this.scheduled=true,setImmediate(()=>{this.processBatch();}));})}getBatchKey(e){let{resource:t,commonWhere:r,...n}=e;return `${t}:${JSON.stringify(r??{})}:${JSON.stringify(n??{})}`}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(r){t.forEach(n=>{n.reject(r);});}}async executeBatchedRequests(e){var p,u;if(e.length===0)return;let t=e[0],{resource:r,commonWhere:n,include:a,sort:s}=t,o=e.length===1?t.limit:void 0,c=e.map(h=>h.uniqueWhere).filter(h=>h!==void 0),m=n,y=(p=Object.entries(c[0]??{})[0])==null?void 0:p[0];if(c.length>0){let h=c.map(d=>d[y]).filter(d=>d!=null);h.length>0&&(m=_(n,{[y]:{$in:h}}));}let l=await this.storage.rawFind({resource:r,where:m,include:a,sort:s,limit:o});for(let h of e){let d={};if(h.uniqueWhere){let[f,g]=Object.entries(h.uniqueWhere)[0];for(let[x,v]of Object.entries(l))((u=v.value[f])==null?void 0:u.value)===g&&(d[x]=v);}else Object.assign(d,l);h.resolve(d);}}};var Re=ue(Q());var W=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()}),G=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()})),et=G.superRefine((i,e)=>{i.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(),resourceId:zod.z.string().optional()}),C=Te.extend({procedure:zod.z.string(),payload:zod.z.any().optional()}),P=Te.extend({procedure:zod.z.enum(["INSERT","UPDATE"]),payload:et});zod.z.union([P,C]);var ge=W.omit({resource:true}),Z=C.omit({id:true,type:true,resource:true,procedure:true}),H=P.omit({id:true,type:true,resource:true,procedure:true});zod.z.union([H,Z]);var xe=i=>{let e=i.logger;return async t=>{var r;try{let n=typeof t.headers.getSetCookie=="function"?Object.fromEntries(t.headers):t.headers,a={headers:n,cookies:n.cookie?Re.default.parse(n.cookie):{}},s=new URL(t.url),o=s.pathname.split("/"),c=s.searchParams,m=rt__default.default.parse(c.toString()),y=await((r=i.contextProvider)==null?void 0:r.call(i,{transport:"HTTP",headers:a.headers,cookies:a.cookies,queryParams:m}))??{};if(t.method==="GET"){let l=o[o.length-1],{success:p,data:u,error:h}=ge.safeParse(m);if(!p)return Response.json({message:"Invalid query",code:"INVALID_QUERY",details:h},{status:400});let d=await i.handleQuery({req:{...a,...u,type:"QUERY",resource:l,context:y,queryParams:m}});return !d||!d.data?Response.json({message:"Invalid resource",code:"INVALID_RESOURCE"},{status:400}):Response.json(d.data)}if(t.method==="POST")try{let l=o[o.length-1],p=o[o.length-2],u=t.body?await t.json():{},h;if(l==="insert"||l==="update"){let{success:f,data:g,error:x}=H.safeParse(u);if(!f)return Response.json({message:"Invalid mutation",code:"INVALID_REQUEST",details:x},{status:400});h=g;}else {let{success:f,data:g,error:x}=Z.safeParse(u);if(!f)return Response.json({message:"Invalid mutation",code:"INVALID_REQUEST",details:x},{status:400});h=g;}let d=await i.handleMutation({req:{...a,type:"MUTATE",resource:p,input:h.payload,context:y,resourceId:h.resourceId,procedure:l==="insert"||l==="update"?l.toUpperCase():l,queryParams:{}}});return Response.json(d)}catch(l){return e.error("Error parsing mutation from the client:",l),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 e.error("Unexpected error:",n),Response.json({message:"Internal server error",code:"INTERNAL_SERVER_ERROR"},{status:500})}}};var we=ue(Q());var A=zod.z.string(),nt=zod.z.object({id:A,type:zod.z.literal("SUBSCRIBE"),resource:zod.z.string()}),it=W.extend({id:A,type:zod.z.literal("QUERY")}),be=P.extend({id:A}),at=C.extend({id:A}),ot=zod.z.union([at,be]),ve=zod.z.union([nt,it,ot]),st=zod.z.object({id:A,type:zod.z.literal("REJECT"),resource:zod.z.string(),message:zod.z.string().optional()}),ct=zod.z.object({id:A,type:zod.z.literal("REPLY"),data:zod.z.any()});zod.z.union([st,ct,be]);zod.z.object({resource:zod.z.string(),data:zod.z.record(zod.z.string(),G)});var Se=i=>{let e={},t={},r=i.logger;return i.subscribeToMutations(n=>{let a=n;!a.resourceId||!a.payload||(r.debug("Mutation propagated:",a),Object.entries(t[a.resource]??{}).forEach(([s,o])=>{var c;(c=e[s])==null||c.send(JSON.stringify({...a,id:a.id??B()}));}));}),(n,a)=>{var l;let s=p=>{n.send(JSON.stringify(p));},o=B(),c={headers:a.headers,cookies:typeof a.headers.cookie=="string"?we.default.parse(a.headers.cookie):{}},m=rt.parse(a.url.split("?")[1]),y=(l=i.contextProvider)==null?void 0:l.call(i,{transport:"WEBSOCKET",headers:c.headers,cookies:c.cookies,queryParams:m});e[o]=n,r.info("Client connected:",o),n.on("message",async p=>{try{r.debug("Message received from the client:",p);let u=ve.parse(JSON.parse(p.toString()));if(u.type==="SUBSCRIBE"){let{resource:h}=u;t[h]||(t[h]={}),t[h][o]={};}else if(u.type==="QUERY"){let{resource:h}=u,d=await i.handleQuery({req:{...c,type:"QUERY",resource:h,context:await y??{},queryParams:m}});if(!d||!d.data)throw new Error("Invalid resource");s({id:u.id,type:"REPLY",data:{resource:h,data:Object.fromEntries(Object.entries(d.data??{}).map(([f,g])=>[f,g.value]))}});}else if(u.type==="MUTATE"){let{resource:h}=u;r.debug("Received mutation from client:",u);try{let d=await i.handleMutation({req:{...c,type:"MUTATE",resource:h,input:u.payload,context:{messageId:u.id,...await y??{}},resourceId:u.resourceId,procedure:u.procedure,queryParams:m}});u.procedure&&u.procedure!=="INSERT"&&u.procedure!=="UPDATE"&&s({id:u.id,type:"REPLY",data:d});}catch(d){s({id:u.id,type:"REJECT",resource:h,message:d.message}),r.error("Error parsing mutation from the client:",d);}}}catch(u){r.error("Error handling message from the client:",u);}}),n.on("close",()=>{r.info("Connection closed",o),delete e[o];for(let p of Object.values(t))delete p[o];});}};function Ie(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 Zt=(i,e,t)=>{i.ws(`${(t==null?void 0:t.basePath)??""}/ws`,Se(e)),i.use(`${(t==null?void 0:t.basePath)??""}/`,(r,n)=>{xe(e)(Ie(r)).then(s=>s.json().then(o=>n.status(s.status).send(o)));});};var b=i=>i?Array.isArray(i.value)?i.value.map(t=>b(t)):typeof i.value!="object"||i.value===null||i.value instanceof Date?i.value:Object.fromEntries(Object.entries(i.value).map(([t,r])=>Array.isArray(r)?[t,r.map(n=>b(n))]:[t,b(r)])):void 0;var re=class i{routes;constructor(e){this.routes=e.routes;}static create(e){return new i(e)}},fr=i=>re.create({...i}),dt=i=>({handler:e=>({inputValidator:i??zod.z.undefined(),handler:e})}),ne=class i{resourceSchema;middlewares;customMutations;authorization;constructor(e,t,r){this.resourceSchema=e,this.middlewares=new Set,this.customMutations=t??{},this.authorization=r;}use(...e){for(let t of e)this.middlewares.add(t);return this}withMutations(e){return new i(this.resourceSchema,e({mutation:dt}),this.authorization)}handleQuery=async({req:e,batcher:t})=>await this.wrapInMiddlewares(async r=>{var a,s;let n=(s=(a=this.authorization)==null?void 0:a.read)==null?void 0:s.call(a,{ctx:r.context});if(typeof n=="boolean"&&!n)throw new Error("Not authorized");return {data:await t.rawFind({resource:r.resource,commonWhere:_(r.where,typeof n=="object"?n:void 0),uniqueWhere:r.relationalWhere,include:r.include,limit:r.limit,sort:r.sort})}})(e);handleMutation=async({req:e,db:t,schema:r})=>await this.wrapInMiddlewares(async n=>{if(!n.procedure)throw new Error("Procedure is required for mutations");let a=this.customMutations[n.procedure];if(a){let s=a.inputValidator.parse(n.input);return n.input=s,a.handler({req:n,db:t})}else {if(n.procedure==="INSERT"||n.procedure==="UPDATE")return this.handleSet({req:n,db:t,operation:n.procedure,schema:r});throw new Error(`Unknown procedure: ${n.procedure}`)}})(e);handleSet=async({req:e,db:t,operation:r,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.resource,e.resourceId);if(r==="INSERT"&&a)throw new Error("Resource already exists");if(r==="UPDATE"&&!a)throw new Error("Resource not found");return t.transaction(async({trx:s})=>{var y,l,p,u,h;let[o,c]=this.resourceSchema.mergeMutation("set",e.input,a);if(!c)throw new Error("Mutation rejected");if(r==="INSERT"){let d=await s.rawInsert(e.resource,e.resourceId,o),f=b(d);if(f.id=f.id??e.resourceId,(y=this.authorization)!=null&&y.insert){let g=this.authorization.insert({ctx:e.context,value:f});if(typeof g=="boolean"){if(!g)throw new Error("Not authorized")}else {let x=E(g,e.resource,n),v=Object.keys(x).length>0?await s.rawFindById(e.resource,e.resourceId,x):d,I=b(v);if(I.id=I.id??e.resourceId,!w(I,g))throw new Error("Not authorized")}}return {data:d,acceptedValues:c}}if((p=(l=this.authorization)==null?void 0:l.update)!=null&&p.preMutation){let d=b(a);d.id=d.id??e.resourceId;let f=this.authorization.update.preMutation({ctx:e.context,value:d});if(typeof f=="boolean"){if(!f)throw new Error("Not authorized")}else {let g=E(f,e.resource,n),x=Object.keys(g).length>0?await s.rawFindById(e.resource,e.resourceId,g):a,v=b(x);if(v.id=v.id??e.resourceId,!w(v,f))throw new Error("Not authorized")}}let m=await s.rawUpdate(e.resource,e.resourceId,o);if((h=(u=this.authorization)==null?void 0:u.update)!=null&&h.postMutation){let d=b(m);d.id=d.id??e.resourceId;let f=this.authorization.update.postMutation({ctx:e.context,value:d});if(typeof f=="boolean"){if(!f)throw new Error("Not authorized")}else {let g=E(f,e.resource,n),x=Object.keys(g).length>0?await s.rawFindById(e.resource,e.resourceId,g):m,v=b(x);if(v.id=v.id??e.resourceId,!w(v,f))throw new Error("Not authorized")}}return {data:m,acceptedValues:c}})};wrapInMiddlewares(e){return t=>Array.from(this.middlewares.values()).reduceRight((r,n)=>a=>n({req:a,next:r}),e)(t)}},ie=class i{middlewares;constructor(e=[]){this.middlewares=e;}collectionRoute(e,t){return new ne(e,void 0,t).use(...this.middlewares)}use(...e){return new i([...this.middlewares,...e])}static create(){return new i}},hr=ie.create;var z=class{async insert(e,t){let r=new Date().toISOString();return b(await this.rawInsert(e.name,t.id,{value:Object.fromEntries(Object.entries(t).map(([n,a])=>[n,{value:a,_meta:{timestamp:r}}]))}))}async update(e,t,r){let n=new Date().toISOString(),{id:a,...s}=r;return b(await this.rawUpdate(e.name,t,{value:Object.fromEntries(Object.entries(s).map(([o,c])=>[o,{value:c,_meta:{timestamp:n}}]))}))}};function q(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(o=>q(i,e,t,o)):s?r.$and.map(o=>q(i,e,t,o)):Object.entries(r).map(([o,c])=>{var m,y;if(n.fields[o])return (c==null?void 0:c.$eq)!==void 0?t(`${e}.${o}`,c.$eq===null?"is":"=",c.$eq):(c==null?void 0:c.$in)!==void 0?t(`${e}.${o}`,"in",c.$in):(c==null?void 0:c.$not)!==void 0?((m=c==null?void 0:c.$not)==null?void 0:m.$in)!==void 0?t(`${e}.${o}`,"not in",c.$not.$in):((y=c==null?void 0:c.$not)==null?void 0:y.$eq)!==void 0?t(`${e}.${o}`,c.$not.$eq===null?"is not":"!=",c.$not.$eq):t(`${e}.${o}`,c.$not===null?"is not":"!=",c.$not):(c==null?void 0:c.$gt)!==void 0?t(`${e}.${o}`,">",c.$gt):(c==null?void 0:c.$gte)!==void 0?t(`${e}.${o}`,">=",c.$gte):(c==null?void 0:c.$lt)!==void 0?t(`${e}.${o}`,"<",c.$lt):(c==null?void 0:c.$lte)!==void 0?t(`${e}.${o}`,"<=",c.$lte):t(`${e}.${o}`,c===null?"is":"=",c);if(n.relations[o]){let l=n.relations[o],p=l.entity.name;return l.type==="many"?t.exists(ae(i,p,t.selectFrom(p).select("id").whereRef(l.foreignColumn,"=",`${e}.id`),c)):q(i,p,t,c)}return null}).filter(Boolean))}function D(i,e,t,r){let n=i[e];if(!n)throw new Error("Resource not found");if(!r)return t;if(r.$and){for(let a of r.$and)t=D(i,e,t,a);return t}else if(r.$or){for(let a of r.$or)t=D(i,e,t,a);return t}for(let[a,s]of Object.entries(r)){if(!n.relations[a])continue;let o=n.relations[a],c=o.entity.name,m=o.type==="one"?"id":o.foreignColumn,y=o.type==="one"?o.relationalColumn:"id";t=t.leftJoin(c,`${c}.${m}`,`${e}.${y}`),s instanceof Object&&!Array.isArray(s)&&s!==null&&(t=D(i,c,t,s));}return t}function ae(i,e,t,r){return !r||Object.keys(r).length===0?t:(t=D(i,e,t,r),t.where(n=>q(i,e,n,r)))}function U(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],o=s.entity.name,c=r[a],m=s.type==="one"?"id":s.foreignColumn,y=s.type==="one"?s.relationalColumn:"id",l=s.type==="one"?postgres.jsonObjectFrom:postgres.jsonArrayFrom,p=typeof c=="object"&&c!==null;t=t.select(u=>{let h=u.selectFrom(o).selectAll(o).whereRef(`${o}.${m}`,"=",`${e}.${y}`).select(d=>postgres.jsonObjectFrom(d.selectFrom(`${o}_meta`).selectAll(`${o}_meta`).whereRef(`${o}_meta.id`,"=",`${o}.id`)).as("_meta"));return p&&(h=U(i,o,h,c)),l(h).as(a)});}return t}var oe=class i extends z{db;schema;logger;constructor(e,t,r){super(),this.isKyselyLike(e)?this.db=e:this.db=new kysely.Kysely({dialect:new kysely.PostgresDialect({pool:e})}),this.schema=t,this.logger=r,this.rawInsert=this.rawInsert.bind(this),this.rawUpdate=this.rawUpdate.bind(this);}async init(e,t){var n;this.schema=e,this.logger=t;let r=await this.db.introspection.getTables();for(let[a,s]of Object.entries(e)){let o=r.find(y=>y.name===a);o||await this.db.schema.createTable(a).ifNotExists().execute();let c=`${a}_meta`,m=r.find(y=>y.name===c);m||await this.db.schema.createTable(c).ifNotExists().execute();for(let[y,l]of Object.entries(s.fields)){let p=o==null?void 0:o.columns.find(d=>d.name===y),u=l.getStorageFieldType();p?p.dataType!==u.type&&((n=this.logger)==null||n.warn("Column type mismatch:",y,"expected to have type:",u.type,"but has type:",p.dataType)):(await this.db.schema.alterTable(a).addColumn(y,u.type,d=>{let f=d;return u.unique&&(f=f.unique()),u.nullable||(f=f.notNull()),u.references&&(f=f.references(u.references)),u.primary&&(f=f.primaryKey()),u.default!==void 0&&(f=f.defaultTo(u.default)),f}).execute().catch(d=>{var f;throw (f=this.logger)==null||f.error("Error adding column",y,d),d}),u.index&&await this.db.schema.createIndex(`${a}_${y}_index`).on(a).column(y).execute().catch(()=>{})),(m==null?void 0:m.columns.find(d=>d.name===y))||await this.db.schema.alterTable(c).addColumn(y,"varchar",d=>{let f=d;return u.primary&&(f=f.primaryKey().references(`${a}.${y}`)),f}).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=>postgres.jsonObjectFrom(s.selectFrom(`${e}_meta`).selectAll(`${e}_meta`).whereRef(`${e}_meta.id`,"=",`${e}.id`)).as("_meta"));n=U(this.schema,e,n,r);let a=await n.executeTakeFirst();if(a)return this.convertToMaterializedLiveType(a)}async findOne(e,t,r){let n=await this.rawFindById(e.name,t,r==null?void 0:r.include);if(n)return b(n)}async rawFind(e){if(!this.schema)throw new Error("Schema not initialized");let{resource:t,where:r,include:n,limit:a,sort:s}=e,o=this.db.selectFrom(t).selectAll(t).select(l=>postgres.jsonObjectFrom(l.selectFrom(`${t}_meta`).selectAll(`${t}_meta`).whereRef(`${t}_meta.id`,"=",`${t}.id`)).as("_meta"));o=ae(this.schema,t,o,r),o=U(this.schema,t,o,n),a!==void 0&&(o=o.limit(a)),s!==void 0&&s.forEach(l=>{o=o.orderBy(l.key,l.direction);});let c=await o.execute(),m=Object.fromEntries(c.map(l=>{let{id:p}=l;return [p,l]}));return Object.keys(m).length===0?{}:Object.entries(m).reduce((l,[p,u])=>(l[p]=this.convertToMaterializedLiveType(u),l),{})}async find(e,t){let r=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(r).map(([n,a])=>[n,b(a)]))}async rawInsert(e,t,r){var s;let n={},a={};for(let[o,c]of Object.entries(r.value)){let m=(s=c._meta)==null?void 0:s.timestamp;m&&(n[o]=c.value,a[o]=m);}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[o,c]of Object.entries(r.value)){let m=(s=c._meta)==null?void 0:s.timestamp;m&&(n[o]=c.value,a[o]=m);}return await Promise.all([this.db.updateTable(e).set(n).where("id","=",t).execute(),this.db.insertInto(`${e}_meta`).values({...a,id:t}).onConflict(o=>o.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,this.logger),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,o;return r==="_meta"||(r==="id"?t[r]={value:n}:Array.isArray(n)?t[r]={value:n.map(c=>this.convertToMaterializedLiveType(c)),_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:(o=e==null?void 0:e._meta)==null?void 0:o[r]}}),t},{})}}isKyselyLike(e){if(e instanceof kysely.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 se=class i{router;storage;schema;middlewares=new Set;logger;contextProvider;mutationSubscriptions=new Set;constructor(e){var t;this.router=e.router,this.storage=e.storage,this.schema=e.schema,this.logger=le({level:e.logLevel??S.INFO}),(t=e.middlewares)==null||t.forEach(r=>{this.middlewares.add(r);}),this.storage.init(this.schema,this.logger),this.contextProvider=e.contextProvider;}static create(e){return new i(e)}subscribeToMutations(e){return this.mutationSubscriptions.add(e),()=>{this.mutationSubscriptions.delete(e);}}handleQuery(e){let t=new F(this.storage);return this.wrapInMiddlewares(async r=>{var m;let n=Ae(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},s={};for(let y=0;y<n.length;y++){let l=n[y],p=this.router.routes[l.resource];if(!p)throw new Error("Invalid resource");let u=l.getWhere&&l.referenceGetter?l.referenceGetter(s).map(l.getWhere):[void 0],h=(m=s[l.prevStepId??""])==null?void 0:m.flatMap(g=>{var x;return Object.keys(((x=g==null?void 0:g.result)==null?void 0:x.data)??{})}),f=(await Promise.allSettled(u.map(async(g,x)=>{let v=h==null?void 0:h[x],I=await p.handleQuery({req:{type:"QUERY",...l,...a,where:l.where,relationalWhere:g},batcher:t});return {includedBy:v,result:I}}))).flatMap(g=>g.status==="fulfilled"?[g.value]:[]);s[l.stepId]=f;}let o=Object.fromEntries(Object.entries(s).flatMap(([y,l],p)=>l.flatMap(u=>Object.entries(u.result.data).map(([h,d])=>[`${y}.${h}`,{data:d,includedBy:y!=="query"&&u.includedBy?`${y.split(".").slice(0,-1).join(".")}.${u.includedBy}`:void 0,path:y.split(".").slice(-1)[0],isMany:n[p].isMany,collectionName:n[p].collectionName,included:n[p].included}]))));return Object.keys(o).reduceRight((y,l)=>{var h,d;let p=o[l],u=p.path;if(u==="query"&&(y.data[l.replace("query.","")]=p.data),p.included.length)for(let f of p.included)p.data.value[f]??=((d=(h=this.schema[p.collectionName])==null?void 0:h.relations[f])==null?void 0:d.type)==="many"?{value:[]}:{value:null};if(p.includedBy){let f=o[p.includedBy];if(!f)return y;p.isMany?(f.data.value[u]??={value:[]},f.data.value[u].value.push(p.data)):f.data.value[u]=p.data;}return y},{data:{}})})(e.req)}async handleMutation(e){let t=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,schema:this.schema})})(e.req);if(t&&e.req.type==="MUTATE"&&t.acceptedValues&&(e.req.procedure==="INSERT"||e.req.procedure==="UPDATE")&&e.req.resourceId){let n=t.acceptedValues??{},a=e.req,s=a.resourceId;Object.keys(n).length&&s&&this.mutationSubscriptions.forEach(o=>{o({id:e.req.context.messageId,type:"MUTATE",resource:a.resource,payload:n,resourceId:s,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((r,n)=>a=>n({req:a,next:r}),e)(t)}},Cr=se.create;function Ae(i,e,t){let{include:r,where:n,...a}=i,{stepId:s}=t,o=[{...a,...t,where:n}];if(r&&typeof r=="object"&&Object.keys(r).length>0){let c=e[a.resource];if(!c)throw new Error(`Resource ${a.resource} not found`);o.push(...Object.entries(r).flatMap(([m,y])=>{let l=c.relations[m];if(!l)throw new Error(`Relation ${m} not found for resource ${a.resource}`);let p=l.entity.name;return Ae({...a,resource:p,include:y},e,{getWhere:l.type==="one"?u=>({id:u}):u=>({[l.foreignColumn]:u}),referenceGetter:u=>u[s].flatMap(h=>h.result.data?l.type==="one"?Object.values(h.result.data).map(d=>{var f,g;return (g=(f=d.value)==null?void 0:f[l.relationalColumn])==null?void 0:g.value}):Object.keys(h.result.data):[]),stepId:`${s}.${m}`,prevStepId:s,isMany:l.type==="many",collectionName:p,included:typeof y=="object"?Object.keys(y):[]})}));}return o}
2
- exports.Route=ne;exports.RouteFactory=ie;exports.Router=re;exports.SQLStorage=oe;exports.Server=se;exports.Storage=z;exports.expressAdapter=Zt;exports.routeFactory=hr;exports.router=fr;exports.server=Cr;
1
+ 'use strict';var W=require('crypto'),jsXxhash=require('js-xxhash'),gt=require('qs'),zod=require('zod'),kysely=require('kysely'),mysql=require('kysely/helpers/mysql'),postgres=require('kysely/helpers/postgres'),sqlite=require('kysely/helpers/sqlite');function _interopDefault(e){return e&&e.__esModule?e:{default:e}}var W__default=/*#__PURE__*/_interopDefault(W);var gt__default=/*#__PURE__*/_interopDefault(gt);var _e=Object.create;var ve=Object.defineProperty;var Be=Object.getOwnPropertyDescriptor;var We=Object.getOwnPropertyNames;var Ue=Object.getPrototypeOf,He=Object.prototype.hasOwnProperty;var Ge=(i,e)=>()=>(e||i((e={exports:{}}).exports,e),e.exports);var Je=(i,e,t,n)=>{if(e&&typeof e=="object"||typeof e=="function")for(let r of We(e))!He.call(i,r)&&r!==t&&ve(i,r,{get:()=>e[r],enumerable:!(n=Be(e,r))||n.enumerable});return i};var Se=(i,e,t)=>(t=i!=null?_e(Ue(i)):{},Je(ve(t,"default",{value:i,enumerable:true}),i));var ue=Ge(Z=>{Object.defineProperty(Z,"__esModule",{value:true});Z.parse=dt;Z.serialize=yt;var st=/^[\u0021-\u003A\u003C\u003E-\u007E]+$/,at=/^[\u0021-\u003A\u003C-\u007E]*$/,ot=/^([.]?[a-z0-9]([a-z0-9-]{0,61}[a-z0-9])?)([.][a-z0-9]([a-z0-9-]{0,61}[a-z0-9])?)*$/i,ct=/^[\u0020-\u003A\u003D-\u007E]*$/,ut=Object.prototype.toString,lt=(()=>{let i=function(){};return i.prototype=Object.create(null),i})();function dt(i,e){let t=new lt,n=i.length;if(n<2)return t;let r=(e==null?void 0:e.decode)||pt,o=0;do{let c=i.indexOf("=",o);if(c===-1)break;let a=i.indexOf(";",o),s=a===-1?n:a;if(c>s){o=i.lastIndexOf(";",c-1)+1;continue}let u=je(i,o,c),d=Oe(i,c,u),l=i.slice(u,d);if(t[l]===void 0){let y=je(i,c+1,s),p=Oe(i,s,y),h=r(i.slice(y,p));t[l]=h;}o=s+1;}while(o<n);return t}function je(i,e,t){do{let n=i.charCodeAt(e);if(n!==32&&n!==9)return e}while(++e<t);return t}function Oe(i,e,t){for(;e>t;){let n=i.charCodeAt(--e);if(n!==32&&n!==9)return e+1}return t}function yt(i,e,t){let n=(t==null?void 0:t.encode)||encodeURIComponent;if(!st.test(i))throw new TypeError(`argument name is invalid: ${i}`);let r=n(e);if(!at.test(r))throw new TypeError(`argument val is invalid: ${e}`);let o=i+"="+r;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(!ot.test(t.domain))throw new TypeError(`option domain is invalid: ${t.domain}`);o+="; Domain="+t.domain;}if(t.path){if(!ct.test(t.path))throw new TypeError(`option path is invalid: ${t.path}`);o+="; Path="+t.path;}if(t.expires){if(!ft(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 pt(i){if(i.indexOf("%")===-1)return i;try{return decodeURIComponent(i)}catch{return i}}function ft(i){return ut.call(i)==="[object Date]"}});var x=i=>i?Array.isArray(i.value)?i.value.map(t=>x(t)):typeof i.value!="object"||i.value===null||i.value instanceof Date?i.value:Object.fromEntries(Object.entries(i.value).map(([t,n])=>Array.isArray(n)?[t,n.map(r=>x(r))]:[t,x(n)])):void 0;var we="0123456789ABCDEFGHJKMNPQRSTVWXYZ",Q=32;var Ze=16,Ie=10,xe=0xffffffffffff;var A;(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";})(A||(A={}));var E=class extends Error{constructor(e,t){super(`${t} (${e})`),this.name="ULIDError",this.code=e;}};function Ye(i){let e=Math.floor(i()*Q);return e===Q&&(e=Q-1),we.charAt(e)}function Xe(i){var n;let e=et(),t=e&&(e.crypto||e.msCrypto)||(typeof W__default.default<"u"?W__default.default:null);if(typeof(t==null?void 0:t.getRandomValues)=="function")return ()=>{let r=new Uint8Array(1);return t.getRandomValues(r),r[0]/255};if(typeof(t==null?void 0:t.randomBytes)=="function")return ()=>t.randomBytes(1).readUInt8()/255;if((n=W__default.default)!=null&&n.randomBytes)return ()=>W__default.default.randomBytes(1).readUInt8()/255;throw new E(A.PRNGDetectFailure,"Failed to find a reliable PRNG")}function et(){return nt()?self:typeof window<"u"?window:typeof global<"u"?global:typeof globalThis<"u"?globalThis:null}function tt(i,e){let t="";for(;i>0;i--)t=Ye(e)+t;return t}function rt(i,e=Ie){if(isNaN(i))throw new E(A.EncodeTimeValueMalformed,`Time must be a number: ${i}`);if(i>xe)throw new E(A.EncodeTimeSizeExceeded,`Cannot encode a time larger than ${xe}: ${i}`);if(i<0)throw new E(A.EncodeTimeNegative,`Time must be positive: ${i}`);if(Number.isInteger(i)===false)throw new E(A.EncodeTimeValueMalformed,`Time must be an integer: ${i}`);let t,n="";for(let r=e;r>0;r--)t=i%Q,n=we.charAt(t)+n,i=(i-t)/Q;return n}function nt(){return typeof WorkerGlobalScope<"u"&&self instanceof WorkerGlobalScope}function Me(i,e){let t=Xe(),n=Date.now();return rt(n,Ie)+tt(Ze,t)}var H=()=>Me().toLowerCase(),U=i=>({then(e,t){try{if(e){let n=e(i);return n instanceof Promise?n:U(n)}return U(i)}catch(n){if(t){let r=t(n);return r instanceof Promise?r:U(r)}throw n}}}),ae=i=>i instanceof Promise?i:U(i);var q=(...i)=>{let e=i.filter(t=>!!t);return e.length===0?{}:e.length===1?e[0]:{$and:e}};var G=class{storage;queue=new Map;scheduled=false;constructor(e){this.storage=e;}async rawFind({resource:e,commonWhere:t,uniqueWhere:n,...r}){return new Promise((o,c)=>{let a=this.getBatchKey({resource:e,commonWhere:t,...r}),s={resource:e,commonWhere:t,uniqueWhere:n,...r,resolve:o,reject:c};this.queue.has(a)||this.queue.set(a,[]);let u=this.queue.get(a);u&&u.push(s),this.scheduled||(this.scheduled=true,setImmediate(()=>{this.processBatch();}));})}getBatchKey(e){let{resource:t,commonWhere:n,...r}=e;return `${t}:${JSON.stringify(n??{})}:${JSON.stringify(r??{})}`}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(r=>{r.reject(n);});}}async executeBatchedRequests(e){var y;if(e.length===0)return;let t=e[0],{resource:n,commonWhere:r,include:o,sort:c}=t,a=e.length===1?t.limit:void 0,s=e.map(p=>p.uniqueWhere).filter(p=>p!==void 0),u=r,d=(y=Object.entries(s[0]??{})[0])==null?void 0:y[0];if(s.length>0){let p=s.map(h=>h[d]).filter(h=>h!=null);p.length>0&&(u=q(r,{[d]:{$in:p}}));}let l=await this.storage.get({resource:n,where:u,include:o,sort:c,limit:a});for(let p of e){let h=l;if(p.uniqueWhere){let[f,T]=Object.entries(p.uniqueWhere)[0];h=l.filter(g=>{var m;return ((m=g.value[f])==null?void 0:m.value)===T});}p.resolve(h);}}};var j=i=>jsXxhash.xxHash32(JSON.stringify(i)).toString(32),C=(i,e,t)=>{let n={},r=t[e];if(!r)return n;let o=c=>{c.$and?c.$and.forEach(o):c.$or?c.$or.forEach(o):Object.entries(c).forEach(([a,s])=>{var u;if((u=r.relations)!=null&&u[a]&&(n[a]=true,typeof s=="object"&&s!==null&&!Array.isArray(s))){let d=C(s,r.relations[a].entity.name,t);Object.keys(d).length>0&&(n[a]=d);}});};return o(i),n},I=(i,e,t=false)=>Object.entries(e).every(([n,r])=>{if(n==="$and")return r.every(c=>I(i,c,t));if(n==="$or")return r.some(c=>I(i,c,t));let o=(r==null?void 0:r.$eq)!==void 0?r==null?void 0:r.$eq:r;if(typeof r=="object"&&r!==null&&(r==null?void 0:r.$eq)===void 0){if(r.$in!==void 0){let a=i[n];return a===void 0?false:t?!r.$in.includes(a):r.$in.includes(a)}if(r.$not!==void 0&&!t)return I(i,{[n]:r.$not},true);if(r.$gt!==void 0){let a=i[n];return typeof a!="number"?false:t?a<=r.$gt:a>r.$gt}if(r.$gte!==void 0){let a=i[n];return typeof a!="number"?false:t?a<r.$gte:a>=r.$gte}if(r.$lt!==void 0){let a=i[n];return typeof a!="number"?false:t?a>=r.$lt:a<r.$lt}if(r.$lte!==void 0){let a=i[n];return typeof a!="number"?false:t?a>r.$lte:a<=r.$lte}let c=i[n];return !c||typeof c!="object"&&!Array.isArray(c)?false:Array.isArray(c)?t?!c.some(a=>I(a,r,false)):c.some(a=>I(a,r,false)):I(c,r,t)}return t?i[n]!==o:i[n]===o}),L={CRITICAL:0,ERROR:1,WARN:2,INFO:3,DEBUG:4},oe=class{level;prefix;constructor(e={}){this.level=e.level??L.INFO,this.prefix=e.prefix?`[${e.prefix}] `:"";}critical(...e){this.level>=L.CRITICAL&&console.error(`${this.prefix}[CRITICAL]`,...e);}error(...e){this.level>=L.ERROR&&console.error(`${this.prefix}[ERROR]`,...e);}warn(...e){this.level>=L.WARN&&console.warn(`${this.prefix}[WARN]`,...e);}info(...e){this.level>=L.INFO&&console.log(`${this.prefix}[INFO]`,...e);}debug(...e){this.level>=L.DEBUG&&console.log(`${this.prefix}[DEBUG]`,...e);}setLevel(e){this.level=e;}getLevel(){return this.level}},Le=i=>new oe(i);function ce(i){return j({resource:i.query.resource,where:i.query.where,include:i.query.include,stepPath:i.stepPath,isMany:i.isMany,relationName:i.relationName})}var J=class{router;storage;schema;logger;queryNodes=new Map;objectNodes=new Map;constructor(e){this.router=e.router,this.storage=e.storage,this.schema=e.schema,this.logger=e.logger;}getRelationalColumns(e){let t=new Map,n=this.schema[e];if(!(n!=null&&n.relations))return t;for(let[r,o]of Object.entries(n.relations))o.type==="one"&&o.relationalColumn&&t.set(String(o.relationalColumn),{relationName:r,targetResource:o.entity.name});return t}ensureObjectNode(e,t,n){let r=this.objectNodes.get(e);return r?n&&r.matchedQueries.add(n):(r={id:e,type:t,matchedQueries:new Set(n?[n]:[]),referencesObjects:new Map,referencedByObjects:new Map},this.objectNodes.set(e,r)),r}storeRelation(e,t,n,r){let o=this.objectNodes.get(e),c=this.objectNodes.get(t);if(o&&o.referencesObjects.set(n,t),c&&r){let a=c.referencedByObjects.get(r);a||(a=new Set,c.referencedByObjects.set(r,a)),a.add(e);}}removeRelation(e,t,n,r){let o=this.objectNodes.get(e),c=this.objectNodes.get(t);if(o&&o.referencesObjects.delete(n),c&&r){let a=c.referencedByObjects.get(r);a&&(a.delete(e),a.size===0&&c.referencedByObjects.delete(r));}}getInverseRelationName(e,t,n){let r=this.schema[e];if(!(r!=null&&r.relations))return;let o=r.relations[n];if(!o)return;let c=this.schema[t];if(c!=null&&c.relations){if(o.type==="many"&&o.foreignColumn){for(let[a,s]of Object.entries(c.relations))if(s.entity.name===e&&s.type==="one"&&s.relationalColumn===o.foreignColumn)return a}if(o.type==="one"&&o.relationalColumn){for(let[a,s]of Object.entries(c.relations))if(s.entity.name===e&&s.type==="many"&&s.foreignColumn===o.relationalColumn)return a}}}updateRelationsFromMutation(e,t,n,r){let o=this.getRelationalColumns(e),c=this.objectNodes.get(t);if(c)for(let[a,{relationName:s,targetResource:u}]of Array.from(o)){if(!(r&&a in r))continue;let l=this.getInverseRelationName(e,u,s),y=c.referencesObjects.get(s),p=n[a];y!==p&&(y&&this.removeRelation(t,y,s,l),p&&(this.ensureObjectNode(p,u),this.storeRelation(t,p,s,l)));}}get(e,t){let n=this.breakdownQuery({query:e,context:(t==null?void 0:t.context)??{}});return this.resolveQuery(n,{context:(t==null?void 0:t.context)??{},batcher:(t==null?void 0:t.batcher)??new G(this.storage)})}subscribe(e,t,n={}){let r=this.breakdownQuery({query:e,context:n}),o={},c=[];for(let a of r){this.logger.debug("[QueryEngine] Subscribing to step",a.stepPath.join("."));let s=ce(a),u=o[a.stepPath.at(-2)??""],d=a.stepPath.at(-1)??"",l=this.queryNodes.get(s);if(l)l.subscriptions.add(t);else if(l={hash:s,queryStep:a,relationName:d,trackedObjects:new Set,subscriptions:new Set([t]),parentQuery:u,childQueries:new Set},this.queryNodes.set(l.hash,l),u){let y=this.queryNodes.get(u);y&&y.childQueries.add(l.hash);}o[d]=s,c.push(()=>{let y=this.queryNodes.get(s);y&&(y.subscriptions.delete(t),y.subscriptions.size===0&&this.queryNodes.delete(s));});}return ()=>{for(let a of c)a();}}breakdownQuery(e){var T;let{query:t,stepPath:n=[],context:r={},parentResource:o}=e,{include:c}=t,a=n.length===0,s=n.at(-1),u,d,l;if(!a&&o&&s){let g=this.schema[o],m=(T=g==null?void 0:g.relations)==null?void 0:T[s];m&&(l=m.type==="many",m.type==="one"?(u=b=>({id:b}),d=b=>b.map(v=>{var S,M;return (M=(S=v.value)==null?void 0:S[m.relationalColumn])==null?void 0:M.value}).filter(v=>v!==void 0)):(u=b=>({[m.foreignColumn]:b}),d=b=>b.map(v=>{var S,M;return (M=(S=v.value)==null?void 0:S.id)==null?void 0:M.value}).filter(v=>v!==void 0)));}let{include:y,...p}=t,f=[this.router.incrementQueryStep({query:p,stepPath:[...n],getWhere:u,referenceGetter:d,isMany:l,relationName:s},r)];if(c&&typeof c=="object"&&Object.keys(c).length>0){let g=this.schema[t.resource];if(!g)throw new Error(`Resource ${t.resource} not found`);f.push(...Object.entries(c).flatMap(([m,b])=>{let v=g.relations[m];if(!v)throw new Error(`Relation ${m} not found for resource ${t.resource}`);let S=v.entity.name;return this.breakdownQuery({query:{resource:S,include:typeof b=="object"?b:void 0},stepPath:[...n,m],context:r,parentResource:t.resource})}));}return f}resolveQuery(e,t){this.logger.debug("[QueryEngine] Resolving query",e.map(o=>o.stepPath.join(".")).join(" -> "));let n={},r=this.resolveStep(e[0],t).then(o=>{this.logger.debug("[QueryEngine] Resolved step",e[0].stepPath.join("."),"with results count:",o.length),n[e[0].stepPath.join(".")]=[{data:o}];});for(let o=1;o<e.length;o++){let c=e[o],a=c.stepPath.slice(0,-1).join(".");r=r.then(async()=>{var u,d;let s=n[a];if(!s){n[c.stepPath.join(".")]=[];return}if(c.referenceGetter&&c.getWhere){let l=new Map;for(let h of s)for(let f of h.data){let T=(d=(u=f==null?void 0:f.value)==null?void 0:u.id)==null?void 0:d.value;if(!T)continue;let m=c.referenceGetter([f])[0];if(m){l.has(m)||l.set(m,new Set);let b=l.get(m);b&&b.add(T);}}let y=Array.from(l.keys());if(y.length===0){n[c.stepPath.join(".")]=[];return}let p=[];for(let h of y){let f=c.getWhere(h),T={...c,relationalWhere:f},g=await this.resolveStep(T,t),m=l.get(h);if(m)for(let b of Array.from(m))p.push({includedBy:b,data:g});}this.logger.debug("[QueryEngine] Resolved step",c.stepPath.join("."),"with results count:",p.reduce((h,f)=>h+f.data.length,0)),n[c.stepPath.join(".")]=p;}else {let l=await this.resolveStep(c,t);n[c.stepPath.join(".")]=[{data:l}];}});}return r=r.then((()=>(this.logger.debug("[QueryEngine] Assembling results"),this.assembleResults(e,n)))),r}assembleResults(e,t){var c,a,s,u,d;this.logger.debug("[QueryEngine] assembleResults: Starting assembly"),this.logger.debug("[QueryEngine] assembleResults: Plan steps:",e.length),this.logger.debug("[QueryEngine] assembleResults: Step results keys:",Object.keys(t));let n=new Map;for(let l of e){let y=l.stepPath.join("."),p=t[y]??[],h=Object.keys(l.query.include??{});this.logger.debug(`[QueryEngine] assembleResults: Processing step "${y}"`,{resource:l.query.resource,includedRelations:h,resultGroups:p.length,isMany:l.isMany,relationName:l.relationName});for(let f of p){this.logger.debug(`[QueryEngine] assembleResults: Processing result group for "${y}"`,{dataCount:f.data.length,includedBy:f.includedBy});for(let T of f.data){let g=(a=(c=T==null?void 0:T.value)==null?void 0:c.id)==null?void 0:a.value;if(!g){this.logger.debug(`[QueryEngine] assembleResults: Skipping data without id in step "${y}"`);continue}let m=y?`${y}.${g}`:g,b=[];if(l.stepPath.length>0&&f.includedBy){let S=l.stepPath.slice(0,-1).join(".");b=[S?`${S}.${f.includedBy}`:f.includedBy],this.logger.debug(`[QueryEngine] assembleResults: Child entity "${m}" has parent keys:`,b,{stepPath:y,parentStepPath:S,includedBy:f.includedBy});}else this.logger.debug(`[QueryEngine] assembleResults: Root entity "${m}" (no parent)`);let v=n.get(m);if(v){this.logger.debug(`[QueryEngine] assembleResults: Entity "${m}" already exists, adding parent keys:`,b);for(let S of b)v.includedBy.add(S);}else this.logger.debug(`[QueryEngine] assembleResults: Adding new entity "${m}"`,{resource:l.query.resource,path:l.stepPath.at(-1)??"",isMany:l.isMany??false,relationName:l.relationName,includedRelations:h,parentKeys:b}),n.set(m,{data:T,includedBy:new Set(b),path:l.stepPath.at(-1)??"",isMany:l.isMany??false,relationName:l.relationName,resourceName:l.query.resource,includedRelations:h});}}}this.logger.debug(`[QueryEngine] assembleResults: Built entriesMap with ${n.size} entries`),this.logger.debug("[QueryEngine] assembleResults: EntriesMap keys:",Array.from(n.keys()));let r=Array.from(n.entries()),o=[];this.logger.debug(`[QueryEngine] assembleResults: Starting assembly phase with ${r.length} entries`);for(let l=r.length-1;l>=0;l--){let[y,p]=r[l],h=this.schema[p.resourceName];this.logger.debug(`[QueryEngine] assembleResults: Processing entry "${y}"`,{resource:p.resourceName,path:p.path,isMany:p.isMany,relationName:p.relationName,includedRelations:p.includedRelations,parentKeys:Array.from(p.includedBy)});for(let f of p.includedRelations){let T=(u=(s=h==null?void 0:h.relations)==null?void 0:s[f])==null?void 0:u.type,g=!!p.data.value[f];if(this.logger.debug(`[QueryEngine] assembleResults: Checking included relation "${f}" for "${y}"`,{relationType:T,hasRelation:g,resourceHasRelation:!!((d=h==null?void 0:h.relations)!=null&&d[f])}),p.data.value[f])this.logger.debug(`[QueryEngine] assembleResults: Relation "${f}" already exists for "${y}"`,p.data.value[f]);else {let m=T==="many"?{value:[]}:{value:null};p.data.value[f]=m,this.logger.debug(`[QueryEngine] assembleResults: Initialized relation "${f}" for "${y}" with`,m);}}if(p.path===""){this.logger.debug(`[QueryEngine] assembleResults: Adding root entity "${y}" to resultData`),o.push(p.data);continue}this.logger.debug(`[QueryEngine] assembleResults: Attaching "${y}" to ${p.includedBy.size} parent(s)`);for(let f of Array.from(p.includedBy)){let T=n.get(f);if(!T){this.logger.warn(`[QueryEngine] assembleResults: WARNING - Parent "${f}" not found in entriesMap for child "${y}"`);continue}let g=p.relationName??p.path;this.logger.debug(`[QueryEngine] assembleResults: Attaching "${y}" to parent "${f}" via relation "${g}"`,{isMany:p.isMany,parentHasRelation:!!T.data.value[g]}),p.isMany?(T.data.value[g]??={value:[]},T.data.value[g].value.push(p.data),this.logger.debug(`[QueryEngine] assembleResults: Added "${y}" to many relation "${g}" on parent "${f}"`,{arrayLength:T.data.value[g].value.length})):(T.data.value[g]=p.data,this.logger.debug(`[QueryEngine] assembleResults: Set one relation "${g}" on parent "${f}" to "${y}"`));}}return this.logger.debug(`[QueryEngine] assembleResults: Assembly complete. Returning ${o.length} root items`),o}resolveStep(e,t){this.logger.debug("[QueryEngine] Resolving step",e.stepPath.join("."),"with query",JSON.stringify(e.query,null,2),"relationalWhere",JSON.stringify(e.relationalWhere,null,2));let{query:n,relationalWhere:r}=e,o=n.where&&r?q(n.where,r):r??n.where,c=o?{...n,where:o}:n;return ae(this.router.get(c,t)).then(a=>(this.loadStepResults(e,a),a))}loadStepResults(e,t){this.logger.debug("[QueryEngine] Loading step results",e.stepPath.join("."),"with results",JSON.stringify(t,null,2));let n=ce(e),r=this.queryNodes.get(n),o=e.query.resource;if(!r)return;let c=this.getRelationalColumns(o);for(let a of t){let s=x(a),u=s.id;this.ensureObjectNode(u,o,n),r.trackedObjects.add(u);for(let[d,{relationName:l,targetResource:y}]of Array.from(c)){let p=s[d];if(p){this.ensureObjectNode(p,y);let h=this.getInverseRelationName(o,y,l);this.storeRelation(u,p,l,h);}}this.loadNestedRelations(o,u,s),this.logger.debug("[QueryEngine] Loaded nested relations for",u);}}loadNestedRelations(e,t,n){let r=this.schema[e];if(r!=null&&r.relations)for(let[o,c]of Object.entries(r.relations)){let a=n[o];if(!a)continue;let s=c.entity.name,u=this.getInverseRelationName(e,s,o);if(c.type==="one")a&&typeof a=="object"&&a.id&&(this.ensureObjectNode(a.id,s),this.storeRelation(t,a.id,o,u),this.loadNestedRelations(s,a.id,a));else if(c.type==="many"&&Array.isArray(a)){for(let d of a)if(d&&typeof d=="object"&&d.id){this.ensureObjectNode(d.id,s);let l=this.getInverseRelationName(s,e,o);l&&this.storeRelation(d.id,t,l,o),this.loadNestedRelations(s,d.id,d);}}}}buildIncludeFromChildQueries(e){let t=this.queryNodes.get(e);if(!t||t.childQueries.size===0)return {};let n={};for(let r of Array.from(t.childQueries)){let o=this.queryNodes.get(r);if(!o||!o.relationName)continue;let c=this.buildIncludeFromChildQueries(r);n[o.relationName]=Object.keys(c).length>0?c:true;}return n}sendInsertsForTree(e,t,n){var a,s;let r=(s=(a=t==null?void 0:t.value)==null?void 0:a.id)==null?void 0:s.value;if(!r)return;let o={procedure:"INSERT",resource:n,resourceId:r,type:"MUTATE",payload:t.value};for(let u of Array.from(e.subscriptions))try{u(o);}catch(d){this.logger.error("[QueryEngine] Error in subscription callback during sendInsertsForTree",{error:d,queryHash:e.hash,resource:n,resourceId:r,stepPath:e.queryStep.stepPath.join(".")});}e.trackedObjects.add(r),this.ensureObjectNode(r,n,e.hash).matchedQueries.add(e.hash);for(let u of Array.from(e.childQueries)){let d=this.queryNodes.get(u);if(!d||!d.relationName)continue;let l=d.relationName,y=d.queryStep.query.resource,p=t.value[l];if(!p)continue;let h=p.value;if(Array.isArray(h))for(let f of h)this.sendInsertsForTree(d,f,y);else h&&typeof h=="object"&&this.sendInsertsForTree(d,h,y);}}handleMutation(e,t){if(e.procedure==="INSERT"){if(this.objectNodes.has(e.resourceId))return;let n=x(t);if(!n)return;let r={id:e.resourceId,type:e.resource,matchedQueries:new Set,referencesObjects:new Map,referencedByObjects:new Map};this.objectNodes.set(e.resourceId,r);let o=this.getRelationalColumns(e.resource);for(let[a,{relationName:s,targetResource:u}]of Array.from(o)){let d=n[a];if(d){this.ensureObjectNode(d,u);let l=this.getInverseRelationName(e.resource,u,s);this.storeRelation(e.resourceId,d,s,l);}}let c=this.objectNodes.get(e.resourceId);this.getMatchingQueries(e,n).then(a=>{for(let s of a){let u=this.queryNodes.get(s);if(u){u.trackedObjects.add(e.resourceId),c&&c.matchedQueries.add(s);for(let d of Array.from(u.subscriptions))try{d(e);}catch(l){this.logger.error("[QueryEngine] Error in subscription callback during INSERT mutation",{error:l,queryHash:u.hash,resource:e.resource,resourceId:e.resourceId,stepPath:u.queryStep.stepPath.join(".")});}}}});return}if(e.procedure==="UPDATE"){let n=x(t);if(!n)return;let r=this.objectNodes.get(e.resourceId),o=new Set((r==null?void 0:r.matchedQueries)??[]);r||(r={id:e.resourceId,type:e.resource,matchedQueries:new Set,referencesObjects:new Map,referencedByObjects:new Map},this.objectNodes.set(e.resourceId,r)),this.updateRelationsFromMutation(e.resource,e.resourceId,n,e.payload),this.getMatchingQueries(e,n).then(c=>{let a=new Set(c),s=[],u=[],d=[];for(let y of c)o.has(y)?d.push(y):s.push(y);for(let y of Array.from(o))a.has(y)||u.push(y);for(let y of s){let p=this.queryNodes.get(y);p&&p.trackedObjects.add(e.resourceId);}for(let y of u){let p=this.queryNodes.get(y);p&&p.trackedObjects.delete(e.resourceId);}let l=this.objectNodes.get(e.resourceId);if(l){for(let y of s)l.matchedQueries.add(y);for(let y of u)l.matchedQueries.delete(y);}for(let y of [...u,...d]){let p=this.queryNodes.get(y);if(p)for(let h of Array.from(p.subscriptions))try{h(e);}catch(f){this.logger.error("[QueryEngine] Error in subscription callback during UPDATE mutation",{error:f,queryHash:p.hash,resource:e.resource,resourceId:e.resourceId,stepPath:p.queryStep.stepPath.join(".")});}}if(s.length>0)for(let y of s){let p=this.queryNodes.get(y);if(!p)continue;let h=this.buildIncludeFromChildQueries(y);this.get({resource:e.resource,where:{id:e.resourceId},include:Object.keys(h).length>0?h:void 0}).then(f=>{!f||f.length===0||this.sendInsertsForTree(p,f[0],e.resource);});}});return}}getMatchingQueries(e,t){let n=[];for(let r of Array.from(this.queryNodes.values()))r.queryStep.query.resource===e.resource&&n.push(r);return n.length===0?ae([]):Promise.all(n.map(async r=>{let o=r.queryStep.query.where,c=r.queryStep.query.resource,a=e.resourceId,s=this.objectNodes.get(a);if(!s)return {hash:r.hash,matches:false};if(r.relationName){let p=r.parentQuery?this.queryNodes.get(r.parentQuery):void 0,h=p==null?void 0:p.queryStep.query.resource,f=h?this.getInverseRelationName(h,c,r.relationName):void 0,T=f?s.referencesObjects.get(f):void 0;if(!T)return {hash:r.hash,matches:false};let g=this.objectNodes.get(T);return !g||!p||!g.matchedQueries.has(p.hash)?{hash:r.hash,matches:false}:{hash:r.hash,matches:true}}if(!o)return {hash:r.hash,matches:true};let u=C(o,c,this.schema),d=Object.keys(u).length>0;if(!d&&t!==void 0)return {hash:r.hash,matches:I(t,o)};let l=await this.storage.get({resource:c,where:{id:a},include:d?u:void 0});if(!l||l.length===0)return {hash:r.hash,matches:false};let y=x(l[0]);return y?{hash:r.hash,matches:I(y,o)}:{hash:r.hash,matches:false}})).then(r=>r.filter(o=>o.matches).map(o=>o.hash))}};var Ce=Se(ue());var F=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()}),le=zod.z.record(zod.z.string(),zod.z.object({value:zod.z.any().nullable(),_meta:zod.z.object({timestamp:zod.z.string().optional().nullable()}).optional()})),ht=le.superRefine((i,e)=>{i.id&&e.addIssue({code:zod.z.ZodIssueCode.custom,message:"Payload cannot have an id"});}),Ae=zod.z.object({id:zod.z.string().optional(),type:zod.z.literal("MUTATE"),resource:zod.z.string(),resourceId:zod.z.string().optional()}),z=Ae.extend({procedure:zod.z.string(),payload:zod.z.any().optional()}),V=Ae.extend({procedure:zod.z.enum(["INSERT","UPDATE"]),payload:ht});zod.z.union([V,z]);var Ee=F.omit({resource:true}),de=z.omit({id:true,type:true,resource:true,procedure:true}),ye=V.omit({id:true,type:true,resource:true,procedure:true});zod.z.union([ye,de]);var pe=i=>{if(i==null)return i;if(i==="null")return null;if(Array.isArray(i))return i.map(pe);if(typeof i=="object"&&i.constructor===Object){let e={};for(let[t,n]of Object.entries(i))e[t]=pe(n);return e}return i},Ne=i=>{let e=i.logger;return async t=>{var n;try{let r=typeof t.headers.getSetCookie=="function"?Object.fromEntries(t.headers):t.headers,o={headers:r,cookies:r.cookie?Ce.default.parse(r.cookie):{}},c=new URL(t.url),a=c.pathname.split("/"),s=c.searchParams,u=pe(gt__default.default.parse(s.toString())),d=await((n=i.contextProvider)==null?void 0:n.call(i,{transport:"HTTP",headers:o.headers,cookies:o.cookies,queryParams:u}))??{};if(t.method==="GET"){let l=a[a.length-1],{success:y,data:p,error:h}=Ee.safeParse(u);if(!y)return Response.json({message:"Invalid query",code:"INVALID_QUERY",details:h},{status:400});let f=await i.handleQuery({req:{...o,...p,type:"QUERY",resource:l,context:d,queryParams:u}});return !f||!f.data?Response.json({message:"Invalid resource",code:"INVALID_RESOURCE"},{status:400}):Response.json(f.data)}if(t.method==="POST")try{let l=a[a.length-1],y=a[a.length-2],p=t.body?await t.json():{},h;if(l==="insert"||l==="update"){let{success:T,data:g,error:m}=ye.safeParse(p);if(!T)return Response.json({message:"Invalid mutation",code:"INVALID_REQUEST",details:m},{status:400});h=g;}else {let{success:T,data:g,error:m}=de.safeParse(p);if(!T)return Response.json({message:"Invalid mutation",code:"INVALID_REQUEST",details:m},{status:400});h=g;}let f=await i.handleMutation({req:{...o,type:"MUTATE",resource:y,input:h.payload,context:d,resourceId:h.resourceId,procedure:l==="insert"||l==="update"?l.toUpperCase():l,queryParams:{}}});return Response.json(f)}catch(l){return e.error("Error parsing mutation from the client:",l),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 e.error("Unexpected error:",r),Response.json({message:"Internal server error",code:"INTERNAL_SERVER_ERROR"},{status:500})}}};var qe=Se(ue());var N=zod.z.string(),Tt=F.extend({id:N,type:zod.z.literal("SUBSCRIBE")}),bt=F.extend({id:N,type:zod.z.literal("UNSUBSCRIBE")}),Rt=F.extend({id:N,type:zod.z.literal("QUERY")}),$e=V.extend({id:N}),vt=z.extend({id:N}),St=zod.z.union([vt,$e]),Pe=zod.z.union([Tt,Rt,St,bt]),xt=zod.z.object({id:N,type:zod.z.literal("REJECT"),resource:zod.z.string(),message:zod.z.string().optional()}),wt=zod.z.object({id:N,type:zod.z.literal("REPLY"),data:zod.z.any()});zod.z.union([xt,wt,$e]);zod.z.object({resource:zod.z.string(),data:zod.z.array(le)});var Fe=i=>{let e={},t=i.logger;return (n,r)=>{var l;let o=y=>{n.send(JSON.stringify(y));},c=H(),a=new Map,s={headers:r.headers,cookies:typeof r.headers.cookie=="string"?qe.default.parse(r.headers.cookie):{}},u=gt.parse(r.url.split("?")[1]),d=(l=i.contextProvider)==null?void 0:l.call(i,{transport:"WEBSOCKET",headers:s.headers,cookies:s.cookies,queryParams:u});e[c]=n,t.info("Client connected:",c),n.on("message",async y=>{try{t.debug("Message received from the client:",y);let p=Pe.parse(JSON.parse(y.toString()));if(p.type==="SUBSCRIBE"||p.type==="QUERY"){let{type:h,id:f,...T}=p,g=h==="SUBSCRIBE",m=await i.handleQuery({req:{...s,...T,type:"QUERY",context:await d??{},queryParams:u},subscription:g?b=>{var v;!b.resourceId||!b.payload||!Object.keys(b.payload).length||(v=e[c])==null||v.send(JSON.stringify(b));}:void 0});if(!m||!m.data)throw new Error("Invalid resource");g&&m.unsubscribe&&a.set(j(T),m.unsubscribe),o({id:f,type:"REPLY",data:{resource:T.resource,data:(m.data??[]).map(b=>b.value)}});}else if(p.type==="UNSUBSCRIBE"){let{type:h,id:f,...T}=p,g=a.get(j(T));g&&(g(),a.delete(j(T)));}else if(p.type==="MUTATE"){let{resource:h}=p;t.debug("Received mutation from client:",p);try{let f=await i.handleMutation({req:{...s,type:"MUTATE",resource:h,input:p.payload,context:{messageId:p.id,...await d??{}},resourceId:p.resourceId,procedure:p.procedure,queryParams:u}});p.procedure&&p.procedure!=="INSERT"&&p.procedure!=="UPDATE"&&o({id:p.id,type:"REPLY",data:f});}catch(f){o({id:p.id,type:"REJECT",resource:h,message:f.message}),t.error("Error parsing mutation from the client:",f);}}}catch(p){t.error("Error handling message from the client:",p);}}),n.on("close",()=>{t.info("Connection closed",c),delete e[c];for(let y of Array.from(a.values()))y();});}};function ke(i){let e=`${i.protocol}://${i.hostname}${i.url}`,t=new Headers;return Object.entries(i.headers).forEach(([n,r])=>{r&&t.set(n,Array.isArray(r)?r.join(","):r);}),new Request(e,{method:i.method,headers:t,body:i.body&&i.method!=="GET"?JSON.stringify(i.body):void 0})}var Gr=(i,e,t)=>{i.ws(`${(t==null?void 0:t.basePath)??""}/ws`,Fe(e)),i.use(`${(t==null?void 0:t.basePath)??""}/`,(n,r)=>{Ne(e)(ke(n)).then(c=>c.json().then(a=>r.status(c.status).send(a)));});};var fe=class i{routes;hooksRegistry=new Map;constructor(e){this.routes=e.routes;for(let t of Object.values(e.routes)){let n=t;n.hooks&&this.hooksRegistry.set(n.resourceSchema.name,n.hooks);}}static create(e){return new i(e)}getHooks(e){return this.hooksRegistry.get(e)}},en=i=>fe.create({...i}),Lt=(i=>({handler:e=>({inputValidator:i??zod.z.undefined(),handler:e})})),he=class i{resourceSchema;middlewares;customMutations;authorization;hooks;constructor(e,t,n,r){this.resourceSchema=e,this.middlewares=new Set,this.customMutations=t??{},this.authorization=n,this.hooks=r;}use(...e){for(let t of e)this.middlewares.add(t);return this}withMutations(e){return new i(this.resourceSchema,e({mutation:Lt}),this.authorization,this.hooks)}withHooks(e){return new i(this.resourceSchema,this.customMutations,this.authorization,e)}handleQuery=async({req:e,batcher:t})=>await this.wrapInMiddlewares(async n=>{let r={resource:n.resource,where:n.where,include:n.include,lastSyncedAt:n.lastSyncedAt,limit:n.limit,sort:n.sort},o=j(r);return {data:await t.rawFind({resource:n.resource,commonWhere:n.where,uniqueWhere:n.relationalWhere,include:n.include,limit:n.limit,sort:n.sort}),unsubscribe:void 0,queryHash:o}})(e);handleMutation=async({req:e,db:t,schema:n})=>await this.wrapInMiddlewares(async r=>{if(!r.procedure)throw new Error("Procedure is required for mutations");let o=this.customMutations[r.procedure];if(o){let c=o.inputValidator["~standard"].validate(r.input),a=c instanceof Promise?await c:c;if(a.issues){let s=a.issues.map(u=>{var l;let d=(l=u.path)==null?void 0:l.map(y=>typeof y=="object"&&"key"in y?String(y.key):String(y)).join(".");return d?`${d}: ${u.message}`:u.message}).join(", ");throw new Error(`Validation failed: ${s}`)}return r.input=a.value,o.handler({req:r,db:t})}else {if(r.procedure==="INSERT"||r.procedure==="UPDATE")return this.handleSet({req:r,db:t,operation:r.procedure,schema:n});throw new Error(`Unknown procedure: ${r.procedure}`)}})(e);getAuthorizationClause(e){var t,n;return (n=(t=this.authorization)==null?void 0:t.read)==null?void 0:n.call(t,{ctx:e.context})}handleSet=async({req:e,db:t,operation:n,schema:r})=>{if(!e.input)throw new Error("Payload is required");if(!e.resourceId)throw new Error("ResourceId is required");let o=await t.rawFindById(e.resource,e.resourceId);if(n==="INSERT"&&o)throw new Error("Resource already exists");if(n==="UPDATE"&&!o)throw new Error("Resource not found");return t.transaction(async({trx:c})=>{var d,l,y,p,h,f,T;let[a,s]=this.resourceSchema.mergeMutation("set",e.input,o);if(!s)throw new Error("Mutation rejected");if(n==="INSERT"){let g=await c.rawInsert(e.resource,e.resourceId,a,(d=e.context)==null?void 0:d.messageId,e.context),m=x(g);if(m.id=m.id??e.resourceId,(l=this.authorization)!=null&&l.insert){let b=this.authorization.insert({ctx:e.context,value:m});if(typeof b=="boolean"){if(!b)throw new Error("Not authorized")}else {let v=C(b,e.resource,r),S=Object.keys(v).length>0?await c.rawFindById(e.resource,e.resourceId,v):g,M=x(S);if(M.id=M.id??e.resourceId,!I(M,b))throw new Error("Not authorized")}}return {data:g,acceptedValues:s}}if((p=(y=this.authorization)==null?void 0:y.update)!=null&&p.preMutation){let g=x(o);g.id=g.id??e.resourceId;let m=this.authorization.update.preMutation({ctx:e.context,value:g});if(typeof m=="boolean"){if(!m)throw new Error("Not authorized")}else {let b=C(m,e.resource,r),v=Object.keys(b).length>0?await c.rawFindById(e.resource,e.resourceId,b):o,S=x(v);if(S.id=S.id??e.resourceId,!I(S,m))throw new Error("Not authorized")}}let u=await c.rawUpdate(e.resource,e.resourceId,a,(h=e.context)==null?void 0:h.messageId,e.context);if((T=(f=this.authorization)==null?void 0:f.update)!=null&&T.postMutation){let g=x(u);g.id=g.id??e.resourceId;let m=this.authorization.update.postMutation({ctx:e.context,value:g});if(typeof m=="boolean"){if(!m)throw new Error("Not authorized")}else {let b=C(m,e.resource,r),v=Object.keys(b).length>0?await c.rawFindById(e.resource,e.resourceId,b):u,S=x(v);if(S.id=S.id??e.resourceId,!I(S,m))throw new Error("Not authorized")}}return {data:u,acceptedValues:s}})};wrapInMiddlewares(e){return t=>Array.from(this.middlewares.values()).reduceRight((n,r)=>o=>r({req:o,next:n}),e)(t)}},me=class i{middlewares;constructor(e=[]){this.middlewares=e;}collectionRoute(e,t){return new he(e,void 0,t,void 0).use(...this.middlewares)}use(...e){return new i([...this.middlewares,...e])}static create(){return new i}},tn=me.create;var K=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(([r,o])=>[r,{value:o,_meta:{timestamp:n}}]))}))}async update(e,t,n){let r=new Date().toISOString(),{id:o,...c}=n;return x(await this.rawUpdate(e.name,t,{value:Object.fromEntries(Object.entries(c).map(([a,s])=>[a,{value:s,_meta:{timestamp:r}}]))}))}};var $t={postgres:{jsonObjectFrom:postgres.jsonObjectFrom,jsonArrayFrom:postgres.jsonArrayFrom},mysql:{jsonObjectFrom:mysql.jsonObjectFrom,jsonArrayFrom:mysql.jsonArrayFrom},sqlite:{jsonObjectFrom:sqlite.jsonObjectFrom,jsonArrayFrom:sqlite.jsonArrayFrom}};function $(i){var r,o,c;let e=(r=i.getExecutor)==null?void 0:r.call(i),t=e==null?void 0:e.adapter;if(!t)return "postgres";let n=((c=(o=t.constructor)==null?void 0:o.name)==null?void 0:c.toLowerCase())??"";return n.includes("postgres")?"postgres":n.includes("mysql")?"mysql":n.includes("sqlite")?"sqlite":"postgres"}function De(i){let e=$(i);return $t[e]}var Pt="42701";function _(i){var t;if(i.code===Pt)return true;let e=((t=i.message)==null?void 0:t.toLowerCase())||"";return e.includes("already exists")||e.includes("duplicate")||e.includes("already defined")}function B(i,e){return i.some(t=>t.name===e)}function qt(i,e){return (i==null?void 0:i.columns.some(t=>t.name===e))??false}function ge(i,e){let[t]=e.split(".");return B(i,t)}function Qe(i,e,t,n=false){let r=i;return e.unique&&(r=r.unique()),e.nullable||(r=r.notNull()),e.primary&&(r=r.primaryKey()),e.default!==void 0&&(r=r.defaultTo(e.default)),!n&&e.references&&ge(t,e.references)&&(r=r.references(e.references)),r}function Ft(i,e,t,n,r){let o=[];for(let[c,a]of Object.entries(i.fields)){let s=e==null?void 0:e.columns.find(d=>d.name===c),u=a.getStorageFieldType();s?(s.dataType,u.type):(o.push({name:c,storageFieldType:u}),u.references&&!ge(t,u.references)&&n.push({tableName:r,columnName:c,references:u.references}));}return o}async function kt(i,e,t,n,r){if(t.length===0)return;let o=i.schema.createTable(e);for(let{name:c,storageFieldType:a}of t)o=o.addColumn(c,a.type,s=>Qe(s,a,n));await o.execute().catch(c=>{if(!_(c))throw r==null||r.error("Error creating table",e,c),c});}async function Dt(i,e,t,n,r,o){for(let{name:c,storageFieldType:a}of t){let s=a.references?ge(n,a.references):false;await i.schema.alterTable(e).addColumn(c,a.type,u=>Qe(u,a,n,!s)).execute().catch(u=>{if(!_(u))throw o==null||o.error("Error adding column",c,u),u}),a.references&&!s&&!r.some(u=>u.tableName===e&&u.columnName===c)&&r.push({tableName:e,columnName:c,references:a.references}),a.index&&await i.schema.createIndex(`${e}_${c}_index`).on(e).column(c).execute().catch(()=>{});}}async function Qt(i,e,t,n,r,o){let c=`${e}_meta`,a=[];for(let[s,u]of Object.entries(t.fields)){let d=u.getStorageFieldType();qt(n,s)||a.push({name:s,storageFieldType:d});}if(!n&&a.length>0){let s=i.schema.createTable(c);for(let{name:u,storageFieldType:d}of a)s=s.addColumn(u,"varchar",l=>{let y=l;return d.primary&&(y=y.primaryKey(),B(r,e)&&(y=y.references(`${e}.${u}`))),y});await s.execute().catch(u=>{if(!_(u))throw o==null||o.error("Error creating meta table",c,u),u});}else if(n)for(let{name:s,storageFieldType:u}of a)await i.schema.alterTable(c).addColumn(s,"varchar",d=>{let l=d;return u.primary&&(l=l.primaryKey(),B(r,e)&&(l=l.references(`${e}.${s}`))),l}).execute().catch(d=>{if(!_(d))throw o==null||o.error("Error adding meta column",s,d),d});}async function ze(i,e,t,n,r){let[o,c]=n.split("."),a=`${e}_${t}_fk`;try{await kysely.sql`
2
+ ALTER TABLE ${kysely.sql.raw(e)}
3
+ ADD CONSTRAINT ${kysely.sql.raw(a)}
4
+ FOREIGN KEY (${kysely.sql.raw(t)})
5
+ REFERENCES ${kysely.sql.raw(o)} (${kysely.sql.raw(c)})
6
+ `.execute(i);}catch(s){_(s)||r==null||r.warn("Could not add foreign key constraint",e,t,n,s);}}async function zt(i,e,t,n){for(let{tableName:r,columnName:o,references:c}of e){let a=t.find(d=>d.name===r),s=a==null?void 0:a.columns.find(d=>d.name===o),[u]=c.split(".");a&&s&&B(t,u)&&await ze(i,r,o,c,n);}}async function Vt(i,e,t,n){for(let[r,o]of Object.entries(e)){let c=`${r}_meta`,a=t.find(s=>s.name===c);if(a)for(let[s,u]of Object.entries(o.fields)){let d=u.getStorageFieldType(),l=a.columns.find(y=>y.name===s);d.primary&&l&&B(t,r)&&await ze(i,c,s,`${r}.${s}`,n);}}}async function Ve(i,e,t){let n=await i.introspection.getTables(),r=[];for(let[c,a]of Object.entries(e)){let s=n.find(l=>l.name===c),u=n.find(l=>l.name===`${c}_meta`),d=Ft(a,s,n,r,c);if(s)for(let[l,y]of Object.entries(a.fields)){let p=s.columns.find(f=>f.name===l),h=y.getStorageFieldType();p&&p.dataType!==h.type&&(t==null||t.warn("Column type mismatch:",l,"expected to have type:",h.type,"but has type:",p.dataType));}!s&&d.length>0?await kt(i,c,d,n,t):s&&await Dt(i,c,d,n,r,t),await Qt(i,c,a,u,n,t);}let o=await i.introspection.getTables();await zt(i,r,o,t),await Vt(i,e,o,t);}function Y(i,e,t,n){if(!i)throw new Error("Schema not initialized");let r=i[e];if(!r)throw new Error("Resource not found");let o=n.$or,c=n.$and;return (o?t.or:t.and)(o?n.$or.map(a=>Y(i,e,t,a)):c?n.$and.map(a=>Y(i,e,t,a)):Object.entries(n).map(([a,s])=>{var u,d;if(r.fields[a])return (s==null?void 0:s.$eq)!==void 0?t(`${e}.${a}`,s.$eq===null?"is":"=",s.$eq):(s==null?void 0:s.$in)!==void 0?t(`${e}.${a}`,"in",s.$in):(s==null?void 0:s.$not)!==void 0?((u=s==null?void 0:s.$not)==null?void 0:u.$in)!==void 0?t(`${e}.${a}`,"not in",s.$not.$in):((d=s==null?void 0:s.$not)==null?void 0:d.$eq)!==void 0?t(`${e}.${a}`,s.$not.$eq===null?"is not":"!=",s.$not.$eq):t(`${e}.${a}`,s.$not===null?"is not":"!=",s.$not):(s==null?void 0:s.$gt)!==void 0?t(`${e}.${a}`,">",s.$gt):(s==null?void 0:s.$gte)!==void 0?t(`${e}.${a}`,">=",s.$gte):(s==null?void 0:s.$lt)!==void 0?t(`${e}.${a}`,"<",s.$lt):(s==null?void 0:s.$lte)!==void 0?t(`${e}.${a}`,"<=",s.$lte):t(`${e}.${a}`,s===null?"is":"=",s);if(r.relations[a]){let l=r.relations[a],y=l.entity.name;return l.type==="many"?t.exists(Te(i,y,t.selectFrom(y).select("id").whereRef(l.foreignColumn,"=",`${e}.id`),s)):Y(i,y,t,s)}return null}).filter(Boolean))}function X(i,e,t,n){let r=i[e];if(!r)throw new Error("Resource not found");if(!n)return t;if(n.$and){for(let o of n.$and)t=X(i,e,t,o);return t}else if(n.$or){for(let o of n.$or)t=X(i,e,t,o);return t}for(let[o,c]of Object.entries(n)){if(!r.relations[o])continue;let a=r.relations[o],s=a.entity.name,u=a.type==="one"?"id":a.foreignColumn,d=a.type==="one"?a.relationalColumn:"id";t=t.leftJoin(s,`${s}.${u}`,`${e}.${d}`),c instanceof Object&&!Array.isArray(c)&&c!==null&&(t=X(i,s,t,c));}return t}function Te(i,e,t,n){return !n||Object.keys(n).length===0?t:(t=X(i,e,t,n),t.where(r=>Y(i,e,r,n)))}function Kt(i,e,t,n,r){let o=$(r),c=e[t];if(o==="sqlite"&&(c!=null&&c.fields)){let a=Object.keys(c.fields),s=i.selectFrom(n);for(let u of a)s=s.select(`${n}.${u}`);return s}return i.selectFrom(n).selectAll(n)}function _t(i,e,t,n,r){let o=$(r),c=e[t];if(o==="sqlite"&&(c!=null&&c.fields)){let a=Object.keys(c.fields),s=i.selectFrom(n);for(let u of a)s=s.select(`${n}.${u}`);return s}return i.selectFrom(n).selectAll(n)}function ee(i,e,t,n,r,o){if(!n)return t;if(!i)throw new Error("Schema not initialized");let c=i[e];if(!c)throw new Error(`Resource not found: ${e}`);let{jsonObjectFrom:a,jsonArrayFrom:s}=r;for(let u of Object.keys(n)){if(!c.relations[u])throw new Error(`Relation ${u} not found in resource ${e}`);let d=c.relations[u],l=d.entity.name,y=n[u],p=d.type==="one"?"id":d.foreignColumn,h=d.type==="one"?d.relationalColumn:"id",f=d.type==="one"?a:s,T=typeof y=="object"&&y!==null;t=t.select(g=>{let m=`${l}_meta`,b=_t(g,i,l,l,o).whereRef(`${l}.${p}`,"=",`${e}.${h}`).select(v=>a(Kt(v,i,l,m,o).whereRef(`${m}.id`,"=",`${l}.id`)).as("_meta"));return T&&(b=ee(i,l,b,y,r,o)),f(b).as(u)});}return t}var be=class i extends K{db;dialectHelpers;schema;logger;server;mutationStack=[];constructor(e,t,n,r){super(),this.isKyselyLike(e)?this.db=e:this.db=new kysely.Kysely({dialect:new kysely.PostgresDialect({pool:e})}),this.dialectHelpers=De(this.db),this.schema=t,this.logger=n,this.server=r,this.rawInsert=this.rawInsert.bind(this),this.rawUpdate=this.rawUpdate.bind(this);}async init(e,t,n){this.schema=e,this.logger=t,this.server=n,await Ve(this.db,e,t);}selectMetaColumns(e,t,n){var c;let r=$(this.db),o=(c=this.schema)==null?void 0:c[t];if(r==="sqlite"&&(o!=null&&o.fields)){let a=Object.keys(o.fields),s=e.selectFrom(n);for(let u of a)s=s.select(`${n}.${u}`);return s}return e.selectFrom(n).selectAll(n)}async rawFindById(e,t,n){if(!this.schema)throw new Error("Schema not initialized");let r=`${e}_meta`,o=await this.db.selectFrom(e).where("id","=",t).selectAll(e).select(s=>this.dialectHelpers.jsonObjectFrom(this.selectMetaColumns(s,e,r).whereRef(`${r}.id`,"=",`${e}.id`)).as("_meta"));o=ee(this.schema,e,o,n,this.dialectHelpers,this.db);let c=await o.executeTakeFirst();if(!c)return;let a=this.parseRelationalJsonStrings(c,e);return this.convertToMaterializedLiveType(a)}async findOne(e,t,n){let r=await this.rawFindById(e.name,t,n==null?void 0:n.include);if(r)return x(r)}async get(e){if(!this.schema)throw new Error("Schema not initialized");let{resource:t,where:n,include:r,limit:o,sort:c}=e,a=`${t}_meta`,s=this.db.selectFrom(t).selectAll(t).select(d=>this.dialectHelpers.jsonObjectFrom(this.selectMetaColumns(d,t,a).whereRef(`${a}.id`,"=",`${t}.id`)).as("_meta"));s=Te(this.schema,t,s,n),s=ee(this.schema,t,s,r,this.dialectHelpers,this.db),o!==void 0&&(s=s.limit(o)),c!==void 0&&c.forEach(d=>{s=s.orderBy(d.key,d.direction);});let u=await s.execute();return u.length===0?[]:u.map(d=>{let l=this.parseRelationalJsonStrings(d,t);return this.convertToMaterializedLiveType(l)})}async find(e,t){return (await this.get({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})).map(r=>x(r))}async rawInsert(e,t,n,r,o){var l,y,p;let c=(y=(l=this.server)==null?void 0:l.router)==null?void 0:y.getHooks(e),a=n;if(c!=null&&c.beforeInsert){let h=await c.beforeInsert({ctx:o,value:a,db:this});h&&(a=h);}let s={},u={};for(let[h,f]of Object.entries(a.value)){let T=(p=f._meta)==null?void 0:p.timestamp;T&&(s[h]=f.value,u[h]=T);}await this.db.insertInto(e).values({...s,id:t}).execute().then(()=>{this.db.insertInto(`${e}_meta`).values({...u,id:t}).execute();});let d=this.buildMutation(e,t,"INSERT",a,r);return d&&this.trackMutation(d,a),c!=null&&c.afterInsert&&await c.afterInsert({ctx:o,value:a,db:this}),a}async rawUpdate(e,t,n,r,o){var y,p,h;let c=(p=(y=this.server)==null?void 0:y.router)==null?void 0:p.getHooks(e),a;(c!=null&&c.beforeUpdate||c!=null&&c.afterUpdate)&&(a=await this.rawFindById(e,t));let s=n;if(c!=null&&c.beforeUpdate){let f=await c.beforeUpdate({ctx:o,value:s,previousValue:a,db:this});f&&(s=f);}let u={},d={};for(let[f,T]of Object.entries(s.value)){let g=(h=T._meta)==null?void 0:h.timestamp;g&&(u[f]=T.value,d[f]=g);}await Promise.all([this.db.updateTable(e).set(u).where("id","=",t).execute(),this.db.insertInto(`${e}_meta`).values({...d,id:t}).onConflict(f=>f.column("id").doUpdateSet(d)).execute()]);let l=this.buildMutation(e,t,"UPDATE",s,r);if(l){let f=await this.rawFindById(e,t);f&&this.trackMutation(l,f);}if(c!=null&&c.afterUpdate){let f=await this.rawFindById(e,t);f&&await c.afterUpdate({ctx:o,value:f,previousValue:a,db:this});}return s}async transaction(e){if(!this.schema)throw new Error("Schema not initialized");if(this.db.isTransaction){let o=Math.random().toString(36).substring(2,15),c=this.mutationStack,a=[];this.mutationStack=a;let s=await this.db.savepoint(o).execute(),u=false,d=false;try{return await e({trx:this,commit:async()=>{await s.releaseSavepoint(o).execute(),u=!0,c.push(...a);},rollback:async()=>{await s.rollbackToSavepoint(o).execute(),d=!0,a.length=0;}}).then(l=>s.isCommitted||s.isRolledBack||u||d?l:s.releaseSavepoint(o).execute().then(()=>(c.push(...a),l)))}catch(l){throw d||await s.rollbackToSavepoint(o).execute().catch(()=>{}),a.length=0,l}finally{this.mutationStack=c;}}let t=[],n=this.mutationStack;this.mutationStack=t;let r=await this.db.startTransaction().execute();try{let o=new i(r,this.schema,this.logger,this.server);return o.mutationStack=t,await e({trx:o,commit:async()=>{await r.commit().execute(),this.notifyMutations(t);},rollback:async()=>{await r.rollback().execute(),t.length=0;}}).then(c=>r.isCommitted||r.isRolledBack?c:r.commit().execute().then(()=>(this.notifyMutations(t),c)))}catch(o){throw await r.rollback().execute(),t.length=0,o}finally{this.mutationStack=n;}}get internalDB(){return this.db}isRelationalField(e,t){var r;if(!this.schema)return false;let n=this.schema[e];return !!((r=n==null?void 0:n.relations)!=null&&r[t])}parseRelationalJsonStrings(e,t){var o,c;if($(this.db)!=="sqlite"||typeof e!="object"||e===null||e instanceof Date)return e;if(Array.isArray(e))return e.map(a=>this.parseRelationalJsonStrings(a,t));let r={};for(let[a,s]of Object.entries(e))if(a==="_meta"&&typeof s=="string")if(s.startsWith("{")&&s.endsWith("}")||s.startsWith("[")&&s.endsWith("]"))try{r[a]=JSON.parse(s);}catch{r[a]=s;}else r[a]=s;else if(this.isRelationalField(t,a))if(typeof s=="string")if(s.startsWith("{")&&s.endsWith("}")||s.startsWith("[")&&s.endsWith("]"))try{let u=JSON.parse(s);if(this.schema){let d=this.schema[t],l=(o=d==null?void 0:d.relations)==null?void 0:o[a];if(l){let y=l.entity.name;r[a]=this.parseRelationalJsonStrings(u,y);}else r[a]=u;}else r[a]=u;}catch{r[a]=s;}else r[a]=s;else if(typeof s=="object"&&s!==null&&!Array.isArray(s))if(this.schema){let u=this.schema[t],d=(c=u==null?void 0:u.relations)==null?void 0:c[a];if(d){let l=d.entity.name;r[a]=this.parseRelationalJsonStrings(s,l);}else r[a]=s;}else r[a]=s;else Array.isArray(s)?r[a]=s.map(u=>{var d,l;if(typeof u=="string")try{let y=JSON.parse(u);if(this.schema){let p=this.schema[t],h=(d=p==null?void 0:p.relations)==null?void 0:d[a];if(h){let f=h.entity.name;return this.parseRelationalJsonStrings(y,f)}}return y}catch{return u}if(typeof u=="object"&&u!==null&&this.schema){let y=this.schema[t],p=(l=y==null?void 0:y.relations)==null?void 0:l[a];if(p){let h=p.entity.name;return this.parseRelationalJsonStrings(u,h)}}return u}):r[a]=s;else r[a]=s;return r}convertToMaterializedLiveType(e){return {value:Object.entries(e).reduce((t,[n,r])=>{var o,c,a;return n==="_meta"||(n==="id"?t[n]={value:r}:Array.isArray(r)?t[n]={value:r.map(s=>this.convertToMaterializedLiveType(s)),_meta:{timestamp:(o=e==null?void 0:e._meta)==null?void 0:o[n]}}:typeof r=="object"&&r!==null&&!(r instanceof Date)?t[n]={...this.convertToMaterializedLiveType(r),_meta:{timestamp:(c=e==null?void 0:e._meta)==null?void 0:c[n]}}:t[n]={value:r,_meta:{timestamp:(a=e==null?void 0:e._meta)==null?void 0:a[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",r=typeof t.startTransaction=="function",o=typeof t.savepoint=="function",c=typeof t.isTransaction=="boolean"||typeof t.isTransaction=="function";return n&&r||o&&c}buildMutation(e,t,n,r,o){var a;let c={};for(let[s,u]of Object.entries(r.value)){if(s==="id")continue;let d=(a=u._meta)==null?void 0:a.timestamp;d&&(c[s]={value:u.value,_meta:{timestamp:d}});}return Object.keys(c).length===0?null:{id:o??H(),type:"MUTATE",resource:e,resourceId:t,procedure:n,payload:c}}trackMutation(e,t){this.db.isTransaction?this.mutationStack.push({mutation:e,entityData:t}):this.notifyMutations([e],t);}notifyMutations(e,t){if(this.server)if(t!==void 0){let n=e;for(let r of n)this.server.notifySubscribers(r,t);}else {let n=e;for(let{mutation:r,entityData:o}of n)this.server.notifySubscribers(r,o);}}};var Re=class i{router;storage;schema;middlewares=new Set;logger;contextProvider;queryEngine;constructor(e){var t;this.router=e.router,this.storage=e.storage,this.schema=e.schema,this.logger=Le({level:e.logLevel??L.INFO}),(t=e.middlewares)==null||t.forEach(n=>{this.middlewares.add(n);}),this.storage.init(this.schema,this.logger,this),this.contextProvider=e.contextProvider,this.queryEngine=new J({router:{get:async(n,r)=>{var l;let{headers:o,cookies:c,queryParams:a,context:s}=(r==null?void 0:r.context)??{};if(!(r!=null&&r.batcher))throw new Error("Batcher is required");let u={...n,type:"QUERY",headers:o,cookies:c,queryParams:a,context:s},d=await((l=this.router.routes[n.resource])==null?void 0:l.handleQuery({req:u,batcher:r.batcher}));return (d==null?void 0:d.data)??[]},incrementQueryStep:(n,r={})=>{var a;let o=(a=this.router.routes[n.query.resource])==null?void 0:a.getAuthorizationClause({...n.query,type:"QUERY",headers:r.headers,cookies:r.cookies,queryParams:r.queryParams,context:r.context});if(typeof o=="boolean"&&!o)throw new Error("Not authorized");let c=q(n.query.where,typeof o=="object"?o:void 0);return {...n,query:{...n.query,where:c}}}},storage:this.storage,schema:this.schema,logger:this.logger});}static create(e){return new i(e)}handleQuery(e){return this.wrapInMiddlewares(async t=>{let{headers:n,cookies:r,queryParams:o,context:c,...a}=t,s={headers:n,cookies:r,queryParams:o,context:c},u=e.subscription?this.queryEngine.subscribe(a,l=>{var y;(y=e.subscription)==null||y.call(e,l);},s):void 0;return {data:await this.queryEngine.get(a,{context:s}),unsubscribe:u}})(e.req)}async handleMutation(e){return await this.wrapInMiddlewares(async n=>{let r=this.router.routes[n.resource];if(!r)throw new Error("Invalid resource");return r.handleMutation({req:n,db:this.storage,schema:this.schema})})(e.req)}use(e){return this.middlewares.add(e),this}context(e){return this.contextProvider=e,this}notifySubscribers(e,t){this.queryEngine.handleMutation(e,t);}wrapInMiddlewares(e){return t=>Array.from(this.middlewares.values()).reduceRight((n,r)=>o=>r({req:o,next:n}),e)(t)}},jn=Re.create;exports.Route=he;exports.RouteFactory=me;exports.Router=fe;exports.SQLStorage=be;exports.Server=Re;exports.Storage=K;exports.expressAdapter=Gr;exports.routeFactory=tn;exports.router=en;exports.server=jn;
package/dist/server.d.cts CHANGED
@@ -1,8 +1,7 @@
1
1
  import { z } from 'zod';
2
2
  import { L as LiveObjectAny, W as WhereClause, S as Simplify, I as InferLiveObjectWithRelationalIds, M as MaterializedLiveType, a as Schema, b as IncludeClause, c as InferLiveObject, d as InferInsert, e as InferUpdate, f as Logger, g as LogLevel } from './index-BtsKDfTE.cjs';
3
- import * as z3 from 'zod/v3';
4
- import * as z4 from 'zod/v4/core';
5
- import { PostgresPool } from 'kysely';
3
+ import { StandardSchemaV1 } from '@standard-schema/spec';
4
+ import { PostgresPool, Kysely } from 'kysely';
6
5
  import { Application } from 'express-ws';
7
6
 
8
7
  declare const querySchema: z.ZodObject<{
@@ -30,7 +29,7 @@ declare const defaultMutationSchema: z.ZodObject<{
30
29
  UPDATE: "UPDATE";
31
30
  }>;
32
31
  payload: z.ZodRecord<z.ZodString, z.ZodObject<{
33
- value: z.ZodNullable<z.ZodUnion<[z.ZodUnion<[z.ZodUnion<[z.ZodString, z.ZodNumber]>, z.ZodBoolean]>, z.ZodDate]>>;
32
+ value: z.ZodNullable<z.ZodAny>;
34
33
  _meta: z.ZodOptional<z.ZodObject<{
35
34
  timestamp: z.ZodNullable<z.ZodOptional<z.ZodString>>;
36
35
  }, z.core.$strip>>;
@@ -40,7 +39,7 @@ type DefaultMutation = Omit<z.infer<typeof defaultMutationSchema>, "resourceId">
40
39
  resourceId: string;
41
40
  };
42
41
 
43
- type Awaitable<T> = T | Promise<T>;
42
+ type PromiseOrSync<T> = T | Promise<T>;
44
43
 
45
44
  /** biome-ignore-all lint/suspicious/noExplicitAny: false positive */
46
45
  /** biome-ignore-all lint/style/noNonNullAssertion: false positive */
@@ -48,10 +47,12 @@ type Awaitable<T> = T | Promise<T>;
48
47
  type RouteRecord = Record<string, AnyRoute>;
49
48
  declare class Router<TRoutes extends RouteRecord> {
50
49
  readonly routes: TRoutes;
50
+ readonly hooksRegistry: Map<string, Hooks<any>>;
51
51
  private constructor();
52
52
  static create<TRoutes extends RouteRecord>(opts: {
53
53
  routes: TRoutes;
54
54
  }): Router<TRoutes>;
55
+ getHooks(resourceName: string): Hooks<any> | undefined;
55
56
  }
56
57
  declare const router: <TSchema extends Schema<any>, TRoutes extends Record<keyof TSchema, Route<any, any, any>>>(opts: {
57
58
  schema: TSchema;
@@ -59,23 +60,35 @@ declare const router: <TSchema extends Schema<any>, TRoutes extends Record<keyof
59
60
  }) => Router<TRoutes>;
60
61
  type AnyRouter = Router<any>;
61
62
  type QueryResult<TShape extends LiveObjectAny> = {
62
- data: Record<string, MaterializedLiveType<TShape>>;
63
+ data: MaterializedLiveType<TShape>[];
64
+ unsubscribe?: () => void;
63
65
  };
64
66
  type MutationResult<TShape extends LiveObjectAny> = {
65
67
  data: MaterializedLiveType<TShape>;
66
68
  acceptedValues: Record<string, any> | null;
67
69
  };
68
- type Mutation<TInputValidator extends z3.ZodTypeAny | z4.$ZodType, // TODO use StandardSchema instead
69
- TOutput> = {
70
+ type Mutation<TInputValidator extends StandardSchemaV1<any, any> | never, TOutput> = {
70
71
  inputValidator: TInputValidator;
71
72
  handler: (opts: {
72
- req: MutationRequest<z.infer<TInputValidator>>;
73
+ req: MutationRequest<TInputValidator extends StandardSchemaV1<any, any> ? StandardSchemaV1.InferOutput<TInputValidator> : undefined>;
73
74
  db: Storage;
74
75
  }) => TOutput;
75
76
  };
76
- declare const mutationCreator: <TInputValidator extends z3.ZodTypeAny | z4.$ZodType>(validator?: TInputValidator) => {
77
- handler: <THandler extends Mutation<TInputValidator, any>["handler"]>(handler: THandler) => Mutation<TInputValidator, ReturnType<THandler>>;
77
+ type MutationCreator = {
78
+ (): {
79
+ handler: <TOutput>(handler: (opts: {
80
+ req: MutationRequest<undefined>;
81
+ db: Storage;
82
+ }) => TOutput) => Mutation<StandardSchemaV1<any, undefined>, TOutput>;
83
+ };
84
+ <TInputValidator extends StandardSchemaV1<any, any>>(validator: TInputValidator): {
85
+ handler: <THandler extends (opts: {
86
+ req: MutationRequest<StandardSchemaV1.InferOutput<TInputValidator>>;
87
+ db: Storage;
88
+ }) => any>(handler: THandler) => Mutation<TInputValidator, ReturnType<THandler>>;
89
+ };
78
90
  };
91
+ declare const mutationCreator: MutationCreator;
79
92
  type ReadAuthorizationHandler<TShape extends LiveObjectAny> = (opts: {
80
93
  ctx: BaseRequest["context"];
81
94
  }) => WhereClause<TShape> | boolean;
@@ -91,16 +104,47 @@ type Authorization<TShape extends LiveObjectAny> = {
91
104
  postMutation?: MutationAuthorizationHandler<TShape>;
92
105
  };
93
106
  };
107
+ type BeforeInsertHook<TShape extends LiveObjectAny> = (opts: {
108
+ ctx?: Record<string, any>;
109
+ value: MaterializedLiveType<TShape>;
110
+ db: Storage;
111
+ }) => Promise<MaterializedLiveType<TShape> | void> | MaterializedLiveType<TShape> | void;
112
+ type AfterInsertHook<TShape extends LiveObjectAny> = (opts: {
113
+ ctx?: Record<string, any>;
114
+ value: MaterializedLiveType<TShape>;
115
+ db: Storage;
116
+ }) => Promise<void> | void;
117
+ type BeforeUpdateHook<TShape extends LiveObjectAny> = (opts: {
118
+ ctx?: Record<string, any>;
119
+ value: MaterializedLiveType<TShape>;
120
+ previousValue?: MaterializedLiveType<TShape>;
121
+ db: Storage;
122
+ }) => Promise<MaterializedLiveType<TShape> | void> | MaterializedLiveType<TShape> | void;
123
+ type AfterUpdateHook<TShape extends LiveObjectAny> = (opts: {
124
+ ctx?: Record<string, any>;
125
+ value: MaterializedLiveType<TShape>;
126
+ previousValue?: MaterializedLiveType<TShape>;
127
+ db: Storage;
128
+ }) => Promise<void> | void;
129
+ type Hooks<TShape extends LiveObjectAny> = {
130
+ beforeInsert?: BeforeInsertHook<TShape>;
131
+ afterInsert?: AfterInsertHook<TShape>;
132
+ beforeUpdate?: BeforeUpdateHook<TShape>;
133
+ afterUpdate?: AfterUpdateHook<TShape>;
134
+ };
94
135
  declare class Route<TResourceSchema extends LiveObjectAny, TMiddleware extends Middleware<any>, TCustomMutations extends Record<string, Mutation<any, any>>> {
95
136
  readonly resourceSchema: TResourceSchema;
96
137
  readonly middlewares: Set<TMiddleware>;
97
138
  readonly customMutations: TCustomMutations;
98
139
  readonly authorization?: Authorization<TResourceSchema>;
99
- constructor(resourceSchema: TResourceSchema, customMutations?: TCustomMutations, authorization?: Authorization<TResourceSchema>);
140
+ readonly hooks?: Hooks<TResourceSchema>;
141
+ constructor(resourceSchema: TResourceSchema, customMutations?: TCustomMutations, authorization?: Authorization<TResourceSchema>, hooks?: Hooks<TResourceSchema>);
100
142
  use(...middlewares: TMiddleware[]): this;
101
143
  withMutations<T extends Record<string, Mutation<any, any>>>(mutationFactory: (opts: {
102
144
  mutation: typeof mutationCreator;
103
145
  }) => T): Route<TResourceSchema, TMiddleware, T>;
146
+ withHooks(hooks: Hooks<TResourceSchema>): Route<TResourceSchema, TMiddleware, TCustomMutations>;
147
+ getAuthorizationClause(req: QueryRequest): WhereClause<TResourceSchema> | undefined | boolean;
104
148
  private handleSet;
105
149
  private wrapInMiddlewares;
106
150
  }
@@ -116,16 +160,51 @@ type AnyRoute = Route<LiveObjectAny, Middleware<any>, Record<string, any>>;
116
160
 
117
161
  /** biome-ignore-all lint/suspicious/noExplicitAny: false positive */
118
162
 
119
- declare abstract class Storage {
120
- abstract findOne<T extends LiveObjectAny>(resource: T, id: string, options?: {
121
- include?: IncludeClause<T>;
122
- }): Promise<InferLiveObject<T> | undefined>;
123
- abstract find<T extends LiveObjectAny>(resource: T, options?: {
163
+ declare class Batcher {
164
+ private storage;
165
+ private queue;
166
+ private scheduled;
167
+ constructor(storage: Storage);
168
+ rawFind<T extends LiveObjectAny>({ resource, commonWhere, uniqueWhere, ...rest }: {
169
+ resource: string;
170
+ commonWhere?: Record<string, any>;
171
+ uniqueWhere?: Record<string, any>;
172
+ include?: Record<string, any>;
173
+ limit?: number;
174
+ sort?: {
175
+ key: string;
176
+ direction: "asc" | "desc";
177
+ }[];
178
+ }): Promise<MaterializedLiveType<T>[]>;
179
+ private getBatchKey;
180
+ private processBatch;
181
+ private executeBatchedRequests;
182
+ }
183
+
184
+ interface DataSource {
185
+ get(query: RawQueryRequest, extra?: {
186
+ context?: any;
187
+ batcher?: Batcher;
188
+ }): PromiseOrSync<any[]>;
189
+ }
190
+
191
+ /** biome-ignore-all lint/suspicious/noExplicitAny: false positive */
192
+
193
+ declare abstract class Storage implements DataSource {
194
+ abstract findOne<T extends LiveObjectAny, TInclude extends IncludeClause<T> | undefined = undefined>(resource: T, id: string, options?: {
195
+ include?: TInclude;
196
+ }): Promise<InferLiveObject<T, TInclude> | undefined>;
197
+ abstract find<T extends LiveObjectAny, TInclude extends IncludeClause<T> | undefined = undefined>(resource: T, options?: {
124
198
  where?: WhereClause<T>;
125
- include?: IncludeClause<T>;
126
- }): Promise<Record<string, InferLiveObject<T>>>;
199
+ include?: TInclude;
200
+ limit?: number;
201
+ sort?: {
202
+ key: string;
203
+ direction: "asc" | "desc";
204
+ }[];
205
+ }): Promise<InferLiveObject<T, TInclude>[]>;
127
206
  insert<T extends LiveObjectAny>(resource: T, value: Simplify<InferInsert<T>>): Promise<InferLiveObject<T>>;
128
- update<T extends LiveObjectAny>(resource: T, resourceId: string, value: InferUpdate<T>): Promise<InferLiveObject<T>>;
207
+ update<T extends LiveObjectAny>(resource: T, resourceId: string, value: InferUpdate<T>): Promise<Partial<InferLiveObject<T>>>;
129
208
  abstract transaction<T>(fn: (opts: {
130
209
  trx: Storage;
131
210
  commit: () => Promise<void>;
@@ -136,37 +215,58 @@ declare abstract class Storage {
136
215
  /** biome-ignore-all lint/suspicious/noExplicitAny: false positive */
137
216
 
138
217
  declare class SQLStorage extends Storage {
139
- private db;
218
+ private readonly db;
219
+ private readonly dialectHelpers;
140
220
  private schema?;
141
221
  private logger?;
222
+ private server?;
223
+ private mutationStack;
142
224
  constructor(pool: PostgresPool);
143
- findOne<T extends LiveObjectAny>(resource: T, id: string, options?: {
144
- include?: IncludeClause<T>;
145
- }): Promise<InferLiveObject<T> | undefined>;
146
- find<T extends LiveObjectAny>(resource: T, options?: {
225
+ private selectMetaColumns;
226
+ findOne<T extends LiveObjectAny, TInclude extends IncludeClause<T> | undefined = undefined>(resource: T, id: string, options?: {
227
+ include?: TInclude;
228
+ }): Promise<InferLiveObject<T, TInclude> | undefined>;
229
+ find<T extends LiveObjectAny, TInclude extends IncludeClause<T> | undefined = undefined>(resource: T, options?: {
147
230
  where?: WhereClause<T>;
148
- include?: IncludeClause<T>;
231
+ include?: TInclude;
149
232
  limit?: number;
150
233
  sort?: {
151
234
  key: string;
152
235
  direction: "asc" | "desc";
153
236
  }[];
154
- }): Promise<Record<string, InferLiveObject<T>>>;
237
+ }): Promise<InferLiveObject<T, TInclude>[]>;
155
238
  transaction<T>(fn: (opts: {
156
239
  trx: Storage;
157
240
  commit: () => Promise<void>;
158
241
  rollback: () => Promise<void>;
159
242
  }) => Promise<T>): Promise<T>;
243
+ /**
244
+ * Provides direct access to the underlying Kysely database instance.
245
+ *
246
+ * ⚠️ Warning: Direct database operations bypass mutation tracking and
247
+ * subscriber notifications. Use this only when you need to execute
248
+ * queries not supported by the Storage API.
249
+ *
250
+ * @returns The Kysely database instance
251
+ */
252
+ get internalDB(): Kysely<{
253
+ [x: string]: {
254
+ [x: string]: any;
255
+ };
256
+ }>;
257
+ private isRelationalField;
258
+ private parseRelationalJsonStrings;
160
259
  private convertToMaterializedLiveType;
161
260
  private isKyselyLike;
261
+ private buildMutation;
262
+ private trackMutation;
263
+ private notifyMutations;
162
264
  }
163
265
 
164
266
  declare const expressAdapter: (app: Application, server: Server<AnyRouter>, options?: {
165
267
  basePath?: string;
166
268
  }) => void;
167
269
 
168
- /** biome-ignore-all lint/suspicious/noExplicitAny: any's are actually used correctly */
169
-
170
270
  interface BaseRequest {
171
271
  headers: Record<string, string>;
172
272
  cookies: Record<string, string>;
@@ -187,8 +287,7 @@ type Request = QueryRequest | MutationRequest;
187
287
  type ContextProvider = (req: Omit<BaseRequest, "context"> & {
188
288
  transport: "HTTP" | "WEBSOCKET";
189
289
  }) => Record<string, any>;
190
- type MutationHandler = (mutation: DefaultMutation) => void;
191
- type NextFunction<O, R = Request> = (req: R) => Awaitable<O>;
290
+ type NextFunction<O, R = Request> = (req: R) => PromiseOrSync<O>;
192
291
  type Middleware<T = any> = (opts: {
193
292
  req: Request;
194
293
  next: NextFunction<T>;
@@ -200,7 +299,6 @@ declare class Server<TRouter extends AnyRouter> {
200
299
  readonly middlewares: Set<Middleware<any>>;
201
300
  readonly logger: Logger;
202
301
  contextProvider?: ContextProvider;
203
- private mutationSubscriptions;
204
302
  private constructor();
205
303
  static create<TRouter extends AnyRouter>(opts: {
206
304
  router: TRouter;
@@ -210,9 +308,9 @@ declare class Server<TRouter extends AnyRouter> {
210
308
  contextProvider?: ContextProvider;
211
309
  logLevel?: LogLevel;
212
310
  }): Server<TRouter>;
213
- subscribeToMutations(handler: MutationHandler): () => void;
214
311
  handleQuery(opts: {
215
312
  req: QueryRequest;
313
+ subscription?: (mutation: DefaultMutation) => void;
216
314
  }): Promise<QueryResult<any>>;
217
315
  handleMutation(opts: {
218
316
  req: MutationRequest;
@@ -223,4 +321,4 @@ declare class Server<TRouter extends AnyRouter> {
223
321
  }
224
322
  declare const server: typeof Server.create;
225
323
 
226
- 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 };
324
+ export { type AfterInsertHook, type AfterUpdateHook, type AnyRoute, type AnyRouter, type Authorization, type BaseRequest, type BeforeInsertHook, type BeforeUpdateHook, type ContextProvider, type Hooks, type Middleware, type Mutation, type MutationAuthorizationHandler, 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 };