@objectstack/nestjs 2.0.2 → 2.0.4

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1,5 +1,5 @@
1
1
 
2
- > @objectstack/nestjs@2.0.2 build /home/runner/work/spec/spec/packages/adapters/nestjs
2
+ > @objectstack/nestjs@2.0.4 build /home/runner/work/spec/spec/packages/adapters/nestjs
3
3
  > tsup --config ../../../tsup.config.ts
4
4
 
5
5
  CLI Building entry: src/index.ts
@@ -12,13 +12,13 @@
12
12
  CJS Build start
13
13
  ESM You have emitDecoratorMetadata enabled but @swc/core was not installed, skipping swc plugin
14
14
  CJS You have emitDecoratorMetadata enabled but @swc/core was not installed, skipping swc plugin
15
- ESM dist/index.mjs 6.87 KB
16
- ESM dist/index.mjs.map 11.21 KB
17
- ESM ⚡️ Build success in 50ms
18
- CJS dist/index.js 8.49 KB
19
- CJS dist/index.js.map 11.24 KB
20
- CJS ⚡️ Build success in 50ms
15
+ ESM dist/index.mjs 8.15 KB
16
+ ESM dist/index.mjs.map 13.94 KB
17
+ ESM ⚡️ Build success in 38ms
18
+ CJS dist/index.js 9.76 KB
19
+ CJS dist/index.js.map 13.97 KB
20
+ CJS ⚡️ Build success in 45ms
21
21
  DTS Build start
22
- DTS ⚡️ Build success in 6716ms
22
+ DTS ⚡️ Build success in 6492ms
23
23
  DTS dist/index.d.mts 8.75 KB
24
24
  DTS dist/index.d.ts 8.75 KB
package/CHANGELOG.md CHANGED
@@ -1,5 +1,21 @@
1
1
  # @objectstack/nestjs
2
2
 
3
+ ## 2.0.4
4
+
5
+ ### Patch Changes
6
+
7
+ - Patch release for maintenance and stability improvements
8
+ - Updated dependencies
9
+ - @objectstack/runtime@2.0.4
10
+
11
+ ## 2.0.3
12
+
13
+ ### Patch Changes
14
+
15
+ - Patch release for maintenance and stability improvements
16
+ - Updated dependencies
17
+ - @objectstack/runtime@2.0.3
18
+
3
19
  ## 2.0.2
4
20
 
5
21
  ### Patch Changes
package/dist/index.js CHANGED
@@ -121,6 +121,33 @@ var ObjectStackController = class {
121
121
  }
