modelence 0.17.0-dev.0 → 0.18.0-dev.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (43) hide show
  1. package/dist/bin/modelence.js +2 -2
  2. package/dist/bin/modelence.js.map +1 -1
  3. package/dist/chunk-IJ5BS7PM.js +2 -0
  4. package/dist/chunk-IJ5BS7PM.js.map +1 -0
  5. package/dist/chunk-JBTEZOY2.js +2 -0
  6. package/dist/chunk-JBTEZOY2.js.map +1 -0
  7. package/dist/chunk-KU4GANYW.js +3 -0
  8. package/dist/chunk-KU4GANYW.js.map +1 -0
  9. package/dist/chunk-LDVLXJLS.js +3 -0
  10. package/dist/chunk-LDVLXJLS.js.map +1 -0
  11. package/dist/chunk-S2E4OO6G.js +19 -0
  12. package/dist/chunk-S2E4OO6G.js.map +1 -0
  13. package/dist/chunk-S77LRZU6.js +2 -0
  14. package/dist/chunk-S77LRZU6.js.map +1 -0
  15. package/dist/chunk-VFOGVWJK.js +2 -0
  16. package/dist/chunk-VFOGVWJK.js.map +1 -0
  17. package/dist/client.d.ts +64 -5
  18. package/dist/client.js +1 -1
  19. package/dist/client.js.map +1 -1
  20. package/dist/collectCss-4V7I3QV3.js +2 -0
  21. package/dist/collectCss-4V7I3QV3.js.map +1 -0
  22. package/dist/{index-CLpVWWuj.d.ts → index-BBMsjrFN.d.ts} +3 -1
  23. package/dist/index.d.ts +2 -2
  24. package/dist/index.js +1 -1
  25. package/dist/{package-WKD57CCF.js → package-XQQF6NGT.js} +2 -2
  26. package/dist/{package-WKD57CCF.js.map → package-XQQF6NGT.js.map} +1 -1
  27. package/dist/render-57TPXKHI.js +2 -0
  28. package/dist/render-57TPXKHI.js.map +1 -0
  29. package/dist/renderApp-3FQ6KUQ2.js +2 -0
  30. package/dist/renderApp-3FQ6KUQ2.js.map +1 -0
  31. package/dist/server-JB4DTGBT.js +2 -0
  32. package/dist/server-JB4DTGBT.js.map +1 -0
  33. package/dist/server.d.ts +8 -6
  34. package/dist/server.js +1 -18
  35. package/dist/server.js.map +1 -1
  36. package/dist/transport-BOPYDAVH.js +2 -0
  37. package/dist/transport-BOPYDAVH.js.map +1 -0
  38. package/dist/{types-CTjq1HaP.d.ts → types-DV0J5rPI.d.ts} +8 -0
  39. package/dist/{types-DNZiaNSs.d.ts → types-mkbhnQN2.d.ts} +2 -2
  40. package/dist/types.d.ts +3 -3
  41. package/package.json +3 -1
  42. package/dist/chunk-52CM66XJ.js +0 -3
  43. package/dist/chunk-52CM66XJ.js.map +0 -1
