From 34c9e43747b6e2b31b1fbbf8a65f1d110d45a6b2 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Luismi=20Rami=CC=81rez?= Date: Mon, 14 Oct 2024 14:12:43 +0200 Subject: [PATCH] Add Pino transport A Pino transport is now exposed from the package (`AppsignalPinoTransport`). It requires an initialized AppSignal client to work, and also accepts a group as an optional parameter. Attributes are flattened so no data is lost. Usage: ```js import pino from "pino" import { Appsignal, AppsignalPinoTransport } from "@appsignal/nodejs" const logger = pino( AppsignalPinoTransport({ client: Appsignal.client, group: "my-group", }) ) ``` --- .changesets/add-pino-transport.md | 18 ++++++ package-lock.json | 17 +++++ package.json | 1 + src/index.ts | 3 + src/pino_transport.ts | 100 ++++++++++++++++++++++++++++++ 5 files changed, 139 insertions(+) create mode 100644 .changesets/add-pino-transport.md create mode 100644 src/pino_transport.ts diff --git a/.changesets/add-pino-transport.md b/.changesets/add-pino-transport.md new file mode 100644 index 00000000..3fcf0c0d --- /dev/null +++ b/.changesets/add-pino-transport.md @@ -0,0 +1,18 @@ +--- +bump: patch +type: add +--- + +A Pino transport is now available. If Pino is your main logger, you can now use the AppSignal pino transport to send those logs to AppSignal. + +```js +import pino from "pino" +import { Appsignal, AppsignalPinoTransport } from "@appsignal/nodejs" + +const logger = pino( + AppsignalPinoTransport({ + client: Appsignal.client, + group: "application", + }) +) +``` diff --git a/package-lock.json b/package-lock.json index 4a2c68bb..ba2d14e0 100644 --- a/package-lock.json +++ b/package-lock.json @@ -38,6 +38,7 @@ "@prisma/instrumentation": ">= 5.11.0 < 5.14.0", "node-addon-api": "^3.1.0", "node-gyp": "^10.0.0", + "pino-abstract-transport": "^2.0.0", "tslib": "^2.0.3", "winston": "^3.6.0" }, @@ -10700,6 +10701,14 @@ "node": ">=4" } }, + "node_modules/pino-abstract-transport": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/pino-abstract-transport/-/pino-abstract-transport-2.0.0.tgz", + "integrity": "sha512-F63x5tizV6WCh4R6RHyi2Ml+M70DNRXt/+HANowMflpgGFMAym/VKm6G7ZOQRjqN7XbGxK1Lg9t6ZrtzOaivMw==", + "dependencies": { + "split2": "^4.0.0" + } + }, "node_modules/pirates": { "version": "4.0.6", "resolved": "https://registry.npmjs.org/pirates/-/pirates-4.0.6.tgz", @@ -11612,6 +11621,14 @@ "dev": true, "license": "CC0-1.0" }, + "node_modules/split2": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/split2/-/split2-4.2.0.tgz", + "integrity": "sha512-UcjcJOWknrNkF6PLX83qcHM6KHgVKNkV62Y8a5uYDVv9ydGQVwAHMKqHdJje1VTWpljG0WYpCDhrCdAOYH4TWg==", + "engines": { + "node": ">= 10.x" + } + }, "node_modules/sprintf-js": { "version": "1.1.3", "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.1.3.tgz", diff --git a/package.json b/package.json index a3bd2ebb..b8165be6 100644 --- a/package.json +++ b/package.json @@ -40,6 +40,7 @@ "@prisma/instrumentation": ">= 5.11.0 < 5.14.0", "node-addon-api": "^3.1.0", "node-gyp": "^10.0.0", + "pino-abstract-transport": "^2.0.0", "tslib": "^2.0.3", "winston": "^3.6.0" }, diff --git a/src/index.ts b/src/index.ts index 8fb58daf..a1049a52 100644 --- a/src/index.ts +++ b/src/index.ts @@ -13,3 +13,6 @@ export { WinstonTransport } from "./winston_transport" export * from "./helpers" export * as checkIn from "./check_in" export { Heartbeat, heartbeat } from "./heartbeat" + +import AppsignalPinoTransport from "./pino_transport" +export { AppsignalPinoTransport } diff --git a/src/pino_transport.ts b/src/pino_transport.ts new file mode 100644 index 00000000..cb39d471 --- /dev/null +++ b/src/pino_transport.ts @@ -0,0 +1,100 @@ +import build from "pino-abstract-transport" +import { Client } from "./client" + +type PinoTransportOptions = { + client: Client + group: string +} + +const appsignalPinoTransport = ({ + client, + group = "app" +}: PinoTransportOptions) => { + let buffer: Record[] = [] + let timeout: NodeJS.Timeout | null = null + + return build( + async (source: any) => { + for await (const obj of source) { + buffer.push(parseInfo(obj, group)) + + if (!timeout) { + timeout = setTimeout(flush, 1000) + } + } + }, + { close: flush } + ) + + async function flush() { + if (timeout) { + clearTimeout(timeout) + } + timeout = null + + if (buffer) { + const data = buffer + buffer = [] + data.forEach(line => sendLogs(line, client)) + } + } + + async function sendLogs(data: Record, client: Client) { + client.extension.log( + data.group || "app", + data.severity, + 0, + data.msg, + data.attributes + ) + } +} + +function parseInfo( + obj: Record, + group: string +): Record { + const { hostname, level, msg, ...attributes } = obj + + return { + severity: getSeverity(level), + hostname, + group, + msg, + attributes: flattenAttributes(attributes) + } +} + +function flattenAttributes( + attributes: Record, + prefix = "" +): Record { + let result: Record = {} + + for (const key in attributes) { + const newKey = prefix ? `${prefix}.${key}` : key + + if ( + typeof attributes[key] === "object" && + attributes[key] !== null && + !Array.isArray(attributes[key]) + ) { + const flattened = flattenAttributes(attributes[key], newKey) + result = { ...result, ...flattened } + } else { + result[newKey] = attributes[key] + } + } + + return result +} + +function getSeverity(level: number): number { + if (level >= 50) return 6 + if (level >= 40) return 5 + if (level >= 30) return 3 + if (level >= 20) return 2 + return 1 +} + +export = appsignalPinoTransport