"use strict"; define(function (require) { var assert = require("assert"); var jsmime = require("jsmime"); var headeremitter = jsmime.headeremitter; var MockDate = require("test/mock_date"); function arrayTest(data, fn) { fn.toString = function () { let text = Function.prototype.toString.call(this); text = text.replace(/data\[([0-9]*)\]/g, function (m, p) { return JSON.stringify(data[p]); }); return text; }; return test(JSON.stringify(data[0]), fn); } suite("headeremitter", function () { suite("addAddresses", function () { let handler = { reset(expected) { this.output = ""; this.expected = expected; }, deliverData(data) { this.output += data; }, deliverEOF() { assert.equal(this.output, this.expected + "\r\n"); for (let line of this.output.split("\r\n")) { assert.ok(line.length <= 30, "Line is too long"); } }, }; let header_tests = [ [[{ name: "", email: "" }], ""], [[{ name: "", email: "a@example.com" }], "a@example.com"], [ [{ name: "John Doe", email: "a@example.com" }], "John Doe ", ], [ [ { name: "", email: "a@b.c" }, { name: "", email: "b@b.c" }, ], "a@b.c, b@b.c", ], [ [ { name: "JD", email: "a@a.c" }, { name: "SD", email: "b@b.c" }, ], "JD , SD ", ], [ [ { name: "John Doe", email: "a@example.com" }, { name: "Sally Doe", email: "b@example.com" }, ], "John Doe ,\r\n Sally Doe ", ], [ [ { name: "My name is really long and I split somewhere", email: "a@a.c", }, ], "My name is really long and I\r\n split somewhere ", ], // Note that the name is 29 chars here, so adding the email needs a break. [ [{ name: "My name is really really long", email: "a@a.c" }], "My name is really really long\r\n ", ], [ [ { name: "", email: "a@a.c" }, { name: "This name is long", email: "b@b.c" }, ], "a@a.c,\r\n This name is long ", ], [ [ { name: "", email: "a@a.c" }, { name: "This name is also long", email: "b@b.c" }, ], "a@a.c,\r\n This name is also long\r\n ", ], [[{ name: "", email: "hi!bad@all.com" }], '"hi!bad"@all.com'], [[{ name: "", email: '"hi!bad"@all.com' }], '"hi!bad"@all.com'], [[{ name: "Doe, John", email: "a@a.com" }], '"Doe, John" '], // This one violates the line length, so it underquotes instead. [ [ { name: "A really, really long name to quote", email: "a@example.com", }, ], 'A "really," really long name\r\n to quote ', ], [ [ { name: "Group", group: [ { name: "", email: "a@a.c" }, { name: "", email: "b@b.c" }, ], }, ], "Group: a@a.c, b@b.c;", ], [[{ name: "No email address", email: "" }], "No email address"], [ [{ name: "]user[ domain", email: "user@d.com" }], '"]user[ domain" ', ], [ [ { name: "Group", group: [ { name: "]u[ d", email: "a@a.c" }, { name: "]u[ c", email: "b@b.c" }, ], }, ], 'Group: "]u[ d" ,\r\n "]u[ c" ;', ], [ [{ name: "user@domain", email: "user@d.com" }], '"user@domain" ', ], [ [ { name: "Group", group: [ { name: "u@d", email: "a@a.c" }, { name: "u@c", email: "b@b.c" }, ], }, ], 'Group: "u@d" ,\r\n "u@c" ;', ], ]; header_tests.forEach(function (data) { arrayTest(data, function () { let emitter = headeremitter.makeStreamingEmitter(handler, { softMargin: 30, useASCII: false, }); handler.reset(data[1]); emitter.addAddresses(data[0]); emitter.finish(true); }); }); }); suite("addAddresses (RFC 2047)", function () { let handler = { reset(expected) { this.output = ""; this.expected = expected; }, deliverData(data) { this.output += data; }, deliverEOF() { assert.equal(this.output, this.expected + "\r\n"); for (let line of this.output.split("\r\n")) { assert.ok(line.length <= 30, "Line is too long"); } }, }; let header_tests = [ [[{ name: "\u0436", email: "a@a.c" }], "=?UTF-8?B?0LY=?= "], [ [{ name: "dioxyg\u00e8ne", email: "a@a.c" }], "=?UTF-8?Q?dioxyg=C3=A8ne?=\r\n ", ], // Prefer QP if base64 and QP are exactly the same length [ [{ name: "oxyg\u00e8ne", email: "a@a.c" }], // =?UTF-8?B?b3h5Z8OobmU=?= "=?UTF-8?Q?oxyg=C3=A8ne?=\r\n ", ], [ [ { name: "\ud83d\udca9\ud83d\udca9\ud83d\udca9\ud83d\udca9", email: "a@a.c", }, ], "=?UTF-8?B?8J+SqfCfkqnwn5Kp?=\r\n =?UTF-8?B?8J+SqQ==?= ", ], // Bug 1088975: Since the encoded-word should be recognized as an atom, // encode commas. [ [{ name: "B\u00fcg 1088975, FirstName", email: "a@b.c" }], "=?UTF-8?Q?B=C3=BCg_1088975?=\r\n" + " =?UTF-8?Q?=2C_FirstName?=\r\n ", ], ]; header_tests.forEach(function (data) { arrayTest(data, function () { let emitter = headeremitter.makeStreamingEmitter(handler, { softMargin: 30, useASCII: true, }); handler.reset(data[1]); emitter.addAddresses(data[0]); emitter.finish(true); }); }); }); suite("addUnstructured (RFC 2047)", function () { let handler = { reset(expected) { this.output = ""; this.expected = expected; }, deliverData(data) { this.output += data; }, deliverEOF() { assert.equal(this.output, this.expected + "\r\n"); for (let line of this.output.split("\r\n")) { assert.ok(line.length <= 30, "Line is too long"); } }, }; let header_tests = [ ["My house burned down!", "My house burned down!"], // Which of the 32 "special" characters need to be encoded in QP encoding? // Note: Encoding is forced by adding a \x7f at the end. // These 5 don't need encoding: [" ! * + - / \x7f", "=?UTF-8?Q?_!_*_+_-_/_=7F?="], // Bug 1438590: RFC2047 [5. (3)] requests the // encoding of these 27 "special" characters: // " # $ % & ' ( ) , . : ; < = > ? @ [ \ ] ^ _ ` { | } ~. // Note: If there are enough characters for padding, // QP is used and not base64. ['Test " # \x7f', "=?UTF-8?Q?Test_=22_=23_=7F?="], ["Test $ % \x7f", "=?UTF-8?Q?Test_=24_=25_=7F?="], ["Test & ' \x7f", "=?UTF-8?Q?Test_=26_=27_=7F?="], ["Test ( ) \x7f", "=?UTF-8?Q?Test_=28_=29_=7F?="], ["Test , . \x7f", "=?UTF-8?Q?Test_=2C_=2E_=7F?="], ["Test : ; \x7f", "=?UTF-8?Q?Test_=3A_=3B_=7F?="], ["Test < = \x7f", "=?UTF-8?Q?Test_=3C_=3D_=7F?="], ["Test > ? \x7f", "=?UTF-8?Q?Test_=3E_=3F_=7F?="], ["Test @ [ \x7f", "=?UTF-8?Q?Test_=40_=5B_=7F?="], ["Test \\ ] \x7f", "=?UTF-8?Q?Test_=5C_=5D_=7F?="], ["Test ^ _ \x7f", "=?UTF-8?Q?Test_=5E_=5F_=7F?="], ["Test ` { \x7f", "=?UTF-8?Q?Test_=60_=7B_=7F?="], ["Test | } \x7f", "=?UTF-8?Q?Test_=7C_=7D_=7F?="], ["Test ~ \x7f", "=?UTF-8?Q?Test_=7E_=7F?="], // But the 32 printable "special" characters don't need it in the first place! [ "! \" # $ % & ' ( ) * + , - . / : ; < = > ? @ [ \\ ] ^ _ ` { | } ~", "! \" # $ % & ' ( ) * + , - . /\r\n" + " : ; < = > ? @ [ \\ ] ^ _ ` { |\r\n" + " } ~", ], // Test to make sure 2047-encoding chooses the right values. ["\u001f", "=?UTF-8?Q?=1F?="], ["\u001fa", "=?UTF-8?Q?=1Fa?="], ["\u001faa", "=?UTF-8?B?H2Fh?="], ["\u001faaa", "=?UTF-8?Q?=1Faaa?="], ["\u001faaa\u001f", "=?UTF-8?B?H2FhYR8=?="], ["\u001faaa\u001fa", "=?UTF-8?B?H2FhYR9h?="], ["\u001faaa\u001faa", "=?UTF-8?Q?=1Faaa=1Faa?="], ["\u001faaa\u001faa\u001faaaa", "=?UTF-8?B?H2FhYR9hYR9hYWFh?="], // Make sure line breaking works right at the edge cases ["\u001faaa\u001faaaaaaaaa", "=?UTF-8?Q?=1Faaa=1Faaaaaaaaa?="], [ "\u001faaa\u001faaaaaaaaaa", "=?UTF-8?Q?=1Faaa=1Faaaaaaaaa?=\r\n =?UTF-8?Q?a?=", ], // Choose base64/qp independently for each word [ "\ud83d\udca9\ud83d\udca9\ud83d\udca9a", "=?UTF-8?B?8J+SqfCfkqnwn5Kp?=\r\n =?UTF-8?Q?a?=", ], // Don't split a surrogate character! [ "a\ud83d\udca9\ud83d\udca9\ud83d\udca9a", "=?UTF-8?B?YfCfkqnwn5Kp?=\r\n =?UTF-8?B?8J+SqWE=?=", ], // Spacing a UTF-8 string [ "L'oxyg\u00e8ne est un \u00e9l\u00e9ment chimique du groupe des " + "chalcog\u00e8nes", // 1 2 3 // 123456789012345678901234567890 "=?UTF-8?B?TCdveHlnw6huZSBl?=\r\n" + " =?UTF-8?B?c3QgdW4gw6lsw6lt?=\r\n" + " =?UTF-8?Q?ent_chimique_du_g?=\r\n" + " =?UTF-8?Q?roupe_des_chalcog?=\r\n" + " =?UTF-8?B?w6huZXM=?=", ], ]; header_tests.forEach(function (data) { arrayTest(data, function () { let emitter = headeremitter.makeStreamingEmitter(handler, { softMargin: 30, useASCII: true, }); handler.reset(data[1]); emitter.addUnstructured(data[0]); emitter.finish(true); }); }); }); suite("addDate", function () { let handler = { reset(expected) { this.output = ""; this.expected = expected; }, deliverData(data) { this.output += data; }, deliverEOF() { assert.equal(this.output, this.expected + "\r\n"); }, }; let header_tests = [ // Test basic day/month names ["2000-01-01T00:00:00Z", "Sat, 1 Jan 2000 00:00:00 +0000"], ["2000-02-01T00:00:00Z", "Tue, 1 Feb 2000 00:00:00 +0000"], ["2000-03-01T00:00:00Z", "Wed, 1 Mar 2000 00:00:00 +0000"], ["2000-04-01T00:00:00Z", "Sat, 1 Apr 2000 00:00:00 +0000"], ["2000-05-01T00:00:00Z", "Mon, 1 May 2000 00:00:00 +0000"], ["2000-06-01T00:00:00Z", "Thu, 1 Jun 2000 00:00:00 +0000"], ["2000-07-01T00:00:00Z", "Sat, 1 Jul 2000 00:00:00 +0000"], ["2000-08-01T00:00:00Z", "Tue, 1 Aug 2000 00:00:00 +0000"], ["2000-09-01T00:00:00Z", "Fri, 1 Sep 2000 00:00:00 +0000"], ["2000-10-01T00:00:00Z", "Sun, 1 Oct 2000 00:00:00 +0000"], ["2000-11-01T00:00:00Z", "Wed, 1 Nov 2000 00:00:00 +0000"], ["2000-12-01T00:00:00Z", "Fri, 1 Dec 2000 00:00:00 +0000"], // Test timezone offsets ["2000-06-01T12:00:00Z", "Thu, 1 Jun 2000 12:00:00 +0000"], ["2000-06-01T12:00:00+0100", "Thu, 1 Jun 2000 12:00:00 +0100"], ["2000-06-01T12:00:00+0130", "Thu, 1 Jun 2000 12:00:00 +0130"], ["2000-06-01T12:00:00-0100", "Thu, 1 Jun 2000 12:00:00 -0100"], ["2000-06-01T12:00:00-0130", "Thu, 1 Jun 2000 12:00:00 -0130"], ["2000-06-01T12:00:00+1345", "Thu, 1 Jun 2000 12:00:00 +1345"], ["2000-06-01T12:00:00-1200", "Thu, 1 Jun 2000 12:00:00 -1200"], ["2000-06-01T12:00:00+1337", "Thu, 1 Jun 2000 12:00:00 +1337"], ["2000-06-01T12:00:00+0101", "Thu, 1 Jun 2000 12:00:00 +0101"], ["2000-06-01T12:00:00-1337", "Thu, 1 Jun 2000 12:00:00 -1337"], // Try some varying hour, minute, and second amounts, to double-check // padding and time dates. ["2000-06-01T01:02:03Z", "Thu, 1 Jun 2000 01:02:03 +0000"], ["2000-06-01T23:13:17Z", "Thu, 1 Jun 2000 23:13:17 +0000"], ["2000-06-01T00:05:04Z", "Thu, 1 Jun 2000 00:05:04 +0000"], ["2000-06-01T23:59:59Z", "Thu, 1 Jun 2000 23:59:59 +0000"], ["2000-06-01T13:17:40Z", "Thu, 1 Jun 2000 13:17:40 +0000"], ["2000-06-01T11:15:34Z", "Thu, 1 Jun 2000 11:15:34 +0000"], ["2000-06-01T04:09:09Z", "Thu, 1 Jun 2000 04:09:09 +0000"], ["2000-06-01T04:10:10Z", "Thu, 1 Jun 2000 04:10:10 +0000"], ["2000-06-01T09:13:17Z", "Thu, 1 Jun 2000 09:13:17 +0000"], ["2000-06-01T13:12:14Z", "Thu, 1 Jun 2000 13:12:14 +0000"], ["2000-06-01T14:16:48Z", "Thu, 1 Jun 2000 14:16:48 +0000"], // Try varying month, date, and year values. ["2000-01-31T00:00:00Z", "Mon, 31 Jan 2000 00:00:00 +0000"], ["2000-02-28T00:00:00Z", "Mon, 28 Feb 2000 00:00:00 +0000"], ["2000-02-29T00:00:00Z", "Tue, 29 Feb 2000 00:00:00 +0000"], ["2001-02-28T00:00:00Z", "Wed, 28 Feb 2001 00:00:00 +0000"], ["2000-03-31T00:00:00Z", "Fri, 31 Mar 2000 00:00:00 +0000"], ["2000-04-30T00:00:00Z", "Sun, 30 Apr 2000 00:00:00 +0000"], ["2000-05-31T00:00:00Z", "Wed, 31 May 2000 00:00:00 +0000"], ["2000-06-30T00:00:00Z", "Fri, 30 Jun 2000 00:00:00 +0000"], ["2000-07-31T00:00:00Z", "Mon, 31 Jul 2000 00:00:00 +0000"], ["2000-08-31T00:00:00Z", "Thu, 31 Aug 2000 00:00:00 +0000"], ["2000-09-30T00:00:00Z", "Sat, 30 Sep 2000 00:00:00 +0000"], ["2000-10-31T00:00:00Z", "Tue, 31 Oct 2000 00:00:00 +0000"], ["2000-11-30T00:00:00Z", "Thu, 30 Nov 2000 00:00:00 +0000"], ["2000-12-31T00:00:00Z", "Sun, 31 Dec 2000 00:00:00 +0000"], ["1900-01-01T00:00:00Z", "Mon, 1 Jan 1900 00:00:00 +0000"], ["9999-12-31T23:59:59Z", "Fri, 31 Dec 9999 23:59:59 +0000"], // Tests that are not actually missing: // We don't actually need to test daylight savings time issues, so long as // getTimezoneOffset is correct. We've confirmed black-box that the value // is being directly queried on every instance, since we have tests that // make MockDate.getTimezoneOffset return different values. // In addition, ES6 Date objects don't support leap seconds. Invalid dates // per RFC 5322 are handled in a later run of code. ]; header_tests.forEach(function (data) { arrayTest(data, function () { let emitter = headeremitter.makeStreamingEmitter(handler, {}); handler.reset(data[1]); emitter.addDate(new MockDate(data[0])); emitter.finish(true); }); }); // An invalid date should throw an error instead of make a malformed header. test("Invalid dates", function () { let emitter = headeremitter.makeStreamingEmitter(handler, {}); assert.throws(function () { emitter.addDate(new Date(NaN)); }, /Cannot encode an invalid date/); assert.throws(function () { emitter.addDate(new Date("1850-01-01")); }, /Date year is out of encodable range/); assert.throws(function () { emitter.addDate(new Date("10000-01-01")); }, /Cannot encode an invalid date/); }); // Test preferred breaking for the date header. test("Break spot", function () { let emitter = headeremitter.makeStreamingEmitter(handler, { softMargin: 30, }); handler.reset("Overly-Long-Date:\r\n Sat, 1 Jan 2000 00:00:00 +0000"); emitter.addHeaderName("Overly-Long-Date"); emitter.addDate(new MockDate("2000-01-01T00:00:00Z")); emitter.finish(); }); test("Correctness of date", function () { let emitter = headeremitter.makeStreamingEmitter(handler, {}); handler.reset(); let now = new Date(); emitter.addDate(now); emitter.finish(); // All engines can parse the date strings we produce let reparsed = new Date(handler.output); // Now and reparsed should be correct to second-level precision. assert.equal(reparsed.getMilliseconds(), 0); assert.equal(now.getTime() - now.getMilliseconds(), reparsed.getTime()); }); }); suite("Header lengths", function () { let handler = { reset(expected) { this.output = ""; this.expected = expected; }, deliverData(data) { this.output += data; }, deliverEOF() { assert.equal(this.output, this.expected + "\r\n"); }, }; let header_tests = [ [ [{ name: "Supercalifragilisticexpialidocious", email: "a@b.c" }], "Supercalifragilisticexpialidocious\r\n ", ], [ [ { email: "supercalifragilisticexpialidocious@" + "the.longest.domain.name.in.the.world.invalid", }, ], "supercalifragilisticexpialidocious\r\n" + " @the.longest.domain.name.in.the.world.invalid", ], [ [ { name: "Lopadotemachoselachogaleokranioleipsanodrimhypotrimmatosilphi" + "paraomelitokatakechymenokichlepikossyphophattoperisteralektryonoptek" + "ephalliokigklopeleiolagoiosiraiobaphetraganopterygon", email: "a@b.c", }, ], new Error(), ], ]; header_tests.forEach(function (data) { arrayTest(data, function () { let emitter = headeremitter.makeStreamingEmitter(handler, { softMargin: 30, hardMargin: 50, useASCII: false, }); handler.reset(data[1]); if (data[1] instanceof Error) { assert.throws(function () { emitter.addAddresses(data[0]); }, /Cannot encode/); } else { assert.doesNotThrow(function () { emitter.addAddresses(data[0]); }); emitter.finish(true); } }); }); }); }); });