@@ -0,0 +1,2 @@
1
+ import {a,d,m}from'./chunk-IJ5BS7PM.js';import {q as q$1,s,n,t,v as v$1}from'./chunk-VFOGVWJK.js';import A from'react';import K from'react-dom/client';import {jsx}from'react/jsx-runtime';function h(e){return {getConfig(t){return s(`${e}.${t}`)},query(t,...r){let o=r[0]??{};return {queryKey:[e,t,o],queryFn:()=>q$1(`${e}.${t}`,o)}},mutation(t){return {mutationFn:r=>q$1(`${e}.${t}`,r)}},infiniteQuery(t,r){return {queryKey:[e,t,"infinite",r(void 0)],initialPageParam:void 0,queryFn:({pageParam:o})=>q$1(`${e}.${t}`,r(o))}}}}var v=h("_system");var M=class{constructor(t,r){this.category=t,this.onMessage=r;}init(){d()?.on({category:this.category,listener:this.onMessage});}joinChannel(t){d()?.joinChannel({category:this.category,id:t});}leaveChannel(t){d()?.leaveChannel({category:this.category,id:t});}};async function b(e,{filePath:t,contentType:r,visibility:o}){let{url:d,fields:u,filePath:i}=await q$1("_system.files.getUploadUrl",{filePath:t,contentType:r,visibility:o}),s=new FormData;for(let[p,c]of Object.entries(u))s.append(p,c);s.append("file",e);let a=await fetch(d,{method:"POST",body:s});if(!a.ok)throw new Error(`Failed to upload file: HTTP status: ${a.status}`);return {filePath:i}}async function F(e){await q$1("_system.files.deleteFile",{filePath:e});}async function k(e){return q$1("_system.files.downloadFile",{filePath:e})}async function _(e){return q$1("_system.files.getFileUrl",{filePath:e})}var E="useClient"in A?A.useClient(a):a;var q="__MODELENCE_STATE__";function U(){if(typeof document>"u")return null;let e=document.getElementById(q);if(!e)return null;try{return JSON.parse(e.textContent??"")}catch(t){return console.error("Modelence: failed to parse SSR state",t),null}}var C="__modelence_ssr_snapshot__";function Q(e){globalThis[C]=e;}function Ie(){return globalThis[C]??null}function w(e){if(typeof window>"u"){Q(e);return}let{loadingElement:t$1,routesElement:r,favicon:o,errorHandler:d,router:u}=e;d&&n(d),window.addEventListener("unload",()=>{});let i=U(),s=i!==null;i?.session&&(t(i.session),v$1());let a=document.getElementById("root"),p=u?u({children:r}):r,c=jsx(A.StrictMode,{children:jsx(E,{loadingElement:t$1,children:jsx(m,{children:p})})});if(s?K.hydrateRoot(a,c):K.createRoot(a).render(c),o){let T=document.querySelector("link[rel~='icon']");if(T)T.href=o;else {let f=document.createElement("link");f.rel="icon",f.href=o,document.head.appendChild(f);}}}export{h as a,v as b,Ie as c,w as d,M as e,b as f,F as g,k as h,_ as i,E as j};//# sourceMappingURL=chunk-S77LRZU6.js.map
2
+ //# sourceMappingURL=chunk-S77LRZU6.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/client/module.ts","../src/system/client.ts","../src/websocket/clientChannel.ts","../src/files/client.ts","../src/client.ts","../src/client/renderApp.tsx"],"names":["createClientModule","moduleName","key","getConfig","name","rest","args","callMethod","getArgs","pageParam","systemConfig","ClientChannel","category","onMessage","getWebsocketClientProvider","id","uploadFile","file","filePath","contentType","visibility","url","fields","resolvedFilePath","formData","value","uploadResponse","deleteFile","downloadFile","getFileUrl","AppProvider","React","SSR_STATE_SCRIPT_ID","readSsrState","node","e","SNAPSHOT_KEY","setSnapshot","snapshot","_getSsrSnapshot","renderApp","options","loadingElement","routesElement","favicon","errorHandler","router","setErrorHandler","ssrState","isHydrating","hydrateSession","startSessionHeartbeat","container","routedTree","tree","jsx","ModelenceQueryProvider","ReactDOM","link","newLink"],"mappings":"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","file":"chunk-S77LRZU6.js","sourcesContent":["'use client';\n\nimport type { ObjectId } from 'mongodb';\nimport type { ConfigParams, ConfigType, ValueType } from '../config/types';\nimport { callMethod, type MethodArgs } from './method';\nimport type { AnyMethodShape } from '../methods/types';\n\n// Pulls the config store value without importing server-side code\nimport { getConfig as _getClientConfig } from '../config/client';\n\n// ── type helpers ─────────────────────────────────────────────────────────────\n\n/**\n * Recursively maps ObjectId → string to match the sanitized runtime values\n * sent over the wire. Dates are preserved (revived via typeMap on the client).\n */\ntype Sanitized<T> = T extends ObjectId\n ? string\n : T extends Date\n ? Date\n : T extends (infer U)[]\n ? Sanitized<U>[]\n : T extends object\n ? { [K in keyof T]: Sanitized<T[K]> }\n : T;\n\ntype ExtractArgs<M> = M extends (args: infer A, ...rest: any[]) => any // eslint-disable-line @typescript-eslint/no-explicit-any\n ? A\n : M extends { handler: (args: infer A, ...rest: any[]) => any } // eslint-disable-line @typescript-eslint/no-explicit-any\n ? A\n : MethodArgs;\n\ntype ExtractResult<M> = M extends (...args: any[]) => Promise<infer R> // eslint-disable-line @typescript-eslint/no-explicit-any\n ? Sanitized<R>\n : M extends { handler: (...args: any[]) => Promise<infer R> } // eslint-disable-line @typescript-eslint/no-explicit-any\n ? Sanitized<R>\n : unknown;\n\ntype PublicKeyOf<TSchema extends Record<string, ConfigParams>> = {\n [K in keyof TSchema as TSchema[K] extends ConfigParams<ConfigType, true>\n ? string & K\n : never]: ValueType<TSchema[K]['type']>;\n};\n\ntype AnyModule = {\n name: string;\n configSchema: Record<string, ConfigParams>;\n queries: Record<string, AnyMethodShape>;\n mutations: Record<string, AnyMethodShape>;\n};\n\n// ── createClientModule ────────────────────────────────────────────────────────\n\n/**\n * Creates a typed client accessor for a module's public configs, queries, and mutations.\n *\n * Use `import type` to reference the module so no server code is bundled on the client.\n * Arg and return types for queries and mutations are inferred automatically from the\n * server-side handler signatures.\n *\n * @param moduleName - The module's name as passed to `new Module(name, ...)`.\n *\n * @example\n * ```ts\n * // src/client/payments.ts\n * import type paymentsModule from '../server/payments';\n * import { createClientModule } from 'modelence/client';\n *\n * export const payments = createClientModule<typeof paymentsModule>('payments');\n * ```\n *\n * ```ts\n * // src/components/Checkout.tsx\n * import { useQuery, useMutation } from '@tanstack/react-query';\n * import { payments } from '../client/payments';\n *\n * // Typed config — public keys only, private and secret keys excluded:\n * const currency = payments.getConfig('currency'); // string | undefined\n *\n * // Typed query — pass directly to useQuery:\n * const { data: products } = useQuery(payments.query('getProducts', { page: 1 }));\n *\n * // Typed mutation — pass directly to useMutation:\n * const { mutate: charge } = useMutation(payments.mutation('charge'));\n * charge({ amount: 100 }); // args typed from handler signature\n * ```\n */\nexport function createClientModule<TModule extends AnyModule>(moduleName: string) {\n return {\n getConfig<K extends keyof PublicKeyOf<TModule['configSchema']> & string>(\n key: K\n ): PublicKeyOf<TModule['configSchema']>[K] | undefined {\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n return _getClientConfig(`${moduleName}.${key}`) as any;\n },\n\n query<K extends keyof TModule['queries'] & string>(\n name: K,\n ...rest: {} extends ExtractArgs<TModule['queries'][K]>\n ? [args?: ExtractArgs<TModule['queries'][K]>]\n : [args: ExtractArgs<TModule['queries'][K]>]\n ) {\n const args = (rest[0] ?? {}) as ExtractArgs<TModule['queries'][K]>;\n return {\n queryKey: [moduleName, name, args] as const,\n queryFn: (): Promise<ExtractResult<TModule['queries'][K]>> =>\n callMethod<ExtractResult<TModule['queries'][K]>>(\n `${moduleName}.${name}`,\n args as MethodArgs\n ),\n };\n },\n\n mutation<K extends keyof TModule['mutations'] & string>(name: K) {\n return {\n mutationFn: (\n args: ExtractArgs<TModule['mutations'][K]>\n ): Promise<ExtractResult<TModule['mutations'][K]>> =>\n callMethod<ExtractResult<TModule['mutations'][K]>>(\n `${moduleName}.${name}`,\n args as MethodArgs\n ),\n };\n },\n\n /**\n * Returns options for `useInfiniteQuery`. The `getArgs` callback receives the\n * current `pageParam` and returns the args to pass to the query handler.\n * Spread the result into `useInfiniteQuery` alongside `getNextPageParam`.\n *\n * Annotate the `pageParam` type in the callback so TypeScript can infer the\n * page param type — no manual generic needed on `useInfiniteQuery`.\n */\n infiniteQuery<K extends keyof TModule['queries'] & string, TPageParam = unknown>(\n name: K,\n getArgs: (pageParam: TPageParam | undefined) => ExtractArgs<TModule['queries'][K]>\n ) {\n return {\n queryKey: [moduleName, name, 'infinite', getArgs(undefined)] as const,\n // Included so TanStack infers TPageParam from the callback type, not from a bare `undefined`.\n initialPageParam: undefined as TPageParam | undefined,\n queryFn: ({\n pageParam,\n }: {\n pageParam: TPageParam | undefined;\n }): Promise<ExtractResult<TModule['queries'][K]>> =>\n callMethod<ExtractResult<TModule['queries'][K]>>(\n `${moduleName}.${name}`,\n getArgs(pageParam) as MethodArgs\n ),\n };\n },\n };\n}\n","import type systemModule from './index';\nimport { createClientModule } from '../client/module';\n\nexport const systemConfig = createClientModule<typeof systemModule>('_system');\n","import { getWebsocketClientProvider } from './client';\n\nexport class ClientChannel<T = unknown> {\n public readonly category: string;\n private readonly onMessage: (data: T) => void;\n\n constructor(category: string, onMessage: (data: T) => void) {\n this.category = category;\n this.onMessage = onMessage;\n }\n\n init() {\n getWebsocketClientProvider()?.on({\n category: this.category,\n listener: this.onMessage,\n });\n }\n\n joinChannel(id: string) {\n getWebsocketClientProvider()?.joinChannel({\n category: this.category,\n id,\n });\n }\n\n leaveChannel(id: string) {\n getWebsocketClientProvider()?.leaveChannel({\n category: this.category,\n id,\n });\n }\n}\n","import { callMethod } from '../client/method';\nimport type { FileVisibility, GetUploadUrlResult } from './types';\n\ntype UploadFileParams = {\n filePath: string;\n contentType: string;\n visibility: FileVisibility;\n};\n\ntype UploadFileResult = {\n filePath: string;\n};\n\nexport async function uploadFile(\n file: File | Blob,\n { filePath, contentType, visibility }: UploadFileParams\n): Promise<UploadFileResult> {\n const {\n url,\n fields,\n filePath: resolvedFilePath,\n } = await callMethod<GetUploadUrlResult>('_system.files.getUploadUrl', {\n filePath,\n contentType,\n visibility,\n });\n\n const formData = new FormData();\n for (const [key, value] of Object.entries(fields)) {\n formData.append(key, value);\n }\n formData.append('file', file);\n\n const uploadResponse = await fetch(url, {\n method: 'POST',\n body: formData,\n });\n\n if (!uploadResponse.ok) {\n throw new Error(`Failed to upload file: HTTP status: ${uploadResponse.status}`);\n }\n\n return { filePath: resolvedFilePath };\n}\n\nexport async function deleteFile(filePath: string): Promise<void> {\n await callMethod('_system.files.deleteFile', { filePath });\n}\n\nexport async function downloadFile(filePath: string): Promise<{ downloadUrl: string }> {\n return callMethod('_system.files.downloadFile', { filePath });\n}\n\nexport async function getFileUrl(filePath: string): Promise<{ url: string }> {\n return callMethod('_system.files.getFileUrl', { filePath });\n}\n","import React from 'react';\n\nimport { AppProvider as OriginalAppProvider } from './client/AppProvider';\n\nexport { getConfig } from './config/client';\nexport { createClientModule } from './client/module';\nexport type { ValueType } from './config/types';\nexport { systemConfig } from './system/client';\n\nexport const AppProvider =\n 'useClient' in React\n ? // @ts-ignore: React.useClient only exists in Next.js\n React.useClient(OriginalAppProvider)\n : OriginalAppProvider;\n\nexport { renderApp } from './client/renderApp';\nexport { ModelenceQueryProvider } from './client/queryProvider';\nexport {\n modelenceQuery,\n modelenceLiveQuery,\n modelenceMutation,\n createQueryKey,\n connectModelenceQueryClient,\n disconnectModelenceQueryClient,\n ModelenceQueryClient,\n type ModelenceQueryKey,\n} from './client/query';\nexport { callMethod, MethodError, type MethodArgs } from './client/method';\nexport { useSession } from './client/session';\nexport {\n signupWithPassword,\n loginWithPassword,\n verifyEmail,\n updateProfile,\n resendEmailVerification,\n logout,\n sendResetPasswordToken,\n resetPassword,\n linkOAuthProvider,\n unlinkOAuthProvider,\n type UserInfo,\n} from './auth/client';\nexport {\n getWebsocketClientProvider,\n setWebsocketClientProvider,\n startWebsockets,\n subscribeLiveQuery,\n} from './websocket/client';\nexport { ClientChannel } from './websocket/clientChannel';\nexport { getLocalStorageSession } from './client/localStorage';\nexport { uploadFile, deleteFile, downloadFile, getFileUrl } from './files/client';\n","import React from 'react';\nimport ReactDOM from 'react-dom/client';\nimport { AppProvider } from '../client';\nimport { setErrorHandler, ErrorHandler } from './errorHandler';\nimport { hydrateSession, startSessionHeartbeat, type SessionInitPayload } from './session';\nimport { ModelenceQueryProvider } from './queryProvider';\n\nconst SSR_STATE_SCRIPT_ID = '__MODELENCE_STATE__';\n\ntype SsrState = {\n session?: SessionInitPayload;\n};\n\nfunction readSsrState(): SsrState | null {\n if (typeof document === 'undefined') {\n return null;\n }\n\n const node = document.getElementById(SSR_STATE_SCRIPT_ID);\n if (!node) {\n return null;\n }\n\n try {\n return JSON.parse(node.textContent ?? '') as SsrState;\n } catch (e) {\n console.error('Modelence: failed to parse SSR state', e);\n return null;\n }\n}\n\nexport type SsrRouter = (props: {\n children: React.ReactNode;\n location?: string;\n}) => React.ReactElement;\n\nexport interface RenderAppOptions {\n loadingElement: React.ReactNode;\n routesElement: React.ReactNode;\n favicon?: string;\n errorHandler?: ErrorHandler;\n router?: SsrRouter;\n}\n\n// Shared on globalThis because Vite's ssrLoadModule loads the user's entry\n// in a separate module graph from the framework runtime.\nconst SNAPSHOT_KEY = '__modelence_ssr_snapshot__';\n\ntype GlobalWithSnapshot = typeof globalThis & {\n [SNAPSHOT_KEY]?: RenderAppOptions | null;\n};\n\nfunction setSnapshot(snapshot: RenderAppOptions | null) {\n (globalThis as GlobalWithSnapshot)[SNAPSHOT_KEY] = snapshot;\n}\n\n/** @internal Used by the SSR runtime after evaluating the user's entry. */\nexport function _getSsrSnapshot(): RenderAppOptions | null {\n return (globalThis as GlobalWithSnapshot)[SNAPSHOT_KEY] ?? null;\n}\n\nexport function renderApp(options: RenderAppOptions) {\n if (typeof window === 'undefined') {\n setSnapshot(options);\n return;\n }\n\n const { loadingElement, routesElement, favicon, errorHandler, router } = options;\n\n if (errorHandler) {\n setErrorHandler(errorHandler);\n }\n\n window.addEventListener('unload', () => {\n // Empty 'unload' handler prevents bfcache in most browsers.\n });\n\n // ORDER MATTERS: hydrate session BEFORE building the React tree / calling\n // hydrateRoot, so `isSessionInitialized()` is true on the first render and\n // AppProvider's initial `isLoading` matches the server-rendered output.\n const ssrState = readSsrState();\n const isHydrating = ssrState !== null;\n if (ssrState?.session) {\n hydrateSession(ssrState.session);\n startSessionHeartbeat();\n }\n\n const container = document.getElementById('root')!;\n const routedTree = router ? router({ children: routesElement }) : routesElement;\n const tree = (\n <React.StrictMode>\n <AppProvider loadingElement={loadingElement}>\n <ModelenceQueryProvider>{routedTree}</ModelenceQueryProvider>\n </AppProvider>\n </React.StrictMode>\n );\n\n if (isHydrating) {\n ReactDOM.hydrateRoot(container, tree);\n } else {\n ReactDOM.createRoot(container).render(tree);\n }\n\n if (favicon) {\n const link = document.querySelector(\"link[rel~='icon']\") as HTMLLinkElement;\n if (!link) {\n const newLink = document.createElement('link');\n newLink.rel = 'icon';\n newLink.href = favicon;\n document.head.appendChild(newLink);\n } else {\n link.href = favicon;\n }\n }\n}\n"]}
@@ -0,0 +1,2 @@
1
+ import {c as c$1}from'./chunk-5M6FUMUK.js';import {a}from'./chunk-DO5TZLF5.js';import {create}from'zustand';import {z}from'zod';var b="modelence.session";function R(){return typeof globalThis<"u"&&typeof globalThis.localStorage<"u"}function U(){if(!R())return null;let e=globalThis.localStorage.getItem(b);try{return e?JSON.parse(e):null}catch(r){return console.error("Error parsing session from localStorage",r),null}}function g(e){R()&&globalThis.localStorage.setItem(b,JSON.stringify(e));}var C=(e,r)=>{console.error(`Error calling method '${r}':`,e);};function j(e){C=e;}function N(e,r){return C(e,r)}var m=class extends Error{constructor(r,t){super(r),this.name="MethodError",this.status=t;}},k=async(e,r)=>O(`/api/_internal/method/${e}`,r),u="__modelence_call_method_transport__";function A(){return globalThis[u]??k}function D(e){let r=globalThis[u];return globalThis[u]=e,()=>{globalThis[u]=r;}}async function o(e,r={},t={}){try{return await A()(e,r)}catch(n){throw (t.errorHandler??N)(n,e),n}}async function O(e,r){let t=await fetch(e,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify({args:r,authToken:h(),clientInfo:_()})});if(!t.ok){let a=await t.text();throw new m(a,t.status)}let n=await t.text(),s=n?JSON.parse(n):void 0;if(!s)throw new Error("Invalid response from server");return c$1(s.data,s.typeMap)}var P={},w=null;function q(e){w=e;}function $(e){return typeof window>"u"&&w?w(e):P[e]?.value}function y(e){P=e;}var d=create(e=>({user:null,setUser:r=>e({user:r})})),l=false,I=a.seconds(30),E=null,M=z.object({id:z.string(),handle:z.string(),roles:z.array(z.string()),firstName:z.string().optional(),lastName:z.string().optional(),avatarUrl:z.string().optional()});function c(e){if(!e)return null;let r=M.safeParse(e);if(!r.success)return console.error("Session Error: Invalid user payload",r.error),null;let t=r.data;return Object.freeze({...t,firstName:t.firstName??void 0,lastName:t.lastName??void 0,avatarUrl:t.avatarUrl??void 0,hasRole:n=>t.roles.includes(n),requireRole:n=>{if(!t.roles.includes(n))throw new Error(`Access denied - role '${n}' required`)}})}function ee(e){l||(l=true,y(e.configs),g(e.session),d.getState().setUser(c(e.user)));}async function re(){if(l)return;l=true;let{configs:e,session:r,user:t}=await o("_system.session.init");y(e),g(r),d.getState().setUser(c(t)),await T();}async function te(){typeof window>"u"||E!==null||await T();}async function T(){if(!(typeof window>"u")){try{await o("_system.session.heartbeat",{},{errorHandler:()=>{}});}catch{}E=setTimeout(T,I);}}function f(e){let r=c(e);return d.getState().setUser(r),r}function ne(){return l}var S=null;function oe(e){S=e;}function se(e){return c(e)}function ie(){return typeof window>"u"&&S?{user:S()}:{user:d(r=>r.user)}}async function ce(e){let{email:r,password:t,handle:n,firstName:s,lastName:a,avatarUrl:p}=e;await o("_system.user.signupWithPassword",{email:r,password:t,...n!==void 0?{handle:n}:{},...s!==void 0?{firstName:s}:{},...a!==void 0?{lastName:a}:{},...p!==void 0?{avatarUrl:p}:{}});}async function fe(e){let{email:r,password:t}=e,{user:n}=await o("_system.user.loginWithPassword",{email:r,password:t});return f(n)}async function pe(e){let{firstName:r,lastName:t,avatarUrl:n,handle:s}=e,{user:a}=await o("_system.user.updateProfile",{...r!==void 0?{firstName:r}:{},...t!==void 0?{lastName:t}:{},...n!==void 0?{avatarUrl:n}:{},...s!==void 0?{handle:s}:{}});return f(a)}async function ge(e){let{token:r}=e;await o("_system.user.verifyEmail",{token:r});}async function me(e){let{email:r}=e;await o("_system.user.resendEmailVerification",{email:r});}async function he(){await o("_system.user.logout"),f(null);}async function we(e){let{email:r}=e;await o("_system.user.sendResetPasswordToken",{email:r});}async function ye(e){let{token:r,password:t}=e;await o("_system.user.resetPassword",{token:r,password:t});}async function Se(e){let{provider:r}=e,t=h();if(t&&!(await fetch("/api/_internal/auth/set-link-cookie",{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify({authToken:t}),credentials:"include"})).ok)throw new Error("Failed to initialize OAuth linking. Please ensure you are logged in.");window.location.href=`/api/_internal/auth/${r}?mode=link`;}async function Te(e){let{provider:r}=e;await o("_system.user.unlinkOAuthProvider",{provider:r});}function h(){return U()?.authToken}function _(){return typeof window>"u"?{screenWidth:0,screenHeight:0,windowWidth:0,windowHeight:0,pixelRatio:1,orientation:null}:{screenWidth:window.screen.width,screenHeight:window.screen.height,windowWidth:window.innerWidth,windowHeight:window.innerHeight,pixelRatio:window.devicePixelRatio,orientation:window.screen.orientation?.type??null}}export{U as a,ce as b,fe as c,pe as d,ge as e,me as f,he as g,we as h,ye as i,Se as j,Te as k,h as l,_ as m,j as n,m as o,D as p,o as q,q as r,$ as s,ee as t,re as u,te as v,ne as w,oe as x,se as y,ie as z};//# sourceMappingURL=chunk-VFOGVWJK.js.map
2
+ //# sourceMappingURL=chunk-VFOGVWJK.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/client/localStorage.ts","../src/client/errorHandler.ts","../src/client/method.ts","../src/config/client.ts","../src/client/session.ts","../src/auth/client/index.ts"],"names":["SESSION_KEY","hasLocalStorage","getLocalStorageSession","sessionJson","e","setLocalStorageSession","session","errorHandler","error","methodName","setErrorHandler","handler","handleError","MethodError","message","status","defaultTransport","args","call","TRANSPORT_KEY","getTransport","_setCallMethodTransport","next","previous","callMethod","options","endpoint","response","getAuthToken","getClientInfo","text","result","reviveResponseTypes","config","ssrConfigResolver","_setSsrConfigResolver","resolver","getConfig","key","_setConfig","configs","useSessionStore","create","set","user","isInitialized","SESSION_HEARTBEAT_INTERVAL","time","heartbeatTimer","userSchema","z","parseUser","parsedData","role","hydrateSession","payload","initSession","loopSessionHeartbeat","startSessionHeartbeat","setCurrentUser","enrichedUser","isSessionInitialized","ssrSessionResolver","_setSsrSessionResolver","_parseSessionUser","useSession","state","signupWithPassword","email","password","handle","firstName","lastName","avatarUrl","loginWithPassword","updateProfile","verifyEmail","token","resendEmailVerification","logout","sendResetPasswordToken","resetPassword","linkOAuthProvider","provider","unlinkOAuthProvider"],"mappings":"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","file":"chunk-VFOGVWJK.js","sourcesContent":["const SESSION_KEY = 'modelence.session';\n\nfunction hasLocalStorage(): boolean {\n return typeof globalThis !== 'undefined' && typeof globalThis.localStorage !== 'undefined';\n}\n\nexport function getLocalStorageSession() {\n if (!hasLocalStorage()) {\n return null;\n }\n\n const sessionJson = globalThis.localStorage.getItem(SESSION_KEY);\n try {\n return sessionJson ? JSON.parse(sessionJson) : null;\n } catch (e) {\n console.error('Error parsing session from localStorage', e);\n return null;\n }\n}\n\nexport function setLocalStorageSession(session: object) {\n if (!hasLocalStorage()) {\n return;\n }\n\n globalThis.localStorage.setItem(SESSION_KEY, JSON.stringify(session));\n}\n","export type ErrorHandler = (error: Error, methodName: string) => void;\n\nlet errorHandler: ErrorHandler = (error, methodName) => {\n console.error(`Error calling method '${methodName}':`, error);\n};\n\nexport function setErrorHandler(handler: ErrorHandler) {\n errorHandler = handler;\n}\n\nexport function handleError(error: Error, methodName: string) {\n return errorHandler(error, methodName);\n}\n","/*\n The \"use client\" directive is specifically for the Next.js layout component, which is rendered on the server by default.\n Because of this, we are explicitly marking it as a client component, so we can render this component on the client\n and properly initialize config on the client side.\n \n While this is specific to Next.js, it is simply ignored outside of Next.js and should not cause errors.\n*/\n'use client';\n\nimport { getAuthToken, getClientInfo } from '@/auth/client';\nimport { handleError } from '@/client/errorHandler';\nimport { reviveResponseTypes } from '@/methods/serialize';\n\nexport class MethodError extends Error {\n status: number;\n\n constructor(message: string, status: number) {\n super(message);\n this.name = 'MethodError';\n this.status = status;\n }\n}\n\nexport type MethodArgs = Record<string, unknown>;\n\nexport type CallMethodOptions = {\n errorHandler?: (error: Error, methodName: string) => void;\n};\n\n// Defaults to fetch-based HTTP; the SSR runtime swaps in an in-process transport.\nexport type CallMethodTransport = <T = unknown>(methodName: string, args: MethodArgs) => Promise<T>;\n\nconst defaultTransport: CallMethodTransport = async <T>(methodName: string, args: MethodArgs) =>\n call<T>(`/api/_internal/method/${methodName}`, args);\n\n// Shared on globalThis so Vite's `ssrLoadModule` (which loads the user's tree\n// in a separate module graph from the framework runtime) sees the same\n// transport that the SSR runtime installed. Without this, `useQuery` calls\n// during SSR fall through to the default fetch-based transport and never\n// populate the dehydrated cache.\nconst TRANSPORT_KEY = '__modelence_call_method_transport__';\n\ntype GlobalWithTransport = typeof globalThis & {\n [TRANSPORT_KEY]?: CallMethodTransport;\n};\n\nfunction getTransport(): CallMethodTransport {\n return (globalThis as GlobalWithTransport)[TRANSPORT_KEY] ?? defaultTransport;\n}\n\n/** Returns a disposer that restores the previous transport. */\nexport function _setCallMethodTransport(next: CallMethodTransport): () => void {\n const previous = (globalThis as GlobalWithTransport)[TRANSPORT_KEY];\n (globalThis as GlobalWithTransport)[TRANSPORT_KEY] = next;\n return () => {\n (globalThis as GlobalWithTransport)[TRANSPORT_KEY] = previous;\n };\n}\n\nexport async function callMethod<T = unknown>(\n methodName: string,\n args: MethodArgs = {},\n options: CallMethodOptions = {}\n): Promise<T> {\n try {\n return await getTransport()<T>(methodName, args);\n } catch (error) {\n const handler = options.errorHandler ?? handleError;\n handler(error as Error, methodName);\n throw error;\n }\n}\n\nasync function call<T = unknown>(endpoint: string, args: MethodArgs): Promise<T> {\n const response = await fetch(endpoint, {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json',\n },\n body: JSON.stringify({\n args,\n authToken: getAuthToken(),\n clientInfo: getClientInfo(),\n }),\n });\n\n if (!response.ok) {\n const error = await response.text();\n throw new MethodError(error, response.status);\n }\n\n const text = await response.text();\n const result = text ? JSON.parse(text) : undefined;\n if (!result) {\n throw new Error('Invalid response from server');\n }\n\n return reviveResponseTypes(result.data, result.typeMap);\n}\n","import { AppConfig, ConfigKey, Configs } from './types';\n\nlet config: Record<ConfigKey, AppConfig> = {};\n\ntype SsrConfigResolver = (key: ConfigKey) => string | number | boolean | undefined;\nlet ssrConfigResolver: SsrConfigResolver | null = null;\n\n/**\n * @internal\n * Installed once by the SSR runtime. The resolver reads per-request configs\n * from AsyncLocalStorage, so this single global is safe under concurrency.\n */\nexport function _setSsrConfigResolver(resolver: SsrConfigResolver | null) {\n ssrConfigResolver = resolver;\n}\n\n/**\n * @sidebarTitle getConfig (client)\n *\n * @param key\n * @returns\n */\nexport function getConfig(key: ConfigKey): string | number | boolean | undefined {\n if (typeof window === 'undefined' && ssrConfigResolver) {\n return ssrConfigResolver(key);\n }\n return config[key]?.value;\n}\n\nexport function _setConfig(configs: Configs) {\n config = configs;\n}\n","import { create } from 'zustand';\nimport { z } from 'zod';\nimport { callMethod } from './method';\nimport { _setConfig } from '../config/client';\nimport { setLocalStorageSession } from './localStorage';\nimport { time } from '../time';\nimport { Configs } from '../config/types';\n\ntype User = {\n id: string;\n handle: string;\n roles: string[];\n hasRole: (role: string) => boolean;\n requireRole: (role: string) => void;\n firstName?: string;\n lastName?: string;\n avatarUrl?: string;\n};\n\ntype SessionStore = {\n user: User | null;\n setUser: (user: User | null) => void;\n};\n\nexport const useSessionStore = create<SessionStore>((set) => ({\n user: null,\n setUser: (user) => set({ user }),\n}));\n\nlet isInitialized = false;\nconst SESSION_HEARTBEAT_INTERVAL = time.seconds(30);\nlet heartbeatTimer: ReturnType<typeof setTimeout> | null = null;\n\nconst userSchema = z.object({\n id: z.string(),\n handle: z.string(),\n roles: z.array(z.string()),\n firstName: z.string().optional(),\n lastName: z.string().optional(),\n avatarUrl: z.string().optional(),\n});\n\nfunction parseUser(user: unknown): User | null {\n if (!user) {\n return null;\n }\n\n const result = userSchema.safeParse(user);\n\n if (!result.success) {\n console.error('Session Error: Invalid user payload', result.error);\n return null;\n }\n\n const parsedData = result.data;\n\n return Object.freeze({\n ...parsedData,\n firstName: parsedData.firstName ?? undefined,\n lastName: parsedData.lastName ?? undefined,\n avatarUrl: parsedData.avatarUrl ?? undefined,\n hasRole: (role: string) => parsedData.roles.includes(role),\n requireRole: (role: string) => {\n if (!parsedData.roles.includes(role)) {\n throw new Error(`Access denied - role '${role}' required`);\n }\n },\n });\n}\n\nexport type SessionInitPayload = {\n configs: Configs;\n session: object;\n user: object;\n};\n\n/** Hydrate session state from the SSR payload, skipping the network round-trip. */\nexport function hydrateSession(payload: SessionInitPayload) {\n if (isInitialized) {\n return;\n }\n\n isInitialized = true;\n\n _setConfig(payload.configs);\n setLocalStorageSession(payload.session);\n useSessionStore.getState().setUser(parseUser(payload.user));\n}\n\nexport async function initSession() {\n if (isInitialized) {\n return;\n }\n\n isInitialized = true;\n\n const { configs, session, user } = await callMethod<SessionInitPayload>('_system.session.init');\n _setConfig(configs);\n setLocalStorageSession(session);\n\n useSessionStore.getState().setUser(parseUser(user));\n\n await loopSessionHeartbeat();\n}\n\n/** Idempotent, client-only. Auto-started by `initSession`; call explicitly after `hydrateSession`. */\nexport async function startSessionHeartbeat() {\n if (typeof window === 'undefined' || heartbeatTimer !== null) {\n return;\n }\n await loopSessionHeartbeat();\n}\n\nasync function loopSessionHeartbeat() {\n if (typeof window === 'undefined') {\n return;\n }\n\n try {\n await callMethod('_system.session.heartbeat', {}, { errorHandler: () => {} });\n } catch {\n // Silently ignore heartbeat errors - they're expected during HMR/reconnects\n }\n heartbeatTimer = setTimeout(loopSessionHeartbeat, SESSION_HEARTBEAT_INTERVAL);\n}\n\nexport function setCurrentUser(user: unknown) {\n const enrichedUser = parseUser(user);\n useSessionStore.getState().setUser(enrichedUser);\n return enrichedUser;\n}\n\nexport function isSessionInitialized() {\n return isInitialized;\n}\n\nexport function getHeartbeatTimer() {\n return heartbeatTimer;\n}\n\nexport function stopHeartbeatTimer() {\n if (heartbeatTimer) {\n clearTimeout(heartbeatTimer);\n heartbeatTimer = null;\n }\n}\n\ntype SsrSessionResolver = () => User | null;\nlet ssrSessionResolver: SsrSessionResolver | null = null;\n\n/**\n * @internal\n * Installed once by the SSR runtime. The resolver reads per-request state\n * from AsyncLocalStorage, so this single global is safe under concurrency.\n */\nexport function _setSsrSessionResolver(resolver: SsrSessionResolver | null) {\n ssrSessionResolver = resolver;\n}\n\n/** @internal Used by the SSR runtime to enrich a raw user payload. */\nexport function _parseSessionUser(user: unknown): User | null {\n return parseUser(user);\n}\n\n/**\n * `useSession` is a hook that returns the current user, and in the future will also return other details about the current session.\n *\n * @example\n * ```ts\n * import { useSession } from 'modelence/client';\n *\n * function MyComponent() {\n * const { user } = useSession();\n * return <div>{user?.handle}</div>;\n * }\n * ```\n */\nexport function useSession() {\n if (typeof window === 'undefined' && ssrSessionResolver) {\n return { user: ssrSessionResolver() };\n }\n const user = useSessionStore((state) => state.user);\n return { user };\n}\n","import { setCurrentUser } from '@/client/session';\nimport { callMethod } from '@/client/method';\nimport { getLocalStorageSession } from '@/client/localStorage';\nimport { ClientInfo } from '@/methods/types';\nimport { OAuthProvider } from '../types';\n\nexport type UserInfo = {\n id: string;\n handle: string;\n roles: string[];\n hasRole: (role: string) => boolean;\n requireRole: (role: string) => void;\n firstName?: string;\n lastName?: string;\n avatarUrl?: string;\n};\n\ntype RawUserData = {\n id: string;\n handle: string;\n roles: string[];\n firstName?: string;\n lastName?: string;\n avatarUrl?: string;\n};\n\n/**\n * Sign up a new user with an email and password.\n *\n * @example\n * ```ts\n * await signupWithPassword({ email: 'test@example.com', password: '12345678' });\n * await signupWithPassword({ email: 'test@example.com', password: '12345678', handle: 'myhandle', firstName: 'John' });\n * ```\n * @param options.email - The email of the user.\n * @param options.password - The password of the user.\n * @param options.handle - Optional custom handle. If omitted, one is derived from the email.\n * @param options.firstName - Optional first name.\n * @param options.lastName - Optional last name.\n * @param options.avatarUrl - Optional avatar URL.\n */\nexport async function signupWithPassword(options: {\n email: string;\n password: string;\n handle?: string;\n firstName?: string;\n lastName?: string;\n avatarUrl?: string;\n}) {\n const { email, password, handle, firstName, lastName, avatarUrl } = options;\n await callMethod('_system.user.signupWithPassword', {\n email,\n password,\n ...(handle !== undefined ? { handle } : {}),\n ...(firstName !== undefined ? { firstName } : {}),\n ...(lastName !== undefined ? { lastName } : {}),\n ...(avatarUrl !== undefined ? { avatarUrl } : {}),\n });\n}\n\n/**\n * Login a user with an email and password.\n *\n * @example\n * ```ts\n * await loginWithPassword({ email: 'test@example.com', password: '12345678' });\n * ```\n * @param options.email - The email of the user.\n * @param options.password - The password of the user.\n */\nexport async function loginWithPassword(options: { email: string; password: string }) {\n const { email, password } = options;\n const { user } = await callMethod<{ user: RawUserData }>('_system.user.loginWithPassword', {\n email,\n password,\n });\n const enrichedUser = setCurrentUser(user);\n return enrichedUser;\n}\n\n/**\n * Update the current user's profile.\n *\n * @example\n * ```ts\n * await updateProfile({ firstName: 'Atul', lastName: 'Yadav', avatarUrl: 'https://example.com/avatar.jpg', handle: 'atulyadav' });\n * ```\n * @param options.firstName - The first name of the user.\n * @param options.lastName - The last name of the user.\n * @param options.avatarUrl - The avatar URL of the user.\n * @param options.handle - The handle of the user.\n */\nexport async function updateProfile(options: {\n firstName?: string;\n lastName?: string;\n avatarUrl?: string;\n handle?: string;\n}) {\n const { firstName, lastName, avatarUrl, handle } = options;\n const { user } = await callMethod<{ user: RawUserData }>('_system.user.updateProfile', {\n ...(firstName !== undefined ? { firstName } : {}),\n ...(lastName !== undefined ? { lastName } : {}),\n ...(avatarUrl !== undefined ? { avatarUrl } : {}),\n ...(handle !== undefined ? { handle } : {}),\n });\n const enrichedUser = setCurrentUser(user);\n return enrichedUser;\n}\n\n/**\n * Verify user's email with a verification token.\n *\n * @example\n * ```ts\n * await verifyEmail({ token: 'verification-token' });\n * ```\n * @param options.token - The email verification token.\n */\nexport async function verifyEmail(options: { token: string }) {\n const { token } = options;\n await callMethod<{ user: RawUserData }>('_system.user.verifyEmail', { token });\n}\n\n/**\n * Resend the verification email for a given email address.\n * The email is only sent if the address is registered and not yet verified.\n * A generic response is always returned to avoid leaking account information.\n *\n * @example\n * ```ts\n * await resendEmailVerification({ email: 'user@example.com' });\n * ```\n * @param options.email - The email address to resend verification to.\n */\nexport async function resendEmailVerification(options: { email: string }) {\n const { email } = options;\n await callMethod('_system.user.resendEmailVerification', { email });\n}\n\n/**\n * Logout the current user.\n *\n */\nexport async function logout() {\n await callMethod('_system.user.logout');\n setCurrentUser(null);\n}\n\n/**\n * Send reset password token.\n * @param options.email - The email of the user.\n */\nexport async function sendResetPasswordToken(options: { email: string }) {\n const { email } = options;\n await callMethod('_system.user.sendResetPasswordToken', {\n email,\n });\n}\n\n/**\n * Reset password.\n * @param options.token - The password reset token.\n * @param options.password - The new password.\n */\nexport async function resetPassword(options: { token: string; password: string }) {\n const { token, password } = options;\n await callMethod('_system.user.resetPassword', {\n token,\n password,\n });\n}\n\n/**\n * Link an OAuth provider to the currently signed-in user's account.\n * Redirects the browser to the OAuth provider's authorization page.\n * The provider will redirect back and the account will be linked.\n *\n * @example\n * ```ts\n * linkOAuthProvider({ provider: 'google' });\n * ```\n * @param options.provider - The OAuth provider to link ('google' or 'github').\n */\nexport async function linkOAuthProvider(options: { provider: OAuthProvider }): Promise<void> {\n const { provider } = options;\n\n const token = getAuthToken();\n if (token) {\n // Ask the server to set a secure httpOnly cookie for the OAuth linking flow\n const response = await fetch('/api/_internal/auth/set-link-cookie', {\n method: 'POST',\n headers: { 'Content-Type': 'application/json' },\n body: JSON.stringify({ authToken: token }),\n credentials: 'include',\n });\n if (!response.ok) {\n throw new Error('Failed to initialize OAuth linking. Please ensure you are logged in.');\n }\n }\n window.location.href = `/api/_internal/auth/${provider}?mode=link`;\n}\n/**\n * Unlink an OAuth provider from the currently signed-in user's account.\n *\n * @example\n * ```ts\n * await unlinkOAuthProvider({ provider: 'github' });\n * ```\n * @param options.provider - The OAuth provider to unlink ('google' or 'github').\n */\nexport async function unlinkOAuthProvider(options: { provider: OAuthProvider }): Promise<void> {\n const { provider } = options;\n await callMethod('_system.user.unlinkOAuthProvider', { provider });\n}\n\n/**\n * Get the current auth token associated with the current session.\n * @returns The auth token or undefined if not authenticated.\n */\nexport function getAuthToken(): string | undefined {\n return getLocalStorageSession()?.authToken;\n}\n\nexport function getClientInfo(): ClientInfo {\n if (typeof window === 'undefined') {\n return {\n screenWidth: 0,\n screenHeight: 0,\n windowWidth: 0,\n windowHeight: 0,\n pixelRatio: 1,\n orientation: null,\n };\n }\n\n return {\n screenWidth: window.screen.width,\n screenHeight: window.screen.height,\n windowWidth: window.innerWidth,\n windowHeight: window.innerHeight,\n pixelRatio: window.devicePixelRatio,\n orientation: window.screen.orientation?.type ?? null,\n };\n}\n"]}
package/dist/client.d.ts CHANGED
@@ -1,8 +1,11 @@
1
- import { b as ConfigKey, c as ConfigParams, A as AnyMethodShape, V as ValueType, d as ConfigType, O as OAuthProvider, W as WebsocketClientProvider, e as ClientChannel } from './types-CTjq1HaP.js';
1
+ import { b as ConfigKey, c as ConfigParams, A as AnyMethodShape, V as ValueType, d as ConfigType, O as OAuthProvider, W as WebsocketClientProvider, e as ClientChannel } from './types-DV0J5rPI.js';
2
2
  import { ObjectId } from 'mongodb';
