Given that the API will now, after PR 12039, automatically pick the correct factories to use depending on the environment (browser vs. Node.js), we can utilize that in the unit-tests as well. This way we don't have to manually repeat the same initialization code in *multiple* unit-tests. *Note:* The *official* PDF.js API is defined in `src/pdf.js`, hence the new exports in `src/display/api.js` will not affect that. Also, updates the unit-test `FileReaderFactory` helpers similarily. *Drive-by change:* Fix the `CMapReaderFactory` usage in the annotation unit-tests, since the cache should only contain raw data and not a Promise. While this obviously works as-is, having unit-tests that "abuse" the intended data format can easily lead to unnecessary failures if changes are made to the relevant `src/core/` code.
2304 lines
74 KiB
JavaScript
2304 lines
74 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 {
|
||
buildGetDocumentParams,
|
||
DefaultFileReaderFactory,
|
||
TEST_PDFS_PATH,
|
||
} from "./test_utils.js";
|
||
import {
|
||
createPromiseCapability,
|
||
FontType,
|
||
ImageKind,
|
||
InvalidPDFException,
|
||
MissingPDFException,
|
||
OPS,
|
||
PasswordException,
|
||
PasswordResponses,
|
||
PermissionFlag,
|
||
StreamType,
|
||
} from "../../src/shared/util.js";
|
||
import {
|
||
DefaultCanvasFactory,
|
||
getDocument,
|
||
PDFDataRangeTransport,
|
||
PDFDocumentProxy,
|
||
PDFPageProxy,
|
||
PDFWorker,
|
||
} from "../../src/display/api.js";
|
||
import {
|
||
RenderingCancelledException,
|
||
StatTimer,
|
||
} from "../../src/display/display_utils.js";
|
||
import { AutoPrintRegExp } from "../../web/ui_utils.js";
|
||
import { GlobalImageCache } from "../../src/core/image_utils.js";
|
||
import { GlobalWorkerOptions } from "../../src/display/worker_options.js";
|
||
import { isNodeJS } from "../../src/shared/is_node.js";
|
||
import { Metadata } from "../../src/display/metadata.js";
|
||
|
||
describe("api", function () {
|
||
const basicApiFileName = "basicapi.pdf";
|
||
const basicApiFileLength = 105779; // bytes
|
||
const basicApiGetDocumentParams = buildGetDocumentParams(basicApiFileName);
|
||
|
||
let CanvasFactory;
|
||
|
||
beforeAll(function (done) {
|
||
CanvasFactory = new DefaultCanvasFactory();
|
||
done();
|
||
});
|
||
|
||
afterAll(function (done) {
|
||
CanvasFactory = null;
|
||
done();
|
||
});
|
||
|
||
function waitSome(callback) {
|
||
const WAIT_TIMEOUT = 10;
|
||
setTimeout(function () {
|
||
callback();
|
||
}, WAIT_TIMEOUT);
|
||
}
|
||
|
||
describe("getDocument", function () {
|
||
it("creates pdf doc from URL", function (done) {
|
||
const loadingTask = getDocument(basicApiGetDocumentParams);
|
||
|
||
const progressReportedCapability = createPromiseCapability();
|
||
// Attach the callback that is used to report loading progress;
|
||
// similarly to how viewer.js works.
|
||
loadingTask.onProgress = function (progressData) {
|
||
if (!progressReportedCapability.settled) {
|
||
progressReportedCapability.resolve(progressData);
|
||
}
|
||
};
|
||
|
||
const promises = [
|
||
progressReportedCapability.promise,
|
||
loadingTask.promise,
|
||
];
|
||
Promise.all(promises)
|
||
.then(function (data) {
|
||
expect(data[0].loaded / data[0].total >= 0).toEqual(true);
|
||
expect(data[1] instanceof PDFDocumentProxy).toEqual(true);
|
||
expect(loadingTask).toEqual(data[1].loadingTask);
|
||
loadingTask.destroy().then(done);
|
||
})
|
||
.catch(done.fail);
|
||
});
|
||
it("creates pdf doc from URL and aborts before worker initialized", function (done) {
|
||
const loadingTask = getDocument(basicApiGetDocumentParams);
|
||
const destroyed = loadingTask.destroy();
|
||
|
||
loadingTask.promise
|
||
.then(function (reason) {
|
||
done.fail("shall fail loading");
|
||
})
|
||
.catch(function (reason) {
|
||
expect(true).toEqual(true);
|
||
destroyed.then(done);
|
||
});
|
||
});
|
||
it("creates pdf doc from URL and aborts loading after worker initialized", function (done) {
|
||
const loadingTask = getDocument(basicApiGetDocumentParams);
|
||
// This can be somewhat random -- we cannot guarantee perfect
|
||
// 'Terminate' message to the worker before/after setting up pdfManager.
|
||
const destroyed = loadingTask._worker.promise.then(function () {
|
||
return loadingTask.destroy();
|
||
});
|
||
destroyed
|
||
.then(function (data) {
|
||
expect(true).toEqual(true);
|
||
done();
|
||
})
|
||
.catch(done.fail);
|
||
});
|
||
it("creates pdf doc from typed array", function (done) {
|
||
const typedArrayPdfPromise = DefaultFileReaderFactory.fetch({
|
||
path: TEST_PDFS_PATH + basicApiFileName,
|
||
});
|
||
|
||
typedArrayPdfPromise
|
||
.then(typedArrayPdf => {
|
||
// Sanity check to make sure that we fetched the entire PDF file.
|
||
expect(typedArrayPdf.length).toEqual(basicApiFileLength);
|
||
|
||
const loadingTask = getDocument(typedArrayPdf);
|
||
|
||
const progressReportedCapability = createPromiseCapability();
|
||
loadingTask.onProgress = function (data) {
|
||
progressReportedCapability.resolve(data);
|
||
};
|
||
|
||
return Promise.all([
|
||
loadingTask.promise,
|
||
progressReportedCapability.promise,
|
||
]).then(function (data) {
|
||
expect(data[0] instanceof PDFDocumentProxy).toEqual(true);
|
||
expect(data[1].loaded / data[1].total).toEqual(1);
|
||
|
||
loadingTask.destroy().then(done);
|
||
});
|
||
})
|
||
.catch(done.fail);
|
||
});
|
||
it("creates pdf doc from invalid PDF file", function (done) {
|
||
// A severely corrupt PDF file (even Adobe Reader fails to open it).
|
||
const loadingTask = getDocument(buildGetDocumentParams("bug1020226.pdf"));
|
||
loadingTask.promise
|
||
.then(function () {
|
||
done.fail("shall fail loading");
|
||
})
|
||
.catch(function (reason) {
|
||
expect(reason instanceof InvalidPDFException).toEqual(true);
|
||
expect(reason.message).toEqual("Invalid PDF structure.");
|
||
|
||
loadingTask.destroy().then(done);
|
||
});
|
||
});
|
||
it("creates pdf doc from non-existent URL", function (done) {
|
||
const loadingTask = getDocument(
|
||
buildGetDocumentParams("non-existent.pdf")
|
||
);
|
||
loadingTask.promise
|
||
.then(function (error) {
|
||
done.fail("shall fail loading");
|
||
})
|
||
.catch(function (error) {
|
||
expect(error instanceof MissingPDFException).toEqual(true);
|
||
loadingTask.destroy().then(done);
|
||
});
|
||
});
|
||
it("creates pdf doc from PDF file protected with user and owner password", function (done) {
|
||
const loadingTask = getDocument(buildGetDocumentParams("pr6531_1.pdf"));
|
||
|
||
const passwordNeededCapability = createPromiseCapability();
|
||
const passwordIncorrectCapability = createPromiseCapability();
|
||
// Attach the callback that is used to request a password;
|
||
// similarly to how viewer.js handles passwords.
|
||
loadingTask.onPassword = function (updatePassword, reason) {
|
||
if (
|
||
reason === PasswordResponses.NEED_PASSWORD &&
|
||
!passwordNeededCapability.settled
|
||
) {
|
||
passwordNeededCapability.resolve();
|
||
|
||
updatePassword("qwerty"); // Provide an incorrect password.
|
||
return;
|
||
}
|
||
if (
|
||
reason === PasswordResponses.INCORRECT_PASSWORD &&
|
||
!passwordIncorrectCapability.settled
|
||
) {
|
||
passwordIncorrectCapability.resolve();
|
||
|
||
updatePassword("asdfasdf"); // Provide the correct password.
|
||
return;
|
||
}
|
||
// Shouldn't get here.
|
||
expect(false).toEqual(true);
|
||
};
|
||
|
||
const promises = [
|
||
passwordNeededCapability.promise,
|
||
passwordIncorrectCapability.promise,
|
||
loadingTask.promise,
|
||
];
|
||
Promise.all(promises)
|
||
.then(function (data) {
|
||
expect(data[2] instanceof PDFDocumentProxy).toEqual(true);
|
||
loadingTask.destroy().then(done);
|
||
})
|
||
.catch(done.fail);
|
||
});
|
||
it("creates pdf doc from PDF file protected with only a user password", function (done) {
|
||
const filename = "pr6531_2.pdf";
|
||
|
||
const passwordNeededLoadingTask = getDocument(
|
||
buildGetDocumentParams(filename, {
|
||
password: "",
|
||
})
|
||
);
|
||
const result1 = passwordNeededLoadingTask.promise.then(
|
||
function () {
|
||
done.fail("shall fail with no password");
|
||
return Promise.reject(new Error("loadingTask should be rejected"));
|
||
},
|
||
function (data) {
|
||
expect(data instanceof PasswordException).toEqual(true);
|
||
expect(data.code).toEqual(PasswordResponses.NEED_PASSWORD);
|
||
return passwordNeededLoadingTask.destroy();
|
||
}
|
||
);
|
||
|
||
const passwordIncorrectLoadingTask = getDocument(
|
||
buildGetDocumentParams(filename, {
|
||
password: "qwerty",
|
||
})
|
||
);
|
||
const result2 = passwordIncorrectLoadingTask.promise.then(
|
||
function () {
|
||
done.fail("shall fail with wrong password");
|
||
return Promise.reject(new Error("loadingTask should be rejected"));
|
||
},
|
||
function (data) {
|
||
expect(data instanceof PasswordException).toEqual(true);
|
||
expect(data.code).toEqual(PasswordResponses.INCORRECT_PASSWORD);
|
||
return passwordIncorrectLoadingTask.destroy();
|
||
}
|
||
);
|
||
|
||
const passwordAcceptedLoadingTask = getDocument(
|
||
buildGetDocumentParams(filename, {
|
||
password: "asdfasdf",
|
||
})
|
||
);
|
||
const result3 = passwordAcceptedLoadingTask.promise.then(function (data) {
|
||
expect(data instanceof PDFDocumentProxy).toEqual(true);
|
||
return passwordAcceptedLoadingTask.destroy();
|
||
});
|
||
Promise.all([result1, result2, result3])
|
||
.then(function () {
|
||
done();
|
||
})
|
||
.catch(done.fail);
|
||
});
|
||
|
||
it(
|
||
"creates pdf doc from password protected PDF file and aborts/throws " +
|
||
"in the onPassword callback (issue 7806)",
|
||
function (done) {
|
||
const filename = "issue3371.pdf";
|
||
|
||
const passwordNeededLoadingTask = getDocument(
|
||
buildGetDocumentParams(filename)
|
||
);
|
||
const passwordIncorrectLoadingTask = getDocument(
|
||
buildGetDocumentParams(filename, {
|
||
password: "qwerty",
|
||
})
|
||
);
|
||
|
||
let passwordNeededDestroyed;
|
||
passwordNeededLoadingTask.onPassword = function (callback, reason) {
|
||
if (reason === PasswordResponses.NEED_PASSWORD) {
|
||
passwordNeededDestroyed = passwordNeededLoadingTask.destroy();
|
||
return;
|
||
}
|
||
// Shouldn't get here.
|
||
expect(false).toEqual(true);
|
||
};
|
||
const result1 = passwordNeededLoadingTask.promise.then(
|
||
function () {
|
||
done.fail("shall fail since the loadingTask should be destroyed");
|
||
return Promise.reject(new Error("loadingTask should be rejected"));
|
||
},
|
||
function (reason) {
|
||
expect(reason instanceof PasswordException).toEqual(true);
|
||
expect(reason.code).toEqual(PasswordResponses.NEED_PASSWORD);
|
||
return passwordNeededDestroyed;
|
||
}
|
||
);
|
||
|
||
passwordIncorrectLoadingTask.onPassword = function (callback, reason) {
|
||
if (reason === PasswordResponses.INCORRECT_PASSWORD) {
|
||
throw new Error("Incorrect password");
|
||
}
|
||
// Shouldn't get here.
|
||
expect(false).toEqual(true);
|
||
};
|
||
const result2 = passwordIncorrectLoadingTask.promise.then(
|
||
function () {
|
||
done.fail("shall fail since the onPassword callback should throw");
|
||
return Promise.reject(new Error("loadingTask should be rejected"));
|
||
},
|
||
function (reason) {
|
||
expect(reason instanceof PasswordException).toEqual(true);
|
||
expect(reason.code).toEqual(PasswordResponses.INCORRECT_PASSWORD);
|
||
return passwordIncorrectLoadingTask.destroy();
|
||
}
|
||
);
|
||
|
||
Promise.all([result1, result2])
|
||
.then(function () {
|
||
done();
|
||
})
|
||
.catch(done.fail);
|
||
}
|
||
);
|
||
|
||
it("creates pdf doc from empty typed array", function (done) {
|
||
const loadingTask = getDocument(new Uint8Array(0));
|
||
|
||
loadingTask.promise.then(
|
||
function () {
|
||
done.fail("shall not open empty file");
|
||
},
|
||
function (reason) {
|
||
expect(reason instanceof InvalidPDFException);
|
||
expect(reason.message).toEqual(
|
||
"The PDF file is empty, i.e. its size is zero bytes."
|
||
);
|
||
|
||
loadingTask.destroy().then(done);
|
||
}
|
||
);
|
||
});
|
||
});
|
||
|
||
describe("PDFWorker", function () {
|
||
it("worker created or destroyed", function (done) {
|
||
if (isNodeJS) {
|
||
pending("Worker is not supported in Node.js.");
|
||
}
|
||
|
||
const worker = new PDFWorker({ name: "test1" });
|
||
worker.promise
|
||
.then(function () {
|
||
expect(worker.name).toEqual("test1");
|
||
expect(!!worker.port).toEqual(true);
|
||
expect(worker.destroyed).toEqual(false);
|
||
expect(!!worker._webWorker).toEqual(true);
|
||
expect(worker.port === worker._webWorker).toEqual(true);
|
||
|
||
worker.destroy();
|
||
expect(!!worker.port).toEqual(false);
|
||
expect(worker.destroyed).toEqual(true);
|
||
done();
|
||
})
|
||
.catch(done.fail);
|
||
});
|
||
it("worker created or destroyed by getDocument", function (done) {
|
||
if (isNodeJS) {
|
||
pending("Worker is not supported in Node.js.");
|
||
}
|
||
|
||
const loadingTask = getDocument(basicApiGetDocumentParams);
|
||
let worker;
|
||
loadingTask.promise.then(function () {
|
||
worker = loadingTask._worker;
|
||
expect(!!worker).toEqual(true);
|
||
});
|
||
|
||
const destroyPromise = loadingTask.promise.then(function () {
|
||
return loadingTask.destroy();
|
||
});
|
||
destroyPromise
|
||
.then(function () {
|
||
const destroyedWorker = loadingTask._worker;
|
||
expect(!!destroyedWorker).toEqual(false);
|
||
expect(worker.destroyed).toEqual(true);
|
||
done();
|
||
})
|
||
.catch(done.fail);
|
||
});
|
||
it("worker created and can be used in getDocument", function (done) {
|
||
if (isNodeJS) {
|
||
pending("Worker is not supported in Node.js.");
|
||
}
|
||
|
||
const worker = new PDFWorker({ name: "test1" });
|
||
const loadingTask = getDocument(
|
||
buildGetDocumentParams(basicApiFileName, {
|
||
worker,
|
||
})
|
||
);
|
||
loadingTask.promise.then(function () {
|
||
const docWorker = loadingTask._worker;
|
||
expect(!!docWorker).toEqual(false);
|
||
// checking is the same port is used in the MessageHandler
|
||
const messageHandlerPort = loadingTask._transport.messageHandler.comObj;
|
||
expect(messageHandlerPort === worker.port).toEqual(true);
|
||
});
|
||
|
||
const destroyPromise = loadingTask.promise.then(function () {
|
||
return loadingTask.destroy();
|
||
});
|
||
destroyPromise
|
||
.then(function () {
|
||
expect(worker.destroyed).toEqual(false);
|
||
worker.destroy();
|
||
done();
|
||
})
|
||
.catch(done.fail);
|
||
});
|
||
it("creates more than one worker", function (done) {
|
||
if (isNodeJS) {
|
||
pending("Worker is not supported in Node.js.");
|
||
}
|
||
|
||
const worker1 = new PDFWorker({ name: "test1" });
|
||
const worker2 = new PDFWorker({ name: "test2" });
|
||
const worker3 = new PDFWorker({ name: "test3" });
|
||
const ready = Promise.all([
|
||
worker1.promise,
|
||
worker2.promise,
|
||
worker3.promise,
|
||
]);
|
||
ready
|
||
.then(function () {
|
||
expect(
|
||
worker1.port !== worker2.port &&
|
||
worker1.port !== worker3.port &&
|
||
worker2.port !== worker3.port
|
||
).toEqual(true);
|
||
worker1.destroy();
|
||
worker2.destroy();
|
||
worker3.destroy();
|
||
done();
|
||
})
|
||
.catch(done.fail);
|
||
});
|
||
it("gets current workerSrc", function () {
|
||
if (isNodeJS) {
|
||
pending("Worker is not supported in Node.js.");
|
||
}
|
||
|
||
const workerSrc = PDFWorker.getWorkerSrc();
|
||
expect(typeof workerSrc).toEqual("string");
|
||
expect(workerSrc).toEqual(GlobalWorkerOptions.workerSrc);
|
||
});
|
||
});
|
||
describe("PDFDocument", function () {
|
||
let pdfLoadingTask, pdfDocument;
|
||
|
||
beforeAll(function (done) {
|
||
pdfLoadingTask = getDocument(basicApiGetDocumentParams);
|
||
pdfLoadingTask.promise.then(function (data) {
|
||
pdfDocument = data;
|
||
done();
|
||
});
|
||
});
|
||
|
||
afterAll(function (done) {
|
||
pdfLoadingTask.destroy().then(done);
|
||
});
|
||
|
||
it("gets number of pages", function () {
|
||
expect(pdfDocument.numPages).toEqual(3);
|
||
});
|
||
it("gets fingerprint", function () {
|
||
expect(pdfDocument.fingerprint).toEqual(
|
||
"ea8b35919d6279a369e835bde778611b"
|
||
);
|
||
});
|
||
it("gets page", function (done) {
|
||
const promise = pdfDocument.getPage(1);
|
||
promise
|
||
.then(function (data) {
|
||
expect(data instanceof PDFPageProxy).toEqual(true);
|
||
expect(data.pageNumber).toEqual(1);
|
||
done();
|
||
})
|
||
.catch(done.fail);
|
||
});
|
||
it("gets non-existent page", function (done) {
|
||
let outOfRangePromise = pdfDocument.getPage(100);
|
||
let nonIntegerPromise = pdfDocument.getPage(2.5);
|
||
let nonNumberPromise = pdfDocument.getPage("1");
|
||
|
||
outOfRangePromise = outOfRangePromise.then(
|
||
function () {
|
||
throw new Error("shall fail for out-of-range pageNumber parameter");
|
||
},
|
||
function (reason) {
|
||
expect(reason instanceof Error).toEqual(true);
|
||
}
|
||
);
|
||
nonIntegerPromise = nonIntegerPromise.then(
|
||
function () {
|
||
throw new Error("shall fail for non-integer pageNumber parameter");
|
||
},
|
||
function (reason) {
|
||
expect(reason instanceof Error).toEqual(true);
|
||
}
|
||
);
|
||
nonNumberPromise = nonNumberPromise.then(
|
||
function () {
|
||
throw new Error("shall fail for non-number pageNumber parameter");
|
||
},
|
||
function (reason) {
|
||
expect(reason instanceof Error).toEqual(true);
|
||
}
|
||
);
|
||
|
||
Promise.all([outOfRangePromise, nonIntegerPromise, nonNumberPromise])
|
||
.then(function () {
|
||
done();
|
||
})
|
||
.catch(done.fail);
|
||
});
|
||
|
||
it("gets page, from /Pages tree with circular reference", function (done) {
|
||
const loadingTask = getDocument(
|
||
buildGetDocumentParams("Pages-tree-refs.pdf")
|
||
);
|
||
|
||
const page1 = loadingTask.promise.then(function (pdfDoc) {
|
||
return pdfDoc.getPage(1).then(
|
||
function (pdfPage) {
|
||
expect(pdfPage instanceof PDFPageProxy).toEqual(true);
|
||
expect(pdfPage.ref).toEqual({ num: 6, gen: 0 });
|
||
},
|
||
function (reason) {
|
||
throw new Error("shall not fail for valid page");
|
||
}
|
||
);
|
||
});
|
||
|
||
const page2 = loadingTask.promise.then(function (pdfDoc) {
|
||
return pdfDoc.getPage(2).then(
|
||
function (pdfPage) {
|
||
throw new Error("shall fail for invalid page");
|
||
},
|
||
function (reason) {
|
||
expect(reason instanceof Error).toEqual(true);
|
||
expect(reason.message).toEqual(
|
||
"Pages tree contains circular reference."
|
||
);
|
||
}
|
||
);
|
||
});
|
||
|
||
Promise.all([page1, page2]).then(function () {
|
||
loadingTask.destroy().then(done);
|
||
}, done.fail);
|
||
});
|
||
|
||
it("gets page index", function (done) {
|
||
// reference to second page
|
||
const ref = { num: 17, gen: 0 };
|
||
const promise = pdfDocument.getPageIndex(ref);
|
||
promise
|
||
.then(function (pageIndex) {
|
||
expect(pageIndex).toEqual(1);
|
||
done();
|
||
})
|
||
.catch(done.fail);
|
||
});
|
||
it("gets invalid page index", function (done) {
|
||
const ref = { num: 3, gen: 0 }; // Reference to a font dictionary.
|
||
const promise = pdfDocument.getPageIndex(ref);
|
||
promise
|
||
.then(function () {
|
||
done.fail("shall fail for invalid page reference.");
|
||
})
|
||
.catch(function (reason) {
|
||
expect(reason instanceof Error).toEqual(true);
|
||
done();
|
||
});
|
||
});
|
||
|
||
it("gets destinations, from /Dests dictionary", function (done) {
|
||
const promise = pdfDocument.getDestinations();
|
||
promise
|
||
.then(function (data) {
|
||
expect(data).toEqual({
|
||
chapter1: [{ gen: 0, num: 17 }, { name: "XYZ" }, 0, 841.89, null],
|
||
});
|
||
done();
|
||
})
|
||
.catch(done.fail);
|
||
});
|
||
it("gets a destination, from /Dests dictionary", function (done) {
|
||
const promise = pdfDocument.getDestination("chapter1");
|
||
promise
|
||
.then(function (data) {
|
||
expect(data).toEqual([
|
||
{ gen: 0, num: 17 },
|
||
{ name: "XYZ" },
|
||
0,
|
||
841.89,
|
||
null,
|
||
]);
|
||
done();
|
||
})
|
||
.catch(done.fail);
|
||
});
|
||
it("gets a non-existent destination, from /Dests dictionary", function (done) {
|
||
const promise = pdfDocument.getDestination(
|
||
"non-existent-named-destination"
|
||
);
|
||
promise
|
||
.then(function (data) {
|
||
expect(data).toEqual(null);
|
||
done();
|
||
})
|
||
.catch(done.fail);
|
||
});
|
||
|
||
it("gets destinations, from /Names (NameTree) dictionary", function (done) {
|
||
const loadingTask = getDocument(buildGetDocumentParams("issue6204.pdf"));
|
||
const promise = loadingTask.promise.then(function (pdfDoc) {
|
||
return pdfDoc.getDestinations();
|
||
});
|
||
promise
|
||
.then(function (destinations) {
|
||
expect(destinations).toEqual({
|
||
"Page.1": [{ num: 1, gen: 0 }, { name: "XYZ" }, 0, 375, null],
|
||
"Page.2": [{ num: 6, gen: 0 }, { name: "XYZ" }, 0, 375, null],
|
||
});
|
||
|
||
loadingTask.destroy().then(done);
|
||
})
|
||
.catch(done.fail);
|
||
});
|
||
it("gets a destination, from /Names (NameTree) dictionary", function (done) {
|
||
const loadingTask = getDocument(buildGetDocumentParams("issue6204.pdf"));
|
||
const promise = loadingTask.promise.then(function (pdfDoc) {
|
||
return pdfDoc.getDestination("Page.1");
|
||
});
|
||
promise
|
||
.then(function (destination) {
|
||
expect(destination).toEqual([
|
||
{ num: 1, gen: 0 },
|
||
{ name: "XYZ" },
|
||
0,
|
||
375,
|
||
null,
|
||
]);
|
||
|
||
loadingTask.destroy().then(done);
|
||
})
|
||
.catch(done.fail);
|
||
});
|
||
it("gets a non-existent destination, from /Names (NameTree) dictionary", function (done) {
|
||
const loadingTask = getDocument(buildGetDocumentParams("issue6204.pdf"));
|
||
const promise = loadingTask.promise.then(function (pdfDoc) {
|
||
return pdfDoc.getDestination("non-existent-named-destination");
|
||
});
|
||
promise
|
||
.then(function (destination) {
|
||
expect(destination).toEqual(null);
|
||
|
||
loadingTask.destroy().then(done);
|
||
})
|
||
.catch(done.fail);
|
||
});
|
||
|
||
it("gets non-string destination", function (done) {
|
||
let numberPromise = pdfDocument.getDestination(4.3);
|
||
let booleanPromise = pdfDocument.getDestination(true);
|
||
let arrayPromise = pdfDocument.getDestination([
|
||
{ num: 17, gen: 0 },
|
||
{ name: "XYZ" },
|
||
0,
|
||
841.89,
|
||
null,
|
||
]);
|
||
|
||
numberPromise = numberPromise.then(
|
||
function () {
|
||
throw new Error("shall fail for non-string destination.");
|
||
},
|
||
function (reason) {
|
||
expect(reason instanceof Error).toEqual(true);
|
||
}
|
||
);
|
||
booleanPromise = booleanPromise.then(
|
||
function () {
|
||
throw new Error("shall fail for non-string destination.");
|
||
},
|
||
function (reason) {
|
||
expect(reason instanceof Error).toEqual(true);
|
||
}
|
||
);
|
||
arrayPromise = arrayPromise.then(
|
||
function () {
|
||
throw new Error("shall fail for non-string destination.");
|
||
},
|
||
function (reason) {
|
||
expect(reason instanceof Error).toEqual(true);
|
||
}
|
||
);
|
||
|
||
Promise.all([numberPromise, booleanPromise, arrayPromise]).then(
|
||
done,
|
||
done.fail
|
||
);
|
||
});
|
||
|
||
it("gets non-existent page labels", function (done) {
|
||
const promise = pdfDocument.getPageLabels();
|
||
promise
|
||
.then(function (data) {
|
||
expect(data).toEqual(null);
|
||
done();
|
||
})
|
||
.catch(done.fail);
|
||
});
|
||
it("gets page labels", function (done) {
|
||
// PageLabels with Roman/Arabic numerals.
|
||
const loadingTask0 = getDocument(buildGetDocumentParams("bug793632.pdf"));
|
||
const promise0 = loadingTask0.promise.then(function (pdfDoc) {
|
||
return pdfDoc.getPageLabels();
|
||
});
|
||
|
||
// PageLabels with only a label prefix.
|
||
const loadingTask1 = getDocument(buildGetDocumentParams("issue1453.pdf"));
|
||
const promise1 = loadingTask1.promise.then(function (pdfDoc) {
|
||
return pdfDoc.getPageLabels();
|
||
});
|
||
|
||
// PageLabels identical to standard page numbering.
|
||
const loadingTask2 = getDocument(buildGetDocumentParams("rotation.pdf"));
|
||
const promise2 = loadingTask2.promise.then(function (pdfDoc) {
|
||
return pdfDoc.getPageLabels();
|
||
});
|
||
|
||
// PageLabels with bad "Prefix" entries.
|
||
const loadingTask3 = getDocument(
|
||
buildGetDocumentParams("bad-PageLabels.pdf")
|
||
);
|
||
const promise3 = loadingTask3.promise.then(function (pdfDoc) {
|
||
return pdfDoc.getPageLabels();
|
||
});
|
||
|
||
Promise.all([promise0, promise1, promise2, promise3])
|
||
.then(function (pageLabels) {
|
||
expect(pageLabels[0]).toEqual(["i", "ii", "iii", "1"]);
|
||
expect(pageLabels[1]).toEqual(["Front Page1"]);
|
||
expect(pageLabels[2]).toEqual(["1", "2"]);
|
||
expect(pageLabels[3]).toEqual(["X3"]);
|
||
|
||
Promise.all([
|
||
loadingTask0.destroy(),
|
||
loadingTask1.destroy(),
|
||
loadingTask2.destroy(),
|
||
loadingTask3.destroy(),
|
||
]).then(done);
|
||
})
|
||
.catch(done.fail);
|
||
});
|
||
|
||
it("gets default page layout", function (done) {
|
||
const loadingTask = getDocument(
|
||
buildGetDocumentParams("tracemonkey.pdf")
|
||
);
|
||
|
||
loadingTask.promise
|
||
.then(function (pdfDoc) {
|
||
return pdfDoc.getPageLayout();
|
||
})
|
||
.then(function (mode) {
|
||
expect(mode).toEqual("");
|
||
|
||
loadingTask.destroy().then(done);
|
||
})
|
||
.catch(done.fail);
|
||
});
|
||
it("gets non-default page layout", function (done) {
|
||
pdfDocument
|
||
.getPageLayout()
|
||
.then(function (mode) {
|
||
expect(mode).toEqual("SinglePage");
|
||
done();
|
||
})
|
||
.catch(done.fail);
|
||
});
|
||
|
||
it("gets default page mode", function (done) {
|
||
const loadingTask = getDocument(
|
||
buildGetDocumentParams("tracemonkey.pdf")
|
||
);
|
||
|
||
loadingTask.promise
|
||
.then(function (pdfDoc) {
|
||
return pdfDoc.getPageMode();
|
||
})
|
||
.then(function (mode) {
|
||
expect(mode).toEqual("UseNone");
|
||
|
||
loadingTask.destroy().then(done);
|
||
})
|
||
.catch(done.fail);
|
||
});
|
||
it("gets non-default page mode", function (done) {
|
||
pdfDocument
|
||
.getPageMode()
|
||
.then(function (mode) {
|
||
expect(mode).toEqual("UseOutlines");
|
||
done();
|
||
})
|
||
.catch(done.fail);
|
||
});
|
||
|
||
it("gets default viewer preferences", function (done) {
|
||
const loadingTask = getDocument(
|
||
buildGetDocumentParams("tracemonkey.pdf")
|
||
);
|
||
|
||
loadingTask.promise
|
||
.then(function (pdfDoc) {
|
||
return pdfDoc.getViewerPreferences();
|
||
})
|
||
.then(function (prefs) {
|
||
expect(prefs).toEqual(null);
|
||
|
||
loadingTask.destroy().then(done);
|
||
})
|
||
.catch(done.fail);
|
||
});
|
||
it("gets non-default viewer preferences", function (done) {
|
||
pdfDocument
|
||
.getViewerPreferences()
|
||
.then(function (prefs) {
|
||
expect(prefs).toEqual({
|
||
Direction: "L2R",
|
||
});
|
||
done();
|
||
})
|
||
.catch(done.fail);
|
||
});
|
||
|
||
it("gets default open action", function (done) {
|
||
const loadingTask = getDocument(
|
||
buildGetDocumentParams("tracemonkey.pdf")
|
||
);
|
||
|
||
loadingTask.promise
|
||
.then(function (pdfDoc) {
|
||
return pdfDoc.getOpenAction();
|
||
})
|
||
.then(function (openAction) {
|
||
expect(openAction).toEqual(null);
|
||
|
||
loadingTask.destroy().then(done);
|
||
})
|
||
.catch(done.fail);
|
||
});
|
||
it("gets non-default open action (with destination)", function (done) {
|
||
pdfDocument
|
||
.getOpenAction()
|
||
.then(function (openAction) {
|
||
expect(openAction.dest).toEqual([
|
||
{ num: 15, gen: 0 },
|
||
{ name: "FitH" },
|
||
null,
|
||
]);
|
||
expect(openAction.action).toBeUndefined();
|
||
|
||
done();
|
||
})
|
||
.catch(done.fail);
|
||
});
|
||
it("gets non-default open action (with Print action)", function (done) {
|
||
// PDF document with "Print" Named action in the OpenAction dictionary.
|
||
const loadingTask1 = getDocument(
|
||
buildGetDocumentParams("bug1001080.pdf")
|
||
);
|
||
// PDF document with "Print" Named action in the OpenAction dictionary,
|
||
// but the OpenAction dictionary is missing the `Type` entry.
|
||
const loadingTask2 = getDocument(
|
||
buildGetDocumentParams("issue11442_reduced.pdf")
|
||
);
|
||
|
||
const promise1 = loadingTask1.promise
|
||
.then(function (pdfDoc) {
|
||
return pdfDoc.getOpenAction();
|
||
})
|
||
.then(function (openAction) {
|
||
expect(openAction.dest).toBeUndefined();
|
||
expect(openAction.action).toEqual("Print");
|
||
|
||
return loadingTask1.destroy();
|
||
});
|
||
const promise2 = loadingTask2.promise
|
||
.then(function (pdfDoc) {
|
||
return pdfDoc.getOpenAction();
|
||
})
|
||
.then(function (openAction) {
|
||
expect(openAction.dest).toBeUndefined();
|
||
expect(openAction.action).toEqual("Print");
|
||
|
||
return loadingTask2.destroy();
|
||
});
|
||
|
||
Promise.all([promise1, promise2]).then(done, done.fail);
|
||
});
|
||
|
||
it("gets non-existent attachments", function (done) {
|
||
const promise = pdfDocument.getAttachments();
|
||
promise
|
||
.then(function (data) {
|
||
expect(data).toEqual(null);
|
||
done();
|
||
})
|
||
.catch(done.fail);
|
||
});
|
||
it("gets attachments", function (done) {
|
||
const loadingTask = getDocument(buildGetDocumentParams("attachment.pdf"));
|
||
const promise = loadingTask.promise.then(function (pdfDoc) {
|
||
return pdfDoc.getAttachments();
|
||
});
|
||
promise
|
||
.then(function (data) {
|
||
const attachment = data["foo.txt"];
|
||
expect(attachment.filename).toEqual("foo.txt");
|
||
expect(attachment.content).toEqual(
|
||
new Uint8Array([98, 97, 114, 32, 98, 97, 122, 32, 10])
|
||
);
|
||
|
||
loadingTask.destroy().then(done);
|
||
})
|
||
.catch(done.fail);
|
||
});
|
||
|
||
it("gets javascript", function (done) {
|
||
const promise = pdfDocument.getJavaScript();
|
||
promise
|
||
.then(function (data) {
|
||
expect(data).toEqual(null);
|
||
done();
|
||
})
|
||
.catch(done.fail);
|
||
});
|
||
it("gets javascript with printing instructions (JS action)", function (done) {
|
||
// PDF document with "JavaScript" action in the OpenAction dictionary.
|
||
const loadingTask = getDocument(buildGetDocumentParams("issue6106.pdf"));
|
||
const promise = loadingTask.promise.then(function (pdfDoc) {
|
||
return pdfDoc.getJavaScript();
|
||
});
|
||
promise
|
||
.then(function (data) {
|
||
expect(data).toEqual([
|
||
"this.print({bUI:true,bSilent:false,bShrinkToFit:true});",
|
||
]);
|
||
expect(data[0]).toMatch(AutoPrintRegExp);
|
||
loadingTask.destroy().then(done);
|
||
})
|
||
.catch(done.fail);
|
||
});
|
||
|
||
it("gets JSActions (none)", function (done) {
|
||
const promise = pdfDocument.getJSActions();
|
||
promise
|
||
.then(function (data) {
|
||
expect(data).toEqual(null);
|
||
done();
|
||
})
|
||
.catch(done.fail);
|
||
});
|
||
it("gets JSActions", function (done) {
|
||
// PDF document with "JavaScript" action in the OpenAction dictionary.
|
||
const loadingTask = getDocument(
|
||
buildGetDocumentParams("doc_actions.pdf")
|
||
);
|
||
const promise = loadingTask.promise.then(async pdfDoc => {
|
||
const docActions = await pdfDoc.getJSActions();
|
||
const page1 = await pdfDoc.getPage(1);
|
||
const page3 = await pdfDoc.getPage(3);
|
||
const page1Actions = await page1.getJSActions();
|
||
const page3Actions = await page3.getJSActions();
|
||
return [docActions, page1Actions, page3Actions];
|
||
});
|
||
promise
|
||
.then(async ([docActions, page1Actions, page3Actions]) => {
|
||
expect(docActions).toEqual({
|
||
DidPrint: [`this.getField("Text2").value = "DidPrint";`],
|
||
DidSave: [`this.getField("Text2").value = "DidSave";`],
|
||
WillClose: [`this.getField("Text1").value = "WillClose";`],
|
||
WillPrint: [`this.getField("Text1").value = "WillPrint";`],
|
||
WillSave: [`this.getField("Text1").value = "WillSave";`],
|
||
});
|
||
expect(page1Actions).toEqual({
|
||
PageOpen: [`this.getField("Text1").value = "PageOpen 1";`],
|
||
PageClose: [`this.getField("Text2").value = "PageClose 1";`],
|
||
});
|
||
expect(page3Actions).toEqual({
|
||
PageOpen: [`this.getField("Text5").value = "PageOpen 3";`],
|
||
PageClose: [`this.getField("Text6").value = "PageClose 3";`],
|
||
});
|
||
loadingTask.destroy().then(done);
|
||
})
|
||
.catch(done.fail);
|
||
});
|
||
|
||
it("gets non-existent outline", function (done) {
|
||
const loadingTask = getDocument(
|
||
buildGetDocumentParams("tracemonkey.pdf")
|
||
);
|
||
|
||
const promise = loadingTask.promise.then(function (pdfDoc) {
|
||
return pdfDoc.getOutline();
|
||
});
|
||
promise
|
||
.then(function (outline) {
|
||
expect(outline).toEqual(null);
|
||
|
||
loadingTask.destroy().then(done);
|
||
})
|
||
.catch(done.fail);
|
||
});
|
||
it("gets outline", function (done) {
|
||
const promise = pdfDocument.getOutline();
|
||
promise
|
||
.then(function (outline) {
|
||
// Two top level entries.
|
||
expect(Array.isArray(outline)).toEqual(true);
|
||
expect(outline.length).toEqual(2);
|
||
// Make sure some basic attributes are set.
|
||
const outlineItem = outline[1];
|
||
expect(outlineItem.title).toEqual("Chapter 1");
|
||
expect(Array.isArray(outlineItem.dest)).toEqual(true);
|
||
expect(outlineItem.url).toEqual(null);
|
||
expect(outlineItem.unsafeUrl).toBeUndefined();
|
||
expect(outlineItem.newWindow).toBeUndefined();
|
||
|
||
expect(outlineItem.bold).toEqual(true);
|
||
expect(outlineItem.italic).toEqual(false);
|
||
expect(outlineItem.color).toEqual(
|
||
new Uint8ClampedArray([0, 64, 128])
|
||
);
|
||
|
||
expect(outlineItem.items.length).toEqual(1);
|
||
expect(outlineItem.items[0].title).toEqual("Paragraph 1.1");
|
||
done();
|
||
})
|
||
.catch(done.fail);
|
||
});
|
||
it("gets outline containing a url", function (done) {
|
||
const loadingTask = getDocument(buildGetDocumentParams("issue3214.pdf"));
|
||
|
||
loadingTask.promise
|
||
.then(function (pdfDoc) {
|
||
pdfDoc.getOutline().then(function (outline) {
|
||
expect(Array.isArray(outline)).toEqual(true);
|
||
expect(outline.length).toEqual(5);
|
||
|
||
const outlineItemTwo = outline[2];
|
||
expect(typeof outlineItemTwo.title).toEqual("string");
|
||
expect(outlineItemTwo.dest).toEqual(null);
|
||
expect(outlineItemTwo.url).toEqual("http://google.com/");
|
||
expect(outlineItemTwo.unsafeUrl).toEqual("http://google.com");
|
||
expect(outlineItemTwo.newWindow).toBeUndefined();
|
||
|
||
const outlineItemOne = outline[1];
|
||
expect(outlineItemOne.bold).toEqual(false);
|
||
expect(outlineItemOne.italic).toEqual(true);
|
||
expect(outlineItemOne.color).toEqual(
|
||
new Uint8ClampedArray([0, 0, 0])
|
||
);
|
||
|
||
loadingTask.destroy().then(done);
|
||
});
|
||
})
|
||
.catch(done.fail);
|
||
});
|
||
|
||
it("gets non-existent permissions", function (done) {
|
||
pdfDocument
|
||
.getPermissions()
|
||
.then(function (permissions) {
|
||
expect(permissions).toEqual(null);
|
||
|
||
done();
|
||
})
|
||
.catch(done.fail);
|
||
});
|
||
|
||
it("gets permissions", function (done) {
|
||
// Editing not allowed.
|
||
const loadingTask0 = getDocument(
|
||
buildGetDocumentParams("issue9972-1.pdf")
|
||
);
|
||
const promise0 = loadingTask0.promise.then(function (pdfDoc) {
|
||
return pdfDoc.getPermissions();
|
||
});
|
||
|
||
// Printing not allowed.
|
||
const loadingTask1 = getDocument(
|
||
buildGetDocumentParams("issue9972-2.pdf")
|
||
);
|
||
const promise1 = loadingTask1.promise.then(function (pdfDoc) {
|
||
return pdfDoc.getPermissions();
|
||
});
|
||
|
||
// Copying not allowed.
|
||
const loadingTask2 = getDocument(
|
||
buildGetDocumentParams("issue9972-3.pdf")
|
||
);
|
||
const promise2 = loadingTask2.promise.then(function (pdfDoc) {
|
||
return pdfDoc.getPermissions();
|
||
});
|
||
|
||
const totalPermissionCount = Object.keys(PermissionFlag).length;
|
||
Promise.all([promise0, promise1, promise2])
|
||
.then(function (permissions) {
|
||
expect(permissions[0].length).toEqual(totalPermissionCount - 1);
|
||
expect(
|
||
permissions[0].includes(PermissionFlag.MODIFY_CONTENTS)
|
||
).toBeFalsy();
|
||
|
||
expect(permissions[1].length).toEqual(totalPermissionCount - 2);
|
||
expect(permissions[1].includes(PermissionFlag.PRINT)).toBeFalsy();
|
||
expect(
|
||
permissions[1].includes(PermissionFlag.PRINT_HIGH_QUALITY)
|
||
).toBeFalsy();
|
||
|
||
expect(permissions[2].length).toEqual(totalPermissionCount - 1);
|
||
expect(permissions[2].includes(PermissionFlag.COPY)).toBeFalsy();
|
||
|
||
Promise.all([
|
||
loadingTask0.destroy(),
|
||
loadingTask1.destroy(),
|
||
loadingTask2.destroy(),
|
||
]).then(done);
|
||
})
|
||
.catch(done.fail);
|
||
});
|
||
|
||
it("gets metadata", function (done) {
|
||
const promise = pdfDocument.getMetadata();
|
||
promise
|
||
.then(function ({
|
||
info,
|
||
metadata,
|
||
contentDispositionFilename,
|
||
contentLength,
|
||
}) {
|
||
expect(info.Title).toEqual("Basic API Test");
|
||
// Custom, non-standard, information dictionary entries.
|
||
expect(info.Custom).toEqual(undefined);
|
||
// The following are PDF.js specific, non-standard, properties.
|
||
expect(info.PDFFormatVersion).toEqual("1.7");
|
||
expect(info.IsLinearized).toEqual(false);
|
||
expect(info.IsAcroFormPresent).toEqual(false);
|
||
expect(info.IsXFAPresent).toEqual(false);
|
||
expect(info.IsCollectionPresent).toEqual(false);
|
||
|
||
expect(metadata instanceof Metadata).toEqual(true);
|
||
expect(metadata.get("dc:title")).toEqual("Basic API Test");
|
||
|
||
expect(contentDispositionFilename).toEqual(null);
|
||
expect(contentLength).toEqual(basicApiFileLength);
|
||
done();
|
||
})
|
||
.catch(done.fail);
|
||
});
|
||
it("gets metadata, with custom info dict entries", function (done) {
|
||
const loadingTask = getDocument(
|
||
buildGetDocumentParams("tracemonkey.pdf")
|
||
);
|
||
|
||
loadingTask.promise
|
||
.then(function (pdfDoc) {
|
||
return pdfDoc.getMetadata();
|
||
})
|
||
.then(function ({
|
||
info,
|
||
metadata,
|
||
contentDispositionFilename,
|
||
contentLength,
|
||
}) {
|
||
expect(info.Creator).toEqual("TeX");
|
||
expect(info.Producer).toEqual("pdfeTeX-1.21a");
|
||
expect(info.CreationDate).toEqual("D:20090401163925-07'00'");
|
||
// Custom, non-standard, information dictionary entries.
|
||
const custom = info.Custom;
|
||
expect(typeof custom === "object" && custom !== null).toEqual(true);
|
||
|
||
expect(custom["PTEX.Fullbanner"]).toEqual(
|
||
"This is pdfeTeX, " +
|
||
"Version 3.141592-1.21a-2.2 (Web2C 7.5.4) kpathsea version 3.5.6"
|
||
);
|
||
// The following are PDF.js specific, non-standard, properties.
|
||
expect(info.PDFFormatVersion).toEqual("1.4");
|
||
expect(info.IsLinearized).toEqual(false);
|
||
expect(info.IsAcroFormPresent).toEqual(false);
|
||
expect(info.IsXFAPresent).toEqual(false);
|
||
expect(info.IsCollectionPresent).toEqual(false);
|
||
|
||
expect(metadata).toEqual(null);
|
||
expect(contentDispositionFilename).toEqual(null);
|
||
expect(contentLength).toEqual(1016315);
|
||
|
||
loadingTask.destroy().then(done);
|
||
})
|
||
.catch(done.fail);
|
||
});
|
||
it("gets metadata, with missing PDF header (bug 1606566)", function (done) {
|
||
const loadingTask = getDocument(buildGetDocumentParams("bug1606566.pdf"));
|
||
|
||
loadingTask.promise
|
||
.then(function (pdfDoc) {
|
||
return pdfDoc.getMetadata();
|
||
})
|
||
.then(function ({
|
||
info,
|
||
metadata,
|
||
contentDispositionFilename,
|
||
contentLength,
|
||
}) {
|
||
// The following are PDF.js specific, non-standard, properties.
|
||
expect(info.PDFFormatVersion).toEqual(null);
|
||
expect(info.IsLinearized).toEqual(false);
|
||
expect(info.IsAcroFormPresent).toEqual(false);
|
||
expect(info.IsXFAPresent).toEqual(false);
|
||
expect(info.IsCollectionPresent).toEqual(false);
|
||
|
||
expect(metadata).toEqual(null);
|
||
expect(contentDispositionFilename).toEqual(null);
|
||
expect(contentLength).toEqual(624);
|
||
|
||
loadingTask.destroy().then(done);
|
||
})
|
||
.catch(done.fail);
|
||
});
|
||
|
||
it("gets markInfo", function (done) {
|
||
const loadingTask = getDocument(
|
||
buildGetDocumentParams("annotation-line.pdf")
|
||
);
|
||
|
||
loadingTask.promise
|
||
.then(function (pdfDoc) {
|
||
return pdfDoc.getMarkInfo();
|
||
})
|
||
.then(function (info) {
|
||
expect(info.Marked).toEqual(true);
|
||
expect(info.UserProperties).toEqual(false);
|
||
expect(info.Suspects).toEqual(false);
|
||
done();
|
||
})
|
||
.catch(done.fail);
|
||
});
|
||
|
||
it("gets data", function (done) {
|
||
const promise = pdfDocument.getData();
|
||
promise
|
||
.then(function (data) {
|
||
expect(data instanceof Uint8Array).toEqual(true);
|
||
expect(data.length).toEqual(basicApiFileLength);
|
||
done();
|
||
})
|
||
.catch(done.fail);
|
||
});
|
||
it("gets download info", function (done) {
|
||
const promise = pdfDocument.getDownloadInfo();
|
||
promise
|
||
.then(function (data) {
|
||
expect(data).toEqual({ length: basicApiFileLength });
|
||
done();
|
||
})
|
||
.catch(done.fail);
|
||
});
|
||
it("gets document stats", function (done) {
|
||
const promise = pdfDocument.getStats();
|
||
promise
|
||
.then(function (stats) {
|
||
expect(stats).toEqual({ streamTypes: {}, fontTypes: {} });
|
||
done();
|
||
})
|
||
.catch(done.fail);
|
||
});
|
||
|
||
it("cleans up document resources", function (done) {
|
||
const promise = pdfDocument.cleanup();
|
||
promise.then(function () {
|
||
expect(true).toEqual(true);
|
||
done();
|
||
}, done.fail);
|
||
});
|
||
|
||
it("checks that fingerprints are unique", function (done) {
|
||
const loadingTask1 = getDocument(
|
||
buildGetDocumentParams("issue4436r.pdf")
|
||
);
|
||
const loadingTask2 = getDocument(buildGetDocumentParams("issue4575.pdf"));
|
||
|
||
Promise.all([loadingTask1.promise, loadingTask2.promise])
|
||
.then(function (data) {
|
||
const fingerprint1 = data[0].fingerprint;
|
||
const fingerprint2 = data[1].fingerprint;
|
||
|
||
expect(fingerprint1).not.toEqual(fingerprint2);
|
||
|
||
expect(fingerprint1).toEqual("2f695a83d6e7553c24fc08b7ac69712d");
|
||
expect(fingerprint2).toEqual("04c7126b34a46b6d4d6e7a1eff7edcb6");
|
||
|
||
Promise.all([loadingTask1.destroy(), loadingTask2.destroy()]).then(
|
||
done
|
||
);
|
||
})
|
||
.catch(done.fail);
|
||
});
|
||
|
||
describe("Cross-origin", function () {
|
||
let loadingTask;
|
||
function _checkCanLoad(expectSuccess, filename, options) {
|
||
if (isNodeJS) {
|
||
pending("Cannot simulate cross-origin requests in Node.js");
|
||
}
|
||
const params = buildGetDocumentParams(filename, options);
|
||
const url = new URL(params.url);
|
||
if (url.hostname === "localhost") {
|
||
url.hostname = "127.0.0.1";
|
||
} else if (params.url.hostname === "127.0.0.1") {
|
||
url.hostname = "localhost";
|
||
} else {
|
||
pending("Can only run cross-origin test on localhost!");
|
||
}
|
||
params.url = url.href;
|
||
loadingTask = getDocument(params);
|
||
return loadingTask.promise
|
||
.then(function (pdf) {
|
||
return pdf.destroy();
|
||
})
|
||
.then(
|
||
function () {
|
||
expect(expectSuccess).toEqual(true);
|
||
},
|
||
function (error) {
|
||
if (expectSuccess) {
|
||
// For ease of debugging.
|
||
expect(error).toEqual("There should not be any error");
|
||
}
|
||
expect(expectSuccess).toEqual(false);
|
||
}
|
||
);
|
||
}
|
||
function testCanLoad(filename, options) {
|
||
return _checkCanLoad(true, filename, options);
|
||
}
|
||
function testCannotLoad(filename, options) {
|
||
return _checkCanLoad(false, filename, options);
|
||
}
|
||
afterEach(function (done) {
|
||
if (loadingTask && !loadingTask.destroyed) {
|
||
loadingTask.destroy().then(done);
|
||
} else {
|
||
done();
|
||
}
|
||
});
|
||
it("server disallows cors", function (done) {
|
||
testCannotLoad("basicapi.pdf").then(done);
|
||
});
|
||
it("server allows cors without credentials, default withCredentials", function (done) {
|
||
testCanLoad("basicapi.pdf?cors=withoutCredentials").then(done);
|
||
});
|
||
it("server allows cors without credentials, and withCredentials=false", function (done) {
|
||
testCanLoad("basicapi.pdf?cors=withoutCredentials", {
|
||
withCredentials: false,
|
||
}).then(done);
|
||
});
|
||
it("server allows cors without credentials, but withCredentials=true", function (done) {
|
||
testCannotLoad("basicapi.pdf?cors=withoutCredentials", {
|
||
withCredentials: true,
|
||
}).then(done);
|
||
});
|
||
it("server allows cors with credentials, and withCredentials=true", function (done) {
|
||
testCanLoad("basicapi.pdf?cors=withCredentials", {
|
||
withCredentials: true,
|
||
}).then(done);
|
||
});
|
||
it("server allows cors with credentials, and withCredentials=false", function (done) {
|
||
// The server supports even more than we need, so if the previous tests
|
||
// pass, then this should pass for sure.
|
||
// The only case where this test fails is when the server does not reply
|
||
// with the Access-Control-Allow-Origin header.
|
||
testCanLoad("basicapi.pdf?cors=withCredentials", {
|
||
withCredentials: false,
|
||
}).then(done);
|
||
});
|
||
});
|
||
});
|
||
describe("Page", function () {
|
||
let pdfLoadingTask, pdfDocument, page;
|
||
|
||
beforeAll(function (done) {
|
||
pdfLoadingTask = getDocument(basicApiGetDocumentParams);
|
||
pdfLoadingTask.promise
|
||
.then(function (doc) {
|
||
pdfDocument = doc;
|
||
pdfDocument.getPage(1).then(function (data) {
|
||
page = data;
|
||
done();
|
||
});
|
||
})
|
||
.catch(done.fail);
|
||
});
|
||
|
||
afterAll(function (done) {
|
||
pdfLoadingTask.destroy().then(done);
|
||
});
|
||
|
||
it("gets page number", function () {
|
||
expect(page.pageNumber).toEqual(1);
|
||
});
|
||
it("gets rotate", function () {
|
||
expect(page.rotate).toEqual(0);
|
||
});
|
||
it("gets ref", function () {
|
||
expect(page.ref).toEqual({ num: 15, gen: 0 });
|
||
});
|
||
it("gets userUnit", function () {
|
||
expect(page.userUnit).toEqual(1.0);
|
||
});
|
||
|
||
it("gets view", function () {
|
||
expect(page.view).toEqual([0, 0, 595.28, 841.89]);
|
||
});
|
||
it("gets view, with empty/invalid bounding boxes", function (done) {
|
||
const viewLoadingTask = getDocument(
|
||
buildGetDocumentParams("boundingBox_invalid.pdf")
|
||
);
|
||
|
||
viewLoadingTask.promise
|
||
.then(pdfDoc => {
|
||
const numPages = pdfDoc.numPages;
|
||
expect(numPages).toEqual(3);
|
||
|
||
const viewPromises = [];
|
||
for (let i = 0; i < numPages; i++) {
|
||
viewPromises[i] = pdfDoc.getPage(i + 1).then(pdfPage => {
|
||
return pdfPage.view;
|
||
});
|
||
}
|
||
|
||
Promise.all(viewPromises).then(([page1, page2, page3]) => {
|
||
expect(page1).toEqual([0, 0, 612, 792]);
|
||
expect(page2).toEqual([0, 0, 800, 600]);
|
||
expect(page3).toEqual([0, 0, 600, 800]);
|
||
|
||
viewLoadingTask.destroy().then(done);
|
||
});
|
||
})
|
||
.catch(done.fail);
|
||
});
|
||
|
||
it("gets viewport", function () {
|
||
const viewport = page.getViewport({ scale: 1.5, rotation: 90 });
|
||
expect(viewport.viewBox).toEqual(page.view);
|
||
expect(viewport.scale).toEqual(1.5);
|
||
expect(viewport.rotation).toEqual(90);
|
||
expect(viewport.transform).toEqual([0, 1.5, 1.5, 0, 0, 0]);
|
||
expect(viewport.width).toEqual(1262.835);
|
||
expect(viewport.height).toEqual(892.92);
|
||
});
|
||
it('gets viewport with "offsetX/offsetY" arguments', function () {
|
||
const viewport = page.getViewport({
|
||
scale: 1,
|
||
rotation: 0,
|
||
offsetX: 100,
|
||
offsetY: -100,
|
||
});
|
||
expect(viewport.transform).toEqual([1, 0, 0, -1, 100, 741.89]);
|
||
});
|
||
it('gets viewport respecting "dontFlip" argument', function () {
|
||
const scale = 1,
|
||
rotation = 0;
|
||
const viewport = page.getViewport({ scale, rotation });
|
||
const dontFlipViewport = page.getViewport({
|
||
scale,
|
||
rotation,
|
||
dontFlip: true,
|
||
});
|
||
|
||
expect(dontFlipViewport).not.toEqual(viewport);
|
||
expect(dontFlipViewport).toEqual(viewport.clone({ dontFlip: true }));
|
||
|
||
expect(viewport.transform).toEqual([1, 0, 0, -1, 0, 841.89]);
|
||
expect(dontFlipViewport.transform).toEqual([1, 0, -0, 1, 0, 0]);
|
||
});
|
||
it("gets viewport with invalid rotation", function () {
|
||
expect(function () {
|
||
page.getViewport({ scale: 1, rotation: 45 });
|
||
}).toThrow(
|
||
new Error(
|
||
"PageViewport: Invalid rotation, must be a multiple of 90 degrees."
|
||
)
|
||
);
|
||
});
|
||
|
||
it("gets annotations", function (done) {
|
||
const defaultPromise = page.getAnnotations().then(function (data) {
|
||
expect(data.length).toEqual(4);
|
||
});
|
||
|
||
const displayPromise = page
|
||
.getAnnotations({ intent: "display" })
|
||
.then(function (data) {
|
||
expect(data.length).toEqual(4);
|
||
});
|
||
|
||
const printPromise = page
|
||
.getAnnotations({ intent: "print" })
|
||
.then(function (data) {
|
||
expect(data.length).toEqual(4);
|
||
});
|
||
Promise.all([defaultPromise, displayPromise, printPromise])
|
||
.then(function () {
|
||
done();
|
||
})
|
||
.catch(done.fail);
|
||
});
|
||
|
||
it("gets annotations containing relative URLs (bug 766086)", function (done) {
|
||
const filename = "bug766086.pdf";
|
||
|
||
const defaultLoadingTask = getDocument(buildGetDocumentParams(filename));
|
||
const defaultPromise = defaultLoadingTask.promise.then(function (pdfDoc) {
|
||
return pdfDoc.getPage(1).then(function (pdfPage) {
|
||
return pdfPage.getAnnotations();
|
||
});
|
||
});
|
||
|
||
const docBaseUrlLoadingTask = getDocument(
|
||
buildGetDocumentParams(filename, {
|
||
docBaseUrl: "http://www.example.com/test/pdfs/qwerty.pdf",
|
||
})
|
||
);
|
||
const docBaseUrlPromise = docBaseUrlLoadingTask.promise.then(function (
|
||
pdfDoc
|
||
) {
|
||
return pdfDoc.getPage(1).then(function (pdfPage) {
|
||
return pdfPage.getAnnotations();
|
||
});
|
||
});
|
||
|
||
const invalidDocBaseUrlLoadingTask = getDocument(
|
||
buildGetDocumentParams(filename, {
|
||
docBaseUrl: "qwerty.pdf",
|
||
})
|
||
);
|
||
const invalidDocBaseUrlPromise = invalidDocBaseUrlLoadingTask.promise.then(
|
||
function (pdfDoc) {
|
||
return pdfDoc.getPage(1).then(function (pdfPage) {
|
||
return pdfPage.getAnnotations();
|
||
});
|
||
}
|
||
);
|
||
|
||
Promise.all([defaultPromise, docBaseUrlPromise, invalidDocBaseUrlPromise])
|
||
.then(function (data) {
|
||
const defaultAnnotations = data[0];
|
||
const docBaseUrlAnnotations = data[1];
|
||
const invalidDocBaseUrlAnnotations = data[2];
|
||
|
||
expect(defaultAnnotations[0].url).toBeUndefined();
|
||
expect(defaultAnnotations[0].unsafeUrl).toEqual(
|
||
"../../0021/002156/215675E.pdf#15"
|
||
);
|
||
|
||
expect(docBaseUrlAnnotations[0].url).toEqual(
|
||
"http://www.example.com/0021/002156/215675E.pdf#15"
|
||
);
|
||
expect(docBaseUrlAnnotations[0].unsafeUrl).toEqual(
|
||
"../../0021/002156/215675E.pdf#15"
|
||
);
|
||
|
||
expect(invalidDocBaseUrlAnnotations[0].url).toBeUndefined();
|
||
expect(invalidDocBaseUrlAnnotations[0].unsafeUrl).toEqual(
|
||
"../../0021/002156/215675E.pdf#15"
|
||
);
|
||
|
||
Promise.all([
|
||
defaultLoadingTask.destroy(),
|
||
docBaseUrlLoadingTask.destroy(),
|
||
invalidDocBaseUrlLoadingTask.destroy(),
|
||
]).then(done);
|
||
})
|
||
.catch(done.fail);
|
||
});
|
||
|
||
it("gets text content", function (done) {
|
||
const defaultPromise = page.getTextContent();
|
||
const parametersPromise = page.getTextContent({
|
||
normalizeWhitespace: true,
|
||
disableCombineTextItems: true,
|
||
});
|
||
|
||
const promises = [defaultPromise, parametersPromise];
|
||
Promise.all(promises)
|
||
.then(function (data) {
|
||
expect(!!data[0].items).toEqual(true);
|
||
expect(data[0].items.length).toEqual(7);
|
||
expect(!!data[0].styles).toEqual(true);
|
||
|
||
// A simple check that ensures the two `textContent` object match.
|
||
expect(JSON.stringify(data[0])).toEqual(JSON.stringify(data[1]));
|
||
done();
|
||
})
|
||
.catch(done.fail);
|
||
});
|
||
|
||
it("gets text content, with correct properties (issue 8276)", function (done) {
|
||
const loadingTask = getDocument(
|
||
buildGetDocumentParams("issue8276_reduced.pdf")
|
||
);
|
||
|
||
loadingTask.promise
|
||
.then(pdfDoc => {
|
||
pdfDoc.getPage(1).then(pdfPage => {
|
||
pdfPage.getTextContent().then(({ items, styles }) => {
|
||
expect(items.length).toEqual(1);
|
||
expect(Object.keys(styles)).toEqual(["Times"]);
|
||
|
||
expect(items[0]).toEqual({
|
||
dir: "ltr",
|
||
fontName: "Times",
|
||
height: 18,
|
||
str: "Issue 8276",
|
||
transform: [18, 0, 0, 18, 441.81, 708.4499999999999],
|
||
width: 77.49,
|
||
});
|
||
expect(styles.Times).toEqual({
|
||
fontFamily: "serif",
|
||
ascent: NaN,
|
||
descent: NaN,
|
||
vertical: false,
|
||
});
|
||
|
||
loadingTask.destroy().then(done);
|
||
});
|
||
});
|
||
})
|
||
.catch(done.fail);
|
||
});
|
||
|
||
it("gets operator list", function (done) {
|
||
const promise = page.getOperatorList();
|
||
promise
|
||
.then(function (oplist) {
|
||
expect(!!oplist.fnArray).toEqual(true);
|
||
expect(!!oplist.argsArray).toEqual(true);
|
||
expect(oplist.lastChunk).toEqual(true);
|
||
done();
|
||
})
|
||
.catch(done.fail);
|
||
});
|
||
|
||
it("gets operatorList with JPEG image (issue 4888)", function (done) {
|
||
const loadingTask = getDocument(buildGetDocumentParams("cmykjpeg.pdf"));
|
||
|
||
loadingTask.promise
|
||
.then(pdfDoc => {
|
||
pdfDoc.getPage(1).then(pdfPage => {
|
||
pdfPage.getOperatorList().then(opList => {
|
||
const imgIndex = opList.fnArray.indexOf(OPS.paintImageXObject);
|
||
const imgArgs = opList.argsArray[imgIndex];
|
||
const { data } = pdfPage.objs.get(imgArgs[0]);
|
||
|
||
expect(data instanceof Uint8ClampedArray).toEqual(true);
|
||
expect(data.length).toEqual(90000);
|
||
|
||
loadingTask.destroy().then(done);
|
||
});
|
||
});
|
||
})
|
||
.catch(done.fail);
|
||
});
|
||
|
||
it(
|
||
"gets operatorList, from corrupt PDF file (issue 8702), " +
|
||
"with/without `stopAtErrors` set",
|
||
function (done) {
|
||
const loadingTask1 = getDocument(
|
||
buildGetDocumentParams("issue8702.pdf", {
|
||
stopAtErrors: false, // The default value.
|
||
})
|
||
);
|
||
const loadingTask2 = getDocument(
|
||
buildGetDocumentParams("issue8702.pdf", {
|
||
stopAtErrors: true,
|
||
})
|
||
);
|
||
|
||
const result1 = loadingTask1.promise.then(pdfDoc => {
|
||
return pdfDoc.getPage(1).then(pdfPage => {
|
||
return pdfPage.getOperatorList().then(opList => {
|
||
expect(opList.fnArray.length).toBeGreaterThan(100);
|
||
expect(opList.argsArray.length).toBeGreaterThan(100);
|
||
expect(opList.lastChunk).toEqual(true);
|
||
|
||
return loadingTask1.destroy();
|
||
});
|
||
});
|
||
});
|
||
|
||
const result2 = loadingTask2.promise.then(pdfDoc => {
|
||
return pdfDoc.getPage(1).then(pdfPage => {
|
||
return pdfPage.getOperatorList().then(opList => {
|
||
expect(opList.fnArray.length).toEqual(0);
|
||
expect(opList.argsArray.length).toEqual(0);
|
||
expect(opList.lastChunk).toEqual(true);
|
||
|
||
return loadingTask2.destroy();
|
||
});
|
||
});
|
||
}); |
|
|
Promise.all([result1, result2]).then(done, done.fail);
|
||
}
|
||
);
|
||
|
||
it("gets document stats after parsing page", function (done) {
|
||
const promise = page.getOperatorList().then(function () {
|
||
return pdfDocument.getStats();
|
||
});
|
||
const expectedStreamTypes = {};
|
||
expectedStreamTypes[StreamType.FLATE] = true;
|
||
const expectedFontTypes = {};
|
||
expectedFontTypes[FontType.TYPE1] = true;
|
||
expectedFontTypes[FontType.CIDFONTTYPE2] = true;
|
||
|
||
promise
|
||
.then(function (stats) {
|
||
expect(stats).toEqual({
|
||
streamTypes: expectedStreamTypes,
|
||
fontTypes: expectedFontTypes,
|
||
});
|
||
done();
|
||
})
|
||
.catch(done.fail);
|
||
});
|
||
|
||
it("gets page stats after parsing page, without `pdfBug` set", function (done) {
|
||
page
|
||
.getOperatorList()
|
||
.then(opList => {
|
||
return page.stats;
|
||
})
|
||
.then(stats => {
|
||
expect(stats).toEqual(null);
|
||
done();
|
||
}, done.fail);
|
||
});
|
||
it("gets page stats after parsing page, with `pdfBug` set", function (done) {
|
||
const loadingTask = getDocument(
|
||
buildGetDocumentParams(basicApiFileName, { pdfBug: true })
|
||
);
|
||
|
||
loadingTask.promise
|
||
.then(pdfDoc => {
|
||
return pdfDoc.getPage(1).then(pdfPage => {
|
||
return pdfPage.getOperatorList().then(opList => {
|
||
return pdfPage.stats;
|
||
});
|
||
});
|
||
})
|
||
.then(stats => {
|
||
expect(stats instanceof StatTimer).toEqual(true);
|
||
expect(stats.times.length).toEqual(1);
|
||
|
||
const [statEntry] = stats.times;
|
||
expect(statEntry.name).toEqual("Page Request");
|
||
expect(statEntry.end - statEntry.start).toBeGreaterThanOrEqual(0);
|
||
|
||
loadingTask.destroy().then(done);
|
||
}, done.fail);
|
||
});
|
||
it("gets page stats after rendering page, with `pdfBug` set", function (done) {
|
||
const loadingTask = getDocument(
|
||
buildGetDocumentParams(basicApiFileName, { pdfBug: true })
|
||
);
|
||
let canvasAndCtx;
|
||
|
||
loadingTask.promise
|
||
.then(pdfDoc => {
|
||
return pdfDoc.getPage(1).then(pdfPage => {
|
||
const viewport = pdfPage.getViewport({ scale: 1 });
|
||
canvasAndCtx = CanvasFactory.create(
|
||
viewport.width,
|
||
viewport.height
|
||
);
|
||
|
||
const renderTask = pdfPage.render({
|
||
canvasContext: canvasAndCtx.context,
|
||
canvasFactory: CanvasFactory,
|
||
viewport,
|
||
});
|
||
return renderTask.promise.then(() => {
|
||
return pdfPage.stats;
|
||
});
|
||
});
|
||
})
|
||
.then(stats => {
|
||
expect(stats instanceof StatTimer).toEqual(true);
|
||
expect(stats.times.length).toEqual(3);
|
||
|
||
const [statEntryOne, statEntryTwo, statEntryThree] = stats.times;
|
||
expect(statEntryOne.name).toEqual("Page Request");
|
||
expect(statEntryOne.end - statEntryOne.start).toBeGreaterThanOrEqual(
|
||
0
|
||
);
|
||
|
||
expect(statEntryTwo.name).toEqual("Rendering");
|
||
expect(statEntryTwo.end - statEntryTwo.start).toBeGreaterThan(0);
|
||
|
||
expect(statEntryThree.name).toEqual("Overall");
|
||
expect(statEntryThree.end - statEntryThree.start).toBeGreaterThan(0);
|
||
|
||
CanvasFactory.destroy(canvasAndCtx);
|
||
loadingTask.destroy().then(done);
|
||
}, done.fail);
|
||
});
|
||
|
||
it("cancels rendering of page", function (done) {
|
||
const viewport = page.getViewport({ scale: 1 });
|
||
const canvasAndCtx = CanvasFactory.create(
|
||
viewport.width,
|
||
viewport.height
|
||
);
|
||
|
||
const renderTask = page.render({
|
||
canvasContext: canvasAndCtx.context,
|
||
canvasFactory: CanvasFactory,
|
||
viewport,
|
||
});
|
||
renderTask.cancel();
|
||
|
||
renderTask.promise
|
||
.then(function () {
|
||
done.fail("shall cancel rendering");
|
||
})
|
||
.catch(function (error) {
|
||
expect(error instanceof RenderingCancelledException).toEqual(true);
|
||
expect(error.message).toEqual("Rendering cancelled, page 1");
|
||
expect(error.type).toEqual("canvas");
|
||
|
||
CanvasFactory.destroy(canvasAndCtx);
|
||
done();
|
||
});
|
||
});
|
||
|
||
it("re-render page, using the same canvas, after cancelling rendering", function (done) {
|
||
const viewport = page.getViewport({ scale: 1 });
|
||
const canvasAndCtx = CanvasFactory.create(
|
||
viewport.width,
|
||
viewport.height
|
||
);
|
||
|
||
const renderTask = page.render({
|
||
canvasContext: canvasAndCtx.context,
|
||
canvasFactory: CanvasFactory,
|
||
viewport,
|
||
});
|
||
renderTask.cancel();
|
||
|
||
renderTask.promise
|
||
.then(
|
||
() => {
|
||
throw new Error("shall cancel rendering");
|
||
},
|
||
reason => {
|
||
expect(reason instanceof RenderingCancelledException).toEqual(true);
|
||
}
|
||
)
|
||
.then(() => {
|
||
const reRenderTask = page.render({
|
||
canvasContext: canvasAndCtx.context,
|
||
canvasFactory: CanvasFactory,
|
||
viewport,
|
||
});
|
||
return reRenderTask.promise;
|
||
})
|
||
.then(() => {
|
||
CanvasFactory.destroy(canvasAndCtx);
|
||
done();
|
||
}, done.fail);
|
||
});
|
||
|
||
it("multiple render() on the same canvas", function (done) {
|
||
const optionalContentConfigPromise = pdfDocument.getOptionalContentConfig();
|
||
|
||
const viewport = page.getViewport({ scale: 1 });
|
||
const canvasAndCtx = CanvasFactory.create(
|
||
viewport.width,
|
||
viewport.height
|
||
);
|
||
|
||
const renderTask1 = page.render({
|
||
canvasContext: canvasAndCtx.context,
|
||
canvasFactory: CanvasFactory,
|
||
viewport,
|
||
optionalContentConfigPromise,
|
||
});
|
||
const renderTask2 = page.render({
|
||
canvasContext: canvasAndCtx.context,
|
||
canvasFactory: CanvasFactory,
|
||
viewport,
|
||
optionalContentConfigPromise,
|
||
});
|
||
|
||
Promise.all([
|
||
renderTask1.promise,
|
||
renderTask2.promise.then(
|
||
() => {
|
||
done.fail("shall fail rendering");
|
||
},
|
||
reason => {
|
||
/* it fails because we already using this canvas */
|
||
expect(/multiple render\(\)/.test(reason.message)).toEqual(true);
|
||
}
|
||
),
|
||
]).then(done);
|
||
});
|
||
|
||
it("cleans up document resources after rendering of page", function (done) {
|
||
const loadingTask = getDocument(buildGetDocumentParams(basicApiFileName));
|
||
let canvasAndCtx;
|
||
|
||
loadingTask.promise
|
||
.then(pdfDoc => {
|
||
return pdfDoc.getPage(1).then(pdfPage => {
|
||
const viewport = pdfPage.getViewport({ scale: 1 });
|
||
canvasAndCtx = CanvasFactory.create(
|
||
viewport.width,
|
||
viewport.height
|
||
);
|
||
|
||
const renderTask = pdfPage.render({
|
||
canvasContext: canvasAndCtx.context,
|
||
canvasFactory: CanvasFactory,
|
||
viewport,
|
||
});
|
||
return renderTask.promise.then(() => {
|
||
return pdfDoc.cleanup();
|
||
});
|
||
});
|
||
})
|
||
.then(() => {
|
||
expect(true).toEqual(true);
|
||
|
||
CanvasFactory.destroy(canvasAndCtx);
|
||
loadingTask.destroy().then(done);
|
||
}, done.fail);
|
||
});
|
||
|
||
it("cleans up document resources during rendering of page", function (done) {
|
||
const loadingTask = getDocument(
|
||
buildGetDocumentParams("tracemonkey.pdf")
|
||
);
|
||
let canvasAndCtx;
|
||
|
||
loadingTask.promise
|
||
.then(pdfDoc => {
|
||
return pdfDoc.getPage(1).then(pdfPage => {
|
||
const viewport = pdfPage.getViewport({ scale: 1 });
|
||
canvasAndCtx = CanvasFactory.create(
|
||
viewport.width,
|
||
viewport.height
|
||
);
|
||
|
||
const renderTask = pdfPage.render({
|
||
canvasContext: canvasAndCtx.context,
|
||
canvasFactory: CanvasFactory,
|
||
viewport,
|
||
});
|
||
|
||
renderTask.onContinue = function (cont) {
|
||
waitSome(cont);
|
||
};
|
||
|
||
return pdfDoc
|
||
.cleanup()
|
||
.then(
|
||
() => {
|
||
throw new Error("shall fail cleanup");
|
||
},
|
||
reason => {
|
||
expect(reason instanceof Error).toEqual(true);
|
||
expect(reason.message).toEqual(
|
||
"startCleanup: Page 1 is currently rendering."
|
||
);
|
||
}
|
||
)
|
||
.then(() => {
|
||
return renderTask.promise;
|
||
})
|
||
.then(() => {
|
||
CanvasFactory.destroy(canvasAndCtx);
|
||
loadingTask.destroy().then(done);
|
||
});
|
||
});
|
||
})
|
||
.catch(done.fail);
|
||
});
|
||
|
||
it("caches image resources at the document/page level as expected (issue 11878)", async function (done) {
|
||
const { NUM_PAGES_THRESHOLD } = GlobalImageCache,
|
||
EXPECTED_WIDTH = 2550,
|
||
EXPECTED_HEIGHT = 3300;
|
||
|
||
const loadingTask = getDocument(buildGetDocumentParams("issue11878.pdf"));
|
||
let firstImgData = null;
|
||
|
||
try {
|
||
const pdfDoc = await loadingTask.promise;
|
||
|
||
for (let i = 1; i <= pdfDoc.numPages; i++) {
|
||
const pdfPage = await pdfDoc.getPage(i);
|
||
const opList = await pdfPage.getOperatorList();
|
||
|
||
const { commonObjs, objs } = pdfPage;
|
||
const imgIndex = opList.fnArray.indexOf(OPS.paintImageXObject);
|
||
const [objId, width, height] = opList.argsArray[imgIndex];
|
||
|
||
if (i < NUM_PAGES_THRESHOLD) {
|
||
expect(objId).toEqual(`img_p${i - 1}_1`);
|
||
|
||
expect(objs.has(objId)).toEqual(true);
|
||
expect(commonObjs.has(objId)).toEqual(false);
|
||
} else {
|
||
expect(objId).toEqual(
|
||
`g_${loadingTask.docId}_img_p${NUM_PAGES_THRESHOLD - 1}_1`
|
||
);
|
||
|
||
expect(objs.has(objId)).toEqual(false);
|
||
expect(commonObjs.has(objId)).toEqual(true);
|
||
}
|
||
expect(width).toEqual(EXPECTED_WIDTH);
|
||
expect(height).toEqual(EXPECTED_HEIGHT);
|
||
|
||
// Ensure that the actual image data is identical for all pages.
|
||
if (i === 1) {
|
||
firstImgData = objs.get(objId);
|
||
|
||
expect(firstImgData.width).toEqual(EXPECTED_WIDTH);
|
||
expect(firstImgData.height).toEqual(EXPECTED_HEIGHT);
|
||
|
||
expect(firstImgData.kind).toEqual(ImageKind.RGB_24BPP);
|
||
expect(firstImgData.data instanceof Uint8ClampedArray).toEqual(
|
||
true
|
||
);
|
||
expect(firstImgData.data.length).toEqual(25245000);
|
||
} else {
|
||
const objsPool = i >= NUM_PAGES_THRESHOLD ? commonObjs : objs;
|
||
const currentImgData = objsPool.get(objId);
|
||
|
||
expect(currentImgData.width).toEqual(firstImgData.width);
|
||
expect(currentImgData.height).toEqual(firstImgData.height);
|
||
|
||
expect(currentImgData.kind).toEqual(firstImgData.kind);
|
||
expect(currentImgData.data instanceof Uint8ClampedArray).toEqual(
|
||
true
|
||
);
|
||
expect(
|
||
currentImgData.data.every((value, index) => {
|
||
return value === firstImgData.data[index];
|
||
})
|
||
).toEqual(true);
|
||
}
|
||
}
|
||
|
||
await loadingTask.destroy();
|
||
firstImgData = null;
|
||
done();
|
||
} catch (ex) {
|
||
done.fail(ex);
|
||
}
|
||
});
|
||
});
|
||
describe("Multiple `getDocument` instances", function () {
|
||
// Regression test for https://github.com/mozilla/pdf.js/issues/6205
|
||
// A PDF using the Helvetica font.
|
||
const pdf1 = buildGetDocumentParams("tracemonkey.pdf");
|
||
// A PDF using the Times font.
|
||
const pdf2 = buildGetDocumentParams("TAMReview.pdf");
|
||
// A PDF using the Arial font.
|
||
const pdf3 = buildGetDocumentParams("issue6068.pdf");
|
||
const loadingTasks = [];
|
||
|
||
// Render the first page of the given PDF file.
|
||
// Fulfills the promise with the base64-encoded version of the PDF.
|
||
async function renderPDF(filename) {
|
||
const loadingTask = getDocument(filename);
|
||
loadingTasks.push(loadingTask);
|
||
const pdf = await loadingTask.promise;
|
||
const page = await pdf.getPage(1);
|
||
const viewport = page.getViewport({ scale: 1.2 });
|
||
const canvasAndCtx = CanvasFactory.create(
|
||
viewport.width,
|
||
viewport.height
|
||
);
|
||
const renderTask = page.render({
|
||
canvasContext: canvasAndCtx.context,
|
||
canvasFactory: CanvasFactory,
|
||
viewport,
|
||
});
|
||
await renderTask.promise;
|
||
const data = canvasAndCtx.canvas.toDataURL();
|
||
CanvasFactory.destroy(canvasAndCtx);
|
||
return data;
|
||
}
|
||
|
||
afterEach(function (done) {
|
||
// Issue 6205 reported an issue with font rendering, so clear the loaded
|
||
// fonts so that we can see whether loading PDFs in parallel does not
|
||
// cause any issues with the rendered fonts.
|
||
const destroyPromises = loadingTasks.map(function (loadingTask) {
|
||
return loadingTask.destroy();
|
||
});
|
||
Promise.all(destroyPromises).then(done);
|
||
});
|
||
|
||
it("should correctly render PDFs in parallel", function (done) {
|
||
let baseline1, baseline2, baseline3;
|
||
const promiseDone = renderPDF(pdf1)
|
||
.then(function (data1) {
|
||
baseline1 = data1;
|
||
return renderPDF(pdf2);
|
||
})
|
||
.then(function (data2) {
|
||
baseline2 = data2;
|
||
return renderPDF(pdf3);
|
||
})
|
||
.then(function (data3) {
|
||
baseline3 = data3;
|
||
return Promise.all([
|
||
renderPDF(pdf1),
|
||
renderPDF(pdf2),
|
||
renderPDF(pdf3),
|
||
]);
|
||
})
|
||
.then(function (dataUrls) {
|
||
expect(dataUrls[0]).toEqual(baseline1);
|
||
expect(dataUrls[1]).toEqual(baseline2);
|
||
expect(dataUrls[2]).toEqual(baseline3);
|
||
return true;
|
||
});
|
||
promiseDone
|
||
.then(function () {
|
||
done();
|
||
})
|
||
.catch(done.fail);
|
||
});
|
||
});
|
||
|
||
describe("PDFDataRangeTransport", function () {
|
||
let dataPromise;
|
||
|
||
beforeAll(function (done) {
|
||
const fileName = "tracemonkey.pdf";
|
||
dataPromise = DefaultFileReaderFactory.fetch({
|
||
path: TEST_PDFS_PATH + fileName,
|
||
});
|
||
done();
|
||
});
|
||
|
||
afterAll(function () {
|
||
dataPromise = null;
|
||
});
|
||
|
||
it("should fetch document info and page using ranges", function (done) {
|
||
const initialDataLength = 4000;
|
||
let fetches = 0,
|
||
loadingTask;
|
||
|
||
dataPromise
|
||
.then(function (data) {
|
||
const initialData = data.subarray(0, initialDataLength);
|
||
const transport = new PDFDataRangeTransport(data.length, initialData);
|
||
transport.requestDataRange = function (begin, end) {
|
||
fetches++;
|
||
waitSome(function () {
|
||
transport.onDataProgress(4000);
|
||
transport.onDataRange(begin, data.subarray(begin, end));
|
||
});
|
||
};
|
||
loadingTask = getDocument(transport);
|
||
return loadingTask.promise;
|
||
})
|
||
.then(function (pdfDocument) {
|
||
expect(pdfDocument.numPages).toEqual(14);
|
||
|
||
return pdfDocument.getPage(10);
|
||
})
|
||
.then(function (pdfPage) {
|
||
expect(pdfPage.rotate).toEqual(0);
|
||
expect(fetches).toBeGreaterThan(2);
|
||
|
||
loadingTask.destroy().then(done);
|
||
})
|
||
.catch(done.fail);
|
||
});
|
||
|
||
it("should fetch document info and page using range and streaming", function (done) {
|
||
const initialDataLength = 4000;
|
||
let fetches = 0,
|
||
loadingTask;
|
||
|
||
dataPromise
|
||
.then(function (data) {
|
||
const initialData = data.subarray(0, initialDataLength);
|
||
const transport = new PDFDataRangeTransport(data.length, initialData);
|
||
transport.requestDataRange = function (begin, end) {
|
||
fetches++;
|
||
if (fetches === 1) {
|
||
// Send rest of the data on first range request.
|
||
transport.onDataProgressiveRead(data.subarray(initialDataLength));
|
||
}
|
||
waitSome(function () {
|
||
transport.onDataRange(begin, data.subarray(begin, end));
|
||
});
|
||
};
|
||
loadingTask = getDocument(transport);
|
||
return loadingTask.promise;
|
||
})
|
||
.then(function (pdfDocument) {
|
||
expect(pdfDocument.numPages).toEqual(14);
|
||
|
||
return pdfDocument.getPage(10);
|
||
})
|
||
.then(function (pdfPage) {
|
||
expect(pdfPage.rotate).toEqual(0);
|
||
expect(fetches).toEqual(1);
|
||
|
||
waitSome(function () {
|
||
loadingTask.destroy().then(done);
|
||
});
|
||
})
|
||
.catch(done.fail);
|
||
});
|
||
|
||
it(
|
||
"should fetch document info and page, without range, " +
|
||
"using complete initialData",
|
||
function (done) {
|
||
let fetches = 0,
|
||
loadingTask;
|
||
|
||
dataPromise
|
||
.then(function (data) {
|
||
const transport = new PDFDataRangeTransport(
|
||
data.length,
|
||
data,
|
||
/* progressiveDone = */ true
|
||
);
|
||
transport.requestDataRange = function (begin, end) {
|
||
fetches++;
|
||
};
|
||
loadingTask = getDocument({ disableRange: true, range: transport });
|
||
return loadingTask.promise;
|
||
})
|
||
.then(function (pdfDocument) {
|
||
expect(pdfDocument.numPages).toEqual(14);
|
||
|
||
return pdfDocument.getPage(10);
|
||
})
|
||
.then(function (pdfPage) {
|
||
expect(pdfPage.rotate).toEqual(0);
|
||
expect(fetches).toEqual(0);
|
||
|
||
loadingTask.destroy().then(done);
|
||
})
|
||
.catch(done.fail);
|
||
}
|
||
);
|
||
});
|
||
});
|