@objectstack/nestjs 1.0.9 → 1.0.11

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@1.0.9 build /home/runner/work/spec/spec/packages/adapters/nestjs
2
+ > @objectstack/nestjs@1.0.11 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.89 KB
16
- ESM dist/index.mjs.map 11.16 KB
17
- ESM ⚡️ Build success in 37ms
18
- CJS dist/index.js 8.50 KB
19
- CJS dist/index.js.map 11.18 KB
20
- CJS ⚡️ Build success in 37ms
15
+ CJS dist/index.js 8.49 KB
16
+ CJS dist/index.js.map 11.17 KB
17
+ CJS ⚡️ Build success in 55ms
18
+ ESM dist/index.mjs 6.87 KB
19
+ ESM dist/index.mjs.map 11.14 KB
20
+ ESM ⚡️ Build success in 55ms
21
21
  DTS Build start
22
- DTS ⚡️ Build success in 6952ms
22
+ DTS ⚡️ Build success in 6947ms
23
23
  DTS dist/index.d.mts 1.95 KB
24
24
  DTS dist/index.d.ts 1.95 KB
package/CHANGELOG.md CHANGED
@@ -1,5 +1,17 @@
1
1
  # @objectstack/nestjs
2
2
 
3
+ ## 1.0.11
4
+
5
+ ### Patch Changes
6
+
7
+ - @objectstack/runtime@1.0.11
8
+
9
+ ## 1.0.10
10
+
11
+ ### Patch Changes
12
+
13
+ - @objectstack/runtime@1.0.10
14
+
3
15
  ## 1.0.9
4
16
 
5
17
  ### Patch Changes
package/dist/index.js CHANGED
@@ -131,8 +131,8 @@ var ObjectStackController = class {
131
131
  async metadata(req, res, body) {
132
132
  try {
133
133
  let path = req.params[0] || "";
134
- if (req.url.includes("/metadata")) {
135
- path = req.url.split("/metadata")[1].split("?")[0];
134
+ if (req.url.includes("/meta")) {
135
+ path = req.url.split("/meta")[1].split("?")[0];
136
136
  }
137
137
  const payload = body || req.body;
138
138
  const result = await this.service.dispatcher.handleMetadata(path, { request: req }, req.method, payload);
@@ -183,7 +183,7 @@ __decorateClass([
183
183
  __decorateParam(2, (0, import_common.Body)())
184
184
  ], ObjectStackController.prototype, "auth", 1);
185
185
  __decorateClass([
186
- (0, import_common.All)("metadata*"),
186
+ (0, import_common.All)("meta*"),
187
187
  __decorateParam(0, (0, import_common.Req)()),
188
188
  __decorateParam(1, (0, import_common.Res)()),
189
189
  __decorateParam(2, (0, import_common.Body)())
package/dist/index.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/index.ts"],"sourcesContent":["import { 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('metadata*')\n async metadata(@Req() req: any, @Res() res: any, @Body() body?: any) {\n try {\n // /api/metadata/objects -> objects\n let path = req.params[0] || ''; \n if (req.url.includes('/metadata')) {\n path = req.url.split('/metadata')[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":["import { 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"]}
package/dist/index.mjs CHANGED
@@ -105,8 +105,8 @@ var ObjectStackController = class {
105
105
  async metadata(req, res, body) {
106
106
  try {
107
107
  let path = req.params[0] || "";
108
- if (req.url.includes("/metadata")) {
109
- path = req.url.split("/metadata")[1].split("?")[0];
108
+ if (req.url.includes("/meta")) {
109
+ path = req.url.split("/meta")[1].split("?")[0];
110
110
  }
111
111
  const payload = body || req.body;
112
112
  const result = await this.service.dispatcher.handleMetadata(path, { request: req }, req.method, payload);
@@ -157,7 +157,7 @@ __decorateClass([
157
157
  __decorateParam(2, Body())
158
158
  ], ObjectStackController.prototype, "auth", 1);
159
159
  __decorateClass([
160
- All("metadata*"),
160
+ All("meta*"),
161
161
  __decorateParam(0, Req()),
162
162
  __decorateParam(1, Res()),
163
163
  __decorateParam(2, Body())
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/index.ts"],"sourcesContent":["import { 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('metadata*')\n async metadata(@Req() req: any, @Res() res: any, @Body() body?: any) {\n try {\n // /api/metadata/objects -> objects\n let path = req.params[0] || ''; \n if (req.url.includes('/metadata')) {\n path = req.url.split('/metadata')[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":["import { 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":[]}
package/package.json CHANGED
@@ -1,19 +1,19 @@
1
1
  {
2
2
  "name": "@objectstack/nestjs",
3
- "version": "1.0.9",
3
+ "version": "1.0.11",
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": "^10.0.0",
9
9
  "@nestjs/core": "^10.0.0",
10
- "@objectstack/runtime": "1.0.9"
10
+ "@objectstack/runtime": "1.0.11"
11
11
  },
12
12
  "devDependencies": {
13
13
  "@nestjs/common": "^10.0.0",
14
14
  "@nestjs/core": "^10.0.0",
15
15
  "typescript": "^5.0.0",
16
- "@objectstack/runtime": "1.0.9"
16
+ "@objectstack/runtime": "1.0.11"
17
17
  },
18
18
  "scripts": {
19
19
  "build": "tsup --config ../../../tsup.config.ts"
package/src/index.ts CHANGED
@@ -113,13 +113,13 @@ export class ObjectStackController {
113
113
  }
114
114
 
115
115
  // Metadata
116
- @All('metadata*')
116
+ @All('meta*')
117
117
  async metadata(@Req() req: any, @Res() res: any, @Body() body?: any) {
118
118
  try {
119
- // /api/metadata/objects -> objects
119
+ // /api/meta/objects -> objects
120
120
  let path = req.params[0] || '';
121
- if (req.url.includes('/metadata')) {
122
- path = req.url.split('/metadata')[1].split('?')[0];
121
+ if (req.url.includes('/meta')) {
122
+ path = req.url.split('/meta')[1].split('?')[0];
123
123
  }
124
124
 
125
125
  // Use injected body or fallback to req.body