3
- import React from 'react';
3
+ import React, { ReactNode } from 'react';
4
+ import * as react_jsx_runtime from 'react/jsx-runtime';
5
+ import { QueryClient, DehydratedState } from '@tanstack/react-query';
4
6
  import { F as FileVisibility } from './types-DLDzAym7.js';
5
7
  import 'http';
8
+ import 'express';
6
9
 
7
10
  /**
8
11
  * @sidebarTitle getConfig (client)
@@ -127,12 +130,68 @@ declare const systemConfig: {
127
130
 
128
131
  type ErrorHandler = (error: Error, methodName: string) => void;
129
132
 
130
- declare function renderApp({ loadingElement, routesElement, favicon, errorHandler, }: {
133
+ type SsrRouter = (props: {
134
+ children: React.ReactNode;
135
+ location?: string;
136
+ }) => React.ReactElement;
137
+ interface RenderAppOptions {
131
138
  loadingElement: React.ReactNode;
132
139
  routesElement: React.ReactNode;
133
140
  favicon?: string;
134
141
  errorHandler?: ErrorHandler;
135
- }): void;
142
+ router?: SsrRouter;
143
+ }
144
+ declare function renderApp(options: RenderAppOptions): void;
145
+
146
+ interface ModelenceQueryProviderProps {
147
+ children: ReactNode;
148
+ client?: QueryClient;
149
+ dehydratedState?: DehydratedState;
150
+ }
151
+ declare function ModelenceQueryProvider({ children, client, dehydratedState, }: ModelenceQueryProviderProps): react_jsx_runtime.JSX.Element;
152
+
153
+ type Args = Record<string, unknown>;
154
+ /**
155
+ * Connects a `QueryClient` to Modelence's live-query layer. Auto-called by
156
+ * `<ModelenceQueryProvider>`; only call manually if you mount your own
157
+ * `<QueryClientProvider>`.
158
+ */
159
+ declare function connectModelenceQueryClient(queryClient: QueryClient): void;
160
+ declare function disconnectModelenceQueryClient(): void;
161
+ /** @deprecated Use `connectModelenceQueryClient(queryClient)` instead. */
162
+ declare class ModelenceQueryClient {
163
+ connect(queryClient: QueryClient): void;
164
+ }
165
+ /**
166
+ * @example
167
+ * ```tsx
168
+ * const { data } = useQuery(modelenceQuery('todo.getAll'));
169
+ * ```
170
+ */
171
+ declare function modelenceQuery<T = unknown>(methodName: string, args?: Args): {
172
+ queryKey: (string | Args)[];
173
+ queryFn: () => Promise<T>;
174
+ };
175
+ /**
176
+ * Live query — data updates in real time as the underlying collection changes.
177
+ * Requires a `QueryClient` connected via `<ModelenceQueryProvider>` or
178
+ * `connectModelenceQueryClient(...)`.
179
+ */
180
+ declare function modelenceLiveQuery<T = unknown>(methodName: string, args?: Args): {
181
+ queryKey: readonly ["live", string, Args];
182
+ queryFn: () => Promise<T>;
183
+ staleTime: number;
184
+ refetchOnWindowFocus: boolean;
185
+ refetchOnMount: boolean;
186
+ refetchOnReconnect: boolean;
187
+ gcTime: number;
188
+ };
189
+ declare function modelenceMutation<T = unknown>(methodName: string, defaultArgs?: Args): {
190
+ mutationFn: (variables?: Args) => Promise<T>;
191
+ };
192
+ type ModelenceQueryKey<T extends string, U extends Args = Args> = readonly [T, U];
193
+ /** Builds a query key matching `modelenceQuery(...)` for cache operations. */
194
+ declare function createQueryKey<T extends string, U extends Args = Args>(methodName: T, args?: U): ModelenceQueryKey<T, U>;
136
195
 