122
122
  async auth(req, res, body) {
123
123
  try {
124
+ const kernel = this.service.getKernel();
125
+ const authService = typeof kernel.getService === "function" ? kernel.getService("auth") : null;
126
+ if (authService && typeof authService.handleRequest === "function") {
127
+ const protocol = req.protocol || "http";
128
+ const host = req.get?.("host") || req.headers?.host || "localhost";
129
+ const url = `${protocol}://${host}${req.originalUrl || req.url}`;
130
+ const headers = new Headers();
131
+ if (req.headers) {
132
+ Object.entries(req.headers).forEach(([k, v]) => {
133
+ if (typeof v === "string") headers.set(k, v);
134
+ else if (Array.isArray(v)) headers.set(k, v.join(", "));
135
+ });
136
+ }
137
+ const init = { method: req.method, headers };
138
+ if (req.method !== "GET" && req.method !== "HEAD" && body) {
139
+ init.body = JSON.stringify(body);
140
+ if (!headers.has("content-type")) {
141
+ headers.set("content-type", "application/json");
142
+ }
143
+ }
144
+ const webRequest = new Request(url, init);
145
+ const response = await authService.handleRequest(webRequest);
146
+ res.status(response.status);
147
+ response.headers.forEach((v, k) => res.setHeader(k, v));
148
+ const text = await response.text();
149
+ return res.send(text);
150
+ }
124
151
  const path = req.params[0] || req.url.split("/auth/")[1]?.split("?")[0] || "";
125
152
  const result = await this.service.dispatcher.handleAuth(path, req.method, body, { request: req, response: res });
126
153
  return this.normalizeResponse(result, res);
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 { DynamicModule, Module, Global, Inject, Provider, Controller, Post, Get, Body, Query, Req, Res, All, createParamDecorator, ExecutionContext } from '@nestjs/common';\nimport { Injectable } from '@nestjs/common';\nimport { ObjectKernel, HttpDispatcher, HttpDispatcherResult } from '@objectstack/runtime';\n\nexport const OBJECT_KERNEL = 'OBJECT_KERNEL';\n\nexport const ConnectReq = createParamDecorator(\n (_data: unknown, ctx: ExecutionContext) => {\n return ctx.switchToHttp().getRequest();\n },\n);\n\n// --- Service ---\n\n@Injectable()\nexport class ObjectStackService {\n public dispatcher: HttpDispatcher;\n\n constructor(@Inject(OBJECT_KERNEL) private readonly kernel: ObjectKernel) {\n this.dispatcher = new HttpDispatcher(kernel);\n }\n\n getKernel() {\n return this.kernel;\n }\n}\n\n// --- Controller ---\n\n@Controller('api')\nexport class ObjectStackController {\n constructor(private readonly service: ObjectStackService) {}\n\n private async normalizeResponse(result: HttpDispatcherResult, res: any) {\n if (result.handled) {\n if (result.response) {\n res.status(result.response.status);\n if (result.response.headers) {\n Object.entries(result.response.headers).forEach(([k, v]) => res.setHeader(k, v));\n }\n return res.json(result.response.body);\n }\n if (result.result) {\n const response = result.result;\n \n // Handle redirect\n if (response.type === 'redirect' && response.url) {\n return res.redirect(response.url);\n }\n\n // Handle stream\n if (response.type === 'stream' && response.stream) {\n if (response.headers) {\n Object.entries(response.headers).forEach(([k, v]) => res.setHeader(k, v));\n }\n response.stream.pipe(res);\n return;\n }\n \n // If response is a standard Response object\n if (response && typeof response.status === 'number' && typeof response.text === 'function') {\n res.status(response.status);\n if (response.headers && typeof response.headers.forEach === 'function') {\n response.headers.forEach((v: string, k: string) => res.setHeader(k, v));\n }\n const text = await response.text();\n res.send(text);\n return;\n }\n return res.status(200).json(response);\n }\n }\n return res.status(404).json({ success: false, error: { message: 'Not Found', code: 404 } });\n }\n\n private async handleError(err: any, res: any) {\n return res.status(err.statusCode || 500).json({ \n success: false, \n error: { \n message: err.message || 'Internal Server Error', \n code: err.statusCode || 500,\n details: err.details \n } \n });\n }\n\n // --- Discovery Endpoint ---\n @Get()\n discovery() {\n return { data: this.service.dispatcher.getDiscoveryInfo('/api') };\n }\n\n @Post('graphql')\n async graphql(@Body() body: any, @Req() req: any, @Res() res: any) {\n try {\n const result = await this.service.dispatcher.handleGraphQL(body, { request: req });\n return res.json(result);\n } catch (err) {\n return this.handleError(err, res);\n }\n }\n\n // Auth (Generic Auth Handler)\n @All('auth/*')\n async auth(@Req() req: any, @Res() res: any, @Body() body: any) {\n try {\n const path = req.params[0] || req.url.split('/auth/')[1]?.split('?')[0] || '';\n const result = await this.service.dispatcher.handleAuth(path, req.method, body, { request: req, response: res });\n return this.normalizeResponse(result, res);\n } catch (err: any) {\n return this.handleError(err, res);\n }\n }\n\n // Metadata\n @All('meta*')\n async metadata(@Req() req: any, @Res() res: any, @Body() body?: any) {\n try {\n // /api/meta/objects -> objects\n let path = req.params[0] || ''; \n if (req.url.includes('/meta')) {\n path = req.url.split('/meta')[1].split('?')[0];\n }\n \n // Use injected body or fallback to req.body\n const payload = body || req.body;\n \n const result = await this.service.dispatcher.handleMetadata(path, { request: req }, req.method, payload);\n return this.normalizeResponse(result, res);\n } catch (err) {\n return this.handleError(err, res);\n }\n }\n\n // Data\n @All('data*')\n async data(@Req() req: any, @Res() res: any, @Body() body: any, @Query() query: any) {\n try {\n let path = req.params[0] || '';\n if (req.url.includes('/data')) {\n path = req.url.substring(req.url.indexOf('/data') + 5).split('?')[0];\n }\n \n const result = await this.service.dispatcher.handleData(path, req.method, body, query, { request: req });\n return this.normalizeResponse(result, res);\n } catch (err) {\n return this.handleError(err, res);\n }\n }\n\n // Storage\n @All('storage*')\n async storage(@Req() req: any, @Res() res: any) {\n try {\n let path = req.params[0] || '';\n if (req.url.includes('/storage')) {\n path = req.url.substring(req.url.indexOf('/storage') + 8).split('?')[0];\n }\n\n // Handle File for NestJS (Express/Fastify)\n const file = req.file || req.files?.file;\n \n const result = await this.service.dispatcher.handleStorage(path, req.method, file, { request: req });\n return this.normalizeResponse(result, res);\n } catch (err) {\n return this.handleError(err, res);\n }\n }\n}\n\n// --- Discovery Controller ---\n\n@Controller('.well-known')\nexport class DiscoveryController {\n @Get('objectstack')\n discover(@Res() res: any) {\n return res.redirect('/api');\n }\n}\n\n// --- Module ---\n\n@Global()\n@Module({})\nexport class ObjectStackModule {\n static forRoot(kernel: ObjectKernel): DynamicModule {\n const kernelProvider: Provider = {\n provide: OBJECT_KERNEL,\n useValue: kernel,\n };\n\n return {\n module: ObjectStackModule,\n controllers: [ObjectStackController, DiscoveryController],\n providers: [kernelProvider, ObjectStackService],\n exports: [kernelProvider, ObjectStackService],\n };\n }\n}\n"],"mappings":"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","names":["import_common"]}
1
+ {"version":3,"sources":["../src/index.ts"],"sourcesContent":["// Copyright (c) 2025 ObjectStack. Licensed under the Apache-2.0 license.\n\nimport { DynamicModule, Module, Global, Inject, Provider, Controller, Post, Get, Body, Query, Req, Res, All, createParamDecorator, ExecutionContext } from '@nestjs/common';\nimport { Injectable } from '@nestjs/common';\nimport { ObjectKernel, HttpDispatcher, HttpDispatcherResult } from '@objectstack/runtime';\n\nexport const OBJECT_KERNEL = 'OBJECT_KERNEL';\n\nexport const ConnectReq = createParamDecorator(\n (_data: unknown, ctx: ExecutionContext) => {\n return ctx.switchToHttp().getRequest();\n },\n);\n\n/**\n * Auth service interface with handleRequest method\n */\ninterface AuthService {\n handleRequest(request: Request): Promise<Response>;\n}\n\n// --- Service ---\n\n@Injectable()\nexport class ObjectStackService {\n public dispatcher: HttpDispatcher;\n\n constructor(@Inject(OBJECT_KERNEL) private readonly kernel: ObjectKernel) {\n this.dispatcher = new HttpDispatcher(kernel);\n }\n\n getKernel() {\n return this.kernel;\n }\n}\n\n// --- Controller ---\n\n@Controller('api')\nexport class ObjectStackController {\n constructor(private readonly service: ObjectStackService) {}\n\n private async normalizeResponse(result: HttpDispatcherResult, res: any) {\n if (result.handled) {\n if (result.response) {\n res.status(result.response.status);\n if (result.response.headers) {\n Object.entries(result.response.headers).forEach(([k, v]) => res.setHeader(k, v));\n }\n return res.json(result.response.body);\n }\n if (result.result) {\n const response = result.result;\n \n // Handle redirect\n if (response.type === 'redirect' && response.url) {\n return res.redirect(response.url);\n }\n\n // Handle stream\n if (response.type === 'stream' && response.stream) {\n if (response.headers) {\n Object.entries(response.headers).forEach(([k, v]) => res.setHeader(k, v));\n }\n response.stream.pipe(res);\n return;\n }\n \n // If response is a standard Response object\n if (response && typeof response.status === 'number' && typeof response.text === 'function') {\n res.status(response.status);\n if (response.headers && typeof response.headers.forEach === 'function') {\n response.headers.forEach((v: string, k: string) => res.setHeader(k, v));\n }\n const text = await response.text();\n res.send(text);\n return;\n }\n return res.status(200).json(response);\n }\n }\n return res.status(404).json({ success: false, error: { message: 'Not Found', code: 404 } });\n }\n\n private async handleError(err: any, res: any) {\n return res.status(err.statusCode || 500).json({ \n success: false, \n error: { \n message: err.message || 'Internal Server Error', \n code: err.statusCode || 500,\n details: err.details \n } \n });\n }\n\n // --- Discovery Endpoint ---\n @Get()\n discovery() {\n return { data: this.service.dispatcher.getDiscoveryInfo('/api') };\n }\n\n @Post('graphql')\n async graphql(@Body() body: any, @Req() req: any, @Res() res: any) {\n try {\n const result = await this.service.dispatcher.handleGraphQL(body, { request: req });\n return res.json(result);\n } catch (err) {\n return this.handleError(err, res);\n }\n }\n\n // Auth (Generic Auth Handler)\n @All('auth/*')\n async auth(@Req() req: any, @Res() res: any, @Body() body: any) {\n try {\n // Try AuthPlugin service first (preferred path)\n const kernel = this.service.getKernel();\n const authService = typeof kernel.getService === 'function'\n ? kernel.getService<AuthService>('auth')\n : null;\n\n if (authService && typeof authService.handleRequest === 'function') {\n // Construct a Web standard Request from the Express/Fastify request\n const protocol = req.protocol || 'http';\n const host = req.get?.('host') || req.headers?.host || 'localhost';\n const url = `${protocol}://${host}${req.originalUrl || req.url}`;\n const headers = new Headers();\n if (req.headers) {\n Object.entries(req.headers).forEach(([k, v]) => {\n if (typeof v === 'string') headers.set(k, v);\n else if (Array.isArray(v)) headers.set(k, v.join(', '));\n });\n }\n const init: RequestInit = { method: req.method, headers };\n if (req.method !== 'GET' && req.method !== 'HEAD' && body) {\n init.body = JSON.stringify(body);\n if (!headers.has('content-type')) {\n headers.set('content-type', 'application/json');\n }\n }\n const webRequest = new Request(url, init);\n const response = await authService.handleRequest(webRequest);\n\n // Convert Web Response to Express/Fastify response\n res.status(response.status);\n response.headers.forEach((v: string, k: string) => res.setHeader(k, v));\n const text = await response.text();\n return res.send(text);\n }\n\n // Fallback to legacy dispatcher\n const path = req.params[0] || req.url.split('/auth/')[1]?.split('?')[0] || '';\n const result = await this.service.dispatcher.handleAuth(path, req.method, body, { request: req, response: res });\n return this.normalizeResponse(result, res);\n } catch (err: any) {\n return this.handleError(err, res);\n }\n }\n\n // Metadata\n @All('meta*')\n async metadata(@Req() req: any, @Res() res: any, @Body() body?: any) {\n try {\n // /api/meta/objects -> objects\n let path = req.params[0] || ''; \n if (req.url.includes('/meta')) {\n path = req.url.split('/meta')[1].split('?')[0];\n }\n \n // Use injected body or fallback to req.body\n const payload = body || req.body;\n \n const result = await this.service.dispatcher.handleMetadata(path, { request: req }, req.method, payload);\n return this.normalizeResponse(result, res);\n } catch (err) {\n return this.handleError(err, res);\n }\n }\n\n // Data\n @All('data*')\n async data(@Req() req: any, @Res() res: any, @Body() body: any, @Query() query: any) {\n try {\n let path = req.params[0] || '';\n if (req.url.includes('/data')) {\n path = req.url.substring(req.url.indexOf('/data') + 5).split('?')[0];\n }\n \n const result = await this.service.dispatcher.handleData(path, req.method, body, query, { request: req });\n return this.normalizeResponse(result, res);\n } catch (err) {\n return this.handleError(err, res);\n }\n }\n\n // Storage\n @All('storage*')\n async storage(@Req() req: any, @Res() res: any) {\n try {\n let path = req.params[0] || '';\n if (req.url.includes('/storage')) {\n path = req.url.substring(req.url.indexOf('/storage') + 8).split('?')[0];\n }\n\n // Handle File for NestJS (Express/Fastify)\n const file = req.file || req.files?.file;\n \n const result = await this.service.dispatcher.handleStorage(path, req.method, file, { request: req });\n return this.normalizeResponse(result, res);\n } catch (err) {\n return this.handleError(err, res);\n }\n }\n}\n\n// --- Discovery Controller ---\n\n@Controller('.well-known')\nexport class DiscoveryController {\n @Get('objectstack')\n discover(@Res() res: any) {\n return res.redirect('/api');\n }\n}\n\n// --- Module ---\n\n@Global()\n@Module({})\nexport class ObjectStackModule {\n static forRoot(kernel: ObjectKernel): DynamicModule {\n const kernelProvider: Provider = {\n provide: OBJECT_KERNEL,\n useValue: kernel,\n };\n\n return {\n module: ObjectStackModule,\n controllers: [ObjectStackController, DiscoveryController],\n providers: [kernelProvider, ObjectStackService],\n exports: [kernelProvider, ObjectStackService],\n };\n }\n}\n"],"mappings":"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","names":["import_common"]}
package/dist/index.mjs CHANGED
@@ -95,6 +95,33 @@ var ObjectStackController = class {
95
95
  }
96
96
  async auth(req, res, body) {
97
97
  try {
98
+ const kernel = this.service.getKernel();
99
+ const authService = typeof kernel.getService === "function" ? kernel.getService("auth") : null;
100
+ if (authService && typeof authService.handleRequest === "function") {
101
+ const protocol = req.protocol || "http";
102
+ const host = req.get?.("host") || req.headers?.host || "localhost";
103
+ const url = `${protocol}://${host}${req.originalUrl || req.url}`;
104
+ const headers = new Headers();
105
+ if (req.headers) {
106
+ Object.entries(req.headers).forEach(([k, v]) => {
107
+ if (typeof v === "string") headers.set(k, v);
108
+ else if (Array.isArray(v)) headers.set(k, v.join(", "));
109
+ });
110
+ }
111
+ const init = { method: req.method, headers };
112
+ if (req.method !== "GET" && req.method !== "HEAD" && body) {
113
+ init.body = JSON.stringify(body);
114
+ if (!headers.has("content-type")) {
115
+ headers.set("content-type", "application/json");
116
+ }
117
+ }
118
+ const webRequest = new Request(url, init);
119
+ const response = await authService.handleRequest(webRequest);
120
+ res.status(response.status);
121
+ response.headers.forEach((v, k) => res.setHeader(k, v));
122
+ const text = await response.text();
123
+ return res.send(text);
124
+ }
98
125
  const path = req.params[0] || req.url.split("/auth/")[1]?.split("?")[0] || "";
99
126
  const result = await this.service.dispatcher.handleAuth(path, req.method, body, { request: req, response: res });
100
127
  return this.normalizeResponse(result, res);
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/index.ts"],"sourcesContent":["// Copyright (c) 2025 ObjectStack. Licensed under the Apache-2.0 license.\n\nimport { DynamicModule, Module, Global, Inject, Provider, Controller, Post, Get, Body, Query, Req, Res, All, createParamDecorator, ExecutionContext } from '@nestjs/common';\nimport { Injectable } from '@nestjs/common';\nimport { ObjectKernel, HttpDispatcher, HttpDispatcherResult } from '@objectstack/runtime';\n\nexport const OBJECT_KERNEL = 'OBJECT_KERNEL';\n\nexport const ConnectReq = createParamDecorator(\n (_data: unknown, ctx: ExecutionContext) => {\n return ctx.switchToHttp().getRequest();\n },\n);\n\n// --- Service ---\n\n@Injectable()\nexport class ObjectStackService {\n public dispatcher: HttpDispatcher;\n\n constructor(@Inject(OBJECT_KERNEL) private readonly kernel: ObjectKernel) {\n this.dispatcher = new HttpDispatcher(kernel);\n }\n\n getKernel() {\n return this.kernel;\n }\n}\n\n// --- Controller ---\n\n@Controller('api')\nexport class ObjectStackController {\n constructor(private readonly service: ObjectStackService) {}\n\n private async normalizeResponse(result: HttpDispatcherResult, res: any) {\n if (result.handled) {\n if (result.response) {\n res.status(result.response.status);\n if (result.response.headers) {\n Object.entries(result.response.headers).forEach(([k, v]) => res.setHeader(k, v));\n }\n return res.json(result.response.body);\n }\n if (result.result) {\n const response = result.result;\n \n // Handle redirect\n if (response.type === 'redirect' && response.url) {\n return res.redirect(response.url);\n }\n\n // Handle stream\n if (response.type === 'stream' && response.stream) {\n if (response.headers) {\n Object.entries(response.headers).forEach(([k, v]) => res.setHeader(k, v));\n }\n response.stream.pipe(res);\n return;\n }\n \n // If response is a standard Response object\n if (response && typeof response.status === 'number' && typeof response.text === 'function') {\n res.status(response.status);\n if (response.headers && typeof response.headers.forEach === 'function') {\n response.headers.forEach((v: string, k: string) => res.setHeader(k, v));\n }\n const text = await response.text();\n res.send(text);\n return;\n }\n return res.status(200).json(response);\n }\n }\n return res.status(404).json({ success: false, error: { message: 'Not Found', code: 404 } });\n }\n\n private async handleError(err: any, res: any) {\n return res.status(err.statusCode || 500).json({ \n success: false, \n error: { \n message: err.message || 'Internal Server Error', \n code: err.statusCode || 500,\n details: err.details \n } \n });\n }\n\n // --- Discovery Endpoint ---\n @Get()\n discovery() {\n return { data: this.service.dispatcher.getDiscoveryInfo('/api') };\n }\n\n @Post('graphql')\n async graphql(@Body() body: any, @Req() req: any, @Res() res: any) {\n try {\n const result = await this.service.dispatcher.handleGraphQL(body, { request: req });\n return res.json(result);\n } catch (err) {\n return this.handleError(err, res);\n }\n }\n\n // Auth (Generic Auth Handler)\n @All('auth/*')\n async auth(@Req() req: any, @Res() res: any, @Body() body: any) {\n try {\n const path = req.params[0] || req.url.split('/auth/')[1]?.split('?')[0] || '';\n const result = await this.service.dispatcher.handleAuth(path, req.method, body, { request: req, response: res });\n return this.normalizeResponse(result, res);\n } catch (err: any) {\n return this.handleError(err, res);\n }\n }\n\n // Metadata\n @All('meta*')\n async metadata(@Req() req: any, @Res() res: any, @Body() body?: any) {\n try {\n // /api/meta/objects -> objects\n let path = req.params[0] || ''; \n if (req.url.includes('/meta')) {\n path = req.url.split('/meta')[1].split('?')[0];\n }\n \n // Use injected body or fallback to req.body\n const payload = body || req.body;\n \n const result = await this.service.dispatcher.handleMetadata(path, { request: req }, req.method, payload);\n return this.normalizeResponse(result, res);\n } catch (err) {\n return this.handleError(err, res);\n }\n }\n\n // Data\n @All('data*')\n async data(@Req() req: any, @Res() res: any, @Body() body: any, @Query() query: any) {\n try {\n let path = req.params[0] || '';\n if (req.url.includes('/data')) {\n path = req.url.substring(req.url.indexOf('/data') + 5).split('?')[0];\n }\n \n const result = await this.service.dispatcher.handleData(path, req.method, body, query, { request: req });\n return this.normalizeResponse(result, res);\n } catch (err) {\n return this.handleError(err, res);\n }\n }\n\n // Storage\n @All('storage*')\n async storage(@Req() req: any, @Res() res: any) {\n try {\n let path = req.params[0] || '';\n if (req.url.includes('/storage')) {\n path = req.url.substring(req.url.indexOf('/storage') + 8).split('?')[0];\n }\n\n // Handle File for NestJS (Express/Fastify)\n const file = req.file || req.files?.file;\n \n const result = await this.service.dispatcher.handleStorage(path, req.method, file, { request: req });\n return this.normalizeResponse(result, res);\n } catch (err) {\n return this.handleError(err, res);\n }\n }\n}\n\n// --- Discovery Controller ---\n\n@Controller('.well-known')\nexport class DiscoveryController {\n @Get('objectstack')\n discover(@Res() res: any) {\n return res.redirect('/api');\n }\n}\n\n// --- Module ---\n\n@Global()\n@Module({})\nexport class ObjectStackModule {\n static forRoot(kernel: ObjectKernel): DynamicModule {\n const kernelProvider: Provider = {\n provide: OBJECT_KERNEL,\n useValue: kernel,\n };\n\n return {\n module: ObjectStackModule,\n controllers: [ObjectStackController, DiscoveryController],\n providers: [kernelProvider, ObjectStackService],\n exports: [kernelProvider, ObjectStackService],\n };\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 { DynamicModule, Module, Global, Inject, Provider, Controller, Post, Get, Body, Query, Req, Res, All, createParamDecorator, ExecutionContext } from '@nestjs/common';\nimport { Injectable } from '@nestjs/common';\nimport { ObjectKernel, HttpDispatcher, HttpDispatcherResult } from '@objectstack/runtime';\n\nexport const OBJECT_KERNEL = 'OBJECT_KERNEL';\n\nexport const ConnectReq = createParamDecorator(\n (_data: unknown, ctx: ExecutionContext) => {\n return ctx.switchToHttp().getRequest();\n },\n);\n\n/**\n * Auth service interface with handleRequest method\n */\ninterface AuthService {\n handleRequest(request: Request): Promise<Response>;\n}\n\n// --- Service ---\n\n@Injectable()\nexport class ObjectStackService {\n public dispatcher: HttpDispatcher;\n\n constructor(@Inject(OBJECT_KERNEL) private readonly kernel: ObjectKernel) {\n this.dispatcher = new HttpDispatcher(kernel);\n }\n\n getKernel() {\n return this.kernel;\n }\n}\n\n// --- Controller ---\n\n@Controller('api')\nexport class ObjectStackController {\n constructor(private readonly service: ObjectStackService) {}\n\n private async normalizeResponse(result: HttpDispatcherResult, res: any) {\n if (result.handled) {\n if (result.response) {\n res.status(result.response.status);\n if (result.response.headers) {\n Object.entries(result.response.headers).forEach(([k, v]) => res.setHeader(k, v));\n }\n return res.json(result.response.body);\n }\n if (result.result) {\n const response = result.result;\n \n // Handle redirect\n if (response.type === 'redirect' && response.url) {\n return res.redirect(response.url);\n }\n\n // Handle stream\n if (response.type === 'stream' && response.stream) {\n if (response.headers) {\n Object.entries(response.headers).forEach(([k, v]) => res.setHeader(k, v));\n }\n response.stream.pipe(res);\n return;\n }\n \n // If response is a standard Response object\n if (response && typeof response.status === 'number' && typeof response.text === 'function') {\n res.status(response.status);\n if (response.headers && typeof response.headers.forEach === 'function') {\n response.headers.forEach((v: string, k: string) => res.setHeader(k, v));\n }\n const text = await response.text();\n res.send(text);\n return;\n }\n return res.status(200).json(response);\n }\n }\n return res.status(404).json({ success: false, error: { message: 'Not Found', code: 404 } });\n }\n\n private async handleError(err: any, res: any) {\n return res.status(err.statusCode || 500).json({ \n success: false, \n error: { \n message: err.message || 'Internal Server Error', \n code: err.statusCode || 500,\n details: err.details \n } \n });\n }\n\n // --- Discovery Endpoint ---\n @Get()\n discovery() {\n return { data: this.service.dispatcher.getDiscoveryInfo('/api') };\n }\n\n @Post('graphql')\n async graphql(@Body() body: any, @Req() req: any, @Res() res: any) {\n try {\n const result = await this.service.dispatcher.handleGraphQL(body, { request: req });\n return res.json(result);\n } catch (err) {\n return this.handleError(err, res);\n }\n }\n\n // Auth (Generic Auth Handler)\n @All('auth/*')\n async auth(@Req() req: any, @Res() res: any, @Body() body: any) {\n try {\n // Try AuthPlugin service first (preferred path)\n const kernel = this.service.getKernel();\n const authService = typeof kernel.getService === 'function'\n ? kernel.getService<AuthService>('auth')\n : null;\n\n if (authService && typeof authService.handleRequest === 'function') {\n // Construct a Web standard Request from the Express/Fastify request\n const protocol = req.protocol || 'http';\n const host = req.get?.('host') || req.headers?.host || 'localhost';\n const url = `${protocol}://${host}${req.originalUrl || req.url}`;\n const headers = new Headers();\n if (req.headers) {\n Object.entries(req.headers).forEach(([k, v]) => {\n if (typeof v === 'string') headers.set(k, v);\n else if (Array.isArray(v)) headers.set(k, v.join(', '));\n });\n }\n const init: RequestInit = { method: req.method, headers };\n if (req.method !== 'GET' && req.method !== 'HEAD' && body) {\n init.body = JSON.stringify(body);\n if (!headers.has('content-type')) {\n headers.set('content-type', 'application/json');\n }\n }\n const webRequest = new Request(url, init);\n const response = await authService.handleRequest(webRequest);\n\n // Convert Web Response to Express/Fastify response\n res.status(response.status);\n response.headers.forEach((v: string, k: string) => res.setHeader(k, v));\n const text = await response.text();\n return res.send(text);\n }\n\n // Fallback to legacy dispatcher\n const path = req.params[0] || req.url.split('/auth/')[1]?.split('?')[0] || '';\n const result = await this.service.dispatcher.handleAuth(path, req.method, body, { request: req, response: res });\n return this.normalizeResponse(result, res);\n } catch (err: any) {\n return this.handleError(err, res);\n }\n }\n\n // Metadata\n @All('meta*')\n async metadata(@Req() req: any, @Res() res: any, @Body() body?: any) {\n try {\n // /api/meta/objects -> objects\n let path = req.params[0] || ''; \n if (req.url.includes('/meta')) {\n path = req.url.split('/meta')[1].split('?')[0];\n }\n \n // Use injected body or fallback to req.body\n const payload = body || req.body;\n \n const result = await this.service.dispatcher.handleMetadata(path, { request: req }, req.method, payload);\n return this.normalizeResponse(result, res);\n } catch (err) {\n return this.handleError(err, res);\n }\n }\n\n // Data\n @All('data*')\n async data(@Req() req: any, @Res() res: any, @Body() body: any, @Query() query: any) {\n try {\n let path = req.params[0] || '';\n if (req.url.includes('/data')) {\n path = req.url.substring(req.url.indexOf('/data') + 5).split('?')[0];\n }\n \n const result = await this.service.dispatcher.handleData(path, req.method, body, query, { request: req });\n return this.normalizeResponse(result, res);\n } catch (err) {\n return this.handleError(err, res);\n }\n }\n\n // Storage\n @All('storage*')\n async storage(@Req() req: any, @Res() res: any) {\n try {\n let path = req.params[0] || '';\n if (req.url.includes('/storage')) {\n path = req.url.substring(req.url.indexOf('/storage') + 8).split('?')[0];\n }\n\n // Handle File for NestJS (Express/Fastify)\n const file = req.file || req.files?.file;\n \n const result = await this.service.dispatcher.handleStorage(path, req.method, file, { request: req });\n return this.normalizeResponse(result, res);\n } catch (err) {\n return this.handleError(err, res);\n }\n }\n}\n\n// --- Discovery Controller ---\n\n@Controller('.well-known')\nexport class DiscoveryController {\n @Get('objectstack')\n discover(@Res() res: any) {\n return res.redirect('/api');\n }\n}\n\n// --- Module ---\n\n@Global()\n@Module({})\nexport class ObjectStackModule {\n static forRoot(kernel: ObjectKernel): DynamicModule {\n const kernelProvider: Provider = {\n provide: OBJECT_KERNEL,\n useValue: kernel,\n };\n\n return {\n module: ObjectStackModule,\n controllers: [ObjectStackController, DiscoveryController],\n providers: [kernelProvider, ObjectStackService],\n exports: [kernelProvider, ObjectStackService],\n };\n }\n}\n"],"mappings":"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","names":[]}
package/package.json CHANGED
@@ -1,20 +1,20 @@
1
1
  {
2
2
  "name": "@objectstack/nestjs",
3
- "version": "2.0.2",
3
+ "version": "2.0.4",
4
4
  "license": "Apache-2.0",
5
5
  "main": "dist/index.js",
6
6
  "types": "dist/index.d.ts",
7
7
  "peerDependencies": {
8
8
  "@nestjs/common": "^11.1.13",
9
9
  "@nestjs/core": "^11.1.13",
10
- "@objectstack/runtime": "2.0.2"
10
+ "@objectstack/runtime": "2.0.4"
11
11
  },
12
12
  "devDependencies": {
13
13
  "@nestjs/common": "^11.1.13",
14
14
  "@nestjs/core": "^11.1.13",
15
15
  "typescript": "^5.0.0",
16
16
  "vitest": "^4.0.18",
17
- "@objectstack/runtime": "2.0.2"
17
+ "@objectstack/runtime": "2.0.4"
18
18
  },
19
19
  "scripts": {
20
20
  "build": "tsup --config ../../../tsup.config.ts",
package/src/index.ts CHANGED
@@ -12,6 +12,13 @@ export const ConnectReq = createParamDecorator(
12
12
  },
13
13
  );
14
14
 
15
+ /**
16
+ * Auth service interface with handleRequest method
17
+ */
18
+ interface AuthService {
19
+ handleRequest(request: Request): Promise<Response>;
20
+ }
21
+
15
22
  // --- Service ---
16
23
 
17
24
  @Injectable()
@@ -106,6 +113,42 @@ export class ObjectStackController {
106
113
  @All('auth/*')
107
114
  async auth(@Req() req: any, @Res() res: any, @Body() body: any) {
108
115
  try {
116
+ // Try AuthPlugin service first (preferred path)
117
+ const kernel = this.service.getKernel();
118
+ const authService = typeof kernel.getService === 'function'
119
+ ? kernel.getService<AuthService>('auth')
120
+ : null;
121
+
122
+ if (authService && typeof authService.handleRequest === 'function') {
123
+ // Construct a Web standard Request from the Express/Fastify request
124
+ const protocol = req.protocol || 'http';
125
+ const host = req.get?.('host') || req.headers?.host || 'localhost';
126
+ const url = `${protocol}://${host}${req.originalUrl || req.url}`;
127
+ const headers = new Headers();
128
+ if (req.headers) {
129
+ Object.entries(req.headers).forEach(([k, v]) => {
130
+ if (typeof v === 'string') headers.set(k, v);
131
+ else if (Array.isArray(v)) headers.set(k, v.join(', '));
132
+ });
133
+ }
134
+ const init: RequestInit = { method: req.method, headers };
135
+ if (req.method !== 'GET' && req.method !== 'HEAD' && body) {
136
+ init.body = JSON.stringify(body);
137
+ if (!headers.has('content-type')) {
138
+ headers.set('content-type', 'application/json');
139
+ }
140
+ }
141
+ const webRequest = new Request(url, init);
142
+ const response = await authService.handleRequest(webRequest);
143
+
144
+ // Convert Web Response to Express/Fastify response
145
+ res.status(response.status);
146
+ response.headers.forEach((v: string, k: string) => res.setHeader(k, v));
147
+ const text = await response.text();
148
+ return res.send(text);
149
+ }
150
+
151
+ // Fallback to legacy dispatcher
109
152
  const path = req.params[0] || req.url.split('/auth/')[1]?.split('?')[0] || '';
110
153
  const result = await this.service.dispatcher.handleAuth(path, req.method, body, { request: req, response: res });
111
154
  return this.normalizeResponse(result, res);
@@ -190,6 +190,109 @@ describe('ObjectStackController', () => {
190
190
  });
191
191
  });
192
192
 
193
+ describe('auth() via AuthPlugin service', () => {
194
+ it('uses kernel.getService("auth") when available', async () => {
195
+ const mockHandleRequest = vi.fn().mockResolvedValue(
196
+ new Response(JSON.stringify({ user: { id: '1' } }), {
197
+ status: 200,
198
+ headers: new Headers({ 'Content-Type': 'application/json' }),
199
+ }),
200
+ );
201
+ const kernelWithAuth = {
202
+ ...createMockKernel(),
203
+ getService: vi.fn().mockReturnValue({ handleRequest: mockHandleRequest }),
204
+ };
205
+ const svc = new ObjectStackService(kernelWithAuth);
206
+ const ctrl = new ObjectStackController(svc);
207
+ const r = createMockRes();
208
+ const req = {
209
+ params: { 0: 'sign-in/email' },
210
+ url: '/api/auth/sign-in/email',
211
+ method: 'POST',
212
+ protocol: 'http',
213
+ get: (key: string) => key === 'host' ? 'localhost' : undefined,
214
+ headers: { 'content-type': 'application/json' },
215
+ originalUrl: '/api/auth/sign-in/email',
216
+ };
217
+
218
+ await ctrl.auth(req, r, { email: 'a@b.com', password: 'pass' });
219
+
220
+ expect(kernelWithAuth.getService).toHaveBeenCalledWith('auth');
221
+ expect(mockHandleRequest).toHaveBeenCalledWith(expect.any(Request));
222
+ expect(r._status).toBe(200);
223
+ });
224
+
225
+ it('falls back to dispatcher when auth service is not available', async () => {
226
+ const kernelWithoutAuth = {
227
+ ...createMockKernel(),
228
+ getService: vi.fn().mockReturnValue(null),
229
+ };
230
+ const svc = new ObjectStackService(kernelWithoutAuth);
231
+ const ctrl = new ObjectStackController(svc);
232
+ const r = createMockRes();
233
+ const req = { params: { 0: 'login' }, url: '/api/auth/login', method: 'POST' };
234
+
235
+ await ctrl.auth(req, r, { email: 'a@b.com' });
236
+
237
+ expect(svc.dispatcher.handleAuth).toHaveBeenCalled();
238
+ });
239
+
240
+ it('forwards GET requests to auth service', async () => {
241
+ const mockHandleRequest = vi.fn().mockResolvedValue(
242
+ new Response(JSON.stringify({ session: { token: 'abc' } }), {
243
+ status: 200,
244
+ headers: new Headers({ 'Content-Type': 'application/json' }),
245
+ }),
246
+ );
247
+ const kernelWithAuth = {
248
+ ...createMockKernel(),
249
+ getService: vi.fn().mockReturnValue({ handleRequest: mockHandleRequest }),
250
+ };
251
+ const svc = new ObjectStackService(kernelWithAuth);
252
+ const ctrl = new ObjectStackController(svc);
253
+ const r = createMockRes();
254
+ const req = {
255
+ params: { 0: 'get-session' },
256
+ url: '/api/auth/get-session',
257
+ method: 'GET',
258
+ protocol: 'http',
259
+ get: (key: string) => key === 'host' ? 'localhost' : undefined,
260
+ headers: {},
261
+ originalUrl: '/api/auth/get-session',
262
+ };
263
+
264
+ await ctrl.auth(req, r, {});
265
+
266
+ expect(mockHandleRequest).toHaveBeenCalled();
267
+ expect(r._status).toBe(200);
268
+ });
269
+
270
+ it('returns error when auth service throws', async () => {
271
+ const mockHandleRequest = vi.fn().mockRejectedValue(new Error('Auth failed'));
272
+ const kernelWithAuth = {
273
+ ...createMockKernel(),
274
+ getService: vi.fn().mockReturnValue({ handleRequest: mockHandleRequest }),
275
+ };
276
+ const svc = new ObjectStackService(kernelWithAuth);
277
+ const ctrl = new ObjectStackController(svc);
278
+ const r = createMockRes();
279
+ const req = {
280
+ params: { 0: 'sign-in/email' },
281
+ url: '/api/auth/sign-in/email',
282
+ method: 'POST',
283
+ protocol: 'http',
284
+ get: (key: string) => key === 'host' ? 'localhost' : undefined,
285
+ headers: {},
286
+ originalUrl: '/api/auth/sign-in/email',
287
+ };
288
+
289
+ await ctrl.auth(req, r, {});
290
+
291
+ expect(r._status).toBe(500);
292
+ expect(r._body.success).toBe(false);
293
+ });
294
+ });
295
+
193
296
  describe('metadata()', () => {
194
297
  it('dispatches to handleMetadata with extracted path', async () => {
195
298
  const req = { params: { 0: '' }, url: '/api/meta/objects', method: 'GET' };