2017-04-08 02:29:14 +08:00
|
|
|
'use strict';
|
|
|
|
|
|
|
|
const EventEmitter = require('events');
|
|
|
|
const net = require('net');
|
|
|
|
const tls = require('tls');
|
|
|
|
const tlsOptions = require('../imap-core/lib/tls-options');
|
|
|
|
const shared = require('nodemailer/lib/shared');
|
|
|
|
const POP3Connection = require('./pop3-connection');
|
|
|
|
|
|
|
|
const CLOSE_TIMEOUT = 1 * 1000; // how much to wait until pending connections are terminated
|
|
|
|
|
|
|
|
class POP3Server extends EventEmitter {
|
|
|
|
constructor(options) {
|
|
|
|
super();
|
|
|
|
|
|
|
|
this.options = options || {};
|
|
|
|
|
2017-04-08 03:38:52 +08:00
|
|
|
// apply shorthand handlers
|
2017-04-09 06:18:31 +08:00
|
|
|
['onAuth', 'onListMessages', 'onFetchMessage', 'onUpdate'].forEach(handler => {
|
2017-04-08 03:38:52 +08:00
|
|
|
if (typeof this.options[handler] === 'function') {
|
|
|
|
this[handler] = this.options[handler];
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2017-04-08 02:29:14 +08:00
|
|
|
/**
|
|
|
|
* 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();
|
|
|
|
|
|
|
|
// 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'
|
|
|
|
});
|
|
|
|
|
2017-06-03 14:51:58 +08:00
|
|
|
this.server = (this.options.secure ? tls : net).createServer(this.options, socket => this._onConnect(socket));
|
2017-04-08 02:29:14 +08:00
|
|
|
|
|
|
|
this._setListeners();
|
|
|
|
}
|
|
|
|
|
|
|
|
_setListeners() {
|
|
|
|
this.server.on('listening', () => this._onListening());
|
|
|
|
this.server.on('close', () => this._onClose());
|
|
|
|
this.server.on('error', err => this._onError(err));
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* 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: 'POP3'
|
|
|
|
},
|
|
|
|
'%s%s Server listening on %s:%s',
|
|
|
|
this.options.secure ? 'Secure ' : '',
|
|
|
|
'POP3',
|
|
|
|
address.family === 'IPv4' ? address.address : '[' + address.address + ']',
|
2017-06-03 14:51:58 +08:00
|
|
|
address.port
|
|
|
|
);
|
2017-04-08 02:29:14 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Called when server is closed
|
|
|
|
*
|
|
|
|
* @event
|
|
|
|
*/
|
|
|
|
_onClose() {
|
2017-06-03 14:51:58 +08:00
|
|
|
this.logger.info(
|
|
|
|
{
|
|
|
|
tnx: 'closed'
|
|
|
|
},
|
|
|
|
'POP3 Server closed'
|
|
|
|
);
|
2017-04-08 02:29:14 +08:00
|
|
|
this.emit('close');
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Called when an error occurs with the server
|
|
|
|
*
|
|
|
|
* @event
|
|
|
|
*/
|
|
|
|
_onError(err) {
|
|
|
|
this.emit('error', err);
|
|
|
|
}
|
|
|
|
|
|
|
|
_onConnect(socket) {
|
|
|
|
let connection = new POP3Connection(this, socket);
|
|
|
|
this.connections.add(connection);
|
|
|
|
connection.once('error', err => {
|
|
|
|
this.connections.delete(connection);
|
|
|
|
this._onError(err);
|
|
|
|
});
|
|
|
|
connection.once('close', () => {
|
|
|
|
this.connections.delete(connection);
|
|
|
|
});
|
|
|
|
connection.init();
|
|
|
|
}
|
|
|
|
|
|
|
|
close(callback) {
|
|
|
|
let connections = this.connections.size;
|
|
|
|
let timeout = this.options.closeTimeout || CLOSE_TIMEOUT;
|
|
|
|
|
|
|
|
// stop accepting new connections
|
|
|
|
this.server.close(() => {
|
|
|
|
clearTimeout(this._closeTimeout);
|
|
|
|
if (typeof callback === 'function') {
|
|
|
|
return callback();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
// close active connections
|
|
|
|
if (connections) {
|
2017-06-03 14:51:58 +08:00
|
|
|
this.logger.info(
|
|
|
|
{
|
|
|
|
tnx: 'close'
|
|
|
|
},
|
|
|
|
'Server closing with %s pending connection%s, waiting %s seconds before terminating',
|
|
|
|
connections,
|
|
|
|
connections !== 1 ? 's' : '',
|
|
|
|
timeout / 1000
|
|
|
|
);
|
2017-04-08 02:29:14 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
this._closeTimeout = setTimeout(() => {
|
|
|
|
connections = this.connections.size;
|
|
|
|
if (connections) {
|
2017-06-03 14:51:58 +08:00
|
|
|
this.logger.info(
|
|
|
|
{
|
|
|
|
tnx: 'close'
|
|
|
|
},
|
|
|
|
'Closing %s pending connection%s to close the server',
|
|
|
|
connections,
|
|
|
|
connections !== 1 ? 's' : ''
|
|
|
|
);
|
2017-04-08 02:29:14 +08:00
|
|
|
|
|
|
|
this.connections.forEach(connection => {
|
|
|
|
connection.close();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}, timeout);
|
|
|
|
}
|
|
|
|
|
2017-04-08 03:38:52 +08:00
|
|
|
/**
|
|
|
|
* Authentication handler. Override this
|
|
|
|
*
|
|
|
|
* @param {Object} auth Authentication options
|
2017-04-08 17:39:07 +08:00
|
|
|
* @param {Object} session Session object
|
2017-04-08 03:38:52 +08:00
|
|
|
* @param {Function} callback Callback to run once the user is authenticated
|
|
|
|
*/
|
|
|
|
onAuth(auth, session, callback) {
|
|
|
|
return callback(null, {
|
|
|
|
message: 'Authentication not implemented'
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2017-04-09 06:18:31 +08:00
|
|
|
// called when a message body needs to be fetched
|
|
|
|
onFetchMessage(id, session, callback) {
|
|
|
|
// should return a stream object
|
|
|
|
return callback(null, false);
|
|
|
|
}
|
|
|
|
|
|
|
|
// called when session is finished and messages need to be updated/deleted
|
|
|
|
onUpdate(update, session, callback) {
|
|
|
|
return callback(null, false);
|
|
|
|
}
|
|
|
|
|
2017-04-08 17:39:07 +08:00
|
|
|
/**
|
|
|
|
* Message listing handler. Override this
|
|
|
|
*
|
|
|
|
* @param {Object} session Session object
|
|
|
|
* @param {Function} callback Callback to run with message listing
|
|
|
|
*/
|
|
|
|
onListMessages(session, callback) {
|
2017-04-09 06:18:31 +08:00
|
|
|
// messages are objects {id: 'abc', size: 123, seen: false}
|
2017-04-08 17:39:07 +08:00
|
|
|
return callback(null, {
|
|
|
|
messages: [],
|
|
|
|
count: 0,
|
|
|
|
size: 0
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2017-04-08 02:29:14 +08:00
|
|
|
listen(...args) {
|
|
|
|
this.server.listen(...args);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
module.exports = POP3Server;
|