wildduck/test/filtering-test.js
2017-10-30 13:41:53 +02:00

586 lines
27 KiB
JavaScript

/*eslint no-unused-expressions: 0, prefer-arrow-callback: 0, no-console: 0 */
/* global before */
'use strict';
process.env.NODE_TLS_REJECT_UNAUTHORIZED = '0';
const crypto = require('crypto');
//const util = require('util');
const chai = require('chai');
const request = require('request');
const fs = require('fs');
const BrowserBox = require('browserbox');
const simpleParser = require('mailparser').simpleParser;
const nodemailer = require('nodemailer');
const transporter = nodemailer.createTransport({
lmtp: true,
host: 'localhost',
port: 2424,
logger: false,
debug: false,
tls: {
rejectUnauthorized: false
}
});
const expect = chai.expect;
chai.config.includeStack = true;
const URL = 'http://localhost:8080';
const user2PubKey = fs.readFileSync(__dirname + '/fixtures/user2-public.key', 'utf-8');
const user3PubKey = fs.readFileSync(__dirname + '/fixtures/user3-public.key', 'utf-8');
describe('Send multiple messages', function() {
this.timeout(100 * 1000); // eslint-disable-line
let userIds = [];
before(done => {
request.post(
URL + '/users',
{
json: {
username: 'user1',
password: 'secretpass',
address: 'user1@example.com',
name: 'user1'
}
},
(err, meta, response) => {
expect(err).to.not.exist;
expect(response.success).to.be.true;
userIds.push(response.id);
request.post(
URL + '/users',
{
json: {
username: 'user2',
password: 'secretpass',
address: 'user2@example.com',
name: 'user2',
pubKey: user2PubKey,
encryptMessages: true,
encryptForwarded: true
}
},
(err, meta, response) => {
expect(err).to.not.exist;
expect(response.success).to.be.true;
userIds.push(response.id);
request.post(
URL + '/users',
{
json: {
username: 'user3',
password: 'secretpass',
address: 'user3@example.com',
name: 'user3',
pubKey: user3PubKey,
encryptMessages: true,
encryptForwarded: true
}
},
(err, meta, response) => {
expect(err).to.not.exist;
expect(response.success).to.be.true;
userIds.push(response.id);
request.post(
URL + '/users',
{
json: {
username: 'user4',
password: 'secretpass',
address: 'user4@example.com',
name: 'user4',
pubKey: user2PubKey,
encryptMessages: false,
encryptForwarded: true
}
},
(err, meta, response) => {
expect(err).to.not.exist;
expect(response.success).to.be.true;
userIds.push(response.id);
request.post(
URL + '/users',
{
json: {
username: 'user5',
password: 'secretpass',
address: 'user5@example.com',
name: 'user5'
}
},
(err, meta, response) => {
expect(err).to.not.exist;
expect(response.success).to.be.true;
userIds.push(response.id);
done();
}
);
}
);
}
);
}
);
}
);
});
it('Should have users set', done => {
expect(userIds.length).to.equal(5);
done();
});
it('Send mail to all users', done => {
let recipients = ['user1@example.com', 'user2@example.com', 'user3@example.com', 'user4@example.com', 'user5@example.com'];
let subject = 'Test ööö message [' + Date.now() + ']';
transporter.sendMail(
{
envelope: {
from: 'andris@kreata.ee',
to: recipients
},
headers: {
// set to Yes to send this message to Junk folder
'x-rspamd-spam': 'No'
},
from: 'Kärbes 🐧 <andris@kreata.ee>',
to: recipients.map((rcpt, i) => ({ name: 'User #' + (i + 1), address: rcpt })),
subject,
text: 'Hello world! Current time is ' + new Date().toString(),
html:
'<p>Hello world! Current time is <em>' +
new Date().toString() +
'</em> <img src="cid:note@example.com"/> <img src="http://www.neti.ee/img/neti-logo-2015-1.png"></p>',
attachments: [
// attachment as plaintext
{
filename: 'notes.txt',
content: 'Some notes about this e-mail',
contentType: 'text/plain' // optional, would be detected from the filename
},
// Small Binary Buffer attachment, should be kept with message
{
filename: 'image.png',
content: new Buffer(
'iVBORw0KGgoAAAANSUhEUgAAABAAAAAQAQMAAAAlPW0iAAAABlBMVEUAAAD/' +
'//+l2Z/dAAAAM0lEQVR4nGP4/5/h/1+G/58ZDrAz3D/McH8yw83NDDeNGe4U' +
'g9C9zwz3gVLMDA/A6P9/AFGGFyjOXZtQAAAAAElFTkSuQmCC',
'base64'
),
cid: 'note@example.com' // should be as unique as possible
},
// Large Binary Buffer attachment, should be kept separately
{
path: __dirname + '/../examples/swan.jpg',
filename: 'swän.jpg'
}
]
},
(err, info) => {
expect(err).to.not.exist;
expect(info.accepted).to.deep.equal(['user1@example.com', 'user2@example.com', 'user3@example.com', 'user4@example.com', 'user5@example.com']);
let getFirstMessage = (userId, callback) => {
request(URL + '/users/' + userId + '/mailboxes', { json: true }, (err, meta, response) => {
expect(err).to.not.exist;
expect(response.success).to.be.true;
let inbox = response.results.find(mbox => mbox.path === 'INBOX');
request(URL + '/users/' + userId + '/mailboxes/' + inbox.id + '/messages', { json: true }, (err, meta, response) => {
expect(err).to.not.exist;
expect(response.success).to.be.true;
let message = response.results[0];
expect(message).to.exist;
request(URL + '/users/' + userId + '/mailboxes/' + inbox.id + '/messages/' + message.id, { json: true }, (err, meta, message) => {
expect(err).to.not.exist;
let processAttachments = next => {
let pos = 0;
let getAttachments = () => {
if (pos >= message.attachments.length) {
return next();
}
let attachment = message.attachments[pos++];
request(URL + attachment.url, { encoding: null }, (err, meta, raw) => {
expect(err).to.not.exist;
attachment.raw = raw;
setImmediate(getAttachments);
});
};
setImmediate(getAttachments);
};
processAttachments(() => {
request(
URL + '/users/' + userId + '/mailboxes/' + inbox.id + '/messages/' + message.id + '/message.eml',
(err, meta, raw) => {
expect(err).to.not.exist;
message.raw = raw;
simpleParser(raw, (err, parsed) => {
expect(err).to.not.exist;
message.parsed = parsed;
callback(null, message);
});
}
);
});
});
});
});
};
let checkNormalUsers = next => {
let npos = 0;
let nusers = [1, 4, 5];
let checkUser = () => {
if (npos >= nusers.length) {
return next();
}
let user = nusers[npos++];
getFirstMessage(userIds[user - 1], (err, message) => {
expect(err).to.not.exist;
expect(message.subject).to.equal(subject);
expect(message.attachments.length).to.equal(3);
expect(message.parsed.attachments.length).to.equal(3);
for (let i = 0; i < message.attachments.length; i++) {
let hashA = crypto
.createHash('md5')
.update(message.attachments[i].raw)
.digest('hex');
let hashB = crypto
.createHash('md5')
.update(message.parsed.attachments[i].content)
.digest('hex');
expect(hashA).equal(hashB);
}
expect(message.parsed.to.value).deep.equal([
{ address: 'user1@example.com', name: 'User #1' },
{ address: 'user2@example.com', name: 'User #2' },
{ address: 'user3@example.com', name: 'User #3' },
{ address: 'user4@example.com', name: 'User #4' },
{ address: 'user5@example.com', name: 'User #5' }
]);
expect(message.parsed.headers.get('delivered-to').value[0].address).equal('user' + user + '@example.com');
setImmediate(checkUser);
});
};
setImmediate(checkUser);
};
let checkEncryptedUsers = next => {
let npos = 0;
let nusers = [2, 3];
let checkUser = () => {
if (npos >= nusers.length) {
return next();
}
let user = nusers[npos++];
getFirstMessage(userIds[user - 1], (err, message) => {
expect(err).to.not.exist;
expect(message.subject).to.equal(subject);
expect(message.parsed.to.value).deep.equal([
{ address: 'user1@example.com', name: 'User #1' },
{ address: 'user2@example.com', name: 'User #2' },
{ address: 'user3@example.com', name: 'User #3' },
{ address: 'user4@example.com', name: 'User #4' },
{ address: 'user5@example.com', name: 'User #5' }
]);
expect(message.parsed.headers.get('delivered-to').value[0].address).equal('user' + user + '@example.com');
expect(message.parsed.attachments.length).equal(2);
expect(message.parsed.attachments[0].contentType).equal('application/pgp-encrypted');
expect(message.parsed.attachments[0].content.toString()).equal('Version: 1\r\n');
expect(message.parsed.attachments[1].contentType).equal('application/octet-stream');
expect(message.parsed.attachments[1].filename).equal('encrypted.asc');
expect(message.parsed.attachments[1].size).gte(1000000);
setImmediate(checkUser);
});
};
setImmediate(checkUser);
};
checkNormalUsers(() => checkEncryptedUsers(() => done()));
}
);
});
it('Send should send mail to spam', done => {
let recipients = ['user1@example.com', 'user2@example.com', 'user3@example.com', 'user4@example.com', 'user5@example.com'];
let subject = 'Test ööö message [' + Date.now() + ']';
transporter.sendMail(
{
envelope: {
from: 'andris@kreata.ee',
to: recipients
},
headers: {
// set to Yes to send this message to Junk folder
'x-rspamd-spam': 'Yes'
},
from: 'Kärbes 🐧 <andris@kreata.ee>',
to: recipients.map((rcpt, i) => ({ name: 'User #' + (i + 1), address: rcpt })),
subject,
text: 'Hello world! Current time is ' + new Date().toString(),
html:
'<p>Hello world! Current time is <em>' +
new Date().toString() +
'</em> <img src="cid:note@example.com"/> <img src="http://www.neti.ee/img/neti-logo-2015-1.png"></p>',
attachments: [
// attachment as plaintext
{
filename: 'notes.txt',
content: 'Some notes about this e-mail',
contentType: 'text/plain' // optional, would be detected from the filename
},
// Small Binary Buffer attachment, should be kept with message
{
filename: 'image.png',
content: new Buffer(
'iVBORw0KGgoAAAANSUhEUgAAABAAAAAQAQMAAAAlPW0iAAAABlBMVEUAAAD/' +
'//+l2Z/dAAAAM0lEQVR4nGP4/5/h/1+G/58ZDrAz3D/McH8yw83NDDeNGe4U' +
'g9C9zwz3gVLMDA/A6P9/AFGGFyjOXZtQAAAAAElFTkSuQmCC',
'base64'
),
cid: 'note@example.com' // should be as unique as possible
},
// Large Binary Buffer attachment, should be kept separately
{
path: __dirname + '/../examples/swan.jpg',
filename: 'swän.jpg'
}
]
},
(err, info) => {
expect(err).to.not.exist;
expect(info.accepted).to.deep.equal(['user1@example.com', 'user2@example.com', 'user3@example.com', 'user4@example.com', 'user5@example.com']);
let getFirstMessage = (userId, callback) => {
request(URL + '/users/' + userId + '/mailboxes', { json: true }, (err, meta, response) => {
expect(err).to.not.exist;
expect(response.success).to.be.true;
let inbox = response.results.find(mbox => mbox.specialUse === '\\Junk');
request(URL + '/users/' + userId + '/mailboxes/' + inbox.id + '/messages', { json: true }, (err, meta, response) => {
expect(err).to.not.exist;
expect(response.success).to.be.true;
let message = response.results[0];
expect(message).to.exist;
request(URL + '/users/' + userId + '/mailboxes/' + inbox.id + '/messages/' + message.id, { json: true }, (err, meta, message) => {
expect(err).to.not.exist;
let processAttachments = next => {
let pos = 0;
let getAttachments = () => {
if (pos >= message.attachments.length) {
return next();
}
let attachment = message.attachments[pos++];
request(URL + attachment.url, { encoding: null }, (err, meta, raw) => {
expect(err).to.not.exist;
attachment.raw = raw;
setImmediate(getAttachments);
});
};
setImmediate(getAttachments);
};
processAttachments(() => {
request(
URL + '/users/' + userId + '/mailboxes/' + inbox.id + '/messages/' + message.id + '/message.eml',
(err, meta, raw) => {
expect(err).to.not.exist;
message.raw = raw;
simpleParser(raw, (err, parsed) => {
expect(err).to.not.exist;
message.parsed = parsed;
callback(null, message);
});
}
);
});
});
});
});
};
let checkNormalUsers = next => {
let npos = 0;
let nusers = [1, 4, 5];
let checkUser = () => {
if (npos >= nusers.length) {
return next();
}
let user = nusers[npos++];
getFirstMessage(userIds[user - 1], (err, message) => {
expect(err).to.not.exist;
expect(message.subject).to.equal(subject);
expect(message.attachments.length).to.equal(3);
expect(message.parsed.attachments.length).to.equal(3);
for (let i = 0; i < message.attachments.length; i++) {
let hashA = crypto
.createHash('md5')
.update(message.attachments[i].raw)
.digest('hex');
let hashB = crypto
.createHash('md5')
.update(message.parsed.attachments[i].content)
.digest('hex');
expect(hashA).equal(hashB);
}
expect(message.parsed.to.value).deep.equal([
{ address: 'user1@example.com', name: 'User #1' },
{ address: 'user2@example.com', name: 'User #2' },
{ address: 'user3@example.com', name: 'User #3' },
{ address: 'user4@example.com', name: 'User #4' },
{ address: 'user5@example.com', name: 'User #5' }
]);
expect(message.parsed.headers.get('delivered-to').value[0].address).equal('user' + user + '@example.com');
setImmediate(checkUser);
});
};
setImmediate(checkUser);
};
let checkEncryptedUsers = next => {
let npos = 0;
let nusers = [2, 3];
let checkUser = () => {
if (npos >= nusers.length) {
return next();
}
let user = nusers[npos++];
getFirstMessage(userIds[user - 1], (err, message) => {
expect(err).to.not.exist;
expect(message.subject).to.equal(subject);
expect(message.parsed.to.value).deep.equal([
{ address: 'user1@example.com', name: 'User #1' },
{ address: 'user2@example.com', name: 'User #2' },
{ address: 'user3@example.com', name: 'User #3' },
{ address: 'user4@example.com', name: 'User #4' },
{ address: 'user5@example.com', name: 'User #5' }
]);
expect(message.parsed.headers.get('delivered-to').value[0].address).equal('user' + user + '@example.com');
expect(message.parsed.attachments.length).equal(2);
expect(message.parsed.attachments[0].contentType).equal('application/pgp-encrypted');
expect(message.parsed.attachments[0].content.toString()).equal('Version: 1\r\n');
expect(message.parsed.attachments[1].contentType).equal('application/octet-stream');
expect(message.parsed.attachments[1].filename).equal('encrypted.asc');
expect(message.parsed.attachments[1].size).gte(1000000);
setImmediate(checkUser);
});
};
setImmediate(checkUser);
};
checkNormalUsers(() => checkEncryptedUsers(() => done()));
}
);
});
it('should fetch messages from IMAP', done => {
let imagePng = new Buffer(
'iVBORw0KGgoAAAANSUhEUgAAABAAAAAQAQMAAAAlPW0iAAAABlBMVEUAAAD/' +
'//+l2Z/dAAAAM0lEQVR4nGP4/5/h/1+G/58ZDrAz3D/McH8yw83NDDeNGe4U' +
'g9C9zwz3gVLMDA/A6P9/AFGGFyjOXZtQAAAAAElFTkSuQmCC',
'base64'
);
let textTxt = 'Some notes about this e-mail';
let swanJpg = fs.readFileSync(__dirname + '/../examples/swan.jpg');
let checksums = [
crypto
.createHash('md5')
.update(imagePng)
.digest('hex'),
crypto
.createHash('md5')
.update(Buffer.from(textTxt))
.digest('hex'),
crypto
.createHash('md5')
.update(swanJpg)
.digest('hex')
];
const client = new BrowserBox('localhost', 9993, {
useSecureTransport: true,
auth: {
user: 'user4',
pass: 'secretpass'
},
id: {
name: 'My Client',
version: '0.1'
},
tls: {
rejectUnauthorized: false
}
});
client.onerror = err => {
expect(err).to.not.exist;
};
client.onclose = done;
client.onauth = () => {
client.listMailboxes((err, result) => {
expect(err).to.not.exist;
let folders = result.children.map(mbox => ({ name: mbox.name, specialUse: mbox.specialUse || false }));
expect(folders).to.deep.equal([
{ name: 'INBOX', specialUse: false },
{ name: 'Archive', specialUse: '\\Archive' },
{ name: 'Drafts', specialUse: '\\Drafts' },
{ name: 'Junk', specialUse: '\\Junk' },
{ name: 'Sent Mail', specialUse: '\\Sent' },
{ name: 'Trash', specialUse: '\\Trash' }
]);
client.selectMailbox('INBOX', { condstore: true }, (err, result) => {
expect(err).to.not.exist;
expect(result.exists).gte(1);
client.listMessages(result.exists, ['uid', 'flags', 'body.peek[]'], (err, messages) => {
expect(err).to.not.exist;
expect(messages.length).equal(1);
let messageInfo = messages[0];
simpleParser(messageInfo['body[]'], (err, parsed) => {
expect(err).to.not.exist;
checksums.forEach((checksum, i) => {
expect(checksum).to.equal(parsed.attachments[i].checksum);
});
client.close();
});
});
});
});
};
client.connect();
});
});