/* eslint global-require: 0, no-console: 0 */ 'use strict'; const config = require('wild-config'); const Gelf = require('gelf'); const os = require('os'); const util = require('util'); let bugsnag; let loggelf; let component; let hostname; module.exports.gelf = {}; if (config.bugsnagCode) { bugsnag = require('bugsnag'); bugsnag.register(config.bugsnagCode); } else { let gelfconf = (config && config.log && config.log.gelf) || {}; component = gelfconf.component || 'wildduck'; hostname = gelfconf.hostname || os.hostname(); module.exports.gelf.handler = gelfconf.enabled ? new Gelf(gelfconf.options) : { emit: (channel, message) => console.error(util.inspect(message, false, 3)) }; loggelf = (...args) => { let err = args.shift() || {}; if (err.code === 'ECONNRESET') { // just ignore return; } let message = { short_message: component.toUpperCase() + ' [Exception] ' + (err.message || ''), full_message: err.stack, _exception: 'yes', _error: err.message }; let limitLength = val => { let str = (val || '').toString().trim(); if (str.length > 256) { str = str.substr(0, 256) + '…'; } return str; }; Object.keys(err).forEach(key => { let vKey = '_' + key; if (!message[vKey] && typeof err[key] !== 'object') { message[vKey] = limitLength(err[key]); } }); for (let extra of args) { Object.keys(extra || {}).forEach(key => { let vKey = '_' + key; if (!message[vKey] && typeof extra[key] !== 'object') { message[vKey] = limitLength(extra[key]); } }); } message.facility = component; // facility is deprecated but set by the driver if not provided message.host = hostname; message.timestamp = Date.now() / 1000; message._component = component; Object.keys(message).forEach(key => { if (!message[key]) { delete message[key]; } }); try { module.exports.gelf.handler.emit('gelf.log', message); } catch (err) { // might fail on non-JSONizable input try { console.error(err); console.error(util.inspect(message, false, 3)); } catch (err) { //ignore } } }; } module.exports.notify = (...args) => { if (bugsnag) { bugsnag.notify(...args); } else { loggelf(...args); } }; module.exports.notifyConnection = (connection, ...args) => { let err = args[0]; let metaData = args[1] || {}; if (connection) { if (connection.selected) { metaData.selected = connection.selected.mailbox; } if (connection.session.user) { metaData.userId = connection.session.user.id.toString(); } metaData.remoteAddress = connection.session.remoteAddress; metaData.isUTF8Enabled = !!connection.acceptUTF8Enabled; } Object.keys(err.meta || {}).forEach(key => { metaData[key] = err.meta[key]; }); args[1] = metaData; if (bugsnag) { bugsnag.notify(...args); } else { loggelf(...args); } }; module.exports.intercept = (...args) => { if (bugsnag) { return bugsnag.intercept(...args); } let cb; if (args.length) { cb = args[args.length - 1]; if (typeof cb === 'function') { args[args.length - 1] = function(...returnArgs) { if (returnArgs.length > 1 && returnArgs[0]) { console.error(returnArgs[0]); } return cb(...returnArgs); }; } } }; module.exports.setGelf = gelf => { module.exports.gelf.handler = gelf; };