@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.
- package/.turbo/turbo-build.log +8 -8
- package/CHANGELOG.md +16 -0
- package/dist/index.js +27 -0
- package/dist/index.js.map +1 -1
- package/dist/index.mjs +27 -0
- package/dist/index.mjs.map +1 -1
- package/package.json +3 -3
- package/src/index.ts +43 -0
- package/src/nestjs.test.ts +103 -0
package/.turbo/turbo-build.log
CHANGED
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
|
|
2
|
-
> @objectstack/nestjs@2.0.
|
|
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
|
[34mCLI[39m Building entry: src/index.ts
|
|
@@ -12,13 +12,13 @@
|
|
|
12
12
|
[34mCJS[39m Build start
|
|
13
13
|
[33mESM[39m [33mYou have emitDecoratorMetadata enabled but @swc/core was not installed, skipping swc plugin[39m
|
|
14
14
|
[33mCJS[39m [33mYou have emitDecoratorMetadata enabled but @swc/core was not installed, skipping swc plugin[39m
|
|
15
|
-
[32mESM[39m [1mdist/index.mjs [22m[
|
|
16
|
-
[32mESM[39m [1mdist/index.mjs.map [22m[
|
|
17
|
-
[32mESM[39m ⚡️ Build success in
|
|
18
|
-
[32mCJS[39m [1mdist/index.js [22m[
|
|
19
|
-
[32mCJS[39m [1mdist/index.js.map [22m[
|
|
20
|
-
[32mCJS[39m ⚡️ Build success in
|
|
15
|
+
[32mESM[39m [1mdist/index.mjs [22m[32m8.15 KB[39m
|
|
16
|
+
[32mESM[39m [1mdist/index.mjs.map [22m[32m13.94 KB[39m
|
|
17
|
+
[32mESM[39m ⚡️ Build success in 38ms
|
|
18
|
+
[32mCJS[39m [1mdist/index.js [22m[32m9.76 KB[39m
|
|
19
|
+
[32mCJS[39m [1mdist/index.js.map [22m[32m13.97 KB[39m
|
|
20
|
+
[32mCJS[39m ⚡️ Build success in 45ms
|
|
21
21
|
[34mDTS[39m Build start
|
|
22
|
-
[32mDTS[39m ⚡️ Build success in
|
|
22
|
+
[32mDTS[39m ⚡️ Build success in 6492ms
|
|
23
23
|
[32mDTS[39m [1mdist/index.d.mts [22m[32m8.75 KB[39m
|
|
24
24
|
[32mDTS[39m [1mdist/index.d.ts [22m[32m8.75 KB[39m
|
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":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAEA,oBAA2J;AAC3J,IAAAA,iBAA2B;AAC3B,qBAAmE;AAE5D,IAAM,gBAAgB;AAEtB,IAAM,iBAAa;AAAA,EACxB,CAAC,OAAgB,QAA0B;AACzC,WAAO,IAAI,aAAa,EAAE,WAAW;AAAA,EACvC;AACF;AAKO,IAAM,qBAAN,MAAyB;AAAA,EAG9B,YAAoD,QAAsB;AAAtB;AAClD,SAAK,aAAa,IAAI,8BAAe,MAAM;AAAA,EAC7C;AAAA,EAEA,YAAY;AACV,WAAO,KAAK;AAAA,EACd;AACF;AAVa,qBAAN;AAAA,MADN,2BAAW;AAAA,EAIG,6CAAO,aAAa;AAAA,GAHtB;AAeN,IAAM,wBAAN,MAA4B;AAAA,EACjC,YAA6B,SAA6B;AAA7B;AAAA,EAA8B;AAAA,EAE3D,MAAc,kBAAkB,QAA8B,KAAU;AACpE,QAAI,OAAO,SAAS;AAChB,UAAI,OAAO,UAAU;AAChB,YAAI,OAAO,OAAO,SAAS,MAAM;AACjC,YAAI,OAAO,SAAS,SAAS;AACzB,iBAAO,QAAQ,OAAO,SAAS,OAAO,EAAE,QAAQ,CAAC,CAAC,GAAG,CAAC,MAAM,IAAI,UAAU,GAAG,CAAC,CAAC;AAAA,QACnF;AACA,eAAO,IAAI,KAAK,OAAO,SAAS,IAAI;AAAA,MACzC;AACA,UAAI,OAAO,QAAQ;AAChB,cAAM,WAAW,OAAO;AAGxB,YAAI,SAAS,SAAS,cAAc,SAAS,KAAK;AAC9C,iBAAO,IAAI,SAAS,SAAS,GAAG;AAAA,QACpC;AAGA,YAAI,SAAS,SAAS,YAAY,SAAS,QAAQ;AAC/C,cAAI,SAAS,SAAS;AAClB,mBAAO,QAAQ,SAAS,OAAO,EAAE,QAAQ,CAAC,CAAC,GAAG,CAAC,MAAM,IAAI,UAAU,GAAG,CAAC,CAAC;AAAA,UAC5E;AACA,mBAAS,OAAO,KAAK,GAAG;AACxB;AAAA,QACJ;AAGA,YAAI,YAAY,OAAO,SAAS,WAAW,YAAY,OAAO,SAAS,SAAS,YAAY;AACvF,cAAI,OAAO,SAAS,MAAM;AAC1B,cAAI,SAAS,WAAW,OAAO,SAAS,QAAQ,YAAY,YAAY;AACpE,qBAAS,QAAQ,QAAQ,CAAC,GAAW,MAAc,IAAI,UAAU,GAAG,CAAC,CAAC;AAAA,UAC1E;AACA,gBAAM,OAAO,MAAM,SAAS,KAAK;AACjC,cAAI,KAAK,IAAI;AACb;AAAA,QACL;AACA,eAAO,IAAI,OAAO,GAAG,EAAE,KAAK,QAAQ;AAAA,MACvC;AAAA,IACJ;AACA,WAAO,IAAI,OAAO,GAAG,EAAE,KAAK,EAAE,SAAS,OAAO,OAAO,EAAE,SAAS,aAAa,MAAM,IAAI,EAAE,CAAC;AAAA,EAC9F;AAAA,EAEA,MAAc,YAAY,KAAU,KAAU;AAC1C,WAAO,IAAI,OAAO,IAAI,cAAc,GAAG,EAAE,KAAK;AAAA,MAC1C,SAAS;AAAA,MACT,OAAO;AAAA,QACH,SAAS,IAAI,WAAW;AAAA,QACxB,MAAM,IAAI,cAAc;AAAA,QACxB,SAAS,IAAI;AAAA,MACjB;AAAA,IACJ,CAAC;AAAA,EACL;AAAA,EAIA,YAAY;AACV,WAAO,EAAE,MAAM,KAAK,QAAQ,WAAW,iBAAiB,MAAM,EAAE;AAAA,EAClE;AAAA,EAGA,MAAM,QAAgB,MAAkB,KAAiB,KAAU;AACjE,QAAI;AACA,YAAM,SAAS,MAAM,KAAK,QAAQ,WAAW,cAAc,MAAM,EAAE,SAAS,IAAI,CAAC;AACjF,aAAO,IAAI,KAAK,MAAM;AAAA,IAC1B,SAAS,KAAK;AACV,aAAO,KAAK,YAAY,KAAK,GAAG;AAAA,IACpC;AAAA,EACF;AAAA,EAIA,MAAM,KAAY,KAAiB,KAAkB,MAAW;AAC9D,QAAI;AACA,YAAM,OAAO,IAAI,OAAO,CAAC,KAAK,IAAI,IAAI,MAAM,QAAQ,EAAE,CAAC,GAAG,MAAM,GAAG,EAAE,CAAC,KAAK;AAC3E,YAAM,SAAS,MAAM,KAAK,QAAQ,WAAW,WAAW,MAAM,IAAI,QAAQ,MAAM,EAAE,SAAS,KAAK,UAAU,IAAI,CAAC;AAC/G,aAAO,KAAK,kBAAkB,QAAQ,GAAG;AAAA,IAC7C,SAAS,KAAU;AACf,aAAO,KAAK,YAAY,KAAK,GAAG;AAAA,IACpC;AAAA,EACF;AAAA,EAIA,MAAM,SAAgB,KAAiB,KAAkB,MAAY;AACjE,QAAI;AAEA,UAAI,OAAO,IAAI,OAAO,CAAC,KAAK;AAC5B,UAAI,IAAI,IAAI,SAAS,OAAO,GAAG;AAC5B,eAAO,IAAI,IAAI,MAAM,OAAO,EAAE,CAAC,EAAE,MAAM,GAAG,EAAE,CAAC;AAAA,MAChD;AAGA,YAAM,UAAU,QAAQ,IAAI;AAE5B,YAAM,SAAS,MAAM,KAAK,QAAQ,WAAW,eAAe,MAAM,EAAE,SAAS,IAAI,GAAG,IAAI,QAAQ,OAAO;AACvG,aAAO,KAAK,kBAAkB,QAAQ,GAAG;AAAA,IAC7C,SAAS,KAAK;AACV,aAAO,KAAK,YAAY,KAAK,GAAG;AAAA,IACpC;AAAA,EACJ;AAAA,EAIA,MAAM,KAAY,KAAiB,KAAkB,MAAoB,OAAY;AACjF,QAAI;AACA,UAAI,OAAO,IAAI,OAAO,CAAC,KAAK;AAC5B,UAAI,IAAI,IAAI,SAAS,OAAO,GAAG;AAC5B,eAAO,IAAI,IAAI,UAAU,IAAI,IAAI,QAAQ,OAAO,IAAI,CAAC,EAAE,MAAM,GAAG,EAAE,CAAC;AAAA,MACtE;AAEA,YAAM,SAAS,MAAM,KAAK,QAAQ,WAAW,WAAW,MAAM,IAAI,QAAQ,MAAM,OAAO,EAAE,SAAS,IAAI,CAAC;AACvG,aAAO,KAAK,kBAAkB,QAAQ,GAAG;AAAA,IAC7C,SAAS,KAAK;AACV,aAAO,KAAK,YAAY,KAAK,GAAG;AAAA,IACpC;AAAA,EACJ;AAAA,EAIA,MAAM,QAAe,KAAiB,KAAU;AAC5C,QAAI;AACA,UAAI,OAAO,IAAI,OAAO,CAAC,KAAK;AAC5B,UAAI,IAAI,IAAI,SAAS,UAAU,GAAG;AAC/B,eAAO,IAAI,IAAI,UAAU,IAAI,IAAI,QAAQ,UAAU,IAAI,CAAC,EAAE,MAAM,GAAG,EAAE,CAAC;AAAA,MACzE;AAGA,YAAM,OAAO,IAAI,QAAQ,IAAI,OAAO;AAEpC,YAAM,SAAS,MAAM,KAAK,QAAQ,WAAW,cAAc,MAAM,IAAI,QAAQ,MAAM,EAAE,SAAS,IAAI,CAAC;AACnG,aAAO,KAAK,kBAAkB,QAAQ,GAAG;AAAA,IAC7C,SAAS,KAAK;AACV,aAAO,KAAK,YAAY,KAAK,GAAG;AAAA,IACpC;AAAA,EACJ;AACF;AAhFE;AAAA,MADC,mBAAI;AAAA,GAzDM,sBA0DX;AAKM;AAAA,MADL,oBAAK,SAAS;AAAA,EACA,2CAAK;AAAA,EAAc,0CAAI;AAAA,EAAa,0CAAI;AAAA,GA/D5C,sBA+DL;AAWA;AAAA,MADL,mBAAI,QAAQ;AAAA,EACD,0CAAI;AAAA,EAAa,0CAAI;AAAA,EAAa,2CAAK;AAAA,GA1ExC,sBA0EL;AAYA;AAAA,MADL,mBAAI,OAAO;AAAA,EACI,0CAAI;AAAA,EAAa,0CAAI;AAAA,EAAa,2CAAK;AAAA,GAtF5C,sBAsFL;AAoBA;AAAA,MADL,mBAAI,OAAO;AAAA,EACA,0CAAI;AAAA,EAAa,0CAAI;AAAA,EAAa,2CAAK;AAAA,EAAc,4CAAM;AAAA,GA1G5D,sBA0GL;AAgBA;AAAA,MADL,mBAAI,UAAU;AAAA,EACA,0CAAI;AAAA,EAAa,0CAAI;AAAA,GA1HzB,sBA0HL;AA1HK,wBAAN;AAAA,MADN,0BAAW,KAAK;AAAA,GACJ;AA+IN,IAAM,sBAAN,MAA0B;AAAA,EAE/B,SAAgB,KAAU;AACxB,WAAO,IAAI,SAAS,MAAM;AAAA,EAC5B;AACF;AAHE;AAAA,MADC,mBAAI,aAAa;AAAA,EACR,0CAAI;AAAA,GAFH,oBAEX;AAFW,sBAAN;AAAA,MADN,0BAAW,aAAa;AAAA,GACZ;AAWN,IAAM,oBAAN,MAAwB;AAAA,EAC7B,OAAO,QAAQ,QAAqC;AAClD,UAAM,iBAA2B;AAAA,MAC/B,SAAS;AAAA,MACT,UAAU;AAAA,IACZ;AAEA,WAAO;AAAA,MACL,QAAQ;AAAA,MACR,aAAa,CAAC,uBAAuB,mBAAmB;AAAA,MACxD,WAAW,CAAC,gBAAgB,kBAAkB;AAAA,MAC9C,SAAS,CAAC,gBAAgB,kBAAkB;AAAA,IAC9C;AAAA,EACF;AACF;AAda,oBAAN;AAAA,MAFN,sBAAO;AAAA,MACP,sBAAO,CAAC,CAAC;AAAA,GACG;","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);
|
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 { 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.
|
|
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.
|
|
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.
|
|
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);
|
package/src/nestjs.test.ts
CHANGED
|
@@ -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' };
|