'use strict'; const net = require('net'); const tls = require('tls'); const IMAPConnection = require('./imap-connection').IMAPConnection; const tlsOptions = require('./tls-options'); const EventEmitter = require('events').EventEmitter; const shared = require('nodemailer/lib/shared'); const util = require('util'); const CLOSE_TIMEOUT = 1 * 1000; // how much to wait until pending connections are terminated /** * Creates a IMAP server instance. * * @constructor * @param {Object} options Connection and IMAP optionsž */ class IMAPServer extends EventEmitter { constructor(options) { super(); this.options = options || {}; // apply TLS defaults if needed if (this.options.secure) { this.options = tlsOptions(this.options); } this.logger = shared.getLogger(this.options, { component: this.options.component || 'pop3-server' }); /** * Timeout after close has been called until pending connections are forcibly closed */ this._closeTimeout = false; /** * A set of all currently open connections */ this.connections = new Set(); // setup server listener and connection handler this.server = (this.options.secure ? tls : net).createServer(this.options, socket => { let connection = new IMAPConnection(this, socket); this.connections.add(connection); connection.on('error', this._onError.bind(this)); connection.init(); }); this._setListeners(); } /** * Start listening on selected port and interface */ listen(...args) { this.server.listen(...args); } /** * Closes the server * * @param {Function} callback Callback to run once the server is fully closed */ close(callback) { let connections = this.connections.size; let timeout = this.options.closeTimeout || CLOSE_TIMEOUT; // stop accepting new connections this.server.close(() => { clearTimeout(this._closeTimeout); callback(); }); // close active connections if (connections) { this.logger.info( { tnx: 'close' }, 'Server closing with %s pending connection%s, waiting %s seconds before terminating', connections, connections !== 1 ? 's' : '', timeout / 1000 ); } this._closeTimeout = setTimeout(() => { connections = this.connections.size; if (connections) { this.logger.info( { tnx: 'close' }, 'Closing %s pending connection%s to close the server', connections, connections !== 1 ? 's' : '' ); this.connections.forEach(connection => { connection.send('* BYE System shutdown'); connection.close(); }); } }, timeout); } // PRIVATE METHODS /** * Generates a bunyan-like logger that prints to console * * @returns {Object} Bunyan logger instance */ _createDefaultLogger() { let logger = { _print: (...args) => { let level = args.shift(); let message; if (args.length > 1) { message = util.format(...args); } else { message = args[0]; } console.log('[%s] %s: %s', new Date().toISOString().substr(0, 19).replace(/T/, ' '), level.toUpperCase(), message); // eslint-disable-line no-console } }; logger.info = logger._print.bind(null, 'info'); logger.debug = logger._print.bind(null, 'debug'); logger.error = logger._print.bind(null, 'error'); return logger; } /** * Setup server event handlers */ _setListeners() { this.server.on('listening', this._onListening.bind(this)); this.server.on('close', this._onClose.bind(this)); this.server.on('error', this._onError.bind(this)); } /** * Called when server started listening * * @event */ _onListening() { let address = this.server.address(); this.logger.info( // { tnx: 'listen', host: address.address, port: address.port, secure: !!this.options.secure, protocol: 'IMAP' }, '%sIMAP Server listening on %s:%s', this.options.secure ? 'Secure ' : '', address.family === 'IPv4' ? address.address : '[' + address.address + ']', address.port ); } /** * Called when server is closed * * @event */ _onClose() { this.logger.info( { tnx: 'closed' }, 'IMAP Server closed' ); this.emit('close'); } /** * Called when an error occurs with the server * * @event */ _onError(err) { this.emit('error', err); } } // Expose to the world module.exports.IMAPServer = IMAPServer;