diff --git a/docs/api.md b/docs/api.md
index ac6c45123..3efe63782 100644
--- a/docs/api.md
+++ b/docs/api.md
@@ -107,6 +107,7 @@ Default: `100`
Option to limit stringification of properties/elements when logging a specific object/array with circular references.
+
#### `mixin` (Function):
Default: `undefined`
@@ -130,7 +131,8 @@ logger.info('world')
```
The result of `mixin()` is supposed to be a _new_ object. For performance reason, the object returned by `mixin()` will be mutated by pino.
-In the following example, passing `mergingObject` argument to the first `info` call will mutate the global `mixin` object:
+In the following example, passing `mergingObject` argument to the first `info` call will mutate the global `mixin` object by default:
+(* See [`mixinMergeStrategy` option](#opt-mixin-merge-strategy)):
```js
const mixin = {
appName: 'My app'
@@ -154,6 +156,67 @@ logger.info('Message 2')
If the `mixin` feature is being used merely to add static metadata to each log message,
then a [child logger ⇗](/docs/child-loggers.md) should be used instead.
+
+#### `mixinMergeStrategy` (Function):
+
+Default: `undefined`
+
+If provided, the `mixinMergeStrategy` function is called each time one of the active
+logging methods is called. The first parameter is the value `mergeObject` or an empty object,
+the second parameter is the value resulting from `mixin()` (* See [`mixin` option](#opt-mixin) or an empty object.
+The function must synchronously return an object.
+
+```js
+// Default strategy, `mergeObject` has priority
+const logger = pino({
+ mixin() {
+ return { tag: 'docker' }
+ },
+ // mixinMergeStrategy(mergeObject, mixinObject) {
+ // return Object.assign(mixinMeta, mergeObject)
+ // }
+})
+
+logger.info({
+ tag: 'local'
+}, 'Message')
+// {"level":30,"time":1591195061437,"pid":16012,"hostname":"x","tag":"local","msg":"Message"}
+```
+
+```js
+// Custom mutable strategy, `mixin` has priority
+const logger = pino({
+ mixin() {
+ return { tag: 'k8s' }
+ },
+ mixinMergeStrategy(mergeObject, mixinObject) {
+ return Object.assign(mergeObject, mixinObject)
+ }
+})
+
+logger.info({
+ tag: 'local'
+}, 'Message')
+// {"level":30,"time":1591195061437,"pid":16012,"hostname":"x","tag":"k8s","msg":"Message"}
+```
+
+```js
+// Custom immutable strategy, `mixin` has priority
+const logger = pino({
+ mixin() {
+ return { tag: 'k8s' }
+ },
+ mixinMergeStrategy(mergeObject, mixinObject) {
+ return Object.assign({}, mergeObject, mixinObject)
+ }
+})
+
+logger.info({
+ tag: 'local'
+}, 'Message')
+// {"level":30,"time":1591195061437,"pid":16012,"hostname":"x","tag":"k8s","msg":"Message"}
+```
+
#### `redact` (Array | Object):
diff --git a/lib/proto.js b/lib/proto.js
index ae00e38f9..445de5477 100644
--- a/lib/proto.js
+++ b/lib/proto.js
@@ -13,6 +13,7 @@ const {
mixinSym,
asJsonSym,
writeSym,
+ mixinMergeStrategySym,
timeSym,
timeSliceIndexSym,
streamSym,
@@ -158,20 +159,33 @@ function setBindings (newBindings) {
delete this[parsedChindingsSym]
}
+/**
+ * Default strategy for creating `mergeObject` from arguments and the result from `mixin()`.
+ * Fields from `mergeObject` have higher priority in this strategy.
+ *
+ * @param {Object} mergeObject The object a user has supplied to the logging function.
+ * @param {Object} mixinObject The result of the `mixin` method.
+ * @return {Object}
+ */
+function defaultMixinMergeStrategy (mergeObject, mixinObject) {
+ return Object.assign(mixinObject, mergeObject)
+}
+
function write (_obj, msg, num) {
const t = this[timeSym]()
const mixin = this[mixinSym]
+ const mixinMergeStrategy = this[mixinMergeStrategySym] || defaultMixinMergeStrategy
let obj
if (_obj === undefined || _obj === null) {
obj = mixin ? mixin({}) : {}
} else if (_obj instanceof Error) {
- obj = Object.assign(mixin ? mixin({}) : {}, { err: _obj })
+ obj = mixinMergeStrategy({ err: _obj }, mixin ? mixin(_obj) : {})
if (msg === undefined) {
msg = _obj.message
}
} else {
- obj = Object.assign(mixin ? mixin({}) : {}, _obj)
+ obj = mixinMergeStrategy(_obj, mixin ? mixin(_obj) : {})
if (msg === undefined && _obj.err) {
msg = _obj.err.message
}
diff --git a/lib/symbols.js b/lib/symbols.js
index 2845cbac0..75195ff07 100644
--- a/lib/symbols.js
+++ b/lib/symbols.js
@@ -26,6 +26,7 @@ const formatOptsSym = Symbol('pino.formatOpts')
const messageKeySym = Symbol('pino.messageKey')
const nestedKeySym = Symbol('pino.nestedKey')
const nestedKeyStrSym = Symbol('pino.nestedKeyStr')
+const mixinMergeStrategySym = Symbol('pino.mixinMergeStrategy')
const wildcardFirstSym = Symbol('pino.wildcardFirst')
@@ -64,5 +65,6 @@ module.exports = {
useOnlyCustomLevelsSym,
formattersSym,
hooksSym,
- nestedKeyStrSym
+ nestedKeyStrSym,
+ mixinMergeStrategySym
}
diff --git a/pino.d.ts b/pino.d.ts
index c46d18ab3..ce0cd51a4 100644
--- a/pino.d.ts
+++ b/pino.d.ts
@@ -31,7 +31,8 @@ import type { WorkerOptions } from "worker_threads";
type ThreadStream = any
type TimeFn = () => string;
-type MixinFn = () => object;
+type MixinFn = (mergeObject: object) => object;
+type MixinMergeStrategyFn = (mergeObject: object, mixinObject: object) => object;
type CustomLevelLogger = Options extends { customLevels: Record } ? Record : Record
@@ -107,7 +108,7 @@ interface LoggerExtras extends EventEmitter {
declare namespace pino {
//// Exported types and interfaces
-
+
interface BaseLogger {
/**
* Set this property to the desired logging level. In order of priority, available levels are:
@@ -125,7 +126,7 @@ declare namespace pino {
* You can pass `'silent'` to disable logging.
*/
level: pino.LevelWithSilent | string;
-
+
/**
* Log at `'fatal'` level the given msg. If the first argument is an object, all its properties will be included in the JSON line.
* If more args follows `msg`, these will be used to format `msg` using `util.format`.
@@ -199,14 +200,14 @@ declare namespace pino {
type SerializerFn = (value: any) => any;
type WriteFn = (o: object) => void;
-
+
type LevelChangeEventListener = (
lvl: LevelWithSilent | string,
val: number,
prevLvl: LevelWithSilent | string,
prevVal: number,
) => void;
-
+
type LogDescriptor = Record;
type Logger = BaseLogger & LoggerExtras & CustomLevelLogger;
@@ -331,6 +332,14 @@ declare namespace pino {
*/
mixin?: MixinFn;
+ /**
+ * If provided, the `mixinMergeStrategy` function is called each time one of the active
+ * logging methods is called. The first parameter is the value `mergeObject` or an empty object,
+ * the second parameter is the value resulting from `mixin()` or an empty object.
+ * The function must synchronously return an object.
+ */
+ mixinMergeStrategy?: MixinMergeStrategyFn
+
/**
* As an array, the redact option specifies paths that should have their values redacted from any log output.
*
@@ -666,12 +675,12 @@ declare namespace pino {
readonly formattersSym: unique symbol;
readonly hooksSym: unique symbol;
};
-
+
/**
* Exposes the Pino package version. Also available on the logger instance.
*/
export const version: string;
-
+
/**
* Provides functions for generating the timestamp property in the log output. You can set the `timestamp` option during
* initialization to one of these functions to adjust the output format. Alternatively, you can specify your own time function.
@@ -696,7 +705,7 @@ declare namespace pino {
*/
isoTime: TimeFn;
};
-
+
//// Exported functions
/**
@@ -750,7 +759,7 @@ declare function pino(options
* @returns a new logger instance.
*/
declare function pino(options: Options, stream: DestinationStream): Logger;
-
+
// Pass through all the top-level exports, allows `import {version} from "pino"`
// Constants and functions
diff --git a/pino.js b/pino.js
index 315e8d46f..2afe6d267 100644
--- a/pino.js
+++ b/pino.js
@@ -39,7 +39,8 @@ const {
useOnlyCustomLevelsSym,
formattersSym,
hooksSym,
- nestedKeyStrSym
+ nestedKeyStrSym,
+ mixinMergeStrategySym
} = symbols
const { epochTime, nullTime } = time
const { pid } = process
@@ -94,6 +95,7 @@ function pino (...args) {
level,
customLevels,
mixin,
+ mixinMergeStrategy,
useOnlyCustomLevels,
formatters,
hooks,
@@ -166,6 +168,7 @@ function pino (...args) {
[nestedKeyStrSym]: nestedKey ? `,${JSON.stringify(nestedKey)}:{` : '',
[serializersSym]: serializers,
[mixinSym]: mixin,
+ [mixinMergeStrategySym]: mixinMergeStrategy,
[chindingsSym]: chindings,
[formattersSym]: allFormatters,
[hooksSym]: hooks,
diff --git a/test/mixin-merge-strategy.test.js b/test/mixin-merge-strategy.test.js
new file mode 100644
index 000000000..3c6dfe45d
--- /dev/null
+++ b/test/mixin-merge-strategy.test.js
@@ -0,0 +1,55 @@
+'use strict'
+
+const { test } = require('tap')
+const { sink, once } = require('./helper')
+const pino = require('../')
+
+const level = 50
+const name = 'error'
+
+test('default merge strategy', async ({ ok, same }) => {
+ const stream = sink()
+ const instance = pino({
+ base: {},
+ mixin () {
+ return { tag: 'k8s' }
+ }
+ }, stream)
+ instance.level = name
+ instance[name]({
+ tag: 'local'
+ }, 'test')
+ const result = await once(stream, 'data')
+ ok(new Date(result.time) <= new Date(), 'time is greater than Date.now()')
+ delete result.time
+ same(result, {
+ level,
+ msg: 'test',
+ tag: 'local'
+ })
+})
+
+test('custom merge strategy with mixin priority', async ({ ok, same }) => {
+ const stream = sink()
+ const instance = pino({
+ base: {},
+ mixin () {
+ return { tag: 'k8s' }
+ },
+ mixinMergeStrategy (mergeObject, mixinObject) {
+ return Object.assign(mergeObject, mixinObject)
+ }
+ }, stream)
+ instance.level = name
+ instance[name]({
+ tag: 'local'
+ }, 'test')
+ const result = await once(stream, 'data')
+ ok(new Date(result.time) <= new Date(), 'time is greater than Date.now()')
+ delete result.time
+ same(result, {
+ level,
+ msg: 'test',
+ tag: 'k8s'
+ })
+})