1
0
mirror of https://github.com/eleith/emailjs.git synced 2024-06-28 09:39:02 +00:00
emailjs/test/client.ts

293 lines
6.9 KiB
TypeScript
Raw Normal View History

2020-04-21 03:20:42 +00:00
import test from 'ava';
2020-05-27 12:49:21 +00:00
import { simpleParser } from 'mailparser';
import { SMTPServer } from 'smtp-server';
2020-04-21 03:20:42 +00:00
2020-05-27 18:15:55 +00:00
import { DEFAULT_TIMEOUT, SMTPClient, Message } from '../email';
2020-04-21 03:20:42 +00:00
type UnPromisify<T> = T extends Promise<infer U> ? U : T;
const port = 2526;
2020-05-27 18:15:55 +00:00
const client = new SMTPClient({
2020-04-23 04:26:49 +00:00
port,
user: 'pooh',
password: 'honey',
ssl: true,
});
2020-05-27 15:59:53 +00:00
const server = new SMTPServer({ secure: true });
2020-04-21 03:20:42 +00:00
2020-04-23 04:26:49 +00:00
const send = (
2020-05-27 12:34:05 +00:00
message: Message,
2020-05-27 12:49:21 +00:00
verify: (mail: UnPromisify<ReturnType<typeof simpleParser>>) => void,
2020-05-01 16:25:32 +00:00
done: () => void
2020-04-23 04:26:49 +00:00
) => {
2020-05-27 10:49:11 +00:00
server.onData = (stream, _session, callback: () => void) => {
2020-06-20 00:10:10 +00:00
simpleParser(stream, {
skipHtmlToText: true,
skipTextToHtml: true,
skipImageLinks: true,
} as Record<string, unknown>)
.then(verify)
.finally(done);
2020-04-21 03:20:42 +00:00
stream.on('end', callback);
};
2020-04-23 04:26:49 +00:00
client.send(message, (err) => {
2020-04-21 03:20:42 +00:00
if (err) {
throw err;
}
});
};
2020-05-01 16:25:32 +00:00
test.before.cb((t) => {
2020-04-25 03:03:31 +00:00
server.listen(port, function () {
server.onAuth = function (auth, _session, callback) {
2020-05-27 15:59:53 +00:00
if (auth.username === 'pooh' && auth.password === 'honey') {
2020-04-21 03:20:42 +00:00
callback(null, { user: 'pooh' });
} else {
return callback(new Error('invalid user / pass'));
}
};
2020-04-25 20:17:35 +00:00
t.end();
2020-04-21 03:20:42 +00:00
});
});
2020-05-01 16:25:32 +00:00
test.after.cb((t) => server.close(t.end));
2020-04-21 03:20:42 +00:00
2020-05-27 10:49:11 +00:00
test.cb('client invokes callback exactly once for invalid connection', (t) => {
t.plan(1);
2020-05-27 18:15:55 +00:00
const client = new SMTPClient({ host: 'bar.baz' });
2020-04-21 03:20:42 +00:00
const msg = {
2020-05-27 10:49:11 +00:00
from: 'foo@bar.baz',
to: 'foo@bar.baz',
subject: 'hello world',
text: 'hello world',
2020-04-21 03:20:42 +00:00
};
2020-05-27 12:34:05 +00:00
client.send(new Message(msg), (err) => {
2020-05-27 10:49:11 +00:00
t.not(err, null);
t.end();
});
});
test('client has a default connection timeout', (t) => {
const connectionOptions = {
user: 'username',
password: 'password',
host: '127.0.0.1',
port: 1234,
timeout: undefined as number | null | undefined,
};
2020-05-27 18:15:55 +00:00
t.is(new SMTPClient(connectionOptions).smtp.timeout, DEFAULT_TIMEOUT);
2020-05-27 10:49:11 +00:00
connectionOptions.timeout = null;
2020-05-27 18:15:55 +00:00
t.is(new SMTPClient(connectionOptions).smtp.timeout, DEFAULT_TIMEOUT);
2020-04-21 03:20:42 +00:00
2020-05-27 10:49:11 +00:00
connectionOptions.timeout = undefined;
2020-05-27 18:15:55 +00:00
t.is(new SMTPClient(connectionOptions).smtp.timeout, DEFAULT_TIMEOUT);
2020-04-21 03:20:42 +00:00
});
test('client deduplicates recipients', (t) => {
const msg = {
from: 'zelda@gmail.com',
to: 'gannon@gmail.com',
cc: 'gannon@gmail.com',
bcc: 'gannon@gmail.com',
};
const stack = client.createMessageStack(new Message(msg));
t.true(stack.to.length === 1);
t.is(stack.to[0].address, 'gannon@gmail.com');
});
test.cb('client accepts array recipients', (t) => {
const msg = new Message({
from: 'zelda@gmail.com',
to: ['gannon1@gmail.com'],
cc: ['gannon2@gmail.com'],
bcc: ['gannon3@gmail.com'],
});
msg.header.to = [msg.header.to as string];
msg.header.cc = [msg.header.cc as string];
msg.header.bcc = [msg.header.bcc as string];
msg.valid((isValid) => {
t.true(isValid);
const stack = client.createMessageStack(msg);
t.is(stack.to.length, 3);
t.deepEqual(
stack.to.map((x) => x.address),
['gannon1@gmail.com', 'gannon2@gmail.com', 'gannon3@gmail.com']
);
t.end();
});
});
test.cb('client accepts array sender', (t) => {
const msg = new Message({
from: ['zelda@gmail.com'],
to: ['gannon1@gmail.com'],
});
msg.header.from = [msg.header.from as string];
msg.valid((isValid) => {
t.true(isValid);
t.end();
});
});
2020-05-27 06:03:59 +00:00
test.cb('client rejects message without `from` header', (t) => {
const msg = {
subject: 'this is a test TEXT message from emailjs',
text: "It is hard to be brave when you're only a Very Small Animal.",
};
2020-05-27 12:34:05 +00:00
client.send(new Message(msg), (err) => {
2020-05-27 06:03:59 +00:00
t.true(err instanceof Error);
t.is(err?.message, 'Message must have a `from` header');
t.end();
});
});
2020-05-27 05:53:45 +00:00
test.cb('client rejects message without `to`, `cc`, or `bcc` header', (t) => {
const msg = {
subject: 'this is a test TEXT message from emailjs',
from: 'piglet@gmail.com',
text: "It is hard to be brave when you're only a Very Small Animal.",
};
2020-05-27 12:34:05 +00:00
client.send(new Message(msg), (err) => {
2020-05-27 05:53:45 +00:00
t.true(err instanceof Error);
t.is(
err?.message,
'Message must have at least one `to`, `cc`, or `bcc` header'
);
t.end();
});
});
2020-05-27 05:47:24 +00:00
2020-05-27 05:53:45 +00:00
test.cb('client allows message with only `cc` recipient header', (t) => {
2020-05-27 05:47:24 +00:00
const msg = {
subject: 'this is a test TEXT message from emailjs',
from: 'piglet@gmail.com',
cc: 'pooh@gmail.com',
text: "It is hard to be brave when you're only a Very Small Animal.",
};
send(
2020-05-27 12:34:05 +00:00
new Message(msg),
2020-05-27 05:47:24 +00:00
(mail) => {
t.is(mail.text, msg.text + '\n\n\n');
t.is(mail.subject, msg.subject);
t.is(mail.from?.text, msg.from);
t.is(mail.cc?.text, msg.cc);
},
t.end
);
});
2020-05-27 05:53:45 +00:00
test.cb('client allows message with only `bcc` recipient header', (t) => {
2020-05-27 05:47:24 +00:00
const msg = {
subject: 'this is a test TEXT message from emailjs',
from: 'piglet@gmail.com',
bcc: 'pooh@gmail.com',
text: "It is hard to be brave when you're only a Very Small Animal.",
};
send(
2020-05-27 12:34:05 +00:00
new Message(msg),
2020-05-27 05:47:24 +00:00
(mail) => {
t.is(mail.text, msg.text + '\n\n\n');
t.is(mail.subject, msg.subject);
t.is(mail.from?.text, msg.from);
2020-05-27 11:31:32 +00:00
t.is(mail.bcc, undefined);
2020-05-27 05:47:24 +00:00
},
t.end
);
});
2020-05-27 05:53:45 +00:00
test('client constructor throws if `password` supplied without `user`', (t) => {
2020-05-27 18:15:55 +00:00
t.notThrows(() => new SMTPClient({ user: 'anything', password: 'anything' }));
t.throws(() => new SMTPClient({ password: 'anything' }));
t.throws(
() =>
2020-05-27 18:15:55 +00:00
new SMTPClient({ username: 'anything', password: 'anything' } as Record<
string,
unknown
>)
);
});
test.cb('client supports greylisting', (t) => {
t.plan(2);
const msg = {
subject: 'this is a test TEXT message from emailjs',
from: 'piglet@gmail.com',
bcc: 'pooh@gmail.com',
text: "It is hard to be brave when you're only a Very Small Animal.",
};
const { onRcptTo } = server;
server.onRcptTo = (_address, _session, callback) => {
server.onRcptTo = (a, s, cb) => {
t.pass();
onRcptTo(a, s, cb);
};
const err = new Error('greylist');
((err as never) as { responseCode: number }).responseCode = 450;
callback(err);
};
client.send(new Message(msg), (err) => {
if (err) {
t.fail();
}
t.pass();
t.end();
});
});
test.cb('client only responds once to greylisting', (t) => {
t.plan(3);
const msg = {
subject: 'this is a test TEXT message from emailjs',
from: 'piglet@gmail.com',
bcc: 'pooh@gmail.com',
text: "It is hard to be brave when you're only a Very Small Animal.",
};
const greylistPort = 2527;
const greylistClient = new SMTPClient({
port: greylistPort,
user: 'pooh',
password: 'honey',
ssl: true,
});
const greylistServer = new SMTPServer({
secure: true,
onRcptTo(_address, _session, callback) {
t.pass();
const err = new Error('greylist');
((err as never) as { responseCode: number }).responseCode = 450;
callback(err);
},
onAuth(auth, _session, callback) {
if (auth.username === 'pooh' && auth.password === 'honey') {
callback(null, { user: 'pooh' });
} else {
return callback(new Error('invalid user / pass'));
}
},
});
greylistServer.listen(greylistPort, () => {
greylistClient.send(new Message(msg), (err) => {
if (err) {
t.pass();
t.end();
} else {
t.fail();
}
});
});
});