Spaces:
Runtime error
Runtime error
| /*! | |
| * depd | |
| * Copyright(c) 2014-2018 Douglas Christopher Wilson | |
| * MIT Licensed | |
| */ | |
| /** | |
| * Module dependencies. | |
| */ | |
| var relative = require('path').relative | |
| /** | |
| * Module exports. | |
| */ | |
| module.exports = depd | |
| /** | |
| * Get the path to base files on. | |
| */ | |
| var basePath = process.cwd() | |
| /** | |
| * Determine if namespace is contained in the string. | |
| */ | |
| function containsNamespace (str, namespace) { | |
| var vals = str.split(/[ ,]+/) | |
| var ns = String(namespace).toLowerCase() | |
| for (var i = 0; i < vals.length; i++) { | |
| var val = vals[i] | |
| // namespace contained | |
| if (val && (val === '*' || val.toLowerCase() === ns)) { | |
| return true | |
| } | |
| } | |
| return false | |
| } | |
| /** | |
| * Convert a data descriptor to accessor descriptor. | |
| */ | |
| function convertDataDescriptorToAccessor (obj, prop, message) { | |
| var descriptor = Object.getOwnPropertyDescriptor(obj, prop) | |
| var value = descriptor.value | |
| descriptor.get = function getter () { return value } | |
| if (descriptor.writable) { | |
| descriptor.set = function setter (val) { return (value = val) } | |
| } | |
| delete descriptor.value | |
| delete descriptor.writable | |
| Object.defineProperty(obj, prop, descriptor) | |
| return descriptor | |
| } | |
| /** | |
| * Create arguments string to keep arity. | |
| */ | |
| function createArgumentsString (arity) { | |
| var str = '' | |
| for (var i = 0; i < arity; i++) { | |
| str += ', arg' + i | |
| } | |
| return str.substr(2) | |
| } | |
| /** | |
| * Create stack string from stack. | |
| */ | |
| function createStackString (stack) { | |
| var str = this.name + ': ' + this.namespace | |
| if (this.message) { | |
| str += ' deprecated ' + this.message | |
| } | |
| for (var i = 0; i < stack.length; i++) { | |
| str += '\n at ' + stack[i].toString() | |
| } | |
| return str | |
| } | |
| /** | |
| * Create deprecate for namespace in caller. | |
| */ | |
| function depd (namespace) { | |
| if (!namespace) { | |
| throw new TypeError('argument namespace is required') | |
| } | |
| var stack = getStack() | |
| var site = callSiteLocation(stack[1]) | |
| var file = site[0] | |
| function deprecate (message) { | |
| // call to self as log | |
| log.call(deprecate, message) | |
| } | |
| deprecate._file = file | |
| deprecate._ignored = isignored(namespace) | |
| deprecate._namespace = namespace | |
| deprecate._traced = istraced(namespace) | |
| deprecate._warned = Object.create(null) | |
| deprecate.function = wrapfunction | |
| deprecate.property = wrapproperty | |
| return deprecate | |
| } | |
| /** | |
| * Determine if event emitter has listeners of a given type. | |
| * | |
| * The way to do this check is done three different ways in Node.js >= 0.8 | |
| * so this consolidates them into a minimal set using instance methods. | |
| * | |
| * @param {EventEmitter} emitter | |
| * @param {string} type | |
| * @returns {boolean} | |
| * @private | |
| */ | |
| function eehaslisteners (emitter, type) { | |
| var count = typeof emitter.listenerCount !== 'function' | |
| ? emitter.listeners(type).length | |
| : emitter.listenerCount(type) | |
| return count > 0 | |
| } | |
| /** | |
| * Determine if namespace is ignored. | |
| */ | |
| function isignored (namespace) { | |
| if (process.noDeprecation) { | |
| // --no-deprecation support | |
| return true | |
| } | |
| var str = process.env.NO_DEPRECATION || '' | |
| // namespace ignored | |
| return containsNamespace(str, namespace) | |
| } | |
| /** | |
| * Determine if namespace is traced. | |
| */ | |
| function istraced (namespace) { | |
| if (process.traceDeprecation) { | |
| // --trace-deprecation support | |
| return true | |
| } | |
| var str = process.env.TRACE_DEPRECATION || '' | |
| // namespace traced | |
| return containsNamespace(str, namespace) | |
| } | |
| /** | |
| * Display deprecation message. | |
| */ | |
| function log (message, site) { | |
| var haslisteners = eehaslisteners(process, 'deprecation') | |
| // abort early if no destination | |
| if (!haslisteners && this._ignored) { | |
| return | |
| } | |
| var caller | |
| var callFile | |
| var callSite | |
| var depSite | |
| var i = 0 | |
| var seen = false | |
| var stack = getStack() | |
| var file = this._file | |
| if (site) { | |
| // provided site | |
| depSite = site | |
| callSite = callSiteLocation(stack[1]) | |
| callSite.name = depSite.name | |
| file = callSite[0] | |
| } else { | |
| // get call site | |
| i = 2 | |
| depSite = callSiteLocation(stack[i]) | |
| callSite = depSite | |
| } | |
| // get caller of deprecated thing in relation to file | |
| for (; i < stack.length; i++) { | |
| caller = callSiteLocation(stack[i]) | |
| callFile = caller[0] | |
| if (callFile === file) { | |
| seen = true | |
| } else if (callFile === this._file) { | |
| file = this._file | |
| } else if (seen) { | |
| break | |
| } | |
| } | |
| var key = caller | |
| ? depSite.join(':') + '__' + caller.join(':') | |
| : undefined | |
| if (key !== undefined && key in this._warned) { | |
| // already warned | |
| return | |
| } | |
| this._warned[key] = true | |
| // generate automatic message from call site | |
| var msg = message | |
| if (!msg) { | |
| msg = callSite === depSite || !callSite.name | |
| ? defaultMessage(depSite) | |
| : defaultMessage(callSite) | |
| } | |
| // emit deprecation if listeners exist | |
| if (haslisteners) { | |
| var err = DeprecationError(this._namespace, msg, stack.slice(i)) | |
| process.emit('deprecation', err) | |
| return | |
| } | |
| // format and write message | |
| var format = process.stderr.isTTY | |
| ? formatColor | |
| : formatPlain | |
| var output = format.call(this, msg, caller, stack.slice(i)) | |
| process.stderr.write(output + '\n', 'utf8') | |
| } | |
| /** | |
| * Get call site location as array. | |
| */ | |
| function callSiteLocation (callSite) { | |
| var file = callSite.getFileName() || '<anonymous>' | |
| var line = callSite.getLineNumber() | |
| var colm = callSite.getColumnNumber() | |
| if (callSite.isEval()) { | |
| file = callSite.getEvalOrigin() + ', ' + file | |
| } | |
| var site = [file, line, colm] | |
| site.callSite = callSite | |
| site.name = callSite.getFunctionName() | |
| return site | |
| } | |
| /** | |
| * Generate a default message from the site. | |
| */ | |
| function defaultMessage (site) { | |
| var callSite = site.callSite | |
| var funcName = site.name | |
| // make useful anonymous name | |
| if (!funcName) { | |
| funcName = '<anonymous@' + formatLocation(site) + '>' | |
| } | |
| var context = callSite.getThis() | |
| var typeName = context && callSite.getTypeName() | |
| // ignore useless type name | |
| if (typeName === 'Object') { | |
| typeName = undefined | |
| } | |
| // make useful type name | |
| if (typeName === 'Function') { | |
| typeName = context.name || typeName | |
| } | |
| return typeName && callSite.getMethodName() | |
| ? typeName + '.' + funcName | |
| : funcName | |
| } | |
| /** | |
| * Format deprecation message without color. | |
| */ | |
| function formatPlain (msg, caller, stack) { | |
| var timestamp = new Date().toUTCString() | |
| var formatted = timestamp + | |
| ' ' + this._namespace + | |
| ' deprecated ' + msg | |
| // add stack trace | |
| if (this._traced) { | |
| for (var i = 0; i < stack.length; i++) { | |
| formatted += '\n at ' + stack[i].toString() | |
| } | |
| return formatted | |
| } | |
| if (caller) { | |
| formatted += ' at ' + formatLocation(caller) | |
| } | |
| return formatted | |
| } | |
| /** | |
| * Format deprecation message with color. | |
| */ | |
| function formatColor (msg, caller, stack) { | |
| var formatted = '\x1b[36;1m' + this._namespace + '\x1b[22;39m' + // bold cyan | |
| ' \x1b[33;1mdeprecated\x1b[22;39m' + // bold yellow | |
| ' \x1b[0m' + msg + '\x1b[39m' // reset | |
| // add stack trace | |
| if (this._traced) { | |
| for (var i = 0; i < stack.length; i++) { | |
| formatted += '\n \x1b[36mat ' + stack[i].toString() + '\x1b[39m' // cyan | |
| } | |
| return formatted | |
| } | |
| if (caller) { | |
| formatted += ' \x1b[36m' + formatLocation(caller) + '\x1b[39m' // cyan | |
| } | |
| return formatted | |
| } | |
| /** | |
| * Format call site location. | |
| */ | |
| function formatLocation (callSite) { | |
| return relative(basePath, callSite[0]) + | |
| ':' + callSite[1] + | |
| ':' + callSite[2] | |
| } | |
| /** | |
| * Get the stack as array of call sites. | |
| */ | |
| function getStack () { | |
| var limit = Error.stackTraceLimit | |
| var obj = {} | |
| var prep = Error.prepareStackTrace | |
| Error.prepareStackTrace = prepareObjectStackTrace | |
| Error.stackTraceLimit = Math.max(10, limit) | |
| // capture the stack | |
| Error.captureStackTrace(obj) | |
| // slice this function off the top | |
| var stack = obj.stack.slice(1) | |
| Error.prepareStackTrace = prep | |
| Error.stackTraceLimit = limit | |
| return stack | |
| } | |
| /** | |
| * Capture call site stack from v8. | |
| */ | |
| function prepareObjectStackTrace (obj, stack) { | |
| return stack | |
| } | |
| /** | |
| * Return a wrapped function in a deprecation message. | |
| */ | |
| function wrapfunction (fn, message) { | |
| if (typeof fn !== 'function') { | |
| throw new TypeError('argument fn must be a function') | |
| } | |
| var args = createArgumentsString(fn.length) | |
| var stack = getStack() | |
| var site = callSiteLocation(stack[1]) | |
| site.name = fn.name | |
| // eslint-disable-next-line no-new-func | |
| var deprecatedfn = new Function('fn', 'log', 'deprecate', 'message', 'site', | |
| '"use strict"\n' + | |
| 'return function (' + args + ') {' + | |
| 'log.call(deprecate, message, site)\n' + | |
| 'return fn.apply(this, arguments)\n' + | |
| '}')(fn, log, this, message, site) | |
| return deprecatedfn | |
| } | |
| /** | |
| * Wrap property in a deprecation message. | |
| */ | |
| function wrapproperty (obj, prop, message) { | |
| if (!obj || (typeof obj !== 'object' && typeof obj !== 'function')) { | |
| throw new TypeError('argument obj must be object') | |
| } | |
| var descriptor = Object.getOwnPropertyDescriptor(obj, prop) | |
| if (!descriptor) { | |
| throw new TypeError('must call property on owner object') | |
| } | |
| if (!descriptor.configurable) { | |
| throw new TypeError('property must be configurable') | |
| } | |
| var deprecate = this | |
| var stack = getStack() | |
| var site = callSiteLocation(stack[1]) | |
| // set site name | |
| site.name = prop | |
| // convert data descriptor | |
| if ('value' in descriptor) { | |
| descriptor = convertDataDescriptorToAccessor(obj, prop, message) | |
| } | |
| var get = descriptor.get | |
| var set = descriptor.set | |
| // wrap getter | |
| if (typeof get === 'function') { | |
| descriptor.get = function getter () { | |
| log.call(deprecate, message, site) | |
| return get.apply(this, arguments) | |
| } | |
| } | |
| // wrap setter | |
| if (typeof set === 'function') { | |
| descriptor.set = function setter () { | |
| log.call(deprecate, message, site) | |
| return set.apply(this, arguments) | |
| } | |
| } | |
| Object.defineProperty(obj, prop, descriptor) | |
| } | |
| /** | |
| * Create DeprecationError for deprecation | |
| */ | |
| function DeprecationError (namespace, message, stack) { | |
| var error = new Error() | |
| var stackString | |
| Object.defineProperty(error, 'constructor', { | |
| value: DeprecationError | |
| }) | |
| Object.defineProperty(error, 'message', { | |
| configurable: true, | |
| enumerable: false, | |
| value: message, | |
| writable: true | |
| }) | |
| Object.defineProperty(error, 'name', { | |
| enumerable: false, | |
| configurable: true, | |
| value: 'DeprecationError', | |
| writable: true | |
| }) | |
| Object.defineProperty(error, 'namespace', { | |
| configurable: true, | |
| enumerable: false, | |
| value: namespace, | |
| writable: true | |
| }) | |
| Object.defineProperty(error, 'stack', { | |
| configurable: true, | |
| enumerable: false, | |
| get: function () { | |
| if (stackString !== undefined) { | |
| return stackString | |
| } | |
| // prepare stack trace | |
| return (stackString = createStackString.call(this, stack)) | |
| }, | |
| set: function setter (val) { | |
| stackString = val | |
| } | |
| }) | |
| return error | |
| } | |