'use strict'; const { safeToString } = process.binding('util'); const promiseRejectEvent = process._promiseRejectEvent; const hasBeenNotifiedProperty = new WeakMap(); const promiseToGuidProperty = new WeakMap(); const pendingUnhandledRejections = []; let lastPromiseId = 1; exports.setup = setupPromises; function getAsynchronousRejectionWarningObject(uid) { return new Error('Promise rejection was handled ' + `asynchronously (rejection id: ${uid})`); } function setupPromises(scheduleMicrotasks) { let deprecationWarned = false; process._setupPromises(function(event, promise, reason) { if (event === promiseRejectEvent.unhandled) unhandledRejection(promise, reason); else if (event === promiseRejectEvent.handled) rejectionHandled(promise); else require('assert').fail(null, null, 'unexpected PromiseRejectEvent'); }); function unhandledRejection(promise, reason) { hasBeenNotifiedProperty.set(promise, false); promiseToGuidProperty.set(promise, lastPromiseId++); addPendingUnhandledRejection(promise, reason); } function rejectionHandled(promise) { const hasBeenNotified = hasBeenNotifiedProperty.get(promise); if (hasBeenNotified !== undefined) { hasBeenNotifiedProperty.delete(promise); const uid = promiseToGuidProperty.get(promise); promiseToGuidProperty.delete(promise); if (hasBeenNotified === true) { let warning = null; if (!process.listenerCount('rejectionHandled')) { // Generate the warning object early to get a good stack trace. warning = getAsynchronousRejectionWarningObject(uid); } process.nextTick(function() { if (!process.emit('rejectionHandled', promise)) { if (warning === null) warning = getAsynchronousRejectionWarningObject(uid); warning.name = 'PromiseRejectionHandledWarning'; warning.id = uid; process.emitWarning(warning); } }); } } } function emitWarning(uid, reason) { const warning = new Error( `Unhandled promise rejection (rejection id: ${uid}): ` + safeToString(reason)); warning.name = 'UnhandledPromiseRejectionWarning'; warning.id = uid; try { if (reason instanceof Error) { warning.stack = reason.stack; } } catch (err) { // ignored } process.emitWarning(warning); if (!deprecationWarned) { deprecationWarned = true; process.emitWarning( 'Unhandled promise rejections are deprecated. In the future, ' + 'promise rejections that are not handled will terminate the ' + 'Node.js process with a non-zero exit code.', 'DeprecationWarning', 'DEP0018'); } } function emitPendingUnhandledRejections() { let hadListeners = false; while (pendingUnhandledRejections.length > 0) { const promise = pendingUnhandledRejections.shift(); const reason = pendingUnhandledRejections.shift(); if (hasBeenNotifiedProperty.get(promise) === false) { hasBeenNotifiedProperty.set(promise, true); const uid = promiseToGuidProperty.get(promise); if (!process.emit('unhandledRejection', reason, promise)) { emitWarning(uid, reason); } else { hadListeners = true; } } } return hadListeners; } function addPendingUnhandledRejection(promise, reason) { pendingUnhandledRejections.push(promise, reason); scheduleMicrotasks(); } return emitPendingUnhandledRejections; }