addb4cb12b
Rather than using a temporary Array to manually create repeated strings, we can use `String.prototype.repeat()` instead. The reason that we didn't use this from the start is most likely because some browsers, notably IE, didn't support this; note https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/repeat#browser_compatibility
418 lines
13 KiB
JavaScript
418 lines
13 KiB
JavaScript
/* 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.
|
|
*/
|
|
|
|
import { createIdFactory, XRefMock } from "./test_utils.js";
|
|
import { Dict, Name } from "../../src/core/primitives.js";
|
|
import { FormatError, OPS } from "../../src/shared/util.js";
|
|
import { Stream, StringStream } from "../../src/core/stream.js";
|
|
import { OperatorList } from "../../src/core/operator_list.js";
|
|
import { PartialEvaluator } from "../../src/core/evaluator.js";
|
|
import { WorkerTask } from "../../src/core/worker.js";
|
|
|
|
describe("evaluator", function () {
|
|
function HandlerMock() {
|
|
this.inputs = [];
|
|
}
|
|
HandlerMock.prototype = {
|
|
send(name, data) {
|
|
this.inputs.push({ name, data });
|
|
},
|
|
};
|
|
function ResourcesMock() {}
|
|
ResourcesMock.prototype = {
|
|
get(name) {
|
|
return this[name];
|
|
},
|
|
};
|
|
|
|
async function runOperatorListCheck(evaluator, stream, resources) {
|
|
const operatorList = new OperatorList();
|
|
const task = new WorkerTask("OperatorListCheck");
|
|
await evaluator.getOperatorList({
|
|
stream,
|
|
task,
|
|
resources,
|
|
operatorList,
|
|
});
|
|
return operatorList;
|
|
}
|
|
|
|
let partialEvaluator;
|
|
|
|
beforeAll(function () {
|
|
partialEvaluator = new PartialEvaluator({
|
|
xref: new XRefMock(),
|
|
handler: new HandlerMock(),
|
|
pageIndex: 0,
|
|
idFactory: createIdFactory(/* pageIndex = */ 0),
|
|
});
|
|
});
|
|
|
|
afterAll(function () {
|
|
partialEvaluator = null;
|
|
});
|
|
|
|
describe("splitCombinedOperations", function () {
|
|
it("should reject unknown operations", async function () {
|
|
const stream = new StringStream("fTT");
|
|
const result = await runOperatorListCheck(
|
|
partialEvaluator,
|
|
stream,
|
|
new ResourcesMock()
|
|
);
|
|
expect(!!result.fnArray && !!result.argsArray).toEqual(true);
|
|
expect(result.fnArray.length).toEqual(1);
|
|
expect(result.fnArray[0]).toEqual(OPS.fill);
|
|
expect(result.argsArray[0]).toEqual(null);
|
|
});
|
|
|
|
it("should handle one operation", async function () {
|
|
const stream = new StringStream("Q");
|
|
const result = await runOperatorListCheck(
|
|
partialEvaluator,
|
|
stream,
|
|
new ResourcesMock()
|
|
);
|
|
expect(!!result.fnArray && !!result.argsArray).toEqual(true);
|
|
expect(result.fnArray.length).toEqual(1);
|
|
expect(result.fnArray[0]).toEqual(OPS.restore);
|
|
});
|
|
|
|
it("should handle two glued operations", async function () {
|
|
const imgDict = new Dict();
|
|
imgDict.set("Subtype", Name.get("Image"));
|
|
imgDict.set("Width", 1);
|
|
imgDict.set("Height", 1);
|
|
|
|
const imgStream = new Stream([0]);
|
|
imgStream.dict = imgDict;
|
|
|
|
const xObject = new Dict();
|
|
xObject.set("Res1", imgStream);
|
|
|
|
const resources = new ResourcesMock();
|
|
resources.XObject = xObject;
|
|
|
|
const stream = new StringStream("/Res1 DoQ");
|
|
const result = await runOperatorListCheck(
|
|
partialEvaluator,
|
|
stream,
|
|
resources
|
|
);
|
|
expect(result.fnArray.length).toEqual(3);
|
|
expect(result.fnArray[0]).toEqual(OPS.dependency);
|
|
expect(result.fnArray[1]).toEqual(OPS.paintImageXObject);
|
|
expect(result.fnArray[2]).toEqual(OPS.restore);
|
|
expect(result.argsArray.length).toEqual(3);
|
|
expect(result.argsArray[0]).toEqual(["img_p0_1"]);
|
|
expect(result.argsArray[1]).toEqual(["img_p0_1", 1, 1]);
|
|
expect(result.argsArray[2]).toEqual(null);
|
|
});
|
|
|
|
it("should handle three glued operations", async function () {
|
|
const stream = new StringStream("fff");
|
|
const result = await runOperatorListCheck(
|
|
partialEvaluator,
|
|
stream,
|
|
new ResourcesMock()
|
|
);
|
|
expect(!!result.fnArray && !!result.argsArray).toEqual(true);
|
|
expect(result.fnArray.length).toEqual(3);
|
|
expect(result.fnArray[0]).toEqual(OPS.fill);
|
|
expect(result.fnArray[1]).toEqual(OPS.fill);
|
|
expect(result.fnArray[2]).toEqual(OPS.fill);
|
|
});
|
|
|
|
it("should handle three glued operations #2", async function () {
|
|
const resources = new ResourcesMock();
|
|
resources.Res1 = {};
|
|
const stream = new StringStream("B*Bf*");
|
|
const result = await runOperatorListCheck(
|
|
partialEvaluator,
|
|
stream,
|
|
resources
|
|
);
|
|
expect(!!result.fnArray && !!result.argsArray).toEqual(true);
|
|
expect(result.fnArray.length).toEqual(3);
|
|
expect(result.fnArray[0]).toEqual(OPS.eoFillStroke);
|
|
expect(result.fnArray[1]).toEqual(OPS.fillStroke);
|
|
expect(result.fnArray[2]).toEqual(OPS.eoFill);
|
|
});
|
|
|
|
it("should handle glued operations and operands", async function () {
|
|
const stream = new StringStream("f5 Ts");
|
|
const result = await runOperatorListCheck(
|
|
partialEvaluator,
|
|
stream,
|
|
new ResourcesMock()
|
|
);
|
|
expect(!!result.fnArray && !!result.argsArray).toEqual(true);
|
|
expect(result.fnArray.length).toEqual(2);
|
|
expect(result.fnArray[0]).toEqual(OPS.fill);
|
|
expect(result.fnArray[1]).toEqual(OPS.setTextRise);
|
|
expect(result.argsArray.length).toEqual(2);
|
|
expect(result.argsArray[1].length).toEqual(1);
|
|
expect(result.argsArray[1][0]).toEqual(5);
|
|
});
|
|
|
|
it("should handle glued operations and literals", async function () {
|
|
const stream = new StringStream("trueifalserinulln");
|
|
const result = await runOperatorListCheck(
|
|
partialEvaluator,
|
|
stream,
|
|
new ResourcesMock()
|
|
);
|
|
expect(!!result.fnArray && !!result.argsArray).toEqual(true);
|
|
expect(result.fnArray.length).toEqual(3);
|
|
expect(result.fnArray[0]).toEqual(OPS.setFlatness);
|
|
expect(result.fnArray[1]).toEqual(OPS.setRenderingIntent);
|
|
expect(result.fnArray[2]).toEqual(OPS.endPath);
|
|
expect(result.argsArray.length).toEqual(3);
|
|
expect(result.argsArray[0].length).toEqual(1);
|
|
expect(result.argsArray[0][0]).toEqual(true);
|
|
expect(result.argsArray[1].length).toEqual(1);
|
|
expect(result.argsArray[1][0]).toEqual(false);
|
|
expect(result.argsArray[2]).toEqual(null);
|
|
});
|
|
});
|
|
|
|
describe("validateNumberOfArgs", function () {
|
|
it("should execute if correct number of arguments", async function () {
|
|
const stream = new StringStream("5 1 d0");
|
|
const result = await runOperatorListCheck(
|
|
partialEvaluator,
|
|
stream,
|
|
new ResourcesMock()
|
|
);
|
|
expect(result.argsArray[0][0]).toEqual(5);
|
|
expect(result.argsArray[0][1]).toEqual(1);
|
|
expect(result.fnArray[0]).toEqual(OPS.setCharWidth);
|
|
});
|
|
|
|
it("should execute if too many arguments", async function () {
|
|
const stream = new StringStream("5 1 4 d0");
|
|
const result = await runOperatorListCheck(
|
|
partialEvaluator,
|
|
stream,
|
|
new ResourcesMock()
|
|
);
|
|
expect(result.argsArray[0][0]).toEqual(1);
|
|
expect(result.argsArray[0][1]).toEqual(4);
|
|
expect(result.fnArray[0]).toEqual(OPS.setCharWidth);
|
|
});
|
|
|
|
it("should execute if nested commands", async function () {
|
|
const gState = new Dict();
|
|
gState.set("LW", 2);
|
|
gState.set("CA", 0.5);
|
|
|
|
const extGState = new Dict();
|
|
extGState.set("GS2", gState);
|
|
|
|
const resources = new ResourcesMock();
|
|
resources.ExtGState = extGState;
|
|
|
|
const stream = new StringStream("/F2 /GS2 gs 5.711 Tf");
|
|
const result = await runOperatorListCheck(
|
|
partialEvaluator,
|
|
stream,
|
|
resources
|
|
);
|
|
expect(result.fnArray.length).toEqual(3);
|
|
expect(result.fnArray[0]).toEqual(OPS.setGState);
|
|
expect(result.fnArray[1]).toEqual(OPS.dependency);
|
|
expect(result.fnArray[2]).toEqual(OPS.setFont);
|
|
expect(result.argsArray.length).toEqual(3);
|
|
expect(result.argsArray[0]).toEqual([
|
|
[
|
|
["LW", 2],
|
|
["CA", 0.5],
|
|
],
|
|
]);
|
|
expect(result.argsArray[1]).toEqual(["g_font_error"]);
|
|
expect(result.argsArray[2]).toEqual(["g_font_error", 5.711]);
|
|
});
|
|
|
|
it("should skip if too few arguments", async function () {
|
|
const stream = new StringStream("5 d0");
|
|
const result = await runOperatorListCheck(
|
|
partialEvaluator,
|
|
stream,
|
|
new ResourcesMock()
|
|
);
|
|
expect(result.argsArray).toEqual([]);
|
|
expect(result.fnArray).toEqual([]);
|
|
});
|
|
|
|
it(
|
|
"should error if (many) path operators have too few arguments " +
|
|
"(bug 1443140)",
|
|
async function () {
|
|
const NUM_INVALID_OPS = 25;
|
|
|
|
// Non-path operators, should be ignored.
|
|
const invalidMoveText = "10 Td\n".repeat(NUM_INVALID_OPS);
|
|
const moveTextStream = new StringStream(invalidMoveText);
|
|
const result = await runOperatorListCheck(
|
|
partialEvaluator,
|
|
moveTextStream,
|
|
new ResourcesMock()
|
|
);
|
|
expect(result.argsArray).toEqual([]);
|
|
expect(result.fnArray).toEqual([]);
|
|
|
|
// Path operators, should throw error.
|
|
const invalidLineTo = "20 l\n".repeat(NUM_INVALID_OPS);
|
|
const lineToStream = new StringStream(invalidLineTo);
|
|
|
|
try {
|
|
await runOperatorListCheck(
|
|
partialEvaluator,
|
|
lineToStream,
|
|
new ResourcesMock()
|
|
);
|
|
|
|
// Shouldn't get here.
|
|
expect(false).toEqual(true);
|
|
} catch (reason) {
|
|
expect(reason instanceof FormatError).toEqual(true);
|
|
expect(reason.message).toEqual(
|
|
"Invalid command l: expected 2 args, but received 1 args."
|
|
);
|
|
}
|
|
}
|
|
);
|
|
|
|
it("should close opened saves", async function () {
|
|
const stream = new StringStream("qq");
|
|
const result = await runOperatorListCheck(
|
|
partialEvaluator,
|
|
stream,
|
|
new ResourcesMock()
|
|
);
|
|
expect(!!result.fnArray && !!result.argsArray).toEqual(true);
|
|
expect(result.fnArray.length).toEqual(4);
|
|
expect(result.fnArray[0]).toEqual(OPS.save);
|
|
expect(result.fnArray[1]).toEqual(OPS.save);
|
|
expect(result.fnArray[2]).toEqual(OPS.restore);
|
|
expect(result.fnArray[3]).toEqual(OPS.restore);
|
|
});
|
|
|
|
it("should error on paintXObject if name is missing", async function () {
|
|
const stream = new StringStream("/ Do");
|
|
|
|
try {
|
|
await runOperatorListCheck(
|
|
partialEvaluator,
|
|
stream,
|
|
new ResourcesMock()
|
|
);
|
|
|
|
// Shouldn't get here.
|
|
expect(false).toEqual(true);
|
|
} catch (reason) {
|
|
expect(reason instanceof FormatError).toEqual(true);
|
|
expect(reason.message).toEqual("XObject should be a stream");
|
|
}
|
|
});
|
|
|
|
it("should skip paintXObject if subtype is PS", async function () {
|
|
const xobjStreamDict = new Dict();
|
|
xobjStreamDict.set("Subtype", Name.get("PS"));
|
|
const xobjStream = new Stream([], 0, 0, xobjStreamDict);
|
|
|
|
const xobjs = new Dict();
|
|
xobjs.set("Res1", xobjStream);
|
|
|
|
const resources = new Dict();
|
|
resources.set("XObject", xobjs);
|
|
|
|
const stream = new StringStream("/Res1 Do");
|
|
const result = await runOperatorListCheck(
|
|
partialEvaluator,
|
|
stream,
|
|
resources
|
|
);
|
|
expect(result.argsArray).toEqual([]);
|
|
expect(result.fnArray).toEqual([]);
|
|
});
|
|
});
|
|
|
|
describe("thread control", function () {
|
|
it("should abort operator list parsing", async function () {
|
|
const stream = new StringStream("qqQQ");
|
|
const resources = new ResourcesMock();
|
|
const result = new OperatorList();
|
|
const task = new WorkerTask("OperatorListAbort");
|
|
task.terminate();
|
|
|
|
try {
|
|
await partialEvaluator.getOperatorList({
|
|
stream,
|
|
task,
|
|
resources,
|
|
operatorList: result,
|
|
});
|
|
|
|
// Shouldn't get here.
|
|
expect(false).toEqual(true);
|
|
} catch (_) {
|
|
expect(!!result.fnArray && !!result.argsArray).toEqual(true);
|
|
expect(result.fnArray.length).toEqual(0);
|
|
}
|
|
});
|
|
|
|
it("should abort text content parsing", async function () {
|
|
const resources = new ResourcesMock();
|
|
const stream = new StringStream("qqQQ");
|
|
const task = new WorkerTask("TextContentAbort");
|
|
task.terminate();
|
|
|
|
try {
|
|
await partialEvaluator.getTextContent({
|
|
stream,
|
|
task,
|
|
resources,
|
|
});
|
|
|
|
// Shouldn't get here.
|
|
expect(false).toEqual(true);
|
|
} catch (_) {
|
|
expect(true).toEqual(true);
|
|
}
|
|
});
|
|
});
|
|
|
|
describe("operator list", function () {
|
|
class StreamSinkMock {
|
|
enqueue() {}
|
|
}
|
|
|
|
it("should get correct total length after flushing", function () {
|
|
const operatorList = new OperatorList(null, new StreamSinkMock());
|
|
operatorList.addOp(OPS.save, null);
|
|
operatorList.addOp(OPS.restore, null);
|
|
|
|
expect(operatorList.totalLength).toEqual(2);
|
|
expect(operatorList.length).toEqual(2);
|
|
|
|
operatorList.flush();
|
|
|
|
expect(operatorList.totalLength).toEqual(2);
|
|
expect(operatorList.length).toEqual(0);
|
|
});
|
|
});
|
|
});
|