From 10536b0680b10075e5919979ad2473a6406ee54d Mon Sep 17 00:00:00 2001 From: Matteo Collina Date: Sun, 9 Dec 2018 16:45:40 +0100 Subject: [PATCH 1/5] Make pino.destination sync --- lib/tools.js | 4 ++-- package.json | 10 +++++----- pino.js | 4 ++-- test/destination-exit.test.js | 23 +++++++++++++++++++++++ test/final.test.js | 22 +++++++++++----------- test/fixtures/destination-exit.js | 8 ++++++++ 6 files changed, 51 insertions(+), 20 deletions(-) create mode 100644 test/destination-exit.test.js create mode 100644 test/fixtures/destination-exit.js diff --git a/lib/tools.js b/lib/tools.js index a553c877e..ad4086a78 100644 --- a/lib/tools.js +++ b/lib/tools.js @@ -267,8 +267,8 @@ function hasBeenTampered (stream) { return stream.write !== stream.constructor.prototype.write } -function buildSafeSonicBoom (dest, buffer = 0) { - const stream = new SonicBoom(dest, buffer) +function buildSafeSonicBoom (dest, buffer = 0, sync = true) { + const stream = new SonicBoom(dest, buffer, sync) stream.on('error', filterBrokenPipe) return stream diff --git a/package.json b/package.json index f5099f5f3..3f76941ba 100644 --- a/package.json +++ b/package.json @@ -69,7 +69,7 @@ "fresh-require": "^1.0.3", "log": "^1.4.0", "loglevel": "^1.6.1", - "pino-pretty": "^2.2.4", + "pino-pretty": "^2.4.0", "pre-commit": "^1.2.2", "proxyquire": "^2.1.0", "qodaa": "^1.0.1", @@ -79,17 +79,17 @@ "steed": "^1.1.3", "tap": "^12.1.0", "tape": "^4.9.0", - "through2": "^2.0.5", + "through2": "^3.0.0", "winston": "^3.1.0" }, "dependencies": { "fast-json-parse": "^1.0.3", - "fast-redact": "^1.4.0", + "fast-redact": "^1.4.2", "fast-safe-stringify": "^2.0.6", - "flatstr": "^1.0.5", + "flatstr": "^1.0.9", "pino-std-serializers": "^2.3.0", "pump": "^3.0.0", "quick-format-unescaped": "^3.0.0", - "sonic-boom": "^0.6.3" + "sonic-boom": "^0.7.1" } } diff --git a/pino.js b/pino.js index 0600ca1da..072d1e8d3 100644 --- a/pino.js +++ b/pino.js @@ -119,8 +119,8 @@ function pino (...args) { return instance } -pino.extreme = (dest = process.stdout.fd) => buildSafeSonicBoom(dest, 4096) -pino.destination = (dest = process.stdout.fd) => buildSafeSonicBoom(dest) +pino.extreme = (dest = process.stdout.fd) => buildSafeSonicBoom(dest, 4096, false) +pino.destination = (dest = process.stdout.fd) => buildSafeSonicBoom(dest, 0, true) pino.final = final pino.levels = mappings() diff --git a/test/destination-exit.test.js b/test/destination-exit.test.js new file mode 100644 index 000000000..7fdd1f5fd --- /dev/null +++ b/test/destination-exit.test.js @@ -0,0 +1,23 @@ +'use strict' + +const { test } = require('tap') +const { join } = require('path') +const execa = require('execa') +const writer = require('flush-write-stream') +const { once } = require('./helper') + +// https://github.com/pinojs/pino/issues/542 +test('log everything when calling process.exit(0)', async ({ isNot }) => { + var actual = '' + const child = execa(process.argv[0], [join(__dirname, 'fixtures', 'destination-exit.js')]) + + child.stdout.pipe(writer((s, enc, cb) => { + actual += s + cb() + })) + + await once(child, 'close') + + isNot(actual.match(/hello/), null) + isNot(actual.match(/world/), null) +}) diff --git a/test/final.test.js b/test/final.test.js index 7daeba521..2b757dce1 100644 --- a/test/final.test.js +++ b/test/final.test.js @@ -24,13 +24,13 @@ test('throws if the supplied handler is not a function', async ({ throws }) => { }, Error('if supplied, the handler parameter should be a function')) }) -test('throws if not supplied logger with pino.destination or pino.extreme instance', async ({ throws, doesNotThrow }) => { +test('throws if not supplied logger with pino.extreme or pino.extreme instance', async ({ throws, doesNotThrow }) => { throws(() => { pino.final(pino(fs.createWriteStream('/dev/null')), () => {}) }, Error('final requires a stream that has a flushSync method, such as pino.destination and pino.extreme')) doesNotThrow(() => { - pino.final(pino(pino.destination()), () => {}) + pino.final(pino(pino.extreme()), () => {}) }) doesNotThrow(() => { @@ -39,11 +39,11 @@ test('throws if not supplied logger with pino.destination or pino.extreme instan }) test('returns an exit listener function', async ({ is }) => { - is(typeof pino.final(pino(pino.destination()), () => {}), 'function') + is(typeof pino.final(pino(pino.extreme()), () => {}), 'function') }) test('listener function immediately sync flushes when fired', async ({ pass, fail }) => { - const dest = pino.destination('/dev/null') + const dest = pino.extreme('/dev/null') var passed = false dest.flushSync = () => { passed = true @@ -55,14 +55,14 @@ test('listener function immediately sync flushes when fired', async ({ pass, fai }) test('swallows the non-ready error', async ({ doesNotThrow }) => { - const dest = pino.destination('/dev/null') + const dest = pino.extreme('/dev/null') doesNotThrow(() => { pino.final(pino(dest), () => {})() }) }) test('listener function triggers handler function parameter', async ({ pass, fail }) => { - const dest = pino.destination('/dev/null') + const dest = pino.extreme('/dev/null') var passed = false pino.final(pino(dest), () => { passed = true @@ -73,14 +73,14 @@ test('listener function triggers handler function parameter', async ({ pass, fai }) test('passes any error to the handler', async ({ is }) => { - const dest = pino.destination('/dev/null') + const dest = pino.extreme('/dev/null') pino.final(pino(dest), (err) => { is(err.message, 'test') })(Error('test')) }) test('passes a specialized final logger instance', async ({ is, isNot, error }) => { - const dest = pino.destination('/dev/null') + const dest = pino.extreme('/dev/null') const logger = pino(dest) pino.final(logger, (err, finalLogger) => { error(err) @@ -104,7 +104,7 @@ test('passes a specialized final logger instance', async ({ is, isNot, error }) }) test('returns a specialized final logger instance if no handler is passed', async ({ is, isNot }) => { - const dest = pino.destination('/dev/null') + const dest = pino.extreme('/dev/null') const logger = pino(dest) const finalLogger = pino.final(logger) is(typeof finalLogger.trace, 'function') @@ -126,7 +126,7 @@ test('returns a specialized final logger instance if no handler is passed', asyn }) test('final logger instances synchronously flush after a log method call', async ({ pass, fail, error }) => { - const dest = pino.destination('/dev/null') + const dest = pino.extreme('/dev/null') const logger = pino(dest) var passed = false var count = 0 @@ -146,7 +146,7 @@ test('final logger instances synchronously flush after a log method call', async }) test('also instruments custom log methods', async ({ pass, fail, error }) => { - const dest = pino.destination('/dev/null') + const dest = pino.extreme('/dev/null') const logger = pino({ customLevels: { foo: 35 diff --git a/test/fixtures/destination-exit.js b/test/fixtures/destination-exit.js new file mode 100644 index 000000000..c14ab07f0 --- /dev/null +++ b/test/fixtures/destination-exit.js @@ -0,0 +1,8 @@ +global.process = { __proto__: process, pid: 123456 } +Date.now = function () { return 1459875739796 } +require('os').hostname = function () { return 'abcdefghijklmnopqr' } +var pino = require(require.resolve('./../../')) +var logger = pino({}, pino.destination(1)) +logger.info('hello') +logger.info('world') +process.exit(0) From e1c9528590f9723a46f083a667e9d4c22592101c Mon Sep 17 00:00:00 2001 From: Matteo Collina Date: Tue, 11 Dec 2018 11:06:45 +0100 Subject: [PATCH 2/5] Remove lambda hack, as it is not needed anymore with sync destination --- lib/tools.js | 7 +------ test/stdout-protection.test.js | 23 ----------------------- 2 files changed, 1 insertion(+), 29 deletions(-) diff --git a/lib/tools.js b/lib/tools.js index ad4086a78..c297f109b 100644 --- a/lib/tools.js +++ b/lib/tools.js @@ -21,11 +21,6 @@ const { streamSym } = require('./symbols') -// FIXME we need to check if we are using AWS Lambda -// because we cannot use pino.destination/Sonic Boom -// there, as data might get lost during freeze -const isAWSLambda = !!process.env.LAMBDA_TASK_ROOT - function noop () {} function genLog (z) { @@ -311,7 +306,7 @@ function createArgsNormalizer (defaultOptions) { const { enabled, prettyPrint, prettifier, messageKey } = opts if (enabled === false) opts.level = 'silent' stream = stream || process.stdout - if (stream === process.stdout && stream.fd >= 0 && !isAWSLambda && !hasBeenTampered(stream)) { + if (stream === process.stdout && stream.fd >= 0 && !hasBeenTampered(stream)) { stream = buildSafeSonicBoom(stream.fd) } if (prettyPrint) { diff --git a/test/stdout-protection.test.js b/test/stdout-protection.test.js index 963dba2dd..1a9778526 100644 --- a/test/stdout-protection.test.js +++ b/test/stdout-protection.test.js @@ -1,34 +1,11 @@ 'use strict' const { test } = require('tap') -const pino = require('..') -const { symbols } = pino const { join } = require('path') const { fork } = require('child_process') const { once } = require('./helper') const writer = require('flush-write-stream') -test('do not use SonicBoom on AWS Lambda', async (t) => { - // hack, we are on lambda now - process.env.LAMBDA_TASK_ROOT = __dirname - - // we need to clear the cache - delete require.cache[require.resolve('../')] - delete require.cache[require.resolve('../lib/tools.js')] - const pino = require('..') - - // restore in the next tick - process.nextTick(() => { - delete process.env.LAMBDA_TASK_ROOT - }) - - const logger = pino() - - // this test verifies that the default constructor of pino - // does not use SonicBoom on lambda by default - t.equal(logger[symbols.streamSym], process.stdout) -}) - test('do not use SonicBoom is someone tampered with process.stdout.write', async ({ isNot }) => { var actual = '' const child = fork(join(__dirname, 'fixtures', 'stdout-hack-protection.js'), { silent: true }) From 8fcec898451f2a5d9b3914273f3baa99663fb567 Mon Sep 17 00:00:00 2001 From: David Mark Clements Date: Tue, 11 Dec 2018 11:07:37 +0100 Subject: [PATCH 3/5] Update test/final.test.js Co-Authored-By: mcollina --- test/final.test.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/test/final.test.js b/test/final.test.js index 2b757dce1..40f4b1f80 100644 --- a/test/final.test.js +++ b/test/final.test.js @@ -24,7 +24,7 @@ test('throws if the supplied handler is not a function', async ({ throws }) => { }, Error('if supplied, the handler parameter should be a function')) }) -test('throws if not supplied logger with pino.extreme or pino.extreme instance', async ({ throws, doesNotThrow }) => { +test('throws if not supplied logger with pino.extreme instance', async ({ throws, doesNotThrow }) => { throws(() => { pino.final(pino(fs.createWriteStream('/dev/null')), () => {}) }, Error('final requires a stream that has a flushSync method, such as pino.destination and pino.extreme')) From 0184977bf108835f67d2afeb3abdf7eb00f07ae7 Mon Sep 17 00:00:00 2001 From: Matteo Collina Date: Tue, 11 Dec 2018 11:08:39 +0100 Subject: [PATCH 4/5] Updated docs --- docs/api.md | 12 +++++------- 1 file changed, 5 insertions(+), 7 deletions(-) diff --git a/docs/api.md b/docs/api.md index 41565b809..314e26d15 100644 --- a/docs/api.md +++ b/docs/api.md @@ -662,9 +662,6 @@ By default, `pino.destination` will use `process.stdout.fd` (1) as the file desc A `pino.destination` instance can also be used to reopen closed files (for example, for some log rotation scenarios), see [Reopening log files](/docs/help.md#reopening). -On AWS Lambda we recommend to call `destination.flushSync()` at the end -of each function execution to avoid losing data. - * See [`destination` parameter](#destination) * See [`sonic-boom` ⇗](https://github.com/mcollina/sonic-boom) * See [Reopening log files](/docs/help.md#reopening) @@ -707,13 +704,14 @@ or create an exit listener function. The `finalLogger` is a specialist logger that synchronously flushes on every write. This is important to guarantee final log writes, -both when using `pino.destination` or `pino.extreme` targets. +both when using `pino.extreme` target. Since final log writes cannot be guaranteed with normal Node.js streams, if the `destination` parameter of the `logger` supplied to `pino.final` -is a Node.js stream `pino.final` will throw. It's highly recommended -for both performance and safety to use `pino.destination`/`pino.extreme` -destinations. +is a Node.js stream `pino.final` will throw. + +The use of `pino.final` with `pino.destination` is not needed, as +`pino.destination` writes things synchronously. #### `pino.final(logger, handler) => Function` From 68597d34147c6f39e16b39476a970b574a0ba663 Mon Sep 17 00:00:00 2001 From: Matteo Collina Date: Tue, 11 Dec 2018 11:14:59 +0100 Subject: [PATCH 5/5] Added missing tests --- test/destination-exit.test.js | 23 ------------- test/exit.test.js | 53 +++++++++++++++++++++++++++++ test/final.test.js | 12 +++++++ test/fixtures/extreme-exit.js | 9 +++++ test/fixtures/extreme-flush-exit.js | 10 ++++++ 5 files changed, 84 insertions(+), 23 deletions(-) delete mode 100644 test/destination-exit.test.js create mode 100644 test/exit.test.js create mode 100644 test/fixtures/extreme-exit.js create mode 100644 test/fixtures/extreme-flush-exit.js diff --git a/test/destination-exit.test.js b/test/destination-exit.test.js deleted file mode 100644 index 7fdd1f5fd..000000000 --- a/test/destination-exit.test.js +++ /dev/null @@ -1,23 +0,0 @@ -'use strict' - -const { test } = require('tap') -const { join } = require('path') -const execa = require('execa') -const writer = require('flush-write-stream') -const { once } = require('./helper') - -// https://github.com/pinojs/pino/issues/542 -test('log everything when calling process.exit(0)', async ({ isNot }) => { - var actual = '' - const child = execa(process.argv[0], [join(__dirname, 'fixtures', 'destination-exit.js')]) - - child.stdout.pipe(writer((s, enc, cb) => { - actual += s - cb() - })) - - await once(child, 'close') - - isNot(actual.match(/hello/), null) - isNot(actual.match(/world/), null) -}) diff --git a/test/exit.test.js b/test/exit.test.js new file mode 100644 index 000000000..fe7542a32 --- /dev/null +++ b/test/exit.test.js @@ -0,0 +1,53 @@ +'use strict' + +const { test } = require('tap') +const { join } = require('path') +const execa = require('execa') +const writer = require('flush-write-stream') +const { once } = require('./helper') + +// https://github.com/pinojs/pino/issues/542 +test('pino.destination log everything when calling process.exit(0)', async ({ isNot }) => { + var actual = '' + const child = execa(process.argv[0], [join(__dirname, 'fixtures', 'destination-exit.js')]) + + child.stdout.pipe(writer((s, enc, cb) => { + actual += s + cb() + })) + + await once(child, 'close') + + isNot(actual.match(/hello/), null) + isNot(actual.match(/world/), null) +}) + +test('pino.extreme does not log everything when calling process.exit(0)', async ({ is }) => { + var actual = '' + const child = execa(process.argv[0], [join(__dirname, 'fixtures', 'extreme-exit.js')]) + + child.stdout.pipe(writer((s, enc, cb) => { + actual += s + cb() + })) + + await once(child, 'close') + + is(actual.match(/hello/), null) + is(actual.match(/world/), null) +}) + +test('pino.extreme logs everything when calling flushSync', async ({ isNot }) => { + var actual = '' + const child = execa(process.argv[0], [join(__dirname, 'fixtures', 'extreme-flush-exit.js')]) + + child.stdout.pipe(writer((s, enc, cb) => { + actual += s + cb() + })) + + await once(child, 'close') + + isNot(actual.match(/hello/), null) + isNot(actual.match(/world/), null) +}) diff --git a/test/final.test.js b/test/final.test.js index 40f4b1f80..abceb5380 100644 --- a/test/final.test.js +++ b/test/final.test.js @@ -54,6 +54,18 @@ test('listener function immediately sync flushes when fired', async ({ pass, fai if (passed === false) fail('flushSync not called') }) +test('listener function immediately sync flushes when fired (pino.destination)', async ({ pass, fail }) => { + const dest = pino.destination('/dev/null') + var passed = false + dest.flushSync = () => { + passed = true + pass('flushSync called') + } + pino.final(pino(dest), () => {})() + await sleep(10) + if (passed === false) fail('flushSync not called') +}) + test('swallows the non-ready error', async ({ doesNotThrow }) => { const dest = pino.extreme('/dev/null') doesNotThrow(() => { diff --git a/test/fixtures/extreme-exit.js b/test/fixtures/extreme-exit.js new file mode 100644 index 000000000..a90cc67c9 --- /dev/null +++ b/test/fixtures/extreme-exit.js @@ -0,0 +1,9 @@ +global.process = { __proto__: process, pid: 123456 } +Date.now = function () { return 1459875739796 } +require('os').hostname = function () { return 'abcdefghijklmnopqr' } +var pino = require(require.resolve('./../../')) +var dest = pino.extreme(1) +var logger = pino({}, dest) +logger.info('hello') +logger.info('world') +process.exit(0) diff --git a/test/fixtures/extreme-flush-exit.js b/test/fixtures/extreme-flush-exit.js new file mode 100644 index 000000000..3abf7e0ec --- /dev/null +++ b/test/fixtures/extreme-flush-exit.js @@ -0,0 +1,10 @@ +global.process = { __proto__: process, pid: 123456 } +Date.now = function () { return 1459875739796 } +require('os').hostname = function () { return 'abcdefghijklmnopqr' } +var pino = require(require.resolve('./../../')) +var dest = pino.extreme(1) +var logger = pino({}, dest) +logger.info('hello') +logger.info('world') +dest.flushSync() +process.exit(0)