2020-08-04 02:44:04 +09:00
|
|
|
/* Copyright 2020 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 { Dict, Name, Ref } from "../../src/core/primitives.js";
|
|
|
|
import { incrementalUpdate, writeDict } from "../../src/core/writer.js";
|
|
|
|
import { bytesToString } from "../../src/shared/util.js";
|
|
|
|
import { StringStream } from "../../src/core/stream.js";
|
|
|
|
|
|
|
|
describe("Writer", function () {
|
|
|
|
describe("Incremental update", function () {
|
2023-04-28 04:50:27 +09:00
|
|
|
it("should update a file with new objects", async function () {
|
2020-08-04 02:44:04 +09:00
|
|
|
const originalData = new Uint8Array();
|
|
|
|
const newRefs = [
|
|
|
|
{ ref: Ref.get(123, 0x2d), data: "abc\n" },
|
|
|
|
{ ref: Ref.get(456, 0x4e), data: "defg\n" },
|
|
|
|
];
|
|
|
|
const xrefInfo = {
|
|
|
|
newRef: Ref.get(789, 0),
|
|
|
|
startXRef: 314,
|
|
|
|
fileIds: ["id", ""],
|
|
|
|
rootRef: null,
|
|
|
|
infoRef: null,
|
2021-04-22 19:08:56 +09:00
|
|
|
encryptRef: null,
|
2020-08-04 02:44:04 +09:00
|
|
|
filename: "foo.pdf",
|
|
|
|
info: {},
|
|
|
|
};
|
|
|
|
|
2023-04-28 04:50:27 +09:00
|
|
|
let data = await incrementalUpdate({ originalData, xrefInfo, newRefs });
|
2020-08-04 02:44:04 +09:00
|
|
|
data = bytesToString(data);
|
|
|
|
|
|
|
|
const expected =
|
|
|
|
"\nabc\n" +
|
|
|
|
"defg\n" +
|
|
|
|
"789 0 obj\n" +
|
|
|
|
"<< /Size 790 /Prev 314 /Type /XRef /Index [0 1 123 1 456 1 789 1] " +
|
|
|
|
"/ID [(id) (\x01#Eg\x89\xab\xcd\xef\xfe\xdc\xba\x98vT2\x10)] " +
|
|
|
|
"/W [1 1 2] /Length 16>> stream\n" +
|
|
|
|
"\x00\x01\xff\xff" +
|
|
|
|
"\x01\x01\x00\x2d" +
|
|
|
|
"\x01\x05\x00\x4e" +
|
|
|
|
"\x01\x0a\x00\x00\n" +
|
|
|
|
"endstream\n" +
|
|
|
|
"endobj\n" +
|
|
|
|
"startxref\n" +
|
|
|
|
"10\n" +
|
|
|
|
"%%EOF\n";
|
|
|
|
|
|
|
|
expect(data).toEqual(expected);
|
|
|
|
});
|
2021-04-22 19:08:56 +09:00
|
|
|
|
2023-04-28 04:50:27 +09:00
|
|
|
it("should update a file, missing the /ID-entry, with new objects", async function () {
|
2021-04-22 19:08:56 +09:00
|
|
|
const originalData = new Uint8Array();
|
|
|
|
const newRefs = [{ ref: Ref.get(123, 0x2d), data: "abc\n" }];
|
|
|
|
const xrefInfo = {
|
|
|
|
newRef: Ref.get(789, 0),
|
|
|
|
startXRef: 314,
|
|
|
|
fileIds: null,
|
|
|
|
rootRef: null,
|
|
|
|
infoRef: null,
|
|
|
|
encryptRef: null,
|
|
|
|
filename: "foo.pdf",
|
|
|
|
info: {},
|
|
|
|
};
|
|
|
|
|
2023-04-28 04:50:27 +09:00
|
|
|
let data = await incrementalUpdate({ originalData, xrefInfo, newRefs });
|
2021-04-22 19:08:56 +09:00
|
|
|
data = bytesToString(data);
|
|
|
|
|
|
|
|
const expected =
|
|
|
|
"\nabc\n" +
|
|
|
|
"789 0 obj\n" +
|
|
|
|
"<< /Size 790 /Prev 314 /Type /XRef /Index [0 1 123 1 789 1] " +
|
|
|
|
"/W [1 1 2] /Length 12>> stream\n" +
|
|
|
|
"\x00\x01\xff\xff" +
|
|
|
|
"\x01\x01\x00\x2d" +
|
|
|
|
"\x01\x05\x00\x00\n" +
|
|
|
|
"endstream\n" +
|
|
|
|
"endobj\n" +
|
|
|
|
"startxref\n" +
|
|
|
|
"5\n" +
|
|
|
|
"%%EOF\n";
|
|
|
|
|
|
|
|
expect(data).toEqual(expected);
|
|
|
|
});
|
2020-08-04 02:44:04 +09:00
|
|
|
});
|
|
|
|
|
|
|
|
describe("writeDict", function () {
|
2023-04-28 04:50:27 +09:00
|
|
|
it("should write a Dict", async function () {
|
2020-08-04 02:44:04 +09:00
|
|
|
const dict = new Dict(null);
|
|
|
|
dict.set("A", Name.get("B"));
|
|
|
|
dict.set("B", Ref.get(123, 456));
|
|
|
|
dict.set("C", 789);
|
|
|
|
dict.set("D", "hello world");
|
|
|
|
dict.set("E", "(hello\\world)");
|
|
|
|
dict.set("F", [1.23001, 4.50001, 6]);
|
|
|
|
|
|
|
|
const gdict = new Dict(null);
|
|
|
|
gdict.set("H", 123.00001);
|
|
|
|
const string = "a stream";
|
|
|
|
const stream = new StringStream(string);
|
|
|
|
stream.dict = new Dict(null);
|
|
|
|
stream.dict.set("Length", string.length);
|
|
|
|
gdict.set("I", stream);
|
|
|
|
|
|
|
|
dict.set("G", gdict);
|
2021-09-10 21:10:21 +09:00
|
|
|
dict.set("J", true);
|
|
|
|
dict.set("K", false);
|
2020-08-04 02:44:04 +09:00
|
|
|
|
[src/core/writer.js] Support `null` values in the `writeValue` function
*This fixes something that I noticed, having recently looked at both the `Lexer.getObj` and `writeValue` code.*
Please note that I unfortunately don't have an example of a form where saving fails without this patch. However, given its overall simplicity and that unit-tests are added, it's hopefully deemed useful to fix this potential issue pro-actively rather than waiting for a bug report.
At this point one might, and rightly so, wonder if there's actually any real-world PDF documents where a `null` value is being used?
Unfortunately the answer is *yes*, and we have a couple of examples in the test-suite (although none of those are related to forms); please see: `issue1015`, `issue2642`, `issue10402`, `issue12823`, `issue13823`, and `pr12564`.
2021-09-13 01:13:36 +09:00
|
|
|
dict.set("NullArr", [null, 10]);
|
|
|
|
dict.set("NullVal", null);
|
|
|
|
|
2020-08-04 02:44:04 +09:00
|
|
|
const buffer = [];
|
2023-04-28 04:50:27 +09:00
|
|
|
await writeDict(dict, buffer, null);
|
2020-08-04 02:44:04 +09:00
|
|
|
|
|
|
|
const expected =
|
|
|
|
"<< /A /B /B 123 456 R /C 789 /D (hello world) " +
|
|
|
|
"/E (\\(hello\\\\world\\)) /F [1.23 4.5 6] " +
|
|
|
|
"/G << /H 123 /I << /Length 8>> stream\n" +
|
|
|
|
"a stream\n" +
|
2022-10-19 00:07:47 +09:00
|
|
|
"endstream>> /J true /K false " +
|
[src/core/writer.js] Support `null` values in the `writeValue` function
*This fixes something that I noticed, having recently looked at both the `Lexer.getObj` and `writeValue` code.*
Please note that I unfortunately don't have an example of a form where saving fails without this patch. However, given its overall simplicity and that unit-tests are added, it's hopefully deemed useful to fix this potential issue pro-actively rather than waiting for a bug report.
At this point one might, and rightly so, wonder if there's actually any real-world PDF documents where a `null` value is being used?
Unfortunately the answer is *yes*, and we have a couple of examples in the test-suite (although none of those are related to forms); please see: `issue1015`, `issue2642`, `issue10402`, `issue12823`, `issue13823`, and `pr12564`.
2021-09-13 01:13:36 +09:00
|
|
|
"/NullArr [null 10] /NullVal null>>";
|
2020-08-04 02:44:04 +09:00
|
|
|
|
|
|
|
expect(buffer.join("")).toEqual(expected);
|
|
|
|
});
|
2020-09-10 01:39:14 +09:00
|
|
|
|
2023-04-28 04:50:27 +09:00
|
|
|
it("should write a Dict in escaping PDF names", async function () {
|
2020-09-10 01:39:14 +09:00
|
|
|
const dict = new Dict(null);
|
2020-09-11 19:25:05 +09:00
|
|
|
dict.set("\xfeA#", Name.get("hello"));
|
2020-09-10 01:39:14 +09:00
|
|
|
dict.set("B", Name.get("#hello"));
|
|
|
|
dict.set("C", Name.get("he\xfello\xff"));
|
|
|
|
|
|
|
|
const buffer = [];
|
2023-04-28 04:50:27 +09:00
|
|
|
await writeDict(dict, buffer, null);
|
2020-09-10 01:39:14 +09:00
|
|
|
|
2020-09-11 19:25:05 +09:00
|
|
|
const expected = "<< /#feA#23 /hello /B /#23hello /C /he#fello#ff>>";
|
2020-09-10 01:39:14 +09:00
|
|
|
|
|
|
|
expect(buffer.join("")).toEqual(expected);
|
|
|
|
});
|
2020-08-04 02:44:04 +09:00
|
|
|
});
|
2021-09-03 21:28:31 +09:00
|
|
|
|
|
|
|
describe("XFA", function () {
|
2023-04-28 04:50:27 +09:00
|
|
|
it("should update AcroForm when no datasets in XFA array", async function () {
|
2021-09-03 21:28:31 +09:00
|
|
|
const originalData = new Uint8Array();
|
|
|
|
const newRefs = [];
|
|
|
|
|
|
|
|
const acroForm = new Dict(null);
|
|
|
|
acroForm.set("XFA", [
|
|
|
|
"preamble",
|
|
|
|
Ref.get(123, 0),
|
|
|
|
"postamble",
|
|
|
|
Ref.get(456, 0),
|
|
|
|
]);
|
|
|
|
const acroFormRef = Ref.get(789, 0);
|
2021-09-24 01:18:55 +09:00
|
|
|
const xfaDatasetsRef = Ref.get(101112, 0);
|
2021-09-03 21:28:31 +09:00
|
|
|
const xfaData = "<hello>world</hello>";
|
|
|
|
|
|
|
|
const xrefInfo = {
|
|
|
|
newRef: Ref.get(131415, 0),
|
|
|
|
startXRef: 314,
|
|
|
|
fileIds: null,
|
|
|
|
rootRef: null,
|
|
|
|
infoRef: null,
|
|
|
|
encryptRef: null,
|
|
|
|
filename: "foo.pdf",
|
|
|
|
info: {},
|
|
|
|
};
|
|
|
|
|
2023-04-28 04:50:27 +09:00
|
|
|
let data = await incrementalUpdate({
|
2021-09-03 21:28:31 +09:00
|
|
|
originalData,
|
|
|
|
xrefInfo,
|
|
|
|
newRefs,
|
2021-09-24 01:18:55 +09:00
|
|
|
hasXfa: true,
|
|
|
|
xfaDatasetsRef,
|
|
|
|
hasXfaDatasetsEntry: false,
|
2021-09-03 21:28:31 +09:00
|
|
|
acroFormRef,
|
|
|
|
acroForm,
|
|
|
|
xfaData,
|
|
|
|
xref: {},
|
|
|
|
});
|
|
|
|
data = bytesToString(data);
|
|
|
|
|
|
|
|
const expected =
|
|
|
|
"\n" +
|
|
|
|
"789 0 obj\n" +
|
|
|
|
"<< /XFA [(preamble) 123 0 R (datasets) 101112 0 R (postamble) 456 0 R]>>\n" +
|
2022-10-19 00:07:47 +09:00
|
|
|
"endobj\n" +
|
2021-09-03 21:28:31 +09:00
|
|
|
"101112 0 obj\n" +
|
|
|
|
"<< /Type /EmbeddedFile /Length 20>>\n" +
|
|
|
|
"stream\n" +
|
|
|
|
"<hello>world</hello>\n" +
|
|
|
|
"endstream\n" +
|
|
|
|
"endobj\n" +
|
|
|
|
"131415 0 obj\n" +
|
|
|
|
"<< /Size 131416 /Prev 314 /Type /XRef /Index [0 1 789 1 101112 1 131415 1] /W [1 1 2] /Length 16>> stream\n" +
|
2022-10-19 00:07:47 +09:00
|
|
|
"\u0000\u0001ÿÿ\u0001\u0001\u0000\u0000\u0001[\u0000\u0000\u0001¹\u0000\u0000\n" +
|
2021-09-03 21:28:31 +09:00
|
|
|
"endstream\n" +
|
|
|
|
"endobj\n" +
|
|
|
|
"startxref\n" +
|
2022-10-19 00:07:47 +09:00
|
|
|
"185\n" +
|
2021-09-03 21:28:31 +09:00
|
|
|
"%%EOF\n";
|
|
|
|
|
|
|
|
expect(data).toEqual(expected);
|
|
|
|
});
|
|
|
|
});
|
2020-08-04 02:44:04 +09:00
|
|
|
});
|