2017-01-10 01:40:57 +09:00
|
|
|
|
/* Copyright 2017 Mozilla Foundation
|
|
|
|
|
*
|
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
|
*
|
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
*
|
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
|
* limitations under the License.
|
|
|
|
|
*/
|
2012-06-24 05:35:59 +09:00
|
|
|
|
|
2017-04-17 05:30:27 +09:00
|
|
|
|
import {
|
2019-02-24 00:14:31 +09:00
|
|
|
|
bytesToString,
|
|
|
|
|
createValidAbsoluteUrl,
|
2020-08-04 02:44:04 +09:00
|
|
|
|
getModificationDate,
|
2022-03-08 01:41:41 +09:00
|
|
|
|
PromiseCapability,
|
2019-10-14 20:19:41 +09:00
|
|
|
|
string32,
|
|
|
|
|
stringToBytes,
|
|
|
|
|
stringToPDFString,
|
2020-01-02 20:00:16 +09:00
|
|
|
|
} from "../../src/shared/util.js";
|
Enable auto-formatting of the entire code-base using Prettier (issue 11444)
Note that Prettier, purposely, has only limited [configuration options](https://prettier.io/docs/en/options.html). The configuration file is based on [the one in `mozilla central`](https://searchfox.org/mozilla-central/source/.prettierrc) with just a few additions (to avoid future breakage if the defaults ever changes).
Prettier is being used for a couple of reasons:
- To be consistent with `mozilla-central`, where Prettier is already in use across the tree.
- To ensure a *consistent* coding style everywhere, which is automatically enforced during linting (since Prettier is used as an ESLint plugin). This thus ends "all" formatting disussions once and for all, removing the need for review comments on most stylistic matters.
Many ESLint options are now redundant, and I've tried my best to remove all the now unnecessary options (but I may have missed some).
Note also that since Prettier considers the `printWidth` option as a guide, rather than a hard rule, this patch resorts to a small hack in the ESLint config to ensure that *comments* won't become too long.
*Please note:* This patch is generated automatically, by appending the `--fix` argument to the ESLint call used in the `gulp lint` task. It will thus require some additional clean-up, which will be done in a *separate* commit.
(On a more personal note, I'll readily admit that some of the changes Prettier makes are *extremely* ugly. However, in the name of consistency we'll probably have to live with that.)
2019-12-25 23:59:37 +09:00
|
|
|
|
|
2020-04-14 19:28:14 +09:00
|
|
|
|
describe("util", function () {
|
|
|
|
|
describe("bytesToString", function () {
|
|
|
|
|
it("handles non-array arguments", function () {
|
|
|
|
|
expect(function () {
|
2017-12-03 04:25:51 +09:00
|
|
|
|
bytesToString(null);
|
|
|
|
|
}).toThrow(new Error("Invalid argument for bytesToString"));
|
|
|
|
|
});
|
|
|
|
|
|
2020-04-14 19:28:14 +09:00
|
|
|
|
it("handles array arguments with a length not exceeding the maximum", function () {
|
2017-12-03 04:25:51 +09:00
|
|
|
|
expect(bytesToString(new Uint8Array([]))).toEqual("");
|
|
|
|
|
expect(bytesToString(new Uint8Array([102, 111, 111]))).toEqual("foo");
|
|
|
|
|
});
|
|
|
|
|
|
2020-04-14 19:28:14 +09:00
|
|
|
|
it("handles array arguments with a length exceeding the maximum", function () {
|
2017-12-03 04:25:51 +09:00
|
|
|
|
const length = 10000; // Larger than MAX_ARGUMENT_COUNT = 8192.
|
|
|
|
|
|
|
|
|
|
// Create an array with `length` 'a' character codes.
|
2020-01-24 17:48:21 +09:00
|
|
|
|
const bytes = new Uint8Array(length);
|
2017-12-03 04:25:51 +09:00
|
|
|
|
for (let i = 0; i < length; i++) {
|
|
|
|
|
bytes[i] = "a".charCodeAt(0);
|
|
|
|
|
}
|
|
|
|
|
|
2022-03-30 22:38:07 +09:00
|
|
|
|
// Create a string with `length` 'a' characters.
|
|
|
|
|
const string = "a".repeat(length);
|
2017-12-03 04:25:51 +09:00
|
|
|
|
|
|
|
|
|
expect(bytesToString(bytes)).toEqual(string);
|
|
|
|
|
});
|
|
|
|
|
});
|
|
|
|
|
|
2020-04-14 19:28:14 +09:00
|
|
|
|
describe("string32", function () {
|
|
|
|
|
it("converts unsigned 32-bit integers to strings", function () {
|
2019-02-11 02:58:52 +09:00
|
|
|
|
expect(string32(0x74727565)).toEqual("true");
|
|
|
|
|
expect(string32(0x74797031)).toEqual("typ1");
|
|
|
|
|
expect(string32(0x4f54544f)).toEqual("OTTO");
|
|
|
|
|
});
|
|
|
|
|
});
|
|
|
|
|
|
2020-04-14 19:28:14 +09:00
|
|
|
|
describe("stringToBytes", function () {
|
|
|
|
|
it("handles non-string arguments", function () {
|
|
|
|
|
expect(function () {
|
2017-12-03 04:25:51 +09:00
|
|
|
|
stringToBytes(null);
|
|
|
|
|
}).toThrow(new Error("Invalid argument for stringToBytes"));
|
|
|
|
|
});
|
|
|
|
|
|
2020-04-14 19:28:14 +09:00
|
|
|
|
it("handles string arguments", function () {
|
2017-12-03 04:25:51 +09:00
|
|
|
|
expect(stringToBytes("")).toEqual(new Uint8Array([]));
|
|
|
|
|
expect(stringToBytes("foo")).toEqual(new Uint8Array([102, 111, 111]));
|
|
|
|
|
});
|
|
|
|
|
});
|
|
|
|
|
|
2020-04-14 19:28:14 +09:00
|
|
|
|
describe("stringToPDFString", function () {
|
|
|
|
|
it("handles ISO Latin 1 strings", function () {
|
2020-01-24 17:48:21 +09:00
|
|
|
|
const str = "\x8Dstring\x8E";
|
2015-09-10 19:23:45 +09:00
|
|
|
|
expect(stringToPDFString(str)).toEqual("\u201Cstring\u201D");
|
|
|
|
|
});
|
|
|
|
|
|
2020-04-14 19:28:14 +09:00
|
|
|
|
it("handles UTF-16 big-endian strings", function () {
|
2020-01-24 17:48:21 +09:00
|
|
|
|
const str = "\xFE\xFF\x00\x73\x00\x74\x00\x72\x00\x69\x00\x6E\x00\x67";
|
2015-09-10 19:23:45 +09:00
|
|
|
|
expect(stringToPDFString(str)).toEqual("string");
|
|
|
|
|
});
|
|
|
|
|
|
2023-11-25 04:17:15 +09:00
|
|
|
|
it("handles incomplete UTF-16 big-endian strings", function () {
|
|
|
|
|
const str = "\xFE\xFF\x00\x73\x00\x74\x00\x72\x00\x69\x00\x6E\x00";
|
|
|
|
|
expect(stringToPDFString(str)).toEqual("strin");
|
|
|
|
|
});
|
|
|
|
|
|
2020-04-14 19:28:14 +09:00
|
|
|
|
it("handles UTF-16 little-endian strings", function () {
|
2020-01-24 17:48:21 +09:00
|
|
|
|
const str = "\xFF\xFE\x73\x00\x74\x00\x72\x00\x69\x00\x6E\x00\x67\x00";
|
2019-11-05 19:52:30 +09:00
|
|
|
|
expect(stringToPDFString(str)).toEqual("string");
|
|
|
|
|
});
|
|
|
|
|
|
2023-11-25 04:17:15 +09:00
|
|
|
|
it("handles incomplete UTF-16 little-endian strings", function () {
|
|
|
|
|
const str = "\xFF\xFE\x73\x00\x74\x00\x72\x00\x69\x00\x6E\x00\x67";
|
|
|
|
|
expect(stringToPDFString(str)).toEqual("strin");
|
|
|
|
|
});
|
|
|
|
|
|
2022-01-15 01:58:47 +09:00
|
|
|
|
it("handles UTF-8 strings", function () {
|
|
|
|
|
const simpleStr = "\xEF\xBB\xBF\x73\x74\x72\x69\x6E\x67";
|
|
|
|
|
expect(stringToPDFString(simpleStr)).toEqual("string");
|
|
|
|
|
|
|
|
|
|
const complexStr =
|
|
|
|
|
"\xEF\xBB\xBF\xE8\xA1\xA8\xE3\x83\x9D\xE3\x81\x82\x41\xE9\xB7\x97" +
|
|
|
|
|
"\xC5\x92\xC3\xA9\xEF\xBC\xA2\xE9\x80\x8D\xC3\x9C\xC3\x9F\xC2\xAA" +
|
|
|
|
|
"\xC4\x85\xC3\xB1\xE4\xB8\x82\xE3\x90\x80\xF0\xA0\x80\x80";
|
|
|
|
|
expect(stringToPDFString(complexStr)).toEqual(
|
|
|
|
|
"表ポあA鷗ŒéB逍Üߪąñ丂㐀𠀀"
|
|
|
|
|
);
|
|
|
|
|
});
|
|
|
|
|
|
2020-04-14 19:28:14 +09:00
|
|
|
|
it("handles empty strings", function () {
|
2015-09-10 19:23:45 +09:00
|
|
|
|
// ISO Latin 1
|
2020-01-24 17:48:21 +09:00
|
|
|
|
const str1 = "";
|
2015-09-10 19:23:45 +09:00
|
|
|
|
expect(stringToPDFString(str1)).toEqual("");
|
|
|
|
|
|
|
|
|
|
// UTF-16BE
|
2020-01-24 17:48:21 +09:00
|
|
|
|
const str2 = "\xFE\xFF";
|
2015-09-10 19:23:45 +09:00
|
|
|
|
expect(stringToPDFString(str2)).toEqual("");
|
2019-11-05 19:52:30 +09:00
|
|
|
|
|
|
|
|
|
// UTF-16LE
|
2020-01-24 17:48:21 +09:00
|
|
|
|
const str3 = "\xFF\xFE";
|
2019-11-05 19:52:30 +09:00
|
|
|
|
expect(stringToPDFString(str3)).toEqual("");
|
2022-01-15 01:58:47 +09:00
|
|
|
|
|
|
|
|
|
// UTF-8
|
|
|
|
|
const str4 = "\xEF\xBB\xBF";
|
|
|
|
|
expect(stringToPDFString(str4)).toEqual("");
|
2015-09-10 19:23:45 +09:00
|
|
|
|
});
|
2023-11-25 04:17:15 +09:00
|
|
|
|
|
|
|
|
|
it("handles strings with language code", function () {
|
|
|
|
|
// ISO Latin 1
|
|
|
|
|
const str1 = "hello \x1benUS\x1bworld";
|
|
|
|
|
expect(stringToPDFString(str1)).toEqual("hello world");
|
|
|
|
|
|
|
|
|
|
// UTF-16BE
|
|
|
|
|
const str2 =
|
|
|
|
|
"\xFE\xFF\x00h\x00e\x00l\x00l\x00o\x00 \x00\x1b\x00e\x00n\x00U\x00S\x00\x1b\x00w\x00o\x00r\x00l\x00d";
|
|
|
|
|
expect(stringToPDFString(str2)).toEqual("hello world");
|
|
|
|
|
|
|
|
|
|
// UTF-16LE
|
|
|
|
|
const str3 =
|
|
|
|
|
"\xFF\xFEh\x00e\x00l\x00l\x00o\x00 \x00\x1b\x00e\x00n\x00U\x00S\x00\x1b\x00w\x00o\x00r\x00l\x00d\x00";
|
|
|
|
|
expect(stringToPDFString(str3)).toEqual("hello world");
|
|
|
|
|
});
|
2015-09-10 19:23:45 +09:00
|
|
|
|
});
|
2015-09-30 19:59:50 +09:00
|
|
|
|
|
2020-04-14 19:28:14 +09:00
|
|
|
|
describe("ReadableStream", function () {
|
|
|
|
|
it("should return an Object", function () {
|
2020-01-24 17:48:21 +09:00
|
|
|
|
const readable = new ReadableStream();
|
2017-04-14 00:16:52 +09:00
|
|
|
|
expect(typeof readable).toEqual("object");
|
|
|
|
|
});
|
|
|
|
|
|
2020-04-14 19:28:14 +09:00
|
|
|
|
it("should have property getReader", function () {
|
2020-01-24 17:48:21 +09:00
|
|
|
|
const readable = new ReadableStream();
|
2017-04-14 00:16:52 +09:00
|
|
|
|
expect(typeof readable.getReader).toEqual("function");
|
|
|
|
|
});
|
|
|
|
|
});
|
2018-08-02 17:32:06 +09:00
|
|
|
|
|
2020-04-14 19:28:14 +09:00
|
|
|
|
describe("URL", function () {
|
|
|
|
|
it("should return an Object", function () {
|
2018-08-02 17:32:06 +09:00
|
|
|
|
const url = new URL("https://example.com");
|
|
|
|
|
expect(typeof url).toEqual("object");
|
|
|
|
|
});
|
|
|
|
|
|
2020-04-14 19:28:14 +09:00
|
|
|
|
it("should have property `href`", function () {
|
2018-08-02 17:32:06 +09:00
|
|
|
|
const url = new URL("https://example.com");
|
|
|
|
|
expect(typeof url.href).toEqual("string");
|
|
|
|
|
});
|
|
|
|
|
});
|
2018-09-11 22:16:07 +09:00
|
|
|
|
|
2020-04-14 19:28:14 +09:00
|
|
|
|
describe("createValidAbsoluteUrl", function () {
|
|
|
|
|
it("handles invalid URLs", function () {
|
2018-09-11 22:16:07 +09:00
|
|
|
|
expect(createValidAbsoluteUrl(undefined, undefined)).toEqual(null);
|
|
|
|
|
expect(createValidAbsoluteUrl(null, null)).toEqual(null);
|
|
|
|
|
expect(createValidAbsoluteUrl("/foo", "/bar")).toEqual(null);
|
|
|
|
|
});
|
|
|
|
|
|
2020-06-30 00:15:14 +09:00
|
|
|
|
it("handles URLs that do not use an allowed protocol", function () {
|
2018-09-11 22:16:07 +09:00
|
|
|
|
expect(createValidAbsoluteUrl("magnet:?foo", null)).toEqual(null);
|
|
|
|
|
});
|
|
|
|
|
|
2020-06-30 00:15:14 +09:00
|
|
|
|
it("correctly creates a valid URL for allowed protocols", function () {
|
2018-09-11 22:16:07 +09:00
|
|
|
|
// `http` protocol
|
|
|
|
|
expect(
|
|
|
|
|
createValidAbsoluteUrl("http://www.mozilla.org/foo", null)
|
|
|
|
|
).toEqual(new URL("http://www.mozilla.org/foo"));
|
|
|
|
|
expect(createValidAbsoluteUrl("/foo", "http://www.mozilla.org")).toEqual(
|
|
|
|
|
new URL("http://www.mozilla.org/foo")
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// `https` protocol
|
|
|
|
|
expect(
|
|
|
|
|
createValidAbsoluteUrl("https://www.mozilla.org/foo", null)
|
|
|
|
|
).toEqual(new URL("https://www.mozilla.org/foo"));
|
|
|
|
|
expect(createValidAbsoluteUrl("/foo", "https://www.mozilla.org")).toEqual(
|
|
|
|
|
new URL("https://www.mozilla.org/foo")
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// `ftp` protocol
|
|
|
|
|
expect(createValidAbsoluteUrl("ftp://www.mozilla.org/foo", null)).toEqual(
|
|
|
|
|
new URL("ftp://www.mozilla.org/foo")
|
|
|
|
|
);
|
|
|
|
|
expect(createValidAbsoluteUrl("/foo", "ftp://www.mozilla.org")).toEqual(
|
|
|
|
|
new URL("ftp://www.mozilla.org/foo")
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// `mailto` protocol (base URLs have no meaning and should yield `null`)
|
|
|
|
|
expect(createValidAbsoluteUrl("mailto:foo@bar.baz", null)).toEqual(
|
|
|
|
|
new URL("mailto:foo@bar.baz")
|
|
|
|
|
);
|
|
|
|
|
expect(createValidAbsoluteUrl("/foo", "mailto:foo@bar.baz")).toEqual(
|
|
|
|
|
null
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// `tel` protocol (base URLs have no meaning and should yield `null`)
|
|
|
|
|
expect(createValidAbsoluteUrl("tel:+0123456789", null)).toEqual(
|
|
|
|
|
new URL("tel:+0123456789")
|
|
|
|
|
);
|
|
|
|
|
expect(createValidAbsoluteUrl("/foo", "tel:0123456789")).toEqual(null);
|
|
|
|
|
});
|
|
|
|
|
});
|
2019-02-02 21:10:06 +09:00
|
|
|
|
|
2022-03-08 01:41:41 +09:00
|
|
|
|
describe("PromiseCapability", function () {
|
2021-04-12 02:26:38 +09:00
|
|
|
|
it("should resolve with correct data", async function () {
|
2022-03-08 01:41:41 +09:00
|
|
|
|
const promiseCapability = new PromiseCapability();
|
2019-02-02 21:10:06 +09:00
|
|
|
|
expect(promiseCapability.settled).toEqual(false);
|
|
|
|
|
|
|
|
|
|
promiseCapability.resolve({ test: "abc" });
|
|
|
|
|
|
2021-04-12 02:26:38 +09:00
|
|
|
|
const data = await promiseCapability.promise;
|
|
|
|
|
expect(promiseCapability.settled).toEqual(true);
|
|
|
|
|
expect(data).toEqual({ test: "abc" });
|
2019-02-02 21:10:06 +09:00
|
|
|
|
});
|
|
|
|
|
|
2021-04-12 02:26:38 +09:00
|
|
|
|
it("should reject with correct reason", async function () {
|
2022-03-08 01:41:41 +09:00
|
|
|
|
const promiseCapability = new PromiseCapability();
|
2019-02-02 21:10:06 +09:00
|
|
|
|
expect(promiseCapability.settled).toEqual(false);
|
|
|
|
|
|
|
|
|
|
promiseCapability.reject(new Error("reason"));
|
|
|
|
|
|
2021-04-12 02:26:38 +09:00
|
|
|
|
try {
|
|
|
|
|
await promiseCapability.promise;
|
2019-02-02 21:10:06 +09:00
|
|
|
|
|
2021-04-12 02:26:38 +09:00
|
|
|
|
// Shouldn't get here.
|
|
|
|
|
expect(false).toEqual(true);
|
|
|
|
|
} catch (reason) {
|
|
|
|
|
expect(promiseCapability.settled).toEqual(true);
|
2019-02-02 21:10:06 +09:00
|
|
|
|
expect(reason instanceof Error).toEqual(true);
|
|
|
|
|
expect(reason.message).toEqual("reason");
|
2021-04-12 02:26:38 +09:00
|
|
|
|
}
|
2019-02-02 21:10:06 +09:00
|
|
|
|
});
|
|
|
|
|
});
|
2020-08-05 21:40:31 +09:00
|
|
|
|
|
2020-08-04 02:44:04 +09:00
|
|
|
|
describe("getModificationDate", function () {
|
|
|
|
|
it("should get a correctly formatted date", function () {
|
|
|
|
|
const date = new Date(Date.UTC(3141, 5, 9, 2, 6, 53));
|
2020-09-10 00:56:28 +09:00
|
|
|
|
expect(getModificationDate(date)).toEqual("31410609020653");
|
2020-08-04 02:44:04 +09:00
|
|
|
|
});
|
|
|
|
|
});
|
2012-06-24 05:35:59 +09:00
|
|
|
|
});
|