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