@autofleet/fastify-boilerplate 2.1.13 → 2.1.14
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.d.ts +7 -1
- package/dist/index.js +1 -1
- package/dist/index.js.map +1 -1
- package/package.json +2 -2
package/dist/index.d.ts
CHANGED
|
@@ -10,6 +10,12 @@ declare module "fastify" {
|
|
|
10
10
|
logger: LoggerInstanceManager;
|
|
11
11
|
}
|
|
12
12
|
}
|
|
13
|
+
type RemoveIndexSignature<T> = { [K in keyof T as string extends K ? never : number extends K ? never : symbol extends K ? never : K]: T[K] };
|
|
14
|
+
type Clients = NonNullable<HealthManagerOptions["clients"]>;
|
|
15
|
+
interface ClientsWithoutServer extends Omit<RemoveIndexSignature<Clients>, "server"> {
|
|
16
|
+
[key: string]: Clients[string];
|
|
17
|
+
server?: never;
|
|
18
|
+
}
|
|
13
19
|
interface FastifyBoilerPlateOptions {
|
|
14
20
|
/** The name of the Fastify application. */
|
|
15
21
|
name?: string;
|
|
@@ -30,7 +36,7 @@ interface FastifyBoilerPlateOptions {
|
|
|
30
36
|
* Note: The server is automatically attached by the plugin, so don't include it in the clients configuration.
|
|
31
37
|
*/
|
|
32
38
|
healthManagerOptions: Omit<HealthManagerOptions, "logger" | "clients"> & {
|
|
33
|
-
clients?:
|
|
39
|
+
clients?: ClientsWithoutServer;
|
|
34
40
|
};
|
|
35
41
|
}
|
|
36
42
|
declare const fastifyBoilerplatePlugin: FastifyPluginAsync<FastifyBoilerPlateOptions>;
|
package/dist/index.js
CHANGED
|
@@ -1,2 +1,2 @@
|
|
|
1
|
-
import*as e from"zod";import t from"@fastify/helmet";import n from"fastify-plugin";import r from"@fastify/compress";import{handleErrorFastify as i}from"@autofleet/errors";import{authFromUserIdHeaderPlugin as a}from"@autofleet/zehut";import{HealthManager as o}from"@autofleet/nitur";import{fastifyZodOpenApiPlugin as s,fastifyZodOpenApiTransform as c,fastifyZodOpenApiTransformObject as l,serializerCompiler as u,validatorCompiler as d}from"fastify-zod-openapi";const f=n((e,t,n)=>{let r=new WeakMap;function i(e){if(!r.has(e))return;let{boundOnceRequestClosed:t,boundOnceRequestError:n}=r.get(e);t&&e.raw.removeListener(`close`,t),n&&e.raw.removeListener(`error`,n)}function a(){if(i(this),r.has(this)&&this.raw.destroyed){let{controller:e}=r.get(this);if(e.signal.aborted)return;e.abort()}}function o(e){if(i(this),r.has(this)){let{controller:t}=r.get(this);t.abort(e)}}e.decorateRequest(`signal`,{getter(){let{raw:e,id:t}=this;if(e.socket.destroyed===!0)throw Error(`Socket for request with ID '${t}' already closed`);if(r.has(this))return r.get(this).controller.signal;let n=a.bind(this),i=o.bind(this),s=new AbortController;return r.set(this,{controller:s,boundOnceRequestClosed:n,boundOnceRequestError:i}),e.once(`close`,n),e.once(`error`,i),s.signal}}),e.addHook(`onResponse`,(e,t,n)=>{r.has(e)&&(i(e),r.delete(e)),n()}),n()}),{version:p}={name:`@autofleet/fastify-boilerplate`,version:`2.1.
|
|
1
|
+
import*as e from"zod";import t from"@fastify/helmet";import n from"fastify-plugin";import r from"@fastify/compress";import{handleErrorFastify as i}from"@autofleet/errors";import{authFromUserIdHeaderPlugin as a}from"@autofleet/zehut";import{HealthManager as o}from"@autofleet/nitur";import{fastifyZodOpenApiPlugin as s,fastifyZodOpenApiTransform as c,fastifyZodOpenApiTransformObject as l,serializerCompiler as u,validatorCompiler as d}from"fastify-zod-openapi";const f=n((e,t,n)=>{let r=new WeakMap;function i(e){if(!r.has(e))return;let{boundOnceRequestClosed:t,boundOnceRequestError:n}=r.get(e);t&&e.raw.removeListener(`close`,t),n&&e.raw.removeListener(`error`,n)}function a(){if(i(this),r.has(this)&&this.raw.destroyed){let{controller:e}=r.get(this);if(e.signal.aborted)return;e.abort()}}function o(e){if(i(this),r.has(this)){let{controller:t}=r.get(this);t.abort(e)}}e.decorateRequest(`signal`,{getter(){let{raw:e,id:t}=this;if(e.socket.destroyed===!0)throw Error(`Socket for request with ID '${t}' already closed`);if(r.has(this))return r.get(this).controller.signal;let n=a.bind(this),i=o.bind(this),s=new AbortController;return r.set(this,{controller:s,boundOnceRequestClosed:n,boundOnceRequestError:i}),e.once(`close`,n),e.once(`error`,i),s.signal}}),e.addHook(`onResponse`,(e,t,n)=>{r.has(e)&&(i(e),r.delete(e)),n()}),n()}),{version:p}={name:`@autofleet/fastify-boilerplate`,version:`2.1.14`,type:`module`,main:`./dist/index.js`,module:`./dist/index.js`,types:`./dist/index.d.ts`,exports:{".":{import:{types:`./dist/index.d.ts`,default:`./dist/index.js`}}},scripts:{build:`pnpm -w tsdown -W ./packages/fastify-boilerplate`,test:`vitest`,coverage:`vitest --coverage`},repository:{type:`git`,url:`git+https://github.com/Autofleet/autorepo.git`},author:``,license:`Proprietary`,bugs:{url:`https://github.com/Autofleet/autorepo/issues`},homepage:`https://github.com/Autofleet/autorepo/tree/master/packages/fastify-boilerplate#readme`,engines:{node:`>=22`},peerDependencies:{"@autofleet/logger":`*`},devDependencies:{"@autofleet/logger":`workspace:^`,fastify:`catalog:`,"fastify-plugin":`catalog:`,"zod-openapi":`catalog:`},files:[`dist`],dependencies:{"@autofleet/errors":`workspace:^`,"@autofleet/nitur":`workspace:^`,"@autofleet/zehut":`workspace:^`,"@fastify/compress":`catalog:`,"@fastify/helmet":`catalog:`,"@fastify/swagger":`catalog:`,"@fastify/swagger-ui":`catalog:`,"fastify-zod-openapi":`catalog:`,zod:`catalog:`}},m=n(async(n,m)=>{let{logger:h,name:g,openApiName:_,version:v,tags:y,eagerLoadUserPermissions:b=!0,customPermissionLoader:x,healthManagerOptions:S}=m;if(n.decorateRequest(`logger`,{getter:()=>h}),n.register(i,{fallbackMsg:`Unhandled error caught`}),n.register(t),n.register(a,{eagerLoadUserPermissions:b,customPermissionLoader:x}),n.register(r,{global:!0,threshold:1024}),n.setValidatorCompiler(d),n.setSerializerCompiler(u),y?.length){let[{default:e},{default:t}]=await Promise.all([import(`@fastify/swagger`),import(`@fastify/swagger-ui`)]);await n.register(s),await n.register(e,{openapi:{openapi:`3.1.1`,info:{title:_||`${g} API`,version:v||`1.0.0`},tags:[{name:`Server status`,description:`General endpoints indicating server's status`},...y]},transform:c,transformObject:l}),await n.register(t)}let C=new Date;if(n.withTypeProvider().route({method:`GET`,url:`/`,logLevel:`warn`,...y?.length&&{schema:{tags:[`Server status`],description:`Get server status`,response:{200:{content:{"application/json":{schema:e.object({name:e.literal(g).optional().meta({description:`The name of ${g}`}),version:e.literal(v).optional().meta({description:`The version in package.json file of ${g}`}),boilerPlateVersion:e.literal(p).optional().meta({description:`The version in package.json file of the fastify boilerplate`}),serverRunningSince:e.date().optional().meta({description:`The date when the server started running`}),commit:e.string().meta({description:`The commit SHA of the current deployment`})})}}}}}},handler:()=>({name:g,version:v,serverRunningSince:C,boilerPlateVersion:p,commit:process.env.DD_GIT_COMMIT_SHA||`unknown`})}),S){let t=new o({...S,logger:h});n.withTypeProvider().route({method:`GET`,url:`/alive`,logLevel:`warn`,...y?.length&&{schema:{tags:[`Server status`],description:`Server liveness status`,response:{200:{content:{"application/json":{schema:e.object({status:e.literal(`ok`)})}}},503:{content:{"application/json":{schema:e.object({status:e.literal(`ERROR`),errors:e.array(e.string())})}}}}}},handler:()=>t.aliveCheck()}),n.withTypeProvider().route({method:`GET`,url:`/ready`,logLevel:`warn`,...y?.length&&{schema:{tags:[`Server status`],description:`Server readiness status`,response:{200:{content:{"application/json":{schema:e.object({status:e.literal(`ok`)})}}},503:{content:{"application/json":{schema:e.object({status:e.literal(`ERROR`),errors:e.array(e.string())})}}}}}},handler:()=>t.readyCheck()}),n.addHook(`onReady`,function(){this.server&&(t.attachServer({connection:this.server}),h.debug(`[FastifyBoilerplate] HTTP server attached to HealthManager for graceful shutdown`))})}n.register(f)});export{m as fastifyBoilerplatePlugin};
|
|
2
2
|
//# sourceMappingURL=index.js.map
|
package/dist/index.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.js","names":["fastifyRequestSignal: FastifyPluginCallback","fastifyRequestSignalPlugin: FastifyPluginCallback","pkgJson","fastifyBoilerplate: FastifyPluginAsync<FastifyBoilerPlateOptions>","name","version","healthManager: HealthManager","fastifyBoilerplatePlugin: FastifyPluginAsync<FastifyBoilerPlateOptions>"],"sources":["../src/cancelSignalProvider.ts","../package.json","../src/index.ts"],"sourcesContent":["import fastifyPlugin from 'fastify-plugin';\nimport type { FastifyRequest, FastifyPluginCallback } from 'fastify';\n\ndeclare module 'fastify' {\n interface FastifyRequest {\n /**\n * The AbortSignal associated with the request.\n * This signal can be used to abort operations related to the request.\n * It is automatically aborted when the request is closed or destroyed.\n *\n * This signal is created lazily, meaning it is only created if this field is accessed.\n *\n * Make sure to handle the signal carefully in your request handlers to avoid memory leaks or unexpected behavior.\n * If you add an event listener, make sure to remove it when the request is done.\n * Consider using the disposable {@link https://nodejs.org/docs/latest/api/events.html#eventsaddabortlistenersignal-listener `addAbortListener`} method.\n */\n signal: AbortSignal;\n }\n}\n\nconst fastifyRequestSignal: FastifyPluginCallback = (fastify, _options, next) => {\n const controllers = new WeakMap<FastifyRequest, { controller: AbortController; boundOnceRequestClosed: typeof onceRequestClosed; boundOnceRequestError: typeof onceRequestError; }>();\n\n function removeRequestListeners(req: FastifyRequest): void {\n if (!controllers.has(req)) {\n return;\n }\n const { boundOnceRequestClosed, boundOnceRequestError } = controllers.get(req)!;\n if (boundOnceRequestClosed) {\n req.raw.removeListener('close', boundOnceRequestClosed);\n }\n if (boundOnceRequestError) {\n req.raw.removeListener('error', boundOnceRequestError);\n }\n }\n\n function onceRequestClosed(this: FastifyRequest) {\n removeRequestListeners(this);\n if (controllers.has(this) && this.raw.destroyed) {\n const { controller } = controllers.get(this)!;\n if (controller.signal.aborted) {\n return;\n }\n controller.abort();\n }\n }\n function onceRequestError(this: FastifyRequest, err: Error) {\n removeRequestListeners(this);\n if (controllers.has(this)) {\n const { controller } = controllers.get(this)!;\n controller.abort(err);\n }\n }\n\n fastify.decorateRequest('signal', {\n getter() {\n const { raw, id } = this;\n if (raw.socket.destroyed === true) {\n throw new Error(`Socket for request with ID '${id}' already closed`);\n }\n if (controllers.has(this)) {\n // Since this is a getter, we can return the existing controller's signal, if the signal was already created.\n return controllers.get(this)!.controller.signal;\n }\n const boundOnceRequestClosed = onceRequestClosed.bind(this);\n const boundOnceRequestError = onceRequestError.bind(this);\n const controller = new AbortController();\n controllers.set(this, { controller, boundOnceRequestClosed, boundOnceRequestError });\n\n raw.once('close', boundOnceRequestClosed);\n raw.once('error', boundOnceRequestError);\n return controller.signal;\n },\n });\n\n fastify.addHook('onResponse', (request, _reply, done) => {\n if (controllers.has(request)) {\n removeRequestListeners(request);\n controllers.delete(request);\n }\n done();\n });\n\n next();\n};\n\nexport const fastifyRequestSignalPlugin: FastifyPluginCallback = fastifyPlugin(fastifyRequestSignal);\n","","import * as z from 'zod';\nimport helmet from '@fastify/helmet';\nimport fastifyPlugin from 'fastify-plugin';\nimport type { FastifyPluginAsync } from 'fastify';\nimport compression from '@fastify/compress';\nimport type { ZodOpenApiVersion } from 'zod-openapi';\nimport { handleErrorFastify } from '@autofleet/errors';\nimport { authFromUserIdHeaderPlugin } from '@autofleet/zehut';\nimport type { LoggerInstanceManager } from '@autofleet/logger';\nimport { HealthManager, type HealthManagerOptions } from '@autofleet/nitur';\nimport type { FastifyDynamicSwaggerOptions } from '@fastify/swagger';\nimport {\n fastifyZodOpenApiPlugin, fastifyZodOpenApiTransform, fastifyZodOpenApiTransformObject, serializerCompiler, validatorCompiler,\n type FastifyZodOpenApiSchema, type FastifyZodOpenApiTypeProvider,\n} from 'fastify-zod-openapi';\nimport { fastifyRequestSignalPlugin } from './cancelSignalProvider.js';\nimport pkgJson from '../package.json' with { type: 'json' };\n\nconst { version: boilerPlateVersion } = pkgJson;\n\ndeclare module 'fastify' {\n interface FastifyRequest {\n logger: LoggerInstanceManager;\n }\n}\n\nexport interface FastifyBoilerPlateOptions {\n /** The name of the Fastify application. */\n name?: string;\n /** The name to show in the swagger UI. @default `${name} API` */\n openApiName?: string;\n /** The version of the MS. */\n version?: string;\n /** The logger instance to use across the application. */\n logger: LoggerInstanceManager;\n /** When tags are provided, a few plugins will be added for fastify to include OpenAPI docs, and a SwaggerUI. */\n tags?: NonNullable<FastifyDynamicSwaggerOptions['openapi']>['tags'];\n /** Should user permissions be loaded eagerly. @default true */\n eagerLoadUserPermissions?: boolean;\n /** Custom permission loader function to be used in the `zehut` authorization plugin. */\n customPermissionLoader?: Parameters<typeof authFromUserIdHeaderPlugin>[1]['customPermissionLoader'];\n /**\n * HealthManager options to create a HealthManager instance for Kubernetes liveness/readiness probes and graceful shutdown.\n * Note: The server is automatically attached by the plugin, so don't include it in the clients configuration.\n */\n healthManagerOptions: Omit<HealthManagerOptions, 'logger' | 'clients'> & {\n clients?: Omit<HealthManagerOptions['clients'], 'server'>;\n };\n}\n\nconst fastifyBoilerplate: FastifyPluginAsync<FastifyBoilerPlateOptions> = async (fastify, options) => {\n const { logger, name, openApiName, version, tags, eagerLoadUserPermissions = true, customPermissionLoader, healthManagerOptions } = options;\n fastify.decorateRequest('logger', { getter: () => logger });\n\n fastify.register(handleErrorFastify, { fallbackMsg: 'Unhandled error caught' });\n\n fastify.register(helmet);\n\n fastify.register(authFromUserIdHeaderPlugin, { eagerLoadUserPermissions, customPermissionLoader });\n\n fastify.register(compression, { global: true, threshold: 1024 });\n\n // #region zod + swagger\n fastify.setValidatorCompiler(validatorCompiler);\n fastify.setSerializerCompiler(serializerCompiler);\n if (tags?.length) {\n const [{ default: fastifySwagger }, { default: fastifySwaggerUI }] = await Promise.all([\n import('@fastify/swagger'),\n import('@fastify/swagger-ui'),\n ]);\n await fastify.register(fastifyZodOpenApiPlugin);\n await fastify.register(fastifySwagger, {\n openapi: {\n openapi: '3.1.1' satisfies ZodOpenApiVersion,\n info: {\n title: openApiName || `${name} API`,\n /* v8 ignore next */\n version: version || '1.0.0',\n },\n tags: [\n { name: 'Server status', description: 'General endpoints indicating server\\'s status' },\n ...tags,\n ],\n },\n transform: fastifyZodOpenApiTransform,\n transformObject: fastifyZodOpenApiTransformObject,\n });\n await fastify.register(fastifySwaggerUI);\n }\n // #endregion\n\n const serverRunningSince = new Date();\n fastify.withTypeProvider<FastifyZodOpenApiTypeProvider>().route({\n method: 'GET',\n url: '/',\n logLevel: 'warn',\n ...(tags?.length && {\n schema: {\n tags: ['Server status'],\n description: 'Get server status',\n response: {\n 200: {\n content: {\n 'application/json': {\n schema: z.object({\n name: z.literal(name).optional().meta({ description: `The name of ${name}` }),\n version: z.literal(version).optional().meta({ description: `The version in package.json file of ${name}` }),\n boilerPlateVersion: z.literal(boilerPlateVersion).optional().meta({ description: 'The version in package.json file of the fastify boilerplate' }),\n serverRunningSince: z.date().optional().meta({ description: 'The date when the server started running' }),\n commit: z.string().meta({ description: 'The commit SHA of the current deployment' }),\n }),\n },\n },\n },\n },\n } satisfies FastifyZodOpenApiSchema,\n }),\n handler: () => ({\n name,\n version,\n serverRunningSince,\n boilerPlateVersion,\n commit: process.env.DD_GIT_COMMIT_SHA || 'unknown',\n }),\n });\n\n if (healthManagerOptions) {\n const healthManager: HealthManager = new HealthManager({ ...healthManagerOptions, logger });\n fastify.withTypeProvider<FastifyZodOpenApiTypeProvider>().route({\n method: 'GET',\n url: '/alive',\n logLevel: 'warn',\n ...(tags?.length && {\n schema: {\n tags: ['Server status'],\n description: 'Server liveness status',\n response: {\n 200: {\n content: {\n 'application/json': {\n schema: z.object({ status: z.literal('ok') }),\n },\n },\n },\n 503: {\n content: {\n 'application/json': {\n schema: z.object({\n status: z.literal('ERROR'),\n errors: z.array(z.string()),\n }),\n },\n },\n },\n },\n } satisfies FastifyZodOpenApiSchema,\n }),\n handler: () => healthManager.aliveCheck(),\n });\n\n fastify.withTypeProvider<FastifyZodOpenApiTypeProvider>().route({\n method: 'GET',\n url: '/ready',\n logLevel: 'warn',\n ...(tags?.length && {\n schema: {\n tags: ['Server status'],\n description: 'Server readiness status',\n response: {\n 200: {\n content: {\n 'application/json': {\n schema: z.object({ status: z.literal('ok') }),\n },\n },\n },\n 503: {\n content: {\n 'application/json': {\n schema: z.object({\n status: z.literal('ERROR'),\n errors: z.array(z.string()),\n }),\n },\n },\n },\n },\n } satisfies FastifyZodOpenApiSchema,\n }),\n handler: () => healthManager.readyCheck(),\n });\n\n fastify.addHook('onReady', function () {\n if (this.server) {\n healthManager.attachServer({ connection: this.server });\n logger.debug('[FastifyBoilerplate] HTTP server attached to HealthManager for graceful shutdown');\n }\n });\n }\n\n fastify.register(fastifyRequestSignalPlugin);\n};\n\nexport const fastifyBoilerplatePlugin: FastifyPluginAsync<FastifyBoilerPlateOptions> = fastifyPlugin(fastifyBoilerplate);\n"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"index.js","names":["fastifyRequestSignal: FastifyPluginCallback","fastifyRequestSignalPlugin: FastifyPluginCallback","pkgJson","fastifyBoilerplate: FastifyPluginAsync<FastifyBoilerPlateOptions>","name","version","healthManager: HealthManager","fastifyBoilerplatePlugin: FastifyPluginAsync<FastifyBoilerPlateOptions>"],"sources":["../src/cancelSignalProvider.ts","../package.json","../src/index.ts"],"sourcesContent":["import fastifyPlugin from 'fastify-plugin';\nimport type { FastifyRequest, FastifyPluginCallback } from 'fastify';\n\ndeclare module 'fastify' {\n interface FastifyRequest {\n /**\n * The AbortSignal associated with the request.\n * This signal can be used to abort operations related to the request.\n * It is automatically aborted when the request is closed or destroyed.\n *\n * This signal is created lazily, meaning it is only created if this field is accessed.\n *\n * Make sure to handle the signal carefully in your request handlers to avoid memory leaks or unexpected behavior.\n * If you add an event listener, make sure to remove it when the request is done.\n * Consider using the disposable {@link https://nodejs.org/docs/latest/api/events.html#eventsaddabortlistenersignal-listener `addAbortListener`} method.\n */\n signal: AbortSignal;\n }\n}\n\nconst fastifyRequestSignal: FastifyPluginCallback = (fastify, _options, next) => {\n const controllers = new WeakMap<FastifyRequest, { controller: AbortController; boundOnceRequestClosed: typeof onceRequestClosed; boundOnceRequestError: typeof onceRequestError; }>();\n\n function removeRequestListeners(req: FastifyRequest): void {\n if (!controllers.has(req)) {\n return;\n }\n const { boundOnceRequestClosed, boundOnceRequestError } = controllers.get(req)!;\n if (boundOnceRequestClosed) {\n req.raw.removeListener('close', boundOnceRequestClosed);\n }\n if (boundOnceRequestError) {\n req.raw.removeListener('error', boundOnceRequestError);\n }\n }\n\n function onceRequestClosed(this: FastifyRequest) {\n removeRequestListeners(this);\n if (controllers.has(this) && this.raw.destroyed) {\n const { controller } = controllers.get(this)!;\n if (controller.signal.aborted) {\n return;\n }\n controller.abort();\n }\n }\n function onceRequestError(this: FastifyRequest, err: Error) {\n removeRequestListeners(this);\n if (controllers.has(this)) {\n const { controller } = controllers.get(this)!;\n controller.abort(err);\n }\n }\n\n fastify.decorateRequest('signal', {\n getter() {\n const { raw, id } = this;\n if (raw.socket.destroyed === true) {\n throw new Error(`Socket for request with ID '${id}' already closed`);\n }\n if (controllers.has(this)) {\n // Since this is a getter, we can return the existing controller's signal, if the signal was already created.\n return controllers.get(this)!.controller.signal;\n }\n const boundOnceRequestClosed = onceRequestClosed.bind(this);\n const boundOnceRequestError = onceRequestError.bind(this);\n const controller = new AbortController();\n controllers.set(this, { controller, boundOnceRequestClosed, boundOnceRequestError });\n\n raw.once('close', boundOnceRequestClosed);\n raw.once('error', boundOnceRequestError);\n return controller.signal;\n },\n });\n\n fastify.addHook('onResponse', (request, _reply, done) => {\n if (controllers.has(request)) {\n removeRequestListeners(request);\n controllers.delete(request);\n }\n done();\n });\n\n next();\n};\n\nexport const fastifyRequestSignalPlugin: FastifyPluginCallback = fastifyPlugin(fastifyRequestSignal);\n","","import * as z from 'zod';\nimport helmet from '@fastify/helmet';\nimport fastifyPlugin from 'fastify-plugin';\nimport type { FastifyPluginAsync } from 'fastify';\nimport compression from '@fastify/compress';\nimport type { ZodOpenApiVersion } from 'zod-openapi';\nimport { handleErrorFastify } from '@autofleet/errors';\nimport { authFromUserIdHeaderPlugin } from '@autofleet/zehut';\nimport type { LoggerInstanceManager } from '@autofleet/logger';\nimport { HealthManager, type HealthManagerOptions } from '@autofleet/nitur';\nimport type { FastifyDynamicSwaggerOptions } from '@fastify/swagger';\nimport {\n fastifyZodOpenApiPlugin, fastifyZodOpenApiTransform, fastifyZodOpenApiTransformObject, serializerCompiler, validatorCompiler,\n type FastifyZodOpenApiSchema, type FastifyZodOpenApiTypeProvider,\n} from 'fastify-zod-openapi';\nimport { fastifyRequestSignalPlugin } from './cancelSignalProvider.js';\nimport pkgJson from '../package.json' with { type: 'json' };\n\nconst { version: boilerPlateVersion } = pkgJson;\n\ndeclare module 'fastify' {\n interface FastifyRequest {\n logger: LoggerInstanceManager;\n }\n}\n\ntype RemoveIndexSignature<T> = {\n [K in keyof T as string extends K ? never : number extends K ? never : symbol extends K ? never : K]: T[K];\n};\ntype Clients = NonNullable<HealthManagerOptions['clients']>;\ninterface ClientsWithoutServer extends Omit<RemoveIndexSignature<Clients>, 'server'> {\n [key: string]: Clients[string];\n server?: never;\n}\nexport interface FastifyBoilerPlateOptions {\n /** The name of the Fastify application. */\n name?: string;\n /** The name to show in the swagger UI. @default `${name} API` */\n openApiName?: string;\n /** The version of the MS. */\n version?: string;\n /** The logger instance to use across the application. */\n logger: LoggerInstanceManager;\n /** When tags are provided, a few plugins will be added for fastify to include OpenAPI docs, and a SwaggerUI. */\n tags?: NonNullable<FastifyDynamicSwaggerOptions['openapi']>['tags'];\n /** Should user permissions be loaded eagerly. @default true */\n eagerLoadUserPermissions?: boolean;\n /** Custom permission loader function to be used in the `zehut` authorization plugin. */\n customPermissionLoader?: Parameters<typeof authFromUserIdHeaderPlugin>[1]['customPermissionLoader'];\n /**\n * HealthManager options to create a HealthManager instance for Kubernetes liveness/readiness probes and graceful shutdown.\n * Note: The server is automatically attached by the plugin, so don't include it in the clients configuration.\n */\n healthManagerOptions: Omit<HealthManagerOptions, 'logger' | 'clients'> & {\n clients?: ClientsWithoutServer;\n };\n}\n\nconst fastifyBoilerplate: FastifyPluginAsync<FastifyBoilerPlateOptions> = async (fastify, options) => {\n const { logger, name, openApiName, version, tags, eagerLoadUserPermissions = true, customPermissionLoader, healthManagerOptions } = options;\n fastify.decorateRequest('logger', { getter: () => logger });\n\n fastify.register(handleErrorFastify, { fallbackMsg: 'Unhandled error caught' });\n\n fastify.register(helmet);\n\n fastify.register(authFromUserIdHeaderPlugin, { eagerLoadUserPermissions, customPermissionLoader });\n\n fastify.register(compression, { global: true, threshold: 1024 });\n\n // #region zod + swagger\n fastify.setValidatorCompiler(validatorCompiler);\n fastify.setSerializerCompiler(serializerCompiler);\n if (tags?.length) {\n const [{ default: fastifySwagger }, { default: fastifySwaggerUI }] = await Promise.all([\n import('@fastify/swagger'),\n import('@fastify/swagger-ui'),\n ]);\n await fastify.register(fastifyZodOpenApiPlugin);\n await fastify.register(fastifySwagger, {\n openapi: {\n openapi: '3.1.1' satisfies ZodOpenApiVersion,\n info: {\n title: openApiName || `${name} API`,\n /* v8 ignore next */\n version: version || '1.0.0',\n },\n tags: [\n { name: 'Server status', description: 'General endpoints indicating server\\'s status' },\n ...tags,\n ],\n },\n transform: fastifyZodOpenApiTransform,\n transformObject: fastifyZodOpenApiTransformObject,\n });\n await fastify.register(fastifySwaggerUI);\n }\n // #endregion\n\n const serverRunningSince = new Date();\n fastify.withTypeProvider<FastifyZodOpenApiTypeProvider>().route({\n method: 'GET',\n url: '/',\n logLevel: 'warn',\n ...(tags?.length && {\n schema: {\n tags: ['Server status'],\n description: 'Get server status',\n response: {\n 200: {\n content: {\n 'application/json': {\n schema: z.object({\n name: z.literal(name).optional().meta({ description: `The name of ${name}` }),\n version: z.literal(version).optional().meta({ description: `The version in package.json file of ${name}` }),\n boilerPlateVersion: z.literal(boilerPlateVersion).optional().meta({ description: 'The version in package.json file of the fastify boilerplate' }),\n serverRunningSince: z.date().optional().meta({ description: 'The date when the server started running' }),\n commit: z.string().meta({ description: 'The commit SHA of the current deployment' }),\n }),\n },\n },\n },\n },\n } satisfies FastifyZodOpenApiSchema,\n }),\n handler: () => ({\n name,\n version,\n serverRunningSince,\n boilerPlateVersion,\n commit: process.env.DD_GIT_COMMIT_SHA || 'unknown',\n }),\n });\n\n if (healthManagerOptions) {\n const healthManager: HealthManager = new HealthManager({ ...healthManagerOptions, logger });\n fastify.withTypeProvider<FastifyZodOpenApiTypeProvider>().route({\n method: 'GET',\n url: '/alive',\n logLevel: 'warn',\n ...(tags?.length && {\n schema: {\n tags: ['Server status'],\n description: 'Server liveness status',\n response: {\n 200: {\n content: {\n 'application/json': {\n schema: z.object({ status: z.literal('ok') }),\n },\n },\n },\n 503: {\n content: {\n 'application/json': {\n schema: z.object({\n status: z.literal('ERROR'),\n errors: z.array(z.string()),\n }),\n },\n },\n },\n },\n } satisfies FastifyZodOpenApiSchema,\n }),\n handler: () => healthManager.aliveCheck(),\n });\n\n fastify.withTypeProvider<FastifyZodOpenApiTypeProvider>().route({\n method: 'GET',\n url: '/ready',\n logLevel: 'warn',\n ...(tags?.length && {\n schema: {\n tags: ['Server status'],\n description: 'Server readiness status',\n response: {\n 200: {\n content: {\n 'application/json': {\n schema: z.object({ status: z.literal('ok') }),\n },\n },\n },\n 503: {\n content: {\n 'application/json': {\n schema: z.object({\n status: z.literal('ERROR'),\n errors: z.array(z.string()),\n }),\n },\n },\n },\n },\n } satisfies FastifyZodOpenApiSchema,\n }),\n handler: () => healthManager.readyCheck(),\n });\n\n fastify.addHook('onReady', function () {\n if (this.server) {\n healthManager.attachServer({ connection: this.server });\n logger.debug('[FastifyBoilerplate] HTTP server attached to HealthManager for graceful shutdown');\n }\n });\n }\n\n fastify.register(fastifyRequestSignalPlugin);\n};\n\nexport const fastifyBoilerplatePlugin: FastifyPluginAsync<FastifyBoilerPlateOptions> = fastifyPlugin(fastifyBoilerplate);\n"],"mappings":"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"}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@autofleet/fastify-boilerplate",
|
|
3
|
-
"version": "2.1.
|
|
3
|
+
"version": "2.1.14",
|
|
4
4
|
"type": "module",
|
|
5
5
|
"main": "./dist/index.js",
|
|
6
6
|
"module": "./dist/index.js",
|
|
@@ -46,7 +46,7 @@
|
|
|
46
46
|
"fastify-zod-openapi": "^5.5.0",
|
|
47
47
|
"zod": "^4.2.1",
|
|
48
48
|
"@autofleet/errors": "^3.1.41",
|
|
49
|
-
"@autofleet/nitur": "^2.
|
|
49
|
+
"@autofleet/nitur": "^2.4.0",
|
|
50
50
|
"@autofleet/zehut": "^4.8.0"
|
|
51
51
|
},
|
|
52
52
|
"scripts": {
|