nesties 1.1.3 → 1.1.5
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/dist/index.cjs +34 -16
- package/dist/index.cjs.map +3 -3
- package/dist/index.d.ts +1 -0
- package/dist/index.mjs +38 -20
- package/dist/index.mjs.map +4 -4
- package/dist/src/abortable-module/abort-signal.provider.d.ts +3 -3
- package/dist/src/abortable-module/abortable.module.d.ts +2 -2
- package/dist/src/abortable-module/abortable.token.d.ts +5 -5
- package/dist/src/create-provider.d.ts +13 -0
- package/index.ts +1 -0
- package/package.json +1 -1
- package/build.js +0 -167
package/dist/index.cjs
CHANGED
|
@@ -23,6 +23,7 @@ var __decorateClass = (decorators, target, key, kind) => {
|
|
|
23
23
|
if (kind && result) __defProp(target, key, result);
|
|
24
24
|
return result;
|
|
25
25
|
};
|
|
26
|
+
var __decorateParam = (index, decorator) => (target, key) => decorator(target, key, index);
|
|
26
27
|
|
|
27
28
|
// index.ts
|
|
28
29
|
var index_exports = {};
|
|
@@ -57,6 +58,7 @@ __export(index_exports, {
|
|
|
57
58
|
TokenGuard: () => TokenGuard,
|
|
58
59
|
abortableToken: () => abortableToken,
|
|
59
60
|
createAbortableProvider: () => createAbortableProvider,
|
|
61
|
+
createProvider: () => createProvider,
|
|
60
62
|
fromAbortable: () => fromAbortable,
|
|
61
63
|
getClassFromClassOrArray: () => getClassFromClassOrArray,
|
|
62
64
|
takeUntilAbort: () => takeUntilAbort
|
|
@@ -251,6 +253,7 @@ var DataBody = createDataPipeDec(import_common2.Body);
|
|
|
251
253
|
|
|
252
254
|
// src/token.guard.ts
|
|
253
255
|
var import_common3 = require("@nestjs/common");
|
|
256
|
+
var import_config = require("@nestjs/config");
|
|
254
257
|
var import_swagger4 = require("@nestjs/swagger");
|
|
255
258
|
var TokenGuard = class {
|
|
256
259
|
constructor(config) {
|
|
@@ -267,7 +270,8 @@ var TokenGuard = class {
|
|
|
267
270
|
}
|
|
268
271
|
};
|
|
269
272
|
TokenGuard = __decorateClass([
|
|
270
|
-
(0, import_common3.Injectable)()
|
|
273
|
+
(0, import_common3.Injectable)(),
|
|
274
|
+
__decorateParam(0, (0, import_common3.Inject)(import_config.ConfigService))
|
|
271
275
|
], TokenGuard);
|
|
272
276
|
var RequireToken = () => MergeClassOrMethodDecorators([
|
|
273
277
|
(0, import_common3.UseGuards)(TokenGuard),
|
|
@@ -361,15 +365,27 @@ var import_nfkit = require("nfkit");
|
|
|
361
365
|
// src/abortable-module/abort-signal.provider.ts
|
|
362
366
|
var import_common5 = require("@nestjs/common");
|
|
363
367
|
var import_core = require("@nestjs/core");
|
|
364
|
-
|
|
365
|
-
|
|
366
|
-
|
|
367
|
-
|
|
368
|
-
|
|
369
|
-
|
|
370
|
-
|
|
371
|
-
}
|
|
368
|
+
|
|
369
|
+
// src/create-provider.ts
|
|
370
|
+
var createProvider = (options, factory) => {
|
|
371
|
+
return {
|
|
372
|
+
useFactory: factory,
|
|
373
|
+
...options
|
|
374
|
+
};
|
|
372
375
|
};
|
|
376
|
+
|
|
377
|
+
// src/abortable-module/abort-signal.provider.ts
|
|
378
|
+
var ABORT_SIGNAL = Symbol(
|
|
379
|
+
"ABORT_SIGNAL"
|
|
380
|
+
);
|
|
381
|
+
var AbortSignalProvider = createProvider(
|
|
382
|
+
{
|
|
383
|
+
provide: ABORT_SIGNAL,
|
|
384
|
+
scope: import_common5.Scope.REQUEST,
|
|
385
|
+
inject: [import_core.REQUEST]
|
|
386
|
+
},
|
|
387
|
+
createAbortSignalFromHttp
|
|
388
|
+
);
|
|
373
389
|
var InjectAbortSignal = () => (0, import_common5.Inject)(ABORT_SIGNAL);
|
|
374
390
|
|
|
375
391
|
// src/abortable-module/abortable.token.ts
|
|
@@ -399,12 +415,13 @@ function InjectAbortable(token) {
|
|
|
399
415
|
}
|
|
400
416
|
function createAbortableProvider(token, opts) {
|
|
401
417
|
const provide = abortableToken(token);
|
|
402
|
-
return
|
|
403
|
-
|
|
404
|
-
|
|
405
|
-
|
|
406
|
-
|
|
407
|
-
|
|
418
|
+
return createProvider(
|
|
419
|
+
{
|
|
420
|
+
provide,
|
|
421
|
+
scope: import_common6.Scope.REQUEST,
|
|
422
|
+
inject: [import_core2.ModuleRef, import_core2.REQUEST, ABORT_SIGNAL]
|
|
423
|
+
},
|
|
424
|
+
async (moduleRef, req, signal) => {
|
|
408
425
|
const ctxId = import_core2.ContextIdFactory.getByRequest(req);
|
|
409
426
|
const svc = await moduleRef.resolve(token, ctxId, { strict: false });
|
|
410
427
|
if (svc == null) {
|
|
@@ -416,7 +433,7 @@ function createAbortableProvider(token, opts) {
|
|
|
416
433
|
}
|
|
417
434
|
return (0, import_nfkit.abortable)(svc, signal, opts);
|
|
418
435
|
}
|
|
419
|
-
|
|
436
|
+
);
|
|
420
437
|
}
|
|
421
438
|
|
|
422
439
|
// src/abortable-module/abortable.module.ts
|
|
@@ -476,6 +493,7 @@ AbortableModule = __decorateClass([
|
|
|
476
493
|
TokenGuard,
|
|
477
494
|
abortableToken,
|
|
478
495
|
createAbortableProvider,
|
|
496
|
+
createProvider,
|
|
479
497
|
fromAbortable,
|
|
480
498
|
getClassFromClassOrArray,
|
|
481
499
|
takeUntilAbort
|
package/dist/index.cjs.map
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
|
-
"sources": ["../index.ts", "../src/insert-field.ts", "../src/merge.ts", "../src/return-message.ts", "../src/openapi.ts", "../src/pipe.ts", "../src/token.guard.ts", "../src/abort-utils.ts", "../src/abort-http-signal.ts", "../src/abortable-module/abortable.token.ts", "../src/abortable-module/abort-signal.provider.ts", "../src/abortable-module/abortable.module.ts"],
|
|
4
|
-
"sourcesContent": ["export * from './src/insert-field';\nexport * from './src/merge';\nexport * from './src/return-message';\nexport * from './src/openapi';\nexport * from './src/pipe';\nexport * from './src/token.guard';\nexport * from './src/abort-utils';\nexport * from './src/abortable-module';\n", "import { ApiProperty, ApiPropertyOptions } from '@nestjs/swagger';\n\nexport type AnyClass = new (...args: any[]) => any;\nexport type ClassOrArray = AnyClass | [AnyClass];\nexport type ClassType<T> = new (...args: any[]) => T;\nexport type TypeFromClass<T> = T extends new (...args: any[]) => infer U\n ? U\n : never;\nexport type ParamsFromClass<T> = T extends new (...args: infer U) => any\n ? U\n : never;\nexport type ParseType<IC extends ClassOrArray> = IC extends [infer U]\n ? TypeFromClass<U>[]\n : TypeFromClass<IC>;\n\nexport function getClassFromClassOrArray(o: ClassOrArray) {\n return o instanceof Array ? o[0] : o;\n}\n\nexport interface InsertOptions<C extends ClassOrArray = ClassOrArray> {\n type: C;\n options?: ApiPropertyOptions;\n}\n\ntype TypeFromInsertOptions<O extends InsertOptions> =\n O extends InsertOptions<infer C>\n ?\n | ParseType<C>\n | (O extends { options: { required: true } } ? never : undefined)\n : never;\n\ntype Merge<T, U> = {\n [K in keyof T | keyof U]: K extends keyof T\n ? T[K]\n : K extends keyof U\n ? U[K]\n : never;\n};\n\nexport function InsertField<\n C extends AnyClass,\n M extends Record<string, InsertOptions>,\n>(\n cl: C,\n map: M,\n newName?: string,\n): new (...args: ParamsFromClass<C>) => Merge<\n {\n [F in keyof M]: TypeFromInsertOptions<M[F]>;\n },\n TypeFromClass<C>\n> {\n const extendedCl = class extends cl {};\n for (const key in map) {\n ApiProperty({\n type: map[key].type,\n ...(map[key].options || {}),\n })(extendedCl.prototype, key);\n }\n Object.defineProperty(extendedCl, 'name', {\n value: newName || cl.name,\n });\n return extendedCl;\n}\n", "export function MergePropertyDecorators(\n decs: PropertyDecorator[],\n): PropertyDecorator {\n return (obj, key) => {\n for (const dec of decs) {\n dec(obj, key);\n }\n };\n}\n\nexport function MergeMethodDecorators(\n decs: MethodDecorator[],\n): MethodDecorator {\n return (obj, key, descriptor) => {\n for (const dec of decs) {\n dec(obj, key, descriptor);\n }\n };\n}\n\nexport function MergeClassDecorators(decs: ClassDecorator[]): ClassDecorator {\n return (obj) => {\n for (const dec of decs) {\n dec(obj);\n }\n };\n}\n\nexport function MergeParameterDecorators(\n decs: ParameterDecorator[],\n): ParameterDecorator {\n return (obj, key, index) => {\n for (const dec of decs) {\n dec(obj, key, index);\n }\n };\n}\n\nexport function MergeClassOrMethodDecorators(\n decs: (ClassDecorator & MethodDecorator)[],\n): ClassDecorator & MethodDecorator {\n return (obj, key?, descriptor?) => {\n if (descriptor) {\n for (const dec of decs) {\n dec(obj, key, descriptor);\n }\n } else {\n for (const dec of decs) {\n dec(obj);\n }\n }\n };\n}\n", "import { ApiProperty } from '@nestjs/swagger';\nimport { HttpException } from '@nestjs/common';\nimport {\n AnyClass,\n ClassOrArray,\n getClassFromClassOrArray,\n InsertField,\n ParseType,\n} from './insert-field';\n\nexport interface PageSettingsWise {\n pageCount: number;\n recordsPerPage: number;\n}\n\nexport interface BlankReturnMessage {\n statusCode: number;\n message: string;\n success: boolean;\n timestamp: Date;\n}\n\nexport interface ReturnMessage<T> extends BlankReturnMessage {\n data?: T;\n}\n\nexport class BlankReturnMessageDto implements BlankReturnMessage {\n @ApiProperty({ description: 'Return code', type: Number })\n statusCode: number;\n @ApiProperty({ description: 'Return message', type: String })\n message: string;\n @ApiProperty({ description: 'Whether success.', type: Boolean })\n success: boolean;\n @ApiProperty({ description: 'Return timestamp', type: Date })\n timestamp: Date;\n constructor(statusCode: number, message?: string) {\n this.statusCode = statusCode;\n this.message = message || 'success';\n this.success = statusCode < 400;\n this.timestamp = new Date();\n }\n\n toException() {\n return new HttpException(this, this.statusCode);\n }\n}\n\nexport class BlankPaginatedReturnMessageDto\n extends BlankReturnMessageDto\n implements PageSettingsWise\n{\n @ApiProperty({ description: 'Total record count.', type: Number })\n total: number;\n @ApiProperty({ description: 'Total page count.', type: Number })\n totalPages: number;\n @ApiProperty({ description: 'Current page.', type: Number })\n pageCount: number;\n @ApiProperty({ description: 'Records per page.', type: Number })\n recordsPerPage: number;\n constructor(\n statusCode: number,\n message: string,\n total: number,\n pageSettings: PageSettingsWise,\n ) {\n super(statusCode, message);\n this.total = total;\n this.pageCount = pageSettings.pageCount;\n this.recordsPerPage = pageSettings.recordsPerPage;\n this.totalPages = Math.ceil(total / pageSettings.recordsPerPage);\n }\n}\n\nexport class GenericReturnMessageDto<T>\n extends BlankReturnMessageDto\n implements ReturnMessage<T>\n{\n data?: T;\n constructor(statusCode: number, message?: string, data?: T) {\n super(statusCode, message);\n this.data = data;\n }\n}\n\nexport function ReturnMessageDto<T extends ClassOrArray>(\n type: T,\n): new (\n statusCode: number,\n message: string,\n data: ParseType<T>,\n) => GenericReturnMessageDto<ParseType<T>> {\n return InsertField(\n GenericReturnMessageDto,\n {\n data: {\n type,\n options: {\n required: false,\n description: 'Return data.',\n },\n },\n },\n `${getClassFromClassOrArray(type).name}${Array.isArray(type) ? 'Array' : ''}ReturnMessageDto`,\n );\n}\n\nexport class GenericPaginatedReturnMessageDto<T>\n extends BlankPaginatedReturnMessageDto\n implements PageSettingsWise, ReturnMessage<T[]>\n{\n data: T[];\n constructor(\n statusCode: number,\n message: string,\n data: T[],\n total: number,\n pageSettings: PageSettingsWise,\n ) {\n super(statusCode, message, total, pageSettings);\n this.data = data;\n }\n}\n\nexport function PaginatedReturnMessageDto<T extends AnyClass>(\n type: T,\n): new (\n statusCode: number,\n message: string,\n data: InstanceType<T>[],\n total: number,\n pageSettings: PageSettingsWise,\n) => GenericPaginatedReturnMessageDto<InstanceType<T>> {\n return InsertField(\n GenericPaginatedReturnMessageDto,\n {\n data: {\n type: [type],\n options: {\n required: false,\n description: 'Return data.',\n },\n },\n },\n `${getClassFromClassOrArray(type).name}PaginatedReturnMessageDto`,\n );\n}\n\nexport class StringReturnMessageDto extends GenericReturnMessageDto<string> {\n @ApiProperty({ description: 'Return data.', type: String, required: false })\n data: string;\n}\n", "import { BlankReturnMessageDto, ReturnMessageDto } from './return-message';\nimport { ApiResponse, ApiResponseOptions } from '@nestjs/swagger';\nimport { ClassOrArray } from './insert-field';\n\nexport const ApiTypeResponse = (\n // eslint-disable-next-line @typescript-eslint/ban-types\n type: ClassOrArray,\n options: ApiResponseOptions = {},\n) =>\n ApiResponse({\n status: 200,\n type: ReturnMessageDto(type),\n ...options,\n });\n\nexport const ApiBlankResponse = (options: ApiResponseOptions = {}) =>\n ApiResponse({\n status: 200,\n type: BlankReturnMessageDto,\n ...options,\n });\n\nexport const ApiError = (status: number, description: string) =>\n ApiBlankResponse({ status, description });\n\nexport const ApiErrorTyped = (\n status: number,\n description: string,\n type: ClassOrArray,\n) =>\n ApiTypeResponse(type, {\n status,\n description,\n });\n", "import {\n Body,\n PipeTransform,\n Query,\n Type,\n ValidationPipe,\n} from '@nestjs/common';\n\nexport const DataPipe = () =>\n new ValidationPipe({\n transform: true,\n transformOptions: { enableImplicitConversion: true },\n });\n\nconst createDataPipeDec =\n (\n nestFieldDec: (\n ...pipes: (Type<PipeTransform> | PipeTransform)[]\n ) => ParameterDecorator,\n ) =>\n (...extraPipes: (Type<PipeTransform> | PipeTransform)[]) =>\n nestFieldDec(DataPipe(), ...extraPipes);\n\nexport const DataQuery = createDataPipeDec(Query);\nexport const DataBody = createDataPipeDec(Body);\n", "import {\n CanActivate,\n ExecutionContext,\n Injectable,\n UseGuards,\n} from '@nestjs/common';\nimport { ConfigService } from '@nestjs/config';\nimport { ApiHeader } from '@nestjs/swagger';\nimport { BlankReturnMessageDto } from './return-message';\nimport { MergeClassOrMethodDecorators } from './merge';\nimport { ApiError } from './openapi';\n\n@Injectable()\nexport class TokenGuard implements CanActivate {\n private token = this.config.get<string>('SERVER_TOKEN');\n constructor(private config: ConfigService) {}\n\n async canActivate(context: ExecutionContext) {\n const request = context.switchToHttp().getRequest();\n const token = request.headers['x-server-token'];\n if (this.token && token !== this.token) {\n throw new BlankReturnMessageDto(401, 'Unauthorized').toException();\n }\n return true;\n }\n}\n\nexport const RequireToken = () =>\n MergeClassOrMethodDecorators([\n UseGuards(TokenGuard),\n ApiHeader({\n name: 'x-server-token',\n description: 'Server token',\n required: false,\n }),\n ApiError(401, 'Incorrect server token provided'),\n ]);\n", "import { Observable, takeUntil } from 'rxjs';\nimport { createParamDecorator, ExecutionContext } from '@nestjs/common';\nimport { IncomingMessage } from 'node:http';\nimport { BlankReturnMessageDto } from './return-message';\nimport type { Request, Response } from 'express';\nimport { createAbortSignalFromHttp } from './abort-http-signal';\n\nexport type AbortableFn<T> = (ac: AbortController) => Promise<T>;\n\nexport const fromAbortable = <T>(fn: AbortableFn<T>): Observable<T> => {\n return new Observable<T>((subscriber) => {\n const ac = new AbortController();\n\n fn(ac).then(\n (value) => {\n if (!ac.signal.aborted && !subscriber.closed) {\n subscriber.next(value);\n subscriber.complete();\n }\n },\n (err) => {\n if (ac.signal.aborted) {\n if (!subscriber.closed) subscriber.complete();\n return;\n }\n if (!subscriber.closed) subscriber.error(err);\n },\n );\n\n return () => {\n if (!ac.signal.aborted) ac.abort();\n };\n });\n};\n\nexport const takeUntilAbort = <T>(signal: AbortSignal) => {\n return (source: Observable<T>) => {\n return source.pipe(\n takeUntil(\n new Observable<void>((subscriber) => {\n const onAbort = () => subscriber.next();\n if (signal.aborted) {\n subscriber.next();\n } else {\n signal.addEventListener('abort', onAbort, { once: true });\n }\n return () => signal.removeEventListener('abort', onAbort);\n }),\n ),\n );\n };\n};\n\nexport const As = createParamDecorator(\n (_data: unknown, ctx: ExecutionContext) => {\n const req = ctx.switchToHttp().getRequest<Request>();\n return createAbortSignalFromHttp(req);\n },\n);\n", "// abort-http-signal.ts\nimport type { IncomingMessage } from 'http';\nimport type { Request as ExpressReq } from 'express';\nimport type { FastifyRequest } from 'fastify';\nimport { BlankReturnMessageDto } from './return-message';\n\ntype AnyReq = ExpressReq | FastifyRequest | IncomingMessage;\n\nfunction toRawReq(req: AnyReq): IncomingMessage {\n // Fastify: req.raw \u662F IncomingMessage\uFF1BExpress: req \u672C\u8EAB\u5C31\u662F IncomingMessage\n if ((req as any)?.raw?.socket) return (req as any).raw as IncomingMessage;\n return req as IncomingMessage;\n}\n\n/**\n * \u4EC5\u901A\u8FC7 req \u81EA\u52A8\u9002\u914D Express/Fastify\uFF0C\u7ED1\u5B9A HTTP \u53D6\u6D88\u5230 AbortSignal\u3002\n * - \u4F18\u5148\u76D1\u542C 'aborted'\uFF08\u5BA2\u6237\u7AEF\u4E2D\u65AD\u6700\u53EF\u9760\u7684\u4FE1\u53F7\uFF09\n * - \u53EF\u9009\u515C\u5E95\uFF1A\u5728 socket 'close' \u65F6\uFF0C\u4EC5\u5F53\u8BF7\u6C42\u672A\u5B8C\u6574\u63A5\u6536/\u6216\u5DF2\u6807\u8BB0 aborted \u65F6\uFF0C\u624D\u89E6\u53D1 abort\uFF0C\u907F\u514D\u6B63\u5E38\u5B8C\u6210\u7684\u8BEF\u4F24\n * - reason \u56FA\u5B9A\u4E3A 499 HttpException\n *\n * \u6CE8\u610F\uFF1A\u82E5\u4F60\u7684 abortable() \u5728 signal.reason \u662F Error \u65F6\u4F1A\u201C\u539F\u6837 throw reason\u201D\uFF0C\n * \u90A3\u4E48\u4E0A\u5C42\u4F1A\u6536\u5230 HttpException(499)\u3002\u82E5\u4ECD\u60F3\u7EDF\u4E00\u629B AbortedError\uFF0C\u53EF\u6539\u4E3A\uFF1A\n * const reason = new AbortedError('Request aborted', httpErr);\n * \u5E76\u8BA9 throwIfAborted \u5148\u629B AbortedError\uFF08\u4FDD\u7559 cause\uFF09\u3002\n */\nexport function createAbortSignalFromHttp(req: AnyReq): AbortSignal {\n const rawReq = toRawReq(req);\n const ac = new AbortController();\n\n const makeReason = () =>\n new BlankReturnMessageDto(499, 'Request aborted').toException();\n\n const abortOnce = () => {\n if (!ac.signal.aborted) ac.abort(makeReason());\n cleanup();\n };\n\n const onClose = () => {\n abortOnce();\n };\n\n const cleanup = () => {\n rawReq.off?.('close', onClose);\n };\n\n rawReq.once?.('close', onClose);\n\n return ac.signal;\n}\n", "import { Inject, Provider, Scope } from '@nestjs/common';\nimport { abortable, AbortableOpts } from 'nfkit';\nimport { ABORT_SIGNAL } from './abort-signal.provider';\nimport { ContextIdFactory, ModuleRef, REQUEST } from '@nestjs/core';\n\nexport type ProviderToken<T = any> =\n | string\n | symbol\n | (new (...args: any[]) => T);\n\nconst tokenMemo = new Map<any, symbol>();\nexport const abortableToken = (token: ProviderToken) => {\n if (tokenMemo.has(token)) return tokenMemo.get(token)!;\n const name = typeof token === 'function' ? token.name : String(token);\n const sym = Symbol.for(`Abortable(${name})`);\n tokenMemo.set(token, sym);\n return sym;\n};\n\n/**\n * \u652F\u6301\u4E24\u79CD\u7528\u6CD5\uFF1A\n * @InjectAbortable(SomeService)\n * @InjectAbortable() // \u81EA\u52A8\u63A8\u65AD\u7C7B\u578B\n */\nexport function InjectAbortable(token?: ProviderToken): ParameterDecorator {\n return (target, propertyKey, parameterIndex) => {\n let actualToken = token;\n\n if (!actualToken) {\n // \u5229\u7528 reflect-metadata \u83B7\u53D6\u53C2\u6570\u7C7B\u578B\n const paramTypes: any[] =\n Reflect.getMetadata('design:paramtypes', target, propertyKey) || [];\n actualToken = paramTypes[parameterIndex];\n if (!actualToken) {\n throw new Error(\n `@InjectAbortable() cannot infer type from metadata: ${target.constructor?.name}[${parameterIndex}]`,\n );\n }\n }\n\n Inject(abortableToken(actualToken))(target, propertyKey, parameterIndex);\n };\n}\n\nexport function createAbortableProvider<T>(\n token: ProviderToken<T>,\n opts?: AbortableOpts,\n): Provider {\n const provide = abortableToken(token);\n\n return {\n provide,\n scope: Scope.REQUEST,\n inject: [ModuleRef, REQUEST, ABORT_SIGNAL],\n // \u26A0\uFE0F \u6CE8\u610F\uFF1A\u7528 async + resolve + contextId + strict:false\n useFactory: async (\n moduleRef: ModuleRef,\n req: Request,\n signal: AbortSignal,\n ) => {\n // \u8BA9\u89E3\u6790\u4E0E\u5F53\u524D\u8BF7\u6C42\u4E0A\u4E0B\u6587\u7ED1\u5B9A\uFF08\u652F\u6301 request/transient \u4F5C\u7528\u57DF\uFF09\n const ctxId = ContextIdFactory.getByRequest(req);\n // \u4E25\u683C\u6A21\u5F0F\u5173\u95ED\uFF0C\u5141\u8BB8\u8DE8\u6A21\u5757\u8FB9\u754C\u89E3\u6790\uFF08\u89E3\u51B3\u6D4B\u8BD5\u91CC forFeature \u5B50\u6A21\u5757\u770B\u4E0D\u5230 DemoService \u7684\u60C5\u51B5\uFF09\n const svc = await moduleRef.resolve<T>(token, ctxId, { strict: false });\n if (svc == null) {\n throw new Error(\n `Abortable: provider \"${String(\n (token as any).name ?? token,\n )}\" not found in container (even with strict:false)`,\n );\n }\n return abortable<T>(svc, signal, opts);\n },\n };\n}\n", "import { Inject, Provider, Scope } from '@nestjs/common';\nimport { REQUEST } from '@nestjs/core';\nimport type { Request as ExpressReq } from 'express';\nimport { createAbortSignalFromHttp } from '../abort-http-signal';\n\nexport const ABORT_SIGNAL = Symbol('ABORT_SIGNAL');\nexport const AbortSignalProvider: Provider = {\n provide: ABORT_SIGNAL,\n scope: Scope.REQUEST,\n inject: [REQUEST],\n useFactory: (req: ExpressReq) => {\n return createAbortSignalFromHttp(req);\n },\n};\n\nexport const InjectAbortSignal = () => Inject(ABORT_SIGNAL);\n", "import { DynamicModule, Module, Provider } from '@nestjs/common';\nimport { ProviderToken, createAbortableProvider } from './abortable.token';\nimport { AbortableOpts } from 'nfkit';\nimport { AbortSignalProvider } from './abort-signal.provider';\n\nexport interface AbortableModuleOptions {\n abortableOptions?: AbortableOpts;\n}\n\n@Module({})\nexport class AbortableModule {\n static forRoot(): DynamicModule {\n return {\n module: AbortableModule,\n providers: [AbortSignalProvider],\n exports: [AbortSignalProvider],\n global: true,\n };\n }\n\n static forFeature(\n tokens: ProviderToken[],\n options?: AbortableModuleOptions,\n ): DynamicModule {\n const providers: Provider[] = tokens.map((token) =>\n createAbortableProvider(token, options?.abortableOptions),\n );\n\n return {\n module: AbortableModule,\n providers: [...providers],\n exports: [...providers],\n };\n }\n}\n"],
|
|
5
|
-
"mappings": "
|
|
3
|
+
"sources": ["../index.ts", "../src/insert-field.ts", "../src/merge.ts", "../src/return-message.ts", "../src/openapi.ts", "../src/pipe.ts", "../src/token.guard.ts", "../src/abort-utils.ts", "../src/abort-http-signal.ts", "../src/abortable-module/abortable.token.ts", "../src/abortable-module/abort-signal.provider.ts", "../src/create-provider.ts", "../src/abortable-module/abortable.module.ts"],
|
|
4
|
+
"sourcesContent": ["export * from './src/insert-field';\nexport * from './src/merge';\nexport * from './src/return-message';\nexport * from './src/openapi';\nexport * from './src/pipe';\nexport * from './src/token.guard';\nexport * from './src/abort-utils';\nexport * from './src/abortable-module';\nexport * from './src/create-provider';\n", "import { ApiProperty, ApiPropertyOptions } from '@nestjs/swagger';\n\nexport type AnyClass = new (...args: any[]) => any;\nexport type ClassOrArray = AnyClass | [AnyClass];\nexport type ClassType<T> = new (...args: any[]) => T;\nexport type TypeFromClass<T> = T extends new (...args: any[]) => infer U\n ? U\n : never;\nexport type ParamsFromClass<T> = T extends new (...args: infer U) => any\n ? U\n : never;\nexport type ParseType<IC extends ClassOrArray> = IC extends [infer U]\n ? TypeFromClass<U>[]\n : TypeFromClass<IC>;\n\nexport function getClassFromClassOrArray(o: ClassOrArray) {\n return o instanceof Array ? o[0] : o;\n}\n\nexport interface InsertOptions<C extends ClassOrArray = ClassOrArray> {\n type: C;\n options?: ApiPropertyOptions;\n}\n\ntype TypeFromInsertOptions<O extends InsertOptions> =\n O extends InsertOptions<infer C>\n ?\n | ParseType<C>\n | (O extends { options: { required: true } } ? never : undefined)\n : never;\n\ntype Merge<T, U> = {\n [K in keyof T | keyof U]: K extends keyof T\n ? T[K]\n : K extends keyof U\n ? U[K]\n : never;\n};\n\nexport function InsertField<\n C extends AnyClass,\n M extends Record<string, InsertOptions>,\n>(\n cl: C,\n map: M,\n newName?: string,\n): new (...args: ParamsFromClass<C>) => Merge<\n {\n [F in keyof M]: TypeFromInsertOptions<M[F]>;\n },\n TypeFromClass<C>\n> {\n const extendedCl = class extends cl {};\n for (const key in map) {\n ApiProperty({\n type: map[key].type,\n ...(map[key].options || {}),\n })(extendedCl.prototype, key);\n }\n Object.defineProperty(extendedCl, 'name', {\n value: newName || cl.name,\n });\n return extendedCl;\n}\n", "export function MergePropertyDecorators(\n decs: PropertyDecorator[],\n): PropertyDecorator {\n return (obj, key) => {\n for (const dec of decs) {\n dec(obj, key);\n }\n };\n}\n\nexport function MergeMethodDecorators(\n decs: MethodDecorator[],\n): MethodDecorator {\n return (obj, key, descriptor) => {\n for (const dec of decs) {\n dec(obj, key, descriptor);\n }\n };\n}\n\nexport function MergeClassDecorators(decs: ClassDecorator[]): ClassDecorator {\n return (obj) => {\n for (const dec of decs) {\n dec(obj);\n }\n };\n}\n\nexport function MergeParameterDecorators(\n decs: ParameterDecorator[],\n): ParameterDecorator {\n return (obj, key, index) => {\n for (const dec of decs) {\n dec(obj, key, index);\n }\n };\n}\n\nexport function MergeClassOrMethodDecorators(\n decs: (ClassDecorator & MethodDecorator)[],\n): ClassDecorator & MethodDecorator {\n return (obj, key?, descriptor?) => {\n if (descriptor) {\n for (const dec of decs) {\n dec(obj, key, descriptor);\n }\n } else {\n for (const dec of decs) {\n dec(obj);\n }\n }\n };\n}\n", "import { ApiProperty } from '@nestjs/swagger';\nimport { HttpException } from '@nestjs/common';\nimport {\n AnyClass,\n ClassOrArray,\n getClassFromClassOrArray,\n InsertField,\n ParseType,\n} from './insert-field';\n\nexport interface PageSettingsWise {\n pageCount: number;\n recordsPerPage: number;\n}\n\nexport interface BlankReturnMessage {\n statusCode: number;\n message: string;\n success: boolean;\n timestamp: Date;\n}\n\nexport interface ReturnMessage<T> extends BlankReturnMessage {\n data?: T;\n}\n\nexport class BlankReturnMessageDto implements BlankReturnMessage {\n @ApiProperty({ description: 'Return code', type: Number })\n statusCode: number;\n @ApiProperty({ description: 'Return message', type: String })\n message: string;\n @ApiProperty({ description: 'Whether success.', type: Boolean })\n success: boolean;\n @ApiProperty({ description: 'Return timestamp', type: Date })\n timestamp: Date;\n constructor(statusCode: number, message?: string) {\n this.statusCode = statusCode;\n this.message = message || 'success';\n this.success = statusCode < 400;\n this.timestamp = new Date();\n }\n\n toException() {\n return new HttpException(this, this.statusCode);\n }\n}\n\nexport class BlankPaginatedReturnMessageDto\n extends BlankReturnMessageDto\n implements PageSettingsWise\n{\n @ApiProperty({ description: 'Total record count.', type: Number })\n total: number;\n @ApiProperty({ description: 'Total page count.', type: Number })\n totalPages: number;\n @ApiProperty({ description: 'Current page.', type: Number })\n pageCount: number;\n @ApiProperty({ description: 'Records per page.', type: Number })\n recordsPerPage: number;\n constructor(\n statusCode: number,\n message: string,\n total: number,\n pageSettings: PageSettingsWise,\n ) {\n super(statusCode, message);\n this.total = total;\n this.pageCount = pageSettings.pageCount;\n this.recordsPerPage = pageSettings.recordsPerPage;\n this.totalPages = Math.ceil(total / pageSettings.recordsPerPage);\n }\n}\n\nexport class GenericReturnMessageDto<T>\n extends BlankReturnMessageDto\n implements ReturnMessage<T>\n{\n data?: T;\n constructor(statusCode: number, message?: string, data?: T) {\n super(statusCode, message);\n this.data = data;\n }\n}\n\nexport function ReturnMessageDto<T extends ClassOrArray>(\n type: T,\n): new (\n statusCode: number,\n message: string,\n data: ParseType<T>,\n) => GenericReturnMessageDto<ParseType<T>> {\n return InsertField(\n GenericReturnMessageDto,\n {\n data: {\n type,\n options: {\n required: false,\n description: 'Return data.',\n },\n },\n },\n `${getClassFromClassOrArray(type).name}${Array.isArray(type) ? 'Array' : ''}ReturnMessageDto`,\n );\n}\n\nexport class GenericPaginatedReturnMessageDto<T>\n extends BlankPaginatedReturnMessageDto\n implements PageSettingsWise, ReturnMessage<T[]>\n{\n data: T[];\n constructor(\n statusCode: number,\n message: string,\n data: T[],\n total: number,\n pageSettings: PageSettingsWise,\n ) {\n super(statusCode, message, total, pageSettings);\n this.data = data;\n }\n}\n\nexport function PaginatedReturnMessageDto<T extends AnyClass>(\n type: T,\n): new (\n statusCode: number,\n message: string,\n data: InstanceType<T>[],\n total: number,\n pageSettings: PageSettingsWise,\n) => GenericPaginatedReturnMessageDto<InstanceType<T>> {\n return InsertField(\n GenericPaginatedReturnMessageDto,\n {\n data: {\n type: [type],\n options: {\n required: false,\n description: 'Return data.',\n },\n },\n },\n `${getClassFromClassOrArray(type).name}PaginatedReturnMessageDto`,\n );\n}\n\nexport class StringReturnMessageDto extends GenericReturnMessageDto<string> {\n @ApiProperty({ description: 'Return data.', type: String, required: false })\n data: string;\n}\n", "import { BlankReturnMessageDto, ReturnMessageDto } from './return-message';\nimport { ApiResponse, ApiResponseOptions } from '@nestjs/swagger';\nimport { ClassOrArray } from './insert-field';\n\nexport const ApiTypeResponse = (\n // eslint-disable-next-line @typescript-eslint/ban-types\n type: ClassOrArray,\n options: ApiResponseOptions = {},\n) =>\n ApiResponse({\n status: 200,\n type: ReturnMessageDto(type),\n ...options,\n });\n\nexport const ApiBlankResponse = (options: ApiResponseOptions = {}) =>\n ApiResponse({\n status: 200,\n type: BlankReturnMessageDto,\n ...options,\n });\n\nexport const ApiError = (status: number, description: string) =>\n ApiBlankResponse({ status, description });\n\nexport const ApiErrorTyped = (\n status: number,\n description: string,\n type: ClassOrArray,\n) =>\n ApiTypeResponse(type, {\n status,\n description,\n });\n", "import {\n Body,\n PipeTransform,\n Query,\n Type,\n ValidationPipe,\n} from '@nestjs/common';\n\nexport const DataPipe = () =>\n new ValidationPipe({\n transform: true,\n transformOptions: { enableImplicitConversion: true },\n });\n\nconst createDataPipeDec =\n (\n nestFieldDec: (\n ...pipes: (Type<PipeTransform> | PipeTransform)[]\n ) => ParameterDecorator,\n ) =>\n (...extraPipes: (Type<PipeTransform> | PipeTransform)[]) =>\n nestFieldDec(DataPipe(), ...extraPipes);\n\nexport const DataQuery = createDataPipeDec(Query);\nexport const DataBody = createDataPipeDec(Body);\n", "import {\n CanActivate,\n ExecutionContext,\n Inject,\n Injectable,\n UseGuards,\n} from '@nestjs/common';\nimport { ConfigService } from '@nestjs/config';\nimport { ApiHeader } from '@nestjs/swagger';\nimport { BlankReturnMessageDto } from './return-message';\nimport { MergeClassOrMethodDecorators } from './merge';\nimport { ApiError } from './openapi';\n\n@Injectable()\nexport class TokenGuard implements CanActivate {\n private token = this.config.get<string>('SERVER_TOKEN');\n constructor(@Inject(ConfigService) private config: ConfigService) {}\n\n async canActivate(context: ExecutionContext) {\n const request = context.switchToHttp().getRequest();\n const token = request.headers['x-server-token'];\n if (this.token && token !== this.token) {\n throw new BlankReturnMessageDto(401, 'Unauthorized').toException();\n }\n return true;\n }\n}\n\nexport const RequireToken = () =>\n MergeClassOrMethodDecorators([\n UseGuards(TokenGuard),\n ApiHeader({\n name: 'x-server-token',\n description: 'Server token',\n required: false,\n }),\n ApiError(401, 'Incorrect server token provided'),\n ]);\n", "import { Observable, takeUntil } from 'rxjs';\nimport { createParamDecorator, ExecutionContext } from '@nestjs/common';\nimport { IncomingMessage } from 'node:http';\nimport { BlankReturnMessageDto } from './return-message';\nimport type { Request, Response } from 'express';\nimport { createAbortSignalFromHttp } from './abort-http-signal';\n\nexport type AbortableFn<T> = (ac: AbortController) => Promise<T>;\n\nexport const fromAbortable = <T>(fn: AbortableFn<T>): Observable<T> => {\n return new Observable<T>((subscriber) => {\n const ac = new AbortController();\n\n fn(ac).then(\n (value) => {\n if (!ac.signal.aborted && !subscriber.closed) {\n subscriber.next(value);\n subscriber.complete();\n }\n },\n (err) => {\n if (ac.signal.aborted) {\n if (!subscriber.closed) subscriber.complete();\n return;\n }\n if (!subscriber.closed) subscriber.error(err);\n },\n );\n\n return () => {\n if (!ac.signal.aborted) ac.abort();\n };\n });\n};\n\nexport const takeUntilAbort = <T>(signal: AbortSignal) => {\n return (source: Observable<T>) => {\n return source.pipe(\n takeUntil(\n new Observable<void>((subscriber) => {\n const onAbort = () => subscriber.next();\n if (signal.aborted) {\n subscriber.next();\n } else {\n signal.addEventListener('abort', onAbort, { once: true });\n }\n return () => signal.removeEventListener('abort', onAbort);\n }),\n ),\n );\n };\n};\n\nexport const As = createParamDecorator(\n (_data: unknown, ctx: ExecutionContext) => {\n const req = ctx.switchToHttp().getRequest<Request>();\n return createAbortSignalFromHttp(req);\n },\n);\n", "// abort-http-signal.ts\nimport type { IncomingMessage } from 'http';\nimport type { Request as ExpressReq } from 'express';\nimport type { FastifyRequest } from 'fastify';\nimport { BlankReturnMessageDto } from './return-message';\n\ntype AnyReq = ExpressReq | FastifyRequest | IncomingMessage;\n\nfunction toRawReq(req: AnyReq): IncomingMessage {\n // Fastify: req.raw \u662F IncomingMessage\uFF1BExpress: req \u672C\u8EAB\u5C31\u662F IncomingMessage\n if ((req as any)?.raw?.socket) return (req as any).raw as IncomingMessage;\n return req as IncomingMessage;\n}\n\n/**\n * \u4EC5\u901A\u8FC7 req \u81EA\u52A8\u9002\u914D Express/Fastify\uFF0C\u7ED1\u5B9A HTTP \u53D6\u6D88\u5230 AbortSignal\u3002\n * - \u4F18\u5148\u76D1\u542C 'aborted'\uFF08\u5BA2\u6237\u7AEF\u4E2D\u65AD\u6700\u53EF\u9760\u7684\u4FE1\u53F7\uFF09\n * - \u53EF\u9009\u515C\u5E95\uFF1A\u5728 socket 'close' \u65F6\uFF0C\u4EC5\u5F53\u8BF7\u6C42\u672A\u5B8C\u6574\u63A5\u6536/\u6216\u5DF2\u6807\u8BB0 aborted \u65F6\uFF0C\u624D\u89E6\u53D1 abort\uFF0C\u907F\u514D\u6B63\u5E38\u5B8C\u6210\u7684\u8BEF\u4F24\n * - reason \u56FA\u5B9A\u4E3A 499 HttpException\n *\n * \u6CE8\u610F\uFF1A\u82E5\u4F60\u7684 abortable() \u5728 signal.reason \u662F Error \u65F6\u4F1A\u201C\u539F\u6837 throw reason\u201D\uFF0C\n * \u90A3\u4E48\u4E0A\u5C42\u4F1A\u6536\u5230 HttpException(499)\u3002\u82E5\u4ECD\u60F3\u7EDF\u4E00\u629B AbortedError\uFF0C\u53EF\u6539\u4E3A\uFF1A\n * const reason = new AbortedError('Request aborted', httpErr);\n * \u5E76\u8BA9 throwIfAborted \u5148\u629B AbortedError\uFF08\u4FDD\u7559 cause\uFF09\u3002\n */\nexport function createAbortSignalFromHttp(req: AnyReq): AbortSignal {\n const rawReq = toRawReq(req);\n const ac = new AbortController();\n\n const makeReason = () =>\n new BlankReturnMessageDto(499, 'Request aborted').toException();\n\n const abortOnce = () => {\n if (!ac.signal.aborted) ac.abort(makeReason());\n cleanup();\n };\n\n const onClose = () => {\n abortOnce();\n };\n\n const cleanup = () => {\n rawReq.off?.('close', onClose);\n };\n\n rawReq.once?.('close', onClose);\n\n return ac.signal;\n}\n", "import { Inject, Scope } from '@nestjs/common';\nimport { abortable, AbortableOpts } from 'nfkit';\nimport { ABORT_SIGNAL } from './abort-signal.provider';\nimport { ContextIdFactory, ModuleRef, REQUEST } from '@nestjs/core';\nimport { createProvider } from '../create-provider';\nimport { InjectionToken } from '@nestjs/common/interfaces/modules/injection-token.interface';\n\nconst tokenMemo = new Map<any, symbol>();\nexport const abortableToken = (token: InjectionToken) => {\n if (tokenMemo.has(token)) return tokenMemo.get(token)!;\n const name = typeof token === 'function' ? token.name : String(token);\n const sym = Symbol.for(`Abortable(${name})`);\n tokenMemo.set(token, sym);\n return sym;\n};\n\n/**\n * \u652F\u6301\u4E24\u79CD\u7528\u6CD5\uFF1A\n * @InjectAbortable(SomeService)\n * @InjectAbortable() // \u81EA\u52A8\u63A8\u65AD\u7C7B\u578B\n */\nexport function InjectAbortable(token?: InjectionToken): ParameterDecorator {\n return (target, propertyKey, parameterIndex) => {\n let actualToken = token;\n\n if (!actualToken) {\n // \u5229\u7528 reflect-metadata \u83B7\u53D6\u53C2\u6570\u7C7B\u578B\n const paramTypes: any[] =\n Reflect.getMetadata('design:paramtypes', target, propertyKey) || [];\n actualToken = paramTypes[parameterIndex];\n if (!actualToken) {\n throw new Error(\n `@InjectAbortable() cannot infer type from metadata: ${target.constructor?.name}[${parameterIndex}]`,\n );\n }\n }\n\n Inject(abortableToken(actualToken))(target, propertyKey, parameterIndex);\n };\n}\n\nexport function createAbortableProvider<T>(\n token: InjectionToken<T>,\n opts?: AbortableOpts,\n) {\n const provide = abortableToken(token);\n\n return createProvider(\n {\n provide,\n scope: Scope.REQUEST,\n inject: [ModuleRef, REQUEST, ABORT_SIGNAL],\n },\n async (moduleRef, req: Request, signal) => {\n // \u8BA9\u89E3\u6790\u4E0E\u5F53\u524D\u8BF7\u6C42\u4E0A\u4E0B\u6587\u7ED1\u5B9A\uFF08\u652F\u6301 request/transient \u4F5C\u7528\u57DF\uFF09\n const ctxId = ContextIdFactory.getByRequest(req);\n // \u4E25\u683C\u6A21\u5F0F\u5173\u95ED\uFF0C\u5141\u8BB8\u8DE8\u6A21\u5757\u8FB9\u754C\u89E3\u6790\uFF08\u89E3\u51B3\u6D4B\u8BD5\u91CC forFeature \u5B50\u6A21\u5757\u770B\u4E0D\u5230 DemoService \u7684\u60C5\u51B5\uFF09\n const svc = await moduleRef.resolve<T>(token, ctxId, { strict: false });\n if (svc == null) {\n throw new Error(\n `Abortable: provider \"${String(\n (token as any).name ?? token,\n )}\" not found in container (even with strict:false)`,\n );\n }\n return abortable<T>(svc, signal, opts);\n },\n );\n}\n", "import { Inject, Scope, Type } from '@nestjs/common';\nimport { REQUEST } from '@nestjs/core';\nimport { createAbortSignalFromHttp } from '../abort-http-signal';\nimport { createProvider } from '../create-provider';\n\nexport const ABORT_SIGNAL = Symbol(\n 'ABORT_SIGNAL',\n) as unknown as Type<AbortSignal>;\nexport const AbortSignalProvider = createProvider(\n {\n provide: ABORT_SIGNAL,\n scope: Scope.REQUEST,\n inject: [REQUEST],\n },\n createAbortSignalFromHttp,\n);\n\nexport const InjectAbortSignal = () => Inject(ABORT_SIGNAL);\n", "import { FactoryProvider } from '@nestjs/common';\nimport { InjectionToken } from '@nestjs/common/interfaces/modules/injection-token.interface';\n\ntype TypeFromToken<T> = T extends string | symbol\n ? any\n : T extends InjectionToken<infer U>\n ? U\n : any;\n\ntype TokensToTypes<A extends InjectionToken[]> = A extends []\n ? []\n : A extends [infer F, ...infer R]\n ? [TypeFromToken<F>, ...TokensToTypes<R extends InjectionToken[] ? R : []>]\n : [];\n\ntype Awaitable<T> = T | Promise<T>;\n\nexport interface TypedFactoryProvider<A extends InjectionToken[], R>\n extends FactoryProvider<R> {\n inject: A;\n useFactory: (...args: TokensToTypes<A>) => Awaitable<R>;\n}\n\nexport const createProvider = <const A extends InjectionToken[], R>(\n options: { inject: A } & Omit<FactoryProvider<R>, 'inject' | 'useFactory'>,\n factory: (...args: TokensToTypes<A>) => Awaitable<R>,\n): TypedFactoryProvider<A, R> => {\n return {\n useFactory: factory,\n ...options,\n };\n};\n", "import { DynamicModule, Module } from '@nestjs/common';\nimport { createAbortableProvider } from './abortable.token';\nimport { AbortableOpts } from 'nfkit';\nimport { AbortSignalProvider } from './abort-signal.provider';\nimport { InjectionToken } from '@nestjs/common/interfaces/modules/injection-token.interface';\n\nexport interface AbortableModuleOptions {\n abortableOptions?: AbortableOpts;\n}\n\n@Module({})\nexport class AbortableModule {\n static forRoot(): DynamicModule {\n return {\n module: AbortableModule,\n providers: [AbortSignalProvider],\n exports: [AbortSignalProvider],\n global: true,\n };\n }\n\n static forFeature(\n tokens: InjectionToken[],\n options?: AbortableModuleOptions,\n ): DynamicModule {\n const providers = tokens.map((token) =>\n createAbortableProvider(token, options?.abortableOptions),\n );\n\n return {\n module: AbortableModule,\n providers: [...providers],\n exports: [...providers],\n };\n }\n}\n"],
|
|
5
|
+
"mappings": "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",
|
|
6
6
|
"names": ["import_swagger", "import_swagger", "import_common", "import_common", "import_swagger", "import_common", "import_common", "import_common", "import_core", "import_common"]
|
|
7
7
|
}
|
package/dist/index.d.ts
CHANGED
package/dist/index.mjs
CHANGED
|
@@ -8,6 +8,7 @@ var __decorateClass = (decorators, target, key, kind) => {
|
|
|
8
8
|
if (kind && result) __defProp(target, key, result);
|
|
9
9
|
return result;
|
|
10
10
|
};
|
|
11
|
+
var __decorateParam = (index, decorator) => (target, key) => decorator(target, key, index);
|
|
11
12
|
|
|
12
13
|
// src/insert-field.ts
|
|
13
14
|
import { ApiProperty } from "@nestjs/swagger";
|
|
@@ -201,9 +202,11 @@ var DataBody = createDataPipeDec(Body);
|
|
|
201
202
|
|
|
202
203
|
// src/token.guard.ts
|
|
203
204
|
import {
|
|
205
|
+
Inject,
|
|
204
206
|
Injectable,
|
|
205
207
|
UseGuards
|
|
206
208
|
} from "@nestjs/common";
|
|
209
|
+
import { ConfigService } from "@nestjs/config";
|
|
207
210
|
import { ApiHeader } from "@nestjs/swagger";
|
|
208
211
|
var TokenGuard = class {
|
|
209
212
|
constructor(config) {
|
|
@@ -220,7 +223,8 @@ var TokenGuard = class {
|
|
|
220
223
|
}
|
|
221
224
|
};
|
|
222
225
|
TokenGuard = __decorateClass([
|
|
223
|
-
Injectable()
|
|
226
|
+
Injectable(),
|
|
227
|
+
__decorateParam(0, Inject(ConfigService))
|
|
224
228
|
], TokenGuard);
|
|
225
229
|
var RequireToken = () => MergeClassOrMethodDecorators([
|
|
226
230
|
UseGuards(TokenGuard),
|
|
@@ -308,22 +312,34 @@ var As = createParamDecorator(
|
|
|
308
312
|
);
|
|
309
313
|
|
|
310
314
|
// src/abortable-module/abortable.token.ts
|
|
311
|
-
import { Inject as
|
|
315
|
+
import { Inject as Inject3, Scope as Scope2 } from "@nestjs/common";
|
|
312
316
|
import { abortable } from "nfkit";
|
|
313
317
|
|
|
314
318
|
// src/abortable-module/abort-signal.provider.ts
|
|
315
|
-
import { Inject, Scope } from "@nestjs/common";
|
|
319
|
+
import { Inject as Inject2, Scope } from "@nestjs/common";
|
|
316
320
|
import { REQUEST } from "@nestjs/core";
|
|
317
|
-
|
|
318
|
-
|
|
319
|
-
|
|
320
|
-
|
|
321
|
-
|
|
322
|
-
|
|
323
|
-
|
|
324
|
-
}
|
|
321
|
+
|
|
322
|
+
// src/create-provider.ts
|
|
323
|
+
var createProvider = (options, factory) => {
|
|
324
|
+
return {
|
|
325
|
+
useFactory: factory,
|
|
326
|
+
...options
|
|
327
|
+
};
|
|
325
328
|
};
|
|
326
|
-
|
|
329
|
+
|
|
330
|
+
// src/abortable-module/abort-signal.provider.ts
|
|
331
|
+
var ABORT_SIGNAL = Symbol(
|
|
332
|
+
"ABORT_SIGNAL"
|
|
333
|
+
);
|
|
334
|
+
var AbortSignalProvider = createProvider(
|
|
335
|
+
{
|
|
336
|
+
provide: ABORT_SIGNAL,
|
|
337
|
+
scope: Scope.REQUEST,
|
|
338
|
+
inject: [REQUEST]
|
|
339
|
+
},
|
|
340
|
+
createAbortSignalFromHttp
|
|
341
|
+
);
|
|
342
|
+
var InjectAbortSignal = () => Inject2(ABORT_SIGNAL);
|
|
327
343
|
|
|
328
344
|
// src/abortable-module/abortable.token.ts
|
|
329
345
|
import { ContextIdFactory, ModuleRef, REQUEST as REQUEST2 } from "@nestjs/core";
|
|
@@ -347,17 +363,18 @@ function InjectAbortable(token) {
|
|
|
347
363
|
);
|
|
348
364
|
}
|
|
349
365
|
}
|
|
350
|
-
|
|
366
|
+
Inject3(abortableToken(actualToken))(target, propertyKey, parameterIndex);
|
|
351
367
|
};
|
|
352
368
|
}
|
|
353
369
|
function createAbortableProvider(token, opts) {
|
|
354
370
|
const provide = abortableToken(token);
|
|
355
|
-
return
|
|
356
|
-
|
|
357
|
-
|
|
358
|
-
|
|
359
|
-
|
|
360
|
-
|
|
371
|
+
return createProvider(
|
|
372
|
+
{
|
|
373
|
+
provide,
|
|
374
|
+
scope: Scope2.REQUEST,
|
|
375
|
+
inject: [ModuleRef, REQUEST2, ABORT_SIGNAL]
|
|
376
|
+
},
|
|
377
|
+
async (moduleRef, req, signal) => {
|
|
361
378
|
const ctxId = ContextIdFactory.getByRequest(req);
|
|
362
379
|
const svc = await moduleRef.resolve(token, ctxId, { strict: false });
|
|
363
380
|
if (svc == null) {
|
|
@@ -369,7 +386,7 @@ function createAbortableProvider(token, opts) {
|
|
|
369
386
|
}
|
|
370
387
|
return abortable(svc, signal, opts);
|
|
371
388
|
}
|
|
372
|
-
|
|
389
|
+
);
|
|
373
390
|
}
|
|
374
391
|
|
|
375
392
|
// src/abortable-module/abortable.module.ts
|
|
@@ -428,6 +445,7 @@ export {
|
|
|
428
445
|
TokenGuard,
|
|
429
446
|
abortableToken,
|
|
430
447
|
createAbortableProvider,
|
|
448
|
+
createProvider,
|
|
431
449
|
fromAbortable,
|
|
432
450
|
getClassFromClassOrArray,
|
|
433
451
|
takeUntilAbort
|
package/dist/index.mjs.map
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
|
-
"sources": ["../src/insert-field.ts", "../src/merge.ts", "../src/return-message.ts", "../src/openapi.ts", "../src/pipe.ts", "../src/token.guard.ts", "../src/abort-utils.ts", "../src/abort-http-signal.ts", "../src/abortable-module/abortable.token.ts", "../src/abortable-module/abort-signal.provider.ts", "../src/abortable-module/abortable.module.ts"],
|
|
4
|
-
"sourcesContent": ["import { ApiProperty, ApiPropertyOptions } from '@nestjs/swagger';\n\nexport type AnyClass = new (...args: any[]) => any;\nexport type ClassOrArray = AnyClass | [AnyClass];\nexport type ClassType<T> = new (...args: any[]) => T;\nexport type TypeFromClass<T> = T extends new (...args: any[]) => infer U\n ? U\n : never;\nexport type ParamsFromClass<T> = T extends new (...args: infer U) => any\n ? U\n : never;\nexport type ParseType<IC extends ClassOrArray> = IC extends [infer U]\n ? TypeFromClass<U>[]\n : TypeFromClass<IC>;\n\nexport function getClassFromClassOrArray(o: ClassOrArray) {\n return o instanceof Array ? o[0] : o;\n}\n\nexport interface InsertOptions<C extends ClassOrArray = ClassOrArray> {\n type: C;\n options?: ApiPropertyOptions;\n}\n\ntype TypeFromInsertOptions<O extends InsertOptions> =\n O extends InsertOptions<infer C>\n ?\n | ParseType<C>\n | (O extends { options: { required: true } } ? never : undefined)\n : never;\n\ntype Merge<T, U> = {\n [K in keyof T | keyof U]: K extends keyof T\n ? T[K]\n : K extends keyof U\n ? U[K]\n : never;\n};\n\nexport function InsertField<\n C extends AnyClass,\n M extends Record<string, InsertOptions>,\n>(\n cl: C,\n map: M,\n newName?: string,\n): new (...args: ParamsFromClass<C>) => Merge<\n {\n [F in keyof M]: TypeFromInsertOptions<M[F]>;\n },\n TypeFromClass<C>\n> {\n const extendedCl = class extends cl {};\n for (const key in map) {\n ApiProperty({\n type: map[key].type,\n ...(map[key].options || {}),\n })(extendedCl.prototype, key);\n }\n Object.defineProperty(extendedCl, 'name', {\n value: newName || cl.name,\n });\n return extendedCl;\n}\n", "export function MergePropertyDecorators(\n decs: PropertyDecorator[],\n): PropertyDecorator {\n return (obj, key) => {\n for (const dec of decs) {\n dec(obj, key);\n }\n };\n}\n\nexport function MergeMethodDecorators(\n decs: MethodDecorator[],\n): MethodDecorator {\n return (obj, key, descriptor) => {\n for (const dec of decs) {\n dec(obj, key, descriptor);\n }\n };\n}\n\nexport function MergeClassDecorators(decs: ClassDecorator[]): ClassDecorator {\n return (obj) => {\n for (const dec of decs) {\n dec(obj);\n }\n };\n}\n\nexport function MergeParameterDecorators(\n decs: ParameterDecorator[],\n): ParameterDecorator {\n return (obj, key, index) => {\n for (const dec of decs) {\n dec(obj, key, index);\n }\n };\n}\n\nexport function MergeClassOrMethodDecorators(\n decs: (ClassDecorator & MethodDecorator)[],\n): ClassDecorator & MethodDecorator {\n return (obj, key?, descriptor?) => {\n if (descriptor) {\n for (const dec of decs) {\n dec(obj, key, descriptor);\n }\n } else {\n for (const dec of decs) {\n dec(obj);\n }\n }\n };\n}\n", "import { ApiProperty } from '@nestjs/swagger';\nimport { HttpException } from '@nestjs/common';\nimport {\n AnyClass,\n ClassOrArray,\n getClassFromClassOrArray,\n InsertField,\n ParseType,\n} from './insert-field';\n\nexport interface PageSettingsWise {\n pageCount: number;\n recordsPerPage: number;\n}\n\nexport interface BlankReturnMessage {\n statusCode: number;\n message: string;\n success: boolean;\n timestamp: Date;\n}\n\nexport interface ReturnMessage<T> extends BlankReturnMessage {\n data?: T;\n}\n\nexport class BlankReturnMessageDto implements BlankReturnMessage {\n @ApiProperty({ description: 'Return code', type: Number })\n statusCode: number;\n @ApiProperty({ description: 'Return message', type: String })\n message: string;\n @ApiProperty({ description: 'Whether success.', type: Boolean })\n success: boolean;\n @ApiProperty({ description: 'Return timestamp', type: Date })\n timestamp: Date;\n constructor(statusCode: number, message?: string) {\n this.statusCode = statusCode;\n this.message = message || 'success';\n this.success = statusCode < 400;\n this.timestamp = new Date();\n }\n\n toException() {\n return new HttpException(this, this.statusCode);\n }\n}\n\nexport class BlankPaginatedReturnMessageDto\n extends BlankReturnMessageDto\n implements PageSettingsWise\n{\n @ApiProperty({ description: 'Total record count.', type: Number })\n total: number;\n @ApiProperty({ description: 'Total page count.', type: Number })\n totalPages: number;\n @ApiProperty({ description: 'Current page.', type: Number })\n pageCount: number;\n @ApiProperty({ description: 'Records per page.', type: Number })\n recordsPerPage: number;\n constructor(\n statusCode: number,\n message: string,\n total: number,\n pageSettings: PageSettingsWise,\n ) {\n super(statusCode, message);\n this.total = total;\n this.pageCount = pageSettings.pageCount;\n this.recordsPerPage = pageSettings.recordsPerPage;\n this.totalPages = Math.ceil(total / pageSettings.recordsPerPage);\n }\n}\n\nexport class GenericReturnMessageDto<T>\n extends BlankReturnMessageDto\n implements ReturnMessage<T>\n{\n data?: T;\n constructor(statusCode: number, message?: string, data?: T) {\n super(statusCode, message);\n this.data = data;\n }\n}\n\nexport function ReturnMessageDto<T extends ClassOrArray>(\n type: T,\n): new (\n statusCode: number,\n message: string,\n data: ParseType<T>,\n) => GenericReturnMessageDto<ParseType<T>> {\n return InsertField(\n GenericReturnMessageDto,\n {\n data: {\n type,\n options: {\n required: false,\n description: 'Return data.',\n },\n },\n },\n `${getClassFromClassOrArray(type).name}${Array.isArray(type) ? 'Array' : ''}ReturnMessageDto`,\n );\n}\n\nexport class GenericPaginatedReturnMessageDto<T>\n extends BlankPaginatedReturnMessageDto\n implements PageSettingsWise, ReturnMessage<T[]>\n{\n data: T[];\n constructor(\n statusCode: number,\n message: string,\n data: T[],\n total: number,\n pageSettings: PageSettingsWise,\n ) {\n super(statusCode, message, total, pageSettings);\n this.data = data;\n }\n}\n\nexport function PaginatedReturnMessageDto<T extends AnyClass>(\n type: T,\n): new (\n statusCode: number,\n message: string,\n data: InstanceType<T>[],\n total: number,\n pageSettings: PageSettingsWise,\n) => GenericPaginatedReturnMessageDto<InstanceType<T>> {\n return InsertField(\n GenericPaginatedReturnMessageDto,\n {\n data: {\n type: [type],\n options: {\n required: false,\n description: 'Return data.',\n },\n },\n },\n `${getClassFromClassOrArray(type).name}PaginatedReturnMessageDto`,\n );\n}\n\nexport class StringReturnMessageDto extends GenericReturnMessageDto<string> {\n @ApiProperty({ description: 'Return data.', type: String, required: false })\n data: string;\n}\n", "import { BlankReturnMessageDto, ReturnMessageDto } from './return-message';\nimport { ApiResponse, ApiResponseOptions } from '@nestjs/swagger';\nimport { ClassOrArray } from './insert-field';\n\nexport const ApiTypeResponse = (\n // eslint-disable-next-line @typescript-eslint/ban-types\n type: ClassOrArray,\n options: ApiResponseOptions = {},\n) =>\n ApiResponse({\n status: 200,\n type: ReturnMessageDto(type),\n ...options,\n });\n\nexport const ApiBlankResponse = (options: ApiResponseOptions = {}) =>\n ApiResponse({\n status: 200,\n type: BlankReturnMessageDto,\n ...options,\n });\n\nexport const ApiError = (status: number, description: string) =>\n ApiBlankResponse({ status, description });\n\nexport const ApiErrorTyped = (\n status: number,\n description: string,\n type: ClassOrArray,\n) =>\n ApiTypeResponse(type, {\n status,\n description,\n });\n", "import {\n Body,\n PipeTransform,\n Query,\n Type,\n ValidationPipe,\n} from '@nestjs/common';\n\nexport const DataPipe = () =>\n new ValidationPipe({\n transform: true,\n transformOptions: { enableImplicitConversion: true },\n });\n\nconst createDataPipeDec =\n (\n nestFieldDec: (\n ...pipes: (Type<PipeTransform> | PipeTransform)[]\n ) => ParameterDecorator,\n ) =>\n (...extraPipes: (Type<PipeTransform> | PipeTransform)[]) =>\n nestFieldDec(DataPipe(), ...extraPipes);\n\nexport const DataQuery = createDataPipeDec(Query);\nexport const DataBody = createDataPipeDec(Body);\n", "import {\n CanActivate,\n ExecutionContext,\n Injectable,\n UseGuards,\n} from '@nestjs/common';\nimport { ConfigService } from '@nestjs/config';\nimport { ApiHeader } from '@nestjs/swagger';\nimport { BlankReturnMessageDto } from './return-message';\nimport { MergeClassOrMethodDecorators } from './merge';\nimport { ApiError } from './openapi';\n\n@Injectable()\nexport class TokenGuard implements CanActivate {\n private token = this.config.get<string>('SERVER_TOKEN');\n constructor(private config: ConfigService) {}\n\n async canActivate(context: ExecutionContext) {\n const request = context.switchToHttp().getRequest();\n const token = request.headers['x-server-token'];\n if (this.token && token !== this.token) {\n throw new BlankReturnMessageDto(401, 'Unauthorized').toException();\n }\n return true;\n }\n}\n\nexport const RequireToken = () =>\n MergeClassOrMethodDecorators([\n UseGuards(TokenGuard),\n ApiHeader({\n name: 'x-server-token',\n description: 'Server token',\n required: false,\n }),\n ApiError(401, 'Incorrect server token provided'),\n ]);\n", "import { Observable, takeUntil } from 'rxjs';\nimport { createParamDecorator, ExecutionContext } from '@nestjs/common';\nimport { IncomingMessage } from 'node:http';\nimport { BlankReturnMessageDto } from './return-message';\nimport type { Request, Response } from 'express';\nimport { createAbortSignalFromHttp } from './abort-http-signal';\n\nexport type AbortableFn<T> = (ac: AbortController) => Promise<T>;\n\nexport const fromAbortable = <T>(fn: AbortableFn<T>): Observable<T> => {\n return new Observable<T>((subscriber) => {\n const ac = new AbortController();\n\n fn(ac).then(\n (value) => {\n if (!ac.signal.aborted && !subscriber.closed) {\n subscriber.next(value);\n subscriber.complete();\n }\n },\n (err) => {\n if (ac.signal.aborted) {\n if (!subscriber.closed) subscriber.complete();\n return;\n }\n if (!subscriber.closed) subscriber.error(err);\n },\n );\n\n return () => {\n if (!ac.signal.aborted) ac.abort();\n };\n });\n};\n\nexport const takeUntilAbort = <T>(signal: AbortSignal) => {\n return (source: Observable<T>) => {\n return source.pipe(\n takeUntil(\n new Observable<void>((subscriber) => {\n const onAbort = () => subscriber.next();\n if (signal.aborted) {\n subscriber.next();\n } else {\n signal.addEventListener('abort', onAbort, { once: true });\n }\n return () => signal.removeEventListener('abort', onAbort);\n }),\n ),\n );\n };\n};\n\nexport const As = createParamDecorator(\n (_data: unknown, ctx: ExecutionContext) => {\n const req = ctx.switchToHttp().getRequest<Request>();\n return createAbortSignalFromHttp(req);\n },\n);\n", "// abort-http-signal.ts\nimport type { IncomingMessage } from 'http';\nimport type { Request as ExpressReq } from 'express';\nimport type { FastifyRequest } from 'fastify';\nimport { BlankReturnMessageDto } from './return-message';\n\ntype AnyReq = ExpressReq | FastifyRequest | IncomingMessage;\n\nfunction toRawReq(req: AnyReq): IncomingMessage {\n // Fastify: req.raw \u662F IncomingMessage\uFF1BExpress: req \u672C\u8EAB\u5C31\u662F IncomingMessage\n if ((req as any)?.raw?.socket) return (req as any).raw as IncomingMessage;\n return req as IncomingMessage;\n}\n\n/**\n * \u4EC5\u901A\u8FC7 req \u81EA\u52A8\u9002\u914D Express/Fastify\uFF0C\u7ED1\u5B9A HTTP \u53D6\u6D88\u5230 AbortSignal\u3002\n * - \u4F18\u5148\u76D1\u542C 'aborted'\uFF08\u5BA2\u6237\u7AEF\u4E2D\u65AD\u6700\u53EF\u9760\u7684\u4FE1\u53F7\uFF09\n * - \u53EF\u9009\u515C\u5E95\uFF1A\u5728 socket 'close' \u65F6\uFF0C\u4EC5\u5F53\u8BF7\u6C42\u672A\u5B8C\u6574\u63A5\u6536/\u6216\u5DF2\u6807\u8BB0 aborted \u65F6\uFF0C\u624D\u89E6\u53D1 abort\uFF0C\u907F\u514D\u6B63\u5E38\u5B8C\u6210\u7684\u8BEF\u4F24\n * - reason \u56FA\u5B9A\u4E3A 499 HttpException\n *\n * \u6CE8\u610F\uFF1A\u82E5\u4F60\u7684 abortable() \u5728 signal.reason \u662F Error \u65F6\u4F1A\u201C\u539F\u6837 throw reason\u201D\uFF0C\n * \u90A3\u4E48\u4E0A\u5C42\u4F1A\u6536\u5230 HttpException(499)\u3002\u82E5\u4ECD\u60F3\u7EDF\u4E00\u629B AbortedError\uFF0C\u53EF\u6539\u4E3A\uFF1A\n * const reason = new AbortedError('Request aborted', httpErr);\n * \u5E76\u8BA9 throwIfAborted \u5148\u629B AbortedError\uFF08\u4FDD\u7559 cause\uFF09\u3002\n */\nexport function createAbortSignalFromHttp(req: AnyReq): AbortSignal {\n const rawReq = toRawReq(req);\n const ac = new AbortController();\n\n const makeReason = () =>\n new BlankReturnMessageDto(499, 'Request aborted').toException();\n\n const abortOnce = () => {\n if (!ac.signal.aborted) ac.abort(makeReason());\n cleanup();\n };\n\n const onClose = () => {\n abortOnce();\n };\n\n const cleanup = () => {\n rawReq.off?.('close', onClose);\n };\n\n rawReq.once?.('close', onClose);\n\n return ac.signal;\n}\n", "import { Inject, Provider, Scope } from '@nestjs/common';\nimport { abortable, AbortableOpts } from 'nfkit';\nimport { ABORT_SIGNAL } from './abort-signal.provider';\nimport { ContextIdFactory, ModuleRef, REQUEST } from '@nestjs/core';\n\nexport type ProviderToken<T = any> =\n | string\n | symbol\n | (new (...args: any[]) => T);\n\nconst tokenMemo = new Map<any, symbol>();\nexport const abortableToken = (token: ProviderToken) => {\n if (tokenMemo.has(token)) return tokenMemo.get(token)!;\n const name = typeof token === 'function' ? token.name : String(token);\n const sym = Symbol.for(`Abortable(${name})`);\n tokenMemo.set(token, sym);\n return sym;\n};\n\n/**\n * \u652F\u6301\u4E24\u79CD\u7528\u6CD5\uFF1A\n * @InjectAbortable(SomeService)\n * @InjectAbortable() // \u81EA\u52A8\u63A8\u65AD\u7C7B\u578B\n */\nexport function InjectAbortable(token?: ProviderToken): ParameterDecorator {\n return (target, propertyKey, parameterIndex) => {\n let actualToken = token;\n\n if (!actualToken) {\n // \u5229\u7528 reflect-metadata \u83B7\u53D6\u53C2\u6570\u7C7B\u578B\n const paramTypes: any[] =\n Reflect.getMetadata('design:paramtypes', target, propertyKey) || [];\n actualToken = paramTypes[parameterIndex];\n if (!actualToken) {\n throw new Error(\n `@InjectAbortable() cannot infer type from metadata: ${target.constructor?.name}[${parameterIndex}]`,\n );\n }\n }\n\n Inject(abortableToken(actualToken))(target, propertyKey, parameterIndex);\n };\n}\n\nexport function createAbortableProvider<T>(\n token: ProviderToken<T>,\n opts?: AbortableOpts,\n): Provider {\n const provide = abortableToken(token);\n\n return {\n provide,\n scope: Scope.REQUEST,\n inject: [ModuleRef, REQUEST, ABORT_SIGNAL],\n // \u26A0\uFE0F \u6CE8\u610F\uFF1A\u7528 async + resolve + contextId + strict:false\n useFactory: async (\n moduleRef: ModuleRef,\n req: Request,\n signal: AbortSignal,\n ) => {\n // \u8BA9\u89E3\u6790\u4E0E\u5F53\u524D\u8BF7\u6C42\u4E0A\u4E0B\u6587\u7ED1\u5B9A\uFF08\u652F\u6301 request/transient \u4F5C\u7528\u57DF\uFF09\n const ctxId = ContextIdFactory.getByRequest(req);\n // \u4E25\u683C\u6A21\u5F0F\u5173\u95ED\uFF0C\u5141\u8BB8\u8DE8\u6A21\u5757\u8FB9\u754C\u89E3\u6790\uFF08\u89E3\u51B3\u6D4B\u8BD5\u91CC forFeature \u5B50\u6A21\u5757\u770B\u4E0D\u5230 DemoService \u7684\u60C5\u51B5\uFF09\n const svc = await moduleRef.resolve<T>(token, ctxId, { strict: false });\n if (svc == null) {\n throw new Error(\n `Abortable: provider \"${String(\n (token as any).name ?? token,\n )}\" not found in container (even with strict:false)`,\n );\n }\n return abortable<T>(svc, signal, opts);\n },\n };\n}\n", "import { Inject, Provider, Scope } from '@nestjs/common';\nimport { REQUEST } from '@nestjs/core';\nimport type { Request as ExpressReq } from 'express';\nimport { createAbortSignalFromHttp } from '../abort-http-signal';\n\nexport const ABORT_SIGNAL = Symbol('ABORT_SIGNAL');\nexport const AbortSignalProvider: Provider = {\n provide: ABORT_SIGNAL,\n scope: Scope.REQUEST,\n inject: [REQUEST],\n useFactory: (req: ExpressReq) => {\n return createAbortSignalFromHttp(req);\n },\n};\n\nexport const InjectAbortSignal = () => Inject(ABORT_SIGNAL);\n", "import { DynamicModule, Module, Provider } from '@nestjs/common';\nimport { ProviderToken, createAbortableProvider } from './abortable.token';\nimport { AbortableOpts } from 'nfkit';\nimport { AbortSignalProvider } from './abort-signal.provider';\n\nexport interface AbortableModuleOptions {\n abortableOptions?: AbortableOpts;\n}\n\n@Module({})\nexport class AbortableModule {\n static forRoot(): DynamicModule {\n return {\n module: AbortableModule,\n providers: [AbortSignalProvider],\n exports: [AbortSignalProvider],\n global: true,\n };\n }\n\n static forFeature(\n tokens: ProviderToken[],\n options?: AbortableModuleOptions,\n ): DynamicModule {\n const providers: Provider[] = tokens.map((token) =>\n createAbortableProvider(token, options?.abortableOptions),\n );\n\n return {\n module: AbortableModule,\n providers: [...providers],\n exports: [...providers],\n };\n }\n}\n"],
|
|
5
|
-
"mappings": "
|
|
6
|
-
"names": ["ApiProperty", "ApiProperty", "Inject", "Scope", "REQUEST", "Inject", "Scope"]
|
|
3
|
+
"sources": ["../src/insert-field.ts", "../src/merge.ts", "../src/return-message.ts", "../src/openapi.ts", "../src/pipe.ts", "../src/token.guard.ts", "../src/abort-utils.ts", "../src/abort-http-signal.ts", "../src/abortable-module/abortable.token.ts", "../src/abortable-module/abort-signal.provider.ts", "../src/create-provider.ts", "../src/abortable-module/abortable.module.ts"],
|
|
4
|
+
"sourcesContent": ["import { ApiProperty, ApiPropertyOptions } from '@nestjs/swagger';\n\nexport type AnyClass = new (...args: any[]) => any;\nexport type ClassOrArray = AnyClass | [AnyClass];\nexport type ClassType<T> = new (...args: any[]) => T;\nexport type TypeFromClass<T> = T extends new (...args: any[]) => infer U\n ? U\n : never;\nexport type ParamsFromClass<T> = T extends new (...args: infer U) => any\n ? U\n : never;\nexport type ParseType<IC extends ClassOrArray> = IC extends [infer U]\n ? TypeFromClass<U>[]\n : TypeFromClass<IC>;\n\nexport function getClassFromClassOrArray(o: ClassOrArray) {\n return o instanceof Array ? o[0] : o;\n}\n\nexport interface InsertOptions<C extends ClassOrArray = ClassOrArray> {\n type: C;\n options?: ApiPropertyOptions;\n}\n\ntype TypeFromInsertOptions<O extends InsertOptions> =\n O extends InsertOptions<infer C>\n ?\n | ParseType<C>\n | (O extends { options: { required: true } } ? never : undefined)\n : never;\n\ntype Merge<T, U> = {\n [K in keyof T | keyof U]: K extends keyof T\n ? T[K]\n : K extends keyof U\n ? U[K]\n : never;\n};\n\nexport function InsertField<\n C extends AnyClass,\n M extends Record<string, InsertOptions>,\n>(\n cl: C,\n map: M,\n newName?: string,\n): new (...args: ParamsFromClass<C>) => Merge<\n {\n [F in keyof M]: TypeFromInsertOptions<M[F]>;\n },\n TypeFromClass<C>\n> {\n const extendedCl = class extends cl {};\n for (const key in map) {\n ApiProperty({\n type: map[key].type,\n ...(map[key].options || {}),\n })(extendedCl.prototype, key);\n }\n Object.defineProperty(extendedCl, 'name', {\n value: newName || cl.name,\n });\n return extendedCl;\n}\n", "export function MergePropertyDecorators(\n decs: PropertyDecorator[],\n): PropertyDecorator {\n return (obj, key) => {\n for (const dec of decs) {\n dec(obj, key);\n }\n };\n}\n\nexport function MergeMethodDecorators(\n decs: MethodDecorator[],\n): MethodDecorator {\n return (obj, key, descriptor) => {\n for (const dec of decs) {\n dec(obj, key, descriptor);\n }\n };\n}\n\nexport function MergeClassDecorators(decs: ClassDecorator[]): ClassDecorator {\n return (obj) => {\n for (const dec of decs) {\n dec(obj);\n }\n };\n}\n\nexport function MergeParameterDecorators(\n decs: ParameterDecorator[],\n): ParameterDecorator {\n return (obj, key, index) => {\n for (const dec of decs) {\n dec(obj, key, index);\n }\n };\n}\n\nexport function MergeClassOrMethodDecorators(\n decs: (ClassDecorator & MethodDecorator)[],\n): ClassDecorator & MethodDecorator {\n return (obj, key?, descriptor?) => {\n if (descriptor) {\n for (const dec of decs) {\n dec(obj, key, descriptor);\n }\n } else {\n for (const dec of decs) {\n dec(obj);\n }\n }\n };\n}\n", "import { ApiProperty } from '@nestjs/swagger';\nimport { HttpException } from '@nestjs/common';\nimport {\n AnyClass,\n ClassOrArray,\n getClassFromClassOrArray,\n InsertField,\n ParseType,\n} from './insert-field';\n\nexport interface PageSettingsWise {\n pageCount: number;\n recordsPerPage: number;\n}\n\nexport interface BlankReturnMessage {\n statusCode: number;\n message: string;\n success: boolean;\n timestamp: Date;\n}\n\nexport interface ReturnMessage<T> extends BlankReturnMessage {\n data?: T;\n}\n\nexport class BlankReturnMessageDto implements BlankReturnMessage {\n @ApiProperty({ description: 'Return code', type: Number })\n statusCode: number;\n @ApiProperty({ description: 'Return message', type: String })\n message: string;\n @ApiProperty({ description: 'Whether success.', type: Boolean })\n success: boolean;\n @ApiProperty({ description: 'Return timestamp', type: Date })\n timestamp: Date;\n constructor(statusCode: number, message?: string) {\n this.statusCode = statusCode;\n this.message = message || 'success';\n this.success = statusCode < 400;\n this.timestamp = new Date();\n }\n\n toException() {\n return new HttpException(this, this.statusCode);\n }\n}\n\nexport class BlankPaginatedReturnMessageDto\n extends BlankReturnMessageDto\n implements PageSettingsWise\n{\n @ApiProperty({ description: 'Total record count.', type: Number })\n total: number;\n @ApiProperty({ description: 'Total page count.', type: Number })\n totalPages: number;\n @ApiProperty({ description: 'Current page.', type: Number })\n pageCount: number;\n @ApiProperty({ description: 'Records per page.', type: Number })\n recordsPerPage: number;\n constructor(\n statusCode: number,\n message: string,\n total: number,\n pageSettings: PageSettingsWise,\n ) {\n super(statusCode, message);\n this.total = total;\n this.pageCount = pageSettings.pageCount;\n this.recordsPerPage = pageSettings.recordsPerPage;\n this.totalPages = Math.ceil(total / pageSettings.recordsPerPage);\n }\n}\n\nexport class GenericReturnMessageDto<T>\n extends BlankReturnMessageDto\n implements ReturnMessage<T>\n{\n data?: T;\n constructor(statusCode: number, message?: string, data?: T) {\n super(statusCode, message);\n this.data = data;\n }\n}\n\nexport function ReturnMessageDto<T extends ClassOrArray>(\n type: T,\n): new (\n statusCode: number,\n message: string,\n data: ParseType<T>,\n) => GenericReturnMessageDto<ParseType<T>> {\n return InsertField(\n GenericReturnMessageDto,\n {\n data: {\n type,\n options: {\n required: false,\n description: 'Return data.',\n },\n },\n },\n `${getClassFromClassOrArray(type).name}${Array.isArray(type) ? 'Array' : ''}ReturnMessageDto`,\n );\n}\n\nexport class GenericPaginatedReturnMessageDto<T>\n extends BlankPaginatedReturnMessageDto\n implements PageSettingsWise, ReturnMessage<T[]>\n{\n data: T[];\n constructor(\n statusCode: number,\n message: string,\n data: T[],\n total: number,\n pageSettings: PageSettingsWise,\n ) {\n super(statusCode, message, total, pageSettings);\n this.data = data;\n }\n}\n\nexport function PaginatedReturnMessageDto<T extends AnyClass>(\n type: T,\n): new (\n statusCode: number,\n message: string,\n data: InstanceType<T>[],\n total: number,\n pageSettings: PageSettingsWise,\n) => GenericPaginatedReturnMessageDto<InstanceType<T>> {\n return InsertField(\n GenericPaginatedReturnMessageDto,\n {\n data: {\n type: [type],\n options: {\n required: false,\n description: 'Return data.',\n },\n },\n },\n `${getClassFromClassOrArray(type).name}PaginatedReturnMessageDto`,\n );\n}\n\nexport class StringReturnMessageDto extends GenericReturnMessageDto<string> {\n @ApiProperty({ description: 'Return data.', type: String, required: false })\n data: string;\n}\n", "import { BlankReturnMessageDto, ReturnMessageDto } from './return-message';\nimport { ApiResponse, ApiResponseOptions } from '@nestjs/swagger';\nimport { ClassOrArray } from './insert-field';\n\nexport const ApiTypeResponse = (\n // eslint-disable-next-line @typescript-eslint/ban-types\n type: ClassOrArray,\n options: ApiResponseOptions = {},\n) =>\n ApiResponse({\n status: 200,\n type: ReturnMessageDto(type),\n ...options,\n });\n\nexport const ApiBlankResponse = (options: ApiResponseOptions = {}) =>\n ApiResponse({\n status: 200,\n type: BlankReturnMessageDto,\n ...options,\n });\n\nexport const ApiError = (status: number, description: string) =>\n ApiBlankResponse({ status, description });\n\nexport const ApiErrorTyped = (\n status: number,\n description: string,\n type: ClassOrArray,\n) =>\n ApiTypeResponse(type, {\n status,\n description,\n });\n", "import {\n Body,\n PipeTransform,\n Query,\n Type,\n ValidationPipe,\n} from '@nestjs/common';\n\nexport const DataPipe = () =>\n new ValidationPipe({\n transform: true,\n transformOptions: { enableImplicitConversion: true },\n });\n\nconst createDataPipeDec =\n (\n nestFieldDec: (\n ...pipes: (Type<PipeTransform> | PipeTransform)[]\n ) => ParameterDecorator,\n ) =>\n (...extraPipes: (Type<PipeTransform> | PipeTransform)[]) =>\n nestFieldDec(DataPipe(), ...extraPipes);\n\nexport const DataQuery = createDataPipeDec(Query);\nexport const DataBody = createDataPipeDec(Body);\n", "import {\n CanActivate,\n ExecutionContext,\n Inject,\n Injectable,\n UseGuards,\n} from '@nestjs/common';\nimport { ConfigService } from '@nestjs/config';\nimport { ApiHeader } from '@nestjs/swagger';\nimport { BlankReturnMessageDto } from './return-message';\nimport { MergeClassOrMethodDecorators } from './merge';\nimport { ApiError } from './openapi';\n\n@Injectable()\nexport class TokenGuard implements CanActivate {\n private token = this.config.get<string>('SERVER_TOKEN');\n constructor(@Inject(ConfigService) private config: ConfigService) {}\n\n async canActivate(context: ExecutionContext) {\n const request = context.switchToHttp().getRequest();\n const token = request.headers['x-server-token'];\n if (this.token && token !== this.token) {\n throw new BlankReturnMessageDto(401, 'Unauthorized').toException();\n }\n return true;\n }\n}\n\nexport const RequireToken = () =>\n MergeClassOrMethodDecorators([\n UseGuards(TokenGuard),\n ApiHeader({\n name: 'x-server-token',\n description: 'Server token',\n required: false,\n }),\n ApiError(401, 'Incorrect server token provided'),\n ]);\n", "import { Observable, takeUntil } from 'rxjs';\nimport { createParamDecorator, ExecutionContext } from '@nestjs/common';\nimport { IncomingMessage } from 'node:http';\nimport { BlankReturnMessageDto } from './return-message';\nimport type { Request, Response } from 'express';\nimport { createAbortSignalFromHttp } from './abort-http-signal';\n\nexport type AbortableFn<T> = (ac: AbortController) => Promise<T>;\n\nexport const fromAbortable = <T>(fn: AbortableFn<T>): Observable<T> => {\n return new Observable<T>((subscriber) => {\n const ac = new AbortController();\n\n fn(ac).then(\n (value) => {\n if (!ac.signal.aborted && !subscriber.closed) {\n subscriber.next(value);\n subscriber.complete();\n }\n },\n (err) => {\n if (ac.signal.aborted) {\n if (!subscriber.closed) subscriber.complete();\n return;\n }\n if (!subscriber.closed) subscriber.error(err);\n },\n );\n\n return () => {\n if (!ac.signal.aborted) ac.abort();\n };\n });\n};\n\nexport const takeUntilAbort = <T>(signal: AbortSignal) => {\n return (source: Observable<T>) => {\n return source.pipe(\n takeUntil(\n new Observable<void>((subscriber) => {\n const onAbort = () => subscriber.next();\n if (signal.aborted) {\n subscriber.next();\n } else {\n signal.addEventListener('abort', onAbort, { once: true });\n }\n return () => signal.removeEventListener('abort', onAbort);\n }),\n ),\n );\n };\n};\n\nexport const As = createParamDecorator(\n (_data: unknown, ctx: ExecutionContext) => {\n const req = ctx.switchToHttp().getRequest<Request>();\n return createAbortSignalFromHttp(req);\n },\n);\n", "// abort-http-signal.ts\nimport type { IncomingMessage } from 'http';\nimport type { Request as ExpressReq } from 'express';\nimport type { FastifyRequest } from 'fastify';\nimport { BlankReturnMessageDto } from './return-message';\n\ntype AnyReq = ExpressReq | FastifyRequest | IncomingMessage;\n\nfunction toRawReq(req: AnyReq): IncomingMessage {\n // Fastify: req.raw \u662F IncomingMessage\uFF1BExpress: req \u672C\u8EAB\u5C31\u662F IncomingMessage\n if ((req as any)?.raw?.socket) return (req as any).raw as IncomingMessage;\n return req as IncomingMessage;\n}\n\n/**\n * \u4EC5\u901A\u8FC7 req \u81EA\u52A8\u9002\u914D Express/Fastify\uFF0C\u7ED1\u5B9A HTTP \u53D6\u6D88\u5230 AbortSignal\u3002\n * - \u4F18\u5148\u76D1\u542C 'aborted'\uFF08\u5BA2\u6237\u7AEF\u4E2D\u65AD\u6700\u53EF\u9760\u7684\u4FE1\u53F7\uFF09\n * - \u53EF\u9009\u515C\u5E95\uFF1A\u5728 socket 'close' \u65F6\uFF0C\u4EC5\u5F53\u8BF7\u6C42\u672A\u5B8C\u6574\u63A5\u6536/\u6216\u5DF2\u6807\u8BB0 aborted \u65F6\uFF0C\u624D\u89E6\u53D1 abort\uFF0C\u907F\u514D\u6B63\u5E38\u5B8C\u6210\u7684\u8BEF\u4F24\n * - reason \u56FA\u5B9A\u4E3A 499 HttpException\n *\n * \u6CE8\u610F\uFF1A\u82E5\u4F60\u7684 abortable() \u5728 signal.reason \u662F Error \u65F6\u4F1A\u201C\u539F\u6837 throw reason\u201D\uFF0C\n * \u90A3\u4E48\u4E0A\u5C42\u4F1A\u6536\u5230 HttpException(499)\u3002\u82E5\u4ECD\u60F3\u7EDF\u4E00\u629B AbortedError\uFF0C\u53EF\u6539\u4E3A\uFF1A\n * const reason = new AbortedError('Request aborted', httpErr);\n * \u5E76\u8BA9 throwIfAborted \u5148\u629B AbortedError\uFF08\u4FDD\u7559 cause\uFF09\u3002\n */\nexport function createAbortSignalFromHttp(req: AnyReq): AbortSignal {\n const rawReq = toRawReq(req);\n const ac = new AbortController();\n\n const makeReason = () =>\n new BlankReturnMessageDto(499, 'Request aborted').toException();\n\n const abortOnce = () => {\n if (!ac.signal.aborted) ac.abort(makeReason());\n cleanup();\n };\n\n const onClose = () => {\n abortOnce();\n };\n\n const cleanup = () => {\n rawReq.off?.('close', onClose);\n };\n\n rawReq.once?.('close', onClose);\n\n return ac.signal;\n}\n", "import { Inject, Scope } from '@nestjs/common';\nimport { abortable, AbortableOpts } from 'nfkit';\nimport { ABORT_SIGNAL } from './abort-signal.provider';\nimport { ContextIdFactory, ModuleRef, REQUEST } from '@nestjs/core';\nimport { createProvider } from '../create-provider';\nimport { InjectionToken } from '@nestjs/common/interfaces/modules/injection-token.interface';\n\nconst tokenMemo = new Map<any, symbol>();\nexport const abortableToken = (token: InjectionToken) => {\n if (tokenMemo.has(token)) return tokenMemo.get(token)!;\n const name = typeof token === 'function' ? token.name : String(token);\n const sym = Symbol.for(`Abortable(${name})`);\n tokenMemo.set(token, sym);\n return sym;\n};\n\n/**\n * \u652F\u6301\u4E24\u79CD\u7528\u6CD5\uFF1A\n * @InjectAbortable(SomeService)\n * @InjectAbortable() // \u81EA\u52A8\u63A8\u65AD\u7C7B\u578B\n */\nexport function InjectAbortable(token?: InjectionToken): ParameterDecorator {\n return (target, propertyKey, parameterIndex) => {\n let actualToken = token;\n\n if (!actualToken) {\n // \u5229\u7528 reflect-metadata \u83B7\u53D6\u53C2\u6570\u7C7B\u578B\n const paramTypes: any[] =\n Reflect.getMetadata('design:paramtypes', target, propertyKey) || [];\n actualToken = paramTypes[parameterIndex];\n if (!actualToken) {\n throw new Error(\n `@InjectAbortable() cannot infer type from metadata: ${target.constructor?.name}[${parameterIndex}]`,\n );\n }\n }\n\n Inject(abortableToken(actualToken))(target, propertyKey, parameterIndex);\n };\n}\n\nexport function createAbortableProvider<T>(\n token: InjectionToken<T>,\n opts?: AbortableOpts,\n) {\n const provide = abortableToken(token);\n\n return createProvider(\n {\n provide,\n scope: Scope.REQUEST,\n inject: [ModuleRef, REQUEST, ABORT_SIGNAL],\n },\n async (moduleRef, req: Request, signal) => {\n // \u8BA9\u89E3\u6790\u4E0E\u5F53\u524D\u8BF7\u6C42\u4E0A\u4E0B\u6587\u7ED1\u5B9A\uFF08\u652F\u6301 request/transient \u4F5C\u7528\u57DF\uFF09\n const ctxId = ContextIdFactory.getByRequest(req);\n // \u4E25\u683C\u6A21\u5F0F\u5173\u95ED\uFF0C\u5141\u8BB8\u8DE8\u6A21\u5757\u8FB9\u754C\u89E3\u6790\uFF08\u89E3\u51B3\u6D4B\u8BD5\u91CC forFeature \u5B50\u6A21\u5757\u770B\u4E0D\u5230 DemoService \u7684\u60C5\u51B5\uFF09\n const svc = await moduleRef.resolve<T>(token, ctxId, { strict: false });\n if (svc == null) {\n throw new Error(\n `Abortable: provider \"${String(\n (token as any).name ?? token,\n )}\" not found in container (even with strict:false)`,\n );\n }\n return abortable<T>(svc, signal, opts);\n },\n );\n}\n", "import { Inject, Scope, Type } from '@nestjs/common';\nimport { REQUEST } from '@nestjs/core';\nimport { createAbortSignalFromHttp } from '../abort-http-signal';\nimport { createProvider } from '../create-provider';\n\nexport const ABORT_SIGNAL = Symbol(\n 'ABORT_SIGNAL',\n) as unknown as Type<AbortSignal>;\nexport const AbortSignalProvider = createProvider(\n {\n provide: ABORT_SIGNAL,\n scope: Scope.REQUEST,\n inject: [REQUEST],\n },\n createAbortSignalFromHttp,\n);\n\nexport const InjectAbortSignal = () => Inject(ABORT_SIGNAL);\n", "import { FactoryProvider } from '@nestjs/common';\nimport { InjectionToken } from '@nestjs/common/interfaces/modules/injection-token.interface';\n\ntype TypeFromToken<T> = T extends string | symbol\n ? any\n : T extends InjectionToken<infer U>\n ? U\n : any;\n\ntype TokensToTypes<A extends InjectionToken[]> = A extends []\n ? []\n : A extends [infer F, ...infer R]\n ? [TypeFromToken<F>, ...TokensToTypes<R extends InjectionToken[] ? R : []>]\n : [];\n\ntype Awaitable<T> = T | Promise<T>;\n\nexport interface TypedFactoryProvider<A extends InjectionToken[], R>\n extends FactoryProvider<R> {\n inject: A;\n useFactory: (...args: TokensToTypes<A>) => Awaitable<R>;\n}\n\nexport const createProvider = <const A extends InjectionToken[], R>(\n options: { inject: A } & Omit<FactoryProvider<R>, 'inject' | 'useFactory'>,\n factory: (...args: TokensToTypes<A>) => Awaitable<R>,\n): TypedFactoryProvider<A, R> => {\n return {\n useFactory: factory,\n ...options,\n };\n};\n", "import { DynamicModule, Module } from '@nestjs/common';\nimport { createAbortableProvider } from './abortable.token';\nimport { AbortableOpts } from 'nfkit';\nimport { AbortSignalProvider } from './abort-signal.provider';\nimport { InjectionToken } from '@nestjs/common/interfaces/modules/injection-token.interface';\n\nexport interface AbortableModuleOptions {\n abortableOptions?: AbortableOpts;\n}\n\n@Module({})\nexport class AbortableModule {\n static forRoot(): DynamicModule {\n return {\n module: AbortableModule,\n providers: [AbortSignalProvider],\n exports: [AbortSignalProvider],\n global: true,\n };\n }\n\n static forFeature(\n tokens: InjectionToken[],\n options?: AbortableModuleOptions,\n ): DynamicModule {\n const providers = tokens.map((token) =>\n createAbortableProvider(token, options?.abortableOptions),\n );\n\n return {\n module: AbortableModule,\n providers: [...providers],\n exports: [...providers],\n };\n }\n}\n"],
|
|
5
|
+
"mappings": "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",
|
|
6
|
+
"names": ["ApiProperty", "ApiProperty", "Inject", "Scope", "Inject", "Inject", "REQUEST", "Inject", "Scope", "REQUEST"]
|
|
7
7
|
}
|
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import {
|
|
2
|
-
export declare const ABORT_SIGNAL:
|
|
3
|
-
export declare const AbortSignalProvider:
|
|
1
|
+
import { Type } from '@nestjs/common';
|
|
2
|
+
export declare const ABORT_SIGNAL: Type<AbortSignal>;
|
|
3
|
+
export declare const AbortSignalProvider: import("../create-provider").TypedFactoryProvider<["REQUEST"], AbortSignal>;
|
|
4
4
|
export declare const InjectAbortSignal: () => PropertyDecorator & ParameterDecorator;
|
|
@@ -1,10 +1,10 @@
|
|
|
1
1
|
import { DynamicModule } from '@nestjs/common';
|
|
2
|
-
import { ProviderToken } from './abortable.token';
|
|
3
2
|
import { AbortableOpts } from 'nfkit';
|
|
3
|
+
import { InjectionToken } from '@nestjs/common/interfaces/modules/injection-token.interface';
|
|
4
4
|
export interface AbortableModuleOptions {
|
|
5
5
|
abortableOptions?: AbortableOpts;
|
|
6
6
|
}
|
|
7
7
|
export declare class AbortableModule {
|
|
8
8
|
static forRoot(): DynamicModule;
|
|
9
|
-
static forFeature(tokens:
|
|
9
|
+
static forFeature(tokens: InjectionToken[], options?: AbortableModuleOptions): DynamicModule;
|
|
10
10
|
}
|
|
@@ -1,11 +1,11 @@
|
|
|
1
|
-
import { Provider } from '@nestjs/common';
|
|
2
1
|
import { AbortableOpts } from 'nfkit';
|
|
3
|
-
|
|
4
|
-
|
|
2
|
+
import { ModuleRef } from '@nestjs/core';
|
|
3
|
+
import { InjectionToken } from '@nestjs/common/interfaces/modules/injection-token.interface';
|
|
4
|
+
export declare const abortableToken: (token: InjectionToken) => symbol;
|
|
5
5
|
/**
|
|
6
6
|
* 支持两种用法:
|
|
7
7
|
* @InjectAbortable(SomeService)
|
|
8
8
|
* @InjectAbortable() // 自动推断类型
|
|
9
9
|
*/
|
|
10
|
-
export declare function InjectAbortable(token?:
|
|
11
|
-
export declare function createAbortableProvider<T>(token:
|
|
10
|
+
export declare function InjectAbortable(token?: InjectionToken): ParameterDecorator;
|
|
11
|
+
export declare function createAbortableProvider<T>(token: InjectionToken<T>, opts?: AbortableOpts): import("../create-provider").TypedFactoryProvider<[typeof ModuleRef, "REQUEST", import("@nestjs/common").Type<AbortSignal>], T>;
|
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
import { FactoryProvider } from '@nestjs/common';
|
|
2
|
+
import { InjectionToken } from '@nestjs/common/interfaces/modules/injection-token.interface';
|
|
3
|
+
type TypeFromToken<T> = T extends string | symbol ? any : T extends InjectionToken<infer U> ? U : any;
|
|
4
|
+
type TokensToTypes<A extends InjectionToken[]> = A extends [] ? [] : A extends [infer F, ...infer R] ? [TypeFromToken<F>, ...TokensToTypes<R extends InjectionToken[] ? R : []>] : [];
|
|
5
|
+
type Awaitable<T> = T | Promise<T>;
|
|
6
|
+
export interface TypedFactoryProvider<A extends InjectionToken[], R> extends FactoryProvider<R> {
|
|
7
|
+
inject: A;
|
|
8
|
+
useFactory: (...args: TokensToTypes<A>) => Awaitable<R>;
|
|
9
|
+
}
|
|
10
|
+
export declare const createProvider: <const A extends InjectionToken[], R>(options: {
|
|
11
|
+
inject: A;
|
|
12
|
+
} & Omit<FactoryProvider<R>, "inject" | "useFactory">, factory: (...args: TokensToTypes<A>) => Awaitable<R>) => TypedFactoryProvider<A, R>;
|
|
13
|
+
export {};
|
package/index.ts
CHANGED
package/package.json
CHANGED
package/build.js
DELETED
|
@@ -1,167 +0,0 @@
|
|
|
1
|
-
#!/usr/bin/env node
|
|
2
|
-
/* build.js - node build.js [all|cjs|esm|types|clean] */
|
|
3
|
-
const fs = require('fs');
|
|
4
|
-
const path = require('path');
|
|
5
|
-
const { builtinModules } = require('module');
|
|
6
|
-
|
|
7
|
-
const DIST_DIR = 'dist';
|
|
8
|
-
|
|
9
|
-
/* ------------------------- utils ------------------------- */
|
|
10
|
-
function readJSONSafe(file, fallback = {}) {
|
|
11
|
-
try {
|
|
12
|
-
const p = path.resolve(process.cwd(), file);
|
|
13
|
-
const txt = fs.readFileSync(p, 'utf8');
|
|
14
|
-
return JSON.parse(txt);
|
|
15
|
-
} catch {
|
|
16
|
-
return fallback;
|
|
17
|
-
}
|
|
18
|
-
}
|
|
19
|
-
function uniq(arr) { return Array.from(new Set(arr)); }
|
|
20
|
-
function ensureDir(p) { fs.mkdirSync(p, { recursive: true }); }
|
|
21
|
-
function rimraf(p) { if (fs.existsSync(p)) fs.rmSync(p, { recursive: true, force: true }); }
|
|
22
|
-
|
|
23
|
-
function depsAsExternal(pkg) {
|
|
24
|
-
const dep = Object.keys(pkg.dependencies || {});
|
|
25
|
-
const peer = Object.keys(pkg.peerDependencies || {});
|
|
26
|
-
const names = uniq([...dep, ...peer]);
|
|
27
|
-
// 覆盖子路径导入(lodash/fp、react/jsx-runtime)
|
|
28
|
-
return uniq(names.flatMap((n) => [n, `${n}/*`]));
|
|
29
|
-
}
|
|
30
|
-
function nodeBuiltinsExternal() {
|
|
31
|
-
// 既包含 'fs' 也包含 'node:fs' 形式
|
|
32
|
-
return uniq([...builtinModules, ...builtinModules.map((m) => `node:${m}`)]);
|
|
33
|
-
}
|
|
34
|
-
async function loadEsbuild() {
|
|
35
|
-
try { return require('esbuild'); }
|
|
36
|
-
catch { const mod = await import('esbuild'); return mod.build ? mod : mod.default; }
|
|
37
|
-
}
|
|
38
|
-
function tsconfigPath() { return fs.existsSync('tsconfig.json') ? 'tsconfig.json' : undefined; }
|
|
39
|
-
function entryPointsFromPkg(/*pkg*/) { return ['index.ts']; }
|
|
40
|
-
|
|
41
|
-
/* ------------------------- esbuild builds ------------------------- */
|
|
42
|
-
async function buildOne(format, options) {
|
|
43
|
-
const esbuild = await loadEsbuild();
|
|
44
|
-
const { external, tsconfig, entryPoints } = options;
|
|
45
|
-
const isCjs = format === 'cjs';
|
|
46
|
-
const outfile = path.join(DIST_DIR, isCjs ? 'index.cjs' : 'index.mjs');
|
|
47
|
-
|
|
48
|
-
ensureDir(path.dirname(outfile));
|
|
49
|
-
console.log(`[build] ${format} -> ${outfile}`);
|
|
50
|
-
|
|
51
|
-
await esbuild.build({
|
|
52
|
-
entryPoints,
|
|
53
|
-
outfile,
|
|
54
|
-
bundle: true,
|
|
55
|
-
sourcemap: true,
|
|
56
|
-
format, // 'cjs' | 'esm'
|
|
57
|
-
platform: isCjs ? 'node' : 'neutral',
|
|
58
|
-
target: isCjs ? 'es2021' : 'esnext',
|
|
59
|
-
external, // deps + peerDeps + node builtins (含 node:*)
|
|
60
|
-
logLevel: 'info',
|
|
61
|
-
...(tsconfig ? { tsconfig } : {}),
|
|
62
|
-
});
|
|
63
|
-
}
|
|
64
|
-
|
|
65
|
-
/* ------------------------- types via TypeScript API ------------------------- */
|
|
66
|
-
function buildTypesAPI(outDir = DIST_DIR) {
|
|
67
|
-
let ts;
|
|
68
|
-
try { ts = require('typescript'); }
|
|
69
|
-
catch {
|
|
70
|
-
console.error('[types] Missing dependency: typescript');
|
|
71
|
-
process.exit(1);
|
|
72
|
-
}
|
|
73
|
-
|
|
74
|
-
const cfgPath = ts.findConfigFile('./', ts.sys.fileExists, 'tsconfig.json');
|
|
75
|
-
|
|
76
|
-
let fileNames, options;
|
|
77
|
-
if (cfgPath) {
|
|
78
|
-
// 读取 tsconfig.json
|
|
79
|
-
const { config } = ts.readConfigFile(cfgPath, ts.sys.readFile);
|
|
80
|
-
const parsed = ts.parseJsonConfigFileContent(config, ts.sys, path.dirname(cfgPath));
|
|
81
|
-
fileNames = parsed.fileNames;
|
|
82
|
-
options = parsed.options;
|
|
83
|
-
} else {
|
|
84
|
-
// 没有 tsconfig 的降级:仅用 index.ts
|
|
85
|
-
console.warn('[types] tsconfig.json not found; fallback to index.ts with basic options.');
|
|
86
|
-
fileNames = ['index.ts'].filter((f) => fs.existsSync(f));
|
|
87
|
-
options = {
|
|
88
|
-
moduleResolution: 99, // NodeNext(避免引入 enum 名字,用常量值)
|
|
89
|
-
target: 99, // ESNext
|
|
90
|
-
skipLibCheck: true,
|
|
91
|
-
strict: true,
|
|
92
|
-
};
|
|
93
|
-
}
|
|
94
|
-
|
|
95
|
-
// 强制仅输出声明
|
|
96
|
-
options.declaration = true;
|
|
97
|
-
options.emitDeclarationOnly = true;
|
|
98
|
-
options.outDir = outDir;
|
|
99
|
-
// 为了不受 sourceMap/emit 等其它设置影响
|
|
100
|
-
options.noEmitOnError = false;
|
|
101
|
-
|
|
102
|
-
console.log('[types] Generating .d.ts ...');
|
|
103
|
-
const program = ts.createProgram(fileNames, options);
|
|
104
|
-
const pre = ts.getPreEmitDiagnostics(program);
|
|
105
|
-
const emitResult = program.emit();
|
|
106
|
-
const diagnostics = pre.concat(emitResult.diagnostics);
|
|
107
|
-
|
|
108
|
-
if (diagnostics.length) {
|
|
109
|
-
const formatHost = {
|
|
110
|
-
getCanonicalFileName: (p) => p,
|
|
111
|
-
getCurrentDirectory: ts.sys.getCurrentDirectory,
|
|
112
|
-
getNewLine: () => ts.sys.newLine,
|
|
113
|
-
};
|
|
114
|
-
const message = ts.formatDiagnosticsWithColorAndContext(diagnostics, formatHost);
|
|
115
|
-
console.error(message);
|
|
116
|
-
if (emitResult.emitSkipped) {
|
|
117
|
-
throw new Error('[types] Type generation failed.');
|
|
118
|
-
}
|
|
119
|
-
}
|
|
120
|
-
console.log('[types] Declarations generated.');
|
|
121
|
-
}
|
|
122
|
-
|
|
123
|
-
/* ------------------------- main dispatcher ------------------------- */
|
|
124
|
-
(async function main() {
|
|
125
|
-
const sub = (process.argv[2] || 'all').toLowerCase(); // all | cjs | esm | types | clean
|
|
126
|
-
|
|
127
|
-
const pkg = readJSONSafe('package.json');
|
|
128
|
-
const externalFromPkg = depsAsExternal(pkg);
|
|
129
|
-
// 统一 external:依赖 + peer + Node 内置(含 node:*)
|
|
130
|
-
const external = uniq([...externalFromPkg, ...nodeBuiltinsExternal()]);
|
|
131
|
-
const tscPath = tsconfigPath();
|
|
132
|
-
const entryPoints = entryPointsFromPkg(pkg);
|
|
133
|
-
|
|
134
|
-
switch (sub) {
|
|
135
|
-
case 'clean': {
|
|
136
|
-
console.log('[clean] remove dist/');
|
|
137
|
-
rimraf(DIST_DIR);
|
|
138
|
-
break;
|
|
139
|
-
}
|
|
140
|
-
case 'cjs': {
|
|
141
|
-
await buildOne('cjs', { external, tsconfig: tscPath, entryPoints });
|
|
142
|
-
break;
|
|
143
|
-
}
|
|
144
|
-
case 'esm': {
|
|
145
|
-
await buildOne('esm', { external, tsconfig: tscPath, entryPoints });
|
|
146
|
-
break;
|
|
147
|
-
}
|
|
148
|
-
case 'types': {
|
|
149
|
-
ensureDir(DIST_DIR);
|
|
150
|
-
buildTypesAPI(DIST_DIR);
|
|
151
|
-
break;
|
|
152
|
-
}
|
|
153
|
-
case 'all':
|
|
154
|
-
default: {
|
|
155
|
-
console.log('[clean] remove dist/');
|
|
156
|
-
rimraf(DIST_DIR);
|
|
157
|
-
await buildOne('cjs', { external, tsconfig: tscPath, entryPoints });
|
|
158
|
-
await buildOne('esm', { external, tsconfig: tscPath, entryPoints });
|
|
159
|
-
buildTypesAPI(DIST_DIR);
|
|
160
|
-
console.log('[build] Done.');
|
|
161
|
-
break;
|
|
162
|
-
}
|
|
163
|
-
}
|
|
164
|
-
})().catch((err) => {
|
|
165
|
-
console.error('[build] Failed:', err);
|
|
166
|
-
process.exit(1);
|
|
167
|
-
});
|