Merge pull request #15920 from Snuffleupagus/transfer-pdf-data
[api-minor] Enable transferring of TypedArray PDF data by default (PR 15908 follow-up)
This commit is contained in:
		
						commit
						8f3fa18c93
					
				| @ -139,8 +139,12 @@ if (typeof PDFJSDev === "undefined" || !PDFJSDev.test("PRODUCTION")) { | |||||||
|  * @typedef {Object} DocumentInitParameters |  * @typedef {Object} DocumentInitParameters | ||||||
|  * @property {string | URL} [url] - The URL of the PDF. |  * @property {string | URL} [url] - The URL of the PDF. | ||||||
|  * @property {BinaryData} [data] - Binary PDF data. |  * @property {BinaryData} [data] - Binary PDF data. | ||||||
|  *   Use typed arrays (Uint8Array) to improve the memory usage. If PDF data is |  *   Use TypedArrays (Uint8Array) to improve the memory usage. If PDF data is | ||||||
|  *   BASE64-encoded, use `atob()` to convert it to a binary string first. |  *   BASE64-encoded, use `atob()` to convert it to a binary string first. | ||||||
|  |  * | ||||||
|  |  *   NOTE: If TypedArrays are used they will generally be transferred to the | ||||||
|  |  *   worker-thread. This will help reduce main-thread memory usage, however | ||||||
|  |  *   it will take ownership of the TypedArrays. | ||||||
|  * @property {Object} [httpHeaders] - Basic authentication headers. |  * @property {Object} [httpHeaders] - Basic authentication headers. | ||||||
|  * @property {boolean} [withCredentials] - Indicates whether or not |  * @property {boolean} [withCredentials] - Indicates whether or not | ||||||
|  *   cross-site Access-Control requests should be made using credentials such |  *   cross-site Access-Control requests should be made using credentials such | ||||||
| @ -189,12 +193,6 @@ if (typeof PDFJSDev === "undefined" || !PDFJSDev.test("PRODUCTION")) { | |||||||
|  * @property {number} [maxImageSize] - The maximum allowed image size in total |  * @property {number} [maxImageSize] - The maximum allowed image size in total | ||||||
|  *   pixels, i.e. width * height. Images above this value will not be rendered. |  *   pixels, i.e. width * height. Images above this value will not be rendered. | ||||||
|  *   Use -1 for no limit, which is also the default value. |  *   Use -1 for no limit, which is also the default value. | ||||||
|  * @property {boolean} [transferPdfData] - Determines if we can transfer |  | ||||||
|  *   TypedArrays used for loading the PDF file, utilized together with: |  | ||||||
|  *    - The `data`-option, for the `getDocument` function. |  | ||||||
|  *    - The `PDFDataTransportStream` implementation. |  | ||||||
|  *   This will help reduce main-thread memory usage, however it will take |  | ||||||
|  *   ownership of the TypedArrays. The default value is `false`. |  | ||||||
|  * @property {boolean} [isEvalSupported] - Determines if we can evaluate strings |  * @property {boolean} [isEvalSupported] - Determines if we can evaluate strings | ||||||
|  *   as JavaScript. Primarily used to improve performance of font rendering, and |  *   as JavaScript. Primarily used to improve performance of font rendering, and | ||||||
|  *   when parsing PDF functions. The default value is `true`. |  *   when parsing PDF functions. The default value is `true`. | ||||||
| @ -281,20 +279,20 @@ function getDocument(src) { | |||||||
|     worker = null; |     worker = null; | ||||||
| 
 | 
 | ||||||
|   for (const key in source) { |   for (const key in source) { | ||||||
|     const value = source[key]; |     const val = source[key]; | ||||||
| 
 | 
 | ||||||
|     switch (key) { |     switch (key) { | ||||||
|       case "url": |       case "url": | ||||||
|         if (typeof window !== "undefined") { |         if (typeof window !== "undefined") { | ||||||
|           try { |           try { | ||||||
|             // The full path is required in the 'url' field.
 |             // The full path is required in the 'url' field.
 | ||||||
|             params[key] = new URL(value, window.location).href; |             params[key] = new URL(val, window.location).href; | ||||||
|             continue; |             continue; | ||||||
|           } catch (ex) { |           } catch (ex) { | ||||||
|             warn(`Cannot create valid URL: "${ex}".`); |             warn(`Cannot create valid URL: "${ex}".`); | ||||||
|           } |           } | ||||||
|         } else if (typeof value === "string" || value instanceof URL) { |         } else if (typeof val === "string" || val instanceof URL) { | ||||||
|           params[key] = value.toString(); // Support Node.js environments.
 |           params[key] = val.toString(); // Support Node.js environments.
 | ||||||
|           continue; |           continue; | ||||||
|         } |         } | ||||||
|         throw new Error( |         throw new Error( | ||||||
| @ -302,10 +300,10 @@ function getDocument(src) { | |||||||
|             "either string or URL-object is expected in the url property." |             "either string or URL-object is expected in the url property." | ||||||
|         ); |         ); | ||||||
|       case "range": |       case "range": | ||||||
|         rangeTransport = value; |         rangeTransport = val; | ||||||
|         continue; |         continue; | ||||||
|       case "worker": |       case "worker": | ||||||
|         worker = value; |         worker = val; | ||||||
|         continue; |         continue; | ||||||
|       case "data": |       case "data": | ||||||
|         // Converting string or array-like data to Uint8Array.
 |         // Converting string or array-like data to Uint8Array.
 | ||||||
| @ -314,21 +312,24 @@ function getDocument(src) { | |||||||
|           PDFJSDev.test("GENERIC") && |           PDFJSDev.test("GENERIC") && | ||||||
|           isNodeJS && |           isNodeJS && | ||||||
|           typeof Buffer !== "undefined" && // eslint-disable-line no-undef
 |           typeof Buffer !== "undefined" && // eslint-disable-line no-undef
 | ||||||
|           value instanceof Buffer // eslint-disable-line no-undef
 |           val instanceof Buffer // eslint-disable-line no-undef
 | ||||||
|         ) { |         ) { | ||||||
|           params[key] = new Uint8Array(value); |           params[key] = new Uint8Array(val); | ||||||
|         } else if (value instanceof Uint8Array) { |  | ||||||
|           break; // Use the data as-is when it's already a Uint8Array.
 |  | ||||||
|         } else if (typeof value === "string") { |  | ||||||
|           params[key] = stringToBytes(value); |  | ||||||
|         } else if ( |         } else if ( | ||||||
|           typeof value === "object" && |           val instanceof Uint8Array && | ||||||
|           value !== null && |           val.byteLength === val.buffer.byteLength | ||||||
|           !isNaN(value.length) |  | ||||||
|         ) { |         ) { | ||||||
|           params[key] = new Uint8Array(value); |           // Use the data as-is when it's already a Uint8Array that completely
 | ||||||
|         } else if (isArrayBuffer(value)) { |           // "utilizes" its underlying ArrayBuffer, to prevent any possible
 | ||||||
|           params[key] = new Uint8Array(value); |           // issues when transferring it to the worker-thread.
 | ||||||
|  |           break; | ||||||
|  |         } else if (typeof val === "string") { | ||||||
|  |           params[key] = stringToBytes(val); | ||||||
|  |         } else if ( | ||||||
|  |           (typeof val === "object" && val !== null && !isNaN(val.length)) || | ||||||
|  |           isArrayBuffer(val) | ||||||
|  |         ) { | ||||||
|  |           params[key] = new Uint8Array(val); | ||||||
|         } else { |         } else { | ||||||
|           throw new Error( |           throw new Error( | ||||||
|             "Invalid PDF binary data: either TypedArray, " + |             "Invalid PDF binary data: either TypedArray, " + | ||||||
| @ -337,7 +338,7 @@ function getDocument(src) { | |||||||
|         } |         } | ||||||
|         continue; |         continue; | ||||||
|     } |     } | ||||||
|     params[key] = value; |     params[key] = val; | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   params.CMapReaderFactory = |   params.CMapReaderFactory = | ||||||
| @ -345,7 +346,6 @@ function getDocument(src) { | |||||||
|   params.StandardFontDataFactory = |   params.StandardFontDataFactory = | ||||||
|     params.StandardFontDataFactory || DefaultStandardFontDataFactory; |     params.StandardFontDataFactory || DefaultStandardFontDataFactory; | ||||||
|   params.ignoreErrors = params.stopAtErrors !== true; |   params.ignoreErrors = params.stopAtErrors !== true; | ||||||
|   params.transferPdfData = params.transferPdfData === true; |  | ||||||
|   params.fontExtraProperties = params.fontExtraProperties === true; |   params.fontExtraProperties = params.fontExtraProperties === true; | ||||||
|   params.pdfBug = params.pdfBug === true; |   params.pdfBug = params.pdfBug === true; | ||||||
|   params.enableXfa = params.enableXfa === true; |   params.enableXfa = params.enableXfa === true; | ||||||
| @ -443,7 +443,6 @@ function getDocument(src) { | |||||||
|             { |             { | ||||||
|               length: params.length, |               length: params.length, | ||||||
|               initialData: params.initialData, |               initialData: params.initialData, | ||||||
|               transferPdfData: params.transferPdfData, |  | ||||||
|               progressiveDone: params.progressiveDone, |               progressiveDone: params.progressiveDone, | ||||||
|               contentDispositionFilename: params.contentDispositionFilename, |               contentDispositionFilename: params.contentDispositionFilename, | ||||||
|               disableRange: params.disableRange, |               disableRange: params.disableRange, | ||||||
| @ -518,8 +517,7 @@ async function _fetchDocument(worker, source, pdfDataRangeTransport, docId) { | |||||||
|     source.contentDispositionFilename = |     source.contentDispositionFilename = | ||||||
|       pdfDataRangeTransport.contentDispositionFilename; |       pdfDataRangeTransport.contentDispositionFilename; | ||||||
|   } |   } | ||||||
|   const transfers = |   const transfers = source.data ? [source.data.buffer] : null; | ||||||
|     source.transferPdfData && source.data ? [source.data.buffer] : null; |  | ||||||
| 
 | 
 | ||||||
|   const workerId = await worker.messageHandler.sendWithPromise( |   const workerId = await worker.messageHandler.sendWithPromise( | ||||||
|     "GetDocRequest", |     "GetDocRequest", | ||||||
| @ -659,6 +657,10 @@ class PDFDocumentLoadingTask { | |||||||
| 
 | 
 | ||||||
| /** | /** | ||||||
|  * Abstract class to support range requests file loading. |  * Abstract class to support range requests file loading. | ||||||
|  |  * | ||||||
|  |  * NOTE: The TypedArrays passed to the constructor and relevant methods below | ||||||
|  |  * will generally be transferred to the worker-thread. This will help reduce | ||||||
|  |  * main-thread memory usage, however it will take ownership of the TypedArrays. | ||||||
|  */ |  */ | ||||||
| class PDFDataRangeTransport { | class PDFDataRangeTransport { | ||||||
|   /** |   /** | ||||||
|  | |||||||
| @ -18,13 +18,10 @@ import { isPdfFile } from "./display_utils.js"; | |||||||
| 
 | 
 | ||||||
| /** @implements {IPDFStream} */ | /** @implements {IPDFStream} */ | ||||||
| class PDFDataTransportStream { | class PDFDataTransportStream { | ||||||
|   #transferPdfData = false; |  | ||||||
| 
 |  | ||||||
|   constructor( |   constructor( | ||||||
|     { |     { | ||||||
|       length, |       length, | ||||||
|       initialData, |       initialData, | ||||||
|       transferPdfData = false, |  | ||||||
|       progressiveDone = false, |       progressiveDone = false, | ||||||
|       contentDispositionFilename = null, |       contentDispositionFilename = null, | ||||||
|       disableRange = false, |       disableRange = false, | ||||||
| @ -38,14 +35,17 @@ class PDFDataTransportStream { | |||||||
|     ); |     ); | ||||||
| 
 | 
 | ||||||
|     this._queuedChunks = []; |     this._queuedChunks = []; | ||||||
|     this.#transferPdfData = transferPdfData; |  | ||||||
|     this._progressiveDone = progressiveDone; |     this._progressiveDone = progressiveDone; | ||||||
|     this._contentDispositionFilename = contentDispositionFilename; |     this._contentDispositionFilename = contentDispositionFilename; | ||||||
| 
 | 
 | ||||||
|     if (initialData?.length > 0) { |     if (initialData?.length > 0) { | ||||||
|       const buffer = this.#transferPdfData |       // Prevent any possible issues by only transferring a Uint8Array that
 | ||||||
|         ? initialData.buffer |       // completely "utilizes" its underlying ArrayBuffer.
 | ||||||
|         : new Uint8Array(initialData).buffer; |       const buffer = | ||||||
|  |         initialData instanceof Uint8Array && | ||||||
|  |         initialData.byteLength === initialData.buffer.byteLength | ||||||
|  |           ? initialData.buffer | ||||||
|  |           : new Uint8Array(initialData).buffer; | ||||||
|       this._queuedChunks.push(buffer); |       this._queuedChunks.push(buffer); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
| @ -77,8 +77,11 @@ class PDFDataTransportStream { | |||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   _onReceiveData({ begin, chunk }) { |   _onReceiveData({ begin, chunk }) { | ||||||
|  |     // Prevent any possible issues by only transferring a Uint8Array that
 | ||||||
|  |     // completely "utilizes" its underlying ArrayBuffer.
 | ||||||
|     const buffer = |     const buffer = | ||||||
|       this.#transferPdfData && chunk?.length >= 0 |       chunk instanceof Uint8Array && | ||||||
|  |       chunk.byteLength === chunk.buffer.byteLength | ||||||
|         ? chunk.buffer |         ? chunk.buffer | ||||||
|         : new Uint8Array(chunk).buffer; |         : new Uint8Array(chunk).buffer; | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -193,44 +193,10 @@ describe("api", function () { | |||||||
|       expect(data[0] instanceof PDFDocumentProxy).toEqual(true); |       expect(data[0] instanceof PDFDocumentProxy).toEqual(true); | ||||||
|       expect(data[1].loaded / data[1].total).toEqual(1); |       expect(data[1].loaded / data[1].total).toEqual(1); | ||||||
| 
 | 
 | ||||||
|       // Check that the TypedArray wasn't transferred.
 |       if (!isNodeJS) { | ||||||
|       expect(typedArrayPdf.length).toEqual(basicApiFileLength); |         // Check that the TypedArray was transferred.
 | ||||||
| 
 |         expect(typedArrayPdf.length).toEqual(0); | ||||||
|       await loadingTask.destroy(); |  | ||||||
|     }); |  | ||||||
| 
 |  | ||||||
|     it("creates pdf doc from TypedArray, with `transferPdfData` set", async function () { |  | ||||||
|       if (isNodeJS) { |  | ||||||
|         pending("Worker is not supported in Node.js."); |  | ||||||
|       } |       } | ||||||
|       const typedArrayPdf = await DefaultFileReaderFactory.fetch({ |  | ||||||
|         path: TEST_PDFS_PATH + basicApiFileName, |  | ||||||
|       }); |  | ||||||
| 
 |  | ||||||
|       // Sanity check to make sure that we fetched the entire PDF file.
 |  | ||||||
|       expect(typedArrayPdf instanceof Uint8Array).toEqual(true); |  | ||||||
|       expect(typedArrayPdf.length).toEqual(basicApiFileLength); |  | ||||||
| 
 |  | ||||||
|       const loadingTask = getDocument({ |  | ||||||
|         data: typedArrayPdf, |  | ||||||
|         transferPdfData: true, |  | ||||||
|       }); |  | ||||||
|       expect(loadingTask instanceof PDFDocumentLoadingTask).toEqual(true); |  | ||||||
| 
 |  | ||||||
|       const progressReportedCapability = createPromiseCapability(); |  | ||||||
|       loadingTask.onProgress = function (data) { |  | ||||||
|         progressReportedCapability.resolve(data); |  | ||||||
|       }; |  | ||||||
| 
 |  | ||||||
|       const data = await Promise.all([ |  | ||||||
|         loadingTask.promise, |  | ||||||
|         progressReportedCapability.promise, |  | ||||||
|       ]); |  | ||||||
|       expect(data[0] instanceof PDFDocumentProxy).toEqual(true); |  | ||||||
|       expect(data[1].loaded / data[1].total).toEqual(1); |  | ||||||
| 
 |  | ||||||
|       // Check that the TypedArray was transferred.
 |  | ||||||
|       expect(typedArrayPdf.length).toEqual(0); |  | ||||||
| 
 | 
 | ||||||
|       await loadingTask.destroy(); |       await loadingTask.destroy(); | ||||||
|     }); |     }); | ||||||
| @ -259,6 +225,11 @@ describe("api", function () { | |||||||
|       expect(data[0] instanceof PDFDocumentProxy).toEqual(true); |       expect(data[0] instanceof PDFDocumentProxy).toEqual(true); | ||||||
|       expect(data[1].loaded / data[1].total).toEqual(1); |       expect(data[1].loaded / data[1].total).toEqual(1); | ||||||
| 
 | 
 | ||||||
|  |       if (!isNodeJS) { | ||||||
|  |         // Check that the ArrayBuffer was transferred.
 | ||||||
|  |         expect(arrayBufferPdf.byteLength).toEqual(0); | ||||||
|  |       } | ||||||
|  | 
 | ||||||
|       await loadingTask.destroy(); |       await loadingTask.destroy(); | ||||||
|     }); |     }); | ||||||
| 
 | 
 | ||||||
| @ -3275,16 +3246,22 @@ Caron Broadcasting, Inc., an Ohio corporation (“Lessee”).`) | |||||||
| 
 | 
 | ||||||
|     it("should fetch document info and page using ranges", async function () { |     it("should fetch document info and page using ranges", async function () { | ||||||
|       const initialDataLength = 4000; |       const initialDataLength = 4000; | ||||||
|  |       const subArrays = []; | ||||||
|       let fetches = 0; |       let fetches = 0; | ||||||
| 
 | 
 | ||||||
|       const data = await dataPromise; |       const data = await dataPromise; | ||||||
|       const initialData = data.subarray(0, initialDataLength); |       const initialData = new Uint8Array(data.subarray(0, initialDataLength)); | ||||||
|  |       subArrays.push(initialData); | ||||||
|  | 
 | ||||||
|       const transport = new PDFDataRangeTransport(data.length, initialData); |       const transport = new PDFDataRangeTransport(data.length, initialData); | ||||||
|       transport.requestDataRange = function (begin, end) { |       transport.requestDataRange = function (begin, end) { | ||||||
|         fetches++; |         fetches++; | ||||||
|         waitSome(function () { |         waitSome(function () { | ||||||
|           transport.onDataProgress(4000); |           const chunk = new Uint8Array(data.subarray(begin, end)); | ||||||
|           transport.onDataRange(begin, data.subarray(begin, end)); |           subArrays.push(chunk); | ||||||
|  | 
 | ||||||
|  |           transport.onDataProgress(initialDataLength); | ||||||
|  |           transport.onDataRange(begin, chunk); | ||||||
|         }); |         }); | ||||||
|       }; |       }; | ||||||
| 
 | 
 | ||||||
| @ -3296,65 +3273,40 @@ Caron Broadcasting, Inc., an Ohio corporation (“Lessee”).`) | |||||||
|       expect(pdfPage.rotate).toEqual(0); |       expect(pdfPage.rotate).toEqual(0); | ||||||
|       expect(fetches).toBeGreaterThan(2); |       expect(fetches).toBeGreaterThan(2); | ||||||
| 
 | 
 | ||||||
|       // Check that the TypedArray wasn't transferred.
 |       if (!isNodeJS) { | ||||||
|       expect(initialData.length).toEqual(initialDataLength); |         // Check that the TypedArrays were transferred.
 | ||||||
| 
 |         for (const array of subArrays) { | ||||||
|       await loadingTask.destroy(); |           expect(array.length).toEqual(0); | ||||||
|     }); |         } | ||||||
| 
 |  | ||||||
|     it("should fetch document info and page using ranges, with `transferPdfData` set", async function () { |  | ||||||
|       if (isNodeJS) { |  | ||||||
|         pending("Worker is not supported in Node.js."); |  | ||||||
|       } |       } | ||||||
|       const initialDataLength = 4000; |  | ||||||
|       let fetches = 0; |  | ||||||
| 
 |  | ||||||
|       const data = await dataPromise; |  | ||||||
|       const initialData = new Uint8Array(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, |  | ||||||
|             new Uint8Array(data.subarray(begin, end)) |  | ||||||
|           ); |  | ||||||
|         }); |  | ||||||
|       }; |  | ||||||
| 
 |  | ||||||
|       const loadingTask = getDocument({ |  | ||||||
|         range: transport, |  | ||||||
|         transferPdfData: true, |  | ||||||
|       }); |  | ||||||
|       const pdfDocument = await loadingTask.promise; |  | ||||||
|       expect(pdfDocument.numPages).toEqual(14); |  | ||||||
| 
 |  | ||||||
|       const pdfPage = await pdfDocument.getPage(10); |  | ||||||
|       expect(pdfPage.rotate).toEqual(0); |  | ||||||
|       expect(fetches).toBeGreaterThan(2); |  | ||||||
| 
 |  | ||||||
|       // Check that the TypedArray was transferred.
 |  | ||||||
|       expect(initialData.length).toEqual(0); |  | ||||||
| 
 | 
 | ||||||
|       await loadingTask.destroy(); |       await loadingTask.destroy(); | ||||||
|     }); |     }); | ||||||
| 
 | 
 | ||||||
|     it("should fetch document info and page using range and streaming", async function () { |     it("should fetch document info and page using range and streaming", async function () { | ||||||
|       const initialDataLength = 4000; |       const initialDataLength = 4000; | ||||||
|  |       const subArrays = []; | ||||||
|       let fetches = 0; |       let fetches = 0; | ||||||
| 
 | 
 | ||||||
|       const data = await dataPromise; |       const data = await dataPromise; | ||||||
|       const initialData = data.subarray(0, initialDataLength); |       const initialData = new Uint8Array(data.subarray(0, initialDataLength)); | ||||||
|  |       subArrays.push(initialData); | ||||||
|  | 
 | ||||||
|       const transport = new PDFDataRangeTransport(data.length, initialData); |       const transport = new PDFDataRangeTransport(data.length, initialData); | ||||||
|       transport.requestDataRange = function (begin, end) { |       transport.requestDataRange = function (begin, end) { | ||||||
|         fetches++; |         fetches++; | ||||||
|         if (fetches === 1) { |         if (fetches === 1) { | ||||||
|  |           const chunk = new Uint8Array(data.subarray(initialDataLength)); | ||||||
|  |           subArrays.push(chunk); | ||||||
|  | 
 | ||||||
|           // Send rest of the data on first range request.
 |           // Send rest of the data on first range request.
 | ||||||
|           transport.onDataProgressiveRead(data.subarray(initialDataLength)); |           transport.onDataProgressiveRead(chunk); | ||||||
|         } |         } | ||||||
|         waitSome(function () { |         waitSome(function () { | ||||||
|           transport.onDataRange(begin, data.subarray(begin, end)); |           const chunk = new Uint8Array(data.subarray(begin, end)); | ||||||
|  |           subArrays.push(chunk); | ||||||
|  | 
 | ||||||
|  |           transport.onDataRange(begin, chunk); | ||||||
|         }); |         }); | ||||||
|       }; |       }; | ||||||
| 
 | 
 | ||||||
| @ -3369,6 +3321,14 @@ Caron Broadcasting, Inc., an Ohio corporation (“Lessee”).`) | |||||||
|       await new Promise(resolve => { |       await new Promise(resolve => { | ||||||
|         waitSome(resolve); |         waitSome(resolve); | ||||||
|       }); |       }); | ||||||
|  | 
 | ||||||
|  |       if (!isNodeJS) { | ||||||
|  |         // Check that the TypedArrays were transferred.
 | ||||||
|  |         for (const array of subArrays) { | ||||||
|  |           expect(array.length).toEqual(0); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  | 
 | ||||||
|       await loadingTask.destroy(); |       await loadingTask.destroy(); | ||||||
|     }); |     }); | ||||||
| 
 | 
 | ||||||
| @ -3376,12 +3336,16 @@ Caron Broadcasting, Inc., an Ohio corporation (“Lessee”).`) | |||||||
|       "should fetch document info and page, without range, " + |       "should fetch document info and page, without range, " + | ||||||
|         "using complete initialData", |         "using complete initialData", | ||||||
|       async function () { |       async function () { | ||||||
|  |         const subArrays = []; | ||||||
|         let fetches = 0; |         let fetches = 0; | ||||||
| 
 | 
 | ||||||
|         const data = await dataPromise; |         const data = await dataPromise; | ||||||
|  |         const initialData = new Uint8Array(data); | ||||||
|  |         subArrays.push(initialData); | ||||||
|  | 
 | ||||||
|         const transport = new PDFDataRangeTransport( |         const transport = new PDFDataRangeTransport( | ||||||
|           data.length, |           data.length, | ||||||
|           data, |           initialData, | ||||||
|           /* progressiveDone = */ true |           /* progressiveDone = */ true | ||||||
|         ); |         ); | ||||||
|         transport.requestDataRange = function (begin, end) { |         transport.requestDataRange = function (begin, end) { | ||||||
| @ -3399,6 +3363,13 @@ Caron Broadcasting, Inc., an Ohio corporation (“Lessee”).`) | |||||||
|         expect(pdfPage.rotate).toEqual(0); |         expect(pdfPage.rotate).toEqual(0); | ||||||
|         expect(fetches).toEqual(0); |         expect(fetches).toEqual(0); | ||||||
| 
 | 
 | ||||||
|  |         if (!isNodeJS) { | ||||||
|  |           // Check that the TypedArrays were transferred.
 | ||||||
|  |           for (const array of subArrays) { | ||||||
|  |             expect(array.length).toEqual(0); | ||||||
|  |           } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|         await loadingTask.destroy(); |         await loadingTask.destroy(); | ||||||
|       } |       } | ||||||
|     ); |     ); | ||||||
|  | |||||||
| @ -270,11 +270,6 @@ const defaultOptions = { | |||||||
|         : "../web/standard_fonts/", |         : "../web/standard_fonts/", | ||||||
|     kind: OptionKind.API, |     kind: OptionKind.API, | ||||||
|   }, |   }, | ||||||
|   transferPdfData: { |  | ||||||
|     /** @type {boolean} */ |  | ||||||
|     value: typeof PDFJSDev !== "undefined" && PDFJSDev.test("MOZCENTRAL"), |  | ||||||
|     kind: OptionKind.API, |  | ||||||
|   }, |  | ||||||
|   verbosity: { |   verbosity: { | ||||||
|     /** @type {number} */ |     /** @type {number} */ | ||||||
|     value: 1, |     value: 1, | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user