@live-state/sync 0.0.1 → 0.0.3

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/index.d.ts CHANGED
@@ -34,18 +34,18 @@ type InferLiveType<T extends LiveTypeAny> = T["_value"] extends Record<string, L
34
34
  } : T["_value"];
35
35
  type InferIndex<T extends LiveTypeAny> = string;
36
36
 
37
- declare class OptionalLiveType<T extends LiveTypeAny> extends LiveType<T["_value"] | undefined, T["_meta"], T["_encodeInput"], T["_decodeInput"]> {
37
+ declare class NullableLiveType<T extends LiveTypeAny> extends LiveType<T["_value"] | null, T["_meta"], T["_encodeInput"], T["_decodeInput"]> {
38
38
  readonly inner: T;
39
39
  constructor(inner: T);
40
- encodeMutation(mutationType: MutationType, input: T["_value"] | undefined, timestamp: string): T["_decodeInput"];
41
- mergeMutation(mutationType: MutationType, encodedMutation: T["_decodeInput"], materializedShape?: MaterializedLiveType<LiveType<T["_value"] | undefined, T["_meta"], T["_value"] | Partial<T["_value"] | undefined>, T["_decodeInput"]>> | undefined): [
42
- MaterializedLiveType<LiveType<T["_value"] | undefined, T["_meta"], T["_value"] | Partial<T["_value"] | undefined>, T["_decodeInput"]>>,
40
+ encodeMutation(mutationType: MutationType, input: T["_value"] | null, timestamp: string): T["_decodeInput"];
41
+ mergeMutation(mutationType: MutationType, encodedMutation: T["_decodeInput"], materializedShape?: MaterializedLiveType<LiveType<T["_value"] | null, T["_meta"], T["_value"] | Partial<T["_value"] | null>, T["_decodeInput"]>> | undefined): [
42
+ MaterializedLiveType<LiveType<T["_value"] | null, T["_meta"], T["_value"] | Partial<T["_value"] | null>, T["_decodeInput"]>>,
43
43
  T["_decodeInput"] | null
44
44
  ];
45
45
  getStorageFieldType(): StorageFieldType;
46
46
  }
47
47
  type LiveAtomicTypeMeta = {
48
- timestamp: string;
48
+ timestamp: string | null;
49
49
  } & LiveTypeMeta;
50
50
  declare class LiveAtomicType<Value> extends LiveType<Value, LiveAtomicTypeMeta, Value, {
51
51
  value: Value;
@@ -84,7 +84,7 @@ declare class LiveAtomicType<Value> extends LiveType<Value, LiveAtomicTypeMeta,
84
84
  index(): LiveAtomicType<Value>;
85
85
  default(value: Value): LiveAtomicType<Value>;
86
86
  primary(): LiveAtomicType<Value>;
87
- optional(): OptionalLiveType<this>;
87
+ nullable(): NullableLiveType<this>;
88
88
  }
89
89
  declare class LiveNumber extends LiveAtomicType<number> {
90
90
  private constructor();
@@ -111,12 +111,14 @@ declare class LiveTimestamp extends LiveAtomicType<Date> {
111
111
  }
112
112
  declare const timestamp: typeof LiveTimestamp.create;
113
113
 
114
+ /** biome-ignore-all lint/complexity/noBannedTypes: false positive */
115
+
114
116
  type InferLiveObjectWithoutRelations<T extends LiveObjectAny> = {
115
117
  [K in keyof T["fields"]]: InferLiveType<T["fields"][K]>;
116
118
  };
117
- type InferLiveObject<T extends LiveObjectAny> = InferLiveObjectWithoutRelations<T> & {
118
- [K in keyof T["relations"]]: T["relations"][K]["type"] extends "one" ? InferLiveObject<T["relations"][K]["entity"]> : InferLiveObject<T["relations"][K]["entity"]>[];
119
- };
119
+ type InferLiveObject<T extends LiveObjectAny, Include extends IncludeClause<T> | undefined = undefined> = InferLiveObjectWithoutRelations<T> & (Include extends IncludeClause<T> ? {
120
+ [K in keyof T["relations"] as Include[K] extends true ? K : never]: T["relations"][K]["type"] extends "one" ? InferLiveObject<T["relations"][K]["entity"]> : InferLiveObject<T["relations"][K]["entity"]>[];
121
+ } : {});
120
122
  type InferRelationalColumns<T extends Record<string, RelationAny>> = {
121
123
  [K in keyof T as T[K] extends Relation<any, any, any, infer ColumnName, any, any> ? ColumnName extends string ? ColumnName : never : never]: T[K]["type"] extends "one" ? T[K] extends Relation<infer Entity, any, any, any, any, any> ? T[K]["required"] extends true ? InferIndex<Entity> : InferIndex<Entity> | undefined : never : never;
122
124
  };
@@ -136,7 +138,7 @@ declare class LiveObject<TName extends string, TSchema extends Record<string, Li
136
138
  declare const object: typeof LiveObject.create;
137
139
  type LiveObjectAny = LiveObject<string, Record<string, LiveTypeAny>, any>;
138
140
  declare class Relation<TEntity extends LiveObjectAny, TSourceEntity extends LiveObjectAny, TType extends "one" | "many", TRelationalColumn extends keyof TSourceEntity["fields"], TForeignColumn extends keyof TEntity["fields"], TRequired extends boolean> extends LiveType<InferIndex<TEntity>, {
139
- timestamp: string;
141
+ timestamp: string | null;
140
142
  } & LiveTypeMeta> {
141
143
  readonly entity: TEntity;
142
144
  readonly type: TType;
@@ -166,6 +168,13 @@ declare class Relation<TEntity extends LiveObjectAny, TSourceEntity extends Live
166
168
  } | null
167
169
  ];
168
170
  getStorageFieldType(): StorageFieldType;
171
+ toJSON(): {
172
+ entityName: string;
173
+ type: TType;
174
+ required: TRequired;
175
+ relationalColumn: TRelationalColumn | undefined;
176
+ foreignColumn: TForeignColumn | undefined;
177
+ };
169
178
  static createOneFactory<TOriginEntity extends LiveObjectAny>(): <TEntity extends LiveObjectAny, TColumn extends keyof TOriginEntity["fields"], TRequired extends boolean = false>(entity: TEntity, column: TColumn, required?: TRequired) => Relation<TEntity, TOriginEntity, "one", TColumn, never, TRequired>;
170
179
  static createManyFactory<TOriginEntity extends LiveObjectAny>(): <TEntity extends LiveObjectAny, TColumn extends keyof TEntity["fields"], TRequired extends boolean = false>(entity: TEntity, foreignColumn: TColumn, required?: TRequired) => Relation<TEntity, TOriginEntity, "many", never, TColumn, TRequired>;
171
180
  }
@@ -200,10 +209,25 @@ type Schema<TRawSchema extends RawSchema> = {
200
209
  [K in keyof TRawSchema as TRawSchema[K] extends LiveObjectAny ? TRawSchema[K]["name"] : never]: TRawSchema[K] extends LiveObjectAny ? ParseObjectFromSchema<TRawSchema, TRawSchema[K]["name"]> : never;
201
210
  };
202
211
  declare const createSchema: <TRawSchema extends RawSchema>(schema: TRawSchema) => Schema<TRawSchema>;
203
- type WhereClause<T extends LiveObjectAny> = {
204
- [K in keyof T["fields"]]?: InferLiveType<T["fields"][K]>;
212
+ type WhereClause<T extends LiveObjectAny> = ({
213
+ [K in keyof T["fields"]]?: InferLiveType<T["fields"][K]> | ({
214
+ $eq?: InferLiveType<T["fields"][K]>;
215
+ $in?: InferLiveType<T["fields"][K]>[];
216
+ $not?: InferLiveType<T["fields"][K]> | {
217
+ $in?: InferLiveType<T["fields"][K]>[];
218
+ $eq?: InferLiveType<T["fields"][K]>;
219
+ };
220
+ } & (InferLiveType<T["fields"][K]> extends number ? {
221
+ $gt?: InferLiveType<T["fields"][K]>;
222
+ $gte?: InferLiveType<T["fields"][K]>;
223
+ $lt?: InferLiveType<T["fields"][K]>;
224
+ $lte?: InferLiveType<T["fields"][K]>;
225
+ } : {}));
205
226
  } & {
206
227
  [K in keyof T["relations"]]?: WhereClause<T["relations"][K]["entity"]>;
228
+ }) | {
229
+ $and?: WhereClause<T>[];
230
+ $or?: WhereClause<T>[];
207
231
  };
208
232
  type IncludeClause<T extends LiveObjectAny> = {
209
233
  [K in keyof T["relations"]]?: boolean;
package/dist/index.js CHANGED
@@ -1 +1 @@
1
- import'./chunk-EK7ODJWE.js';var o=class{_value;_meta;_encodeInput;_decodeInput};var T=class extends o{inner;constructor(e){super(),this.inner=e;}encodeMutation(e,t,n){return this.inner.encodeMutation(e,t,n)}mergeMutation(e,t,n){return this.inner.mergeMutation(e,t,n)}getStorageFieldType(){return {...this.inner.getStorageFieldType(),nullable:true}}},s=class a extends o{storageType;convertFunc;isIndex;isUnique;defaultValue;foreignReference;isPrimary;constructor(e,t,n,r,i,y,c){super(),this.storageType=e,this.convertFunc=t,this.isIndex=n??false,this.isUnique=r??false,this.defaultValue=i,this.foreignReference=y,this.isPrimary=c??false;}encodeMutation(e,t,n){return {value:t,_meta:{timestamp:n}}}mergeMutation(e,t,n){return n&&n._meta.timestamp.localeCompare(t._meta.timestamp)>=0?[n,null]:[{value:this.convertFunc?this.convertFunc(t.value):t.value,_meta:t._meta},t]}getStorageFieldType(){return {type:this.storageType,nullable:false,index:this.isIndex,unique:this.isUnique,default:this.defaultValue,references:this.foreignReference,primary:this.isPrimary}}unique(){return new a(this.storageType,this.convertFunc,this.isIndex,true,this.defaultValue,this.foreignReference,this.isPrimary)}index(){return new a(this.storageType,this.convertFunc,true,this.isUnique,this.defaultValue,this.foreignReference,this.isPrimary)}default(e){return new a(this.storageType,this.convertFunc,this.isIndex,this.isUnique,e,this.foreignReference,this.isPrimary)}primary(){return new a(this.storageType,this.convertFunc,this.isIndex,this.isUnique,this.defaultValue,this.foreignReference,true)}optional(){return new T(this)}},d=class a extends s{constructor(){super("integer",e=>Number(e));}static create(){return new a}},O=d.create,l=class a extends s{constructor(e){super("varchar",void 0,void 0,void 0,void 0,e);}static create(){return new a}static createId(){return new a().index().unique().primary()}static createReference(e){return new a(e)}},j=l.create,S=l.createId,I=l.createReference,p=class a extends s{constructor(){super("boolean",e=>typeof e=="string"?e.toLowerCase()==="true":!!e);}static create(){return new a}},w=p.create,m=class a extends s{constructor(){super("timestamp",e=>typeof e=="string"?new Date(e):e);}static create(){return new a}},A=m.create;var v=class a extends o{name;fields;relations;constructor(e,t,n){super(),this.name=e,this.fields=t,this.relations=n??{};}encodeMutation(e,t,n){return Object.fromEntries(Object.entries(t).map(([r,i])=>[r,(this.fields[r]??this.relations[r]).encodeMutation("set",i,n)]))}mergeMutation(e,t,n){let r={};return [{value:{...(n==null?void 0:n.value)??{},...Object.fromEntries(Object.entries(t).map(([i,y])=>{let[c,f]=(this.fields[i]??this.relations[i]).mergeMutation(e,y,n==null?void 0:n.value[i]);return f&&(r[i]=f),[i,c]}))}},r]}setRelations(e){return new a(this.name,this.fields,e)}getStorageFieldType(){throw new Error("Method not implemented.")}static create(e,t){return new a(e,t)}},P=v.create,u=class a extends o{entity;type;required;relationalColumn;foreignColumn;sourceEntity;constructor(e,t,n,r,i){super(),this.entity=e,this.type=t,this.required=i??false,this.relationalColumn=n,this.foreignColumn=r;}encodeMutation(e,t,n){if(e!=="set")throw new Error("Mutation type not implemented.");if(this.type==="many")throw new Error("Many not implemented.");return {value:t,_meta:{timestamp:n}}}mergeMutation(e,t,n){if(this.type==="many")throw new Error("Many not implemented.");return n&&n._meta.timestamp.localeCompare(t._meta.timestamp)>=0?[n,null]:[t,t]}getStorageFieldType(){return {type:"varchar",nullable:!this.required,references:`${this.entity.name}.${String(this.foreignColumn??this.relationalColumn??"id")}`}}static createOneFactory(){return (e,t,n)=>new a(e,"one",t,void 0,n??false)}static createManyFactory(){return (e,t,n)=>new a(e,"many",void 0,t,n??false)}},D=(a,e)=>({$type:"relations",objectName:a.name,relations:e({one:u.createOneFactory(),many:u.createManyFactory()})}),h=a=>{if(a)return Array.isArray(a.value)?a.value.map(e=>h(e)):typeof a.value!="object"?a.value:Object.fromEntries(Object.entries(a.value).map(([e,t])=>[e,h(t)]))},k=a=>Object.fromEntries(Object.entries(a).flatMap(([e,t])=>{if(t.$type==="relations")return [];let n=t,r=Object.values(a).find(i=>i.$type==="relations"&&i.objectName===t.name);return r&&(n=n.setRelations(r.relations)),[[n.name,n]]}));export{p as LiveBoolean,d as LiveNumber,v as LiveObject,l as LiveString,m as LiveTimestamp,o as LiveType,u as Relation,w as boolean,D as createRelations,k as createSchema,S as id,h as inferValue,O as number,P as object,I as reference,j as string,A as timestamp};
1
+ export{j as LiveBoolean,d as LiveNumber,n as LiveObject,f as LiveString,l as LiveTimestamp,c as LiveType,p as Relation,k as boolean,q as createRelations,s as createSchema,h as id,r as inferValue,e as number,o as object,i as reference,g as string,m as timestamp}from'./chunk-LLHCJUB6.js';
package/dist/server.cjs CHANGED
@@ -1,2 +1,2 @@
1
- 'use strict';var Se=require('qs'),zod=require('zod'),E=require('crypto'),kysely=require('kysely'),postgres=require('kysely/helpers/postgres');function _interopDefault(e){return e&&e.__esModule?e:{default:e}}var Se__default=/*#__PURE__*/_interopDefault(Se);var E__default=/*#__PURE__*/_interopDefault(E);var oe=Object.create;var U=Object.defineProperty;var ie=Object.getOwnPropertyDescriptor;var se=Object.getOwnPropertyNames;var ce=Object.getPrototypeOf,ue=Object.prototype.hasOwnProperty;var de=(r,e)=>()=>(e||r((e={exports:{}}).exports,e),e.exports);var le=(r,e,t,n)=>{if(e&&typeof e=="object"||typeof e=="function")for(let a of se(e))!ue.call(r,a)&&a!==t&&U(r,a,{get:()=>e[a],enumerable:!(n=ie(e,a))||n.enumerable});return r};var $=(r,e,t)=>(t=r!=null?oe(ce(r)):{},le(U(t,"default",{value:r,enumerable:true}),r));var P=de(I=>{Object.defineProperty(I,"__esModule",{value:true});I.parse=Te;I.serialize=Re;var pe=/^[\u0021-\u003A\u003C\u003E-\u007E]+$/,me=/^[\u0021-\u003A\u003C-\u007E]*$/,ye=/^([.]?[a-z0-9]([a-z0-9-]{0,61}[a-z0-9])?)([.][a-z0-9]([a-z0-9-]{0,61}[a-z0-9])?)*$/i,fe=/^[\u0020-\u003A\u003D-\u007E]*$/,he=Object.prototype.toString,ge=(()=>{let r=function(){};return r.prototype=Object.create(null),r})();function Te(r,e){let t=new ge,n=r.length;if(n<2)return t;let a=(e==null?void 0:e.decode)||xe,o=0;do{let i=r.indexOf("=",o);if(i===-1)break;let s=r.indexOf(";",o),c=s===-1?n:s;if(i>c){o=r.lastIndexOf(";",i-1)+1;continue}let y=V(r,o,i),p=_(r,i,y),u=r.slice(y,p);if(t[u]===void 0){let m=V(r,i+1,c),l=_(r,c,m),d=a(r.slice(m,l));t[u]=d;}o=c+1;}while(o<n);return t}function V(r,e,t){do{let n=r.charCodeAt(e);if(n!==32&&n!==9)return e}while(++e<t);return t}function _(r,e,t){for(;e>t;){let n=r.charCodeAt(--e);if(n!==32&&n!==9)return e+1}return t}function Re(r,e,t){let n=(t==null?void 0:t.encode)||encodeURIComponent;if(!pe.test(r))throw new TypeError(`argument name is invalid: ${r}`);let a=n(e);if(!me.test(a))throw new TypeError(`argument val is invalid: ${e}`);let o=r+"="+a;if(!t)return o;if(t.maxAge!==void 0){if(!Number.isInteger(t.maxAge))throw new TypeError(`option maxAge is invalid: ${t.maxAge}`);o+="; Max-Age="+t.maxAge;}if(t.domain){if(!ye.test(t.domain))throw new TypeError(`option domain is invalid: ${t.domain}`);o+="; Domain="+t.domain;}if(t.path){if(!fe.test(t.path))throw new TypeError(`option path is invalid: ${t.path}`);o+="; Path="+t.path;}if(t.expires){if(!be(t.expires)||!Number.isFinite(t.expires.valueOf()))throw new TypeError(`option expires is invalid: ${t.expires}`);o+="; Expires="+t.expires.toUTCString();}if(t.httpOnly&&(o+="; HttpOnly"),t.secure&&(o+="; Secure"),t.partitioned&&(o+="; Partitioned"),t.priority)switch(typeof t.priority=="string"?t.priority.toLowerCase():void 0){case "low":o+="; Priority=Low";break;case "medium":o+="; Priority=Medium";break;case "high":o+="; Priority=High";break;default:throw new TypeError(`option priority is invalid: ${t.priority}`)}if(t.sameSite)switch(typeof t.sameSite=="string"?t.sameSite.toLowerCase():t.sameSite){case true:case "strict":o+="; SameSite=Strict";break;case "lax":o+="; SameSite=Lax";break;case "none":o+="; SameSite=None";break;default:throw new TypeError(`option sameSite is invalid: ${t.sameSite}`)}return o}function xe(r){if(r.indexOf("%")===-1)return r;try{return decodeURIComponent(r)}catch{return r}}function be(r){return he.call(r)==="[object Date]"}});var H=$(P());var q=zod.z.object({type:zod.z.literal("QUERY"),resource:zod.z.string(),where:zod.z.record(zod.z.any()).optional(),include:zod.z.record(zod.z.any()).optional()}),L=zod.z.record(zod.z.object({value:zod.z.string().or(zod.z.number()).or(zod.z.boolean()).or(zod.z.date()),_meta:zod.z.object({timestamp:zod.z.string().optional()}).optional()})).superRefine((r,e)=>{r.id&&e.addIssue({code:zod.z.ZodIssueCode.custom,message:"Payload cannot have an id"});}),F=zod.z.object({id:zod.z.string().optional(),type:zod.z.literal("MUTATE"),resource:zod.z.string()}),S=F.extend({procedure:zod.z.string(),payload:zod.z.any().optional()}),M=F.extend({resourceId:zod.z.string(),payload:L});zod.z.union([S,M]);var Z=q.omit({type:true,resource:true}),j=S.omit({id:true,type:true,resource:true,procedure:true}),O=M.omit({id:true,type:true,resource:true});zod.z.union([O,j]);var G=r=>async e=>{var t;try{let n=typeof e.headers.getSetCookie=="function"?Object.fromEntries(e.headers):e.headers,a={headers:n,cookies:n.cookie?H.default.parse(n.cookie):{}},o=new URL(e.url),i=o.pathname.split("/"),s=o.searchParams,c=Se__default.default.parse(s.toString()),y=await((t=r.contextProvider)==null?void 0:t.call(r,{transport:"HTTP",headers:a.headers,cookies:a.cookies,query:c}))??{};if(e.method==="GET"){let p=i[i.length-1],{success:u,data:m,error:l}=Z.safeParse(c);if(!u)return Response.json({message:"Invalid query",code:"INVALID_QUERY",details:l},{status:400});let d=await r.handleRequest({req:{...a,type:"QUERY",resourceName:p,context:y,where:m.where,include:m.include,query:c}});return !d||!d.data?Response.json({message:"Invalid resource",code:"INVALID_RESOURCE"},{status:400}):Response.json(d.data)}if(e.method==="POST")try{let p=i[i.length-1],u=i[i.length-2],m=e.body?await e.json():{},l;if(p==="set"){let{success:b,data:g,error:w}=O.safeParse(m);if(!b)return Response.json({message:"Invalid mutation",code:"INVALID_REQUEST",details:w},{status:400});l=g;}else {let{success:b,data:g,error:w}=j.safeParse(m);if(!b)return Response.json({message:"Invalid mutation",code:"INVALID_REQUEST",details:w},{status:400});l=g;}let d=await r.handleRequest({req:{...a,type:"MUTATE",resourceName:u,input:l.payload,context:y,resourceId:l.resourceId,procedure:p!=="set"?p:void 0,query:{}}});return Response.json(d)}catch(p){return console.error("Error parsing mutation from the client:",p),Response.json({message:"Internal server error",code:"INTERNAL_SERVER_ERROR"},{status:500})}return Response.json({message:"Not found",code:"NOT_FOUND"},{status:404})}catch(n){return console.error("Unexpected error:",n),Response.json({message:"Internal server error",code:"INTERNAL_SERVER_ERROR"},{status:500})}};var X=$(P());var T=zod.z.string(),Me=zod.z.object({id:T,type:zod.z.literal("SUBSCRIBE"),resource:zod.z.string()}),ve=zod.z.object({id:T,type:zod.z.literal("SYNC"),lastSyncedAt:zod.z.string().optional(),resources:zod.z.string().array().optional(),where:zod.z.record(zod.z.any()).optional()}),B=M.extend({id:T}),Ie=S.extend({id:T}),Ee=zod.z.union([Ie,B]),Q=zod.z.union([Me,ve,Ee]),Ae=zod.z.object({id:T,type:zod.z.literal("SYNC"),resource:zod.z.string(),data:zod.z.record(L)}),Pe=zod.z.object({id:T,type:zod.z.literal("REJECT"),resource:zod.z.string(),message:zod.z.string().optional()}),Le=zod.z.object({id:T,type:zod.z.literal("REPLY"),data:zod.z.any()});zod.z.union([Ae,Pe,Le,B]);var K="0123456789ABCDEFGHJKMNPQRSTVWXYZ",v=32;var je=16,Y=10,W=0xffffffffffff;var R;(function(r){r.Base32IncorrectEncoding="B32_ENC_INVALID",r.DecodeTimeInvalidCharacter="DEC_TIME_CHAR",r.DecodeTimeValueMalformed="DEC_TIME_MALFORMED",r.EncodeTimeNegative="ENC_TIME_NEG",r.EncodeTimeSizeExceeded="ENC_TIME_SIZE_EXCEED",r.EncodeTimeValueMalformed="ENC_TIME_MALFORMED",r.PRNGDetectFailure="PRNG_DETECT",r.ULIDInvalid="ULID_INVALID",r.Unexpected="UNEXPECTED",r.UUIDInvalid="UUID_INVALID";})(R||(R={}));var x=class extends Error{constructor(e,t){super(`${t} (${e})`),this.name="ULIDError",this.code=e;}};function Oe(r){let e=Math.floor(r()*v);return e===v&&(e=v-1),K.charAt(e)}function Ne(r){var n;let e=Ce(),t=e&&(e.crypto||e.msCrypto)||(typeof E__default.default<"u"?E__default.default:null);if(typeof(t==null?void 0:t.getRandomValues)=="function")return ()=>{let a=new Uint8Array(1);return t.getRandomValues(a),a[0]/255};if(typeof(t==null?void 0:t.randomBytes)=="function")return ()=>t.randomBytes(1).readUInt8()/255;if((n=E__default.default)!=null&&n.randomBytes)return ()=>E__default.default.randomBytes(1).readUInt8()/255;throw new x(R.PRNGDetectFailure,"Failed to find a reliable PRNG")}function Ce(){return De()?self:typeof window<"u"?window:typeof global<"u"?global:typeof globalThis<"u"?globalThis:null}function ze(r,e){let t="";for(;r>0;r--)t=Oe(e)+t;return t}function ke(r,e=Y){if(isNaN(r))throw new x(R.EncodeTimeValueMalformed,`Time must be a number: ${r}`);if(r>W)throw new x(R.EncodeTimeSizeExceeded,`Cannot encode a time larger than ${W}: ${r}`);if(r<0)throw new x(R.EncodeTimeNegative,`Time must be positive: ${r}`);if(Number.isInteger(r)===false)throw new x(R.EncodeTimeValueMalformed,`Time must be an integer: ${r}`);let t,n="";for(let a=e;a>0;a--)t=r%v,n=K.charAt(t)+n,r=(r-t)/v;return n}function De(){return typeof WorkerGlobalScope<"u"&&self instanceof WorkerGlobalScope}function J(r,e){let t=Ne(),n=Date.now();return ke(n,Y)+ze(je,t)}var N=()=>J().toLowerCase();var ee=r=>{let e={},t={};return r.subscribeToMutations(n=>{let a=n;!a.resourceId||!a.payload||(console.log("Mutation propagated:",a),Object.entries(t[a.resource]??{}).forEach(([o,i])=>{var s;(s=e[o])==null||s.send(JSON.stringify({...a,id:a.id??N()}));}));}),(n,a)=>{var p;let o=u=>{n.send(JSON.stringify(u));},i=N(),s={headers:a.headers,cookies:typeof a.headers.cookie=="string"?X.default.parse(a.headers.cookie):{}},c=Se.parse(a.url.split("?")[1]),y=(p=r.contextProvider)==null?void 0:p.call(r,{transport:"WEBSOCKET",headers:s.headers,cookies:s.cookies,query:c});e[i]=n,console.log("Client connected:",i),n.on("message",async u=>{try{console.log("Message received from the client:",u);let m=Q.parse(JSON.parse(u.toString()));if(m.type==="SUBSCRIBE"){let{resource:l}=m;t[l]||(t[l]={}),t[l][i]={};}else if(m.type==="SYNC"){let{resources:l}=m,d=l??Object.keys(r.schema);console.log("Syncing resources:",d),await Promise.all(d.map(async b=>{let g=await r.handleRequest({req:{...s,type:"QUERY",resourceName:b,context:await y??{},query:c}});if(!g||!g.data)throw new Error("Invalid resource");o({id:m.id,type:"SYNC",resource:b,data:Object.fromEntries(Object.entries(g.data??{}).map(([w,ae])=>[w,ae.value]))});}));}else if(m.type==="MUTATE"){let{resource:l}=m;console.log("Received mutation from client:",m);try{let d=await r.handleRequest({req:{...s,type:"MUTATE",resourceName:l,input:m.payload,context:{messageId:m.id,...await y??{}},resourceId:m.resourceId,procedure:m.procedure,query:c}});m.procedure&&o({id:m.id,type:"REPLY",data:d});}catch(d){o({id:m.id,type:"REJECT",resource:l,message:d.message}),console.error("Error parsing mutation from the client:",d);}}}catch(m){console.error("Error handling message from the client:",m);}}),n.on("close",()=>{console.log("Connection closed",i),delete e[i];for(let u of Object.values(t))delete u[i];});}};function te(r){let e=`${r.protocol}://${r.hostname}${r.url}`,t=new Headers;return Object.entries(r.headers).forEach(([n,a])=>{a&&t.set(n,Array.isArray(a)?a.join(","):a);}),new Request(e,{method:r.method,headers:t,body:r.body&&r.method!=="GET"?JSON.stringify(r.body):void 0})}var bt=(r,e,t)=>{r.ws(`${(t==null?void 0:t.basePath)??""}/ws`,ee(e)),r.use(`${(t==null?void 0:t.basePath)??""}/`,(n,a)=>{G(e)(te(n)).then(i=>i.json().then(s=>a.status(i.status).send(s)));});};var C=class r{routes;constructor(e){this.routes=e.routes;}static create(e){return new r(e)}},vt=r=>C.create({...r}),Ve=r=>({handler:e=>({inputValidator:r??zod.z.undefined(),handler:e})}),z=class r{_resourceSchema;resourceName;middlewares;customMutations;constructor(e,t){this.resourceName=e,this.middlewares=new Set,this.customMutations=t??{};}handleFind=async({req:e,db:t})=>({data:await t.find(e.resourceName,e.where,e.include),acceptedValues:null});handleSet=async({req:e,db:t,schema:n})=>{if(!e.input)throw new Error("Payload is required");if(!e.resourceId)throw new Error("ResourceId is required");let a=await t.findById(e.resourceName,e.resourceId),[o,i]=n[this.resourceName].mergeMutation("set",e.input,a);if(!i)throw new Error("Mutation rejected");return {data:await t.upsert(e.resourceName,e.resourceId,o),acceptedValues:i}};async handleRequest(e){let t=n=>(()=>{if(n.type==="QUERY")return this.handleFind({req:n,db:e.db,schema:e.schema});if(n.type==="MUTATE")if(n.procedure){if(this.customMutations[n.procedure]){let a=this.customMutations[n.procedure].inputValidator.parse(n.input);return n.input=a,this.customMutations[n.procedure].handler({req:n,db:e.db,schema:e.schema})}}else return this.handleSet({req:n,db:e.db,schema:e.schema});throw new Error("Invalid request")})();return await Array.from(this.middlewares.values()).reduceRight((n,a)=>o=>a({req:o,next:n}),async n=>t(n))(e.req)}use(...e){for(let t of e)this.middlewares.add(t);return this}withMutations(e){return new r(this.resourceName,e({mutation:Ve}))}},k=class r{middlewares;constructor(e=[]){this.middlewares=e;}createBasicRoute(e){return new z(e.name).use(...this.middlewares)}use(...e){return new r([...this.middlewares,...e])}static create(){return new r}},It=k.create;var A=class{},re=class extends A{storage={};async updateSchema(e){this.storage=Object.entries(e).reduce((t,[n,a])=>(t[a.name]={},t),{});}async findById(e,t){var n;return (n=this.storage[e])==null?void 0:n[t]}async find(e,t,n){return this.storage[e]??{}}async upsert(e,t,n){return this.storage[e]??={},this.storage[e][t]=n,n}},ne=class extends A{db;schema;constructor(e){super(),this.db=new kysely.Kysely({dialect:new kysely.PostgresDialect({pool:e})});}async updateSchema(e){this.schema=e;let t=await this.db.introspection.getTables();for(let[n,a]of Object.entries(e)){let o=t.find(c=>c.name===n);o||await this.db.schema.createTable(n).ifNotExists().execute();let i=`${n}_meta`,s=t.find(c=>c.name===i);s||await this.db.schema.createTable(i).ifNotExists().execute();for(let[c,y]of Object.entries(a.fields)){let p=o==null?void 0:o.columns.find(l=>l.name===c),u=y.getStorageFieldType();p?p.dataType!==u.type&&console.error("Column type mismatch:",c,"expected to have type:",u.type,"but has type:",p.dataType):(await this.db.schema.alterTable(n).addColumn(c,u.type,l=>{let d=l;return u.unique&&(d=d.unique()),u.nullable||(d=d.notNull()),u.references&&(d=d.references(u.references)),u.primary&&(d=d.primaryKey()),u.default!==void 0&&(d=d.defaultTo(u.default)),d}).execute().catch(l=>{throw console.error("Error adding column",c,l),l}),u.index&&await this.db.schema.createIndex(`${n}_${c}_index`).on(n).column(c).execute().catch(l=>{})),(s==null?void 0:s.columns.find(l=>l.name===c))||await this.db.schema.alterTable(i).addColumn(c,"varchar",l=>{let d=l;return u.primary&&(d=d.primaryKey().references(`${n}.${c}`)),d}).execute();}}}async findById(e,t){let n=await this.db.selectFrom(e).where("id","=",t).selectAll(e).executeTakeFirst(),a=await this.db.selectFrom(`${e}_meta`).where("id","=",t).selectAll(`${e}_meta`).executeTakeFirst();if(!(!n||!a))return this.convertToMaterializedLiveType(n,a)}async find(e,t,n){let a=this.applyWhere(e,this.db.selectFrom(e).selectAll(e),t);a=this.applyInclude(e,a,n);let o=await a.execute(),i=Object.fromEntries(o.map(y=>{let{id:p,...u}=y;return [p,u]}));if(Object.keys(i).length===0)return {};let s=Object.fromEntries((await this.db.selectFrom(`${e}_meta`).selectAll().where("id","in",Object.keys(i)).execute()).map(y=>{let{id:p,...u}=y;return [p,u]}));return Object.entries(i).reduce((y,[p,u])=>(s[p]&&(y[p]=this.convertToMaterializedLiveType(u,s[p])),y),{})}async upsert(e,t,n){return await this.db.transaction().execute(async a=>{let o=!!await a.selectFrom(e).select("id").where("id","=",t).executeTakeFirst(),i={},s={};for(let[c,y]of Object.entries(n.value))i[c]=y.value,s[c]=y._meta.timestamp;o?await Promise.all([a.updateTable(e).set(i).where("id","=",t).execute(),a.updateTable(`${e}_meta`).set(s).where("id","=",t).execute()]):await Promise.all([a.insertInto(e).values({...i,id:t}).execute(),a.insertInto(`${e}_meta`).values({...s,id:t}).execute()]);}),n}convertToMaterializedLiveType(e,t){return {value:Object.fromEntries(Object.entries(e).flatMap(([n,a])=>t?[[n,{value:a,_meta:{timestamp:t==null?void 0:t[n]}}]]:[]))}}applyWhere(e,t,n){if(!n)return t;if(!this.schema)throw new Error("Schema not initialized");let a=this.schema[e];if(!a)throw new Error("Resource not found");for(let[o,i]of Object.entries(n))if(a.fields[o])t=t.where(`${e}.${o}`,"=",i);else if(a.relations[o]){let s=a.relations[o],c=s.entity.name,y=s.type==="one"?"id":s.foreignColumn,p=s.type==="one"?s.relationalColumn:"id";t=t.leftJoin(c,`${c}.${y}`,`${e}.${p}`),t=this.applyWhere(c,t,i);}return t}applyInclude(e,t,n){if(!n)return t;if(!this.schema)throw new Error("Schema not initialized");let a=this.schema[e];if(!a)throw new Error(`Resource not found: ${e}`);for(let[o,i]of Object.entries(n)){if(!a.relations[o])throw new Error(`Relation ${o} not found in resource ${e}`);let s=a.relations[o],c=s.entity.name,y=s.type==="one"?"id":s.foreignColumn,p=s.type==="one"?s.relationalColumn:"id";t=t.select(u=>postgres.jsonArrayFrom(u.selectFrom(c).selectAll(c).whereRef(`${c}.${y}`,"=",`${e}.${p}`)).as(o));}return t}};var D=class r{router;storage;schema;middlewares=new Set;contextProvider;mutationSubscriptions=new Set;constructor(e){var t;this.router=e.router,this.storage=e.storage,this.schema=e.schema,(t=e.middlewares)==null||t.forEach(n=>this.middlewares.add(n)),this.storage.updateSchema(this.schema),this.contextProvider=e.contextProvider;}static create(e){return new r(e)}subscribeToMutations(e){return this.mutationSubscriptions.add(e),()=>{this.mutationSubscriptions.delete(e);}}async handleRequest(e){if(!this.router.routes[e.req.resourceName])throw new Error("Invalid resource");let t=await Array.from(this.middlewares.values()).reduceRight((n,a)=>o=>a({req:o,next:n}),async n=>this.router.routes[e.req.resourceName].handleRequest({req:n,db:this.storage,schema:this.schema}))(e.req);return t&&e.req.type==="MUTATE"&&t.acceptedValues&&Object.keys(t.acceptedValues).length>0&&this.mutationSubscriptions.forEach(n=>{n({id:e.req.context.messageId,type:"MUTATE",resource:e.req.resourceName,payload:t.acceptedValues??{},resourceId:e.req.resourceId});}),t}use(e){return this.middlewares.add(e),this}context(e){return this.contextProvider=e,this}},Nt=D.create;
2
- exports.InMemoryStorage=re;exports.Route=z;exports.RouteFactory=k;exports.Router=C;exports.SQLStorage=ne;exports.Server=D;exports.Storage=A;exports.expressAdapter=bt;exports.routeFactory=It;exports.router=vt;exports.server=Nt;
1
+ 'use strict';var je=require('qs'),zod=require('zod'),j=require('crypto'),kysely=require('kysely'),postgres=require('kysely/helpers/postgres');function _interopDefault(e){return e&&e.__esModule?e:{default:e}}var je__default=/*#__PURE__*/_interopDefault(je);var j__default=/*#__PURE__*/_interopDefault(j);var pe=Object.create;var Q=Object.defineProperty;var me=Object.getOwnPropertyDescriptor;var Te=Object.getOwnPropertyNames;var fe=Object.getPrototypeOf,he=Object.prototype.hasOwnProperty;var ge=(n,e)=>()=>(e||n((e={exports:{}}).exports,e),e.exports);var Re=(n,e,t,r)=>{if(e&&typeof e=="object"||typeof e=="function")for(let a of Te(e))!he.call(n,a)&&a!==t&&Q(n,a,{get:()=>e[a],enumerable:!(r=me(e,a))||r.enumerable});return n};var B=(n,e,t)=>(t=n!=null?pe(fe(n)):{},Re(Q(t,"default",{value:n,enumerable:true}),n));var P=ge(E=>{Object.defineProperty(E,"__esModule",{value:true});E.parse=Ie;E.serialize=Le;var ve=/^[\u0021-\u003A\u003C\u003E-\u007E]+$/,xe=/^[\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,we=/^[\u0020-\u003A\u003D-\u007E]*$/,Me=Object.prototype.toString,Se=(()=>{let n=function(){};return n.prototype=Object.create(null),n})();function Ie(n,e){let t=new Se,r=n.length;if(r<2)return t;let a=(e==null?void 0:e.decode)||Ae,i=0;do{let o=n.indexOf("=",i);if(o===-1)break;let c=n.indexOf(";",i),s=c===-1?r:c;if(o>s){i=n.lastIndexOf(";",o-1)+1;continue}let T=J(n,i,o),m=Y(n,o,T),d=n.slice(T,m);if(t[d]===void 0){let y=J(n,o+1,s),u=Y(n,s,y),l=a(n.slice(y,u));t[d]=l;}i=s+1;}while(i<r);return t}function J(n,e,t){do{let r=n.charCodeAt(e);if(r!==32&&r!==9)return e}while(++e<t);return t}function Y(n,e,t){for(;e>t;){let r=n.charCodeAt(--e);if(r!==32&&r!==9)return e+1}return t}function Le(n,e,t){let r=(t==null?void 0:t.encode)||encodeURIComponent;if(!ve.test(n))throw new TypeError(`argument name is invalid: ${n}`);let a=r(e);if(!xe.test(a))throw new TypeError(`argument val is invalid: ${e}`);let i=n+"="+a;if(!t)return i;if(t.maxAge!==void 0){if(!Number.isInteger(t.maxAge))throw new TypeError(`option maxAge is invalid: ${t.maxAge}`);i+="; Max-Age="+t.maxAge;}if(t.domain){if(!be.test(t.domain))throw new TypeError(`option domain is invalid: ${t.domain}`);i+="; Domain="+t.domain;}if(t.path){if(!we.test(t.path))throw new TypeError(`option path is invalid: ${t.path}`);i+="; Path="+t.path;}if(t.expires){if(!Ee(t.expires)||!Number.isFinite(t.expires.valueOf()))throw new TypeError(`option expires is invalid: ${t.expires}`);i+="; Expires="+t.expires.toUTCString();}if(t.httpOnly&&(i+="; HttpOnly"),t.secure&&(i+="; Secure"),t.partitioned&&(i+="; Partitioned"),t.priority)switch(typeof t.priority=="string"?t.priority.toLowerCase():void 0){case "low":i+="; Priority=Low";break;case "medium":i+="; Priority=Medium";break;case "high":i+="; 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":i+="; SameSite=Strict";break;case "lax":i+="; SameSite=Lax";break;case "none":i+="; SameSite=None";break;default:throw new TypeError(`option sameSite is invalid: ${t.sameSite}`)}return i}function Ae(n){if(n.indexOf("%")===-1)return n;try{return decodeURIComponent(n)}catch{return n}}function Ee(n){return Me.call(n)==="[object Date]"}});var te=B(P());var O=zod.z.object({resource:zod.z.string(),where:zod.z.record(zod.z.any()).optional(),include:zod.z.record(zod.z.any()).optional(),lastSyncedAt:zod.z.string().optional(),limit:zod.z.number().optional(),sort:zod.z.array(zod.z.object({key:zod.z.string(),direction:zod.z.enum(["asc","desc"])})).optional()}),V=zod.z.record(zod.z.object({value:zod.z.string().or(zod.z.number()).or(zod.z.boolean()).or(zod.z.date()).nullable(),_meta:zod.z.object({timestamp:zod.z.string().optional().nullable()}).optional()})).superRefine((n,e)=>{n.id&&e.addIssue({code:zod.z.ZodIssueCode.custom,message:"Payload cannot have an id"});}),X=zod.z.object({id:zod.z.string().optional(),type:zod.z.literal("MUTATE"),resource:zod.z.string()}),S=X.extend({procedure:zod.z.string(),payload:zod.z.any().optional()}),I=X.extend({resourceId:zod.z.string(),payload:V});zod.z.union([S,I]);var ee=O.omit({resource:true}),N=S.omit({id:true,type:true,resource:true,procedure:true}),F=I.omit({id:true,type:true,resource:true});zod.z.union([F,N]);var ne=n=>async e=>{var t;try{let r=typeof e.headers.getSetCookie=="function"?Object.fromEntries(e.headers):e.headers,a={headers:r,cookies:r.cookie?te.default.parse(r.cookie):{}},i=new URL(e.url),o=i.pathname.split("/"),c=i.searchParams,s=je__default.default.parse(c.toString()),T=await((t=n.contextProvider)==null?void 0:t.call(n,{transport:"HTTP",headers:a.headers,cookies:a.cookies,query:s}))??{};if(e.method==="GET"){let m=o[o.length-1],{success:d,data:y,error:u}=ee.safeParse(s);if(!d)return Response.json({message:"Invalid query",code:"INVALID_QUERY",details:u},{status:400});let l=await n.handleRequest({req:{...a,type:"QUERY",resourceName:m,context:T,where:y.where,include:y.include,query:s}});return !l||!l.data?Response.json({message:"Invalid resource",code:"INVALID_RESOURCE"},{status:400}):Response.json(l.data)}if(e.method==="POST")try{let m=o[o.length-1],d=o[o.length-2],y=e.body?await e.json():{},u;if(m==="set"){let{success:w,data:M,error:_}=F.safeParse(y);if(!w)return Response.json({message:"Invalid mutation",code:"INVALID_REQUEST",details:_},{status:400});u=M;}else {let{success:w,data:M,error:_}=N.safeParse(y);if(!w)return Response.json({message:"Invalid mutation",code:"INVALID_REQUEST",details:_},{status:400});u=M;}let l=await n.handleRequest({req:{...a,type:"MUTATE",resourceName:d,input:u.payload,context:T,resourceId:u.resourceId,procedure:m!=="set"?m:void 0,query:{}}});return Response.json(l)}catch(m){return console.error("Error parsing mutation from the client:",m),Response.json({message:"Internal server error",code:"INTERNAL_SERVER_ERROR"},{status:500})}return Response.json({message:"Not found",code:"NOT_FOUND"},{status:404})}catch(r){return console.error("Unexpected error:",r),Response.json({message:"Internal server error",code:"INTERNAL_SERVER_ERROR"},{status:500})}};var ue=B(P());var x=zod.z.string(),Ce=zod.z.object({id:x,type:zod.z.literal("SUBSCRIBE"),resource:zod.z.string()}),_e=O.extend({id:x,type:zod.z.literal("QUERY")}),re=I.extend({id:x}),Pe=S.extend({id:x}),Ve=zod.z.union([Pe,re]),ae=zod.z.union([Ce,_e,Ve]),Ne=zod.z.object({id:x,type:zod.z.literal("REJECT"),resource:zod.z.string(),message:zod.z.string().optional()}),Fe=zod.z.object({id:x,type:zod.z.literal("REPLY"),data:zod.z.any()});zod.z.union([Ne,Fe,re]);zod.z.object({resource:zod.z.string(),data:zod.z.record(V)});var oe="0123456789ABCDEFGHJKMNPQRSTVWXYZ",L=32;var $e=16,se=10,ie=0xffffffffffff;var h;(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";})(h||(h={}));var g=class extends Error{constructor(e,t){super(`${t} (${e})`),this.name="ULIDError",this.code=e;}};function Ke(n){let e=Math.floor(n()*L);return e===L&&(e=L-1),oe.charAt(e)}function De(n){var r;let e=qe(),t=e&&(e.crypto||e.msCrypto)||(typeof j__default.default<"u"?j__default.default:null);if(typeof(t==null?void 0:t.getRandomValues)=="function")return ()=>{let a=new Uint8Array(1);return t.getRandomValues(a),a[0]/255};if(typeof(t==null?void 0:t.randomBytes)=="function")return ()=>t.randomBytes(1).readUInt8()/255;if((r=j__default.default)!=null&&r.randomBytes)return ()=>j__default.default.randomBytes(1).readUInt8()/255;throw new g(h.PRNGDetectFailure,"Failed to find a reliable PRNG")}function qe(){return ze()?self:typeof window<"u"?window:typeof global<"u"?global:typeof globalThis<"u"?globalThis:null}function Ue(n,e){let t="";for(;n>0;n--)t=Ke(e)+t;return t}function ke(n,e=se){if(isNaN(n))throw new g(h.EncodeTimeValueMalformed,`Time must be a number: ${n}`);if(n>ie)throw new g(h.EncodeTimeSizeExceeded,`Cannot encode a time larger than ${ie}: ${n}`);if(n<0)throw new g(h.EncodeTimeNegative,`Time must be positive: ${n}`);if(Number.isInteger(n)===false)throw new g(h.EncodeTimeValueMalformed,`Time must be an integer: ${n}`);let t,r="";for(let a=e;a>0;a--)t=n%L,r=oe.charAt(t)+r,n=(n-t)/L;return r}function ze(){return typeof WorkerGlobalScope<"u"&&self instanceof WorkerGlobalScope}function ce(n,e){let t=De(),r=Date.now();return ke(r,se)+Ue($e,t)}var $=()=>ce().toLowerCase();var le=n=>{let e={},t={};return n.subscribeToMutations(r=>{let a=r;!a.resourceId||!a.payload||(console.log("Mutation propagated:",a),Object.entries(t[a.resource]??{}).forEach(([i,o])=>{var c;(c=e[i])==null||c.send(JSON.stringify({...a,id:a.id??$()}));}));}),(r,a)=>{var m;let i=d=>{r.send(JSON.stringify(d));},o=$(),c={headers:a.headers,cookies:typeof a.headers.cookie=="string"?ue.default.parse(a.headers.cookie):{}},s=je.parse(a.url.split("?")[1]),T=(m=n.contextProvider)==null?void 0:m.call(n,{transport:"WEBSOCKET",headers:c.headers,cookies:c.cookies,query:s});e[o]=r,console.log("Client connected:",o),r.on("message",async d=>{try{console.log("Message received from the client:",d);let y=ae.parse(JSON.parse(d.toString()));if(y.type==="SUBSCRIBE"){let{resource:u}=y;t[u]||(t[u]={}),t[u][o]={};}else if(y.type==="QUERY"){let{resource:u}=y,l=await n.handleRequest({req:{...c,type:"QUERY",resourceName:u,context:await T??{},query:s}});if(!l||!l.data)throw new Error("Invalid resource");i({id:y.id,type:"REPLY",data:{resource:u,data:Object.fromEntries(Object.entries(l.data??{}).map(([w,M])=>[w,M.value]))}});}else if(y.type==="MUTATE"){let{resource:u}=y;console.log("Received mutation from client:",y);try{let l=await n.handleRequest({req:{...c,type:"MUTATE",resourceName:u,input:y.payload,context:{messageId:y.id,...await T??{}},resourceId:y.resourceId,procedure:y.procedure,query:s}});y.procedure&&i({id:y.id,type:"REPLY",data:l});}catch(l){i({id:y.id,type:"REJECT",resource:u,message:l.message}),console.error("Error parsing mutation from the client:",l);}}}catch(y){console.error("Error handling message from the client:",y);}}),r.on("close",()=>{console.log("Connection closed",o),delete e[o];for(let d of Object.values(t))delete d[o];});}};function de(n){let e=`${n.protocol}://${n.hostname}${n.url}`,t=new Headers;return Object.entries(n.headers).forEach(([r,a])=>{a&&t.set(r,Array.isArray(a)?a.join(","):a);}),new Request(e,{method:n.method,headers:t,body:n.body&&n.method!=="GET"?JSON.stringify(n.body):void 0})}var It=(n,e,t)=>{n.ws(`${(t==null?void 0:t.basePath)??""}/ws`,le(e)),n.use(`${(t==null?void 0:t.basePath)??""}/`,(r,a)=>{ne(e)(de(r)).then(o=>o.json().then(c=>a.status(o.status).send(c)));});};var K=class n{routes;constructor(e){this.routes=e.routes;}static create(e){return new n(e)}},Et=n=>K.create({...n}),He=n=>({handler:e=>({inputValidator:n??zod.z.undefined(),handler:e})}),D=class n{_resourceSchema;resourceName;middlewares;customMutations;constructor(e,t){this.resourceName=e,this.middlewares=new Set,this.customMutations=t??{};}handleFind=async({req:e,db:t})=>({data:await t.rawFind(e.resourceName,e.where,e.include),acceptedValues:null});handleSet=async({req:e,db:t,schema:r})=>{if(!e.input)throw new Error("Payload is required");if(!e.resourceId)throw new Error("ResourceId is required");let a=await t.rawFindById(e.resourceName,e.resourceId),[i,o]=r[this.resourceName].mergeMutation("set",e.input,a);if(!o)throw new Error("Mutation rejected");return {data:await t.rawUpsert(e.resourceName,e.resourceId,i),acceptedValues:o}};async handleRequest(e){let t=r=>(()=>{if(r.type==="QUERY")return this.handleFind({req:r,db:e.db,schema:e.schema});if(r.type==="MUTATE")if(r.procedure){if(this.customMutations[r.procedure]){let a=this.customMutations[r.procedure].inputValidator.parse(r.input);return r.input=a,this.customMutations[r.procedure].handler({req:r,db:e.db,schema:e.schema})}}else return this.handleSet({req:r,db:e.db,schema:e.schema});throw new Error("Invalid request")})();return await Array.from(this.middlewares.values()).reduceRight((r,a)=>i=>a({req:i,next:r}),async r=>t(r))(e.req)}use(...e){for(let t of e)this.middlewares.add(t);return this}withMutations(e){return new n(this.resourceName,e({mutation:He}))}},q=class n{middlewares;constructor(e=[]){this.middlewares=e;}createBasicRoute(e){return new D(e.name).use(...this.middlewares)}use(...e){return new n([...this.middlewares,...e])}static create(){return new n}},Ot=q.create;var v=n=>{if(n)return Array.isArray(n.value)?n.value.map(e=>v(e)):typeof n.value!="object"||n.value===null||n.value instanceof Date?n.value:Object.fromEntries(Object.entries(n.value).map(([e,t])=>[e,v(t)]))};var H=class{async insert(e,t){let r=new Date().toISOString();return v(await this.rawUpsert(e.name,t.id,{value:Object.fromEntries(Object.entries(t).map(([a,i])=>[a,{value:i,_meta:{timestamp:r}}]))}))}async update(e,t,r){let a=new Date().toISOString(),{id:i,...o}=r;return v(await this.rawUpsert(e.name,t,{value:Object.fromEntries(Object.entries(o).map(([c,s])=>[c,{value:s,_meta:{timestamp:a}}]))}))}},ye=class extends H{db;schema;constructor(e){super(),this.db=new kysely.Kysely({dialect:new kysely.PostgresDialect({pool:e})});}async updateSchema(e){this.schema=e;let t=await this.db.introspection.getTables();for(let[r,a]of Object.entries(e)){let i=t.find(s=>s.name===r);i||await this.db.schema.createTable(r).ifNotExists().execute();let o=`${r}_meta`,c=t.find(s=>s.name===o);c||await this.db.schema.createTable(o).ifNotExists().execute();for(let[s,T]of Object.entries(a.fields)){let m=i==null?void 0:i.columns.find(u=>u.name===s),d=T.getStorageFieldType();m?m.dataType!==d.type&&console.error("Column type mismatch:",s,"expected to have type:",d.type,"but has type:",m.dataType):(await this.db.schema.alterTable(r).addColumn(s,d.type,u=>{let l=u;return d.unique&&(l=l.unique()),d.nullable||(l=l.notNull()),d.references&&(l=l.references(d.references)),d.primary&&(l=l.primaryKey()),d.default!==void 0&&(l=l.defaultTo(d.default)),l}).execute().catch(u=>{throw console.error("Error adding column",s,u),u}),d.index&&await this.db.schema.createIndex(`${r}_${s}_index`).on(r).column(s).execute().catch(u=>{})),(c==null?void 0:c.columns.find(u=>u.name===s))||await this.db.schema.alterTable(o).addColumn(s,"varchar",u=>{let l=u;return d.primary&&(l=l.primaryKey().references(`${r}.${s}`)),l}).execute();}}}async rawFindById(e,t,r){let a=await this.db.selectFrom(e).where("id","=",t).selectAll(e).select(o=>postgres.jsonObjectFrom(o.selectFrom(`${e}_meta`).selectAll(`${e}_meta`).whereRef(`${e}_meta.id`,"=",`${e}.id`)).as("_meta"));a=this.applyInclude(e,a,r);let i=await a.executeTakeFirst();if(i)return this.convertToMaterializedLiveType(i)}async findOne(e,t,r){let a=await this.rawFindById(e.name,t,r==null?void 0:r.include);if(a)return v(a)}async rawFind(e,t,r){let a=this.db.selectFrom(e).selectAll(e).select(s=>postgres.jsonObjectFrom(s.selectFrom(`${e}_meta`).selectAll(`${e}_meta`).whereRef(`${e}_meta.id`,"=",`${e}.id`)).as("_meta"));a=this.applyWhere(e,a,t),a=this.applyInclude(e,a,r);let i=await a.execute(),o=Object.fromEntries(i.map(s=>{let{id:T,...m}=s;return [T,m]}));return Object.keys(o).length===0?{}:Object.entries(o).reduce((s,[T,m])=>(s[T]=this.convertToMaterializedLiveType(m),s),{})}async find(e,t){let r=await this.rawFind(e.name,t==null?void 0:t.where,t==null?void 0:t.include);return Object.fromEntries(Object.entries(r).map(([a,i])=>[a,v(i)]))}async rawUpsert(e,t,r){return await this.db.transaction().execute(async a=>{var s;let i=!!await a.selectFrom(e).select("id").where("id","=",t).executeTakeFirst(),o={},c={};for(let[T,m]of Object.entries(r.value)){let d=(s=m._meta)==null?void 0:s.timestamp;d&&(o[T]=m.value,c[T]=d);}i?await Promise.all([a.updateTable(e).set(o).where("id","=",t).execute(),a.updateTable(`${e}_meta`).set(c).where("id","=",t).execute()]):await Promise.all([a.insertInto(e).values({...o,id:t}).execute(),a.insertInto(`${e}_meta`).values({...c,id:t}).execute()]);}),r}convertToMaterializedLiveType(e){if(!e._meta)throw new Error("Missing _meta");return {value:Object.entries(e).reduce((t,[r,a])=>{var i,o,c;return r==="_meta"||(r==="id"?t[r]={value:a}:Array.isArray(a)?t[r]={value:a.map(s=>this.convertToMaterializedLiveType(s)),_meta:{timestamp:(i=e==null?void 0:e._meta)==null?void 0:i[r]}}:typeof a=="object"&&a!==null&&!(a instanceof Date)?t[r]={...this.convertToMaterializedLiveType(a),_meta:{timestamp:(o=e==null?void 0:e._meta)==null?void 0:o[r]}}:t[r]={value:a,_meta:{timestamp:(c=e==null?void 0:e._meta)==null?void 0:c[r]}}),t},{})}}applyWhere(e,t,r){if(!r)return t;if(!this.schema)throw new Error("Schema not initialized");let a=this.schema[e];if(!a)throw new Error("Resource not found");for(let[i,o]of Object.entries(r))if(a.fields[i])t=t.where(`${e}.${i}`,"=",o);else if(a.relations[i]){let c=a.relations[i],s=c.entity.name,T=c.type==="one"?"id":c.foreignColumn,m=c.type==="one"?c.relationalColumn:"id";t=t.leftJoin(s,`${s}.${T}`,`${e}.${m}`),t=this.applyWhere(s,t,o);}return t}applyInclude(e,t,r){if(!r)return t;if(!this.schema)throw new Error("Schema not initialized");let a=this.schema[e];if(!a)throw new Error(`Resource not found: ${e}`);for(let[i,o]of Object.entries(r)){if(!a.relations[i])throw new Error(`Relation ${i} not found in resource ${e}`);let c=a.relations[i],s=c.entity.name,T=c.type==="one"?"id":c.foreignColumn,m=c.type==="one"?c.relationalColumn:"id",d=c.type==="one"?postgres.jsonObjectFrom:postgres.jsonArrayFrom;t=t.select(y=>d(y.selectFrom(s).selectAll(s).whereRef(`${s}.${T}`,"=",`${e}.${m}`).select(u=>postgres.jsonObjectFrom(u.selectFrom(`${s}_meta`).selectAll(`${s}_meta`).whereRef(`${s}_meta.id`,"=",`${s}.id`)).as("_meta"))).as(i));}return t}};var G=class n{router;storage;schema;middlewares=new Set;contextProvider;mutationSubscriptions=new Set;constructor(e){var t;this.router=e.router,this.storage=e.storage,this.schema=e.schema,(t=e.middlewares)==null||t.forEach(r=>{this.middlewares.add(r);}),this.storage.updateSchema(this.schema),this.contextProvider=e.contextProvider;}static create(e){return new n(e)}subscribeToMutations(e){return this.mutationSubscriptions.add(e),()=>{this.mutationSubscriptions.delete(e);}}async handleRequest(e){if(!this.router.routes[e.req.resourceName])throw new Error("Invalid resource");let t=await Array.from(this.middlewares.values()).reduceRight((r,a)=>i=>a({req:i,next:r}),async r=>this.router.routes[e.req.resourceName].handleRequest({req:r,db:this.storage,schema:this.schema}))(e.req);return t&&e.req.type==="MUTATE"&&t.acceptedValues&&Object.keys(t.acceptedValues).length>0&&this.mutationSubscriptions.forEach(r=>{r({id:e.req.context.messageId,type:"MUTATE",resource:e.req.resourceName,payload:t.acceptedValues??{},resourceId:e.req.resourceId});}),t}use(e){return this.middlewares.add(e),this}context(e){return this.contextProvider=e,this}},Bt=G.create;
2
+ exports.Route=D;exports.RouteFactory=q;exports.Router=K;exports.SQLStorage=ye;exports.Server=G;exports.Storage=H;exports.expressAdapter=It;exports.routeFactory=Ot;exports.router=Et;exports.server=Bt;
package/dist/server.d.cts CHANGED
@@ -1,5 +1,5 @@
1
1
  import { z, ZodTypeAny } from 'zod';
2
- import { LiveObjectAny, Schema, MaterializedLiveType, WhereClause, IncludeClause } from './index.cjs';
2
+ import { LiveObjectAny, Schema, MaterializedLiveType, IncludeClause, InferLiveObject, WhereClause, LiveObjectMutationInput } from './index.cjs';
3
3
  import { PostgresPool } from 'kysely';
4
4
  import { Application } from 'express-ws';
5
5
 
@@ -29,33 +29,33 @@ declare const mutationSchema: z.ZodUnion<[z.ZodObject<z.objectUtil.extendShape<{
29
29
  }, {
30
30
  resourceId: z.ZodString;
31
31
  payload: z.ZodEffects<z.ZodRecord<z.ZodString, z.ZodObject<{
32
- value: z.ZodUnion<[z.ZodUnion<[z.ZodUnion<[z.ZodString, z.ZodNumber]>, z.ZodBoolean]>, z.ZodDate]>;
32
+ value: z.ZodNullable<z.ZodUnion<[z.ZodUnion<[z.ZodUnion<[z.ZodString, z.ZodNumber]>, z.ZodBoolean]>, z.ZodDate]>>;
33
33
  _meta: z.ZodOptional<z.ZodObject<{
34
- timestamp: z.ZodOptional<z.ZodString>;
34
+ timestamp: z.ZodNullable<z.ZodOptional<z.ZodString>>;
35
35
  }, "strip", z.ZodTypeAny, {
36
- timestamp?: string | undefined;
36
+ timestamp?: string | null | undefined;
37
37
  }, {
38
- timestamp?: string | undefined;
38
+ timestamp?: string | null | undefined;
39
39
  }>>;
40
40
  }, "strip", z.ZodTypeAny, {
41
- value: string | number | boolean | Date;
41
+ value: string | number | boolean | Date | null;
42
42
  _meta?: {
43
- timestamp?: string | undefined;
43
+ timestamp?: string | null | undefined;
44
44
  } | undefined;
45
45
  }, {
46
- value: string | number | boolean | Date;
46
+ value: string | number | boolean | Date | null;
47
47
  _meta?: {
48
- timestamp?: string | undefined;
48
+ timestamp?: string | null | undefined;
49
49
  } | undefined;
50
50
  }>>, Record<string, {
51
- value: string | number | boolean | Date;
51
+ value: string | number | boolean | Date | null;
52
52
  _meta?: {
53
- timestamp?: string | undefined;
53
+ timestamp?: string | null | undefined;
54
54
  } | undefined;
55
55
  }>, Record<string, {
56
- value: string | number | boolean | Date;
56
+ value: string | number | boolean | Date | null;
57
57
  _meta?: {
58
- timestamp?: string | undefined;
58
+ timestamp?: string | null | undefined;
59
59
  } | undefined;
60
60
  }>>;
61
61
  }>, "strip", z.ZodTypeAny, {
@@ -63,9 +63,9 @@ declare const mutationSchema: z.ZodUnion<[z.ZodObject<z.objectUtil.extendShape<{
63
63
  resourceId: string;
64
64
  resource: string;
65
65
  payload: Record<string, {
66
- value: string | number | boolean | Date;
66
+ value: string | number | boolean | Date | null;
67
67
  _meta?: {
68
- timestamp?: string | undefined;
68
+ timestamp?: string | null | undefined;
69
69
  } | undefined;
70
70
  }>;
71
71
  id?: string | undefined;
@@ -74,9 +74,9 @@ declare const mutationSchema: z.ZodUnion<[z.ZodObject<z.objectUtil.extendShape<{
74
74
  resourceId: string;
75
75
  resource: string;
76
76
  payload: Record<string, {
77
- value: string | number | boolean | Date;
77
+ value: string | number | boolean | Date | null;
78
78
  _meta?: {
79
- timestamp?: string | undefined;
79
+ timestamp?: string | null | undefined;
80
80
  } | undefined;
81
81
  }>;
82
82
  id?: string | undefined;
@@ -144,27 +144,32 @@ declare class RouteFactory {
144
144
  declare const routeFactory: typeof RouteFactory.create;
145
145
  type AnyRoute = Route<LiveObjectAny, Middleware<any>, Record<string, any>>;
146
146
 
147
+ type Simplify<T> = T extends Record<string, unknown> ? {
148
+ [K in keyof T]: Simplify<T[K]>;
149
+ } : T;
150
+
147
151
  declare abstract class Storage {
148
- abstract updateSchema(opts: Schema<any>): Promise<void>;
149
- abstract findById<T extends LiveObjectAny>(resourceName: string, id: string): Promise<MaterializedLiveType<T> | undefined>;
150
- abstract find<T extends LiveObjectAny>(resourceName: string, where?: Record<string, any>, include?: Record<string, any>): Promise<Record<string, MaterializedLiveType<T>>>;
151
- abstract upsert<T extends LiveObjectAny>(resourceName: string, resourceId: string, value: MaterializedLiveType<T>): Promise<MaterializedLiveType<T>>;
152
- }
153
- declare class InMemoryStorage extends Storage {
154
- private storage;
155
- updateSchema(opts: Schema<any>): Promise<void>;
156
- findById<T extends LiveObjectAny>(resourceName: string, id: string): Promise<MaterializedLiveType<T> | undefined>;
157
- find<T extends LiveObjectAny>(resourceName: string, where?: Record<string, any>, include?: Record<string, any>): Promise<Record<string, MaterializedLiveType<T>>>;
158
- upsert<T extends LiveObjectAny>(resourceName: string, resourceId: string, value: MaterializedLiveType<T>): Promise<MaterializedLiveType<T>>;
152
+ abstract findOne<T extends LiveObjectAny>(resource: T, id: string, options?: {
153
+ include?: IncludeClause<T>;
154
+ }): Promise<InferLiveObject<T> | undefined>;
155
+ abstract find<T extends LiveObjectAny>(resource: T, options?: {
156
+ where?: WhereClause<T>;
157
+ include?: IncludeClause<T>;
158
+ }): Promise<Record<string, InferLiveObject<T>>>;
159
+ insert<T extends LiveObjectAny>(resource: T, value: Simplify<LiveObjectMutationInput<T>>): Promise<InferLiveObject<T>>;
160
+ update<T extends LiveObjectAny>(resource: T, resourceId: string, value: LiveObjectMutationInput<T>): Promise<InferLiveObject<T>>;
159
161
  }
160
162
  declare class SQLStorage extends Storage {
161
163
  private db;
162
164
  private schema?;
163
165
  constructor(pool: PostgresPool);
164
- updateSchema(opts: Schema<any>): Promise<void>;
165
- findById<T extends LiveObjectAny>(resourceName: string, id: string): Promise<MaterializedLiveType<T> | undefined>;
166
- find<T extends LiveObjectAny>(resourceName: string, where?: WhereClause<T>, include?: IncludeClause<T>): Promise<Record<string, MaterializedLiveType<T>>>;
167
- upsert<T extends LiveObjectAny>(resourceName: string, resourceId: string, value: MaterializedLiveType<T>): Promise<MaterializedLiveType<T>>;
166
+ findOne<T extends LiveObjectAny>(resource: T, id: string, options?: {
167
+ include?: IncludeClause<T>;
168
+ }): Promise<InferLiveObject<T> | undefined>;
169
+ find<T extends LiveObjectAny>(resource: T, options?: {
170
+ where?: WhereClause<T>;
171
+ include?: IncludeClause<T>;
172
+ }): Promise<Record<string, InferLiveObject<T>>>;
168
173
  private convertToMaterializedLiveType;
169
174
  private applyWhere;
170
175
  private applyInclude;
@@ -221,4 +226,4 @@ declare class Server<TRouter extends AnyRouter> {
221
226
  }
222
227
  declare const server: typeof Server.create;
223
228
 
224
- export { type AnyRoute, type AnyRouter, type ContextProvider, InMemoryStorage, type Middleware, type Mutation, type MutationHandler, type MutationResult, type NextFunction, type ParsedRequest, type QueryResult, type RequestHandler, type RequestType, Route, RouteFactory, type RouteRecord, Router, SQLStorage, Server, Storage, expressAdapter, routeFactory, router, server };
229
+ export { type AnyRoute, type AnyRouter, type ContextProvider, type Middleware, type Mutation, type MutationHandler, type MutationResult, type NextFunction, type ParsedRequest, type QueryResult, type RequestHandler, type RequestType, Route, RouteFactory, type RouteRecord, Router, SQLStorage, Server, Storage, expressAdapter, routeFactory, router, server };
package/dist/server.d.ts CHANGED
@@ -1,5 +1,5 @@
1
1
  import { z, ZodTypeAny } from 'zod';
2
- import { LiveObjectAny, Schema, MaterializedLiveType, WhereClause, IncludeClause } from './index.js';
2
+ import { LiveObjectAny, Schema, MaterializedLiveType, IncludeClause, InferLiveObject, WhereClause, LiveObjectMutationInput } from './index.js';
3
3
  import { PostgresPool } from 'kysely';
4
4
  import { Application } from 'express-ws';
5
5
 
@@ -29,33 +29,33 @@ declare const mutationSchema: z.ZodUnion<[z.ZodObject<z.objectUtil.extendShape<{
29
29
  }, {
30
30
  resourceId: z.ZodString;
31
31
  payload: z.ZodEffects<z.ZodRecord<z.ZodString, z.ZodObject<{
32
- value: z.ZodUnion<[z.ZodUnion<[z.ZodUnion<[z.ZodString, z.ZodNumber]>, z.ZodBoolean]>, z.ZodDate]>;
32
+ value: z.ZodNullable<z.ZodUnion<[z.ZodUnion<[z.ZodUnion<[z.ZodString, z.ZodNumber]>, z.ZodBoolean]>, z.ZodDate]>>;
33
33
  _meta: z.ZodOptional<z.ZodObject<{
34
- timestamp: z.ZodOptional<z.ZodString>;
34
+ timestamp: z.ZodNullable<z.ZodOptional<z.ZodString>>;
35
35
  }, "strip", z.ZodTypeAny, {
36
- timestamp?: string | undefined;
36
+ timestamp?: string | null | undefined;
37
37
  }, {
38
- timestamp?: string | undefined;
38
+ timestamp?: string | null | undefined;
39
39
  }>>;
40
40
  }, "strip", z.ZodTypeAny, {
41
- value: string | number | boolean | Date;
41
+ value: string | number | boolean | Date | null;
42
42
  _meta?: {
43
- timestamp?: string | undefined;
43
+ timestamp?: string | null | undefined;
44
44
  } | undefined;
45
45
  }, {
46
- value: string | number | boolean | Date;
46
+ value: string | number | boolean | Date | null;
47
47
  _meta?: {
48
- timestamp?: string | undefined;
48
+ timestamp?: string | null | undefined;
49
49
  } | undefined;
50
50
  }>>, Record<string, {
51
- value: string | number | boolean | Date;
51
+ value: string | number | boolean | Date | null;
52
52
  _meta?: {
53
- timestamp?: string | undefined;
53
+ timestamp?: string | null | undefined;
54
54
  } | undefined;
55
55
  }>, Record<string, {
56
- value: string | number | boolean | Date;
56
+ value: string | number | boolean | Date | null;
57
57
  _meta?: {
58
- timestamp?: string | undefined;
58
+ timestamp?: string | null | undefined;
59
59
  } | undefined;
60
60
  }>>;
61
61
  }>, "strip", z.ZodTypeAny, {
@@ -63,9 +63,9 @@ declare const mutationSchema: z.ZodUnion<[z.ZodObject<z.objectUtil.extendShape<{
63
63
  resourceId: string;
64
64
  resource: string;
65
65
  payload: Record<string, {
66
- value: string | number | boolean | Date;
66
+ value: string | number | boolean | Date | null;
67
67
  _meta?: {
68
- timestamp?: string | undefined;
68
+ timestamp?: string | null | undefined;
69
69
  } | undefined;
70
70
  }>;
71
71
  id?: string | undefined;
@@ -74,9 +74,9 @@ declare const mutationSchema: z.ZodUnion<[z.ZodObject<z.objectUtil.extendShape<{
74
74
  resourceId: string;
75
75
  resource: string;
76
76
  payload: Record<string, {
77
- value: string | number | boolean | Date;
77
+ value: string | number | boolean | Date | null;
78
78
  _meta?: {
79
- timestamp?: string | undefined;
79
+ timestamp?: string | null | undefined;
80
80
  } | undefined;
81
81
  }>;
82
82
  id?: string | undefined;
@@ -144,27 +144,32 @@ declare class RouteFactory {
144
144
  declare const routeFactory: typeof RouteFactory.create;
145
145
  type AnyRoute = Route<LiveObjectAny, Middleware<any>, Record<string, any>>;
146
146
 
147
+ type Simplify<T> = T extends Record<string, unknown> ? {
148
+ [K in keyof T]: Simplify<T[K]>;
149
+ } : T;
150
+
147
151
  declare abstract class Storage {
148
- abstract updateSchema(opts: Schema<any>): Promise<void>;
149
- abstract findById<T extends LiveObjectAny>(resourceName: string, id: string): Promise<MaterializedLiveType<T> | undefined>;
150
- abstract find<T extends LiveObjectAny>(resourceName: string, where?: Record<string, any>, include?: Record<string, any>): Promise<Record<string, MaterializedLiveType<T>>>;
151
- abstract upsert<T extends LiveObjectAny>(resourceName: string, resourceId: string, value: MaterializedLiveType<T>): Promise<MaterializedLiveType<T>>;
152
- }
153
- declare class InMemoryStorage extends Storage {
154
- private storage;
155
- updateSchema(opts: Schema<any>): Promise<void>;
156
- findById<T extends LiveObjectAny>(resourceName: string, id: string): Promise<MaterializedLiveType<T> | undefined>;
157
- find<T extends LiveObjectAny>(resourceName: string, where?: Record<string, any>, include?: Record<string, any>): Promise<Record<string, MaterializedLiveType<T>>>;
158
- upsert<T extends LiveObjectAny>(resourceName: string, resourceId: string, value: MaterializedLiveType<T>): Promise<MaterializedLiveType<T>>;
152
+ abstract findOne<T extends LiveObjectAny>(resource: T, id: string, options?: {
153
+ include?: IncludeClause<T>;
154
+ }): Promise<InferLiveObject<T> | undefined>;
155
+ abstract find<T extends LiveObjectAny>(resource: T, options?: {
156
+ where?: WhereClause<T>;
157
+ include?: IncludeClause<T>;
158
+ }): Promise<Record<string, InferLiveObject<T>>>;
159
+ insert<T extends LiveObjectAny>(resource: T, value: Simplify<LiveObjectMutationInput<T>>): Promise<InferLiveObject<T>>;
160
+ update<T extends LiveObjectAny>(resource: T, resourceId: string, value: LiveObjectMutationInput<T>): Promise<InferLiveObject<T>>;
159
161
  }
160
162
  declare class SQLStorage extends Storage {
161
163
  private db;
162
164
  private schema?;
163
165
  constructor(pool: PostgresPool);
164
- updateSchema(opts: Schema<any>): Promise<void>;
165
- findById<T extends LiveObjectAny>(resourceName: string, id: string): Promise<MaterializedLiveType<T> | undefined>;
166
- find<T extends LiveObjectAny>(resourceName: string, where?: WhereClause<T>, include?: IncludeClause<T>): Promise<Record<string, MaterializedLiveType<T>>>;
167
- upsert<T extends LiveObjectAny>(resourceName: string, resourceId: string, value: MaterializedLiveType<T>): Promise<MaterializedLiveType<T>>;
166
+ findOne<T extends LiveObjectAny>(resource: T, id: string, options?: {
167
+ include?: IncludeClause<T>;
168
+ }): Promise<InferLiveObject<T> | undefined>;
169
+ find<T extends LiveObjectAny>(resource: T, options?: {
170
+ where?: WhereClause<T>;
171
+ include?: IncludeClause<T>;
172
+ }): Promise<Record<string, InferLiveObject<T>>>;
168
173
  private convertToMaterializedLiveType;
169
174
  private applyWhere;
170
175
  private applyInclude;
@@ -221,4 +226,4 @@ declare class Server<TRouter extends AnyRouter> {
221
226
  }
222
227
  declare const server: typeof Server.create;
223
228
 
224
- export { type AnyRoute, type AnyRouter, type ContextProvider, InMemoryStorage, type Middleware, type Mutation, type MutationHandler, type MutationResult, type NextFunction, type ParsedRequest, type QueryResult, type RequestHandler, type RequestType, Route, RouteFactory, type RouteRecord, Router, SQLStorage, Server, Storage, expressAdapter, routeFactory, router, server };
229
+ export { type AnyRoute, type AnyRouter, type ContextProvider, type Middleware, type Mutation, type MutationHandler, type MutationResult, type NextFunction, type ParsedRequest, type QueryResult, type RequestHandler, type RequestType, Route, RouteFactory, type RouteRecord, Router, SQLStorage, Server, Storage, expressAdapter, routeFactory, router, server };
package/dist/server.js CHANGED
@@ -1,2 +1,2 @@
1
- import {a,b}from'./chunk-EK7ODJWE.js';import he,{parse}from'qs';import {z as z$1}from'zod';import E from'node:crypto';import {Kysely,PostgresDialect}from'kysely';import {jsonArrayFrom}from'kysely/helpers/postgres';var P=a(I=>{Object.defineProperty(I,"__esModule",{value:true});I.parse=le;I.serialize=pe;var oe=/^[\u0021-\u003A\u003C\u003E-\u007E]+$/,ie=/^[\u0021-\u003A\u003C-\u007E]*$/,se=/^([.]?[a-z0-9]([a-z0-9-]{0,61}[a-z0-9])?)([.][a-z0-9]([a-z0-9-]{0,61}[a-z0-9])?)*$/i,ce=/^[\u0020-\u003A\u003D-\u007E]*$/,ue=Object.prototype.toString,de=(()=>{let r=function(){};return r.prototype=Object.create(null),r})();function le(r,e){let t=new de,n=r.length;if(n<2)return t;let a=(e==null?void 0:e.decode)||me,o=0;do{let i=r.indexOf("=",o);if(i===-1)break;let s=r.indexOf(";",o),c=s===-1?n:s;if(i>c){o=r.lastIndexOf(";",i-1)+1;continue}let y=$(r,o,i),p=V(r,i,y),u=r.slice(y,p);if(t[u]===void 0){let m=$(r,i+1,c),l=V(r,c,m),d=a(r.slice(m,l));t[u]=d;}o=c+1;}while(o<n);return t}function $(r,e,t){do{let n=r.charCodeAt(e);if(n!==32&&n!==9)return e}while(++e<t);return t}function V(r,e,t){for(;e>t;){let n=r.charCodeAt(--e);if(n!==32&&n!==9)return e+1}return t}function pe(r,e,t){let n=(t==null?void 0:t.encode)||encodeURIComponent;if(!oe.test(r))throw new TypeError(`argument name is invalid: ${r}`);let a=n(e);if(!ie.test(a))throw new TypeError(`argument val is invalid: ${e}`);let o=r+"="+a;if(!t)return o;if(t.maxAge!==void 0){if(!Number.isInteger(t.maxAge))throw new TypeError(`option maxAge is invalid: ${t.maxAge}`);o+="; Max-Age="+t.maxAge;}if(t.domain){if(!se.test(t.domain))throw new TypeError(`option domain is invalid: ${t.domain}`);o+="; Domain="+t.domain;}if(t.path){if(!ce.test(t.path))throw new TypeError(`option path is invalid: ${t.path}`);o+="; Path="+t.path;}if(t.expires){if(!ye(t.expires)||!Number.isFinite(t.expires.valueOf()))throw new TypeError(`option expires is invalid: ${t.expires}`);o+="; Expires="+t.expires.toUTCString();}if(t.httpOnly&&(o+="; HttpOnly"),t.secure&&(o+="; Secure"),t.partitioned&&(o+="; Partitioned"),t.priority)switch(typeof t.priority=="string"?t.priority.toLowerCase():void 0){case "low":o+="; Priority=Low";break;case "medium":o+="; Priority=Medium";break;case "high":o+="; Priority=High";break;default:throw new TypeError(`option priority is invalid: ${t.priority}`)}if(t.sameSite)switch(typeof t.sameSite=="string"?t.sameSite.toLowerCase():t.sameSite){case true:case "strict":o+="; SameSite=Strict";break;case "lax":o+="; SameSite=Lax";break;case "none":o+="; SameSite=None";break;default:throw new TypeError(`option sameSite is invalid: ${t.sameSite}`)}return o}function me(r){if(r.indexOf("%")===-1)return r;try{return decodeURIComponent(r)}catch{return r}}function ye(r){return ue.call(r)==="[object Date]"}});var Z=b(P(),1);var _=z$1.object({type:z$1.literal("QUERY"),resource:z$1.string(),where:z$1.record(z$1.any()).optional(),include:z$1.record(z$1.any()).optional()}),L=z$1.record(z$1.object({value:z$1.string().or(z$1.number()).or(z$1.boolean()).or(z$1.date()),_meta:z$1.object({timestamp:z$1.string().optional()}).optional()})).superRefine((r,e)=>{r.id&&e.addIssue({code:z$1.ZodIssueCode.custom,message:"Payload cannot have an id"});}),q=z$1.object({id:z$1.string().optional(),type:z$1.literal("MUTATE"),resource:z$1.string()}),S=q.extend({procedure:z$1.string(),payload:z$1.any().optional()}),M=q.extend({resourceId:z$1.string(),payload:L});z$1.union([S,M]);var F=_.omit({type:true,resource:true}),j=S.omit({id:true,type:true,resource:true,procedure:true}),O=M.omit({id:true,type:true,resource:true});z$1.union([O,j]);var H=r=>async e=>{var t;try{let n=typeof e.headers.getSetCookie=="function"?Object.fromEntries(e.headers):e.headers,a={headers:n,cookies:n.cookie?Z.default.parse(n.cookie):{}},o=new URL(e.url),i=o.pathname.split("/"),s=o.searchParams,c=he.parse(s.toString()),y=await((t=r.contextProvider)==null?void 0:t.call(r,{transport:"HTTP",headers:a.headers,cookies:a.cookies,query:c}))??{};if(e.method==="GET"){let p=i[i.length-1],{success:u,data:m,error:l}=F.safeParse(c);if(!u)return Response.json({message:"Invalid query",code:"INVALID_QUERY",details:l},{status:400});let d=await r.handleRequest({req:{...a,type:"QUERY",resourceName:p,context:y,where:m.where,include:m.include,query:c}});return !d||!d.data?Response.json({message:"Invalid resource",code:"INVALID_RESOURCE"},{status:400}):Response.json(d.data)}if(e.method==="POST")try{let p=i[i.length-1],u=i[i.length-2],m=e.body?await e.json():{},l;if(p==="set"){let{success:b,data:g,error:w}=O.safeParse(m);if(!b)return Response.json({message:"Invalid mutation",code:"INVALID_REQUEST",details:w},{status:400});l=g;}else {let{success:b,data:g,error:w}=j.safeParse(m);if(!b)return Response.json({message:"Invalid mutation",code:"INVALID_REQUEST",details:w},{status:400});l=g;}let d=await r.handleRequest({req:{...a,type:"MUTATE",resourceName:u,input:l.payload,context:y,resourceId:l.resourceId,procedure:p!=="set"?p:void 0,query:{}}});return Response.json(d)}catch(p){return console.error("Error parsing mutation from the client:",p),Response.json({message:"Internal server error",code:"INTERNAL_SERVER_ERROR"},{status:500})}return Response.json({message:"Not found",code:"NOT_FOUND"},{status:404})}catch(n){return console.error("Unexpected error:",n),Response.json({message:"Internal server error",code:"INTERNAL_SERVER_ERROR"},{status:500})}};var J=b(P(),1);var T=z$1.string(),ge=z$1.object({id:T,type:z$1.literal("SUBSCRIBE"),resource:z$1.string()}),Te=z$1.object({id:T,type:z$1.literal("SYNC"),lastSyncedAt:z$1.string().optional(),resources:z$1.string().array().optional(),where:z$1.record(z$1.any()).optional()}),G=M.extend({id:T}),Re=S.extend({id:T}),xe=z$1.union([Re,G]),B=z$1.union([ge,Te,xe]),be=z$1.object({id:T,type:z$1.literal("SYNC"),resource:z$1.string(),data:z$1.record(L)}),we=z$1.object({id:T,type:z$1.literal("REJECT"),resource:z$1.string(),message:z$1.string().optional()}),Se=z$1.object({id:T,type:z$1.literal("REPLY"),data:z$1.any()});z$1.union([be,we,Se,G]);var W="0123456789ABCDEFGHJKMNPQRSTVWXYZ",v=32;var Me=16,K=10,Q=0xffffffffffff;var R;(function(r){r.Base32IncorrectEncoding="B32_ENC_INVALID",r.DecodeTimeInvalidCharacter="DEC_TIME_CHAR",r.DecodeTimeValueMalformed="DEC_TIME_MALFORMED",r.EncodeTimeNegative="ENC_TIME_NEG",r.EncodeTimeSizeExceeded="ENC_TIME_SIZE_EXCEED",r.EncodeTimeValueMalformed="ENC_TIME_MALFORMED",r.PRNGDetectFailure="PRNG_DETECT",r.ULIDInvalid="ULID_INVALID",r.Unexpected="UNEXPECTED",r.UUIDInvalid="UUID_INVALID";})(R||(R={}));var x=class extends Error{constructor(e,t){super(`${t} (${e})`),this.name="ULIDError",this.code=e;}};function ve(r){let e=Math.floor(r()*v);return e===v&&(e=v-1),W.charAt(e)}function Ie(r){var n;let e=Ee(),t=e&&(e.crypto||e.msCrypto)||(typeof E<"u"?E:null);if(typeof(t==null?void 0:t.getRandomValues)=="function")return ()=>{let a=new Uint8Array(1);return t.getRandomValues(a),a[0]/255};if(typeof(t==null?void 0:t.randomBytes)=="function")return ()=>t.randomBytes(1).readUInt8()/255;if((n=E)!=null&&n.randomBytes)return ()=>E.randomBytes(1).readUInt8()/255;throw new x(R.PRNGDetectFailure,"Failed to find a reliable PRNG")}function Ee(){return Le()?self:typeof window<"u"?window:typeof global<"u"?global:typeof globalThis<"u"?globalThis:null}function Ae(r,e){let t="";for(;r>0;r--)t=ve(e)+t;return t}function Pe(r,e=K){if(isNaN(r))throw new x(R.EncodeTimeValueMalformed,`Time must be a number: ${r}`);if(r>Q)throw new x(R.EncodeTimeSizeExceeded,`Cannot encode a time larger than ${Q}: ${r}`);if(r<0)throw new x(R.EncodeTimeNegative,`Time must be positive: ${r}`);if(Number.isInteger(r)===false)throw new x(R.EncodeTimeValueMalformed,`Time must be an integer: ${r}`);let t,n="";for(let a=e;a>0;a--)t=r%v,n=W.charAt(t)+n,r=(r-t)/v;return n}function Le(){return typeof WorkerGlobalScope<"u"&&self instanceof WorkerGlobalScope}function Y(r,e){let t=Ie(),n=Date.now();return Pe(n,K)+Ae(Me,t)}var N=()=>Y().toLowerCase();var X=r=>{let e={},t={};return r.subscribeToMutations(n=>{let a=n;!a.resourceId||!a.payload||(console.log("Mutation propagated:",a),Object.entries(t[a.resource]??{}).forEach(([o,i])=>{var s;(s=e[o])==null||s.send(JSON.stringify({...a,id:a.id??N()}));}));}),(n,a)=>{var p;let o=u=>{n.send(JSON.stringify(u));},i=N(),s={headers:a.headers,cookies:typeof a.headers.cookie=="string"?J.default.parse(a.headers.cookie):{}},c=parse(a.url.split("?")[1]),y=(p=r.contextProvider)==null?void 0:p.call(r,{transport:"WEBSOCKET",headers:s.headers,cookies:s.cookies,query:c});e[i]=n,console.log("Client connected:",i),n.on("message",async u=>{try{console.log("Message received from the client:",u);let m=B.parse(JSON.parse(u.toString()));if(m.type==="SUBSCRIBE"){let{resource:l}=m;t[l]||(t[l]={}),t[l][i]={};}else if(m.type==="SYNC"){let{resources:l}=m,d=l??Object.keys(r.schema);console.log("Syncing resources:",d),await Promise.all(d.map(async b=>{let g=await r.handleRequest({req:{...s,type:"QUERY",resourceName:b,context:await y??{},query:c}});if(!g||!g.data)throw new Error("Invalid resource");o({id:m.id,type:"SYNC",resource:b,data:Object.fromEntries(Object.entries(g.data??{}).map(([w,ne])=>[w,ne.value]))});}));}else if(m.type==="MUTATE"){let{resource:l}=m;console.log("Received mutation from client:",m);try{let d=await r.handleRequest({req:{...s,type:"MUTATE",resourceName:l,input:m.payload,context:{messageId:m.id,...await y??{}},resourceId:m.resourceId,procedure:m.procedure,query:c}});m.procedure&&o({id:m.id,type:"REPLY",data:d});}catch(d){o({id:m.id,type:"REJECT",resource:l,message:d.message}),console.error("Error parsing mutation from the client:",d);}}}catch(m){console.error("Error handling message from the client:",m);}}),n.on("close",()=>{console.log("Connection closed",i),delete e[i];for(let u of Object.values(t))delete u[i];});}};function ee(r){let e=`${r.protocol}://${r.hostname}${r.url}`,t=new Headers;return Object.entries(r.headers).forEach(([n,a])=>{a&&t.set(n,Array.isArray(a)?a.join(","):a);}),new Request(e,{method:r.method,headers:t,body:r.body&&r.method!=="GET"?JSON.stringify(r.body):void 0})}var mt=(r,e,t)=>{r.ws(`${(t==null?void 0:t.basePath)??""}/ws`,X(e)),r.use(`${(t==null?void 0:t.basePath)??""}/`,(n,a)=>{H(e)(ee(n)).then(i=>i.json().then(s=>a.status(i.status).send(s)));});};var C=class r{routes;constructor(e){this.routes=e.routes;}static create(e){return new r(e)}},gt=r=>C.create({...r}),Ne=r=>({handler:e=>({inputValidator:r??z$1.undefined(),handler:e})}),z=class r{_resourceSchema;resourceName;middlewares;customMutations;constructor(e,t){this.resourceName=e,this.middlewares=new Set,this.customMutations=t??{};}handleFind=async({req:e,db:t})=>({data:await t.find(e.resourceName,e.where,e.include),acceptedValues:null});handleSet=async({req:e,db:t,schema:n})=>{if(!e.input)throw new Error("Payload is required");if(!e.resourceId)throw new Error("ResourceId is required");let a=await t.findById(e.resourceName,e.resourceId),[o,i]=n[this.resourceName].mergeMutation("set",e.input,a);if(!i)throw new Error("Mutation rejected");return {data:await t.upsert(e.resourceName,e.resourceId,o),acceptedValues:i}};async handleRequest(e){let t=n=>(()=>{if(n.type==="QUERY")return this.handleFind({req:n,db:e.db,schema:e.schema});if(n.type==="MUTATE")if(n.procedure){if(this.customMutations[n.procedure]){let a=this.customMutations[n.procedure].inputValidator.parse(n.input);return n.input=a,this.customMutations[n.procedure].handler({req:n,db:e.db,schema:e.schema})}}else return this.handleSet({req:n,db:e.db,schema:e.schema});throw new Error("Invalid request")})();return await Array.from(this.middlewares.values()).reduceRight((n,a)=>o=>a({req:o,next:n}),async n=>t(n))(e.req)}use(...e){for(let t of e)this.middlewares.add(t);return this}withMutations(e){return new r(this.resourceName,e({mutation:Ne}))}},k=class r{middlewares;constructor(e=[]){this.middlewares=e;}createBasicRoute(e){return new z(e.name).use(...this.middlewares)}use(...e){return new r([...this.middlewares,...e])}static create(){return new r}},Tt=k.create;var A=class{},te=class extends A{storage={};async updateSchema(e){this.storage=Object.entries(e).reduce((t,[n,a])=>(t[a.name]={},t),{});}async findById(e,t){var n;return (n=this.storage[e])==null?void 0:n[t]}async find(e,t,n){return this.storage[e]??{}}async upsert(e,t,n){return this.storage[e]??={},this.storage[e][t]=n,n}},re=class extends A{db;schema;constructor(e){super(),this.db=new Kysely({dialect:new PostgresDialect({pool:e})});}async updateSchema(e){this.schema=e;let t=await this.db.introspection.getTables();for(let[n,a]of Object.entries(e)){let o=t.find(c=>c.name===n);o||await this.db.schema.createTable(n).ifNotExists().execute();let i=`${n}_meta`,s=t.find(c=>c.name===i);s||await this.db.schema.createTable(i).ifNotExists().execute();for(let[c,y]of Object.entries(a.fields)){let p=o==null?void 0:o.columns.find(l=>l.name===c),u=y.getStorageFieldType();p?p.dataType!==u.type&&console.error("Column type mismatch:",c,"expected to have type:",u.type,"but has type:",p.dataType):(await this.db.schema.alterTable(n).addColumn(c,u.type,l=>{let d=l;return u.unique&&(d=d.unique()),u.nullable||(d=d.notNull()),u.references&&(d=d.references(u.references)),u.primary&&(d=d.primaryKey()),u.default!==void 0&&(d=d.defaultTo(u.default)),d}).execute().catch(l=>{throw console.error("Error adding column",c,l),l}),u.index&&await this.db.schema.createIndex(`${n}_${c}_index`).on(n).column(c).execute().catch(l=>{})),(s==null?void 0:s.columns.find(l=>l.name===c))||await this.db.schema.alterTable(i).addColumn(c,"varchar",l=>{let d=l;return u.primary&&(d=d.primaryKey().references(`${n}.${c}`)),d}).execute();}}}async findById(e,t){let n=await this.db.selectFrom(e).where("id","=",t).selectAll(e).executeTakeFirst(),a=await this.db.selectFrom(`${e}_meta`).where("id","=",t).selectAll(`${e}_meta`).executeTakeFirst();if(!(!n||!a))return this.convertToMaterializedLiveType(n,a)}async find(e,t,n){let a=this.applyWhere(e,this.db.selectFrom(e).selectAll(e),t);a=this.applyInclude(e,a,n);let o=await a.execute(),i=Object.fromEntries(o.map(y=>{let{id:p,...u}=y;return [p,u]}));if(Object.keys(i).length===0)return {};let s=Object.fromEntries((await this.db.selectFrom(`${e}_meta`).selectAll().where("id","in",Object.keys(i)).execute()).map(y=>{let{id:p,...u}=y;return [p,u]}));return Object.entries(i).reduce((y,[p,u])=>(s[p]&&(y[p]=this.convertToMaterializedLiveType(u,s[p])),y),{})}async upsert(e,t,n){return await this.db.transaction().execute(async a=>{let o=!!await a.selectFrom(e).select("id").where("id","=",t).executeTakeFirst(),i={},s={};for(let[c,y]of Object.entries(n.value))i[c]=y.value,s[c]=y._meta.timestamp;o?await Promise.all([a.updateTable(e).set(i).where("id","=",t).execute(),a.updateTable(`${e}_meta`).set(s).where("id","=",t).execute()]):await Promise.all([a.insertInto(e).values({...i,id:t}).execute(),a.insertInto(`${e}_meta`).values({...s,id:t}).execute()]);}),n}convertToMaterializedLiveType(e,t){return {value:Object.fromEntries(Object.entries(e).flatMap(([n,a])=>t?[[n,{value:a,_meta:{timestamp:t==null?void 0:t[n]}}]]:[]))}}applyWhere(e,t,n){if(!n)return t;if(!this.schema)throw new Error("Schema not initialized");let a=this.schema[e];if(!a)throw new Error("Resource not found");for(let[o,i]of Object.entries(n))if(a.fields[o])t=t.where(`${e}.${o}`,"=",i);else if(a.relations[o]){let s=a.relations[o],c=s.entity.name,y=s.type==="one"?"id":s.foreignColumn,p=s.type==="one"?s.relationalColumn:"id";t=t.leftJoin(c,`${c}.${y}`,`${e}.${p}`),t=this.applyWhere(c,t,i);}return t}applyInclude(e,t,n){if(!n)return t;if(!this.schema)throw new Error("Schema not initialized");let a=this.schema[e];if(!a)throw new Error(`Resource not found: ${e}`);for(let[o,i]of Object.entries(n)){if(!a.relations[o])throw new Error(`Relation ${o} not found in resource ${e}`);let s=a.relations[o],c=s.entity.name,y=s.type==="one"?"id":s.foreignColumn,p=s.type==="one"?s.relationalColumn:"id";t=t.select(u=>jsonArrayFrom(u.selectFrom(c).selectAll(c).whereRef(`${c}.${y}`,"=",`${e}.${p}`)).as(o));}return t}};var D=class r{router;storage;schema;middlewares=new Set;contextProvider;mutationSubscriptions=new Set;constructor(e){var t;this.router=e.router,this.storage=e.storage,this.schema=e.schema,(t=e.middlewares)==null||t.forEach(n=>this.middlewares.add(n)),this.storage.updateSchema(this.schema),this.contextProvider=e.contextProvider;}static create(e){return new r(e)}subscribeToMutations(e){return this.mutationSubscriptions.add(e),()=>{this.mutationSubscriptions.delete(e);}}async handleRequest(e){if(!this.router.routes[e.req.resourceName])throw new Error("Invalid resource");let t=await Array.from(this.middlewares.values()).reduceRight((n,a)=>o=>a({req:o,next:n}),async n=>this.router.routes[e.req.resourceName].handleRequest({req:n,db:this.storage,schema:this.schema}))(e.req);return t&&e.req.type==="MUTATE"&&t.acceptedValues&&Object.keys(t.acceptedValues).length>0&&this.mutationSubscriptions.forEach(n=>{n({id:e.req.context.messageId,type:"MUTATE",resource:e.req.resourceName,payload:t.acceptedValues??{},resourceId:e.req.resourceId});}),t}use(e){return this.middlewares.add(e),this}context(e){return this.contextProvider=e,this}},vt=D.create;
2
- export{te as InMemoryStorage,z as Route,k as RouteFactory,C as Router,re as SQLStorage,D as Server,A as Storage,mt as expressAdapter,Tt as routeFactory,gt as router,vt as server};
1
+ import {a,b,r}from'./chunk-LLHCJUB6.js';import he,{parse}from'qs';import {z as z$1}from'zod';import A from'node:crypto';import {Kysely,PostgresDialect}from'kysely';import {jsonObjectFrom,jsonArrayFrom}from'kysely/helpers/postgres';var L=a(v=>{Object.defineProperty(v,"__esModule",{value:true});v.parse=le;v.serialize=me;var ie=/^[\u0021-\u003A\u003C\u003E-\u007E]+$/,oe=/^[\u0021-\u003A\u003C-\u007E]*$/,se=/^([.]?[a-z0-9]([a-z0-9-]{0,61}[a-z0-9])?)([.][a-z0-9]([a-z0-9-]{0,61}[a-z0-9])?)*$/i,ce=/^[\u0020-\u003A\u003D-\u007E]*$/,ue=Object.prototype.toString,de=(()=>{let r=function(){};return r.prototype=Object.create(null),r})();function le(r,e){let t=new de,n=r.length;if(n<2)return t;let a=(e==null?void 0:e.decode)||pe,i=0;do{let s=r.indexOf("=",i);if(s===-1)break;let c=r.indexOf(";",i),o=c===-1?n:c;if(s>o){i=r.lastIndexOf(";",s-1)+1;continue}let f=F(r,i,s),y=q(r,s,f),l=r.slice(f,y);if(t[l]===void 0){let m=F(r,s+1,o),u=q(r,o,m),d=a(r.slice(m,u));t[l]=d;}i=o+1;}while(i<n);return t}function F(r,e,t){do{let n=r.charCodeAt(e);if(n!==32&&n!==9)return e}while(++e<t);return t}function q(r,e,t){for(;e>t;){let n=r.charCodeAt(--e);if(n!==32&&n!==9)return e+1}return t}function me(r,e,t){let n=(t==null?void 0:t.encode)||encodeURIComponent;if(!ie.test(r))throw new TypeError(`argument name is invalid: ${r}`);let a=n(e);if(!oe.test(a))throw new TypeError(`argument val is invalid: ${e}`);let i=r+"="+a;if(!t)return i;if(t.maxAge!==void 0){if(!Number.isInteger(t.maxAge))throw new TypeError(`option maxAge is invalid: ${t.maxAge}`);i+="; Max-Age="+t.maxAge;}if(t.domain){if(!se.test(t.domain))throw new TypeError(`option domain is invalid: ${t.domain}`);i+="; Domain="+t.domain;}if(t.path){if(!ce.test(t.path))throw new TypeError(`option path is invalid: ${t.path}`);i+="; Path="+t.path;}if(t.expires){if(!ye(t.expires)||!Number.isFinite(t.expires.valueOf()))throw new TypeError(`option expires is invalid: ${t.expires}`);i+="; Expires="+t.expires.toUTCString();}if(t.httpOnly&&(i+="; HttpOnly"),t.secure&&(i+="; Secure"),t.partitioned&&(i+="; Partitioned"),t.priority)switch(typeof t.priority=="string"?t.priority.toLowerCase():void 0){case "low":i+="; Priority=Low";break;case "medium":i+="; Priority=Medium";break;case "high":i+="; 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":i+="; SameSite=Strict";break;case "lax":i+="; SameSite=Lax";break;case "none":i+="; SameSite=None";break;default:throw new TypeError(`option sameSite is invalid: ${t.sameSite}`)}return i}function pe(r){if(r.indexOf("%")===-1)return r;try{return decodeURIComponent(r)}catch{return r}}function ye(r){return ue.call(r)==="[object Date]"}});var G=b(L(),1);var E=z$1.object({resource:z$1.string(),where:z$1.record(z$1.any()).optional(),include:z$1.record(z$1.any()).optional(),lastSyncedAt:z$1.string().optional(),limit:z$1.number().optional(),sort:z$1.array(z$1.object({key:z$1.string(),direction:z$1.enum(["asc","desc"])})).optional()}),P=z$1.record(z$1.object({value:z$1.string().or(z$1.number()).or(z$1.boolean()).or(z$1.date()).nullable(),_meta:z$1.object({timestamp:z$1.string().optional().nullable()}).optional()})).superRefine((r,e)=>{r.id&&e.addIssue({code:z$1.ZodIssueCode.custom,message:"Payload cannot have an id"});}),Z=z$1.object({id:z$1.string().optional(),type:z$1.literal("MUTATE"),resource:z$1.string()}),S=Z.extend({procedure:z$1.string(),payload:z$1.any().optional()}),M=Z.extend({resourceId:z$1.string(),payload:P});z$1.union([S,M]);var H=E.omit({resource:true}),C=S.omit({id:true,type:true,resource:true,procedure:true}),N=M.omit({id:true,type:true,resource:true});z$1.union([N,C]);var Q=r=>async e=>{var t;try{let n=typeof e.headers.getSetCookie=="function"?Object.fromEntries(e.headers):e.headers,a={headers:n,cookies:n.cookie?G.default.parse(n.cookie):{}},i=new URL(e.url),s=i.pathname.split("/"),c=i.searchParams,o=he.parse(c.toString()),f=await((t=r.contextProvider)==null?void 0:t.call(r,{transport:"HTTP",headers:a.headers,cookies:a.cookies,query:o}))??{};if(e.method==="GET"){let y=s[s.length-1],{success:l,data:m,error:u}=H.safeParse(o);if(!l)return Response.json({message:"Invalid query",code:"INVALID_QUERY",details:u},{status:400});let d=await r.handleRequest({req:{...a,type:"QUERY",resourceName:y,context:f,where:m.where,include:m.include,query:o}});return !d||!d.data?Response.json({message:"Invalid resource",code:"INVALID_RESOURCE"},{status:400}):Response.json(d.data)}if(e.method==="POST")try{let y=s[s.length-1],l=s[s.length-2],m=e.body?await e.json():{},u;if(y==="set"){let{success:b,data:w,error:O}=N.safeParse(m);if(!b)return Response.json({message:"Invalid mutation",code:"INVALID_REQUEST",details:O},{status:400});u=w;}else {let{success:b,data:w,error:O}=C.safeParse(m);if(!b)return Response.json({message:"Invalid mutation",code:"INVALID_REQUEST",details:O},{status:400});u=w;}let d=await r.handleRequest({req:{...a,type:"MUTATE",resourceName:l,input:u.payload,context:f,resourceId:u.resourceId,procedure:y!=="set"?y:void 0,query:{}}});return Response.json(d)}catch(y){return console.error("Error parsing mutation from the client:",y),Response.json({message:"Internal server error",code:"INTERNAL_SERVER_ERROR"},{status:500})}return Response.json({message:"Not found",code:"NOT_FOUND"},{status:404})}catch(n){return console.error("Unexpected error:",n),Response.json({message:"Internal server error",code:"INTERNAL_SERVER_ERROR"},{status:500})}};var ee=b(L(),1);var R=z$1.string(),Te=z$1.object({id:R,type:z$1.literal("SUBSCRIBE"),resource:z$1.string()}),ge=E.extend({id:R,type:z$1.literal("QUERY")}),B=M.extend({id:R}),Re=S.extend({id:R}),be=z$1.union([Re,B]),W=z$1.union([Te,ge,be]),we=z$1.object({id:R,type:z$1.literal("REJECT"),resource:z$1.string(),message:z$1.string().optional()}),xe=z$1.object({id:R,type:z$1.literal("REPLY"),data:z$1.any()});z$1.union([we,xe,B]);z$1.object({resource:z$1.string(),data:z$1.record(P)});var Y="0123456789ABCDEFGHJKMNPQRSTVWXYZ",I=32;var Se=16,J=10,K=0xffffffffffff;var T;(function(r){r.Base32IncorrectEncoding="B32_ENC_INVALID",r.DecodeTimeInvalidCharacter="DEC_TIME_CHAR",r.DecodeTimeValueMalformed="DEC_TIME_MALFORMED",r.EncodeTimeNegative="ENC_TIME_NEG",r.EncodeTimeSizeExceeded="ENC_TIME_SIZE_EXCEED",r.EncodeTimeValueMalformed="ENC_TIME_MALFORMED",r.PRNGDetectFailure="PRNG_DETECT",r.ULIDInvalid="ULID_INVALID",r.Unexpected="UNEXPECTED",r.UUIDInvalid="UUID_INVALID";})(T||(T={}));var g=class extends Error{constructor(e,t){super(`${t} (${e})`),this.name="ULIDError",this.code=e;}};function Me(r){let e=Math.floor(r()*I);return e===I&&(e=I-1),Y.charAt(e)}function Ie(r){var n;let e=ve(),t=e&&(e.crypto||e.msCrypto)||(typeof A<"u"?A:null);if(typeof(t==null?void 0:t.getRandomValues)=="function")return ()=>{let a=new Uint8Array(1);return t.getRandomValues(a),a[0]/255};if(typeof(t==null?void 0:t.randomBytes)=="function")return ()=>t.randomBytes(1).readUInt8()/255;if((n=A)!=null&&n.randomBytes)return ()=>A.randomBytes(1).readUInt8()/255;throw new g(T.PRNGDetectFailure,"Failed to find a reliable PRNG")}function ve(){return je()?self:typeof window<"u"?window:typeof global<"u"?global:typeof globalThis<"u"?globalThis:null}function Ee(r,e){let t="";for(;r>0;r--)t=Me(e)+t;return t}function Ae(r,e=J){if(isNaN(r))throw new g(T.EncodeTimeValueMalformed,`Time must be a number: ${r}`);if(r>K)throw new g(T.EncodeTimeSizeExceeded,`Cannot encode a time larger than ${K}: ${r}`);if(r<0)throw new g(T.EncodeTimeNegative,`Time must be positive: ${r}`);if(Number.isInteger(r)===false)throw new g(T.EncodeTimeValueMalformed,`Time must be an integer: ${r}`);let t,n="";for(let a=e;a>0;a--)t=r%I,n=Y.charAt(t)+n,r=(r-t)/I;return n}function je(){return typeof WorkerGlobalScope<"u"&&self instanceof WorkerGlobalScope}function X(r,e){let t=Ie(),n=Date.now();return Ae(n,J)+Ee(Se,t)}var _=()=>X().toLowerCase();var te=r=>{let e={},t={};return r.subscribeToMutations(n=>{let a=n;!a.resourceId||!a.payload||(console.log("Mutation propagated:",a),Object.entries(t[a.resource]??{}).forEach(([i,s])=>{var c;(c=e[i])==null||c.send(JSON.stringify({...a,id:a.id??_()}));}));}),(n,a)=>{var y;let i=l=>{n.send(JSON.stringify(l));},s=_(),c={headers:a.headers,cookies:typeof a.headers.cookie=="string"?ee.default.parse(a.headers.cookie):{}},o=parse(a.url.split("?")[1]),f=(y=r.contextProvider)==null?void 0:y.call(r,{transport:"WEBSOCKET",headers:c.headers,cookies:c.cookies,query:o});e[s]=n,console.log("Client connected:",s),n.on("message",async l=>{try{console.log("Message received from the client:",l);let m=W.parse(JSON.parse(l.toString()));if(m.type==="SUBSCRIBE"){let{resource:u}=m;t[u]||(t[u]={}),t[u][s]={};}else if(m.type==="QUERY"){let{resource:u}=m,d=await r.handleRequest({req:{...c,type:"QUERY",resourceName:u,context:await f??{},query:o}});if(!d||!d.data)throw new Error("Invalid resource");i({id:m.id,type:"REPLY",data:{resource:u,data:Object.fromEntries(Object.entries(d.data??{}).map(([b,w])=>[b,w.value]))}});}else if(m.type==="MUTATE"){let{resource:u}=m;console.log("Received mutation from client:",m);try{let d=await r.handleRequest({req:{...c,type:"MUTATE",resourceName:u,input:m.payload,context:{messageId:m.id,...await f??{}},resourceId:m.resourceId,procedure:m.procedure,query:o}});m.procedure&&i({id:m.id,type:"REPLY",data:d});}catch(d){i({id:m.id,type:"REJECT",resource:u,message:d.message}),console.error("Error parsing mutation from the client:",d);}}}catch(m){console.error("Error handling message from the client:",m);}}),n.on("close",()=>{console.log("Connection closed",s),delete e[s];for(let l of Object.values(t))delete l[s];});}};function re(r){let e=`${r.protocol}://${r.hostname}${r.url}`,t=new Headers;return Object.entries(r.headers).forEach(([n,a])=>{a&&t.set(n,Array.isArray(a)?a.join(","):a);}),new Request(e,{method:r.method,headers:t,body:r.body&&r.method!=="GET"?JSON.stringify(r.body):void 0})}var dt=(r,e,t)=>{r.ws(`${(t==null?void 0:t.basePath)??""}/ws`,te(e)),r.use(`${(t==null?void 0:t.basePath)??""}/`,(n,a)=>{Q(e)(re(n)).then(s=>s.json().then(c=>a.status(s.status).send(c)));});};var $=class r{routes;constructor(e){this.routes=e.routes;}static create(e){return new r(e)}},pt=r=>$.create({...r}),Pe=r=>({handler:e=>({inputValidator:r??z$1.undefined(),handler:e})}),U=class r{_resourceSchema;resourceName;middlewares;customMutations;constructor(e,t){this.resourceName=e,this.middlewares=new Set,this.customMutations=t??{};}handleFind=async({req:e,db:t})=>({data:await t.rawFind(e.resourceName,e.where,e.include),acceptedValues:null});handleSet=async({req:e,db:t,schema:n})=>{if(!e.input)throw new Error("Payload is required");if(!e.resourceId)throw new Error("ResourceId is required");let a=await t.rawFindById(e.resourceName,e.resourceId),[i,s]=n[this.resourceName].mergeMutation("set",e.input,a);if(!s)throw new Error("Mutation rejected");return {data:await t.rawUpsert(e.resourceName,e.resourceId,i),acceptedValues:s}};async handleRequest(e){let t=n=>(()=>{if(n.type==="QUERY")return this.handleFind({req:n,db:e.db,schema:e.schema});if(n.type==="MUTATE")if(n.procedure){if(this.customMutations[n.procedure]){let a=this.customMutations[n.procedure].inputValidator.parse(n.input);return n.input=a,this.customMutations[n.procedure].handler({req:n,db:e.db,schema:e.schema})}}else return this.handleSet({req:n,db:e.db,schema:e.schema});throw new Error("Invalid request")})();return await Array.from(this.middlewares.values()).reduceRight((n,a)=>i=>a({req:i,next:n}),async n=>t(n))(e.req)}use(...e){for(let t of e)this.middlewares.add(t);return this}withMutations(e){return new r(this.resourceName,e({mutation:Pe}))}},D=class r{middlewares;constructor(e=[]){this.middlewares=e;}createBasicRoute(e){return new U(e.name).use(...this.middlewares)}use(...e){return new r([...this.middlewares,...e])}static create(){return new r}},yt=D.create;var z=class{async insert(e,t){let n=new Date().toISOString();return r(await this.rawUpsert(e.name,t.id,{value:Object.fromEntries(Object.entries(t).map(([a,i])=>[a,{value:i,_meta:{timestamp:n}}]))}))}async update(e,t,n){let a=new Date().toISOString(),{id:i,...s}=n;return r(await this.rawUpsert(e.name,t,{value:Object.fromEntries(Object.entries(s).map(([c,o])=>[c,{value:o,_meta:{timestamp:a}}]))}))}},ne=class extends z{db;schema;constructor(e){super(),this.db=new Kysely({dialect:new PostgresDialect({pool:e})});}async updateSchema(e){this.schema=e;let t=await this.db.introspection.getTables();for(let[n,a]of Object.entries(e)){let i=t.find(o=>o.name===n);i||await this.db.schema.createTable(n).ifNotExists().execute();let s=`${n}_meta`,c=t.find(o=>o.name===s);c||await this.db.schema.createTable(s).ifNotExists().execute();for(let[o,f]of Object.entries(a.fields)){let y=i==null?void 0:i.columns.find(u=>u.name===o),l=f.getStorageFieldType();y?y.dataType!==l.type&&console.error("Column type mismatch:",o,"expected to have type:",l.type,"but has type:",y.dataType):(await this.db.schema.alterTable(n).addColumn(o,l.type,u=>{let d=u;return l.unique&&(d=d.unique()),l.nullable||(d=d.notNull()),l.references&&(d=d.references(l.references)),l.primary&&(d=d.primaryKey()),l.default!==void 0&&(d=d.defaultTo(l.default)),d}).execute().catch(u=>{throw console.error("Error adding column",o,u),u}),l.index&&await this.db.schema.createIndex(`${n}_${o}_index`).on(n).column(o).execute().catch(u=>{})),(c==null?void 0:c.columns.find(u=>u.name===o))||await this.db.schema.alterTable(s).addColumn(o,"varchar",u=>{let d=u;return l.primary&&(d=d.primaryKey().references(`${n}.${o}`)),d}).execute();}}}async rawFindById(e,t,n){let a=await this.db.selectFrom(e).where("id","=",t).selectAll(e).select(s=>jsonObjectFrom(s.selectFrom(`${e}_meta`).selectAll(`${e}_meta`).whereRef(`${e}_meta.id`,"=",`${e}.id`)).as("_meta"));a=this.applyInclude(e,a,n);let i=await a.executeTakeFirst();if(i)return this.convertToMaterializedLiveType(i)}async findOne(e,t,n){let a=await this.rawFindById(e.name,t,n==null?void 0:n.include);if(a)return r(a)}async rawFind(e,t,n){let a=this.db.selectFrom(e).selectAll(e).select(o=>jsonObjectFrom(o.selectFrom(`${e}_meta`).selectAll(`${e}_meta`).whereRef(`${e}_meta.id`,"=",`${e}.id`)).as("_meta"));a=this.applyWhere(e,a,t),a=this.applyInclude(e,a,n);let i=await a.execute(),s=Object.fromEntries(i.map(o=>{let{id:f,...y}=o;return [f,y]}));return Object.keys(s).length===0?{}:Object.entries(s).reduce((o,[f,y])=>(o[f]=this.convertToMaterializedLiveType(y),o),{})}async find(e,t){let n=await this.rawFind(e.name,t==null?void 0:t.where,t==null?void 0:t.include);return Object.fromEntries(Object.entries(n).map(([a,i])=>[a,r(i)]))}async rawUpsert(e,t,n){return await this.db.transaction().execute(async a=>{var o;let i=!!await a.selectFrom(e).select("id").where("id","=",t).executeTakeFirst(),s={},c={};for(let[f,y]of Object.entries(n.value)){let l=(o=y._meta)==null?void 0:o.timestamp;l&&(s[f]=y.value,c[f]=l);}i?await Promise.all([a.updateTable(e).set(s).where("id","=",t).execute(),a.updateTable(`${e}_meta`).set(c).where("id","=",t).execute()]):await Promise.all([a.insertInto(e).values({...s,id:t}).execute(),a.insertInto(`${e}_meta`).values({...c,id:t}).execute()]);}),n}convertToMaterializedLiveType(e){if(!e._meta)throw new Error("Missing _meta");return {value:Object.entries(e).reduce((t,[n,a])=>{var i,s,c;return n==="_meta"||(n==="id"?t[n]={value:a}:Array.isArray(a)?t[n]={value:a.map(o=>this.convertToMaterializedLiveType(o)),_meta:{timestamp:(i=e==null?void 0:e._meta)==null?void 0:i[n]}}:typeof a=="object"&&a!==null&&!(a instanceof Date)?t[n]={...this.convertToMaterializedLiveType(a),_meta:{timestamp:(s=e==null?void 0:e._meta)==null?void 0:s[n]}}:t[n]={value:a,_meta:{timestamp:(c=e==null?void 0:e._meta)==null?void 0:c[n]}}),t},{})}}applyWhere(e,t,n){if(!n)return t;if(!this.schema)throw new Error("Schema not initialized");let a=this.schema[e];if(!a)throw new Error("Resource not found");for(let[i,s]of Object.entries(n))if(a.fields[i])t=t.where(`${e}.${i}`,"=",s);else if(a.relations[i]){let c=a.relations[i],o=c.entity.name,f=c.type==="one"?"id":c.foreignColumn,y=c.type==="one"?c.relationalColumn:"id";t=t.leftJoin(o,`${o}.${f}`,`${e}.${y}`),t=this.applyWhere(o,t,s);}return t}applyInclude(e,t,n){if(!n)return t;if(!this.schema)throw new Error("Schema not initialized");let a=this.schema[e];if(!a)throw new Error(`Resource not found: ${e}`);for(let[i,s]of Object.entries(n)){if(!a.relations[i])throw new Error(`Relation ${i} not found in resource ${e}`);let c=a.relations[i],o=c.entity.name,f=c.type==="one"?"id":c.foreignColumn,y=c.type==="one"?c.relationalColumn:"id",l=c.type==="one"?jsonObjectFrom:jsonArrayFrom;t=t.select(m=>l(m.selectFrom(o).selectAll(o).whereRef(`${o}.${f}`,"=",`${e}.${y}`).select(u=>jsonObjectFrom(u.selectFrom(`${o}_meta`).selectAll(`${o}_meta`).whereRef(`${o}_meta.id`,"=",`${o}.id`)).as("_meta"))).as(i));}return t}};var k=class r{router;storage;schema;middlewares=new Set;contextProvider;mutationSubscriptions=new Set;constructor(e){var t;this.router=e.router,this.storage=e.storage,this.schema=e.schema,(t=e.middlewares)==null||t.forEach(n=>{this.middlewares.add(n);}),this.storage.updateSchema(this.schema),this.contextProvider=e.contextProvider;}static create(e){return new r(e)}subscribeToMutations(e){return this.mutationSubscriptions.add(e),()=>{this.mutationSubscriptions.delete(e);}}async handleRequest(e){if(!this.router.routes[e.req.resourceName])throw new Error("Invalid resource");let t=await Array.from(this.middlewares.values()).reduceRight((n,a)=>i=>a({req:i,next:n}),async n=>this.router.routes[e.req.resourceName].handleRequest({req:n,db:this.storage,schema:this.schema}))(e.req);return t&&e.req.type==="MUTATE"&&t.acceptedValues&&Object.keys(t.acceptedValues).length>0&&this.mutationSubscriptions.forEach(n=>{n({id:e.req.context.messageId,type:"MUTATE",resource:e.req.resourceName,payload:t.acceptedValues??{},resourceId:e.req.resourceId});}),t}use(e){return this.middlewares.add(e),this}context(e){return this.contextProvider=e,this}},bt=k.create;
2
+ export{U as Route,D as RouteFactory,$ as Router,ne as SQLStorage,k as Server,z as Storage,dt as expressAdapter,yt as routeFactory,pt as router,bt as server};