6d4d402a78
Given that this helper function is only used on the worker-thread, there's no reason to duplicate it in both of the *built* `pdf.js` and `pdf.worker.js` files.
472 lines
15 KiB
JavaScript
472 lines
15 KiB
JavaScript
/* Copyright 2019 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.
|
|
*/
|
|
|
|
import {
|
|
arrayBuffersToBytes,
|
|
encodeToXmlString,
|
|
escapePDFName,
|
|
escapeString,
|
|
getInheritableProperty,
|
|
isAscii,
|
|
isWhiteSpace,
|
|
log2,
|
|
numberToString,
|
|
parseXFAPath,
|
|
recoverJsURL,
|
|
stringToUTF16HexString,
|
|
stringToUTF16String,
|
|
toRomanNumerals,
|
|
validateCSSFont,
|
|
} from "../../src/core/core_utils.js";
|
|
import { Dict, Ref } from "../../src/core/primitives.js";
|
|
import { XRefMock } from "./test_utils.js";
|
|
|
|
describe("core_utils", function () {
|
|
describe("arrayBuffersToBytes", function () {
|
|
it("handles zero ArrayBuffers", function () {
|
|
const bytes = arrayBuffersToBytes([]);
|
|
|
|
expect(bytes).toEqual(new Uint8Array(0));
|
|
});
|
|
|
|
it("handles one ArrayBuffer", function () {
|
|
const buffer = new Uint8Array([1, 2, 3]).buffer;
|
|
const bytes = arrayBuffersToBytes([buffer]);
|
|
|
|
expect(bytes).toEqual(new Uint8Array([1, 2, 3]));
|
|
// Ensure that the fast-path works correctly.
|
|
expect(bytes.buffer).toBe(buffer);
|
|
});
|
|
|
|
it("handles multiple ArrayBuffers", function () {
|
|
const buffer1 = new Uint8Array([1, 2, 3]).buffer,
|
|
buffer2 = new Uint8Array(0).buffer,
|
|
buffer3 = new Uint8Array([4, 5]).buffer;
|
|
const bytes = arrayBuffersToBytes([buffer1, buffer2, buffer3]);
|
|
|
|
expect(bytes).toEqual(new Uint8Array([1, 2, 3, 4, 5]));
|
|
});
|
|
});
|
|
|
|
describe("getInheritableProperty", function () {
|
|
it("handles non-dictionary arguments", function () {
|
|
expect(getInheritableProperty({ dict: null, key: "foo" })).toEqual(
|
|
undefined
|
|
);
|
|
expect(getInheritableProperty({ dict: undefined, key: "foo" })).toEqual(
|
|
undefined
|
|
);
|
|
});
|
|
|
|
it("handles dictionaries that do not contain the property", function () {
|
|
// Empty dictionary.
|
|
const emptyDict = new Dict();
|
|
expect(getInheritableProperty({ dict: emptyDict, key: "foo" })).toEqual(
|
|
undefined
|
|
);
|
|
|
|
// Filled dictionary with a different property.
|
|
const filledDict = new Dict();
|
|
filledDict.set("bar", "baz");
|
|
expect(getInheritableProperty({ dict: filledDict, key: "foo" })).toEqual(
|
|
undefined
|
|
);
|
|
});
|
|
|
|
it("fetches the property if it is not inherited", function () {
|
|
const ref = Ref.get(10, 0);
|
|
const xref = new XRefMock([{ ref, data: "quux" }]);
|
|
const dict = new Dict(xref);
|
|
|
|
// Regular values should be fetched.
|
|
dict.set("foo", "bar");
|
|
expect(getInheritableProperty({ dict, key: "foo" })).toEqual("bar");
|
|
|
|
// Array value should be fetched (with references resolved).
|
|
dict.set("baz", ["qux", ref]);
|
|
expect(
|
|
getInheritableProperty({ dict, key: "baz", getArray: true })
|
|
).toEqual(["qux", "quux"]);
|
|
});
|
|
|
|
it("fetches the property if it is inherited and present on one level", function () {
|
|
const ref = Ref.get(10, 0);
|
|
const xref = new XRefMock([{ ref, data: "quux" }]);
|
|
const firstDict = new Dict(xref);
|
|
const secondDict = new Dict(xref);
|
|
firstDict.set("Parent", secondDict);
|
|
|
|
// Regular values should be fetched.
|
|
secondDict.set("foo", "bar");
|
|
expect(getInheritableProperty({ dict: firstDict, key: "foo" })).toEqual(
|
|
"bar"
|
|
);
|
|
|
|
// Array value should be fetched (with references resolved).
|
|
secondDict.set("baz", ["qux", ref]);
|
|
expect(
|
|
getInheritableProperty({ dict: firstDict, key: "baz", getArray: true })
|
|
).toEqual(["qux", "quux"]);
|
|
});
|
|
|
|
it("fetches the property if it is inherited and present on multiple levels", function () {
|
|
const ref = Ref.get(10, 0);
|
|
const xref = new XRefMock([{ ref, data: "quux" }]);
|
|
const firstDict = new Dict(xref);
|
|
const secondDict = new Dict(xref);
|
|
firstDict.set("Parent", secondDict);
|
|
|
|
// Regular values should be fetched.
|
|
firstDict.set("foo", "bar1");
|
|
secondDict.set("foo", "bar2");
|
|
expect(getInheritableProperty({ dict: firstDict, key: "foo" })).toEqual(
|
|
"bar1"
|
|
);
|
|
expect(
|
|
getInheritableProperty({
|
|
dict: firstDict,
|
|
key: "foo",
|
|
getArray: false,
|
|
stopWhenFound: false,
|
|
})
|
|
).toEqual(["bar1", "bar2"]);
|
|
|
|
// Array value should be fetched (with references resolved).
|
|
firstDict.set("baz", ["qux1", ref]);
|
|
secondDict.set("baz", ["qux2", ref]);
|
|
expect(
|
|
getInheritableProperty({
|
|
dict: firstDict,
|
|
key: "baz",
|
|
getArray: true,
|
|
stopWhenFound: false,
|
|
})
|
|
).toEqual([
|
|
["qux1", "quux"],
|
|
["qux2", "quux"],
|
|
]);
|
|
});
|
|
});
|
|
|
|
describe("toRomanNumerals", function () {
|
|
it("handles invalid arguments", function () {
|
|
for (const input of ["foo", -1, 0]) {
|
|
expect(function () {
|
|
toRomanNumerals(input);
|
|
}).toThrow(new Error("The number should be a positive integer."));
|
|
}
|
|
});
|
|
|
|
it("converts numbers to uppercase Roman numerals", function () {
|
|
expect(toRomanNumerals(1)).toEqual("I");
|
|
expect(toRomanNumerals(6)).toEqual("VI");
|
|
expect(toRomanNumerals(7)).toEqual("VII");
|
|
expect(toRomanNumerals(8)).toEqual("VIII");
|
|
expect(toRomanNumerals(10)).toEqual("X");
|
|
expect(toRomanNumerals(40)).toEqual("XL");
|
|
expect(toRomanNumerals(100)).toEqual("C");
|
|
expect(toRomanNumerals(500)).toEqual("D");
|
|
expect(toRomanNumerals(1000)).toEqual("M");
|
|
expect(toRomanNumerals(2019)).toEqual("MMXIX");
|
|
});
|
|
|
|
it("converts numbers to lowercase Roman numerals", function () {
|
|
expect(toRomanNumerals(1, /* lowercase = */ true)).toEqual("i");
|
|
expect(toRomanNumerals(6, /* lowercase = */ true)).toEqual("vi");
|
|
expect(toRomanNumerals(7, /* lowercase = */ true)).toEqual("vii");
|
|
expect(toRomanNumerals(8, /* lowercase = */ true)).toEqual("viii");
|
|
expect(toRomanNumerals(10, /* lowercase = */ true)).toEqual("x");
|
|
expect(toRomanNumerals(40, /* lowercase = */ true)).toEqual("xl");
|
|
expect(toRomanNumerals(100, /* lowercase = */ true)).toEqual("c");
|
|
expect(toRomanNumerals(500, /* lowercase = */ true)).toEqual("d");
|
|
expect(toRomanNumerals(1000, /* lowercase = */ true)).toEqual("m");
|
|
expect(toRomanNumerals(2019, /* lowercase = */ true)).toEqual("mmxix");
|
|
});
|
|
});
|
|
|
|
describe("log2", function () {
|
|
it("handles values smaller than/equal to zero", function () {
|
|
expect(log2(0)).toEqual(0);
|
|
expect(log2(-1)).toEqual(0);
|
|
});
|
|
|
|
it("handles values larger than zero", function () {
|
|
expect(log2(1)).toEqual(0);
|
|
expect(log2(2)).toEqual(1);
|
|
expect(log2(3)).toEqual(2);
|
|
expect(log2(3.14)).toEqual(2);
|
|
});
|
|
});
|
|
|
|
describe("numberToString", function () {
|
|
it("should stringify integers", function () {
|
|
expect(numberToString(1)).toEqual("1");
|
|
expect(numberToString(0)).toEqual("0");
|
|
expect(numberToString(-1)).toEqual("-1");
|
|
});
|
|
|
|
it("should stringify floats", function () {
|
|
expect(numberToString(1.0)).toEqual("1");
|
|
expect(numberToString(1.2)).toEqual("1.2");
|
|
expect(numberToString(1.23)).toEqual("1.23");
|
|
expect(numberToString(1.234)).toEqual("1.23");
|
|
});
|
|
});
|
|
|
|
describe("isWhiteSpace", function () {
|
|
it("handles space characters", function () {
|
|
expect(isWhiteSpace(0x20)).toEqual(true);
|
|
expect(isWhiteSpace(0x09)).toEqual(true);
|
|
expect(isWhiteSpace(0x0d)).toEqual(true);
|
|
expect(isWhiteSpace(0x0a)).toEqual(true);
|
|
});
|
|
|
|
it("handles non-space characters", function () {
|
|
expect(isWhiteSpace(0x0b)).toEqual(false);
|
|
expect(isWhiteSpace(null)).toEqual(false);
|
|
expect(isWhiteSpace(undefined)).toEqual(false);
|
|
});
|
|
});
|
|
|
|
describe("parseXFAPath", function () {
|
|
it("should get a correctly parsed path", function () {
|
|
const path = "foo.bar[12].oof[3].rab.FOO[123].BAR[456]";
|
|
expect(parseXFAPath(path)).toEqual([
|
|
{ name: "foo", pos: 0 },
|
|
{ name: "bar", pos: 12 },
|
|
{ name: "oof", pos: 3 },
|
|
{ name: "rab", pos: 0 },
|
|
{ name: "FOO", pos: 123 },
|
|
{ name: "BAR", pos: 456 },
|
|
]);
|
|
});
|
|
});
|
|
|
|
describe("recoverJsURL", function () {
|
|
it("should get valid URLs without `newWindow` property", function () {
|
|
const inputs = [
|
|
"window.open('https://test.local')",
|
|
"window.open('https://test.local', true)",
|
|
"app.launchURL('https://test.local')",
|
|
"app.launchURL('https://test.local', false)",
|
|
"xfa.host.gotoURL('https://test.local')",
|
|
"xfa.host.gotoURL('https://test.local', true)",
|
|
];
|
|
|
|
for (const input of inputs) {
|
|
expect(recoverJsURL(input)).toEqual({
|
|
url: "https://test.local",
|
|
newWindow: false,
|
|
});
|
|
}
|
|
});
|
|
|
|
it("should get valid URLs with `newWindow` property", function () {
|
|
const input = "app.launchURL('https://test.local', true)";
|
|
expect(recoverJsURL(input)).toEqual({
|
|
url: "https://test.local",
|
|
newWindow: true,
|
|
});
|
|
});
|
|
|
|
it("should not get invalid URLs", function () {
|
|
const input = "navigateToUrl('https://test.local')";
|
|
expect(recoverJsURL(input)).toBeNull();
|
|
});
|
|
});
|
|
|
|
describe("escapePDFName", function () {
|
|
it("should escape PDF name", function () {
|
|
expect(escapePDFName("hello")).toEqual("hello");
|
|
expect(escapePDFName("\xfehello")).toEqual("#fehello");
|
|
expect(escapePDFName("he\xfell\xffo")).toEqual("he#fell#ffo");
|
|
expect(escapePDFName("\xfehe\xfell\xffo\xff")).toEqual(
|
|
"#fehe#fell#ffo#ff"
|
|
);
|
|
expect(escapePDFName("#h#e#l#l#o")).toEqual("#23h#23e#23l#23l#23o");
|
|
expect(escapePDFName("#()<>[]{}/%")).toEqual(
|
|
"#23#28#29#3c#3e#5b#5d#7b#7d#2f#25"
|
|
);
|
|
});
|
|
});
|
|
|
|
describe("escapeString", function () {
|
|
it("should escape (, ), \\n, \\r, and \\", function () {
|
|
expect(escapeString("((a\\a))\n(b(b\\b)\rb)")).toEqual(
|
|
"\\(\\(a\\\\a\\)\\)\\n\\(b\\(b\\\\b\\)\\rb\\)"
|
|
);
|
|
});
|
|
});
|
|
|
|
describe("encodeToXmlString", function () {
|
|
it("should get a correctly encoded string with some entities", function () {
|
|
const str = "\"\u0397ell😂' & <W😂rld>";
|
|
expect(encodeToXmlString(str)).toEqual(
|
|
""Ηell😂' & <W😂rld>"
|
|
);
|
|
});
|
|
|
|
it("should get a correctly encoded basic ascii string", function () {
|
|
const str = "hello world";
|
|
expect(encodeToXmlString(str)).toEqual(str);
|
|
});
|
|
});
|
|
|
|
describe("validateCSSFont", function () {
|
|
it("Check font family", function () {
|
|
const cssFontInfo = {
|
|
fontFamily: `"blah blah " blah blah"`,
|
|
fontWeight: 0,
|
|
italicAngle: 0,
|
|
};
|
|
|
|
expect(validateCSSFont(cssFontInfo)).toEqual(false);
|
|
|
|
cssFontInfo.fontFamily = `"blah blah \\" blah blah"`;
|
|
expect(validateCSSFont(cssFontInfo)).toEqual(true);
|
|
|
|
cssFontInfo.fontFamily = `'blah blah ' blah blah'`;
|
|
expect(validateCSSFont(cssFontInfo)).toEqual(false);
|
|
|
|
cssFontInfo.fontFamily = `'blah blah \\' blah blah'`;
|
|
expect(validateCSSFont(cssFontInfo)).toEqual(true);
|
|
|
|
cssFontInfo.fontFamily = `"blah blah `;
|
|
expect(validateCSSFont(cssFontInfo)).toEqual(false);
|
|
|
|
cssFontInfo.fontFamily = `blah blah"`;
|
|
expect(validateCSSFont(cssFontInfo)).toEqual(false);
|
|
|
|
cssFontInfo.fontFamily = `'blah blah `;
|
|
expect(validateCSSFont(cssFontInfo)).toEqual(false);
|
|
|
|
cssFontInfo.fontFamily = `blah blah'`;
|
|
expect(validateCSSFont(cssFontInfo)).toEqual(false);
|
|
|
|
cssFontInfo.fontFamily = "blah blah blah";
|
|
expect(validateCSSFont(cssFontInfo)).toEqual(true);
|
|
|
|
cssFontInfo.fontFamily = "blah 0blah blah";
|
|
expect(validateCSSFont(cssFontInfo)).toEqual(false);
|
|
|
|
cssFontInfo.fontFamily = "blah blah -0blah";
|
|
expect(validateCSSFont(cssFontInfo)).toEqual(false);
|
|
|
|
cssFontInfo.fontFamily = "blah blah --blah";
|
|
expect(validateCSSFont(cssFontInfo)).toEqual(false);
|
|
|
|
cssFontInfo.fontFamily = "blah blah -blah";
|
|
expect(validateCSSFont(cssFontInfo)).toEqual(true);
|
|
|
|
cssFontInfo.fontFamily = "blah fdqAJqjHJK23kl23__--Kj blah";
|
|
expect(validateCSSFont(cssFontInfo)).toEqual(true);
|
|
|
|
cssFontInfo.fontFamily = "blah fdqAJqjH$JK23kl23__--Kj blah";
|
|
expect(validateCSSFont(cssFontInfo)).toEqual(false);
|
|
});
|
|
|
|
it("Check font weight", function () {
|
|
const cssFontInfo = {
|
|
fontFamily: "blah",
|
|
fontWeight: 100,
|
|
italicAngle: 0,
|
|
};
|
|
|
|
validateCSSFont(cssFontInfo);
|
|
expect(cssFontInfo.fontWeight).toEqual("100");
|
|
|
|
cssFontInfo.fontWeight = "700";
|
|
validateCSSFont(cssFontInfo);
|
|
expect(cssFontInfo.fontWeight).toEqual("700");
|
|
|
|
cssFontInfo.fontWeight = "normal";
|
|
validateCSSFont(cssFontInfo);
|
|
expect(cssFontInfo.fontWeight).toEqual("normal");
|
|
|
|
cssFontInfo.fontWeight = 314;
|
|
validateCSSFont(cssFontInfo);
|
|
expect(cssFontInfo.fontWeight).toEqual("400");
|
|
});
|
|
|
|
it("Check italic angle", function () {
|
|
const cssFontInfo = {
|
|
fontFamily: "blah",
|
|
fontWeight: 100,
|
|
italicAngle: 10,
|
|
};
|
|
validateCSSFont(cssFontInfo);
|
|
expect(cssFontInfo.italicAngle).toEqual("10");
|
|
|
|
cssFontInfo.italicAngle = -123;
|
|
validateCSSFont(cssFontInfo);
|
|
expect(cssFontInfo.italicAngle).toEqual("14");
|
|
|
|
cssFontInfo.italicAngle = "91";
|
|
validateCSSFont(cssFontInfo);
|
|
expect(cssFontInfo.italicAngle).toEqual("14");
|
|
|
|
cssFontInfo.italicAngle = 2.718;
|
|
validateCSSFont(cssFontInfo);
|
|
expect(cssFontInfo.italicAngle).toEqual("2.718");
|
|
});
|
|
});
|
|
|
|
describe("isAscii", function () {
|
|
it("handles ascii/non-ascii strings", function () {
|
|
expect(isAscii("hello world")).toEqual(true);
|
|
expect(isAscii("こんにちは世界の")).toEqual(false);
|
|
expect(isAscii("hello world in Japanese is こんにちは世界の")).toEqual(
|
|
false
|
|
);
|
|
});
|
|
});
|
|
|
|
describe("stringToUTF16HexString", function () {
|
|
it("should encode a string in UTF16 hexadecimal format", function () {
|
|
expect(stringToUTF16HexString("hello world")).toEqual(
|
|
"00680065006c006c006f00200077006f0072006c0064"
|
|
);
|
|
|
|
expect(stringToUTF16HexString("こんにちは世界の")).toEqual(
|
|
"30533093306b3061306f4e16754c306e"
|
|
);
|
|
});
|
|
});
|
|
|
|
describe("stringToUTF16String", function () {
|
|
it("should encode a string in UTF16", function () {
|
|
expect(stringToUTF16String("hello world")).toEqual(
|
|
"\0h\0e\0l\0l\0o\0 \0w\0o\0r\0l\0d"
|
|
);
|
|
|
|
expect(stringToUTF16String("こんにちは世界の")).toEqual(
|
|
"\x30\x53\x30\x93\x30\x6b\x30\x61\x30\x6f\x4e\x16\x75\x4c\x30\x6e"
|
|
);
|
|
});
|
|
|
|
it("should encode a string in UTF16BE with a BOM", function () {
|
|
expect(
|
|
stringToUTF16String("hello world", /* bigEndian = */ true)
|
|
).toEqual("\xfe\xff\0h\0e\0l\0l\0o\0 \0w\0o\0r\0l\0d");
|
|
|
|
expect(
|
|
stringToUTF16String("こんにちは世界の", /* bigEndian = */ true)
|
|
).toEqual(
|
|
"\xfe\xff\x30\x53\x30\x93\x30\x6b\x30\x61\x30\x6f\x4e\x16\x75\x4c\x30\x6e"
|
|
);
|
|
});
|
|
});
|
|
});
|