137
196
  type User = {
138
197
  id: string;
@@ -348,4 +407,4 @@ declare function getFileUrl(filePath: string): Promise<{
348
407
 
349
408
  declare const AppProvider: any;
350
409
 
351
- export { AppProvider, ClientChannel, type MethodArgs, MethodError, type UserInfo, ValueType, callMethod, createClientModule, deleteFile, downloadFile, getConfig, getFileUrl, getLocalStorageSession, getWebsocketClientProvider, linkOAuthProvider, loginWithPassword, logout, renderApp, resendEmailVerification, resetPassword, sendResetPasswordToken, setWebsocketClientProvider, signupWithPassword, startWebsockets, subscribeLiveQuery, systemConfig, unlinkOAuthProvider, updateProfile, uploadFile, useSession, verifyEmail };
410
+ export { AppProvider, ClientChannel, type MethodArgs, MethodError, ModelenceQueryClient, type ModelenceQueryKey, ModelenceQueryProvider, type UserInfo, ValueType, callMethod, connectModelenceQueryClient, createClientModule, createQueryKey, deleteFile, disconnectModelenceQueryClient, downloadFile, getConfig, getFileUrl, getLocalStorageSession, getWebsocketClientProvider, linkOAuthProvider, loginWithPassword, logout, modelenceLiveQuery, modelenceMutation, modelenceQuery, renderApp, resendEmailVerification, resetPassword, sendResetPasswordToken, setWebsocketClientProvider, signupWithPassword, startWebsockets, subscribeLiveQuery, systemConfig, unlinkOAuthProvider, updateProfile, uploadFile, useSession, verifyEmail };
package/dist/client.js CHANGED
@@ -1,2 +1,2 @@
1
- import {c as c$1}from'./chunk-5M6FUMUK.js';import {a}from'./chunk-DO5TZLF5.js';import V,{useState,useEffect}from'react';import {create}from'zustand';import {z}from'zod';import {jsx}from'react/jsx-runtime';import me from'react-dom/client';import he from'socket.io-client';function T(){let e=localStorage.getItem("modelence.session");try{return e?JSON.parse(e):null}catch(t){return console.error("Error parsing session from localStorage",t),null}}function K(e){localStorage.setItem("modelence.session",JSON.stringify(e));}async function J(e){let{email:t,password:r,handle:n,firstName:o,lastName:s,avatarUrl:l}=e;await i("_system.user.signupWithPassword",{email:t,password:r,...n!==void 0?{handle:n}:{},...o!==void 0?{firstName:o}:{},...s!==void 0?{lastName:s}:{},...l!==void 0?{avatarUrl:l}:{}});}async function B(e){let{email:t,password:r}=e,{user:n}=await i("_system.user.loginWithPassword",{email:t,password:r});return w(n)}async function G(e){let{firstName:t,lastName:r,avatarUrl:n,handle:o}=e,{user:s}=await i("_system.user.updateProfile",{...t!==void 0?{firstName:t}:{},...r!==void 0?{lastName:r}:{},...n!==void 0?{avatarUrl:n}:{},...o!==void 0?{handle:o}:{}});return w(s)}async function X(e){let{token:t}=e;await i("_system.user.verifyEmail",{token:t});}async function Y(e){let{email:t}=e;await i("_system.user.resendEmailVerification",{email:t});}async function Z(){await i("_system.user.logout"),w(null);}async function ee(e){let{email:t}=e;await i("_system.user.sendResetPasswordToken",{email:t});}async function te(e){let{token:t,password:r}=e;await i("_system.user.resetPassword",{token:t,password:r});}async function re(e){let{provider:t}=e,r=c();if(r&&!(await fetch("/api/_internal/auth/set-link-cookie",{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify({authToken:r}),credentials:"include"})).ok)throw new Error("Failed to initialize OAuth linking. Please ensure you are logged in.");window.location.href=`/api/_internal/auth/${t}?mode=link`;}async function ne(e){let{provider:t}=e;await i("_system.user.unlinkOAuthProvider",{provider:t});}function c(){return T()?.authToken}function p(){return {screenWidth:window.screen.width,screenHeight:window.screen.height,windowWidth:window.innerWidth,windowHeight:window.innerHeight,pixelRatio:window.devicePixelRatio,orientation:window.screen.orientation?.type??null}}var N=(e,t)=>{console.error(`Error calling method '${t}':`,e);};function I(e){N=e;}function O(e,t){return N(e,t)}var x=class extends Error{constructor(t,r){super(t),this.name="MethodError",this.status=r;}};async function i(e,t={},r={}){try{return await oe(`/api/_internal/method/${e}`,t)}catch(n){throw (r.errorHandler??O)(n,e),n}}async function oe(e,t){let r=await fetch(e,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify({args:t,authToken:c(),clientInfo:p()})});if(!r.ok){let s=await r.text();throw new x(s,r.status)}let n=await r.text(),o=n?JSON.parse(n):void 0;if(!o)throw new Error("Invalid response from server");return c$1(o.data,o.typeMap)}var W={};function k(e){return W[e]?.value}function F(e){W=e;}var b=create(e=>({user:null,setUser:t=>e({user:t})})),q=false,se=a.seconds(30),le=z.object({id:z.string(),handle:z.string(),roles:z.array(z.string()),firstName:z.string().optional(),lastName:z.string().optional(),avatarUrl:z.string().optional()});function _(e){if(!e)return null;let t=le.safeParse(e);if(!t.success)return console.error("Session Error: Invalid user payload",t.error),null;let r=t.data;return Object.freeze({...r,firstName:r.firstName??void 0,lastName:r.lastName??void 0,avatarUrl:r.avatarUrl??void 0,hasRole:n=>r.roles.includes(n),requireRole:n=>{if(!r.roles.includes(n))throw new Error(`Access denied - role '${n}' required`)}})}async function L(){if(q)return;q=true;let{configs:e,session:t,user:r}=await i("_system.session.init");F(e),K(t),b.getState().setUser(_(r)),await H();}async function H(){try{await i("_system.session.heartbeat",{},{errorHandler:()=>{}});}catch{}setTimeout(H,se);}function w(e){let t=_(e);return b.getState().setUser(t),t}function de(){return {user:b(t=>t.user)}}var $=false;function C({children:e,loadingElement:t}){let[r,n]=useState(true);return useEffect(()=>{async function o(){$||($=true,await L(),n(false));}o();},[]),r?t??jsx("div",{children:"Loading..."}):e}function S(e){return {getConfig(t){return k(`${e}.${t}`)},query(t,...r){let n=r[0]??{};return {queryKey:[e,t,n],queryFn:()=>i(`${e}.${t}`,n)}},mutation(t){return {mutationFn:r=>i(`${e}.${t}`,r)}},infiniteQuery(t,r){return {queryKey:[e,t,"infinite",r(void 0)],initialPageParam:void 0,queryFn:({pageParam:n})=>i(`${e}.${t}`,r(n))}}}}var pe=S("_system");function ye({loadingElement:e,routesElement:t,favicon:r,errorHandler:n}){if(n&&I(n),window.addEventListener("unload",()=>{}),me.createRoot(document.getElementById("root")).render(jsx(V.StrictMode,{children:jsx(D,{loadingElement:e,children:t})})),r){let o=document.querySelector("link[rel~='icon']");if(o)o.href=r;else {let s=document.createElement("link");s.rel="icon",s.href=r,document.head.appendChild(s);}}}var g=null,m=new Map;function y(){if(!g)throw new Error("WebSocket not initialized. Call startWebsockets() first.");return g}function ve(){let e=c(),t=p();for(let r of m.values())g?.emit("subscribeLiveQuery",{subscriptionId:r.subscriptionId,method:r.method,args:r.args,authToken:e,clientInfo:t});}function we(e){g=he("/",{transports:["websocket"],auth:{token:c()}}),g.on("connect",()=>{m.size>0&&(console.log(`[Modelence] WebSocket reconnected, re-subscribing to ${m.size} live queries`),ve());}),e.channels?.forEach(t=>t.init());}function xe({category:e,listener:t}){y().on(e,t);}function Pe({category:e,listener:t}){y().once(e,t);}function Te({category:e,listener:t}){y().off(e,t);}function M({eventName:e,category:t,id:r}){y().emit(e,`${t}:${r}`);}function ke({category:e,id:t}){M({eventName:"joinChannel",category:e,id:t});}function be({category:e,id:t}){M({eventName:"leaveChannel",category:e,id:t});}var Ce=0;function j(e,t,r,n){let o=`sub-${++Ce}-${Date.now()}`,s=({subscriptionId:u,data:f,typeMap:P})=>{u===o&&r(c$1(f,P));},l=({subscriptionId:u,error:f})=>{u===o&&(console.error(`[Modelence] Live query error for ${e}:`,f),n?.(f));},a=y();return a.on("liveQueryData",s),a.on("liveQueryError",l),m.set(o,{subscriptionId:o,method:e,args:t}),a.connected&&a.emit("subscribeLiveQuery",{subscriptionId:o,method:e,args:t,authToken:c(),clientInfo:p()}),()=>{m.delete(o),a.emit("unsubscribeLiveQuery",{subscriptionId:o}),a.off("liveQueryData",s),a.off("liveQueryError",l);}}var Se={init:we,on:xe,once:Pe,off:Te,emit:M,joinChannel:ke,leaveChannel:be},Q=Se;var R=null;function E(e){R=e;}function h(){return R}function Me(e){if(R){console.warn("WebSocket already initialized. Skipping initialization.");return}let t=e?.provider||Q;E(t);try{t.init({channels:e?.channels});}catch(r){throw E(null),r}}var A=class{constructor(t,r){this.category=t,this.onMessage=r;}init(){h()?.on({category:this.category,listener:this.onMessage});}joinChannel(t){h()?.joinChannel({category:this.category,id:t});}leaveChannel(t){h()?.leaveChannel({category:this.category,id:t});}};async function Ee(e,{filePath:t,contentType:r,visibility:n}){let{url:o,fields:s,filePath:l}=await i("_system.files.getUploadUrl",{filePath:t,contentType:r,visibility:n}),a=new FormData;for(let[f,P]of Object.entries(s))a.append(f,P);a.append("file",e);let u=await fetch(o,{method:"POST",body:a});if(!u.ok)throw new Error(`Failed to upload file: HTTP status: ${u.status}`);return {filePath:l}}async function Re(e){await i("_system.files.deleteFile",{filePath:e});}async function Ae(e){return i("_system.files.downloadFile",{filePath:e})}async function Ue(e){return i("_system.files.getFileUrl",{filePath:e})}var D="useClient"in V?V.useClient(C):C;export{D as AppProvider,A as ClientChannel,x as MethodError,i as callMethod,S as createClientModule,Re as deleteFile,Ae as downloadFile,k as getConfig,Ue as getFileUrl,T as getLocalStorageSession,h as getWebsocketClientProvider,re as linkOAuthProvider,B as loginWithPassword,Z as logout,ye as renderApp,Y as resendEmailVerification,te as resetPassword,ee as sendResetPasswordToken,E as setWebsocketClientProvider,J as signupWithPassword,Me as startWebsockets,j as subscribeLiveQuery,pe as systemConfig,ne as unlinkOAuthProvider,G as updateProfile,Ee as uploadFile,de as useSession,X as verifyEmail};//# sourceMappingURL=client.js.map
1
+ export{j as AppProvider,e as ClientChannel,a as createClientModule,g as deleteFile,h as downloadFile,i as getFileUrl,d as renderApp,b as systemConfig,f as uploadFile}from'./chunk-S77LRZU6.js';export{h as ModelenceQueryClient,m as ModelenceQueryProvider,f as connectModelenceQueryClient,l as createQueryKey,g as disconnectModelenceQueryClient,d as getWebsocketClientProvider,j as modelenceLiveQuery,k as modelenceMutation,i as modelenceQuery,c as setWebsocketClientProvider,e as startWebsockets,b as subscribeLiveQuery}from'./chunk-IJ5BS7PM.js';export{o as MethodError,q as callMethod,s as getConfig,a as getLocalStorageSession,j as linkOAuthProvider,c as loginWithPassword,g as logout,f as resendEmailVerification,i as resetPassword,h as sendResetPasswordToken,b as signupWithPassword,k as unlinkOAuthProvider,d as updateProfile,z as useSession,e as verifyEmail}from'./chunk-VFOGVWJK.js';import'./chunk-5M6FUMUK.js';import'./chunk-DO5TZLF5.js';//# sourceMappingURL=client.js.map
2
2
  //# sourceMappingURL=client.js.map
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/client/localStorage.ts","../src/auth/client/index.ts","../src/client/errorHandler.ts","../src/client/method.ts","../src/config/client.ts","../src/client/session.ts","../src/client/AppProvider.tsx","../src/client/module.ts","../src/system/client.ts","../src/client/renderApp.tsx","../src/websocket/socketio/client.ts","../src/websocket/client.ts","../src/websocket/clientChannel.ts","../src/files/client.ts","../src/client.ts"],"names":["getLocalStorageSession","sessionJson","e","setLocalStorageSession","session","signupWithPassword","options","email","password","handle","firstName","lastName","avatarUrl","callMethod","loginWithPassword","user","setCurrentUser","updateProfile","verifyEmail","token","resendEmailVerification","logout","sendResetPasswordToken","resetPassword","linkOAuthProvider","provider","getAuthToken","unlinkOAuthProvider","getClientInfo","errorHandler","error","methodName","setErrorHandler","handler","handleError","MethodError","message","status","args","call","endpoint","response","text","result","reviveResponseTypes","config","getConfig","key","_setConfig","configs","useSessionStore","create","set","isInitialized","SESSION_HEARTBEAT_INTERVAL","time","userSchema","z","parseUser","parsedData","role","initSession","loopSessionHeartbeat","enrichedUser","useSession","state","AppProvider","children","loadingElement","isLoading","setIsLoading","useState","useEffect","initConfig","jsx","createClientModule","moduleName","name","rest","getArgs","pageParam","systemConfig","renderApp","routesElement","favicon","ReactDOM","React","link","newLink","socketClient","activeLiveSubscriptions","getSocket","resubscribeAll","authToken","clientInfo","sub","init","props","io","channel","on","category","listener","once","off","emit","eventName","id","joinChannel","leaveChannel","liveQueryCounter","subscribeLiveQuery","method","onData","onError","subscriptionId","handleData","sid","data","typeMap","socket","websocketProvider","client_default","websocketClientProvider","setWebsocketClientProvider","getWebsocketClientProvider","startWebsockets","ClientChannel","onMessage","uploadFile","file","filePath","contentType","visibility","url","fields","resolvedFilePath","formData","value","uploadResponse","deleteFile","downloadFile","getFileUrl"],"mappings":"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","file":"client.js","sourcesContent":["export function getLocalStorageSession() {\n const sessionJson = localStorage.getItem('modelence.session');\n try {\n return sessionJson ? JSON.parse(sessionJson) : null;\n } catch (e) {\n console.error('Error parsing session from localStorage', e);\n return null;\n }\n}\n\nexport function setLocalStorageSession(session: object) {\n localStorage.setItem('modelence.session', JSON.stringify(session));\n}\n","import { setCurrentUser } from '@/client/session';\nimport { callMethod } from '@/client/method';\nimport { getLocalStorageSession } from '@/client/localStorage';\nimport { ClientInfo } from '@/methods/types';\nimport { OAuthProvider } from '../types';\n\nexport type UserInfo = {\n id: string;\n handle: string;\n roles: string[];\n hasRole: (role: string) => boolean;\n requireRole: (role: string) => void;\n firstName?: string;\n lastName?: string;\n avatarUrl?: string;\n};\n\ntype RawUserData = {\n id: string;\n handle: string;\n roles: string[];\n firstName?: string;\n lastName?: string;\n avatarUrl?: string;\n};\n\n/**\n * Sign up a new user with an email and password.\n *\n * @example\n * ```ts\n * await signupWithPassword({ email: 'test@example.com', password: '12345678' });\n * await signupWithPassword({ email: 'test@example.com', password: '12345678', handle: 'myhandle', firstName: 'John' });\n * ```\n * @param options.email - The email of the user.\n * @param options.password - The password of the user.\n * @param options.handle - Optional custom handle. If omitted, one is derived from the email.\n * @param options.firstName - Optional first name.\n * @param options.lastName - Optional last name.\n * @param options.avatarUrl - Optional avatar URL.\n */\nexport async function signupWithPassword(options: {\n email: string;\n password: string;\n handle?: string;\n firstName?: string;\n lastName?: string;\n avatarUrl?: string;\n}) {\n const { email, password, handle, firstName, lastName, avatarUrl } = options;\n await callMethod('_system.user.signupWithPassword', {\n email,\n password,\n ...(handle !== undefined ? { handle } : {}),\n ...(firstName !== undefined ? { firstName } : {}),\n ...(lastName !== undefined ? { lastName } : {}),\n ...(avatarUrl !== undefined ? { avatarUrl } : {}),\n });\n}\n\n/**\n * Login a user with an email and password.\n *\n * @example\n * ```ts\n * await loginWithPassword({ email: 'test@example.com', password: '12345678' });\n * ```\n * @param options.email - The email of the user.\n * @param options.password - The password of the user.\n */\nexport async function loginWithPassword(options: { email: string; password: string }) {\n const { email, password } = options;\n const { user } = await callMethod<{ user: RawUserData }>('_system.user.loginWithPassword', {\n email,\n password,\n });\n const enrichedUser = setCurrentUser(user);\n return enrichedUser;\n}\n\n/**\n * Update the current user's profile.\n *\n * @example\n * ```ts\n * await updateProfile({ firstName: 'Atul', lastName: 'Yadav', avatarUrl: 'https://example.com/avatar.jpg', handle: 'atulyadav' });\n * ```\n * @param options.firstName - The first name of the user.\n * @param options.lastName - The last name of the user.\n * @param options.avatarUrl - The avatar URL of the user.\n * @param options.handle - The handle of the user.\n */\nexport async function updateProfile(options: {\n firstName?: string;\n lastName?: string;\n avatarUrl?: string;\n handle?: string;\n}) {\n const { firstName, lastName, avatarUrl, handle } = options;\n const { user } = await callMethod<{ user: RawUserData }>('_system.user.updateProfile', {\n ...(firstName !== undefined ? { firstName } : {}),\n ...(lastName !== undefined ? { lastName } : {}),\n ...(avatarUrl !== undefined ? { avatarUrl } : {}),\n ...(handle !== undefined ? { handle } : {}),\n });\n const enrichedUser = setCurrentUser(user);\n return enrichedUser;\n}\n\n/**\n * Verify user's email with a verification token.\n *\n * @example\n * ```ts\n * await verifyEmail({ token: 'verification-token' });\n * ```\n * @param options.token - The email verification token.\n */\nexport async function verifyEmail(options: { token: string }) {\n const { token } = options;\n await callMethod<{ user: RawUserData }>('_system.user.verifyEmail', { token });\n}\n\n/**\n * Resend the verification email for a given email address.\n * The email is only sent if the address is registered and not yet verified.\n * A generic response is always returned to avoid leaking account information.\n *\n * @example\n * ```ts\n * await resendEmailVerification({ email: 'user@example.com' });\n * ```\n * @param options.email - The email address to resend verification to.\n */\nexport async function resendEmailVerification(options: { email: string }) {\n const { email } = options;\n await callMethod('_system.user.resendEmailVerification', { email });\n}\n\n/**\n * Logout the current user.\n *\n */\nexport async function logout() {\n await callMethod('_system.user.logout');\n setCurrentUser(null);\n}\n\n/**\n * Send reset password token.\n * @param options.email - The email of the user.\n */\nexport async function sendResetPasswordToken(options: { email: string }) {\n const { email } = options;\n await callMethod('_system.user.sendResetPasswordToken', {\n email,\n });\n}\n\n/**\n * Reset password.\n * @param options.token - The password reset token.\n * @param options.password - The new password.\n */\nexport async function resetPassword(options: { token: string; password: string }) {\n const { token, password } = options;\n await callMethod('_system.user.resetPassword', {\n token,\n password,\n });\n}\n\n/**\n * Link an OAuth provider to the currently signed-in user's account.\n * Redirects the browser to the OAuth provider's authorization page.\n * The provider will redirect back and the account will be linked.\n *\n * @example\n * ```ts\n * linkOAuthProvider({ provider: 'google' });\n * ```\n * @param options.provider - The OAuth provider to link ('google' or 'github').\n */\nexport async function linkOAuthProvider(options: { provider: OAuthProvider }): Promise<void> {\n const { provider } = options;\n\n const token = getAuthToken();\n if (token) {\n // Ask the server to set a secure httpOnly cookie for the OAuth linking flow\n const response = await fetch('/api/_internal/auth/set-link-cookie', {\n method: 'POST',\n headers: { 'Content-Type': 'application/json' },\n body: JSON.stringify({ authToken: token }),\n credentials: 'include',\n });\n if (!response.ok) {\n throw new Error('Failed to initialize OAuth linking. Please ensure you are logged in.');\n }\n }\n window.location.href = `/api/_internal/auth/${provider}?mode=link`;\n}\n/**\n * Unlink an OAuth provider from the currently signed-in user's account.\n *\n * @example\n * ```ts\n * await unlinkOAuthProvider({ provider: 'github' });\n * ```\n * @param options.provider - The OAuth provider to unlink ('google' or 'github').\n */\nexport async function unlinkOAuthProvider(options: { provider: OAuthProvider }): Promise<void> {\n const { provider } = options;\n await callMethod('_system.user.unlinkOAuthProvider', { provider });\n}\n\n/**\n * Get the current auth token associated with the current session.\n * @returns The auth token or undefined if not authenticated.\n */\nexport function getAuthToken(): string | undefined {\n return getLocalStorageSession()?.authToken;\n}\n\nexport function getClientInfo(): ClientInfo {\n return {\n screenWidth: window.screen.width,\n screenHeight: window.screen.height,\n windowWidth: window.innerWidth,\n windowHeight: window.innerHeight,\n pixelRatio: window.devicePixelRatio,\n orientation: window.screen.orientation?.type ?? null,\n };\n}\n","export type ErrorHandler = (error: Error, methodName: string) => void;\n\nlet errorHandler: ErrorHandler = (error, methodName) => {\n console.error(`Error calling method '${methodName}':`, error);\n};\n\nexport function setErrorHandler(handler: ErrorHandler) {\n errorHandler = handler;\n}\n\nexport function handleError(error: Error, methodName: string) {\n return errorHandler(error, methodName);\n}\n","/*\n The \"use client\" directive is specifically for the Next.js layout component, which is rendered on the server by default.\n Because of this, we are explicitly marking it as a client component, so we can render this component on the client\n and properly initialize config on the client side.\n \n While this is specific to Next.js, it is simply ignored outside of Next.js and should not cause errors.\n*/\n'use client';\n\nimport { getAuthToken, getClientInfo } from '@/auth/client';\nimport { handleError } from '@/client/errorHandler';\nimport { reviveResponseTypes } from '@/methods/serialize';\n\nexport class MethodError extends Error {\n status: number;\n\n constructor(message: string, status: number) {\n super(message);\n this.name = 'MethodError';\n this.status = status;\n }\n}\n\nexport type MethodArgs = Record<string, unknown>;\n\nexport type CallMethodOptions = {\n errorHandler?: (error: Error, methodName: string) => void;\n};\n\nexport async function callMethod<T = unknown>(\n methodName: string,\n args: MethodArgs = {},\n options: CallMethodOptions = {}\n): Promise<T> {\n try {\n return await call<T>(`/api/_internal/method/${methodName}`, args);\n } catch (error) {\n const handler = options.errorHandler ?? handleError;\n handler(error as Error, methodName);\n throw error;\n }\n}\n\nasync function call<T = unknown>(endpoint: string, args: MethodArgs): Promise<T> {\n const response = await fetch(endpoint, {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json',\n },\n body: JSON.stringify({\n args,\n authToken: getAuthToken(),\n clientInfo: getClientInfo(),\n }),\n });\n\n if (!response.ok) {\n const error = await response.text();\n throw new MethodError(error, response.status);\n }\n\n const text = await response.text();\n const result = text ? JSON.parse(text) : undefined;\n if (!result) {\n throw new Error('Invalid response from server');\n }\n\n return reviveResponseTypes(result.data, result.typeMap);\n}\n","import { AppConfig, ConfigKey, Configs } from './types';\n\nlet config: Record<ConfigKey, AppConfig> = {};\n\n/**\n * @sidebarTitle getConfig (client)\n *\n * @param key\n * @returns\n */\nexport function getConfig(key: ConfigKey): string | number | boolean | undefined {\n return config[key]?.value;\n}\n\nexport function _setConfig(configs: Configs) {\n config = configs;\n}\n","import { create } from 'zustand';\nimport { z } from 'zod';\nimport { callMethod } from './method';\nimport { _setConfig } from '../config/client';\nimport { setLocalStorageSession } from './localStorage';\nimport { time } from '../time';\nimport { Configs } from '../config/types';\n\ntype User = {\n id: string;\n handle: string;\n roles: string[];\n hasRole: (role: string) => boolean;\n requireRole: (role: string) => void;\n firstName?: string;\n lastName?: string;\n avatarUrl?: string;\n};\n\ntype SessionStore = {\n user: User | null;\n setUser: (user: User | null) => void;\n};\n\nexport const useSessionStore = create<SessionStore>((set) => ({\n user: null,\n setUser: (user) => set({ user }),\n}));\n\nlet isInitialized = false;\nconst SESSION_HEARTBEAT_INTERVAL = time.seconds(30);\nlet heartbeatTimer: ReturnType<typeof setTimeout> | null = null;\n\nconst userSchema = z.object({\n id: z.string(),\n handle: z.string(),\n roles: z.array(z.string()),\n firstName: z.string().optional(),\n lastName: z.string().optional(),\n avatarUrl: z.string().optional(),\n});\n\nfunction parseUser(user: unknown): User | null {\n if (!user) {\n return null;\n }\n\n const result = userSchema.safeParse(user);\n\n if (!result.success) {\n console.error('Session Error: Invalid user payload', result.error);\n return null;\n }\n\n const parsedData = result.data;\n\n return Object.freeze({\n ...parsedData,\n firstName: parsedData.firstName ?? undefined,\n lastName: parsedData.lastName ?? undefined,\n avatarUrl: parsedData.avatarUrl ?? undefined,\n hasRole: (role: string) => parsedData.roles.includes(role),\n requireRole: (role: string) => {\n if (!parsedData.roles.includes(role)) {\n throw new Error(`Access denied - role '${role}' required`);\n }\n },\n });\n}\n\nexport async function initSession() {\n if (isInitialized) {\n return;\n }\n\n isInitialized = true;\n\n const { configs, session, user } = await callMethod<{\n configs: Configs;\n session: object;\n user: object;\n }>('_system.session.init');\n _setConfig(configs);\n setLocalStorageSession(session);\n\n useSessionStore.getState().setUser(parseUser(user));\n\n await loopSessionHeartbeat();\n}\n\nasync function loopSessionHeartbeat() {\n try {\n await callMethod('_system.session.heartbeat', {}, { errorHandler: () => {} });\n } catch {\n // Silently ignore heartbeat errors - they're expected during HMR/reconnects\n }\n heartbeatTimer = setTimeout(loopSessionHeartbeat, SESSION_HEARTBEAT_INTERVAL);\n}\n\nexport function setCurrentUser(user: unknown) {\n const enrichedUser = parseUser(user);\n useSessionStore.getState().setUser(enrichedUser);\n return enrichedUser;\n}\n\nexport function getHeartbeatTimer() {\n return heartbeatTimer;\n}\n\nexport function stopHeartbeatTimer() {\n if (heartbeatTimer) {\n clearTimeout(heartbeatTimer);\n heartbeatTimer = null;\n }\n}\n\n/**\n * `useSession` is a hook that returns the current user, and in the future will also return other details about the current session.\n *\n * @example\n * ```ts\n * import { useSession } from 'modelence/client';\n *\n * function MyComponent() {\n * const { user } = useSession();\n * return <div>{user?.handle}</div>;\n * }\n * ```\n */\nexport function useSession() {\n const user = useSessionStore((state) => state.user);\n return { user };\n}\n","/*\n The \"use client\" directive is specifically for the Next.js layout component, which is rendered on the server by default.\n Because of this, we are explicitly marking it as a client component, so we can render this component on the client\n and properly initialize config on the client side.\n \n While this is specific to Next.js, it is simply ignored outside of Next.js and should not cause errors.\n*/\n'use client';\n\nimport React, { useState, useEffect, ReactNode } from 'react';\nimport { initSession } from './session';\n\ninterface AppProviderProps {\n children: ReactNode;\n loadingElement?: ReactNode;\n}\n\nlet isInitialized = false;\n\nexport function AppProvider({ children, loadingElement }: AppProviderProps) {\n const [isLoading, setIsLoading] = useState(true);\n\n useEffect(() => {\n async function initConfig() {\n if (isInitialized) {\n return;\n }\n\n isInitialized = true;\n\n await initSession();\n setIsLoading(false);\n }\n\n initConfig();\n }, []);\n\n if (isLoading) {\n return loadingElement ?? <div>Loading...</div>;\n }\n\n return children;\n}\n","'use client';\n\nimport type { ObjectId } from 'mongodb';\nimport type { ConfigParams, ConfigType, ValueType } from '../config/types';\nimport { callMethod, type MethodArgs } from './method';\nimport type { AnyMethodShape } from '../methods/types';\n\n// Pulls the config store value without importing server-side code\nimport { getConfig as _getClientConfig } from '../config/client';\n\n// ── type helpers ─────────────────────────────────────────────────────────────\n\n/**\n * Recursively maps ObjectId → string to match the sanitized runtime values\n * sent over the wire. Dates are preserved (revived via typeMap on the client).\n */\ntype Sanitized<T> = T extends ObjectId\n ? string\n : T extends Date\n ? Date\n : T extends (infer U)[]\n ? Sanitized<U>[]\n : T extends object\n ? { [K in keyof T]: Sanitized<T[K]> }\n : T;\n\ntype ExtractArgs<M> = M extends (args: infer A, ...rest: any[]) => any // eslint-disable-line @typescript-eslint/no-explicit-any\n ? A\n : M extends { handler: (args: infer A, ...rest: any[]) => any } // eslint-disable-line @typescript-eslint/no-explicit-any\n ? A\n : MethodArgs;\n\ntype ExtractResult<M> = M extends (...args: any[]) => Promise<infer R> // eslint-disable-line @typescript-eslint/no-explicit-any\n ? Sanitized<R>\n : M extends { handler: (...args: any[]) => Promise<infer R> } // eslint-disable-line @typescript-eslint/no-explicit-any\n ? Sanitized<R>\n : unknown;\n\ntype PublicKeyOf<TSchema extends Record<string, ConfigParams>> = {\n [K in keyof TSchema as TSchema[K] extends ConfigParams<ConfigType, true>\n ? string & K\n : never]: ValueType<TSchema[K]['type']>;\n};\n\ntype AnyModule = {\n name: string;\n configSchema: Record<string, ConfigParams>;\n queries: Record<string, AnyMethodShape>;\n mutations: Record<string, AnyMethodShape>;\n};\n\n// ── createClientModule ────────────────────────────────────────────────────────\n\n/**\n * Creates a typed client accessor for a module's public configs, queries, and mutations.\n *\n * Use `import type` to reference the module so no server code is bundled on the client.\n * Arg and return types for queries and mutations are inferred automatically from the\n * server-side handler signatures.\n *\n * @param moduleName - The module's name as passed to `new Module(name, ...)`.\n *\n * @example\n * ```ts\n * // src/client/payments.ts\n * import type paymentsModule from '../server/payments';\n * import { createClientModule } from 'modelence/client';\n *\n * export const payments = createClientModule<typeof paymentsModule>('payments');\n * ```\n *\n * ```ts\n * // src/components/Checkout.tsx\n * import { useQuery, useMutation } from '@tanstack/react-query';\n * import { payments } from '../client/payments';\n *\n * // Typed config — public keys only, private and secret keys excluded:\n * const currency = payments.getConfig('currency'); // string | undefined\n *\n * // Typed query — pass directly to useQuery:\n * const { data: products } = useQuery(payments.query('getProducts', { page: 1 }));\n *\n * // Typed mutation — pass directly to useMutation:\n * const { mutate: charge } = useMutation(payments.mutation('charge'));\n * charge({ amount: 100 }); // args typed from handler signature\n * ```\n */\nexport function createClientModule<TModule extends AnyModule>(moduleName: string) {\n return {\n getConfig<K extends keyof PublicKeyOf<TModule['configSchema']> & string>(\n key: K\n ): PublicKeyOf<TModule['configSchema']>[K] | undefined {\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n return _getClientConfig(`${moduleName}.${key}`) as any;\n },\n\n query<K extends keyof TModule['queries'] & string>(\n name: K,\n ...rest: {} extends ExtractArgs<TModule['queries'][K]>\n ? [args?: ExtractArgs<TModule['queries'][K]>]\n : [args: ExtractArgs<TModule['queries'][K]>]\n ) {\n const args = (rest[0] ?? {}) as ExtractArgs<TModule['queries'][K]>;\n return {\n queryKey: [moduleName, name, args] as const,\n queryFn: (): Promise<ExtractResult<TModule['queries'][K]>> =>\n callMethod<ExtractResult<TModule['queries'][K]>>(\n `${moduleName}.${name}`,\n args as MethodArgs\n ),\n };\n },\n\n mutation<K extends keyof TModule['mutations'] & string>(name: K) {\n return {\n mutationFn: (\n args: ExtractArgs<TModule['mutations'][K]>\n ): Promise<ExtractResult<TModule['mutations'][K]>> =>\n callMethod<ExtractResult<TModule['mutations'][K]>>(\n `${moduleName}.${name}`,\n args as MethodArgs\n ),\n };\n },\n\n /**\n * Returns options for `useInfiniteQuery`. The `getArgs` callback receives the\n * current `pageParam` and returns the args to pass to the query handler.\n * Spread the result into `useInfiniteQuery` alongside `getNextPageParam`.\n *\n * Annotate the `pageParam` type in the callback so TypeScript can infer the\n * page param type — no manual generic needed on `useInfiniteQuery`.\n */\n infiniteQuery<K extends keyof TModule['queries'] & string, TPageParam = unknown>(\n name: K,\n getArgs: (pageParam: TPageParam | undefined) => ExtractArgs<TModule['queries'][K]>\n ) {\n return {\n queryKey: [moduleName, name, 'infinite', getArgs(undefined)] as const,\n // Included so TanStack infers TPageParam from the callback type, not from a bare `undefined`.\n initialPageParam: undefined as TPageParam | undefined,\n queryFn: ({\n pageParam,\n }: {\n pageParam: TPageParam | undefined;\n }): Promise<ExtractResult<TModule['queries'][K]>> =>\n callMethod<ExtractResult<TModule['queries'][K]>>(\n `${moduleName}.${name}`,\n getArgs(pageParam) as MethodArgs\n ),\n };\n },\n };\n}\n","import type systemModule from './index';\nimport { createClientModule } from '../client/module';\n\nexport const systemConfig = createClientModule<typeof systemModule>('_system');\n","import React from 'react';\nimport ReactDOM from 'react-dom/client';\nimport { AppProvider } from '../client';\nimport { setErrorHandler, ErrorHandler } from './errorHandler';\n\nexport function renderApp({\n loadingElement,\n routesElement,\n favicon,\n errorHandler,\n}: {\n loadingElement: React.ReactNode;\n routesElement: React.ReactNode;\n favicon?: string;\n errorHandler?: ErrorHandler;\n}) {\n if (errorHandler) {\n setErrorHandler(errorHandler);\n }\n\n window.addEventListener('unload', () => {\n // The presence of any 'unload' event handler, even empty,\n // prevents bfcache in most browsers\n });\n\n ReactDOM.createRoot(document.getElementById('root')!).render(\n <React.StrictMode>\n <AppProvider loadingElement={loadingElement}>{routesElement}</AppProvider>\n </React.StrictMode>\n );\n\n if (favicon) {\n const link = document.querySelector(\"link[rel~='icon']\") as HTMLLinkElement;\n if (!link) {\n const newLink = document.createElement('link');\n newLink.rel = 'icon';\n newLink.href = favicon;\n document.head.appendChild(newLink);\n } else {\n link.href = favicon;\n }\n }\n}\n","import io, { Socket } from 'socket.io-client';\nimport { WebsocketClientProvider } from '../types';\nimport { ClientChannel } from '../clientChannel';\nimport { getAuthToken, getClientInfo } from '@/auth/client';\nimport { reviveResponseTypes } from '@/methods/serialize';\n\nlet socketClient: Socket | null = null;\n\ninterface ActiveLiveSubscription {\n subscriptionId: string;\n method: string;\n args: Record<string, unknown>;\n}\nconst activeLiveSubscriptions = new Map<string, ActiveLiveSubscription>();\n\nfunction getSocket(): Socket {\n if (!socketClient) {\n throw new Error('WebSocket not initialized. Call startWebsockets() first.');\n }\n return socketClient;\n}\n\nfunction resubscribeAll() {\n const authToken = getAuthToken();\n const clientInfo = getClientInfo();\n for (const sub of activeLiveSubscriptions.values()) {\n socketClient?.emit('subscribeLiveQuery', {\n subscriptionId: sub.subscriptionId,\n method: sub.method,\n args: sub.args,\n authToken,\n clientInfo,\n });\n }\n}\n\nfunction init(props: { channels?: ClientChannel<unknown>[] }) {\n socketClient = io('/', {\n transports: ['websocket'],\n auth: {\n token: getAuthToken(),\n },\n });\n\n // Subscribe to all live queries on connect/reconnect\n socketClient.on('connect', () => {\n if (activeLiveSubscriptions.size > 0) {\n console.log(\n `[Modelence] WebSocket reconnected, re-subscribing to ${activeLiveSubscriptions.size} live queries`\n );\n resubscribeAll();\n }\n });\n\n props.channels?.forEach((channel) => channel.init());\n}\n\nfunction on<T = unknown>({\n category,\n listener,\n}: {\n category: string;\n listener: (data: T) => void;\n}) {\n getSocket().on(category, listener);\n}\n\nfunction once<T = unknown>({\n category,\n listener,\n}: {\n category: string;\n listener: (data: T) => void;\n}) {\n getSocket().once(category, listener);\n}\n\nfunction off<T = unknown>({\n category,\n listener,\n}: {\n category: string;\n listener: (data: T) => void;\n}) {\n getSocket().off(category, listener);\n}\n\nfunction emit({ eventName, category, id }: { eventName: string; category: string; id: string }) {\n getSocket().emit(eventName, `${category}:${id}`);\n}\n\nfunction joinChannel({ category, id }: { category: string; id: string }) {\n emit({\n eventName: 'joinChannel',\n category,\n id,\n });\n}\n\nfunction leaveChannel({ category, id }: { category: string; id: string }) {\n emit({\n eventName: 'leaveChannel',\n category,\n id,\n });\n}\n\nlet liveQueryCounter = 0;\n\nexport function subscribeLiveQuery<T = unknown>(\n method: string,\n args: Record<string, unknown>,\n onData: (data: T) => void,\n onError?: (error: string) => void\n): () => void {\n const subscriptionId = `sub-${++liveQueryCounter}-${Date.now()}`;\n\n const handleData = ({\n subscriptionId: sid,\n data,\n typeMap,\n }: {\n subscriptionId: string;\n data: T;\n typeMap?: Record<string, unknown>;\n }) => {\n if (sid === subscriptionId) {\n onData(reviveResponseTypes(data, typeMap));\n }\n };\n\n const handleError = ({\n subscriptionId: sid,\n error,\n }: {\n subscriptionId: string;\n error: string;\n }) => {\n if (sid === subscriptionId) {\n console.error(`[Modelence] Live query error for ${method}:`, error);\n onError?.(error);\n }\n };\n\n const socket = getSocket();\n socket.on('liveQueryData', handleData);\n socket.on('liveQueryError', handleError);\n\n activeLiveSubscriptions.set(subscriptionId, { subscriptionId, method, args });\n\n // Only emit if already connected; otherwise the connect handler will handle it\n if (socket.connected) {\n socket.emit('subscribeLiveQuery', {\n subscriptionId,\n method,\n args,\n authToken: getAuthToken(),\n clientInfo: getClientInfo(),\n });\n }\n\n // Return unsubscribe function\n return () => {\n activeLiveSubscriptions.delete(subscriptionId);\n socket.emit('unsubscribeLiveQuery', { subscriptionId });\n socket.off('liveQueryData', handleData);\n socket.off('liveQueryError', handleError);\n };\n}\n\nconst websocketProvider: WebsocketClientProvider = {\n init,\n on,\n once,\n off,\n emit,\n joinChannel,\n leaveChannel,\n};\n\nexport default websocketProvider;\n","import { ClientChannel } from './clientChannel';\nimport websocketProvider, { subscribeLiveQuery } from './socketio/client';\nimport { WebsocketClientProvider } from './types';\n\nexport { subscribeLiveQuery };\n\nlet websocketClientProvider: WebsocketClientProvider | null = null;\n\nexport function setWebsocketClientProvider(provider: WebsocketClientProvider | null) {\n websocketClientProvider = provider;\n}\n\nexport function getWebsocketClientProvider() {\n return websocketClientProvider;\n}\n\nexport function startWebsockets(props?: {\n provider?: WebsocketClientProvider;\n channels?: ClientChannel[];\n}) {\n if (websocketClientProvider) {\n console.warn('WebSocket already initialized. Skipping initialization.');\n return;\n }\n\n const provider = props?.provider || websocketProvider;\n setWebsocketClientProvider(provider);\n try {\n provider.init({\n channels: props?.channels,\n });\n } catch (error) {\n setWebsocketClientProvider(null);\n throw error;\n }\n}\n","import { getWebsocketClientProvider } from './client';\n\nexport class ClientChannel<T = unknown> {\n public readonly category: string;\n private readonly onMessage: (data: T) => void;\n\n constructor(category: string, onMessage: (data: T) => void) {\n this.category = category;\n this.onMessage = onMessage;\n }\n\n init() {\n getWebsocketClientProvider()?.on({\n category: this.category,\n listener: this.onMessage,\n });\n }\n\n joinChannel(id: string) {\n getWebsocketClientProvider()?.joinChannel({\n category: this.category,\n id,\n });\n }\n\n leaveChannel(id: string) {\n getWebsocketClientProvider()?.leaveChannel({\n category: this.category,\n id,\n });\n }\n}\n","import { callMethod } from '../client/method';\nimport type { FileVisibility, GetUploadUrlResult } from './types';\n\ntype UploadFileParams = {\n filePath: string;\n contentType: string;\n visibility: FileVisibility;\n};\n\ntype UploadFileResult = {\n filePath: string;\n};\n\nexport async function uploadFile(\n file: File | Blob,\n { filePath, contentType, visibility }: UploadFileParams\n): Promise<UploadFileResult> {\n const {\n url,\n fields,\n filePath: resolvedFilePath,\n } = await callMethod<GetUploadUrlResult>('_system.files.getUploadUrl', {\n filePath,\n contentType,\n visibility,\n });\n\n const formData = new FormData();\n for (const [key, value] of Object.entries(fields)) {\n formData.append(key, value);\n }\n formData.append('file', file);\n\n const uploadResponse = await fetch(url, {\n method: 'POST',\n body: formData,\n });\n\n if (!uploadResponse.ok) {\n throw new Error(`Failed to upload file: HTTP status: ${uploadResponse.status}`);\n }\n\n return { filePath: resolvedFilePath };\n}\n\nexport async function deleteFile(filePath: string): Promise<void> {\n await callMethod('_system.files.deleteFile', { filePath });\n}\n\nexport async function downloadFile(filePath: string): Promise<{ downloadUrl: string }> {\n return callMethod('_system.files.downloadFile', { filePath });\n}\n\nexport async function getFileUrl(filePath: string): Promise<{ url: string }> {\n return callMethod('_system.files.getFileUrl', { filePath });\n}\n","import React from 'react';\n\nimport { AppProvider as OriginalAppProvider } from './client/AppProvider';\n\nexport { getConfig } from './config/client';\nexport { createClientModule } from './client/module';\nexport type { ValueType } from './config/types';\nexport { systemConfig } from './system/client';\n\nexport const AppProvider =\n 'useClient' in React\n ? // @ts-ignore: React.useClient only exists in Next.js\n React.useClient(OriginalAppProvider)\n : OriginalAppProvider;\n\nexport { renderApp } from './client/renderApp';\nexport { callMethod, MethodError, type MethodArgs } from './client/method';\nexport { useSession } from './client/session';\nexport {\n signupWithPassword,\n loginWithPassword,\n verifyEmail,\n updateProfile,\n resendEmailVerification,\n logout,\n sendResetPasswordToken,\n resetPassword,\n linkOAuthProvider,\n unlinkOAuthProvider,\n type UserInfo,\n} from './auth/client';\nexport {\n getWebsocketClientProvider,\n setWebsocketClientProvider,\n startWebsockets,\n subscribeLiveQuery,\n} from './websocket/client';\nexport { ClientChannel } from './websocket/clientChannel';\nexport { getLocalStorageSession } from './client/localStorage';\nexport { uploadFile, deleteFile, downloadFile, getFileUrl } from './files/client';\n"]}
1
+ {"version":3,"sources":[],"names":[],"mappings":"","file":"client.js"}
@@ -0,0 +1,2 @@
1
+ import a from'path';import u from'fs';var l=/\.(css|scss|sass|less|styl|stylus|pcss|postcss)(\?|$)/,p=Object.freeze({hrefs:[],source:"dev"});function y(s,e){let t=s.moduleGraph.getModuleById(d(s,e));if(!t)return p;let n=new Set,o=[],i=new Set,c=r=>{if(!n.has(r)){n.add(r),r.url&&l.test(r.url)&&!i.has(r.url)&&(i.add(r.url),o.push(r.url));for(let f of r.importedModules)c(f);for(let f of r.ssrImportedModules)c(f);}};return c(t),{hrefs:o,source:"dev"}}function d(s,e){let t=s.config.root;return a.join(t,e).replace(/\\/g,"/")}function A(s){let e=a.resolve(process.cwd(),s,".vite","ssr-manifest.json");if(!u.existsSync(e))return {hrefs:[],source:"prod"};let t;try{t=JSON.parse(u.readFileSync(e,"utf-8"));}catch(o){return console.warn("Modelence SSR: failed to parse ssr-manifest.json",o),{hrefs:[],source:"prod"}}let n=new Set;for(let o of Object.values(t))if(Array.isArray(o))for(let i of o)typeof i=="string"&&l.test(i)&&n.add(i);return {hrefs:Array.from(n),source:"prod"}}function v(s){return s.hrefs.length===0?"":s.hrefs.map(e=>`<link rel="stylesheet" href="${h(e)}">`).join("")}function C(s){return s.hrefs.map(e=>`<${e}>; rel=preload; as=style`)}function h(s){return s.replace(/&/g,"&amp;").replace(/"/g,"&quot;")}export{C as buildEarlyHintsLink,y as collectDevCssAssets,A as loadProdCssAssets,v as renderStylesheetLinks};//# sourceMappingURL=collectCss-4V7I3QV3.js.map
2
+ //# sourceMappingURL=collectCss-4V7I3QV3.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/ssr/collectCss.ts"],"names":["CSS_EXTENSION_REGEX","EMPTY_ASSETS","collectDevCssAssets","viteServer","entryVirtualPath","ssrModule","resolveEntryId","seen","hrefs","hrefSet","walk","mod","imported","root","path","loadProdCssAssets","clientBuildDir","manifestPath","fs","manifest","error","assets","asset","renderStylesheetLinks","href","escapeAttribute","buildEarlyHintsLink","value"],"mappings":"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","file":"collectCss-4V7I3QV3.js","sourcesContent":["import path from 'path';\nimport fs from 'fs';\nimport type { ViteDevServer, ModuleNode } from 'vite';\n\nconst CSS_EXTENSION_REGEX = /\\.(css|scss|sass|less|styl|stylus|pcss|postcss)(\\?|$)/;\n\nexport type CssAssetSource = 'dev' | 'prod';\n\nexport interface CssAssets {\n /**\n * Public URLs suitable for emitting as <link rel=\"stylesheet\">. In dev these\n * point at Vite's transform endpoint; in prod they point at the built asset.\n */\n hrefs: string[];\n source: CssAssetSource;\n}\n\nconst EMPTY_ASSETS: CssAssets = Object.freeze({ hrefs: [], source: 'dev' }) as CssAssets;\n\n/**\n * Walk the Vite dev module graph from the SSR entry, collecting every CSS\n * module pulled in by the rendered tree. Returns URLs that the browser can\n * request directly so the dev HTML can ship <link rel=\"stylesheet\"> tags\n * before the client JS runs (kills FOUC in dev).\n */\nexport function collectDevCssAssets(\n viteServer: ViteDevServer,\n entryVirtualPath: string\n): CssAssets {\n const ssrModule = viteServer.moduleGraph.getModuleById(\n resolveEntryId(viteServer, entryVirtualPath)\n );\n if (!ssrModule) {\n return EMPTY_ASSETS;\n }\n\n const seen = new Set<ModuleNode>();\n const hrefs: string[] = [];\n const hrefSet = new Set<string>();\n\n const walk = (mod: ModuleNode) => {\n if (seen.has(mod)) {\n return;\n }\n seen.add(mod);\n\n if (mod.url && CSS_EXTENSION_REGEX.test(mod.url) && !hrefSet.has(mod.url)) {\n hrefSet.add(mod.url);\n hrefs.push(mod.url);\n }\n\n for (const imported of mod.importedModules) {\n walk(imported);\n }\n for (const imported of mod.ssrImportedModules) {\n walk(imported);\n }\n };\n\n walk(ssrModule);\n\n return { hrefs, source: 'dev' };\n}\n\nfunction resolveEntryId(viteServer: ViteDevServer, entryVirtualPath: string): string {\n const root = viteServer.config.root;\n // `entryVirtualPath` is rooted at Vite's `root` (e.g. '/index.tsx' under\n // 'src/client'). The module graph keys modules by absolute file id.\n const absolutePath = path.join(root, entryVirtualPath).replace(/\\\\/g, '/');\n return absolutePath;\n}\n\ntype SsrManifest = Record<string, string[]>;\n\n/**\n * Read the production SSR manifest produced by `vite build --ssr`. The manifest\n * maps every JS module id to the CSS/asset URLs the client must load to render\n * its output. We flatten the unique CSS entries — the SSR render only touched a\n * subset of routes per request, but at framework boot we don't yet have a\n * per-request module list, so we emit the full set. This matches the behavior\n * of a non-streaming build.\n */\nexport function loadProdCssAssets(clientBuildDir: string): CssAssets {\n const manifestPath = path.resolve(process.cwd(), clientBuildDir, '.vite', 'ssr-manifest.json');\n\n if (!fs.existsSync(manifestPath)) {\n return { hrefs: [], source: 'prod' };\n }\n\n let manifest: SsrManifest;\n try {\n manifest = JSON.parse(fs.readFileSync(manifestPath, 'utf-8')) as SsrManifest;\n } catch (error) {\n console.warn('Modelence SSR: failed to parse ssr-manifest.json', error);\n return { hrefs: [], source: 'prod' };\n }\n\n const hrefSet = new Set<string>();\n for (const assets of Object.values(manifest)) {\n if (!Array.isArray(assets)) {\n continue;\n }\n for (const asset of assets) {\n if (typeof asset === 'string' && CSS_EXTENSION_REGEX.test(asset)) {\n hrefSet.add(asset);\n }\n }\n }\n\n return { hrefs: Array.from(hrefSet), source: 'prod' };\n}\n\n/**\n * Render the CSS asset URLs as HTML <link rel=\"stylesheet\"> tags. React 19\n * users who render <link precedence=...> inside components get React Float's\n * dedup + ordering automatically; this helper exists for the framework-level\n * tags injected into the template <head> before the React shell.\n */\nexport function renderStylesheetLinks(assets: CssAssets): string {\n if (assets.hrefs.length === 0) {\n return '';\n }\n return assets.hrefs\n .map((href) => `<link rel=\"stylesheet\" href=\"${escapeAttribute(href)}\">`)\n .join('');\n}\n\n/**\n * Build a Link header value for HTTP 103 Early Hints. The browser starts the\n * stylesheet fetch as soon as it sees this response, before the SSR render\n * has even begun.\n */\nexport function buildEarlyHintsLink(assets: CssAssets): string[] {\n return assets.hrefs.map((href) => `<${href}>; rel=preload; as=style`);\n}\n\nfunction escapeAttribute(value: string): string {\n return value.replace(/&/g, '&amp;').replace(/\"/g, '&quot;');\n}\n"]}
@@ -5,6 +5,8 @@ interface ModelenceConfig {
5
5
  serverDir: string;
6
6
  serverEntry: string;
7
7
  postBuildCommand?: string;
8
+ /** Must match the `ssr` flag passed to `startApp()`. */
9
+ ssr?: boolean;
8
10
  }
9
11
  type ExpressMiddleware = (req: Request, res: Response, next: NextFunction) => void | Promise<void>;
10
12
  interface AppServerInitOptions {
@@ -12,7 +14,7 @@ interface AppServerInitOptions {
12
14
  }
13
15
  interface AppServer {
14
16
  init: (options: AppServerInitOptions) => Promise<void>;
15
- handler: (req: Request, res: Response) => void;
17
+ handler: (req: Request, res: Response) => void | Promise<void>;
16
18
  middlewares?: () => ExpressMiddleware[];
17
19
  }
18
20
 
package/dist/index.d.ts CHANGED
@@ -1,5 +1,5 @@
1
- export { C as ConfigSchema, W as WebsocketClientProvider, a as WebsocketServerProvider } from './types-CTjq1HaP.js';
2
- export { M as ModelenceConfig } from './index-CLpVWWuj.js';
1
+ export { C as ConfigSchema, W as WebsocketClientProvider, a as WebsocketServerProvider } from './types-DV0J5rPI.js';
2
+ export { M as ModelenceConfig } from './index-BBMsjrFN.js';
3
3
  import 'http';
4
4
  import 'mongodb';
5
5
  import 'express';
package/dist/index.js CHANGED
@@ -1,2 +1,2 @@
1
- export{b as AuthError,a as ModelenceError,d as RateLimitError,c as ValidationError}from'./chunk-C3UESBRX.js';export{a as time}from'./chunk-DO5TZLF5.js';import'./chunk-55J6XMHW.js';//# sourceMappingURL=index.js.map
1
+ export{b as AuthError,a as ModelenceError,d as RateLimitError,c as ValidationError}from'./chunk-C3UESBRX.js';import'./chunk-55J6XMHW.js';export{a as time}from'./chunk-DO5TZLF5.js';//# sourceMappingURL=index.js.map
2
2
  //# sourceMappingURL=index.js.map
@@ -1,2 +1,2 @@
1
- export{l as author,i as bin,n as bugs,s as default,q as dependencies,d as description,p as devDependencies,g as exports,h as files,o as homepage,m as license,e as main,b as name,r as peerDependencies,k as repository,j as scripts,a as type,f as types,c as version}from'./chunk-52CM66XJ.js';//# sourceMappingURL=package-WKD57CCF.js.map
2
- //# sourceMappingURL=package-WKD57CCF.js.map
1
+ export{l as author,i as bin,n as bugs,s as default,q as dependencies,d as description,p as devDependencies,g as exports,h as files,o as homepage,m as license,e as main,b as name,r as peerDependencies,k as repository,j as scripts,a as type,f as types,c as version}from'./chunk-KU4GANYW.js';//# sourceMappingURL=package-XQQF6NGT.js.map
2
+ //# sourceMappingURL=package-XQQF6NGT.js.map
@@ -1 +1 @@
1
- {"version":3,"sources":[],"names":[],"mappings":"","file":"package-WKD57CCF.js"}
1
+ {"version":3,"sources":[],"names":[],"mappings":"","file":"package-XQQF6NGT.js"}
@@ -0,0 +1,2 @@
1
+ import {a,b,c}from'./chunk-JBTEZOY2.js';import'./chunk-LDVLXJLS.js';import'./chunk-3SPXJEOR.js';import {a as a$1,m}from'./chunk-IJ5BS7PM.js';import {x,r,y}from'./chunk-VFOGVWJK.js';import'./chunk-5M6FUMUK.js';import'./chunk-DO5TZLF5.js';import {renderToPipeableStream}from'react-dom/server';import {Writable}from'stream';import {QueryClient,dehydrate}from'@tanstack/react-query';import {jsx}from'react/jsx-runtime';var A=()=>{let r=c();return r?y(r.session.user):null},I=r=>{let t=c();if(t)return t.session.configs[r]?.value};function Q(){x(A),r(I);}async function G(r){let{callContext:t,loadingElement:c,routesElement:o,router:n,location:s}=r;Q();let l=await a("_system.session.init",{},t),e=new QueryClient({defaultOptions:{queries:{retry:false,gcTime:0}}}),a$2=n?n({children:o,location:s}):o,i=jsx(a$1,{loadingElement:c,children:jsx(m,{client:e,children:a$2})}),p=await b({callContext:t,queryClient:e,session:{user:l.user,configs:l.configs??{}}},()=>J(i,s)),y=dehydrate(e);return e.clear(),{html:p,sessionState:JSON.stringify({session:l}),queryState:JSON.stringify(y)}}async function L(r){let{callContext:t,loadingElement:c,routesElement:o,router:n,location:s,onShellReady:l,onError:e}=r;Q();let a$2=await a("_system.session.init",{},t),i=new QueryClient({defaultOptions:{queries:{retry:false,gcTime:0}}}),p=n?n({children:o,location:s}):o,y=jsx(a$1,{loadingElement:c,children:jsx(m,{client:i,children:p})}),f=null;await new Promise((m,R)=>{b({callContext:t,queryClient:i,session:{user:a$2.user,configs:a$2.configs??{}}},()=>{let d=renderToPipeableStream(y,{onShellReady(){f=d,l?.(),m(d);},onShellError(S){R(S);},onError(S){e?.(S);}});return d});});let g=null,N=m=>new Promise((R,d)=>{if(!f){d(new Error("SSR stream was not initialized"));return}m.on("finish",()=>{try{let S=dehydrate(i);g=JSON.stringify(S);}finally{i.clear();}R();}),m.on("error",d),f.pipe(m);});return {sessionState:JSON.stringify({session:a$2}),pipe:N,getQueryState:()=>{if(g===null)throw new Error("getQueryState() called before stream finished");return g}}}function J(r,t){return new Promise((c,o)=>{let n="",s=new Writable({write(e,a,i){n+=e.toString(),i();}}),{pipe:l}=renderToPipeableStream(r,{onAllReady(){s.on("finish",()=>c(n)),s.on("error",o),l(s);},onShellError(e){o(e);},onError(e){console.error("SSR onError (non-fatal):",{location:t,error:e});}});})}export{G as renderSsrTree,L as renderSsrTreeStream};//# sourceMappingURL=render-57TPXKHI.js.map
2
+ //# sourceMappingURL=render-57TPXKHI.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/ssr/render.tsx"],"names":["sessionResolver","ctx","getSsrContext","_parseSessionUser","configResolver","key","ensureSsrResolversInstalled","_setSsrSessionResolver","_setSsrConfigResolver","renderSsrTree","options","callContext","loadingElement","routesElement","router","location","sessionPayload","callInProcessMethod","queryClient","QueryClient","routedTree","tree","jsx","AppProvider","ModelenceQueryProvider","html","runWithSsrContext","renderToString","dehydratedState","dehydrate","renderSsrTreeStream","onShellReady","onError","streamRef","resolve","reject","stream","renderToPipeableStream","error","queryStateJson","pipe","destination","buffer","writable","Writable","chunk","_encoding","callback"],"mappings":"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","file":"render-57TPXKHI.js","sourcesContent":["import React from 'react';\nimport { renderToPipeableStream, type PipeableStream } from 'react-dom/server';\nimport { Writable } from 'node:stream';\nimport { QueryClient, dehydrate, type DehydratedState } from '@tanstack/react-query';\nimport { AppProvider } from '../client/AppProvider';\nimport { ModelenceQueryProvider } from '../client/queryProvider';\nimport { getSsrContext, runWithSsrContext } from './context';\nimport { callInProcessMethod } from './callInProcess';\nimport type { Context } from '../methods/types';\nimport {\n _parseSessionUser,\n _setSsrSessionResolver,\n type SessionInitPayload,\n} from '../client/session';\nimport { _setSsrConfigResolver } from '../config/client';\nimport type { ConfigKey, Configs } from '../config/types';\nimport type { SsrRouter } from '../client/renderApp';\n\nconst sessionResolver = () => {\n const ctx = getSsrContext();\n if (!ctx) {\n return null;\n }\n return _parseSessionUser(ctx.session.user);\n};\n\nconst configResolver = (key: ConfigKey) => {\n const ctx = getSsrContext();\n if (!ctx) {\n return undefined;\n }\n return ctx.session.configs[key]?.value;\n};\n\nfunction ensureSsrResolversInstalled() {\n // Idempotent — safe to call repeatedly. Re-installs the framework's own\n // resolvers if they were swapped out (tests, future reload code).\n _setSsrSessionResolver(sessionResolver);\n _setSsrConfigResolver(configResolver);\n}\n\nexport type SsrRenderResult = {\n html: string;\n /** Inline as <script id=\"__MODELENCE_STATE__\">…</script> */\n sessionState: string;\n /** Inline as <script id=\"__MODELENCE_QUERY_STATE__\">…</script> */\n queryState: string;\n};\n\nexport type SsrRenderOptions = {\n callContext: Context;\n loadingElement: React.ReactNode;\n routesElement: React.ReactNode;\n router?: SsrRouter;\n location?: string;\n};\n\nexport type SsrStreamHandle = {\n /** Session bootstrap payload — safe to inline before the React shell flushes. */\n sessionState: string;\n /**\n * Pipe React's HTML stream into the response. Resolves once every Suspense\n * boundary has settled and the stream has finished writing.\n */\n pipe: (destination: Writable) => Promise<void>;\n /**\n * Read the dehydrated query state. Only call AFTER `pipe()` resolves —\n * queries that resolve mid-stream populate the cache during render.\n */\n getQueryState: () => string;\n};\n\nexport type SsrStreamOptions = SsrRenderOptions & {\n /**\n * Called once React has flushed the shell (head + above-fallback content)\n * and the stream is ready to pipe. The framework uses this hook to write\n * the opening template + state scripts before piping React's HTML.\n */\n onShellReady?: () => void;\n /** Non-fatal SSR errors (Suspense fallbacks, etc.). */\n onError?: (error: unknown) => void;\n};\n\nexport async function renderSsrTree(options: SsrRenderOptions): Promise<SsrRenderResult> {\n const { callContext, loadingElement, routesElement, router, location } = options;\n\n ensureSsrResolversInstalled();\n\n const sessionPayload = await callInProcessMethod<SessionInitPayload>(\n '_system.session.init',\n {},\n callContext\n );\n\n const queryClient = new QueryClient({\n defaultOptions: {\n queries: {\n retry: false,\n // Per-request client; release cache + timers immediately after dehydrate.\n gcTime: 0,\n },\n },\n });\n\n const routedTree = router ? router({ children: routesElement, location }) : routesElement;\n\n const tree = (\n <AppProvider loadingElement={loadingElement}>\n <ModelenceQueryProvider client={queryClient}>{routedTree}</ModelenceQueryProvider>\n </AppProvider>\n );\n\n const html = await runWithSsrContext(\n {\n callContext,\n queryClient,\n session: {\n user: sessionPayload.user,\n configs: (sessionPayload.configs as Configs) ?? {},\n },\n },\n () => renderToString(tree, location)\n );\n\n // Safe to dehydrate + clear here: `onAllReady` waits for every Suspense\n // boundary to settle before resolving, so all in-flight queries have\n // populated the cache by this point.\n const dehydratedState: DehydratedState = dehydrate(queryClient);\n queryClient.clear();\n\n return {\n html,\n sessionState: JSON.stringify({ session: sessionPayload }),\n queryState: JSON.stringify(dehydratedState),\n };\n}\n\n/**\n * Streaming variant of {@link renderSsrTree}. Returns a handle that lets the\n * caller flush a template prelude (head + opening shell) as soon as the\n * React shell is ready, pipe the React HTML stream into the response, then\n * append the dehydrated query state once streaming completes.\n *\n * This is what enables fast First Contentful Paint: the browser receives the\n * <head> (with CSS <link> tags) immediately, starts the stylesheet fetch in\n * parallel with the HTML stream, and paints the streamed shell with styles\n * applied — instead of the dev-mode FOUC caused by JS-injected CSS.\n */\nexport async function renderSsrTreeStream(options: SsrStreamOptions): Promise<SsrStreamHandle> {\n const { callContext, loadingElement, routesElement, router, location, onShellReady, onError } =\n options;\n\n ensureSsrResolversInstalled();\n\n const sessionPayload = await callInProcessMethod<SessionInitPayload>(\n '_system.session.init',\n {},\n callContext\n );\n\n const queryClient = new QueryClient({\n defaultOptions: {\n queries: {\n retry: false,\n gcTime: 0,\n },\n },\n });\n\n const routedTree = router ? router({ children: routesElement, location }) : routesElement;\n const tree = (\n <AppProvider loadingElement={loadingElement}>\n <ModelenceQueryProvider client={queryClient}>{routedTree}</ModelenceQueryProvider>\n </AppProvider>\n );\n\n let streamRef: PipeableStream | null = null;\n // The shell-ready promise resolves with the PipeableStream as soon as React\n // has rendered above-fallback content. Errors during the shell render\n // reject it so the caller can fall back to a static response.\n const shellReady = new Promise<PipeableStream>((resolve, reject) => {\n // Run the render inside the SSR context so server-rendered components\n // can resolve session/config/query state from the per-request scope.\n runWithSsrContext(\n {\n callContext,\n queryClient,\n session: {\n user: sessionPayload.user,\n configs: (sessionPayload.configs as Configs) ?? {},\n },\n },\n () => {\n const stream = renderToPipeableStream(tree, {\n onShellReady() {\n streamRef = stream;\n onShellReady?.();\n resolve(stream);\n },\n onShellError(error) {\n reject(error);\n },\n onError(error) {\n onError?.(error);\n },\n });\n return stream;\n }\n );\n });\n\n // Surface shell errors synchronously by awaiting before returning the handle.\n await shellReady;\n\n let queryStateJson: string | null = null;\n\n const pipe = (destination: Writable): Promise<void> => {\n return new Promise((resolve, reject) => {\n if (!streamRef) {\n reject(new Error('SSR stream was not initialized'));\n return;\n }\n\n destination.on('finish', () => {\n // React has finished streaming; dehydrate now so every query that\n // resolved during the stream is captured.\n try {\n const dehydratedState: DehydratedState = dehydrate(queryClient);\n queryStateJson = JSON.stringify(dehydratedState);\n } finally {\n queryClient.clear();\n }\n resolve();\n });\n destination.on('error', reject);\n\n streamRef.pipe(destination);\n });\n };\n\n return {\n sessionState: JSON.stringify({ session: sessionPayload }),\n pipe,\n getQueryState: () => {\n if (queryStateJson === null) {\n throw new Error('getQueryState() called before stream finished');\n }\n return queryStateJson;\n },\n };\n}\n\nfunction renderToString(tree: React.ReactElement, location: string | undefined): Promise<string> {\n return new Promise((resolve, reject) => {\n let buffer = '';\n const writable = new Writable({\n write(chunk, _encoding, callback) {\n buffer += chunk.toString();\n callback();\n },\n });\n\n const { pipe } = renderToPipeableStream(tree, {\n onAllReady() {\n writable.on('finish', () => resolve(buffer));\n writable.on('error', reject);\n pipe(writable);\n },\n onShellError(error) {\n reject(error);\n },\n onError(error) {\n // Non-fatal recoverable errors (Suspense fallbacks, etc.). React still\n // ships HTML, but repeated occurrences can mask real bugs — log with\n // request URL for telemetry follow-up.\n console.error('SSR onError (non-fatal):', { location, error });\n },\n });\n });\n}\n"]}
@@ -0,0 +1,2 @@
1
+ export{c as _getSsrSnapshot,d as renderApp}from'./chunk-S77LRZU6.js';import'./chunk-IJ5BS7PM.js';import'./chunk-VFOGVWJK.js';import'./chunk-5M6FUMUK.js';import'./chunk-DO5TZLF5.js';//# sourceMappingURL=renderApp-3FQ6KUQ2.js.map
2
+ //# sourceMappingURL=renderApp-3FQ6KUQ2.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":[],"names":[],"mappings":"","file":"renderApp-3FQ6KUQ2.js"}
@@ -0,0 +1,2 @@
1
+ export{g as getCallContext,f as startServer}from'./chunk-S2E4OO6G.js';import'./chunk-KU4GANYW.js';import'./chunk-LDVLXJLS.js';import'./chunk-C3UESBRX.js';import'./chunk-3SPXJEOR.js';import'./chunk-5M6FUMUK.js';import'./chunk-DO5TZLF5.js';//# sourceMappingURL=server-JB4DTGBT.js.map
2
+ //# sourceMappingURL=server-JB4DTGBT.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":[],"names":[],"mappings":"","file":"server-JB4DTGBT.js"}
package/dist/server.d.ts CHANGED
@@ -1,8 +1,8 @@
1
- import { A as AppServer } from './index-CLpVWWuj.js';
2
- import { c as ConfigParams, C as ConfigSchema, A as AnyMethodShape, S as ServerChannel, V as ValueType, f as SignupProps, U as UpdateProfileProps, g as AuthSuccessProps, h as AuthErrorProps, i as User, j as OAuthErrorInfo, a as WebsocketServerProvider, R as RoleDefinition, M as MethodDefinition, b as ConfigKey, k as AppConfig, l as Session, m as UserInfo, n as Role } from './types-CTjq1HaP.js';
3
- export { d as ConfigType } from './types-CTjq1HaP.js';
4
- import { S as Store, R as RouteDefinition, C as CronJobInputParams, a as RateLimitRule, E as EmailProvider, I as InferDocumentType, b as RateLimitType, c as EmailPayload } from './types-DNZiaNSs.js';
5
- export { H as HttpMethod, d as RouteHandler, e as RouteParams, f as RouteResponse, s as schema } from './types-DNZiaNSs.js';
1
+ import { A as AppServer } from './index-BBMsjrFN.js';
2
+ import { c as ConfigParams, C as ConfigSchema, A as AnyMethodShape, S as ServerChannel, V as ValueType, f as SignupProps, U as UpdateProfileProps, g as AuthSuccessProps, h as AuthErrorProps, i as User, j as OAuthErrorInfo, a as WebsocketServerProvider, R as RoleDefinition, M as MethodDefinition, b as ConfigKey, k as AppConfig, l as Session, m as UserInfo, n as Role } from './types-DV0J5rPI.js';
3
+ export { d as ConfigType } from './types-DV0J5rPI.js';
4
+ import { S as Store, R as RouteDefinition, C as CronJobInputParams, a as RateLimitRule, E as EmailProvider, I as InferDocumentType, b as RateLimitType, c as EmailPayload } from './types-mkbhnQN2.js';
5
+ export { H as HttpMethod, d as RouteHandler, e as RouteParams, f as RouteResponse, s as schema } from './types-mkbhnQN2.js';
6
6
  import { ObjectId as ObjectId$1 } from 'mongodb';
7
7
  export { ObjectId } from 'mongodb';
8
8
  import * as zod from 'zod';
@@ -331,8 +331,10 @@ type AppOptions = {
331
331
  defaultRoles?: Record<string, string>;
332
332
  migrations?: Array<MigrationScript>;
333
333
  websocket?: WebsocketConfig;
334
+ /** Enable server-side rendering of the user's React tree. */
335
+ ssr?: boolean;
334
336
  };
335
- declare function startApp({ modules, roles, defaultRoles, server, migrations, email, auth, security, websocket, }: AppOptions): Promise<void>;
337
+ declare function startApp({ modules, roles, defaultRoles, server, migrations, email, auth, security, websocket, ssr, }: AppOptions): Promise<void>;
336
338
 
337
339
  /**
338
340
  * Publish function provided to watch handlers.