-
Notifications
You must be signed in to change notification settings - Fork 544
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat(instrumentation-pino): add log sending to Logs Bridge API (#2249)
* feat(instrumentation-pino): add log sending to Logs Bridge API * refactor tests (mostly from separate #2247 PR); lint:fix; some in-progress changes * remove some old dev/debug code * feat!: make it so re-enabled instr after creating a logger will NOT change behaviour for that logger This removes functionality that was there before, so technically could be breaking. The motivation is to have the contract of pino instrumentation be cleaner. With this change a pino Logger instance will not be touched if the PinoInstrumentation is disabled. I.e. we are hands-off as much as possible when disabled. Before this change, even when disabled, the instrumentation would tweak the pino Logger config to have a no-op mixin. If the instrumentation was later enabled, then the mixin would become active (adding trace_id et al fields in a span context). The coming "log sending" to the Logs Bridge API will *not* support this "work if instrumentation is re-enabled later", so I think it is clearer if neither "log sending" nor "log correlation" support this. We can back this out if we think it is important to support a possible future feature of the SDK doing live enabling/disabling of individual instrumentations. * impl disableLogCorrelation config; undo the previous commit so that log-correlation *will* follow the live instr enable/disable state * log sending: first tests; change impl to use pino.multistream * edge case tests; almost complete * more tests and a fix for 'useOnlyCustomLevels: true' usage * lint:fix * refactor some code out to utils file * add some internal docs * update readme * fix lint * avoid a possible flaky error if using pino 'unixTime' and logging in the first half-second since process start Effectively the issue is that this sometimes returns true: node -e 'console.log(Math.round(Date.now() / 1e3) * 1e3 < performance.timeOrigin)' * limit log-sending to pino@7 and later because that's when pino.multistream was added * lint:fix * discuss pino-opentelemetry-transport alternative * fix a mis-merge * update changed deps to their new latest * typo in README Co-authored-by: Hector Hernandez <[email protected]> --------- Co-authored-by: Hector Hernandez <[email protected]> Co-authored-by: Marc Pichler <[email protected]>
- Loading branch information
1 parent
d9d558f
commit 055ef41
Showing
7 changed files
with
784 additions
and
72 deletions.
There are no files selected for viewing
Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.
Oops, something went wrong.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -28,6 +28,7 @@ import { | |
} from '@opentelemetry/instrumentation'; | ||
import { PinoInstrumentationConfig } from './types'; | ||
import { PACKAGE_NAME, PACKAGE_VERSION } from './version'; | ||
import { getTimeConverter, OTelPinoStream } from './log-sending-utils'; | ||
|
||
const pinoVersions = ['>=5.14.0 <10']; | ||
|
||
|
@@ -48,30 +49,73 @@ export class PinoInstrumentation extends InstrumentationBase { | |
const isESM = module[Symbol.toStringTag] === 'Module'; | ||
const moduleExports = isESM ? module.default : module; | ||
const instrumentation = this; | ||
|
||
const patchedPino = Object.assign((...args: unknown[]) => { | ||
if (args.length === 0) { | ||
return moduleExports({ | ||
mixin: instrumentation._getMixinFunction(), | ||
}); | ||
const config = instrumentation.getConfig(); | ||
const isEnabled = instrumentation.isEnabled(); | ||
|
||
const logger = moduleExports(...args); | ||
|
||
// Setup "log correlation" -- injection of `trace_id` et al fields. | ||
// Note: If the Pino logger is configured with `nestedKey`, then | ||
// the `trace_id` et al fields added by `otelMixin` will be nested | ||
// as well. https://getpino.io/#/docs/api?id=mixin-function | ||
const otelMixin = instrumentation._getMixinFunction(); | ||
const mixinSym = moduleExports.symbols.mixinSym; | ||
const origMixin = logger[mixinSym]; | ||
if (origMixin === undefined) { | ||
logger[mixinSym] = otelMixin; | ||
} else { | ||
logger[mixinSym] = (ctx: object, level: number) => { | ||
return Object.assign( | ||
otelMixin(ctx, level), | ||
origMixin(ctx, level) | ||
); | ||
}; | ||
} | ||
|
||
if (args.length === 1) { | ||
// eslint-disable-next-line @typescript-eslint/no-explicit-any | ||
const optsOrStream = args[0] as any; | ||
if ( | ||
typeof optsOrStream === 'string' || | ||
typeof optsOrStream?.write === 'function' | ||
) { | ||
args.splice(0, 0, { | ||
mixin: instrumentation._getMixinFunction(), | ||
}); | ||
return moduleExports(...args); | ||
} | ||
} | ||
// Setup "log sending" -- sending log records to the Logs API. | ||
// This depends on `pino.multistream`, which was added in v7.0.0. | ||
if ( | ||
isEnabled && | ||
!config.disableLogSending && | ||
typeof moduleExports.multistream === 'function' | ||
) { | ||
const otelTimestampFromTime = getTimeConverter( | ||
logger, | ||
moduleExports | ||
); | ||
const otelStream = new OTelPinoStream({ | ||
messageKey: logger[moduleExports.symbols.messageKeySym], | ||
levels: logger.levels, | ||
otelTimestampFromTime, | ||
}); | ||
(otelStream as any)[Symbol.for('pino.metadata')] = true; // for `stream.lastLevel` | ||
|
||
// An error typically indicates a Pino bug, or logger configuration | ||
// bug. `diag.warn` *once* for the first error on the assumption | ||
// subsequent ones stem from the same bug. | ||
otelStream.once('unknown', (line, err) => { | ||
instrumentation._diag.warn( | ||
'could not send pino log line (will only log first occurrence)', | ||
{ line, err } | ||
); | ||
}); | ||
|
||
args[0] = instrumentation._combineOptions(args[0]); | ||
// Use pino's own `multistream` to send to the original stream and | ||
// to the OTel Logs API/SDK. | ||
// https://getpino.io/#/docs/api?id=pinomultistreamstreamsarray-opts-gt-multistreamres | ||
const origStream = logger[moduleExports.symbols.streamSym]; | ||
logger[moduleExports.symbols.streamSym] = moduleExports.multistream( | ||
[ | ||
{ level: logger.level, stream: origStream }, | ||
{ level: logger.level, stream: otelStream }, | ||
], | ||
{ levels: logger.levels.values } | ||
); | ||
} | ||
|
||
return moduleExports(...args); | ||
return logger; | ||
}, moduleExports); | ||
|
||
if (typeof patchedPino.pino === 'function') { | ||
|
@@ -80,6 +124,7 @@ export class PinoInstrumentation extends InstrumentationBase { | |
if (typeof patchedPino.default === 'function') { | ||
patchedPino.default = patchedPino; | ||
} | ||
/* istanbul ignore if */ | ||
if (isESM) { | ||
if (module.pino) { | ||
// This was added in [email protected] (https://github.com/pinojs/pino/pull/936). | ||
|
@@ -122,7 +167,10 @@ export class PinoInstrumentation extends InstrumentationBase { | |
private _getMixinFunction() { | ||
const instrumentation = this; | ||
return function otelMixin(_context: object, level: number) { | ||
if (!instrumentation.isEnabled()) { | ||
if ( | ||
!instrumentation.isEnabled() || | ||
instrumentation.getConfig().disableLogCorrelation | ||
) { | ||
return {}; | ||
} | ||
|
||
|
@@ -151,27 +199,4 @@ export class PinoInstrumentation extends InstrumentationBase { | |
return record; | ||
}; | ||
} | ||
|
||
private _combineOptions(options?: any) { | ||
if (options === undefined) { | ||
return { mixin: this._getMixinFunction() }; | ||
} | ||
|
||
if (options.mixin === undefined) { | ||
options.mixin = this._getMixinFunction(); | ||
return options; | ||
} | ||
|
||
const originalMixin = options.mixin; | ||
const otelMixin = this._getMixinFunction(); | ||
|
||
options.mixin = (context: object, level: number) => { | ||
return Object.assign( | ||
otelMixin(context, level), | ||
originalMixin(context, level) | ||
); | ||
}; | ||
|
||
return options; | ||
} | ||
} |
Oops, something went wrong.