@objectstack/sveltekit 3.2.0 → 3.2.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.turbo/turbo-build.log +8 -8
- package/CHANGELOG.md +12 -0
- package/dist/index.js +10 -1
- package/dist/index.js.map +1 -1
- package/dist/index.mjs +10 -1
- package/dist/index.mjs.map +1 -1
- package/package.json +3 -3
- package/src/index.ts +12 -4
- package/src/sveltekit.test.ts +31 -0
package/.turbo/turbo-build.log
CHANGED
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
|
|
2
|
-
> @objectstack/sveltekit@3.2.
|
|
2
|
+
> @objectstack/sveltekit@3.2.2 build /home/runner/work/spec/spec/packages/adapters/sveltekit
|
|
3
3
|
> tsup --config ../../../tsup.config.ts
|
|
4
4
|
|
|
5
5
|
[34mCLI[39m Building entry: src/index.ts
|
|
@@ -10,13 +10,13 @@
|
|
|
10
10
|
[34mCLI[39m Cleaning output folder
|
|
11
11
|
[34mESM[39m Build start
|
|
12
12
|
[34mCJS[39m Build start
|
|
13
|
-
[
|
|
14
|
-
[
|
|
15
|
-
[
|
|
16
|
-
[
|
|
17
|
-
[
|
|
18
|
-
[
|
|
13
|
+
[32mESM[39m [1mdist/index.mjs [22m[32m5.09 KB[39m
|
|
14
|
+
[32mESM[39m [1mdist/index.mjs.map [22m[32m10.99 KB[39m
|
|
15
|
+
[32mESM[39m ⚡️ Build success in 71ms
|
|
16
|
+
[32mCJS[39m [1mdist/index.js [22m[32m6.15 KB[39m
|
|
17
|
+
[32mCJS[39m [1mdist/index.js.map [22m[32m11.03 KB[39m
|
|
18
|
+
[32mCJS[39m ⚡️ Build success in 86ms
|
|
19
19
|
[34mDTS[39m Build start
|
|
20
|
-
[32mDTS[39m ⚡️ Build success in
|
|
20
|
+
[32mDTS[39m ⚡️ Build success in 12704ms
|
|
21
21
|
[32mDTS[39m [1mdist/index.d.mts [22m[32m1.50 KB[39m
|
|
22
22
|
[32mDTS[39m [1mdist/index.d.ts [22m[32m1.50 KB[39m
|
package/CHANGELOG.md
CHANGED
package/dist/index.js
CHANGED
|
@@ -83,7 +83,16 @@ function createRequestHandler(options) {
|
|
|
83
83
|
try {
|
|
84
84
|
if (segments[0] === "auth") {
|
|
85
85
|
const subPath = segments.slice(1).join("/");
|
|
86
|
-
|
|
86
|
+
let authService = null;
|
|
87
|
+
try {
|
|
88
|
+
if (typeof options.kernel.getServiceAsync === "function") {
|
|
89
|
+
authService = await options.kernel.getServiceAsync("auth");
|
|
90
|
+
} else if (typeof options.kernel.getService === "function") {
|
|
91
|
+
authService = options.kernel.getService("auth");
|
|
92
|
+
}
|
|
93
|
+
} catch {
|
|
94
|
+
authService = null;
|
|
95
|
+
}
|
|
87
96
|
if (authService && typeof authService.handleRequest === "function") {
|
|
88
97
|
return await authService.handleRequest(request);
|
|
89
98
|
}
|
package/dist/index.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/index.ts"],"sourcesContent":["// Copyright (c) 2025 ObjectStack. Licensed under the Apache-2.0 license.\n\nimport { type ObjectKernel, HttpDispatcher, HttpDispatcherResult } from '@objectstack/runtime';\n\nexport interface SvelteKitAdapterOptions {\n kernel: ObjectKernel;\n prefix?: string;\n}\n\n/**\n * Auth service interface with handleRequest method\n */\ninterface AuthService {\n handleRequest(request: Request): Promise<Response>;\n}\n\n/**\n * SvelteKit request event type (minimal interface to avoid hard dependency on @sveltejs/kit types at runtime)\n */\ninterface RequestEvent {\n request: Request;\n url: URL;\n params: Record<string, string>;\n}\n\n/**\n * Creates a SvelteKit request handler for ObjectStack API routes.\n * Use in a catch-all `+server.ts` route like `src/routes/api/[...path]/+server.ts`.\n *\n * @example\n * ```ts\n * // src/routes/api/[...path]/+server.ts\n * import { createRequestHandler } from '@objectstack/sveltekit';\n * import { kernel } from '$lib/kernel';\n *\n * const handler = createRequestHandler({ kernel });\n *\n * export const GET = handler;\n * export const POST = handler;\n * export const PUT = handler;\n * export const PATCH = handler;\n * export const DELETE = handler;\n * ```\n */\nexport function createRequestHandler(options: SvelteKitAdapterOptions) {\n const dispatcher = new HttpDispatcher(options.kernel);\n const prefix = options.prefix || '/api';\n\n const errorJson = (message: string, code: number = 500) => {\n return new Response(JSON.stringify({ success: false, error: { message, code } }), {\n status: code,\n headers: { 'Content-Type': 'application/json' },\n });\n };\n\n const toResponse = (result: HttpDispatcherResult): Response => {\n if (result.handled) {\n if (result.response) {\n const headers = new Headers({ 'Content-Type': 'application/json' });\n if (result.response.headers) {\n Object.entries(result.response.headers).forEach(([k, v]) => headers.set(k, v as string));\n }\n return new Response(JSON.stringify(result.response.body), {\n status: result.response.status,\n headers,\n });\n }\n if (result.result) {\n const res = result.result;\n if (res.type === 'redirect' && res.url) {\n return new Response(null, {\n status: 302,\n headers: { Location: res.url },\n });\n }\n if (res.type === 'stream' && res.stream) {\n const headers = new Headers();\n if (res.headers) {\n Object.entries(res.headers).forEach(([k, v]) => headers.set(k, v as string));\n }\n return new Response(res.stream, { status: 200, headers });\n }\n return new Response(JSON.stringify(res), {\n status: 200,\n headers: { 'Content-Type': 'application/json' },\n });\n }\n }\n return errorJson('Not Found', 404);\n };\n\n return async function handler(event: RequestEvent): Promise<Response> {\n const { request, url } = event;\n const method = request.method;\n const path = url.pathname.substring(prefix.length);\n const segments = path.split('/').filter(Boolean);\n\n // --- Discovery ---\n if (segments.length === 0 && method === 'GET') {\n return new Response(JSON.stringify({ data: dispatcher.getDiscoveryInfo(prefix) }), {\n status: 200,\n headers: { 'Content-Type': 'application/json' },\n });\n }\n\n try {\n // --- Auth ---\n if (segments[0] === 'auth') {\n const subPath = segments.slice(1).join('/');\n\n // Try AuthPlugin service first\n const authService = typeof options.kernel.getService === 'function'\n ? options.kernel.getService<AuthService>('auth')\n : null;\n\n if (authService && typeof authService.handleRequest === 'function') {\n return await authService.handleRequest(request);\n }\n\n // Fallback to dispatcher\n const body = method === 'GET' || method === 'HEAD'\n ? {}\n : await request.json().catch(() => ({}));\n const result = await dispatcher.handleAuth(subPath, method, body, { request });\n return toResponse(result);\n }\n\n // --- GraphQL ---\n if (segments[0] === 'graphql' && method === 'POST') {\n const body = await request.json() as { query: string; variables?: any };\n const result = await dispatcher.handleGraphQL(body, { request });\n return new Response(JSON.stringify(result), {\n status: 200,\n headers: { 'Content-Type': 'application/json' },\n });\n }\n\n // --- Metadata ---\n if (segments[0] === 'meta') {\n const subPath = segments.slice(1).join('/');\n let body: any = undefined;\n if (method === 'PUT' || method === 'POST') {\n body = await request.json().catch(() => ({}));\n }\n const result = await dispatcher.handleMetadata(\n subPath ? `/${subPath}` : '',\n { request },\n method,\n body,\n );\n return toResponse(result);\n }\n\n // --- Data ---\n if (segments[0] === 'data') {\n const subPath = segments.slice(1).join('/');\n let body: any = {};\n if (method === 'POST' || method === 'PATCH') {\n body = await request.json().catch(() => ({}));\n }\n const queryParams: Record<string, any> = {};\n url.searchParams.forEach((val, key) => { queryParams[key] = val; });\n\n const result = await dispatcher.handleData(\n subPath ? `/${subPath}` : '',\n method,\n body,\n queryParams,\n { request },\n );\n return toResponse(result);\n }\n\n // --- Storage ---\n if (segments[0] === 'storage') {\n const subPath = segments.slice(1).join('/');\n let file: any = undefined;\n if (method === 'POST' && subPath === 'upload') {\n const formData = await request.formData();\n file = formData.get('file');\n }\n const result = await dispatcher.handleStorage(\n subPath ? `/${subPath}` : '',\n method,\n file,\n { request },\n );\n return toResponse(result);\n }\n\n return errorJson('Not Found', 404);\n } catch (err: any) {\n return errorJson(err.message || 'Internal Server Error', err.statusCode || 500);\n }\n };\n}\n\n/**\n * Creates a SvelteKit handle hook that attaches the kernel to event.locals.\n *\n * @example\n * ```ts\n * // src/hooks.server.ts\n * import { createHandle } from '@objectstack/sveltekit';\n * export const handle = createHandle({ kernel });\n * ```\n */\nexport function createHandle(options: SvelteKitAdapterOptions) {\n return async function handle({ event, resolve }: { event: any; resolve: (event: any) => Promise<Response> }) {\n event.locals.objectStack = options.kernel;\n return resolve(event);\n };\n}\n"],"mappings":";;;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAEA,qBAAwE;AA0CjE,SAAS,qBAAqB,SAAkC;AACrE,QAAM,aAAa,IAAI,8BAAe,QAAQ,MAAM;AACpD,QAAM,SAAS,QAAQ,UAAU;AAEjC,QAAM,YAAY,CAAC,SAAiB,OAAe,QAAQ;AACzD,WAAO,IAAI,SAAS,KAAK,UAAU,EAAE,SAAS,OAAO,OAAO,EAAE,SAAS,KAAK,EAAE,CAAC,GAAG;AAAA,MAChF,QAAQ;AAAA,MACR,SAAS,EAAE,gBAAgB,mBAAmB;AAAA,IAChD,CAAC;AAAA,EACH;AAEA,QAAM,aAAa,CAAC,WAA2C;AAC7D,QAAI,OAAO,SAAS;AAClB,UAAI,OAAO,UAAU;AACnB,cAAM,UAAU,IAAI,QAAQ,EAAE,gBAAgB,mBAAmB,CAAC;AAClE,YAAI,OAAO,SAAS,SAAS;AAC3B,iBAAO,QAAQ,OAAO,SAAS,OAAO,EAAE,QAAQ,CAAC,CAAC,GAAG,CAAC,MAAM,QAAQ,IAAI,GAAG,CAAW,CAAC;AAAA,QACzF;AACA,eAAO,IAAI,SAAS,KAAK,UAAU,OAAO,SAAS,IAAI,GAAG;AAAA,UACxD,QAAQ,OAAO,SAAS;AAAA,UACxB;AAAA,QACF,CAAC;AAAA,MACH;AACA,UAAI,OAAO,QAAQ;AACjB,cAAM,MAAM,OAAO;AACnB,YAAI,IAAI,SAAS,cAAc,IAAI,KAAK;AACtC,iBAAO,IAAI,SAAS,MAAM;AAAA,YACxB,QAAQ;AAAA,YACR,SAAS,EAAE,UAAU,IAAI,IAAI;AAAA,UAC/B,CAAC;AAAA,QACH;AACA,YAAI,IAAI,SAAS,YAAY,IAAI,QAAQ;AACvC,gBAAM,UAAU,IAAI,QAAQ;AAC5B,cAAI,IAAI,SAAS;AACf,mBAAO,QAAQ,IAAI,OAAO,EAAE,QAAQ,CAAC,CAAC,GAAG,CAAC,MAAM,QAAQ,IAAI,GAAG,CAAW,CAAC;AAAA,UAC7E;AACA,iBAAO,IAAI,SAAS,IAAI,QAAQ,EAAE,QAAQ,KAAK,QAAQ,CAAC;AAAA,QAC1D;AACA,eAAO,IAAI,SAAS,KAAK,UAAU,GAAG,GAAG;AAAA,UACvC,QAAQ;AAAA,UACR,SAAS,EAAE,gBAAgB,mBAAmB;AAAA,QAChD,CAAC;AAAA,MACH;AAAA,IACF;AACA,WAAO,UAAU,aAAa,GAAG;AAAA,EACnC;AAEA,SAAO,eAAe,QAAQ,OAAwC;AACpE,UAAM,EAAE,SAAS,IAAI,IAAI;AACzB,UAAM,SAAS,QAAQ;AACvB,UAAM,OAAO,IAAI,SAAS,UAAU,OAAO,MAAM;AACjD,UAAM,WAAW,KAAK,MAAM,GAAG,EAAE,OAAO,OAAO;AAG/C,QAAI,SAAS,WAAW,KAAK,WAAW,OAAO;AAC7C,aAAO,IAAI,SAAS,KAAK,UAAU,EAAE,MAAM,WAAW,iBAAiB,MAAM,EAAE,CAAC,GAAG;AAAA,QACjF,QAAQ;AAAA,QACR,SAAS,EAAE,gBAAgB,mBAAmB;AAAA,MAChD,CAAC;AAAA,IACH;AAEA,QAAI;AAEF,UAAI,SAAS,CAAC,MAAM,QAAQ;AAC1B,cAAM,UAAU,SAAS,MAAM,CAAC,EAAE,KAAK,GAAG;AAG1C,cAAM,cAAc,OAAO,QAAQ,OAAO,eAAe,aACrD,QAAQ,OAAO,WAAwB,MAAM,IAC7C;AAEJ,YAAI,eAAe,OAAO,YAAY,kBAAkB,YAAY;AAClE,iBAAO,MAAM,YAAY,cAAc,OAAO;AAAA,QAChD;AAGA,cAAM,OAAO,WAAW,SAAS,WAAW,SACxC,CAAC,IACD,MAAM,QAAQ,KAAK,EAAE,MAAM,OAAO,CAAC,EAAE;AACzC,cAAM,SAAS,MAAM,WAAW,WAAW,SAAS,QAAQ,MAAM,EAAE,QAAQ,CAAC;AAC7E,eAAO,WAAW,MAAM;AAAA,MAC1B;AAGA,UAAI,SAAS,CAAC,MAAM,aAAa,WAAW,QAAQ;AAClD,cAAM,OAAO,MAAM,QAAQ,KAAK;AAChC,cAAM,SAAS,MAAM,WAAW,cAAc,MAAM,EAAE,QAAQ,CAAC;AAC/D,eAAO,IAAI,SAAS,KAAK,UAAU,MAAM,GAAG;AAAA,UAC1C,QAAQ;AAAA,UACR,SAAS,EAAE,gBAAgB,mBAAmB;AAAA,QAChD,CAAC;AAAA,MACH;AAGA,UAAI,SAAS,CAAC,MAAM,QAAQ;AAC1B,cAAM,UAAU,SAAS,MAAM,CAAC,EAAE,KAAK,GAAG;AAC1C,YAAI,OAAY;AAChB,YAAI,WAAW,SAAS,WAAW,QAAQ;AACzC,iBAAO,MAAM,QAAQ,KAAK,EAAE,MAAM,OAAO,CAAC,EAAE;AAAA,QAC9C;AACA,cAAM,SAAS,MAAM,WAAW;AAAA,UAC9B,UAAU,IAAI,OAAO,KAAK;AAAA,UAC1B,EAAE,QAAQ;AAAA,UACV;AAAA,UACA;AAAA,QACF;AACA,eAAO,WAAW,MAAM;AAAA,MAC1B;AAGA,UAAI,SAAS,CAAC,MAAM,QAAQ;AAC1B,cAAM,UAAU,SAAS,MAAM,CAAC,EAAE,KAAK,GAAG;AAC1C,YAAI,OAAY,CAAC;AACjB,YAAI,WAAW,UAAU,WAAW,SAAS;AAC3C,iBAAO,MAAM,QAAQ,KAAK,EAAE,MAAM,OAAO,CAAC,EAAE;AAAA,QAC9C;AACA,cAAM,cAAmC,CAAC;AAC1C,YAAI,aAAa,QAAQ,CAAC,KAAK,QAAQ;AAAE,sBAAY,GAAG,IAAI;AAAA,QAAK,CAAC;AAElE,cAAM,SAAS,MAAM,WAAW;AAAA,UAC9B,UAAU,IAAI,OAAO,KAAK;AAAA,UAC1B;AAAA,UACA;AAAA,UACA;AAAA,UACA,EAAE,QAAQ;AAAA,QACZ;AACA,eAAO,WAAW,MAAM;AAAA,MAC1B;AAGA,UAAI,SAAS,CAAC,MAAM,WAAW;AAC7B,cAAM,UAAU,SAAS,MAAM,CAAC,EAAE,KAAK,GAAG;AAC1C,YAAI,OAAY;AAChB,YAAI,WAAW,UAAU,YAAY,UAAU;AAC7C,gBAAM,WAAW,MAAM,QAAQ,SAAS;AACxC,iBAAO,SAAS,IAAI,MAAM;AAAA,QAC5B;AACA,cAAM,SAAS,MAAM,WAAW;AAAA,UAC9B,UAAU,IAAI,OAAO,KAAK;AAAA,UAC1B;AAAA,UACA;AAAA,UACA,EAAE,QAAQ;AAAA,QACZ;AACA,eAAO,WAAW,MAAM;AAAA,MAC1B;AAEA,aAAO,UAAU,aAAa,GAAG;AAAA,IACnC,SAAS,KAAU;AACjB,aAAO,UAAU,IAAI,WAAW,yBAAyB,IAAI,cAAc,GAAG;AAAA,IAChF;AAAA,EACF;AACF;AAYO,SAAS,aAAa,SAAkC;AAC7D,SAAO,eAAe,OAAO,EAAE,OAAO,QAAQ,GAA+D;AAC3G,UAAM,OAAO,cAAc,QAAQ;AACnC,WAAO,QAAQ,KAAK;AAAA,EACtB;AACF;","names":[]}
|
|
1
|
+
{"version":3,"sources":["../src/index.ts"],"sourcesContent":["// Copyright (c) 2025 ObjectStack. Licensed under the Apache-2.0 license.\n\nimport { type ObjectKernel, HttpDispatcher, HttpDispatcherResult } from '@objectstack/runtime';\n\nexport interface SvelteKitAdapterOptions {\n kernel: ObjectKernel;\n prefix?: string;\n}\n\n/**\n * Auth service interface with handleRequest method\n */\ninterface AuthService {\n handleRequest(request: Request): Promise<Response>;\n}\n\n/**\n * SvelteKit request event type (minimal interface to avoid hard dependency on @sveltejs/kit types at runtime)\n */\ninterface RequestEvent {\n request: Request;\n url: URL;\n params: Record<string, string>;\n}\n\n/**\n * Creates a SvelteKit request handler for ObjectStack API routes.\n * Use in a catch-all `+server.ts` route like `src/routes/api/[...path]/+server.ts`.\n *\n * @example\n * ```ts\n * // src/routes/api/[...path]/+server.ts\n * import { createRequestHandler } from '@objectstack/sveltekit';\n * import { kernel } from '$lib/kernel';\n *\n * const handler = createRequestHandler({ kernel });\n *\n * export const GET = handler;\n * export const POST = handler;\n * export const PUT = handler;\n * export const PATCH = handler;\n * export const DELETE = handler;\n * ```\n */\nexport function createRequestHandler(options: SvelteKitAdapterOptions) {\n const dispatcher = new HttpDispatcher(options.kernel);\n const prefix = options.prefix || '/api';\n\n const errorJson = (message: string, code: number = 500) => {\n return new Response(JSON.stringify({ success: false, error: { message, code } }), {\n status: code,\n headers: { 'Content-Type': 'application/json' },\n });\n };\n\n const toResponse = (result: HttpDispatcherResult): Response => {\n if (result.handled) {\n if (result.response) {\n const headers = new Headers({ 'Content-Type': 'application/json' });\n if (result.response.headers) {\n Object.entries(result.response.headers).forEach(([k, v]) => headers.set(k, v as string));\n }\n return new Response(JSON.stringify(result.response.body), {\n status: result.response.status,\n headers,\n });\n }\n if (result.result) {\n const res = result.result;\n if (res.type === 'redirect' && res.url) {\n return new Response(null, {\n status: 302,\n headers: { Location: res.url },\n });\n }\n if (res.type === 'stream' && res.stream) {\n const headers = new Headers();\n if (res.headers) {\n Object.entries(res.headers).forEach(([k, v]) => headers.set(k, v as string));\n }\n return new Response(res.stream, { status: 200, headers });\n }\n return new Response(JSON.stringify(res), {\n status: 200,\n headers: { 'Content-Type': 'application/json' },\n });\n }\n }\n return errorJson('Not Found', 404);\n };\n\n return async function handler(event: RequestEvent): Promise<Response> {\n const { request, url } = event;\n const method = request.method;\n const path = url.pathname.substring(prefix.length);\n const segments = path.split('/').filter(Boolean);\n\n // --- Discovery ---\n if (segments.length === 0 && method === 'GET') {\n return new Response(JSON.stringify({ data: dispatcher.getDiscoveryInfo(prefix) }), {\n status: 200,\n headers: { 'Content-Type': 'application/json' },\n });\n }\n\n try {\n // --- Auth ---\n if (segments[0] === 'auth') {\n const subPath = segments.slice(1).join('/');\n\n // Try AuthPlugin service first (prefer async to support factory-based services)\n let authService: AuthService | null = null;\n try {\n if (typeof options.kernel.getServiceAsync === 'function') {\n authService = await options.kernel.getServiceAsync<AuthService>('auth');\n } else if (typeof options.kernel.getService === 'function') {\n authService = options.kernel.getService<AuthService>('auth');\n }\n } catch {\n // Service not registered — fall through to dispatcher\n authService = null;\n }\n\n if (authService && typeof authService.handleRequest === 'function') {\n return await authService.handleRequest(request);\n }\n\n // Fallback to dispatcher\n const body = method === 'GET' || method === 'HEAD'\n ? {}\n : await request.json().catch(() => ({}));\n const result = await dispatcher.handleAuth(subPath, method, body, { request });\n return toResponse(result);\n }\n\n // --- GraphQL ---\n if (segments[0] === 'graphql' && method === 'POST') {\n const body = await request.json() as { query: string; variables?: any };\n const result = await dispatcher.handleGraphQL(body, { request });\n return new Response(JSON.stringify(result), {\n status: 200,\n headers: { 'Content-Type': 'application/json' },\n });\n }\n\n // --- Metadata ---\n if (segments[0] === 'meta') {\n const subPath = segments.slice(1).join('/');\n let body: any = undefined;\n if (method === 'PUT' || method === 'POST') {\n body = await request.json().catch(() => ({}));\n }\n const result = await dispatcher.handleMetadata(\n subPath ? `/${subPath}` : '',\n { request },\n method,\n body,\n );\n return toResponse(result);\n }\n\n // --- Data ---\n if (segments[0] === 'data') {\n const subPath = segments.slice(1).join('/');\n let body: any = {};\n if (method === 'POST' || method === 'PATCH') {\n body = await request.json().catch(() => ({}));\n }\n const queryParams: Record<string, any> = {};\n url.searchParams.forEach((val, key) => { queryParams[key] = val; });\n\n const result = await dispatcher.handleData(\n subPath ? `/${subPath}` : '',\n method,\n body,\n queryParams,\n { request },\n );\n return toResponse(result);\n }\n\n // --- Storage ---\n if (segments[0] === 'storage') {\n const subPath = segments.slice(1).join('/');\n let file: any = undefined;\n if (method === 'POST' && subPath === 'upload') {\n const formData = await request.formData();\n file = formData.get('file');\n }\n const result = await dispatcher.handleStorage(\n subPath ? `/${subPath}` : '',\n method,\n file,\n { request },\n );\n return toResponse(result);\n }\n\n return errorJson('Not Found', 404);\n } catch (err: any) {\n return errorJson(err.message || 'Internal Server Error', err.statusCode || 500);\n }\n };\n}\n\n/**\n * Creates a SvelteKit handle hook that attaches the kernel to event.locals.\n *\n * @example\n * ```ts\n * // src/hooks.server.ts\n * import { createHandle } from '@objectstack/sveltekit';\n * export const handle = createHandle({ kernel });\n * ```\n */\nexport function createHandle(options: SvelteKitAdapterOptions) {\n return async function handle({ event, resolve }: { event: any; resolve: (event: any) => Promise<Response> }) {\n event.locals.objectStack = options.kernel;\n return resolve(event);\n };\n}\n"],"mappings":"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","names":[]}
|
package/dist/index.mjs
CHANGED
|
@@ -58,7 +58,16 @@ function createRequestHandler(options) {
|
|
|
58
58
|
try {
|
|
59
59
|
if (segments[0] === "auth") {
|
|
60
60
|
const subPath = segments.slice(1).join("/");
|
|
61
|
-
|
|
61
|
+
let authService = null;
|
|
62
|
+
try {
|
|
63
|
+
if (typeof options.kernel.getServiceAsync === "function") {
|
|
64
|
+
authService = await options.kernel.getServiceAsync("auth");
|
|
65
|
+
} else if (typeof options.kernel.getService === "function") {
|
|
66
|
+
authService = options.kernel.getService("auth");
|
|
67
|
+
}
|
|
68
|
+
} catch {
|
|
69
|
+
authService = null;
|
|
70
|
+
}
|
|
62
71
|
if (authService && typeof authService.handleRequest === "function") {
|
|
63
72
|
return await authService.handleRequest(request);
|
|
64
73
|
}
|
package/dist/index.mjs.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/index.ts"],"sourcesContent":["// Copyright (c) 2025 ObjectStack. Licensed under the Apache-2.0 license.\n\nimport { type ObjectKernel, HttpDispatcher, HttpDispatcherResult } from '@objectstack/runtime';\n\nexport interface SvelteKitAdapterOptions {\n kernel: ObjectKernel;\n prefix?: string;\n}\n\n/**\n * Auth service interface with handleRequest method\n */\ninterface AuthService {\n handleRequest(request: Request): Promise<Response>;\n}\n\n/**\n * SvelteKit request event type (minimal interface to avoid hard dependency on @sveltejs/kit types at runtime)\n */\ninterface RequestEvent {\n request: Request;\n url: URL;\n params: Record<string, string>;\n}\n\n/**\n * Creates a SvelteKit request handler for ObjectStack API routes.\n * Use in a catch-all `+server.ts` route like `src/routes/api/[...path]/+server.ts`.\n *\n * @example\n * ```ts\n * // src/routes/api/[...path]/+server.ts\n * import { createRequestHandler } from '@objectstack/sveltekit';\n * import { kernel } from '$lib/kernel';\n *\n * const handler = createRequestHandler({ kernel });\n *\n * export const GET = handler;\n * export const POST = handler;\n * export const PUT = handler;\n * export const PATCH = handler;\n * export const DELETE = handler;\n * ```\n */\nexport function createRequestHandler(options: SvelteKitAdapterOptions) {\n const dispatcher = new HttpDispatcher(options.kernel);\n const prefix = options.prefix || '/api';\n\n const errorJson = (message: string, code: number = 500) => {\n return new Response(JSON.stringify({ success: false, error: { message, code } }), {\n status: code,\n headers: { 'Content-Type': 'application/json' },\n });\n };\n\n const toResponse = (result: HttpDispatcherResult): Response => {\n if (result.handled) {\n if (result.response) {\n const headers = new Headers({ 'Content-Type': 'application/json' });\n if (result.response.headers) {\n Object.entries(result.response.headers).forEach(([k, v]) => headers.set(k, v as string));\n }\n return new Response(JSON.stringify(result.response.body), {\n status: result.response.status,\n headers,\n });\n }\n if (result.result) {\n const res = result.result;\n if (res.type === 'redirect' && res.url) {\n return new Response(null, {\n status: 302,\n headers: { Location: res.url },\n });\n }\n if (res.type === 'stream' && res.stream) {\n const headers = new Headers();\n if (res.headers) {\n Object.entries(res.headers).forEach(([k, v]) => headers.set(k, v as string));\n }\n return new Response(res.stream, { status: 200, headers });\n }\n return new Response(JSON.stringify(res), {\n status: 200,\n headers: { 'Content-Type': 'application/json' },\n });\n }\n }\n return errorJson('Not Found', 404);\n };\n\n return async function handler(event: RequestEvent): Promise<Response> {\n const { request, url } = event;\n const method = request.method;\n const path = url.pathname.substring(prefix.length);\n const segments = path.split('/').filter(Boolean);\n\n // --- Discovery ---\n if (segments.length === 0 && method === 'GET') {\n return new Response(JSON.stringify({ data: dispatcher.getDiscoveryInfo(prefix) }), {\n status: 200,\n headers: { 'Content-Type': 'application/json' },\n });\n }\n\n try {\n // --- Auth ---\n if (segments[0] === 'auth') {\n const subPath = segments.slice(1).join('/');\n\n // Try AuthPlugin service first\n const authService = typeof options.kernel.getService === 'function'\n ? options.kernel.getService<AuthService>('auth')\n : null;\n\n if (authService && typeof authService.handleRequest === 'function') {\n return await authService.handleRequest(request);\n }\n\n // Fallback to dispatcher\n const body = method === 'GET' || method === 'HEAD'\n ? {}\n : await request.json().catch(() => ({}));\n const result = await dispatcher.handleAuth(subPath, method, body, { request });\n return toResponse(result);\n }\n\n // --- GraphQL ---\n if (segments[0] === 'graphql' && method === 'POST') {\n const body = await request.json() as { query: string; variables?: any };\n const result = await dispatcher.handleGraphQL(body, { request });\n return new Response(JSON.stringify(result), {\n status: 200,\n headers: { 'Content-Type': 'application/json' },\n });\n }\n\n // --- Metadata ---\n if (segments[0] === 'meta') {\n const subPath = segments.slice(1).join('/');\n let body: any = undefined;\n if (method === 'PUT' || method === 'POST') {\n body = await request.json().catch(() => ({}));\n }\n const result = await dispatcher.handleMetadata(\n subPath ? `/${subPath}` : '',\n { request },\n method,\n body,\n );\n return toResponse(result);\n }\n\n // --- Data ---\n if (segments[0] === 'data') {\n const subPath = segments.slice(1).join('/');\n let body: any = {};\n if (method === 'POST' || method === 'PATCH') {\n body = await request.json().catch(() => ({}));\n }\n const queryParams: Record<string, any> = {};\n url.searchParams.forEach((val, key) => { queryParams[key] = val; });\n\n const result = await dispatcher.handleData(\n subPath ? `/${subPath}` : '',\n method,\n body,\n queryParams,\n { request },\n );\n return toResponse(result);\n }\n\n // --- Storage ---\n if (segments[0] === 'storage') {\n const subPath = segments.slice(1).join('/');\n let file: any = undefined;\n if (method === 'POST' && subPath === 'upload') {\n const formData = await request.formData();\n file = formData.get('file');\n }\n const result = await dispatcher.handleStorage(\n subPath ? `/${subPath}` : '',\n method,\n file,\n { request },\n );\n return toResponse(result);\n }\n\n return errorJson('Not Found', 404);\n } catch (err: any) {\n return errorJson(err.message || 'Internal Server Error', err.statusCode || 500);\n }\n };\n}\n\n/**\n * Creates a SvelteKit handle hook that attaches the kernel to event.locals.\n *\n * @example\n * ```ts\n * // src/hooks.server.ts\n * import { createHandle } from '@objectstack/sveltekit';\n * export const handle = createHandle({ kernel });\n * ```\n */\nexport function createHandle(options: SvelteKitAdapterOptions) {\n return async function handle({ event, resolve }: { event: any; resolve: (event: any) => Promise<Response> }) {\n event.locals.objectStack = options.kernel;\n return resolve(event);\n };\n}\n"],"mappings":"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","names":[]}
|
|
1
|
+
{"version":3,"sources":["../src/index.ts"],"sourcesContent":["// Copyright (c) 2025 ObjectStack. Licensed under the Apache-2.0 license.\n\nimport { type ObjectKernel, HttpDispatcher, HttpDispatcherResult } from '@objectstack/runtime';\n\nexport interface SvelteKitAdapterOptions {\n kernel: ObjectKernel;\n prefix?: string;\n}\n\n/**\n * Auth service interface with handleRequest method\n */\ninterface AuthService {\n handleRequest(request: Request): Promise<Response>;\n}\n\n/**\n * SvelteKit request event type (minimal interface to avoid hard dependency on @sveltejs/kit types at runtime)\n */\ninterface RequestEvent {\n request: Request;\n url: URL;\n params: Record<string, string>;\n}\n\n/**\n * Creates a SvelteKit request handler for ObjectStack API routes.\n * Use in a catch-all `+server.ts` route like `src/routes/api/[...path]/+server.ts`.\n *\n * @example\n * ```ts\n * // src/routes/api/[...path]/+server.ts\n * import { createRequestHandler } from '@objectstack/sveltekit';\n * import { kernel } from '$lib/kernel';\n *\n * const handler = createRequestHandler({ kernel });\n *\n * export const GET = handler;\n * export const POST = handler;\n * export const PUT = handler;\n * export const PATCH = handler;\n * export const DELETE = handler;\n * ```\n */\nexport function createRequestHandler(options: SvelteKitAdapterOptions) {\n const dispatcher = new HttpDispatcher(options.kernel);\n const prefix = options.prefix || '/api';\n\n const errorJson = (message: string, code: number = 500) => {\n return new Response(JSON.stringify({ success: false, error: { message, code } }), {\n status: code,\n headers: { 'Content-Type': 'application/json' },\n });\n };\n\n const toResponse = (result: HttpDispatcherResult): Response => {\n if (result.handled) {\n if (result.response) {\n const headers = new Headers({ 'Content-Type': 'application/json' });\n if (result.response.headers) {\n Object.entries(result.response.headers).forEach(([k, v]) => headers.set(k, v as string));\n }\n return new Response(JSON.stringify(result.response.body), {\n status: result.response.status,\n headers,\n });\n }\n if (result.result) {\n const res = result.result;\n if (res.type === 'redirect' && res.url) {\n return new Response(null, {\n status: 302,\n headers: { Location: res.url },\n });\n }\n if (res.type === 'stream' && res.stream) {\n const headers = new Headers();\n if (res.headers) {\n Object.entries(res.headers).forEach(([k, v]) => headers.set(k, v as string));\n }\n return new Response(res.stream, { status: 200, headers });\n }\n return new Response(JSON.stringify(res), {\n status: 200,\n headers: { 'Content-Type': 'application/json' },\n });\n }\n }\n return errorJson('Not Found', 404);\n };\n\n return async function handler(event: RequestEvent): Promise<Response> {\n const { request, url } = event;\n const method = request.method;\n const path = url.pathname.substring(prefix.length);\n const segments = path.split('/').filter(Boolean);\n\n // --- Discovery ---\n if (segments.length === 0 && method === 'GET') {\n return new Response(JSON.stringify({ data: dispatcher.getDiscoveryInfo(prefix) }), {\n status: 200,\n headers: { 'Content-Type': 'application/json' },\n });\n }\n\n try {\n // --- Auth ---\n if (segments[0] === 'auth') {\n const subPath = segments.slice(1).join('/');\n\n // Try AuthPlugin service first (prefer async to support factory-based services)\n let authService: AuthService | null = null;\n try {\n if (typeof options.kernel.getServiceAsync === 'function') {\n authService = await options.kernel.getServiceAsync<AuthService>('auth');\n } else if (typeof options.kernel.getService === 'function') {\n authService = options.kernel.getService<AuthService>('auth');\n }\n } catch {\n // Service not registered — fall through to dispatcher\n authService = null;\n }\n\n if (authService && typeof authService.handleRequest === 'function') {\n return await authService.handleRequest(request);\n }\n\n // Fallback to dispatcher\n const body = method === 'GET' || method === 'HEAD'\n ? {}\n : await request.json().catch(() => ({}));\n const result = await dispatcher.handleAuth(subPath, method, body, { request });\n return toResponse(result);\n }\n\n // --- GraphQL ---\n if (segments[0] === 'graphql' && method === 'POST') {\n const body = await request.json() as { query: string; variables?: any };\n const result = await dispatcher.handleGraphQL(body, { request });\n return new Response(JSON.stringify(result), {\n status: 200,\n headers: { 'Content-Type': 'application/json' },\n });\n }\n\n // --- Metadata ---\n if (segments[0] === 'meta') {\n const subPath = segments.slice(1).join('/');\n let body: any = undefined;\n if (method === 'PUT' || method === 'POST') {\n body = await request.json().catch(() => ({}));\n }\n const result = await dispatcher.handleMetadata(\n subPath ? `/${subPath}` : '',\n { request },\n method,\n body,\n );\n return toResponse(result);\n }\n\n // --- Data ---\n if (segments[0] === 'data') {\n const subPath = segments.slice(1).join('/');\n let body: any = {};\n if (method === 'POST' || method === 'PATCH') {\n body = await request.json().catch(() => ({}));\n }\n const queryParams: Record<string, any> = {};\n url.searchParams.forEach((val, key) => { queryParams[key] = val; });\n\n const result = await dispatcher.handleData(\n subPath ? `/${subPath}` : '',\n method,\n body,\n queryParams,\n { request },\n );\n return toResponse(result);\n }\n\n // --- Storage ---\n if (segments[0] === 'storage') {\n const subPath = segments.slice(1).join('/');\n let file: any = undefined;\n if (method === 'POST' && subPath === 'upload') {\n const formData = await request.formData();\n file = formData.get('file');\n }\n const result = await dispatcher.handleStorage(\n subPath ? `/${subPath}` : '',\n method,\n file,\n { request },\n );\n return toResponse(result);\n }\n\n return errorJson('Not Found', 404);\n } catch (err: any) {\n return errorJson(err.message || 'Internal Server Error', err.statusCode || 500);\n }\n };\n}\n\n/**\n * Creates a SvelteKit handle hook that attaches the kernel to event.locals.\n *\n * @example\n * ```ts\n * // src/hooks.server.ts\n * import { createHandle } from '@objectstack/sveltekit';\n * export const handle = createHandle({ kernel });\n * ```\n */\nexport function createHandle(options: SvelteKitAdapterOptions) {\n return async function handle({ event, resolve }: { event: any; resolve: (event: any) => Promise<Response> }) {\n event.locals.objectStack = options.kernel;\n return resolve(event);\n };\n}\n"],"mappings":"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","names":[]}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@objectstack/sveltekit",
|
|
3
|
-
"version": "3.2.
|
|
3
|
+
"version": "3.2.2",
|
|
4
4
|
"license": "Apache-2.0",
|
|
5
5
|
"main": "dist/index.js",
|
|
6
6
|
"types": "dist/index.d.ts",
|
|
@@ -13,13 +13,13 @@
|
|
|
13
13
|
},
|
|
14
14
|
"peerDependencies": {
|
|
15
15
|
"@sveltejs/kit": "^2.53.4",
|
|
16
|
-
"@objectstack/runtime": "^3.2.
|
|
16
|
+
"@objectstack/runtime": "^3.2.2"
|
|
17
17
|
},
|
|
18
18
|
"devDependencies": {
|
|
19
19
|
"@sveltejs/kit": "^2.53.4",
|
|
20
20
|
"typescript": "^5.0.0",
|
|
21
21
|
"vitest": "^4.0.18",
|
|
22
|
-
"@objectstack/runtime": "3.2.
|
|
22
|
+
"@objectstack/runtime": "3.2.2"
|
|
23
23
|
},
|
|
24
24
|
"scripts": {
|
|
25
25
|
"build": "tsup --config ../../../tsup.config.ts",
|
package/src/index.ts
CHANGED
|
@@ -108,10 +108,18 @@ export function createRequestHandler(options: SvelteKitAdapterOptions) {
|
|
|
108
108
|
if (segments[0] === 'auth') {
|
|
109
109
|
const subPath = segments.slice(1).join('/');
|
|
110
110
|
|
|
111
|
-
// Try AuthPlugin service first
|
|
112
|
-
|
|
113
|
-
|
|
114
|
-
|
|
111
|
+
// Try AuthPlugin service first (prefer async to support factory-based services)
|
|
112
|
+
let authService: AuthService | null = null;
|
|
113
|
+
try {
|
|
114
|
+
if (typeof options.kernel.getServiceAsync === 'function') {
|
|
115
|
+
authService = await options.kernel.getServiceAsync<AuthService>('auth');
|
|
116
|
+
} else if (typeof options.kernel.getService === 'function') {
|
|
117
|
+
authService = options.kernel.getService<AuthService>('auth');
|
|
118
|
+
}
|
|
119
|
+
} catch {
|
|
120
|
+
// Service not registered — fall through to dispatcher
|
|
121
|
+
authService = null;
|
|
122
|
+
}
|
|
115
123
|
|
|
116
124
|
if (authService && typeof authService.handleRequest === 'function') {
|
|
117
125
|
return await authService.handleRequest(request);
|
package/src/sveltekit.test.ts
CHANGED
|
@@ -119,6 +119,37 @@ describe('createRequestHandler', () => {
|
|
|
119
119
|
expect(kernelWithAuth.getService).toHaveBeenCalledWith('auth');
|
|
120
120
|
expect(mockHandleRequest).toHaveBeenCalled();
|
|
121
121
|
});
|
|
122
|
+
|
|
123
|
+
it('uses kernel.getServiceAsync("auth") when available (async factory)', async () => {
|
|
124
|
+
const mockHandleRequest = vi.fn().mockResolvedValue(
|
|
125
|
+
new Response(JSON.stringify({ user: { id: '2' } }), {
|
|
126
|
+
status: 200,
|
|
127
|
+
headers: { 'Content-Type': 'application/json' },
|
|
128
|
+
}),
|
|
129
|
+
);
|
|
130
|
+
const kernelWithAsyncAuth = {
|
|
131
|
+
...mockKernel,
|
|
132
|
+
getServiceAsync: vi.fn().mockResolvedValue({ handleRequest: mockHandleRequest }),
|
|
133
|
+
};
|
|
134
|
+
const authHandler = createRequestHandler({ kernel: kernelWithAsyncAuth });
|
|
135
|
+
const event = makeEvent('http://localhost/api/auth/sign-in/email', 'POST', { email: 'a@b.com' });
|
|
136
|
+
const res = await authHandler(event);
|
|
137
|
+
expect(res.status).toBe(200);
|
|
138
|
+
expect(kernelWithAsyncAuth.getServiceAsync).toHaveBeenCalledWith('auth');
|
|
139
|
+
expect(mockHandleRequest).toHaveBeenCalled();
|
|
140
|
+
});
|
|
141
|
+
|
|
142
|
+
it('falls back to dispatcher when getServiceAsync throws', async () => {
|
|
143
|
+
const kernelWithFailingAsync = {
|
|
144
|
+
...mockKernel,
|
|
145
|
+
getServiceAsync: vi.fn().mockRejectedValue(new Error("Service 'auth' not found")),
|
|
146
|
+
};
|
|
147
|
+
const authHandler = createRequestHandler({ kernel: kernelWithFailingAsync });
|
|
148
|
+
const event = makeEvent('http://localhost/api/auth/login', 'POST', { email: 'a@b.com' });
|
|
149
|
+
const res = await authHandler(event);
|
|
150
|
+
expect(res.status).toBe(200);
|
|
151
|
+
expect(mockDispatcher.handleAuth).toHaveBeenCalled();
|
|
152
|
+
});
|
|
122
153
|
});
|
|
123
154
|
|
|
124
155
|
describe('GraphQL', () => {
|