@live-state/sync 0.0.5 → 0.0.6-canary-1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1,9 +1,12 @@
1
- import { A as AnyRouter, C as ClientOptions, a as Client } from './index-D8jFoiy6.js';
1
+ import { C as ClientOptions, A as AnyRouter, a as Client } from './index-D8jFoiy6.js';
2
2
  import 'react/jsx-runtime';
3
3
  import 'zod';
4
4
  import 'zod/v3';
5
5
  import 'zod/v4/core';
6
6
 
7
- declare const createClient: <TRouter extends AnyRouter>(opts: Omit<ClientOptions, "storage">) => Client<TRouter, true>;
7
+ type FetchClientOptions = Omit<ClientOptions, "storage"> & {
8
+ fetchOptions?: RequestInit;
9
+ };
10
+ declare const createClient: <TRouter extends AnyRouter>(opts: FetchClientOptions) => Client<TRouter, true>;
8
11
 
9
- export { createClient };
12
+ export { type FetchClientOptions, createClient };
@@ -1 +1 @@
1
- import {d,b,c,f}from'./chunk-RCXJM33Z.js';import {stringify}from'qs';var u=async(...e)=>{let o=await fetch(...e),t;try{t=await o.json();}catch{t=await o.text().catch(()=>{});}if(!o.ok)throw new Error(`Failed to fetch: ${o.status} ${o.statusText}`,{cause:t});return t},S=e=>{let o={get:async t=>{let n=stringify(t),a=await b(e.credentials)??{},r=await u(`${e.url}/${t.resource}${n?`?${n}`:""}`,{headers:{...a,"Content-Type":"application/json"}});return !r||typeof r!="object"?[]:Object.entries(r).map(([i,s])=>({...f(s),id:i}))},subscribe:()=>{throw new Error("Fetch client does not support subscriptions")}};return {query:Object.entries(e.schema).reduce((t,[n,a])=>(t[n]=c._init(a,o,true),t),{}),mutate:d(()=>{},{apply:async(t,n,a)=>{if(n.length<2)return;if(n.length>2)throw new Error("Trying to access an invalid path");let[r,i]=n,s=await b(e.credentials)??{};if(i==="insert"){let{id:c,...y}=a[0];await u(`${e.url}/${r}/insert`,{method:"POST",headers:{...s,"Content-Type":"application/json"},body:JSON.stringify({resourceId:c,payload:e.schema[r].encodeMutation("set",y,new Date().toISOString())})});return}if(i==="update"){let[c,y]=a,{id:O,...m}=y;await u(`${e.url}/${r}/update`,{method:"POST",headers:{...s,"Content-Type":"application/json"},body:JSON.stringify({resourceId:c,payload:e.schema[r].encodeMutation("set",m,new Date().toISOString())})});return}await u(`${e.url}/${r}/${i}`,{method:"POST",headers:{...s,"Content-Type":"application/json"},body:JSON.stringify({payload:a[0]})});}})}};export{S as createClient};
1
+ import {d as d$1,b,c,f}from'./chunk-RCXJM33Z.js';import {stringify}from'qs';var d=async(e,u,t)=>{let r=o=>{if(!o)return {};if(o instanceof Headers){let y={};return o.forEach((f,O)=>{y[O]=f;}),y}return Array.isArray(o)?Object.fromEntries(o):o},a=r(t==null?void 0:t.headers),n=r(u==null?void 0:u.headers),c={...t,...u,headers:{...a,...n}},i=await fetch(e,c),s;try{s=await i.json();}catch{s=await i.text().catch(()=>{});}if(!i.ok)throw new Error(`Failed to fetch: ${i.status} ${i.statusText}`,{cause:s});return s},$=e=>{let u={get:async t=>{let r=stringify(t),a=await b(e.credentials)??{},n=await d(`${e.url}/${t.resource}${r?`?${r}`:""}`,{headers:{...a,"Content-Type":"application/json"}},e.fetchOptions);return !n||typeof n!="object"?[]:Object.entries(n).map(([c,i])=>({...f(i),id:c}))},subscribe:()=>{throw new Error("Fetch client does not support subscriptions")}};return {query:Object.entries(e.schema).reduce((t,[r,a])=>(t[r]=c._init(a,u,true),t),{}),mutate:d$1(()=>{},{apply:async(t,r,a)=>{if(r.length<2)return;if(r.length>2)throw new Error("Trying to access an invalid path");let[n,c]=r,i=await b(e.credentials)??{};if(c==="insert"){let{id:s,...o}=a[0];await d(`${e.url}/${n}/insert`,{method:"POST",headers:{...i,"Content-Type":"application/json"},body:JSON.stringify({resourceId:s,payload:e.schema[n].encodeMutation("set",o,new Date().toISOString())})},e.fetchOptions);return}if(c==="update"){let[s,o]=a,{id:y,...f}=o;await d(`${e.url}/${n}/update`,{method:"POST",headers:{...i,"Content-Type":"application/json"},body:JSON.stringify({resourceId:s,payload:e.schema[n].encodeMutation("set",f,new Date().toISOString())})},e.fetchOptions);return}await d(`${e.url}/${n}/${c}`,{method:"POST",headers:{...i,"Content-Type":"application/json"},body:JSON.stringify({payload:a[0]})},e.fetchOptions);}})}};export{$ as createClient};
package/dist/server.cjs CHANGED
@@ -1,2 +1,2 @@
1
- 'use strict';require('js-xxhash');var V=require('crypto'),rt=require('qs'),zod=require('zod'),kysely=require('kysely'),postgres=require('kysely/helpers/postgres');function _interopDefault(e){return e&&e.__esModule?e:{default:e}}var V__default=/*#__PURE__*/_interopDefault(V);var rt__default=/*#__PURE__*/_interopDefault(rt);var je=Object.create;var ce=Object.defineProperty;var Ee=Object.getOwnPropertyDescriptor;var $e=Object.getOwnPropertyNames;var Ce=Object.getPrototypeOf,Pe=Object.prototype.hasOwnProperty;var Ne=(i,e)=>()=>(e||i((e={exports:{}}).exports,e),e.exports);var ze=(i,e,t,r)=>{if(e&&typeof e=="object"||typeof e=="function")for(let n of $e(e))!Pe.call(i,n)&&n!==t&&ce(i,n,{get:()=>e[n],enumerable:!(r=Ee(e,n))||r.enumerable});return i};var ue=(i,e,t)=>(t=i!=null?je(Ce(i)):{},ze(ce(t,"default",{value:i,enumerable:true}),i));var Q=Ne(K=>{Object.defineProperty(K,"__esModule",{value:true});K.parse=He;K.serialize=Je;var Ue=/^[\u0021-\u003A\u003C\u003E-\u007E]+$/,ke=/^[\u0021-\u003A\u003C-\u007E]*$/,Be=/^([.]?[a-z0-9]([a-z0-9-]{0,61}[a-z0-9])?)([.][a-z0-9]([a-z0-9-]{0,61}[a-z0-9])?)*$/i,Qe=/^[\u0020-\u003A\u003D-\u007E]*$/,Ge=Object.prototype.toString,Ze=(()=>{let i=function(){};return i.prototype=Object.create(null),i})();function He(i,e){let t=new Ze,r=i.length;if(r<2)return t;let n=(e==null?void 0:e.decode)||Ye,a=0;do{let s=i.indexOf("=",a);if(s===-1)break;let o=i.indexOf(";",a),c=o===-1?r:o;if(s>c){a=i.lastIndexOf(";",s-1)+1;continue}let m=fe(i,a,s),y=he(i,s,m),l=i.slice(m,y);if(t[l]===void 0){let p=fe(i,s+1,c),u=he(i,c,p),h=n(i.slice(p,u));t[l]=h;}a=c+1;}while(a<r);return t}function fe(i,e,t){do{let r=i.charCodeAt(e);if(r!==32&&r!==9)return e}while(++e<t);return t}function he(i,e,t){for(;e>t;){let r=i.charCodeAt(--e);if(r!==32&&r!==9)return e+1}return t}function Je(i,e,t){let r=(t==null?void 0:t.encode)||encodeURIComponent;if(!Ue.test(i))throw new TypeError(`argument name is invalid: ${i}`);let n=r(e);if(!ke.test(n))throw new TypeError(`argument val is invalid: ${e}`);let a=i+"="+n;if(!t)return a;if(t.maxAge!==void 0){if(!Number.isInteger(t.maxAge))throw new TypeError(`option maxAge is invalid: ${t.maxAge}`);a+="; Max-Age="+t.maxAge;}if(t.domain){if(!Be.test(t.domain))throw new TypeError(`option domain is invalid: ${t.domain}`);a+="; Domain="+t.domain;}if(t.path){if(!Qe.test(t.path))throw new TypeError(`option path is invalid: ${t.path}`);a+="; Path="+t.path;}if(t.expires){if(!Xe(t.expires)||!Number.isFinite(t.expires.valueOf()))throw new TypeError(`option expires is invalid: ${t.expires}`);a+="; Expires="+t.expires.toUTCString();}if(t.httpOnly&&(a+="; HttpOnly"),t.secure&&(a+="; Secure"),t.partitioned&&(a+="; Partitioned"),t.priority)switch(typeof t.priority=="string"?t.priority.toLowerCase():void 0){case "low":a+="; Priority=Low";break;case "medium":a+="; Priority=Medium";break;case "high":a+="; Priority=High";break;default:throw new TypeError(`option priority is invalid: ${t.priority}`)}if(t.sameSite)switch(typeof t.sameSite=="string"?t.sameSite.toLowerCase():t.sameSite){case true:case "strict":a+="; SameSite=Strict";break;case "lax":a+="; SameSite=Lax";break;case "none":a+="; SameSite=None";break;default:throw new TypeError(`option sameSite is invalid: ${t.sameSite}`)}return a}function Ye(i){if(i.indexOf("%")===-1)return i;try{return decodeURIComponent(i)}catch{return i}}function Xe(i){return Ge.call(i)==="[object Date]"}});var E=(i,e,t)=>{let r={},n=t[e];if(!n)return r;let a=s=>{s.$and?s.$and.forEach(a):s.$or?s.$or.forEach(a):Object.entries(s).forEach(([o,c])=>{var m;if((m=n.relations)!=null&&m[o]&&(r[o]=true,typeof c=="object"&&c!==null&&!Array.isArray(c))){let y=E(c,n.relations[o].entity.name,t);Object.keys(y).length>0&&(r[o]=y);}});};return a(i),r},w=(i,e,t=false)=>Object.entries(e).every(([r,n])=>{if(r==="$and")return n.every(s=>w(i,s,t));if(r==="$or")return n.some(s=>w(i,s,t));let a=(n==null?void 0:n.$eq)!==void 0?n==null?void 0:n.$eq:n;if(typeof n=="object"&&n!==null&&(n==null?void 0:n.$eq)===void 0){if(n.$in!==void 0){let o=i[r];return o===void 0?false:t?!n.$in.includes(o):n.$in.includes(o)}if(n.$not!==void 0&&!t)return w(i,{[r]:n.$not},true);if(n.$gt!==void 0){let o=i[r];return typeof o!="number"?false:t?o<=n.$gt:o>n.$gt}if(n.$gte!==void 0){let o=i[r];return typeof o!="number"?false:t?o<n.$gte:o>=n.$gte}if(n.$lt!==void 0){let o=i[r];return typeof o!="number"?false:t?o>=n.$lt:o<n.$lt}if(n.$lte!==void 0){let o=i[r];return typeof o!="number"?false:t?o>n.$lte:o<=n.$lte}let s=i[r];return !s||typeof s!="object"&&!Array.isArray(s)?false:Array.isArray(s)?t?!s.some(o=>w(o,n,false)):s.some(o=>w(o,n,false)):w(s,n,t)}return t?i[r]!==a:i[r]===a}),S={CRITICAL:0,ERROR:1,WARN:2,INFO:3,DEBUG:4},k=class{level;prefix;constructor(e={}){this.level=e.level??S.INFO,this.prefix=e.prefix?`[${e.prefix}] `:"";}critical(...e){this.level>=S.CRITICAL&&console.error(`${this.prefix}[CRITICAL]`,...e);}error(...e){this.level>=S.ERROR&&console.error(`${this.prefix}[ERROR]`,...e);}warn(...e){this.level>=S.WARN&&console.warn(`${this.prefix}[WARN]`,...e);}info(...e){this.level>=S.INFO&&console.log(`${this.prefix}[INFO]`,...e);}debug(...e){this.level>=S.DEBUG&&console.log(`${this.prefix}[DEBUG]`,...e);}setLevel(e){this.level=e;}getLevel(){return this.level}},le=i=>new k(i);var ye="0123456789ABCDEFGHJKMNPQRSTVWXYZ",$=32;var Ve=16,pe=10,de=0xffffffffffff;var L;(function(i){i.Base32IncorrectEncoding="B32_ENC_INVALID",i.DecodeTimeInvalidCharacter="DEC_TIME_CHAR",i.DecodeTimeValueMalformed="DEC_TIME_MALFORMED",i.EncodeTimeNegative="ENC_TIME_NEG",i.EncodeTimeSizeExceeded="ENC_TIME_SIZE_EXCEED",i.EncodeTimeValueMalformed="ENC_TIME_MALFORMED",i.PRNGDetectFailure="PRNG_DETECT",i.ULIDInvalid="ULID_INVALID",i.Unexpected="UNEXPECTED",i.UUIDInvalid="UUID_INVALID";})(L||(L={}));var M=class extends Error{constructor(e,t){super(`${t} (${e})`),this.name="ULIDError",this.code=e;}};function _e(i){let e=Math.floor(i()*$);return e===$&&(e=$-1),ye.charAt(e)}function Fe(i){var r;let e=Ke(),t=e&&(e.crypto||e.msCrypto)||(typeof V__default.default<"u"?V__default.default:null);if(typeof(t==null?void 0:t.getRandomValues)=="function")return ()=>{let n=new Uint8Array(1);return t.getRandomValues(n),n[0]/255};if(typeof(t==null?void 0:t.randomBytes)=="function")return ()=>t.randomBytes(1).readUInt8()/255;if((r=V__default.default)!=null&&r.randomBytes)return ()=>V__default.default.randomBytes(1).readUInt8()/255;throw new M(L.PRNGDetectFailure,"Failed to find a reliable PRNG")}function Ke(){return De()?self:typeof window<"u"?window:typeof global<"u"?global:typeof globalThis<"u"?globalThis:null}function We(i,e){let t="";for(;i>0;i--)t=_e(e)+t;return t}function qe(i,e=pe){if(isNaN(i))throw new M(L.EncodeTimeValueMalformed,`Time must be a number: ${i}`);if(i>de)throw new M(L.EncodeTimeSizeExceeded,`Cannot encode a time larger than ${de}: ${i}`);if(i<0)throw new M(L.EncodeTimeNegative,`Time must be positive: ${i}`);if(Number.isInteger(i)===false)throw new M(L.EncodeTimeValueMalformed,`Time must be an integer: ${i}`);let t,r="";for(let n=e;n>0;n--)t=i%$,r=ye.charAt(t)+r,i=(i-t)/$;return r}function De(){return typeof WorkerGlobalScope<"u"&&self instanceof WorkerGlobalScope}function me(i,e){let t=Fe(),r=Date.now();return qe(r,pe)+We(Ve,t)}var B=()=>me().toLowerCase();var _=(...i)=>{let e=i.filter(t=>!!t);return e.length===0?{}:e.length===1?e[0]:{$and:e}};var F=class{storage;queue=new Map;scheduled=false;constructor(e){this.storage=e;}async rawFind({resource:e,commonWhere:t,uniqueWhere:r,...n}){return new Promise((a,s)=>{let o=this.getBatchKey({resource:e,commonWhere:t,...n}),c={resource:e,commonWhere:t,uniqueWhere:r,...n,resolve:a,reject:s};this.queue.has(o)||this.queue.set(o,[]);let m=this.queue.get(o);m&&m.push(c),this.scheduled||(this.scheduled=true,setImmediate(()=>{this.processBatch();}));})}getBatchKey(e){let{resource:t,commonWhere:r,...n}=e;return `${t}:${JSON.stringify(r??{})}:${JSON.stringify(n??{})}`}async processBatch(){this.scheduled=false;let e=Array.from(this.queue.entries());this.queue.clear();for(let[,t]of e)try{await this.executeBatchedRequests(t);}catch(r){t.forEach(n=>{n.reject(r);});}}async executeBatchedRequests(e){var p,u;if(e.length===0)return;let t=e[0],{resource:r,commonWhere:n,include:a,sort:s}=t,o=e.length===1?t.limit:void 0,c=e.map(h=>h.uniqueWhere).filter(h=>h!==void 0),m=n,y=(p=Object.entries(c[0]??{})[0])==null?void 0:p[0];if(c.length>0){let h=c.map(d=>d[y]).filter(d=>d!=null);h.length>0&&(m=_(n,{[y]:{$in:h}}));}let l=await this.storage.rawFind({resource:r,where:m,include:a,sort:s,limit:o});for(let h of e){let d={};if(h.uniqueWhere){let[f,g]=Object.entries(h.uniqueWhere)[0];for(let[x,v]of Object.entries(l))((u=v.value[f])==null?void 0:u.value)===g&&(d[x]=v);}else Object.assign(d,l);h.resolve(d);}}};var Re=ue(Q());var W=zod.z.object({resource:zod.z.string(),where:zod.z.record(zod.z.string(),zod.z.any()).optional(),include:zod.z.record(zod.z.string(),zod.z.any()).optional(),lastSyncedAt:zod.z.string().optional(),limit:zod.z.coerce.number().optional(),sort:zod.z.array(zod.z.object({key:zod.z.string(),direction:zod.z.enum(["asc","desc"])})).optional()}),G=zod.z.record(zod.z.string(),zod.z.object({value:zod.z.string().or(zod.z.number()).or(zod.z.boolean()).or(zod.z.date()).nullable(),_meta:zod.z.object({timestamp:zod.z.string().optional().nullable()}).optional()})),et=G.superRefine((i,e)=>{i.id&&e.addIssue({code:zod.z.ZodIssueCode.custom,message:"Payload cannot have an id"});}),Te=zod.z.object({id:zod.z.string().optional(),type:zod.z.literal("MUTATE"),resource:zod.z.string(),resourceId:zod.z.string().optional()}),C=Te.extend({procedure:zod.z.string(),payload:zod.z.any().optional()}),P=Te.extend({procedure:zod.z.enum(["INSERT","UPDATE"]),payload:et});zod.z.union([P,C]);var ge=W.omit({resource:true}),Z=C.omit({id:true,type:true,resource:true,procedure:true}),H=P.omit({id:true,type:true,resource:true,procedure:true});zod.z.union([H,Z]);var xe=i=>{let e=i.logger;return async t=>{var r;try{let n=typeof t.headers.getSetCookie=="function"?Object.fromEntries(t.headers):t.headers,a={headers:n,cookies:n.cookie?Re.default.parse(n.cookie):{}},s=new URL(t.url),o=s.pathname.split("/"),c=s.searchParams,m=rt__default.default.parse(c.toString()),y=await((r=i.contextProvider)==null?void 0:r.call(i,{transport:"HTTP",headers:a.headers,cookies:a.cookies,queryParams:m}))??{};if(t.method==="GET"){let l=o[o.length-1],{success:p,data:u,error:h}=ge.safeParse(m);if(!p)return Response.json({message:"Invalid query",code:"INVALID_QUERY",details:h},{status:400});let d=await i.handleQuery({req:{...a,...u,type:"QUERY",resource:l,context:y,queryParams:m}});return !d||!d.data?Response.json({message:"Invalid resource",code:"INVALID_RESOURCE"},{status:400}):Response.json(d.data)}if(t.method==="POST")try{let l=o[o.length-1],p=o[o.length-2],u=t.body?await t.json():{},h;if(l==="insert"||l==="update"){let{success:f,data:g,error:x}=H.safeParse(u);if(!f)return Response.json({message:"Invalid mutation",code:"INVALID_REQUEST",details:x},{status:400});h=g;}else {let{success:f,data:g,error:x}=Z.safeParse(u);if(!f)return Response.json({message:"Invalid mutation",code:"INVALID_REQUEST",details:x},{status:400});h=g;}let d=await i.handleMutation({req:{...a,type:"MUTATE",resource:p,input:h.payload,context:y,resourceId:h.resourceId,procedure:l==="insert"||l==="update"?l.toUpperCase():l,queryParams:{}}});return Response.json(d)}catch(l){return e.error("Error parsing mutation from the client:",l),Response.json({message:"Internal server error",code:"INTERNAL_SERVER_ERROR"},{status:500})}return Response.json({message:"Not found",code:"NOT_FOUND"},{status:404})}catch(n){return e.error("Unexpected error:",n),Response.json({message:"Internal server error",code:"INTERNAL_SERVER_ERROR"},{status:500})}}};var we=ue(Q());var A=zod.z.string(),nt=zod.z.object({id:A,type:zod.z.literal("SUBSCRIBE"),resource:zod.z.string()}),it=W.extend({id:A,type:zod.z.literal("QUERY")}),be=P.extend({id:A}),at=C.extend({id:A}),ot=zod.z.union([at,be]),ve=zod.z.union([nt,it,ot]),st=zod.z.object({id:A,type:zod.z.literal("REJECT"),resource:zod.z.string(),message:zod.z.string().optional()}),ct=zod.z.object({id:A,type:zod.z.literal("REPLY"),data:zod.z.any()});zod.z.union([st,ct,be]);zod.z.object({resource:zod.z.string(),data:zod.z.record(zod.z.string(),G)});var Se=i=>{let e={},t={},r=i.logger;return i.subscribeToMutations(n=>{let a=n;!a.resourceId||!a.payload||(r.debug("Mutation propagated:",a),Object.entries(t[a.resource]??{}).forEach(([s,o])=>{var c;(c=e[s])==null||c.send(JSON.stringify({...a,id:a.id??B()}));}));}),(n,a)=>{var l;let s=p=>{n.send(JSON.stringify(p));},o=B(),c={headers:a.headers,cookies:typeof a.headers.cookie=="string"?we.default.parse(a.headers.cookie):{}},m=rt.parse(a.url.split("?")[1]),y=(l=i.contextProvider)==null?void 0:l.call(i,{transport:"WEBSOCKET",headers:c.headers,cookies:c.cookies,queryParams:m});e[o]=n,r.info("Client connected:",o),n.on("message",async p=>{try{r.debug("Message received from the client:",p);let u=ve.parse(JSON.parse(p.toString()));if(u.type==="SUBSCRIBE"){let{resource:h}=u;t[h]||(t[h]={}),t[h][o]={};}else if(u.type==="QUERY"){let{resource:h}=u,d=await i.handleQuery({req:{...c,type:"QUERY",resource:h,context:await y??{},queryParams:m}});if(!d||!d.data)throw new Error("Invalid resource");s({id:u.id,type:"REPLY",data:{resource:h,data:Object.fromEntries(Object.entries(d.data??{}).map(([f,g])=>[f,g.value]))}});}else if(u.type==="MUTATE"){let{resource:h}=u;r.debug("Received mutation from client:",u);try{let d=await i.handleMutation({req:{...c,type:"MUTATE",resource:h,input:u.payload,context:{messageId:u.id,...await y??{}},resourceId:u.resourceId,procedure:u.procedure,queryParams:m}});u.procedure&&u.procedure!=="INSERT"&&u.procedure!=="UPDATE"&&s({id:u.id,type:"REPLY",data:d});}catch(d){s({id:u.id,type:"REJECT",resource:h,message:d.message}),r.error("Error parsing mutation from the client:",d);}}}catch(u){r.error("Error handling message from the client:",u);}}),n.on("close",()=>{r.info("Connection closed",o),delete e[o];for(let p of Object.values(t))delete p[o];});}};function Ie(i){let e=`${i.protocol}://${i.hostname}${i.url}`,t=new Headers;return Object.entries(i.headers).forEach(([r,n])=>{n&&t.set(r,Array.isArray(n)?n.join(","):n);}),new Request(e,{method:i.method,headers:t,body:i.body&&i.method!=="GET"?JSON.stringify(i.body):void 0})}var Zt=(i,e,t)=>{i.ws(`${(t==null?void 0:t.basePath)??""}/ws`,Se(e)),i.use(`${(t==null?void 0:t.basePath)??""}/`,(r,n)=>{xe(e)(Ie(r)).then(s=>s.json().then(o=>n.status(s.status).send(o)));});};var b=i=>i?Array.isArray(i.value)?i.value.map(t=>b(t)):typeof i.value!="object"||i.value===null||i.value instanceof Date?i.value:Object.fromEntries(Object.entries(i.value).map(([t,r])=>Array.isArray(r)?[t,r.map(n=>b(n))]:[t,b(r)])):void 0;var re=class i{routes;constructor(e){this.routes=e.routes;}static create(e){return new i(e)}},fr=i=>re.create({...i}),dt=i=>({handler:e=>({inputValidator:i??zod.z.undefined(),handler:e})}),ne=class i{resourceSchema;middlewares;customMutations;authorization;constructor(e,t,r){this.resourceSchema=e,this.middlewares=new Set,this.customMutations=t??{},this.authorization=r;}use(...e){for(let t of e)this.middlewares.add(t);return this}withMutations(e){return new i(this.resourceSchema,e({mutation:dt}),this.authorization)}handleQuery=async({req:e,batcher:t})=>await this.wrapInMiddlewares(async r=>{var a,s;let n=(s=(a=this.authorization)==null?void 0:a.read)==null?void 0:s.call(a,{ctx:r.context});if(typeof n=="boolean"&&!n)throw new Error("Not authorized");return {data:await t.rawFind({resource:r.resource,commonWhere:_(r.where,typeof n=="object"?n:void 0),uniqueWhere:r.relationalWhere,include:r.include,limit:r.limit,sort:r.sort})}})(e);handleMutation=async({req:e,db:t,schema:r})=>await this.wrapInMiddlewares(async n=>{if(!n.procedure)throw new Error("Procedure is required for mutations");let a=this.customMutations[n.procedure];if(a){let s=a.inputValidator.parse(n.input);return n.input=s,a.handler({req:n,db:t})}else {if(n.procedure==="INSERT"||n.procedure==="UPDATE")return this.handleSet({req:n,db:t,operation:n.procedure,schema:r});throw new Error(`Unknown procedure: ${n.procedure}`)}})(e);handleSet=async({req:e,db:t,operation:r,schema:n})=>{if(!e.input)throw new Error("Payload is required");if(!e.resourceId)throw new Error("ResourceId is required");let a=await t.rawFindById(e.resource,e.resourceId);if(r==="INSERT"&&a)throw new Error("Resource already exists");if(r==="UPDATE"&&!a)throw new Error("Resource not found");return t.transaction(async({trx:s})=>{var y,l,p,u,h;let[o,c]=this.resourceSchema.mergeMutation("set",e.input,a);if(!c)throw new Error("Mutation rejected");if(r==="INSERT"){let d=await s.rawInsert(e.resource,e.resourceId,o),f=b(d);if(f.id=f.id??e.resourceId,(y=this.authorization)!=null&&y.insert){let g=this.authorization.insert({ctx:e.context,value:f});if(typeof g=="boolean"){if(!g)throw new Error("Not authorized")}else {let x=E(g,e.resource,n),v=Object.keys(x).length>0?await s.rawFindById(e.resource,e.resourceId,x):d,I=b(v);if(I.id=I.id??e.resourceId,!w(I,g))throw new Error("Not authorized")}}return {data:d,acceptedValues:c}}if((p=(l=this.authorization)==null?void 0:l.update)!=null&&p.preMutation){let d=b(a);d.id=d.id??e.resourceId;let f=this.authorization.update.preMutation({ctx:e.context,value:d});if(typeof f=="boolean"){if(!f)throw new Error("Not authorized")}else {let g=E(f,e.resource,n),x=Object.keys(g).length>0?await s.rawFindById(e.resource,e.resourceId,g):a,v=b(x);if(v.id=v.id??e.resourceId,!w(v,f))throw new Error("Not authorized")}}let m=await s.rawUpdate(e.resource,e.resourceId,o);if((h=(u=this.authorization)==null?void 0:u.update)!=null&&h.postMutation){let d=b(m);d.id=d.id??e.resourceId;let f=this.authorization.update.postMutation({ctx:e.context,value:d});if(typeof f=="boolean"){if(!f)throw new Error("Not authorized")}else {let g=E(f,e.resource,n),x=Object.keys(g).length>0?await s.rawFindById(e.resource,e.resourceId,g):m,v=b(x);if(v.id=v.id??e.resourceId,!w(v,f))throw new Error("Not authorized")}}return {data:m,acceptedValues:c}})};wrapInMiddlewares(e){return t=>Array.from(this.middlewares.values()).reduceRight((r,n)=>a=>n({req:a,next:r}),e)(t)}},ie=class i{middlewares;constructor(e=[]){this.middlewares=e;}collectionRoute(e,t){return new ne(e,void 0,t).use(...this.middlewares)}use(...e){return new i([...this.middlewares,...e])}static create(){return new i}},hr=ie.create;var z=class{async insert(e,t){let r=new Date().toISOString();return b(await this.rawInsert(e.name,t.id,{value:Object.fromEntries(Object.entries(t).map(([n,a])=>[n,{value:a,_meta:{timestamp:r}}]))}))}async update(e,t,r){let n=new Date().toISOString(),{id:a,...s}=r;return b(await this.rawUpdate(e.name,t,{value:Object.fromEntries(Object.entries(s).map(([o,c])=>[o,{value:c,_meta:{timestamp:n}}]))}))}};function q(i,e,t,r){if(!i)throw new Error("Schema not initialized");let n=i[e];if(!n)throw new Error("Resource not found");let a=r.$or,s=r.$and;return (a?t.or:t.and)(a?r.$or.map(o=>q(i,e,t,o)):s?r.$and.map(o=>q(i,e,t,o)):Object.entries(r).map(([o,c])=>{var m,y;if(n.fields[o])return (c==null?void 0:c.$eq)!==void 0?t(`${e}.${o}`,c.$eq===null?"is":"=",c.$eq):(c==null?void 0:c.$in)!==void 0?t(`${e}.${o}`,"in",c.$in):(c==null?void 0:c.$not)!==void 0?((m=c==null?void 0:c.$not)==null?void 0:m.$in)!==void 0?t(`${e}.${o}`,"not in",c.$not.$in):((y=c==null?void 0:c.$not)==null?void 0:y.$eq)!==void 0?t(`${e}.${o}`,c.$not.$eq===null?"is not":"!=",c.$not.$eq):t(`${e}.${o}`,c.$not===null?"is not":"!=",c.$not):(c==null?void 0:c.$gt)!==void 0?t(`${e}.${o}`,">",c.$gt):(c==null?void 0:c.$gte)!==void 0?t(`${e}.${o}`,">=",c.$gte):(c==null?void 0:c.$lt)!==void 0?t(`${e}.${o}`,"<",c.$lt):(c==null?void 0:c.$lte)!==void 0?t(`${e}.${o}`,"<=",c.$lte):t(`${e}.${o}`,c===null?"is":"=",c);if(n.relations[o]){let l=n.relations[o],p=l.entity.name;return l.type==="many"?t.exists(ae(i,p,t.selectFrom(p).select("id").whereRef(l.foreignColumn,"=",`${e}.id`),c)):q(i,p,t,c)}return null}).filter(Boolean))}function D(i,e,t,r){let n=i[e];if(!n)throw new Error("Resource not found");if(!r)return t;if(r.$and){for(let a of r.$and)t=D(i,e,t,a);return t}else if(r.$or){for(let a of r.$or)t=D(i,e,t,a);return t}for(let[a,s]of Object.entries(r)){if(!n.relations[a])continue;let o=n.relations[a],c=o.entity.name,m=o.type==="one"?"id":o.foreignColumn,y=o.type==="one"?o.relationalColumn:"id";t=t.leftJoin(c,`${c}.${m}`,`${e}.${y}`),s instanceof Object&&!Array.isArray(s)&&s!==null&&(t=D(i,c,t,s));}return t}function ae(i,e,t,r){return !r||Object.keys(r).length===0?t:(t=D(i,e,t,r),t.where(n=>q(i,e,n,r)))}function U(i,e,t,r){if(!r)return t;if(!i)throw new Error("Schema not initialized");let n=i[e];if(!n)throw new Error(`Resource not found: ${e}`);for(let a of Object.keys(r)){if(!n.relations[a])throw new Error(`Relation ${a} not found in resource ${e}`);let s=n.relations[a],o=s.entity.name,c=r[a],m=s.type==="one"?"id":s.foreignColumn,y=s.type==="one"?s.relationalColumn:"id",l=s.type==="one"?postgres.jsonObjectFrom:postgres.jsonArrayFrom,p=typeof c=="object"&&c!==null;t=t.select(u=>{let h=u.selectFrom(o).selectAll(o).whereRef(`${o}.${m}`,"=",`${e}.${y}`).select(d=>postgres.jsonObjectFrom(d.selectFrom(`${o}_meta`).selectAll(`${o}_meta`).whereRef(`${o}_meta.id`,"=",`${o}.id`)).as("_meta"));return p&&(h=U(i,o,h,c)),l(h).as(a)});}return t}var oe=class i extends z{db;schema;logger;constructor(e,t,r){super(),this.isKyselyLike(e)?this.db=e:this.db=new kysely.Kysely({dialect:new kysely.PostgresDialect({pool:e})}),this.schema=t,this.logger=r,this.rawInsert=this.rawInsert.bind(this),this.rawUpdate=this.rawUpdate.bind(this);}async init(e,t){var n;this.schema=e,this.logger=t;let r=await this.db.introspection.getTables();for(let[a,s]of Object.entries(e)){let o=r.find(y=>y.name===a);o||await this.db.schema.createTable(a).ifNotExists().execute();let c=`${a}_meta`,m=r.find(y=>y.name===c);m||await this.db.schema.createTable(c).ifNotExists().execute();for(let[y,l]of Object.entries(s.fields)){let p=o==null?void 0:o.columns.find(d=>d.name===y),u=l.getStorageFieldType();p?p.dataType!==u.type&&((n=this.logger)==null||n.warn("Column type mismatch:",y,"expected to have type:",u.type,"but has type:",p.dataType)):(await this.db.schema.alterTable(a).addColumn(y,u.type,d=>{let f=d;return u.unique&&(f=f.unique()),u.nullable||(f=f.notNull()),u.references&&(f=f.references(u.references)),u.primary&&(f=f.primaryKey()),u.default!==void 0&&(f=f.defaultTo(u.default)),f}).execute().catch(d=>{var f;throw (f=this.logger)==null||f.error("Error adding column",y,d),d}),u.index&&await this.db.schema.createIndex(`${a}_${y}_index`).on(a).column(y).execute().catch(()=>{})),(m==null?void 0:m.columns.find(d=>d.name===y))||await this.db.schema.alterTable(c).addColumn(y,"varchar",d=>{let f=d;return u.primary&&(f=f.primaryKey().references(`${a}.${y}`)),f}).execute();}}}async rawFindById(e,t,r){if(!this.schema)throw new Error("Schema not initialized");let n=await this.db.selectFrom(e).where("id","=",t).selectAll(e).select(s=>postgres.jsonObjectFrom(s.selectFrom(`${e}_meta`).selectAll(`${e}_meta`).whereRef(`${e}_meta.id`,"=",`${e}.id`)).as("_meta"));n=U(this.schema,e,n,r);let a=await n.executeTakeFirst();if(a)return this.convertToMaterializedLiveType(a)}async findOne(e,t,r){let n=await this.rawFindById(e.name,t,r==null?void 0:r.include);if(n)return b(n)}async rawFind(e){if(!this.schema)throw new Error("Schema not initialized");let{resource:t,where:r,include:n,limit:a,sort:s}=e,o=this.db.selectFrom(t).selectAll(t).select(l=>postgres.jsonObjectFrom(l.selectFrom(`${t}_meta`).selectAll(`${t}_meta`).whereRef(`${t}_meta.id`,"=",`${t}.id`)).as("_meta"));o=ae(this.schema,t,o,r),o=U(this.schema,t,o,n),a!==void 0&&(o=o.limit(a)),s!==void 0&&s.forEach(l=>{o=o.orderBy(l.key,l.direction);});let c=await o.execute(),m=Object.fromEntries(c.map(l=>{let{id:p}=l;return [p,l]}));return Object.keys(m).length===0?{}:Object.entries(m).reduce((l,[p,u])=>(l[p]=this.convertToMaterializedLiveType(u),l),{})}async find(e,t){let r=await this.rawFind({resource:e.name,where:t==null?void 0:t.where,include:t==null?void 0:t.include,limit:t==null?void 0:t.limit,sort:t==null?void 0:t.sort});return Object.fromEntries(Object.entries(r).map(([n,a])=>[n,b(a)]))}async rawInsert(e,t,r){var s;let n={},a={};for(let[o,c]of Object.entries(r.value)){let m=(s=c._meta)==null?void 0:s.timestamp;m&&(n[o]=c.value,a[o]=m);}return await this.db.insertInto(e).values({...n,id:t}).execute().then(()=>{this.db.insertInto(`${e}_meta`).values({...a,id:t}).execute();}),r}async rawUpdate(e,t,r){var s;let n={},a={};for(let[o,c]of Object.entries(r.value)){let m=(s=c._meta)==null?void 0:s.timestamp;m&&(n[o]=c.value,a[o]=m);}return await Promise.all([this.db.updateTable(e).set(n).where("id","=",t).execute(),this.db.insertInto(`${e}_meta`).values({...a,id:t}).onConflict(o=>o.column("id").doUpdateSet(a)).execute()]),r}async transaction(e){if(!this.schema)throw new Error("Schema not initialized");if(this.db.isTransaction){let r=Math.random().toString(36).substring(2,15),n=await this.db.savepoint(r).execute();try{return await e({trx:this,commit:()=>n.releaseSavepoint(r).execute().then(()=>{}),rollback:()=>n.rollbackToSavepoint(r).execute().then(()=>{})}).then(a=>n.isCommitted||n.isRolledBack?a:n.releaseSavepoint(r).execute().then(()=>a))}catch(a){throw await n.rollbackToSavepoint(r).execute().catch(()=>{}),a}}let t=await this.db.startTransaction().execute();try{return await e({trx:new i(t,this.schema,this.logger),commit:()=>t.commit().execute(),rollback:()=>t.rollback().execute()}).then(r=>t.isCommitted||t.isRolledBack?r:t.commit().execute().then(()=>r))}catch(r){throw await t.rollback().execute(),r}}convertToMaterializedLiveType(e){return {value:Object.entries(e).reduce((t,[r,n])=>{var a,s,o;return r==="_meta"||(r==="id"?t[r]={value:n}:Array.isArray(n)?t[r]={value:n.map(c=>this.convertToMaterializedLiveType(c)),_meta:{timestamp:(a=e==null?void 0:e._meta)==null?void 0:a[r]}}:typeof n=="object"&&n!==null&&!(n instanceof Date)?t[r]={...this.convertToMaterializedLiveType(n),_meta:{timestamp:(s=e==null?void 0:e._meta)==null?void 0:s[r]}}:t[r]={value:n,_meta:{timestamp:(o=e==null?void 0:e._meta)==null?void 0:o[r]}}),t},{})}}isKyselyLike(e){if(e instanceof kysely.Kysely)return true;if(!e||typeof e!="object")return false;let t=e,r=typeof t.selectFrom=="function",n=typeof t.startTransaction=="function",a=typeof t.savepoint=="function",s=typeof t.isTransaction=="boolean"||typeof t.isTransaction=="function";return r&&n||a&&s}};var se=class i{router;storage;schema;middlewares=new Set;logger;contextProvider;mutationSubscriptions=new Set;constructor(e){var t;this.router=e.router,this.storage=e.storage,this.schema=e.schema,this.logger=le({level:e.logLevel??S.INFO}),(t=e.middlewares)==null||t.forEach(r=>{this.middlewares.add(r);}),this.storage.init(this.schema,this.logger),this.contextProvider=e.contextProvider;}static create(e){return new i(e)}subscribeToMutations(e){return this.mutationSubscriptions.add(e),()=>{this.mutationSubscriptions.delete(e);}}handleQuery(e){let t=new F(this.storage);return this.wrapInMiddlewares(async r=>{var m;let n=Ae(r,this.schema,{stepId:"query",collectionName:r.resource,included:Object.keys(r.include??{})}),a={headers:r.headers,cookies:r.cookies,queryParams:r.queryParams,context:r.context},s={};for(let y=0;y<n.length;y++){let l=n[y],p=this.router.routes[l.resource];if(!p)throw new Error("Invalid resource");let u=l.getWhere&&l.referenceGetter?l.referenceGetter(s).map(l.getWhere):[void 0],h=(m=s[l.prevStepId??""])==null?void 0:m.flatMap(g=>{var x;return Object.keys(((x=g==null?void 0:g.result)==null?void 0:x.data)??{})}),f=(await Promise.allSettled(u.map(async(g,x)=>{let v=h==null?void 0:h[x],I=await p.handleQuery({req:{type:"QUERY",...l,...a,where:l.where,relationalWhere:g},batcher:t});return {includedBy:v,result:I}}))).flatMap(g=>g.status==="fulfilled"?[g.value]:[]);s[l.stepId]=f;}let o=Object.fromEntries(Object.entries(s).flatMap(([y,l],p)=>l.flatMap(u=>Object.entries(u.result.data).map(([h,d])=>[`${y}.${h}`,{data:d,includedBy:y!=="query"&&u.includedBy?`${y.split(".").slice(0,-1).join(".")}.${u.includedBy}`:void 0,path:y.split(".").slice(-1)[0],isMany:n[p].isMany,collectionName:n[p].collectionName,included:n[p].included}]))));return Object.keys(o).reduceRight((y,l)=>{var h,d;let p=o[l],u=p.path;if(u==="query"&&(y.data[l.replace("query.","")]=p.data),p.included.length)for(let f of p.included)p.data.value[f]??=((d=(h=this.schema[p.collectionName])==null?void 0:h.relations[f])==null?void 0:d.type)==="many"?{value:[]}:{value:null};if(p.includedBy){let f=o[p.includedBy];if(!f)return y;p.isMany?(f.data.value[u]??={value:[]},f.data.value[u].value.push(p.data)):f.data.value[u]=p.data;}return y},{data:{}})})(e.req)}async handleMutation(e){let t=await this.wrapInMiddlewares(async r=>{let n=this.router.routes[r.resource];if(!n)throw new Error("Invalid resource");return n.handleMutation({req:r,db:this.storage,schema:this.schema})})(e.req);if(t&&e.req.type==="MUTATE"&&t.acceptedValues&&(e.req.procedure==="INSERT"||e.req.procedure==="UPDATE")&&e.req.resourceId){let n=t.acceptedValues??{},a=e.req,s=a.resourceId;Object.keys(n).length&&s&&this.mutationSubscriptions.forEach(o=>{o({id:e.req.context.messageId,type:"MUTATE",resource:a.resource,payload:n,resourceId:s,procedure:a.procedure});});}return t}use(e){return this.middlewares.add(e),this}context(e){return this.contextProvider=e,this}wrapInMiddlewares(e){return t=>Array.from(this.middlewares.values()).reduceRight((r,n)=>a=>n({req:a,next:r}),e)(t)}},Cr=se.create;function Ae(i,e,t){let{include:r,where:n,...a}=i,{stepId:s}=t,o=[{...a,...t,where:n}];if(r&&typeof r=="object"&&Object.keys(r).length>0){let c=e[a.resource];if(!c)throw new Error(`Resource ${a.resource} not found`);o.push(...Object.entries(r).flatMap(([m,y])=>{let l=c.relations[m];if(!l)throw new Error(`Relation ${m} not found for resource ${a.resource}`);let p=l.entity.name;return Ae({...a,resource:p,include:y},e,{getWhere:l.type==="one"?u=>({id:u}):u=>({[l.foreignColumn]:u}),referenceGetter:u=>u[s].flatMap(h=>h.result.data?l.type==="one"?Object.values(h.result.data).map(d=>{var f,g;return (g=(f=d.value)==null?void 0:f[l.relationalColumn])==null?void 0:g.value}):Object.keys(h.result.data):[]),stepId:`${s}.${m}`,prevStepId:s,isMany:l.type==="many",collectionName:p,included:typeof y=="object"?Object.keys(y):[]})}));}return o}
2
- exports.Route=ne;exports.RouteFactory=ie;exports.Router=re;exports.SQLStorage=oe;exports.Server=se;exports.Storage=z;exports.expressAdapter=Zt;exports.routeFactory=hr;exports.router=fr;exports.server=Cr;
1
+ 'use strict';require('js-xxhash');var k=require('crypto'),dt=require('qs'),zod=require('zod'),kysely=require('kysely'),postgres=require('kysely/helpers/postgres');function _interopDefault(e){return e&&e.__esModule?e:{default:e}}var k__default=/*#__PURE__*/_interopDefault(k);var dt__default=/*#__PURE__*/_interopDefault(dt);var Ke=Object.create;var he=Object.defineProperty;var Fe=Object.getOwnPropertyDescriptor;var We=Object.getOwnPropertyNames;var De=Object.getPrototypeOf,qe=Object.prototype.hasOwnProperty;var Ue=(n,e)=>()=>(e||n((e={exports:{}}).exports,e),e.exports);var ke=(n,e,t,r)=>{if(e&&typeof e=="object"||typeof e=="function")for(let i of We(e))!qe.call(n,i)&&i!==t&&he(n,i,{get:()=>e[i],enumerable:!(r=Fe(e,i))||r.enumerable});return n};var Te=(n,e,t)=>(t=n!=null?Ke(De(n)):{},ke(he(t,"default",{value:n,enumerable:true}),n));var te=Ue(G=>{Object.defineProperty(G,"__esModule",{value:true});G.parse=at;G.serialize=ot;var Xe=/^[\u0021-\u003A\u003C\u003E-\u007E]+$/,et=/^[\u0021-\u003A\u003C-\u007E]*$/,tt=/^([.]?[a-z0-9]([a-z0-9-]{0,61}[a-z0-9])?)([.][a-z0-9]([a-z0-9-]{0,61}[a-z0-9])?)*$/i,rt=/^[\u0020-\u003A\u003D-\u007E]*$/,nt=Object.prototype.toString,it=(()=>{let n=function(){};return n.prototype=Object.create(null),n})();function at(n,e){let t=new it,r=n.length;if(r<2)return t;let i=(e==null?void 0:e.decode)||st,a=0;do{let s=n.indexOf("=",a);if(s===-1)break;let o=n.indexOf(";",a),c=o===-1?r:o;if(s>c){a=n.lastIndexOf(";",s-1)+1;continue}let y=we(n,a,s),m=Se(n,s,y),p=n.slice(y,m);if(t[p]===void 0){let h=we(n,s+1,c),l=Se(n,c,h),f=i(n.slice(h,l));t[p]=f;}a=c+1;}while(a<r);return t}function we(n,e,t){do{let r=n.charCodeAt(e);if(r!==32&&r!==9)return e}while(++e<t);return t}function Se(n,e,t){for(;e>t;){let r=n.charCodeAt(--e);if(r!==32&&r!==9)return e+1}return t}function ot(n,e,t){let r=(t==null?void 0:t.encode)||encodeURIComponent;if(!Xe.test(n))throw new TypeError(`argument name is invalid: ${n}`);let i=r(e);if(!et.test(i))throw new TypeError(`argument val is invalid: ${e}`);let a=n+"="+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(!tt.test(t.domain))throw new TypeError(`option domain is invalid: ${t.domain}`);a+="; Domain="+t.domain;}if(t.path){if(!rt.test(t.path))throw new TypeError(`option path is invalid: ${t.path}`);a+="; Path="+t.path;}if(t.expires){if(!ct(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 st(n){if(n.indexOf("%")===-1)return n;try{return decodeURIComponent(n)}catch{return n}}function ct(n){return nt.call(n)==="[object Date]"}});var K=(n,e,t)=>{let r={},i=t[e];if(!i)return r;let a=s=>{s.$and?s.$and.forEach(a):s.$or?s.$or.forEach(a):Object.entries(s).forEach(([o,c])=>{var y;if((y=i.relations)!=null&&y[o]&&(r[o]=true,typeof c=="object"&&c!==null&&!Array.isArray(c))){let m=K(c,i.relations[o].entity.name,t);Object.keys(m).length>0&&(r[o]=m);}});};return a(n),r},M=(n,e,t=false)=>Object.entries(e).every(([r,i])=>{if(r==="$and")return i.every(s=>M(n,s,t));if(r==="$or")return i.some(s=>M(n,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=n[r];return o===void 0?false:t?!i.$in.includes(o):i.$in.includes(o)}if(i.$not!==void 0&&!t)return M(n,{[r]:i.$not},true);if(i.$gt!==void 0){let o=n[r];return typeof o!="number"?false:t?o<=i.$gt:o>i.$gt}if(i.$gte!==void 0){let o=n[r];return typeof o!="number"?false:t?o<i.$gte:o>=i.$gte}if(i.$lt!==void 0){let o=n[r];return typeof o!="number"?false:t?o>=i.$lt:o<i.$lt}if(i.$lte!==void 0){let o=n[r];return typeof o!="number"?false:t?o>i.$lte:o<=i.$lte}let s=n[r];return !s||typeof s!="object"&&!Array.isArray(s)?false:Array.isArray(s)?t?!s.some(o=>M(o,i,false)):s.some(o=>M(o,i,false)):M(s,i,t)}return t?n[r]!==a:n[r]===a}),j={CRITICAL:0,ERROR:1,WARN:2,INFO:3,DEBUG:4},X=class{level;prefix;constructor(e={}){this.level=e.level??j.INFO,this.prefix=e.prefix?`[${e.prefix}] `:"";}critical(...e){this.level>=j.CRITICAL&&console.error(`${this.prefix}[CRITICAL]`,...e);}error(...e){this.level>=j.ERROR&&console.error(`${this.prefix}[ERROR]`,...e);}warn(...e){this.level>=j.WARN&&console.warn(`${this.prefix}[WARN]`,...e);}info(...e){this.level>=j.INFO&&console.log(`${this.prefix}[INFO]`,...e);}debug(...e){this.level>=j.DEBUG&&console.log(`${this.prefix}[DEBUG]`,...e);}setLevel(e){this.level=e;}getLevel(){return this.level}},ge=n=>new X(n);var xe="0123456789ABCDEFGHJKMNPQRSTVWXYZ",F=32;var Be=16,be=10,Re=0xffffffffffff;var $;(function(n){n.Base32IncorrectEncoding="B32_ENC_INVALID",n.DecodeTimeInvalidCharacter="DEC_TIME_CHAR",n.DecodeTimeValueMalformed="DEC_TIME_MALFORMED",n.EncodeTimeNegative="ENC_TIME_NEG",n.EncodeTimeSizeExceeded="ENC_TIME_SIZE_EXCEED",n.EncodeTimeValueMalformed="ENC_TIME_MALFORMED",n.PRNGDetectFailure="PRNG_DETECT",n.ULIDInvalid="ULID_INVALID",n.Unexpected="UNEXPECTED",n.UUIDInvalid="UUID_INVALID";})($||($={}));var C=class extends Error{constructor(e,t){super(`${t} (${e})`),this.name="ULIDError",this.code=e;}};function Qe(n){let e=Math.floor(n()*F);return e===F&&(e=F-1),xe.charAt(e)}function Ge(n){var r;let e=Ze(),t=e&&(e.crypto||e.msCrypto)||(typeof k__default.default<"u"?k__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((r=k__default.default)!=null&&r.randomBytes)return ()=>k__default.default.randomBytes(1).readUInt8()/255;throw new C($.PRNGDetectFailure,"Failed to find a reliable PRNG")}function Ze(){return Ye()?self:typeof window<"u"?window:typeof global<"u"?global:typeof globalThis<"u"?globalThis:null}function He(n,e){let t="";for(;n>0;n--)t=Qe(e)+t;return t}function Je(n,e=be){if(isNaN(n))throw new C($.EncodeTimeValueMalformed,`Time must be a number: ${n}`);if(n>Re)throw new C($.EncodeTimeSizeExceeded,`Cannot encode a time larger than ${Re}: ${n}`);if(n<0)throw new C($.EncodeTimeNegative,`Time must be positive: ${n}`);if(Number.isInteger(n)===false)throw new C($.EncodeTimeValueMalformed,`Time must be an integer: ${n}`);let t,r="";for(let i=e;i>0;i--)t=n%F,r=xe.charAt(t)+r,n=(n-t)/F;return r}function Ye(){return typeof WorkerGlobalScope<"u"&&self instanceof WorkerGlobalScope}function ve(n,e){let t=Ge(),r=Date.now();return Je(r,be)+He(Be,t)}var ee=()=>ve().toLowerCase();var B=(...n)=>{let e=n.filter(t=>!!t);return e.length===0?{}:e.length===1?e[0]:{$and:e}};var Q=class{storage;queue=new Map;scheduled=false;constructor(e){this.storage=e;}async rawFind({resource:e,commonWhere:t,uniqueWhere:r,...i}){return new Promise((a,s)=>{let o=this.getBatchKey({resource:e,commonWhere:t,...i}),c={resource:e,commonWhere:t,uniqueWhere:r,...i,resolve:a,reject:s};this.queue.has(o)||this.queue.set(o,[]);let y=this.queue.get(o);y&&y.push(c),this.scheduled||(this.scheduled=true,setImmediate(()=>{this.processBatch();}));})}getBatchKey(e){let{resource:t,commonWhere:r,...i}=e;return `${t}:${JSON.stringify(r??{})}:${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(r){t.forEach(i=>{i.reject(r);});}}async executeBatchedRequests(e){var h,l;if(e.length===0)return;let t=e[0],{resource:r,commonWhere:i,include:a,sort:s}=t,o=e.length===1?t.limit:void 0,c=e.map(f=>f.uniqueWhere).filter(f=>f!==void 0),y=i,m=(h=Object.entries(c[0]??{})[0])==null?void 0:h[0];if(c.length>0){let f=c.map(d=>d[m]).filter(d=>d!=null);f.length>0&&(y=B(i,{[m]:{$in:f}}));}let p=await this.storage.rawFind({resource:r,where:y,include:a,sort:s,limit:o});for(let f of e){let d={};if(f.uniqueWhere){let[u,T]=Object.entries(f.uniqueWhere)[0];for(let[g,x]of Object.entries(p))((l=x.value[u])==null?void 0:l.value)===T&&(d[g]=x);}else Object.assign(d,p);f.resolve(d);}}};var Me=Te(te());var Z=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()}),re=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()})),ut=re.superRefine((n,e)=>{n.id&&e.addIssue({code:zod.z.ZodIssueCode.custom,message:"Payload cannot have an id"});}),Ie=zod.z.object({id:zod.z.string().optional(),type:zod.z.literal("MUTATE"),resource:zod.z.string(),resourceId:zod.z.string().optional()}),W=Ie.extend({procedure:zod.z.string(),payload:zod.z.any().optional()}),D=Ie.extend({procedure:zod.z.enum(["INSERT","UPDATE"]),payload:ut});zod.z.union([D,W]);var Le=Z.omit({resource:true}),ne=W.omit({id:true,type:true,resource:true,procedure:true}),ie=D.omit({id:true,type:true,resource:true,procedure:true});zod.z.union([ie,ne]);var Oe=n=>{let e=n.logger;return async t=>{var r;try{let i=typeof t.headers.getSetCookie=="function"?Object.fromEntries(t.headers):t.headers,a={headers:i,cookies:i.cookie?Me.default.parse(i.cookie):{}},s=new URL(t.url),o=s.pathname.split("/"),c=s.searchParams,y=dt__default.default.parse(c.toString()),m=await((r=n.contextProvider)==null?void 0:r.call(n,{transport:"HTTP",headers:a.headers,cookies:a.cookies,queryParams:y}))??{};if(t.method==="GET"){let p=o[o.length-1],{success:h,data:l,error:f}=Le.safeParse(y);if(!h)return Response.json({message:"Invalid query",code:"INVALID_QUERY",details:f},{status:400});let d=await n.handleQuery({req:{...a,...l,type:"QUERY",resource:p,context:m,queryParams:y}});return !d||!d.data?Response.json({message:"Invalid resource",code:"INVALID_RESOURCE"},{status:400}):Response.json(d.data)}if(t.method==="POST")try{let p=o[o.length-1],h=o[o.length-2],l=t.body?await t.json():{},f;if(p==="insert"||p==="update"){let{success:u,data:T,error:g}=ie.safeParse(l);if(!u)return Response.json({message:"Invalid mutation",code:"INVALID_REQUEST",details:g},{status:400});f=T;}else {let{success:u,data:T,error:g}=ne.safeParse(l);if(!u)return Response.json({message:"Invalid mutation",code:"INVALID_REQUEST",details:g},{status:400});f=T;}let d=await n.handleMutation({req:{...a,type:"MUTATE",resource:h,input:f.payload,context:m,resourceId:f.resourceId,procedure:p==="insert"||p==="update"?p.toUpperCase():p,queryParams:{}}});return Response.json(d)}catch(p){return e.error("Error parsing mutation from the client:",p),Response.json({message:"Internal server error",code:"INTERNAL_SERVER_ERROR"},{status:500})}return Response.json({message:"Not found",code:"NOT_FOUND"},{status:404})}catch(i){return e.error("Unexpected error:",i),Response.json({message:"Internal server error",code:"INTERNAL_SERVER_ERROR"},{status:500})}}};var Ee=Te(te());var N=zod.z.string(),yt=zod.z.object({id:N,type:zod.z.literal("SUBSCRIBE"),resource:zod.z.string()}),pt=Z.extend({id:N,type:zod.z.literal("QUERY")}),Ae=D.extend({id:N}),mt=W.extend({id:N}),ft=zod.z.union([mt,Ae]),je=zod.z.union([yt,pt,ft]),ht=zod.z.object({id:N,type:zod.z.literal("REJECT"),resource:zod.z.string(),message:zod.z.string().optional()}),Tt=zod.z.object({id:N,type:zod.z.literal("REPLY"),data:zod.z.any()});zod.z.union([ht,Tt,Ae]);zod.z.object({resource:zod.z.string(),data:zod.z.record(zod.z.string(),re)});var $e=n=>{let e={},t={},r=n.logger;return n.subscribeToMutations(i=>{let a=i;!a.resourceId||!a.payload||(r.debug("Mutation propagated:",a),Object.entries(t[a.resource]??{}).forEach(([s,o])=>{var c;(c=e[s])==null||c.send(JSON.stringify({...a,id:a.id??ee()}));}));}),(i,a)=>{var p;let s=h=>{i.send(JSON.stringify(h));},o=ee(),c={headers:a.headers,cookies:typeof a.headers.cookie=="string"?Ee.default.parse(a.headers.cookie):{}},y=dt.parse(a.url.split("?")[1]),m=(p=n.contextProvider)==null?void 0:p.call(n,{transport:"WEBSOCKET",headers:c.headers,cookies:c.cookies,queryParams:y});e[o]=i,r.info("Client connected:",o),i.on("message",async h=>{try{r.debug("Message received from the client:",h);let l=je.parse(JSON.parse(h.toString()));if(l.type==="SUBSCRIBE"){let{resource:f}=l;t[f]||(t[f]={}),t[f][o]={};}else if(l.type==="QUERY"){let{resource:f}=l,d=await n.handleQuery({req:{...c,type:"QUERY",resource:f,context:await m??{},queryParams:y}});if(!d||!d.data)throw new Error("Invalid resource");s({id:l.id,type:"REPLY",data:{resource:f,data:Object.fromEntries(Object.entries(d.data??{}).map(([u,T])=>[u,T.value]))}});}else if(l.type==="MUTATE"){let{resource:f}=l;r.debug("Received mutation from client:",l);try{let d=await n.handleMutation({req:{...c,type:"MUTATE",resource:f,input:l.payload,context:{messageId:l.id,...await m??{}},resourceId:l.resourceId,procedure:l.procedure,queryParams:y}});l.procedure&&l.procedure!=="INSERT"&&l.procedure!=="UPDATE"&&s({id:l.id,type:"REPLY",data:d});}catch(d){s({id:l.id,type:"REJECT",resource:f,message:d.message}),r.error("Error parsing mutation from the client:",d);}}}catch(l){r.error("Error handling message from the client:",l);}}),i.on("close",()=>{r.info("Connection closed",o),delete e[o];for(let h of Object.values(t))delete h[o];});}};function Ce(n){let e=`${n.protocol}://${n.hostname}${n.url}`,t=new Headers;return Object.entries(n.headers).forEach(([r,i])=>{i&&t.set(r,Array.isArray(i)?i.join(","):i);}),new Request(e,{method:n.method,headers:t,body:n.body&&n.method!=="GET"?JSON.stringify(n.body):void 0})}var nr=(n,e,t)=>{n.ws(`${(t==null?void 0:t.basePath)??""}/ws`,$e(e)),n.use(`${(t==null?void 0:t.basePath)??""}/`,(r,i)=>{Oe(e)(Ce(r)).then(s=>s.json().then(o=>i.status(s.status).send(o)));});};var S=n=>n?Array.isArray(n.value)?n.value.map(t=>S(t)):typeof n.value!="object"||n.value===null||n.value instanceof Date?n.value:Object.fromEntries(Object.entries(n.value).map(([t,r])=>Array.isArray(r)?[t,r.map(i=>S(i))]:[t,S(r)])):void 0;var le=class n{routes;constructor(e){this.routes=e.routes;}static create(e){return new n(e)}},wr=n=>le.create({...n}),xt=n=>({handler:e=>({inputValidator:n??zod.z.undefined(),handler:e})}),de=class n{resourceSchema;middlewares;customMutations;authorization;constructor(e,t,r){this.resourceSchema=e,this.middlewares=new Set,this.customMutations=t??{},this.authorization=r;}use(...e){for(let t of e)this.middlewares.add(t);return this}withMutations(e){return new n(this.resourceSchema,e({mutation:xt}),this.authorization)}handleQuery=async({req:e,batcher:t})=>await this.wrapInMiddlewares(async r=>{var a,s;let i=(s=(a=this.authorization)==null?void 0:a.read)==null?void 0:s.call(a,{ctx:r.context});if(typeof i=="boolean"&&!i)throw new Error("Not authorized");return {data:await t.rawFind({resource:r.resource,commonWhere:B(r.where,typeof i=="object"?i:void 0),uniqueWhere:r.relationalWhere,include:r.include,limit:r.limit,sort:r.sort})}})(e);handleMutation=async({req:e,db:t,schema:r})=>await this.wrapInMiddlewares(async 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:r});throw new Error(`Unknown procedure: ${i.procedure}`)}})(e);handleSet=async({req:e,db:t,operation:r,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(r==="INSERT"&&a)throw new Error("Resource already exists");if(r==="UPDATE"&&!a)throw new Error("Resource not found");return t.transaction(async({trx:s})=>{var m,p,h,l,f;let[o,c]=this.resourceSchema.mergeMutation("set",e.input,a);if(!c)throw new Error("Mutation rejected");if(r==="INSERT"){let d=await s.rawInsert(e.resource,e.resourceId,o),u=S(d);if(u.id=u.id??e.resourceId,(m=this.authorization)!=null&&m.insert){let T=this.authorization.insert({ctx:e.context,value:u});if(typeof T=="boolean"){if(!T)throw new Error("Not authorized")}else {let g=K(T,e.resource,i),x=Object.keys(g).length>0?await s.rawFindById(e.resource,e.resourceId,g):d,b=S(x);if(b.id=b.id??e.resourceId,!M(b,T))throw new Error("Not authorized")}}return {data:d,acceptedValues:c}}if((h=(p=this.authorization)==null?void 0:p.update)!=null&&h.preMutation){let d=S(a);d.id=d.id??e.resourceId;let u=this.authorization.update.preMutation({ctx:e.context,value:d});if(typeof u=="boolean"){if(!u)throw new Error("Not authorized")}else {let T=K(u,e.resource,i),g=Object.keys(T).length>0?await s.rawFindById(e.resource,e.resourceId,T):a,x=S(g);if(x.id=x.id??e.resourceId,!M(x,u))throw new Error("Not authorized")}}let y=await s.rawUpdate(e.resource,e.resourceId,o);if((f=(l=this.authorization)==null?void 0:l.update)!=null&&f.postMutation){let d=S(y);d.id=d.id??e.resourceId;let u=this.authorization.update.postMutation({ctx:e.context,value:d});if(typeof u=="boolean"){if(!u)throw new Error("Not authorized")}else {let T=K(u,e.resource,i),g=Object.keys(T).length>0?await s.rawFindById(e.resource,e.resourceId,T):y,x=S(g);if(x.id=x.id??e.resourceId,!M(x,u))throw new Error("Not authorized")}}return {data:y,acceptedValues:c}})};wrapInMiddlewares(e){return t=>Array.from(this.middlewares.values()).reduceRight((r,i)=>a=>i({req:a,next:r}),e)(t)}},ye=class n{middlewares;constructor(e=[]){this.middlewares=e;}collectionRoute(e,t){return new de(e,void 0,t).use(...this.middlewares)}use(...e){return new n([...this.middlewares,...e])}static create(){return new n}},Sr=ye.create;var U=class{async insert(e,t){let r=new Date().toISOString();return S(await this.rawInsert(e.name,t.id,{value:Object.fromEntries(Object.entries(t).map(([i,a])=>[i,{value:a,_meta:{timestamp:r}}]))}))}async update(e,t,r){let i=new Date().toISOString(),{id:a,...s}=r;return S(await this.rawUpdate(e.name,t,{value:Object.fromEntries(Object.entries(s).map(([o,c])=>[o,{value:c,_meta:{timestamp:i}}]))}))}};function H(n,e,t,r){if(!n)throw new Error("Schema not initialized");let i=n[e];if(!i)throw new Error("Resource not found");let a=r.$or,s=r.$and;return (a?t.or:t.and)(a?r.$or.map(o=>H(n,e,t,o)):s?r.$and.map(o=>H(n,e,t,o)):Object.entries(r).map(([o,c])=>{var y,m;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?((y=c==null?void 0:c.$not)==null?void 0:y.$in)!==void 0?t(`${e}.${o}`,"not in",c.$not.$in):((m=c==null?void 0:c.$not)==null?void 0:m.$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 p=i.relations[o],h=p.entity.name;return p.type==="many"?t.exists(pe(n,h,t.selectFrom(h).select("id").whereRef(p.foreignColumn,"=",`${e}.id`),c)):H(n,h,t,c)}return null}).filter(Boolean))}function J(n,e,t,r){let i=n[e];if(!i)throw new Error("Resource not found");if(!r)return t;if(r.$and){for(let a of r.$and)t=J(n,e,t,a);return t}else if(r.$or){for(let a of r.$or)t=J(n,e,t,a);return t}for(let[a,s]of Object.entries(r)){if(!i.relations[a])continue;let o=i.relations[a],c=o.entity.name,y=o.type==="one"?"id":o.foreignColumn,m=o.type==="one"?o.relationalColumn:"id";t=t.leftJoin(c,`${c}.${y}`,`${e}.${m}`),s instanceof Object&&!Array.isArray(s)&&s!==null&&(t=J(n,c,t,s));}return t}function pe(n,e,t,r){return !r||Object.keys(r).length===0?t:(t=J(n,e,t,r),t.where(i=>H(n,e,i,r)))}function Y(n,e,t,r){if(!r)return t;if(!n)throw new Error("Schema not initialized");let i=n[e];if(!i)throw new Error(`Resource not found: ${e}`);for(let a of Object.keys(r)){if(!i.relations[a])throw new Error(`Relation ${a} not found in resource ${e}`);let s=i.relations[a],o=s.entity.name,c=r[a],y=s.type==="one"?"id":s.foreignColumn,m=s.type==="one"?s.relationalColumn:"id",p=s.type==="one"?postgres.jsonObjectFrom:postgres.jsonArrayFrom,h=typeof c=="object"&&c!==null;t=t.select(l=>{let f=l.selectFrom(o).selectAll(o).whereRef(`${o}.${y}`,"=",`${e}.${m}`).select(d=>postgres.jsonObjectFrom(d.selectFrom(`${o}_meta`).selectAll(`${o}_meta`).whereRef(`${o}_meta.id`,"=",`${o}.id`)).as("_meta"));return h&&(f=Y(n,o,f,c)),p(f).as(a)});}return t}var Ve="42701",me=class n extends U{db;schema;logger;constructor(e,t,r){super(),this.isKyselyLike(e)?this.db=e:this.db=new kysely.Kysely({dialect:new kysely.PostgresDialect({pool:e})}),this.schema=t,this.logger=r,this.rawInsert=this.rawInsert.bind(this),this.rawUpdate=this.rawUpdate.bind(this);}async init(e,t){var i;this.schema=e,this.logger=t;let r=await this.db.introspection.getTables();for(let[a,s]of Object.entries(e)){let o=r.find(m=>m.name===a);o||await this.db.schema.createTable(a).ifNotExists().execute();let c=`${a}_meta`,y=r.find(m=>m.name===c);y||await this.db.schema.createTable(c).ifNotExists().execute();for(let[m,p]of Object.entries(s.fields)){let h=o==null?void 0:o.columns.find(d=>d.name===m),l=p.getStorageFieldType();h?h.dataType!==l.type&&((i=this.logger)==null||i.warn("Column type mismatch:",m,"expected to have type:",l.type,"but has type:",h.dataType)):(await this.db.schema.alterTable(a).addColumn(m,l.type,d=>{let u=d;return l.unique&&(u=u.unique()),l.nullable||(u=u.notNull()),l.references&&(u=u.references(l.references)),l.primary&&(u=u.primaryKey()),l.default!==void 0&&(u=u.defaultTo(l.default)),u}).execute().catch(d=>{var u;if(d.code!==Ve)throw (u=this.logger)==null||u.error("Error adding column",m,d),d}),l.index&&await this.db.schema.createIndex(`${a}_${m}_index`).on(a).column(m).execute().catch(()=>{})),(y==null?void 0:y.columns.find(d=>d.name===m))||await this.db.schema.alterTable(c).addColumn(m,"varchar",d=>{let u=d;return l.primary&&(u=u.primaryKey().references(`${a}.${m}`)),u}).execute().catch(d=>{var u;if(d.code!==Ve)throw (u=this.logger)==null||u.error("Error adding meta column",m,d),d});}}}async rawFindById(e,t,r){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=Y(this.schema,e,i,r);let a=await i.executeTakeFirst();if(a)return this.convertToMaterializedLiveType(a)}async findOne(e,t,r){let i=await this.rawFindById(e.name,t,r==null?void 0:r.include);if(i)return S(i)}async rawFind(e){if(!this.schema)throw new Error("Schema not initialized");let{resource:t,where:r,include:i,limit:a,sort:s}=e,o=this.db.selectFrom(t).selectAll(t).select(p=>postgres.jsonObjectFrom(p.selectFrom(`${t}_meta`).selectAll(`${t}_meta`).whereRef(`${t}_meta.id`,"=",`${t}.id`)).as("_meta"));o=pe(this.schema,t,o,r),o=Y(this.schema,t,o,i),a!==void 0&&(o=o.limit(a)),s!==void 0&&s.forEach(p=>{o=o.orderBy(p.key,p.direction);});let c=await o.execute(),y=Object.fromEntries(c.map(p=>{let{id:h}=p;return [h,p]}));return Object.keys(y).length===0?{}:Object.entries(y).reduce((p,[h,l])=>(p[h]=this.convertToMaterializedLiveType(l),p),{})}async find(e,t){let r=await this.rawFind({resource:e.name,where:t==null?void 0:t.where,include:t==null?void 0:t.include,limit:t==null?void 0:t.limit,sort:t==null?void 0:t.sort});return Object.fromEntries(Object.entries(r).map(([i,a])=>[i,S(a)]))}async rawInsert(e,t,r){var s;let i={},a={};for(let[o,c]of Object.entries(r.value)){let y=(s=c._meta)==null?void 0:s.timestamp;y&&(i[o]=c.value,a[o]=y);}return await this.db.insertInto(e).values({...i,id:t}).execute().then(()=>{this.db.insertInto(`${e}_meta`).values({...a,id:t}).execute();}),r}async rawUpdate(e,t,r){var s;let i={},a={};for(let[o,c]of Object.entries(r.value)){let y=(s=c._meta)==null?void 0:s.timestamp;y&&(i[o]=c.value,a[o]=y);}return await Promise.all([this.db.updateTable(e).set(i).where("id","=",t).execute(),this.db.insertInto(`${e}_meta`).values({...a,id:t}).onConflict(o=>o.column("id").doUpdateSet(a)).execute()]),r}async transaction(e){if(!this.schema)throw new Error("Schema not initialized");if(this.db.isTransaction){let r=Math.random().toString(36).substring(2,15),i=await this.db.savepoint(r).execute();try{return await e({trx:this,commit:()=>i.releaseSavepoint(r).execute().then(()=>{}),rollback:()=>i.rollbackToSavepoint(r).execute().then(()=>{})}).then(a=>i.isCommitted||i.isRolledBack?a:i.releaseSavepoint(r).execute().then(()=>a))}catch(a){throw await i.rollbackToSavepoint(r).execute().catch(()=>{}),a}}let t=await this.db.startTransaction().execute();try{return await e({trx:new n(t,this.schema,this.logger),commit:()=>t.commit().execute(),rollback:()=>t.rollback().execute()}).then(r=>t.isCommitted||t.isRolledBack?r:t.commit().execute().then(()=>r))}catch(r){throw await t.rollback().execute(),r}}convertToMaterializedLiveType(e){return {value:Object.entries(e).reduce((t,[r,i])=>{var a,s,o;return r==="_meta"||(r==="id"?t[r]={value:i}:Array.isArray(i)?t[r]={value:i.map(c=>this.convertToMaterializedLiveType(c)),_meta:{timestamp:(a=e==null?void 0:e._meta)==null?void 0:a[r]}}:typeof i=="object"&&i!==null&&!(i instanceof Date)?t[r]={...this.convertToMaterializedLiveType(i),_meta:{timestamp:(s=e==null?void 0:e._meta)==null?void 0:s[r]}}:t[r]={value:i,_meta:{timestamp:(o=e==null?void 0:e._meta)==null?void 0:o[r]}}),t},{})}}isKyselyLike(e){if(e instanceof kysely.Kysely)return true;if(!e||typeof e!="object")return false;let t=e,r=typeof t.selectFrom=="function",i=typeof t.startTransaction=="function",a=typeof t.savepoint=="function",s=typeof t.isTransaction=="boolean"||typeof t.isTransaction=="function";return r&&i||a&&s}};var fe=class n{router;storage;schema;middlewares=new Set;logger;contextProvider;mutationSubscriptions=new Set;constructor(e){var t;this.router=e.router,this.storage=e.storage,this.schema=e.schema,this.logger=ge({level:e.logLevel??j.INFO}),(t=e.middlewares)==null||t.forEach(r=>{this.middlewares.add(r);}),this.storage.init(this.schema,this.logger),this.contextProvider=e.contextProvider;}static create(e){return new n(e)}subscribeToMutations(e){return this.mutationSubscriptions.add(e),()=>{this.mutationSubscriptions.delete(e);}}handleQuery(e){let t=new Q(this.storage);return this.wrapInMiddlewares(async r=>{var l,f,d;let i=_e(r,this.schema,{stepId:"query",collectionName:r.resource,included:Object.keys(r.include??{})}),a={headers:r.headers,cookies:r.cookies,queryParams:r.queryParams,context:r.context},s={};for(let u=0;u<i.length;u++){let T=i[u],g=this.router.routes[T.resource];if(!g)throw new Error("Invalid resource");let x;if(T.getWhere&&T.referenceGetter){let v=T.referenceGetter(s);x=[];for(let I=0;I<v.length;I++)x.push(T.getWhere(v[I]));}else x=[void 0];let b=s[T.prevStepId??""],L=[];if(b)for(let v=0;v<b.length;v++){let I=b[v],A=Object.keys(((l=I==null?void 0:I.result)==null?void 0:l.data)??{});for(let _=0;_<A.length;_++)L.push(A[_]);}let E=[];for(let v=0;v<x.length;v++){let I=x[v],A=L[v];E.push((async()=>{let _=await g.handleQuery({req:{type:"QUERY",...T,...a,where:T.where,relationalWhere:I},batcher:t});return {includedBy:A,result:_}})());}let O=await Promise.allSettled(E),V=[];for(let v=0;v<O.length;v++){let I=O[v];I.status==="fulfilled"&&V.push(I.value);}s[T.stepId]=V;}let o=new Map,c=0;for(let u in s){let T=s[u],g=i[c];c++;for(let x=0;x<T.length;x++){let b=T[x],L=b.result.data;for(let E in L){let O=L[E],V=`${u}.${E}`,v=[];u!=="query"&&b.includedBy&&(v=[`${g.prevStepId}.${b.includedBy}`]);let I=o.get(V);if(I)for(let A=0;A<v.length;A++)I.includedBy.add(v[A]);else o.set(V,{data:O,includedBy:new Set(v),path:u.split(".").slice(-1)[0],isMany:g.isMany??false,collectionName:g.collectionName,included:g.included});}}}let y=Object.fromEntries(o),m={data:{}},p=Object.keys(y);for(let u=p.length-1;u>=0;u--){let T=p[u],g=y[T],x=g.path;if(x==="query"&&(m.data[T.replace("query.","")]=g.data),g.included.length)for(let b=0;b<g.included.length;b++){let L=g.included[b];g.data.value[L]??=((d=(f=this.schema[g.collectionName])==null?void 0:f.relations[L])==null?void 0:d.type)==="many"?{value:[]}:{value:null};}if(g.includedBy.size>0){let b=Array.from(g.includedBy);for(let L=0;L<b.length;L++){let E=b[L],O=y[E];O&&(g.isMany?(O.data.value[x]??={value:[]},O.data.value[x].value.push(g.data)):O.data.value[x]=g.data);}}}return m})(e.req)}async handleMutation(e){let t=await this.wrapInMiddlewares(async r=>{let i=this.router.routes[r.resource];if(!i)throw new Error("Invalid resource");return i.handleMutation({req:r,db:this.storage,schema:this.schema})})(e.req);if(t&&e.req.type==="MUTATE"&&t.acceptedValues&&(e.req.procedure==="INSERT"||e.req.procedure==="UPDATE")&&e.req.resourceId){let 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}wrapInMiddlewares(e){return t=>Array.from(this.middlewares.values()).reduceRight((r,i)=>a=>i({req:a,next:r}),e)(t)}},Wr=fe.create;function _e(n,e,t){let{include:r,where:i,...a}=n,{stepId:s}=t,o=[{...a,...t,where:i}];if(r&&typeof r=="object"&&Object.keys(r).length>0){let c=e[a.resource];if(!c)throw new Error(`Resource ${a.resource} not found`);o.push(...Object.entries(r).flatMap(([y,m])=>{let p=c.relations[y];if(!p)throw new Error(`Relation ${y} not found for resource ${a.resource}`);let h=p.entity.name;return _e({...a,resource:h,include:m},e,{getWhere:p.type==="one"?l=>({id:l}):l=>({[p.foreignColumn]:l}),referenceGetter:l=>l[s].flatMap(f=>f.result.data?p.type==="one"?Object.values(f.result.data).map(d=>{var u,T;return (T=(u=d.value)==null?void 0:u[p.relationalColumn])==null?void 0:T.value}):Object.keys(f.result.data):[]),stepId:`${s}.${y}`,prevStepId:s,isMany:p.type==="many",collectionName:h,included:typeof m=="object"?Object.keys(m):[]})}));}return o}
2
+ exports.Route=de;exports.RouteFactory=ye;exports.Router=le;exports.SQLStorage=me;exports.Server=fe;exports.Storage=U;exports.expressAdapter=nr;exports.routeFactory=Sr;exports.router=wr;exports.server=Wr;
package/dist/server.js CHANGED
@@ -1,2 +1,2 @@
1
- import {a,b,t,v as v$1,w,y,x}from'./chunk-NJ7LXJAY.js';import C from'node:crypto';import Be,{parse}from'qs';import {z as z$1}from'zod';import {Kysely,PostgresDialect}from'kysely';import {jsonObjectFrom,jsonArrayFrom}from'kysely/helpers/postgres';var _=a(N=>{Object.defineProperty(N,"__esModule",{value:true});N.parse=Pe;N.serialize=ze;var Ae=/^[\u0021-\u003A\u003C\u003E-\u007E]+$/,Oe=/^[\u0021-\u003A\u003C-\u007E]*$/,je=/^([.]?[a-z0-9]([a-z0-9-]{0,61}[a-z0-9])?)([.][a-z0-9]([a-z0-9-]{0,61}[a-z0-9])?)*$/i,Le=/^[\u0020-\u003A\u003D-\u007E]*$/,$e=Object.prototype.toString,Ce=(()=>{let i=function(){};return i.prototype=Object.create(null),i})();function Pe(i,t){let e=new Ce,r=i.length;if(r<2)return e;let n=(t==null?void 0:t.decode)||Ne,o=0;do{let c=i.indexOf("=",o);if(c===-1)break;let a=i.indexOf(";",o),s=a===-1?r:a;if(c>s){o=i.lastIndexOf(";",c-1)+1;continue}let h=oe(i,o,c),y=ae(i,c,h),d=i.slice(h,y);if(e[d]===void 0){let p=oe(i,c+1,s),u=ae(i,s,p),f=n(i.slice(p,u));e[d]=f;}o=s+1;}while(o<r);return e}function oe(i,t,e){do{let r=i.charCodeAt(t);if(r!==32&&r!==9)return t}while(++t<e);return e}function ae(i,t,e){for(;t>e;){let r=i.charCodeAt(--t);if(r!==32&&r!==9)return t+1}return e}function ze(i,t,e){let r=(e==null?void 0:e.encode)||encodeURIComponent;if(!Ae.test(i))throw new TypeError(`argument name is invalid: ${i}`);let n=r(t);if(!Oe.test(n))throw new TypeError(`argument val is invalid: ${t}`);let o=i+"="+n;if(!e)return o;if(e.maxAge!==void 0){if(!Number.isInteger(e.maxAge))throw new TypeError(`option maxAge is invalid: ${e.maxAge}`);o+="; Max-Age="+e.maxAge;}if(e.domain){if(!je.test(e.domain))throw new TypeError(`option domain is invalid: ${e.domain}`);o+="; Domain="+e.domain;}if(e.path){if(!Le.test(e.path))throw new TypeError(`option path is invalid: ${e.path}`);o+="; Path="+e.path;}if(e.expires){if(!Ue(e.expires)||!Number.isFinite(e.expires.valueOf()))throw new TypeError(`option expires is invalid: ${e.expires}`);o+="; Expires="+e.expires.toUTCString();}if(e.httpOnly&&(o+="; HttpOnly"),e.secure&&(o+="; Secure"),e.partitioned&&(o+="; Partitioned"),e.priority)switch(typeof e.priority=="string"?e.priority.toLowerCase():void 0){case "low":o+="; Priority=Low";break;case "medium":o+="; Priority=Medium";break;case "high":o+="; Priority=High";break;default:throw new TypeError(`option priority is invalid: ${e.priority}`)}if(e.sameSite)switch(typeof e.sameSite=="string"?e.sameSite.toLowerCase():e.sameSite){case true:case "strict":o+="; SameSite=Strict";break;case "lax":o+="; SameSite=Lax";break;case "none":o+="; SameSite=None";break;default:throw new TypeError(`option sameSite is invalid: ${e.sameSite}`)}return o}function Ne(i){if(i.indexOf("%")===-1)return i;try{return decodeURIComponent(i)}catch{return i}}function Ue(i){return $e.call(i)==="[object Date]"}});var re="0123456789ABCDEFGHJKMNPQRSTVWXYZ",E=32;var we=16,ne=10,te=0xffffffffffff;var I;(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";})(I||(I={}));var M=class extends Error{constructor(t,e){super(`${e} (${t})`),this.name="ULIDError",this.code=t;}};function xe(i){let t=Math.floor(i()*E);return t===E&&(t=E-1),re.charAt(t)}function Se(i){var r;let t=Ie(),e=t&&(t.crypto||t.msCrypto)||(typeof C<"u"?C:null);if(typeof(e==null?void 0:e.getRandomValues)=="function")return ()=>{let n=new Uint8Array(1);return e.getRandomValues(n),n[0]/255};if(typeof(e==null?void 0:e.randomBytes)=="function")return ()=>e.randomBytes(1).readUInt8()/255;if((r=C)!=null&&r.randomBytes)return ()=>C.randomBytes(1).readUInt8()/255;throw new M(I.PRNGDetectFailure,"Failed to find a reliable PRNG")}function Ie(){return Ee()?self:typeof window<"u"?window:typeof global<"u"?global:typeof globalThis<"u"?globalThis:null}function Me(i,t){let e="";for(;i>0;i--)e=xe(t)+e;return e}function ve(i,t=ne){if(isNaN(i))throw new M(I.EncodeTimeValueMalformed,`Time must be a number: ${i}`);if(i>te)throw new M(I.EncodeTimeSizeExceeded,`Cannot encode a time larger than ${te}: ${i}`);if(i<0)throw new M(I.EncodeTimeNegative,`Time must be positive: ${i}`);if(Number.isInteger(i)===false)throw new M(I.EncodeTimeValueMalformed,`Time must be an integer: ${i}`);let e,r="";for(let n=t;n>0;n--)e=i%E,r=re.charAt(e)+r,i=(i-e)/E;return r}function Ee(){return typeof WorkerGlobalScope<"u"&&self instanceof WorkerGlobalScope}function ie(i,t){let e=Se(),r=Date.now();return ve(r,ne)+Me(we,e)}var D=()=>ie().toLowerCase();var P=(...i)=>{let t=i.filter(e=>!!e);return t.length===0?{}:t.length===1?t[0]:{$and:t}};var z=class{storage;queue=new Map;scheduled=false;constructor(t){this.storage=t;}async rawFind({resource:t,commonWhere:e,uniqueWhere:r,...n}){return new Promise((o,c)=>{let a=this.getBatchKey({resource:t,commonWhere:e,...n}),s={resource:t,commonWhere:e,uniqueWhere:r,...n,resolve:o,reject:c};this.queue.has(a)||this.queue.set(a,[]);let h=this.queue.get(a);h&&h.push(s),this.scheduled||(this.scheduled=true,setImmediate(()=>{this.processBatch();}));})}getBatchKey(t){let{resource:e,commonWhere:r,...n}=t;return `${e}:${JSON.stringify(r??{})}:${JSON.stringify(n??{})}`}async processBatch(){this.scheduled=false;let t=Array.from(this.queue.entries());this.queue.clear();for(let[,e]of t)try{await this.executeBatchedRequests(e);}catch(r){e.forEach(n=>{n.reject(r);});}}async executeBatchedRequests(t){var p,u;if(t.length===0)return;let e=t[0],{resource:r,commonWhere:n,include:o,sort:c}=e,a=t.length===1?e.limit:void 0,s=t.map(f=>f.uniqueWhere).filter(f=>f!==void 0),h=n,y=(p=Object.entries(s[0]??{})[0])==null?void 0:p[0];if(s.length>0){let f=s.map(l=>l[y]).filter(l=>l!=null);f.length>0&&(h=P(n,{[y]:{$in:f}}));}let d=await this.storage.rawFind({resource:r,where:h,include:o,sort:c,limit:a});for(let f of t){let l={};if(f.uniqueWhere){let[m,T]=Object.entries(f.uniqueWhere)[0];for(let[b,w]of Object.entries(d))((u=w.value[m])==null?void 0:u.value)===T&&(l[b]=w);}else Object.assign(l,d);f.resolve(l);}}};var ue=b(_(),1);var U=z$1.object({resource:z$1.string(),where:z$1.record(z$1.string(),z$1.any()).optional(),include:z$1.record(z$1.string(),z$1.any()).optional(),lastSyncedAt:z$1.string().optional(),limit:z$1.coerce.number().optional(),sort:z$1.array(z$1.object({key:z$1.string(),direction:z$1.enum(["asc","desc"])})).optional()}),F=z$1.record(z$1.string(),z$1.object({value:z$1.string().or(z$1.number()).or(z$1.boolean()).or(z$1.date()).nullable(),_meta:z$1.object({timestamp:z$1.string().optional().nullable()}).optional()})),We=F.superRefine((i,t)=>{i.id&&t.addIssue({code:z$1.ZodIssueCode.custom,message:"Payload cannot have an id"});}),se=z$1.object({id:z$1.string().optional(),type:z$1.literal("MUTATE"),resource:z$1.string(),resourceId:z$1.string().optional()}),A=se.extend({procedure:z$1.string(),payload:z$1.any().optional()}),O=se.extend({procedure:z$1.enum(["INSERT","UPDATE"]),payload:We});z$1.union([O,A]);var ce=U.omit({resource:true}),V=A.omit({id:true,type:true,resource:true,procedure:true}),q=O.omit({id:true,type:true,resource:true,procedure:true});z$1.union([q,V]);var de=i=>{let t=i.logger;return async e=>{var r;try{let n=typeof e.headers.getSetCookie=="function"?Object.fromEntries(e.headers):e.headers,o={headers:n,cookies:n.cookie?ue.default.parse(n.cookie):{}},c=new URL(e.url),a=c.pathname.split("/"),s=c.searchParams,h=Be.parse(s.toString()),y=await((r=i.contextProvider)==null?void 0:r.call(i,{transport:"HTTP",headers:o.headers,cookies:o.cookies,queryParams:h}))??{};if(e.method==="GET"){let d=a[a.length-1],{success:p,data:u,error:f}=ce.safeParse(h);if(!p)return Response.json({message:"Invalid query",code:"INVALID_QUERY",details:f},{status:400});let l=await i.handleQuery({req:{...o,...u,type:"QUERY",resource:d,context:y,queryParams:h}});return !l||!l.data?Response.json({message:"Invalid resource",code:"INVALID_RESOURCE"},{status:400}):Response.json(l.data)}if(e.method==="POST")try{let d=a[a.length-1],p=a[a.length-2],u=e.body?await e.json():{},f;if(d==="insert"||d==="update"){let{success:m,data:T,error:b}=q.safeParse(u);if(!m)return Response.json({message:"Invalid mutation",code:"INVALID_REQUEST",details:b},{status:400});f=T;}else {let{success:m,data:T,error:b}=V.safeParse(u);if(!m)return Response.json({message:"Invalid mutation",code:"INVALID_REQUEST",details:b},{status:400});f=T;}let l=await i.handleMutation({req:{...o,type:"MUTATE",resource:p,input:f.payload,context:y,resourceId:f.resourceId,procedure:d==="insert"||d==="update"?d.toUpperCase():d,queryParams:{}}});return Response.json(l)}catch(d){return t.error("Error parsing mutation from the client:",d),Response.json({message:"Internal server error",code:"INTERNAL_SERVER_ERROR"},{status:500})}return Response.json({message:"Not found",code:"NOT_FOUND"},{status:404})}catch(n){return t.error("Unexpected error:",n),Response.json({message:"Internal server error",code:"INTERNAL_SERVER_ERROR"},{status:500})}}};var me=b(_(),1);var v=z$1.string(),De=z$1.object({id:v,type:z$1.literal("SUBSCRIBE"),resource:z$1.string()}),_e=U.extend({id:v,type:z$1.literal("QUERY")}),le=O.extend({id:v}),Fe=A.extend({id:v}),Ve=z$1.union([Fe,le]),pe=z$1.union([De,_e,Ve]),qe=z$1.object({id:v,type:z$1.literal("REJECT"),resource:z$1.string(),message:z$1.string().optional()}),Qe=z$1.object({id:v,type:z$1.literal("REPLY"),data:z$1.any()});z$1.union([qe,Qe,le]);z$1.object({resource:z$1.string(),data:z$1.record(z$1.string(),F)});var ye=i=>{let t={},e={},r=i.logger;return i.subscribeToMutations(n=>{let o=n;!o.resourceId||!o.payload||(r.debug("Mutation propagated:",o),Object.entries(e[o.resource]??{}).forEach(([c,a])=>{var s;(s=t[c])==null||s.send(JSON.stringify({...o,id:o.id??D()}));}));}),(n,o)=>{var d;let c=p=>{n.send(JSON.stringify(p));},a=D(),s={headers:o.headers,cookies:typeof o.headers.cookie=="string"?me.default.parse(o.headers.cookie):{}},h=parse(o.url.split("?")[1]),y=(d=i.contextProvider)==null?void 0:d.call(i,{transport:"WEBSOCKET",headers:s.headers,cookies:s.cookies,queryParams:h});t[a]=n,r.info("Client connected:",a),n.on("message",async p=>{try{r.debug("Message received from the client:",p);let u=pe.parse(JSON.parse(p.toString()));if(u.type==="SUBSCRIBE"){let{resource:f}=u;e[f]||(e[f]={}),e[f][a]={};}else if(u.type==="QUERY"){let{resource:f}=u,l=await i.handleQuery({req:{...s,type:"QUERY",resource:f,context:await y??{},queryParams:h}});if(!l||!l.data)throw new Error("Invalid resource");c({id:u.id,type:"REPLY",data:{resource:f,data:Object.fromEntries(Object.entries(l.data??{}).map(([m,T])=>[m,T.value]))}});}else if(u.type==="MUTATE"){let{resource:f}=u;r.debug("Received mutation from client:",u);try{let l=await i.handleMutation({req:{...s,type:"MUTATE",resource:f,input:u.payload,context:{messageId:u.id,...await y??{}},resourceId:u.resourceId,procedure:u.procedure,queryParams:h}});u.procedure&&u.procedure!=="INSERT"&&u.procedure!=="UPDATE"&&c({id:u.id,type:"REPLY",data:l});}catch(l){c({id:u.id,type:"REJECT",resource:f,message:l.message}),r.error("Error parsing mutation from the client:",l);}}}catch(u){r.error("Error handling message from the client:",u);}}),n.on("close",()=>{r.info("Connection closed",a),delete t[a];for(let p of Object.values(e))delete p[a];});}};function fe(i){let t=`${i.protocol}://${i.hostname}${i.url}`,e=new Headers;return Object.entries(i.headers).forEach(([r,n])=>{n&&e.set(r,Array.isArray(n)?n.join(","):n);}),new Request(t,{method:i.method,headers:e,body:i.body&&i.method!=="GET"?JSON.stringify(i.body):void 0})}var Ot=(i,t,e)=>{i.ws(`${(e==null?void 0:e.basePath)??""}/ws`,ye(t)),i.use(`${(e==null?void 0:e.basePath)??""}/`,(r,n)=>{de(t)(fe(r)).then(c=>c.json().then(a=>n.status(c.status).send(a)));});};var Q=class i{routes;constructor(t){this.routes=t.routes;}static create(t){return new i(t)}},zt=i=>Q.create({...i}),He=i=>({handler:t=>({inputValidator:i??z$1.undefined(),handler:t})}),Z=class i{resourceSchema;middlewares;customMutations;authorization;constructor(t,e,r){this.resourceSchema=t,this.middlewares=new Set,this.customMutations=e??{},this.authorization=r;}use(...t){for(let e of t)this.middlewares.add(e);return this}withMutations(t){return new i(this.resourceSchema,t({mutation:He}),this.authorization)}handleQuery=async({req:t,batcher:e})=>await this.wrapInMiddlewares(async r=>{var o,c;let n=(c=(o=this.authorization)==null?void 0:o.read)==null?void 0:c.call(o,{ctx:r.context});if(typeof n=="boolean"&&!n)throw new Error("Not authorized");return {data:await e.rawFind({resource:r.resource,commonWhere:P(r.where,typeof n=="object"?n:void 0),uniqueWhere:r.relationalWhere,include:r.include,limit:r.limit,sort:r.sort})}})(t);handleMutation=async({req:t,db:e,schema:r})=>await this.wrapInMiddlewares(async n=>{if(!n.procedure)throw new Error("Procedure is required for mutations");let o=this.customMutations[n.procedure];if(o){let c=o.inputValidator.parse(n.input);return n.input=c,o.handler({req:n,db:e})}else {if(n.procedure==="INSERT"||n.procedure==="UPDATE")return this.handleSet({req:n,db:e,operation:n.procedure,schema:r});throw new Error(`Unknown procedure: ${n.procedure}`)}})(t);handleSet=async({req:t$1,db:e,operation:r,schema:n})=>{if(!t$1.input)throw new Error("Payload is required");if(!t$1.resourceId)throw new Error("ResourceId is required");let o=await e.rawFindById(t$1.resource,t$1.resourceId);if(r==="INSERT"&&o)throw new Error("Resource already exists");if(r==="UPDATE"&&!o)throw new Error("Resource not found");return e.transaction(async({trx:c})=>{var y,d,p,u,f;let[a,s]=this.resourceSchema.mergeMutation("set",t$1.input,o);if(!s)throw new Error("Mutation rejected");if(r==="INSERT"){let l=await c.rawInsert(t$1.resource,t$1.resourceId,a),m=t(l);if(m.id=m.id??t$1.resourceId,(y=this.authorization)!=null&&y.insert){let T=this.authorization.insert({ctx:t$1.context,value:m});if(typeof T=="boolean"){if(!T)throw new Error("Not authorized")}else {let b=v$1(T,t$1.resource,n),w$1=Object.keys(b).length>0?await c.rawFindById(t$1.resource,t$1.resourceId,b):l,S=t(w$1);if(S.id=S.id??t$1.resourceId,!w(S,T))throw new Error("Not authorized")}}return {data:l,acceptedValues:s}}if((p=(d=this.authorization)==null?void 0:d.update)!=null&&p.preMutation){let l=t(o);l.id=l.id??t$1.resourceId;let m=this.authorization.update.preMutation({ctx:t$1.context,value:l});if(typeof m=="boolean"){if(!m)throw new Error("Not authorized")}else {let T=v$1(m,t$1.resource,n),b=Object.keys(T).length>0?await c.rawFindById(t$1.resource,t$1.resourceId,T):o,w$1=t(b);if(w$1.id=w$1.id??t$1.resourceId,!w(w$1,m))throw new Error("Not authorized")}}let h=await c.rawUpdate(t$1.resource,t$1.resourceId,a);if((f=(u=this.authorization)==null?void 0:u.update)!=null&&f.postMutation){let l=t(h);l.id=l.id??t$1.resourceId;let m=this.authorization.update.postMutation({ctx:t$1.context,value:l});if(typeof m=="boolean"){if(!m)throw new Error("Not authorized")}else {let T=v$1(m,t$1.resource,n),b=Object.keys(T).length>0?await c.rawFindById(t$1.resource,t$1.resourceId,T):h,w$1=t(b);if(w$1.id=w$1.id??t$1.resourceId,!w(w$1,m))throw new Error("Not authorized")}}return {data:h,acceptedValues:s}})};wrapInMiddlewares(t){return e=>Array.from(this.middlewares.values()).reduceRight((r,n)=>o=>n({req:o,next:r}),t)(e)}},G=class i{middlewares;constructor(t=[]){this.middlewares=t;}collectionRoute(t,e){return new Z(t,void 0,e).use(...this.middlewares)}use(...t){return new i([...this.middlewares,...t])}static create(){return new i}},Nt=G.create;var j=class{async insert(t$1,e){let r=new Date().toISOString();return t(await this.rawInsert(t$1.name,e.id,{value:Object.fromEntries(Object.entries(e).map(([n,o])=>[n,{value:o,_meta:{timestamp:r}}]))}))}async update(t$1,e,r){let n=new Date().toISOString(),{id:o,...c}=r;return t(await this.rawUpdate(t$1.name,e,{value:Object.fromEntries(Object.entries(c).map(([a,s])=>[a,{value:s,_meta:{timestamp:n}}]))}))}};function W(i,t,e,r){if(!i)throw new Error("Schema not initialized");let n=i[t];if(!n)throw new Error("Resource not found");let o=r.$or,c=r.$and;return (o?e.or:e.and)(o?r.$or.map(a=>W(i,t,e,a)):c?r.$and.map(a=>W(i,t,e,a)):Object.entries(r).map(([a,s])=>{var h,y;if(n.fields[a])return (s==null?void 0:s.$eq)!==void 0?e(`${t}.${a}`,s.$eq===null?"is":"=",s.$eq):(s==null?void 0:s.$in)!==void 0?e(`${t}.${a}`,"in",s.$in):(s==null?void 0:s.$not)!==void 0?((h=s==null?void 0:s.$not)==null?void 0:h.$in)!==void 0?e(`${t}.${a}`,"not in",s.$not.$in):((y=s==null?void 0:s.$not)==null?void 0:y.$eq)!==void 0?e(`${t}.${a}`,s.$not.$eq===null?"is not":"!=",s.$not.$eq):e(`${t}.${a}`,s.$not===null?"is not":"!=",s.$not):(s==null?void 0:s.$gt)!==void 0?e(`${t}.${a}`,">",s.$gt):(s==null?void 0:s.$gte)!==void 0?e(`${t}.${a}`,">=",s.$gte):(s==null?void 0:s.$lt)!==void 0?e(`${t}.${a}`,"<",s.$lt):(s==null?void 0:s.$lte)!==void 0?e(`${t}.${a}`,"<=",s.$lte):e(`${t}.${a}`,s===null?"is":"=",s);if(n.relations[a]){let d=n.relations[a],p=d.entity.name;return d.type==="many"?e.exists(H(i,p,e.selectFrom(p).select("id").whereRef(d.foreignColumn,"=",`${t}.id`),s)):W(i,p,e,s)}return null}).filter(Boolean))}function k(i,t,e,r){let n=i[t];if(!n)throw new Error("Resource not found");if(!r)return e;if(r.$and){for(let o of r.$and)e=k(i,t,e,o);return e}else if(r.$or){for(let o of r.$or)e=k(i,t,e,o);return e}for(let[o,c]of Object.entries(r)){if(!n.relations[o])continue;let a=n.relations[o],s=a.entity.name,h=a.type==="one"?"id":a.foreignColumn,y=a.type==="one"?a.relationalColumn:"id";e=e.leftJoin(s,`${s}.${h}`,`${t}.${y}`),c instanceof Object&&!Array.isArray(c)&&c!==null&&(e=k(i,s,e,c));}return e}function H(i,t,e,r){return !r||Object.keys(r).length===0?e:(e=k(i,t,e,r),e.where(n=>W(i,t,n,r)))}function B(i,t,e,r){if(!r)return e;if(!i)throw new Error("Schema not initialized");let n=i[t];if(!n)throw new Error(`Resource not found: ${t}`);for(let o of Object.keys(r)){if(!n.relations[o])throw new Error(`Relation ${o} not found in resource ${t}`);let c=n.relations[o],a=c.entity.name,s=r[o],h=c.type==="one"?"id":c.foreignColumn,y=c.type==="one"?c.relationalColumn:"id",d=c.type==="one"?jsonObjectFrom:jsonArrayFrom,p=typeof s=="object"&&s!==null;e=e.select(u=>{let f=u.selectFrom(a).selectAll(a).whereRef(`${a}.${h}`,"=",`${t}.${y}`).select(l=>jsonObjectFrom(l.selectFrom(`${a}_meta`).selectAll(`${a}_meta`).whereRef(`${a}_meta.id`,"=",`${a}.id`)).as("_meta"));return p&&(f=B(i,a,f,s)),d(f).as(o)});}return e}var K=class i extends j{db;schema;logger;constructor(t,e,r){super(),this.isKyselyLike(t)?this.db=t:this.db=new Kysely({dialect:new PostgresDialect({pool:t})}),this.schema=e,this.logger=r,this.rawInsert=this.rawInsert.bind(this),this.rawUpdate=this.rawUpdate.bind(this);}async init(t,e){var n;this.schema=t,this.logger=e;let r=await this.db.introspection.getTables();for(let[o,c]of Object.entries(t)){let a=r.find(y=>y.name===o);a||await this.db.schema.createTable(o).ifNotExists().execute();let s=`${o}_meta`,h=r.find(y=>y.name===s);h||await this.db.schema.createTable(s).ifNotExists().execute();for(let[y,d]of Object.entries(c.fields)){let p=a==null?void 0:a.columns.find(l=>l.name===y),u=d.getStorageFieldType();p?p.dataType!==u.type&&((n=this.logger)==null||n.warn("Column type mismatch:",y,"expected to have type:",u.type,"but has type:",p.dataType)):(await this.db.schema.alterTable(o).addColumn(y,u.type,l=>{let m=l;return u.unique&&(m=m.unique()),u.nullable||(m=m.notNull()),u.references&&(m=m.references(u.references)),u.primary&&(m=m.primaryKey()),u.default!==void 0&&(m=m.defaultTo(u.default)),m}).execute().catch(l=>{var m;throw (m=this.logger)==null||m.error("Error adding column",y,l),l}),u.index&&await this.db.schema.createIndex(`${o}_${y}_index`).on(o).column(y).execute().catch(()=>{})),(h==null?void 0:h.columns.find(l=>l.name===y))||await this.db.schema.alterTable(s).addColumn(y,"varchar",l=>{let m=l;return u.primary&&(m=m.primaryKey().references(`${o}.${y}`)),m}).execute();}}}async rawFindById(t,e,r){if(!this.schema)throw new Error("Schema not initialized");let n=await this.db.selectFrom(t).where("id","=",e).selectAll(t).select(c=>jsonObjectFrom(c.selectFrom(`${t}_meta`).selectAll(`${t}_meta`).whereRef(`${t}_meta.id`,"=",`${t}.id`)).as("_meta"));n=B(this.schema,t,n,r);let o=await n.executeTakeFirst();if(o)return this.convertToMaterializedLiveType(o)}async findOne(t$1,e,r){let n=await this.rawFindById(t$1.name,e,r==null?void 0:r.include);if(n)return t(n)}async rawFind(t){if(!this.schema)throw new Error("Schema not initialized");let{resource:e,where:r,include:n,limit:o,sort:c}=t,a=this.db.selectFrom(e).selectAll(e).select(d=>jsonObjectFrom(d.selectFrom(`${e}_meta`).selectAll(`${e}_meta`).whereRef(`${e}_meta.id`,"=",`${e}.id`)).as("_meta"));a=H(this.schema,e,a,r),a=B(this.schema,e,a,n),o!==void 0&&(a=a.limit(o)),c!==void 0&&c.forEach(d=>{a=a.orderBy(d.key,d.direction);});let s=await a.execute(),h=Object.fromEntries(s.map(d=>{let{id:p}=d;return [p,d]}));return Object.keys(h).length===0?{}:Object.entries(h).reduce((d,[p,u])=>(d[p]=this.convertToMaterializedLiveType(u),d),{})}async find(t$1,e){let r=await this.rawFind({resource:t$1.name,where:e==null?void 0:e.where,include:e==null?void 0:e.include,limit:e==null?void 0:e.limit,sort:e==null?void 0:e.sort});return Object.fromEntries(Object.entries(r).map(([n,o])=>[n,t(o)]))}async rawInsert(t,e,r){var c;let n={},o={};for(let[a,s]of Object.entries(r.value)){let h=(c=s._meta)==null?void 0:c.timestamp;h&&(n[a]=s.value,o[a]=h);}return await this.db.insertInto(t).values({...n,id:e}).execute().then(()=>{this.db.insertInto(`${t}_meta`).values({...o,id:e}).execute();}),r}async rawUpdate(t,e,r){var c;let n={},o={};for(let[a,s]of Object.entries(r.value)){let h=(c=s._meta)==null?void 0:c.timestamp;h&&(n[a]=s.value,o[a]=h);}return await Promise.all([this.db.updateTable(t).set(n).where("id","=",e).execute(),this.db.insertInto(`${t}_meta`).values({...o,id:e}).onConflict(a=>a.column("id").doUpdateSet(o)).execute()]),r}async transaction(t){if(!this.schema)throw new Error("Schema not initialized");if(this.db.isTransaction){let r=Math.random().toString(36).substring(2,15),n=await this.db.savepoint(r).execute();try{return await t({trx:this,commit:()=>n.releaseSavepoint(r).execute().then(()=>{}),rollback:()=>n.rollbackToSavepoint(r).execute().then(()=>{})}).then(o=>n.isCommitted||n.isRolledBack?o:n.releaseSavepoint(r).execute().then(()=>o))}catch(o){throw await n.rollbackToSavepoint(r).execute().catch(()=>{}),o}}let e=await this.db.startTransaction().execute();try{return await t({trx:new i(e,this.schema,this.logger),commit:()=>e.commit().execute(),rollback:()=>e.rollback().execute()}).then(r=>e.isCommitted||e.isRolledBack?r:e.commit().execute().then(()=>r))}catch(r){throw await e.rollback().execute(),r}}convertToMaterializedLiveType(t){return {value:Object.entries(t).reduce((e,[r,n])=>{var o,c,a;return r==="_meta"||(r==="id"?e[r]={value:n}:Array.isArray(n)?e[r]={value:n.map(s=>this.convertToMaterializedLiveType(s)),_meta:{timestamp:(o=t==null?void 0:t._meta)==null?void 0:o[r]}}:typeof n=="object"&&n!==null&&!(n instanceof Date)?e[r]={...this.convertToMaterializedLiveType(n),_meta:{timestamp:(c=t==null?void 0:t._meta)==null?void 0:c[r]}}:e[r]={value:n,_meta:{timestamp:(a=t==null?void 0:t._meta)==null?void 0:a[r]}}),e},{})}}isKyselyLike(t){if(t instanceof Kysely)return true;if(!t||typeof t!="object")return false;let e=t,r=typeof e.selectFrom=="function",n=typeof e.startTransaction=="function",o=typeof e.savepoint=="function",c=typeof e.isTransaction=="boolean"||typeof e.isTransaction=="function";return r&&n||o&&c}};var J=class i{router;storage;schema;middlewares=new Set;logger;contextProvider;mutationSubscriptions=new Set;constructor(t){var e;this.router=t.router,this.storage=t.storage,this.schema=t.schema,this.logger=y({level:t.logLevel??x.INFO}),(e=t.middlewares)==null||e.forEach(r=>{this.middlewares.add(r);}),this.storage.init(this.schema,this.logger),this.contextProvider=t.contextProvider;}static create(t){return new i(t)}subscribeToMutations(t){return this.mutationSubscriptions.add(t),()=>{this.mutationSubscriptions.delete(t);}}handleQuery(t){let e=new z(this.storage);return this.wrapInMiddlewares(async r=>{var h;let n=Re(r,this.schema,{stepId:"query",collectionName:r.resource,included:Object.keys(r.include??{})}),o={headers:r.headers,cookies:r.cookies,queryParams:r.queryParams,context:r.context},c={};for(let y=0;y<n.length;y++){let d=n[y],p=this.router.routes[d.resource];if(!p)throw new Error("Invalid resource");let u=d.getWhere&&d.referenceGetter?d.referenceGetter(c).map(d.getWhere):[void 0],f=(h=c[d.prevStepId??""])==null?void 0:h.flatMap(T=>{var b;return Object.keys(((b=T==null?void 0:T.result)==null?void 0:b.data)??{})}),m=(await Promise.allSettled(u.map(async(T,b)=>{let w=f==null?void 0:f[b],S=await p.handleQuery({req:{type:"QUERY",...d,...o,where:d.where,relationalWhere:T},batcher:e});return {includedBy:w,result:S}}))).flatMap(T=>T.status==="fulfilled"?[T.value]:[]);c[d.stepId]=m;}let a=Object.fromEntries(Object.entries(c).flatMap(([y,d],p)=>d.flatMap(u=>Object.entries(u.result.data).map(([f,l])=>[`${y}.${f}`,{data:l,includedBy:y!=="query"&&u.includedBy?`${y.split(".").slice(0,-1).join(".")}.${u.includedBy}`:void 0,path:y.split(".").slice(-1)[0],isMany:n[p].isMany,collectionName:n[p].collectionName,included:n[p].included}]))));return Object.keys(a).reduceRight((y,d)=>{var f,l;let p=a[d],u=p.path;if(u==="query"&&(y.data[d.replace("query.","")]=p.data),p.included.length)for(let m of p.included)p.data.value[m]??=((l=(f=this.schema[p.collectionName])==null?void 0:f.relations[m])==null?void 0:l.type)==="many"?{value:[]}:{value:null};if(p.includedBy){let m=a[p.includedBy];if(!m)return y;p.isMany?(m.data.value[u]??={value:[]},m.data.value[u].value.push(p.data)):m.data.value[u]=p.data;}return y},{data:{}})})(t.req)}async handleMutation(t){let e=await this.wrapInMiddlewares(async r=>{let n=this.router.routes[r.resource];if(!n)throw new Error("Invalid resource");return n.handleMutation({req:r,db:this.storage,schema:this.schema})})(t.req);if(e&&t.req.type==="MUTATE"&&e.acceptedValues&&(t.req.procedure==="INSERT"||t.req.procedure==="UPDATE")&&t.req.resourceId){let n=e.acceptedValues??{},o=t.req,c=o.resourceId;Object.keys(n).length&&c&&this.mutationSubscriptions.forEach(a=>{a({id:t.req.context.messageId,type:"MUTATE",resource:o.resource,payload:n,resourceId:c,procedure:o.procedure});});}return e}use(t){return this.middlewares.add(t),this}context(t){return this.contextProvider=t,this}wrapInMiddlewares(t){return e=>Array.from(this.middlewares.values()).reduceRight((r,n)=>o=>n({req:o,next:r}),t)(e)}},Xt=J.create;function Re(i,t,e){let{include:r,where:n,...o}=i,{stepId:c}=e,a=[{...o,...e,where:n}];if(r&&typeof r=="object"&&Object.keys(r).length>0){let s=t[o.resource];if(!s)throw new Error(`Resource ${o.resource} not found`);a.push(...Object.entries(r).flatMap(([h,y])=>{let d=s.relations[h];if(!d)throw new Error(`Relation ${h} not found for resource ${o.resource}`);let p=d.entity.name;return Re({...o,resource:p,include:y},t,{getWhere:d.type==="one"?u=>({id:u}):u=>({[d.foreignColumn]:u}),referenceGetter:u=>u[c].flatMap(f=>f.result.data?d.type==="one"?Object.values(f.result.data).map(l=>{var m,T;return (T=(m=l.value)==null?void 0:m[d.relationalColumn])==null?void 0:T.value}):Object.keys(f.result.data):[]),stepId:`${c}.${h}`,prevStepId:c,isMany:d.type==="many",collectionName:p,included:typeof y=="object"?Object.keys(y):[]})}));}return a}
2
- export{Z as Route,G as RouteFactory,Q as Router,K as SQLStorage,J as Server,j as Storage,Ot as expressAdapter,Nt as routeFactory,zt as router,Xt as server};
1
+ import {a,b,t,v,w,y,x}from'./chunk-NJ7LXJAY.js';import D from'node:crypto';import Ke,{parse}from'qs';import {z as z$1}from'zod';import {Kysely,PostgresDialect}from'kysely';import {jsonObjectFrom,jsonArrayFrom}from'kysely/helpers/postgres';var H=a(V=>{Object.defineProperty(V,"__esModule",{value:true});V.parse=Fe;V.serialize=Ve;var Ue=/^[\u0021-\u003A\u003C\u003E-\u007E]+$/,We=/^[\u0021-\u003A\u003C-\u007E]*$/,ke=/^([.]?[a-z0-9]([a-z0-9-]{0,61}[a-z0-9])?)([.][a-z0-9]([a-z0-9-]{0,61}[a-z0-9])?)*$/i,Be=/^[\u0020-\u003A\u003D-\u007E]*$/,De=Object.prototype.toString,_e=(()=>{let i=function(){};return i.prototype=Object.create(null),i})();function Fe(i,t){let e=new _e,r=i.length;if(r<2)return e;let n=(t==null?void 0:t.decode)||qe,o=0;do{let c=i.indexOf("=",o);if(c===-1)break;let a=i.indexOf(";",o),s=a===-1?r:a;if(c>s){o=i.lastIndexOf(";",c-1)+1;continue}let m=me(i,o,c),f=ye(i,c,m),p=i.slice(m,f);if(e[p]===void 0){let h=me(i,c+1,s),d=ye(i,s,h),y=n(i.slice(h,d));e[p]=y;}o=s+1;}while(o<r);return e}function me(i,t,e){do{let r=i.charCodeAt(t);if(r!==32&&r!==9)return t}while(++t<e);return e}function ye(i,t,e){for(;t>e;){let r=i.charCodeAt(--t);if(r!==32&&r!==9)return t+1}return e}function Ve(i,t,e){let r=(e==null?void 0:e.encode)||encodeURIComponent;if(!Ue.test(i))throw new TypeError(`argument name is invalid: ${i}`);let n=r(t);if(!We.test(n))throw new TypeError(`argument val is invalid: ${t}`);let o=i+"="+n;if(!e)return o;if(e.maxAge!==void 0){if(!Number.isInteger(e.maxAge))throw new TypeError(`option maxAge is invalid: ${e.maxAge}`);o+="; Max-Age="+e.maxAge;}if(e.domain){if(!ke.test(e.domain))throw new TypeError(`option domain is invalid: ${e.domain}`);o+="; Domain="+e.domain;}if(e.path){if(!Be.test(e.path))throw new TypeError(`option path is invalid: ${e.path}`);o+="; Path="+e.path;}if(e.expires){if(!Qe(e.expires)||!Number.isFinite(e.expires.valueOf()))throw new TypeError(`option expires is invalid: ${e.expires}`);o+="; Expires="+e.expires.toUTCString();}if(e.httpOnly&&(o+="; HttpOnly"),e.secure&&(o+="; Secure"),e.partitioned&&(o+="; Partitioned"),e.priority)switch(typeof e.priority=="string"?e.priority.toLowerCase():void 0){case "low":o+="; Priority=Low";break;case "medium":o+="; Priority=Medium";break;case "high":o+="; Priority=High";break;default:throw new TypeError(`option priority is invalid: ${e.priority}`)}if(e.sameSite)switch(typeof e.sameSite=="string"?e.sameSite.toLowerCase():e.sameSite){case true:case "strict":o+="; SameSite=Strict";break;case "lax":o+="; SameSite=Lax";break;case "none":o+="; SameSite=None";break;default:throw new TypeError(`option sameSite is invalid: ${e.sameSite}`)}return o}function qe(i){if(i.indexOf("%")===-1)return i;try{return decodeURIComponent(i)}catch{return i}}function Qe(i){return De.call(i)==="[object Date]"}});var de="0123456789ABCDEFGHJKMNPQRSTVWXYZ",z=32;var Oe=16,le=10,ue=0xffffffffffff;var O;(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";})(O||(O={}));var L=class extends Error{constructor(t,e){super(`${e} (${t})`),this.name="ULIDError",this.code=t;}};function Le(i){let t=Math.floor(i()*z);return t===z&&(t=z-1),de.charAt(t)}function $e(i){var r;let t=Pe(),e=t&&(t.crypto||t.msCrypto)||(typeof D<"u"?D:null);if(typeof(e==null?void 0:e.getRandomValues)=="function")return ()=>{let n=new Uint8Array(1);return e.getRandomValues(n),n[0]/255};if(typeof(e==null?void 0:e.randomBytes)=="function")return ()=>e.randomBytes(1).readUInt8()/255;if((r=D)!=null&&r.randomBytes)return ()=>D.randomBytes(1).readUInt8()/255;throw new L(O.PRNGDetectFailure,"Failed to find a reliable PRNG")}function Pe(){return Ne()?self:typeof window<"u"?window:typeof global<"u"?global:typeof globalThis<"u"?globalThis:null}function Ce(i,t){let e="";for(;i>0;i--)e=Le(t)+e;return e}function ze(i,t=le){if(isNaN(i))throw new L(O.EncodeTimeValueMalformed,`Time must be a number: ${i}`);if(i>ue)throw new L(O.EncodeTimeSizeExceeded,`Cannot encode a time larger than ${ue}: ${i}`);if(i<0)throw new L(O.EncodeTimeNegative,`Time must be positive: ${i}`);if(Number.isInteger(i)===false)throw new L(O.EncodeTimeValueMalformed,`Time must be an integer: ${i}`);let e,r="";for(let n=t;n>0;n--)e=i%z,r=de.charAt(e)+r,i=(i-e)/z;return r}function Ne(){return typeof WorkerGlobalScope<"u"&&self instanceof WorkerGlobalScope}function pe(i,t){let e=$e(),r=Date.now();return ze(r,le)+Ce(Oe,e)}var K=()=>pe().toLowerCase();var _=(...i)=>{let t=i.filter(e=>!!e);return t.length===0?{}:t.length===1?t[0]:{$and:t}};var F=class{storage;queue=new Map;scheduled=false;constructor(t){this.storage=t;}async rawFind({resource:t,commonWhere:e,uniqueWhere:r,...n}){return new Promise((o,c)=>{let a=this.getBatchKey({resource:t,commonWhere:e,...n}),s={resource:t,commonWhere:e,uniqueWhere:r,...n,resolve:o,reject:c};this.queue.has(a)||this.queue.set(a,[]);let m=this.queue.get(a);m&&m.push(s),this.scheduled||(this.scheduled=true,setImmediate(()=>{this.processBatch();}));})}getBatchKey(t){let{resource:e,commonWhere:r,...n}=t;return `${e}:${JSON.stringify(r??{})}:${JSON.stringify(n??{})}`}async processBatch(){this.scheduled=false;let t=Array.from(this.queue.entries());this.queue.clear();for(let[,e]of t)try{await this.executeBatchedRequests(e);}catch(r){e.forEach(n=>{n.reject(r);});}}async executeBatchedRequests(t){var h,d;if(t.length===0)return;let e=t[0],{resource:r,commonWhere:n,include:o,sort:c}=e,a=t.length===1?e.limit:void 0,s=t.map(y=>y.uniqueWhere).filter(y=>y!==void 0),m=n,f=(h=Object.entries(s[0]??{})[0])==null?void 0:h[0];if(s.length>0){let y=s.map(l=>l[f]).filter(l=>l!=null);y.length>0&&(m=_(n,{[f]:{$in:y}}));}let p=await this.storage.rawFind({resource:r,where:m,include:o,sort:c,limit:a});for(let y of t){let l={};if(y.uniqueWhere){let[u,g]=Object.entries(y.uniqueWhere)[0];for(let[T,w]of Object.entries(p))((d=w.value[u])==null?void 0:d.value)===g&&(l[T]=w);}else Object.assign(l,p);y.resolve(l);}}};var ge=b(H(),1);var q=z$1.object({resource:z$1.string(),where:z$1.record(z$1.string(),z$1.any()).optional(),include:z$1.record(z$1.string(),z$1.any()).optional(),lastSyncedAt:z$1.string().optional(),limit:z$1.coerce.number().optional(),sort:z$1.array(z$1.object({key:z$1.string(),direction:z$1.enum(["asc","desc"])})).optional()}),J=z$1.record(z$1.string(),z$1.object({value:z$1.string().or(z$1.number()).or(z$1.boolean()).or(z$1.date()).nullable(),_meta:z$1.object({timestamp:z$1.string().optional().nullable()}).optional()})),Ge=J.superRefine((i,t)=>{i.id&&t.addIssue({code:z$1.ZodIssueCode.custom,message:"Payload cannot have an id"});}),fe=z$1.object({id:z$1.string().optional(),type:z$1.literal("MUTATE"),resource:z$1.string(),resourceId:z$1.string().optional()}),N=fe.extend({procedure:z$1.string(),payload:z$1.any().optional()}),U=fe.extend({procedure:z$1.enum(["INSERT","UPDATE"]),payload:Ge});z$1.union([U,N]);var he=q.omit({resource:true}),Y=N.omit({id:true,type:true,resource:true,procedure:true}),X=U.omit({id:true,type:true,resource:true,procedure:true});z$1.union([X,Y]);var Te=i=>{let t=i.logger;return async e=>{var r;try{let n=typeof e.headers.getSetCookie=="function"?Object.fromEntries(e.headers):e.headers,o={headers:n,cookies:n.cookie?ge.default.parse(n.cookie):{}},c=new URL(e.url),a=c.pathname.split("/"),s=c.searchParams,m=Ke.parse(s.toString()),f=await((r=i.contextProvider)==null?void 0:r.call(i,{transport:"HTTP",headers:o.headers,cookies:o.cookies,queryParams:m}))??{};if(e.method==="GET"){let p=a[a.length-1],{success:h,data:d,error:y}=he.safeParse(m);if(!h)return Response.json({message:"Invalid query",code:"INVALID_QUERY",details:y},{status:400});let l=await i.handleQuery({req:{...o,...d,type:"QUERY",resource:p,context:f,queryParams:m}});return !l||!l.data?Response.json({message:"Invalid resource",code:"INVALID_RESOURCE"},{status:400}):Response.json(l.data)}if(e.method==="POST")try{let p=a[a.length-1],h=a[a.length-2],d=e.body?await e.json():{},y;if(p==="insert"||p==="update"){let{success:u,data:g,error:T}=X.safeParse(d);if(!u)return Response.json({message:"Invalid mutation",code:"INVALID_REQUEST",details:T},{status:400});y=g;}else {let{success:u,data:g,error:T}=Y.safeParse(d);if(!u)return Response.json({message:"Invalid mutation",code:"INVALID_REQUEST",details:T},{status:400});y=g;}let l=await i.handleMutation({req:{...o,type:"MUTATE",resource:h,input:y.payload,context:f,resourceId:y.resourceId,procedure:p==="insert"||p==="update"?p.toUpperCase():p,queryParams:{}}});return Response.json(l)}catch(p){return t.error("Error parsing mutation from the client:",p),Response.json({message:"Internal server error",code:"INTERNAL_SERVER_ERROR"},{status:500})}return Response.json({message:"Not found",code:"NOT_FOUND"},{status:404})}catch(n){return t.error("Unexpected error:",n),Response.json({message:"Internal server error",code:"INTERNAL_SERVER_ERROR"},{status:500})}}};var be=b(H(),1);var $=z$1.string(),He=z$1.object({id:$,type:z$1.literal("SUBSCRIBE"),resource:z$1.string()}),Je=q.extend({id:$,type:z$1.literal("QUERY")}),Re=U.extend({id:$}),Ye=N.extend({id:$}),Xe=z$1.union([Ye,Re]),we=z$1.union([He,Je,Xe]),et=z$1.object({id:$,type:z$1.literal("REJECT"),resource:z$1.string(),message:z$1.string().optional()}),tt=z$1.object({id:$,type:z$1.literal("REPLY"),data:z$1.any()});z$1.union([et,tt,Re]);z$1.object({resource:z$1.string(),data:z$1.record(z$1.string(),J)});var xe=i=>{let t={},e={},r=i.logger;return i.subscribeToMutations(n=>{let o=n;!o.resourceId||!o.payload||(r.debug("Mutation propagated:",o),Object.entries(e[o.resource]??{}).forEach(([c,a])=>{var s;(s=t[c])==null||s.send(JSON.stringify({...o,id:o.id??K()}));}));}),(n,o)=>{var p;let c=h=>{n.send(JSON.stringify(h));},a=K(),s={headers:o.headers,cookies:typeof o.headers.cookie=="string"?be.default.parse(o.headers.cookie):{}},m=parse(o.url.split("?")[1]),f=(p=i.contextProvider)==null?void 0:p.call(i,{transport:"WEBSOCKET",headers:s.headers,cookies:s.cookies,queryParams:m});t[a]=n,r.info("Client connected:",a),n.on("message",async h=>{try{r.debug("Message received from the client:",h);let d=we.parse(JSON.parse(h.toString()));if(d.type==="SUBSCRIBE"){let{resource:y}=d;e[y]||(e[y]={}),e[y][a]={};}else if(d.type==="QUERY"){let{resource:y}=d,l=await i.handleQuery({req:{...s,type:"QUERY",resource:y,context:await f??{},queryParams:m}});if(!l||!l.data)throw new Error("Invalid resource");c({id:d.id,type:"REPLY",data:{resource:y,data:Object.fromEntries(Object.entries(l.data??{}).map(([u,g])=>[u,g.value]))}});}else if(d.type==="MUTATE"){let{resource:y}=d;r.debug("Received mutation from client:",d);try{let l=await i.handleMutation({req:{...s,type:"MUTATE",resource:y,input:d.payload,context:{messageId:d.id,...await f??{}},resourceId:d.resourceId,procedure:d.procedure,queryParams:m}});d.procedure&&d.procedure!=="INSERT"&&d.procedure!=="UPDATE"&&c({id:d.id,type:"REPLY",data:l});}catch(l){c({id:d.id,type:"REJECT",resource:y,message:l.message}),r.error("Error parsing mutation from the client:",l);}}}catch(d){r.error("Error handling message from the client:",d);}}),n.on("close",()=>{r.info("Connection closed",a),delete t[a];for(let h of Object.values(e))delete h[a];});}};function Se(i){let t=`${i.protocol}://${i.hostname}${i.url}`,e=new Headers;return Object.entries(i.headers).forEach(([r,n])=>{n&&e.set(r,Array.isArray(n)?n.join(","):n);}),new Request(t,{method:i.method,headers:e,body:i.body&&i.method!=="GET"?JSON.stringify(i.body):void 0})}var Ut=(i,t,e)=>{i.ws(`${(e==null?void 0:e.basePath)??""}/ws`,xe(t)),i.use(`${(e==null?void 0:e.basePath)??""}/`,(r,n)=>{Te(t)(Se(r)).then(c=>c.json().then(a=>n.status(c.status).send(a)));});};var ee=class i{routes;constructor(t){this.routes=t.routes;}static create(t){return new i(t)}},Ft=i=>ee.create({...i}),it=i=>({handler:t=>({inputValidator:i??z$1.undefined(),handler:t})}),te=class i{resourceSchema;middlewares;customMutations;authorization;constructor(t,e,r){this.resourceSchema=t,this.middlewares=new Set,this.customMutations=e??{},this.authorization=r;}use(...t){for(let e of t)this.middlewares.add(e);return this}withMutations(t){return new i(this.resourceSchema,t({mutation:it}),this.authorization)}handleQuery=async({req:t,batcher:e})=>await this.wrapInMiddlewares(async r=>{var o,c;let n=(c=(o=this.authorization)==null?void 0:o.read)==null?void 0:c.call(o,{ctx:r.context});if(typeof n=="boolean"&&!n)throw new Error("Not authorized");return {data:await e.rawFind({resource:r.resource,commonWhere:_(r.where,typeof n=="object"?n:void 0),uniqueWhere:r.relationalWhere,include:r.include,limit:r.limit,sort:r.sort})}})(t);handleMutation=async({req:t,db:e,schema:r})=>await this.wrapInMiddlewares(async n=>{if(!n.procedure)throw new Error("Procedure is required for mutations");let o=this.customMutations[n.procedure];if(o){let c=o.inputValidator.parse(n.input);return n.input=c,o.handler({req:n,db:e})}else {if(n.procedure==="INSERT"||n.procedure==="UPDATE")return this.handleSet({req:n,db:e,operation:n.procedure,schema:r});throw new Error(`Unknown procedure: ${n.procedure}`)}})(t);handleSet=async({req:t$1,db:e,operation:r,schema:n})=>{if(!t$1.input)throw new Error("Payload is required");if(!t$1.resourceId)throw new Error("ResourceId is required");let o=await e.rawFindById(t$1.resource,t$1.resourceId);if(r==="INSERT"&&o)throw new Error("Resource already exists");if(r==="UPDATE"&&!o)throw new Error("Resource not found");return e.transaction(async({trx:c})=>{var f,p,h,d,y;let[a,s]=this.resourceSchema.mergeMutation("set",t$1.input,o);if(!s)throw new Error("Mutation rejected");if(r==="INSERT"){let l=await c.rawInsert(t$1.resource,t$1.resourceId,a),u=t(l);if(u.id=u.id??t$1.resourceId,(f=this.authorization)!=null&&f.insert){let g=this.authorization.insert({ctx:t$1.context,value:u});if(typeof g=="boolean"){if(!g)throw new Error("Not authorized")}else {let T=v(g,t$1.resource,n),w$1=Object.keys(T).length>0?await c.rawFindById(t$1.resource,t$1.resourceId,T):l,b=t(w$1);if(b.id=b.id??t$1.resourceId,!w(b,g))throw new Error("Not authorized")}}return {data:l,acceptedValues:s}}if((h=(p=this.authorization)==null?void 0:p.update)!=null&&h.preMutation){let l=t(o);l.id=l.id??t$1.resourceId;let u=this.authorization.update.preMutation({ctx:t$1.context,value:l});if(typeof u=="boolean"){if(!u)throw new Error("Not authorized")}else {let g=v(u,t$1.resource,n),T=Object.keys(g).length>0?await c.rawFindById(t$1.resource,t$1.resourceId,g):o,w$1=t(T);if(w$1.id=w$1.id??t$1.resourceId,!w(w$1,u))throw new Error("Not authorized")}}let m=await c.rawUpdate(t$1.resource,t$1.resourceId,a);if((y=(d=this.authorization)==null?void 0:d.update)!=null&&y.postMutation){let l=t(m);l.id=l.id??t$1.resourceId;let u=this.authorization.update.postMutation({ctx:t$1.context,value:l});if(typeof u=="boolean"){if(!u)throw new Error("Not authorized")}else {let g=v(u,t$1.resource,n),T=Object.keys(g).length>0?await c.rawFindById(t$1.resource,t$1.resourceId,g):m,w$1=t(T);if(w$1.id=w$1.id??t$1.resourceId,!w(w$1,u))throw new Error("Not authorized")}}return {data:m,acceptedValues:s}})};wrapInMiddlewares(t){return e=>Array.from(this.middlewares.values()).reduceRight((r,n)=>o=>n({req:o,next:r}),t)(e)}},re=class i{middlewares;constructor(t=[]){this.middlewares=t;}collectionRoute(t,e){return new te(t,void 0,e).use(...this.middlewares)}use(...t){return new i([...this.middlewares,...t])}static create(){return new i}},Vt=re.create;var W=class{async insert(t$1,e){let r=new Date().toISOString();return t(await this.rawInsert(t$1.name,e.id,{value:Object.fromEntries(Object.entries(e).map(([n,o])=>[n,{value:o,_meta:{timestamp:r}}]))}))}async update(t$1,e,r){let n=new Date().toISOString(),{id:o,...c}=r;return t(await this.rawUpdate(t$1.name,e,{value:Object.fromEntries(Object.entries(c).map(([a,s])=>[a,{value:s,_meta:{timestamp:n}}]))}))}};function Q(i,t,e,r){if(!i)throw new Error("Schema not initialized");let n=i[t];if(!n)throw new Error("Resource not found");let o=r.$or,c=r.$and;return (o?e.or:e.and)(o?r.$or.map(a=>Q(i,t,e,a)):c?r.$and.map(a=>Q(i,t,e,a)):Object.entries(r).map(([a,s])=>{var m,f;if(n.fields[a])return (s==null?void 0:s.$eq)!==void 0?e(`${t}.${a}`,s.$eq===null?"is":"=",s.$eq):(s==null?void 0:s.$in)!==void 0?e(`${t}.${a}`,"in",s.$in):(s==null?void 0:s.$not)!==void 0?((m=s==null?void 0:s.$not)==null?void 0:m.$in)!==void 0?e(`${t}.${a}`,"not in",s.$not.$in):((f=s==null?void 0:s.$not)==null?void 0:f.$eq)!==void 0?e(`${t}.${a}`,s.$not.$eq===null?"is not":"!=",s.$not.$eq):e(`${t}.${a}`,s.$not===null?"is not":"!=",s.$not):(s==null?void 0:s.$gt)!==void 0?e(`${t}.${a}`,">",s.$gt):(s==null?void 0:s.$gte)!==void 0?e(`${t}.${a}`,">=",s.$gte):(s==null?void 0:s.$lt)!==void 0?e(`${t}.${a}`,"<",s.$lt):(s==null?void 0:s.$lte)!==void 0?e(`${t}.${a}`,"<=",s.$lte):e(`${t}.${a}`,s===null?"is":"=",s);if(n.relations[a]){let p=n.relations[a],h=p.entity.name;return p.type==="many"?e.exists(ne(i,h,e.selectFrom(h).select("id").whereRef(p.foreignColumn,"=",`${t}.id`),s)):Q(i,h,e,s)}return null}).filter(Boolean))}function G(i,t,e,r){let n=i[t];if(!n)throw new Error("Resource not found");if(!r)return e;if(r.$and){for(let o of r.$and)e=G(i,t,e,o);return e}else if(r.$or){for(let o of r.$or)e=G(i,t,e,o);return e}for(let[o,c]of Object.entries(r)){if(!n.relations[o])continue;let a=n.relations[o],s=a.entity.name,m=a.type==="one"?"id":a.foreignColumn,f=a.type==="one"?a.relationalColumn:"id";e=e.leftJoin(s,`${s}.${m}`,`${t}.${f}`),c instanceof Object&&!Array.isArray(c)&&c!==null&&(e=G(i,s,e,c));}return e}function ne(i,t,e,r){return !r||Object.keys(r).length===0?e:(e=G(i,t,e,r),e.where(n=>Q(i,t,n,r)))}function Z(i,t,e,r){if(!r)return e;if(!i)throw new Error("Schema not initialized");let n=i[t];if(!n)throw new Error(`Resource not found: ${t}`);for(let o of Object.keys(r)){if(!n.relations[o])throw new Error(`Relation ${o} not found in resource ${t}`);let c=n.relations[o],a=c.entity.name,s=r[o],m=c.type==="one"?"id":c.foreignColumn,f=c.type==="one"?c.relationalColumn:"id",p=c.type==="one"?jsonObjectFrom:jsonArrayFrom,h=typeof s=="object"&&s!==null;e=e.select(d=>{let y=d.selectFrom(a).selectAll(a).whereRef(`${a}.${m}`,"=",`${t}.${f}`).select(l=>jsonObjectFrom(l.selectFrom(`${a}_meta`).selectAll(`${a}_meta`).whereRef(`${a}_meta.id`,"=",`${a}.id`)).as("_meta"));return h&&(y=Z(i,a,y,s)),p(y).as(o)});}return e}var Ee="42701",ie=class i extends W{db;schema;logger;constructor(t,e,r){super(),this.isKyselyLike(t)?this.db=t:this.db=new Kysely({dialect:new PostgresDialect({pool:t})}),this.schema=e,this.logger=r,this.rawInsert=this.rawInsert.bind(this),this.rawUpdate=this.rawUpdate.bind(this);}async init(t,e){var n;this.schema=t,this.logger=e;let r=await this.db.introspection.getTables();for(let[o,c]of Object.entries(t)){let a=r.find(f=>f.name===o);a||await this.db.schema.createTable(o).ifNotExists().execute();let s=`${o}_meta`,m=r.find(f=>f.name===s);m||await this.db.schema.createTable(s).ifNotExists().execute();for(let[f,p]of Object.entries(c.fields)){let h=a==null?void 0:a.columns.find(l=>l.name===f),d=p.getStorageFieldType();h?h.dataType!==d.type&&((n=this.logger)==null||n.warn("Column type mismatch:",f,"expected to have type:",d.type,"but has type:",h.dataType)):(await this.db.schema.alterTable(o).addColumn(f,d.type,l=>{let u=l;return d.unique&&(u=u.unique()),d.nullable||(u=u.notNull()),d.references&&(u=u.references(d.references)),d.primary&&(u=u.primaryKey()),d.default!==void 0&&(u=u.defaultTo(d.default)),u}).execute().catch(l=>{var u;if(l.code!==Ee)throw (u=this.logger)==null||u.error("Error adding column",f,l),l}),d.index&&await this.db.schema.createIndex(`${o}_${f}_index`).on(o).column(f).execute().catch(()=>{})),(m==null?void 0:m.columns.find(l=>l.name===f))||await this.db.schema.alterTable(s).addColumn(f,"varchar",l=>{let u=l;return d.primary&&(u=u.primaryKey().references(`${o}.${f}`)),u}).execute().catch(l=>{var u;if(l.code!==Ee)throw (u=this.logger)==null||u.error("Error adding meta column",f,l),l});}}}async rawFindById(t,e,r){if(!this.schema)throw new Error("Schema not initialized");let n=await this.db.selectFrom(t).where("id","=",e).selectAll(t).select(c=>jsonObjectFrom(c.selectFrom(`${t}_meta`).selectAll(`${t}_meta`).whereRef(`${t}_meta.id`,"=",`${t}.id`)).as("_meta"));n=Z(this.schema,t,n,r);let o=await n.executeTakeFirst();if(o)return this.convertToMaterializedLiveType(o)}async findOne(t$1,e,r){let n=await this.rawFindById(t$1.name,e,r==null?void 0:r.include);if(n)return t(n)}async rawFind(t){if(!this.schema)throw new Error("Schema not initialized");let{resource:e,where:r,include:n,limit:o,sort:c}=t,a=this.db.selectFrom(e).selectAll(e).select(p=>jsonObjectFrom(p.selectFrom(`${e}_meta`).selectAll(`${e}_meta`).whereRef(`${e}_meta.id`,"=",`${e}.id`)).as("_meta"));a=ne(this.schema,e,a,r),a=Z(this.schema,e,a,n),o!==void 0&&(a=a.limit(o)),c!==void 0&&c.forEach(p=>{a=a.orderBy(p.key,p.direction);});let s=await a.execute(),m=Object.fromEntries(s.map(p=>{let{id:h}=p;return [h,p]}));return Object.keys(m).length===0?{}:Object.entries(m).reduce((p,[h,d])=>(p[h]=this.convertToMaterializedLiveType(d),p),{})}async find(t$1,e){let r=await this.rawFind({resource:t$1.name,where:e==null?void 0:e.where,include:e==null?void 0:e.include,limit:e==null?void 0:e.limit,sort:e==null?void 0:e.sort});return Object.fromEntries(Object.entries(r).map(([n,o])=>[n,t(o)]))}async rawInsert(t,e,r){var c;let n={},o={};for(let[a,s]of Object.entries(r.value)){let m=(c=s._meta)==null?void 0:c.timestamp;m&&(n[a]=s.value,o[a]=m);}return await this.db.insertInto(t).values({...n,id:e}).execute().then(()=>{this.db.insertInto(`${t}_meta`).values({...o,id:e}).execute();}),r}async rawUpdate(t,e,r){var c;let n={},o={};for(let[a,s]of Object.entries(r.value)){let m=(c=s._meta)==null?void 0:c.timestamp;m&&(n[a]=s.value,o[a]=m);}return await Promise.all([this.db.updateTable(t).set(n).where("id","=",e).execute(),this.db.insertInto(`${t}_meta`).values({...o,id:e}).onConflict(a=>a.column("id").doUpdateSet(o)).execute()]),r}async transaction(t){if(!this.schema)throw new Error("Schema not initialized");if(this.db.isTransaction){let r=Math.random().toString(36).substring(2,15),n=await this.db.savepoint(r).execute();try{return await t({trx:this,commit:()=>n.releaseSavepoint(r).execute().then(()=>{}),rollback:()=>n.rollbackToSavepoint(r).execute().then(()=>{})}).then(o=>n.isCommitted||n.isRolledBack?o:n.releaseSavepoint(r).execute().then(()=>o))}catch(o){throw await n.rollbackToSavepoint(r).execute().catch(()=>{}),o}}let e=await this.db.startTransaction().execute();try{return await t({trx:new i(e,this.schema,this.logger),commit:()=>e.commit().execute(),rollback:()=>e.rollback().execute()}).then(r=>e.isCommitted||e.isRolledBack?r:e.commit().execute().then(()=>r))}catch(r){throw await e.rollback().execute(),r}}convertToMaterializedLiveType(t){return {value:Object.entries(t).reduce((e,[r,n])=>{var o,c,a;return r==="_meta"||(r==="id"?e[r]={value:n}:Array.isArray(n)?e[r]={value:n.map(s=>this.convertToMaterializedLiveType(s)),_meta:{timestamp:(o=t==null?void 0:t._meta)==null?void 0:o[r]}}:typeof n=="object"&&n!==null&&!(n instanceof Date)?e[r]={...this.convertToMaterializedLiveType(n),_meta:{timestamp:(c=t==null?void 0:t._meta)==null?void 0:c[r]}}:e[r]={value:n,_meta:{timestamp:(a=t==null?void 0:t._meta)==null?void 0:a[r]}}),e},{})}}isKyselyLike(t){if(t instanceof Kysely)return true;if(!t||typeof t!="object")return false;let e=t,r=typeof e.selectFrom=="function",n=typeof e.startTransaction=="function",o=typeof e.savepoint=="function",c=typeof e.isTransaction=="boolean"||typeof e.isTransaction=="function";return r&&n||o&&c}};var oe=class i{router;storage;schema;middlewares=new Set;logger;contextProvider;mutationSubscriptions=new Set;constructor(t){var e;this.router=t.router,this.storage=t.storage,this.schema=t.schema,this.logger=y({level:t.logLevel??x.INFO}),(e=t.middlewares)==null||e.forEach(r=>{this.middlewares.add(r);}),this.storage.init(this.schema,this.logger),this.contextProvider=t.contextProvider;}static create(t){return new i(t)}subscribeToMutations(t){return this.mutationSubscriptions.add(t),()=>{this.mutationSubscriptions.delete(t);}}handleQuery(t){let e=new F(this.storage);return this.wrapInMiddlewares(async r=>{var d,y,l;let n=Ae(r,this.schema,{stepId:"query",collectionName:r.resource,included:Object.keys(r.include??{})}),o={headers:r.headers,cookies:r.cookies,queryParams:r.queryParams,context:r.context},c={};for(let u=0;u<n.length;u++){let g=n[u],T=this.router.routes[g.resource];if(!T)throw new Error("Invalid resource");let w;if(g.getWhere&&g.referenceGetter){let x=g.referenceGetter(c);w=[];for(let I=0;I<x.length;I++)w.push(g.getWhere(x[I]));}else w=[void 0];let b=c[g.prevStepId??""],M=[];if(b)for(let x=0;x<b.length;x++){let I=b[x],A=Object.keys(((d=I==null?void 0:I.result)==null?void 0:d.data)??{});for(let C=0;C<A.length;C++)M.push(A[C]);}let j=[];for(let x=0;x<w.length;x++){let I=w[x],A=M[x];j.push((async()=>{let C=await T.handleQuery({req:{type:"QUERY",...g,...o,where:g.where,relationalWhere:I},batcher:e});return {includedBy:A,result:C}})());}let E=await Promise.allSettled(j),P=[];for(let x=0;x<E.length;x++){let I=E[x];I.status==="fulfilled"&&P.push(I.value);}c[g.stepId]=P;}let a=new Map,s=0;for(let u in c){let g=c[u],T=n[s];s++;for(let w=0;w<g.length;w++){let b=g[w],M=b.result.data;for(let j in M){let E=M[j],P=`${u}.${j}`,x=[];u!=="query"&&b.includedBy&&(x=[`${T.prevStepId}.${b.includedBy}`]);let I=a.get(P);if(I)for(let A=0;A<x.length;A++)I.includedBy.add(x[A]);else a.set(P,{data:E,includedBy:new Set(x),path:u.split(".").slice(-1)[0],isMany:T.isMany??false,collectionName:T.collectionName,included:T.included});}}}let m=Object.fromEntries(a),f={data:{}},p=Object.keys(m);for(let u=p.length-1;u>=0;u--){let g=p[u],T=m[g],w=T.path;if(w==="query"&&(f.data[g.replace("query.","")]=T.data),T.included.length)for(let b=0;b<T.included.length;b++){let M=T.included[b];T.data.value[M]??=((l=(y=this.schema[T.collectionName])==null?void 0:y.relations[M])==null?void 0:l.type)==="many"?{value:[]}:{value:null};}if(T.includedBy.size>0){let b=Array.from(T.includedBy);for(let M=0;M<b.length;M++){let j=b[M],E=m[j];E&&(T.isMany?(E.data.value[w]??={value:[]},E.data.value[w].value.push(T.data)):E.data.value[w]=T.data);}}}return f})(t.req)}async handleMutation(t){let e=await this.wrapInMiddlewares(async r=>{let n=this.router.routes[r.resource];if(!n)throw new Error("Invalid resource");return n.handleMutation({req:r,db:this.storage,schema:this.schema})})(t.req);if(e&&t.req.type==="MUTATE"&&e.acceptedValues&&(t.req.procedure==="INSERT"||t.req.procedure==="UPDATE")&&t.req.resourceId){let n=e.acceptedValues??{},o=t.req,c=o.resourceId;Object.keys(n).length&&c&&this.mutationSubscriptions.forEach(a=>{a({id:t.req.context.messageId,type:"MUTATE",resource:o.resource,payload:n,resourceId:c,procedure:o.procedure});});}return e}use(t){return this.middlewares.add(t),this}context(t){return this.contextProvider=t,this}wrapInMiddlewares(t){return e=>Array.from(this.middlewares.values()).reduceRight((r,n)=>o=>n({req:o,next:r}),t)(e)}},sr=oe.create;function Ae(i,t,e){let{include:r,where:n,...o}=i,{stepId:c}=e,a=[{...o,...e,where:n}];if(r&&typeof r=="object"&&Object.keys(r).length>0){let s=t[o.resource];if(!s)throw new Error(`Resource ${o.resource} not found`);a.push(...Object.entries(r).flatMap(([m,f])=>{let p=s.relations[m];if(!p)throw new Error(`Relation ${m} not found for resource ${o.resource}`);let h=p.entity.name;return Ae({...o,resource:h,include:f},t,{getWhere:p.type==="one"?d=>({id:d}):d=>({[p.foreignColumn]:d}),referenceGetter:d=>d[c].flatMap(y=>y.result.data?p.type==="one"?Object.values(y.result.data).map(l=>{var u,g;return (g=(u=l.value)==null?void 0:u[p.relationalColumn])==null?void 0:g.value}):Object.keys(y.result.data):[]),stepId:`${c}.${m}`,prevStepId:c,isMany:p.type==="many",collectionName:h,included:typeof f=="object"?Object.keys(f):[]})}));}return a}
2
+ export{te as Route,re as RouteFactory,ee as Router,ie as SQLStorage,oe as Server,W as Storage,Ut as expressAdapter,Vt as routeFactory,Ft as router,sr as server};
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@live-state/sync",
3
- "version": "0.0.5",
3
+ "version": "0.0.6-canary-1",
4
4
  "private": false,
5
5
  "publishConfig": {
6
6
  "access": "public"
@@ -27,6 +27,7 @@
27
27
  "@types/express": "^4.17.21",
28
28
  "@types/express-ws": "^3.0.5",
29
29
  "@types/node": "^20.11.24",
30
+ "@types/pg": "^8.15.1",
30
31
  "@types/qs": "^6.14.0",
31
32
  "@types/react": "^18.2.62",
32
33
  "@types/react-dom": "^18.2.19",
@@ -35,6 +36,8 @@
35
36
  "cookie": "^1.0.2",
36
37
  "express": "^4.21.2",
37
38
  "jsdom": "^26.1.0",
39
+ "pg": "^8.15.6",
40
+ "express-ws": "^5.0.2",
38
41
  "react": "18.0.0",
39
42
  "react-dom": "^18.2.0",
40
43
  "tsup": "^8.0.2",