UNPKG

apitally

Version:

Simple API monitoring & analytics for REST APIs built with Express, Fastify, NestJS, AdonisJS, Hono, H3, Elysia, Hapi, and Koa.

1 lines 11.7 kB
{"version":3,"sources":["../../src/koa/middleware.ts"],"sourcesContent":["import Koa from \"koa\";\nimport { AsyncLocalStorage } from \"node:async_hooks\";\n\nimport { ApitallyClient } from \"../common/client.js\";\nimport { consumerFromStringOrObject } from \"../common/consumerRegistry.js\";\nimport { getPackageVersion } from \"../common/packageVersions.js\";\nimport type { LogRecord } from \"../common/requestLogger.js\";\nimport { convertBody, convertHeaders } from \"../common/requestLogger.js\";\nimport {\n ApitallyConfig,\n ApitallyConsumer,\n PathInfo,\n StartupData,\n} from \"../common/types.js\";\nimport { patchConsole, patchWinston } from \"../loggers/index.js\";\n\nexport function useApitally(app: Koa, config: ApitallyConfig) {\n const client = new ApitallyClient(config);\n const middleware = getMiddleware(client);\n app.use(middleware);\n\n const setStartupData = (attempt: number = 1) => {\n const appInfo = getAppInfo(app, config.appVersion);\n if (appInfo.paths.length > 0 || attempt >= 10) {\n client.setStartupData(appInfo);\n client.startSync();\n } else {\n setTimeout(() => setStartupData(attempt + 1), 500);\n }\n };\n setTimeout(() => setStartupData(), 500);\n}\n\nfunction getMiddleware(client: ApitallyClient) {\n const logsContext = new AsyncLocalStorage<LogRecord[]>();\n\n if (client.requestLogger.config.captureLogs) {\n patchConsole(logsContext);\n patchWinston(logsContext);\n }\n\n return async (ctx: Koa.Context, next: Koa.Next) => {\n if (!client.isEnabled() || ctx.request.method.toUpperCase() === \"OPTIONS\") {\n await next();\n return;\n }\n\n await logsContext.run([], async () => {\n let path: string | undefined;\n let statusCode: number | undefined;\n let serverError: Error | undefined;\n const startTime = performance.now();\n try {\n await next();\n } catch (error: any) {\n path = getPath(ctx);\n statusCode = error.statusCode || error.status || 500;\n if (path && statusCode === 500 && error instanceof Error) {\n serverError = error;\n client.serverErrorCounter.addServerError({\n consumer: getConsumer(ctx)?.identifier,\n method: ctx.request.method,\n path,\n type: error.name,\n msg: error.message,\n traceback: error.stack || \"\",\n });\n }\n throw error;\n } finally {\n const responseTime = performance.now() - startTime;\n const consumer = getConsumer(ctx);\n client.consumerRegistry.addOrUpdateConsumer(consumer);\n if (!path) {\n path = getPath(ctx);\n }\n if (path) {\n try {\n client.requestCounter.addRequest({\n consumer: consumer?.identifier,\n method: ctx.request.method,\n path,\n statusCode: statusCode || ctx.response.status,\n responseTime,\n requestSize: ctx.request.length,\n responseSize: ctx.response.length,\n });\n } catch (error) {\n client.logger.error(\n \"Error while logging request in Apitally middleware.\",\n { context: ctx, error },\n );\n }\n }\n if (client.requestLogger.enabled) {\n const logs = logsContext.getStore();\n client.requestLogger.logRequest(\n {\n timestamp: Date.now() / 1000,\n method: ctx.request.method,\n path,\n url: ctx.request.href,\n headers: convertHeaders(ctx.request.headers),\n size: ctx.request.length,\n consumer: consumer?.identifier,\n body: convertBody(\n ctx.request.body,\n ctx.request.get(\"content-type\"),\n ),\n },\n {\n statusCode: statusCode || ctx.response.status,\n responseTime: responseTime / 1000,\n headers: convertHeaders(ctx.response.headers),\n size: ctx.response.length,\n body: convertBody(\n ctx.response.body,\n ctx.response.get(\"content-type\"),\n ),\n },\n serverError,\n logs,\n );\n }\n }\n });\n };\n}\n\nfunction getPath(ctx: Koa.Context) {\n return ctx._matchedRoute || ctx.routePath; // _matchedRoute is set by koa-router, routePath is set by koa-route\n}\n\nexport function setConsumer(\n ctx: Koa.Context,\n consumer: ApitallyConsumer | string | null | undefined,\n) {\n ctx.state.apitallyConsumer = consumer || undefined;\n}\n\nfunction getConsumer(ctx: Koa.Context) {\n if (ctx.state.apitallyConsumer) {\n return consumerFromStringOrObject(ctx.state.apitallyConsumer);\n } else if (ctx.state.consumerIdentifier) {\n // For backwards compatibility\n process.emitWarning(\n \"The consumerIdentifier property on the ctx.state object is deprecated. Use apitallyConsumer instead.\",\n \"DeprecationWarning\",\n );\n return consumerFromStringOrObject(ctx.state.consumerIdentifier);\n }\n return null;\n}\n\nfunction getAppInfo(app: Koa, appVersion?: string): StartupData {\n const versions: Array<[string, string]> = [\n [\"nodejs\", process.version.replace(/^v/, \"\")],\n ];\n const koaVersion = getPackageVersion(\"koa\");\n const apitallyVersion = getPackageVersion(\"../..\");\n if (koaVersion) {\n versions.push([\"koa\", koaVersion]);\n }\n if (apitallyVersion) {\n versions.push([\"apitally\", apitallyVersion]);\n }\n if (appVersion) {\n versions.push([\"app\", appVersion]);\n }\n return {\n paths: listEndpoints(app),\n versions: Object.fromEntries(versions),\n client: \"js:koa\",\n };\n}\n\nfunction isKoaRouterMiddleware(middleware: any) {\n return (\n typeof middleware === \"function\" &&\n middleware.router &&\n Array.isArray(middleware.router.stack)\n );\n}\n\nfunction listEndpoints(app: Koa) {\n const endpoints: Array<PathInfo> = [];\n app.middleware.forEach((middleware: any) => {\n if (isKoaRouterMiddleware(middleware)) {\n middleware.router.stack.forEach((layer: any) => {\n if (layer.methods && layer.methods.length > 0) {\n layer.methods.forEach((method: string) => {\n if (![\"HEAD\", \"OPTIONS\"].includes(method.toUpperCase())) {\n endpoints.push({\n method: method.toUpperCase(),\n path: layer.path,\n });\n }\n });\n }\n });\n }\n });\n return endpoints;\n}\n"],"mappings":"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","names":["useApitally","app","config","client","ApitallyClient","middleware","getMiddleware","use","setStartupData","attempt","appInfo","getAppInfo","appVersion","paths","length","startSync","setTimeout","logsContext","AsyncLocalStorage","requestLogger","captureLogs","patchConsole","patchWinston","ctx","next","isEnabled","request","method","toUpperCase","run","path","statusCode","serverError","startTime","performance","now","error","getPath","status","Error","serverErrorCounter","addServerError","consumer","getConsumer","identifier","type","name","msg","message","traceback","stack","responseTime","consumerRegistry","addOrUpdateConsumer","requestCounter","addRequest","response","requestSize","responseSize","logger","context","enabled","logs","getStore","logRequest","timestamp","Date","url","href","headers","convertHeaders","size","body","convertBody","get","_matchedRoute","routePath","setConsumer","state","apitallyConsumer","undefined","consumerFromStringOrObject","consumerIdentifier","process","emitWarning","versions","version","replace","koaVersion","getPackageVersion","apitallyVersion","push","listEndpoints","Object","fromEntries","isKoaRouterMiddleware","router","Array","isArray","endpoints","forEach","layer","methods","includes"]}