From 8602df4957c16f3316aff712785cfa321b27568c Mon Sep 17 00:00:00 2001 From: Jonas Jenwald Date: Wed, 12 Jun 2024 11:19:51 +0200 Subject: [PATCH] [api-minor] Load Node.js packages/polyfills with `process.getBuiltinModule` --- .github/workflows/ci.yml | 1 + .github/workflows/font_tests.yml | 1 + .github/workflows/lint.yml | 1 + .github/workflows/types_tests.yml | 1 + src/display/api.js | 9 --- src/display/node_stream.js | 16 ++-- src/display/node_utils.js | 124 +++++++++++------------------- src/display/stubs.js | 2 - test/types/tsconfig.json | 2 +- test/unit/clitests_helper.js | 4 - test/unit/node_stream_spec.js | 5 +- test/unit/test_utils.js | 10 +-- 12 files changed, 62 insertions(+), 114 deletions(-) diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index 5206c17f33782f..af41ebc961c9f8 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -23,6 +23,7 @@ jobs: uses: actions/setup-node@v4 with: node-version: ${{ matrix.node-version }} + check-latest: true - name: Install dependencies run: npm ci diff --git a/.github/workflows/font_tests.yml b/.github/workflows/font_tests.yml index ff8353b371ae88..a742e385ffb282 100644 --- a/.github/workflows/font_tests.yml +++ b/.github/workflows/font_tests.yml @@ -44,6 +44,7 @@ jobs: uses: actions/setup-node@v4 with: node-version: ${{ matrix.node-version }} + check-latest: true - name: Install dependencies run: npm ci diff --git a/.github/workflows/lint.yml b/.github/workflows/lint.yml index 99b224c6ab9d3d..a11439b8557b59 100644 --- a/.github/workflows/lint.yml +++ b/.github/workflows/lint.yml @@ -23,6 +23,7 @@ jobs: uses: actions/setup-node@v4 with: node-version: ${{ matrix.node-version }} + check-latest: true - name: Install dependencies run: npm ci diff --git a/.github/workflows/types_tests.yml b/.github/workflows/types_tests.yml index f0b96a3be1a1f7..c0f68c86e0213b 100644 --- a/.github/workflows/types_tests.yml +++ b/.github/workflows/types_tests.yml @@ -23,6 +23,7 @@ jobs: uses: actions/setup-node@v4 with: node-version: ${{ matrix.node-version }} + check-latest: true - name: Install dependencies run: npm ci diff --git a/src/display/api.js b/src/display/api.js index 571f1e677f02f1..aca57ed83e9486 100644 --- a/src/display/api.js +++ b/src/display/api.js @@ -58,7 +58,6 @@ import { NodeCanvasFactory, NodeCMapReaderFactory, NodeFilterFactory, - NodePackages, NodeStandardFontDataFactory, } from "display-node_utils"; import { CanvasGraphics } from "./canvas.js"; @@ -2086,14 +2085,6 @@ class PDFWorker { * @type {Promise} */ get promise() { - if ( - typeof PDFJSDev !== "undefined" && - PDFJSDev.test("GENERIC") && - isNodeJS - ) { - // Ensure that all Node.js packages/polyfills have loaded. - return Promise.all([NodePackages.promise, this._readyCapability.promise]); - } return this._readyCapability.promise; } diff --git a/src/display/node_stream.js b/src/display/node_stream.js index 3ffabffe0df139..990f873c519374 100644 --- a/src/display/node_stream.js +++ b/src/display/node_stream.js @@ -12,13 +12,13 @@ * See the License for the specific language governing permissions and * limitations under the License. */ +/* globals process */ import { AbortException, assert, MissingPDFException } from "../shared/util.js"; import { extractFilenameFromHeader, validateRangeRequestCapabilities, } from "./network_utils.js"; -import { NodePackages } from "./node_utils.js"; if (typeof PDFJSDev !== "undefined" && PDFJSDev.test("MOZCENTRAL")) { throw new Error( @@ -29,7 +29,7 @@ if (typeof PDFJSDev !== "undefined" && PDFJSDev.test("MOZCENTRAL")) { const fileUriRegex = /^file:\/\/\/[a-zA-Z]:\//; function parseUrl(sourceUrl) { - const url = NodePackages.get("url"); + const url = process.getBuiltinModule("url"); const parsedUrl = url.parse(sourceUrl); if (parsedUrl.protocol === "file:" || parsedUrl.host) { return parsedUrl; @@ -335,13 +335,13 @@ class PDFNodeStreamFullReader extends BaseFullReader { this._request = null; if (this._url.protocol === "http:") { - const http = NodePackages.get("http"); + const http = process.getBuiltinModule("http"); this._request = http.request( createRequestOptions(this._url, stream.httpHeaders), handleResponse ); } else { - const https = NodePackages.get("https"); + const https = process.getBuiltinModule("https"); this._request = https.request( createRequestOptions(this._url, stream.httpHeaders), handleResponse @@ -384,13 +384,13 @@ class PDFNodeStreamRangeReader extends BaseRangeReader { this._request = null; if (this._url.protocol === "http:") { - const http = NodePackages.get("http"); + const http = process.getBuiltinModule("http"); this._request = http.request( createRequestOptions(this._url, this._httpHeaders), handleResponse ); } else { - const https = NodePackages.get("https"); + const https = process.getBuiltinModule("https"); this._request = https.request( createRequestOptions(this._url, this._httpHeaders), handleResponse @@ -415,7 +415,7 @@ class PDFNodeStreamFsFullReader extends BaseFullReader { path = path.replace(/^\//, ""); } - const fs = NodePackages.get("fs"); + const fs = process.getBuiltinModule("fs"); fs.promises.lstat(path).then( stat => { // Setting right content length. @@ -446,7 +446,7 @@ class PDFNodeStreamFsRangeReader extends BaseRangeReader { path = path.replace(/^\//, ""); } - const fs = NodePackages.get("fs"); + const fs = process.getBuiltinModule("fs"); this._setReadableStream(fs.createReadStream(path, { start, end: end - 1 })); } } diff --git a/src/display/node_utils.js b/src/display/node_utils.js index aca4ad750fad7d..854b71dfc60dfe 100644 --- a/src/display/node_utils.js +++ b/src/display/node_utils.js @@ -12,6 +12,7 @@ * See the License for the specific language governing permissions and * limitations under the License. */ +/* globals process */ import { BaseCanvasFactory, @@ -27,90 +28,51 @@ if (typeof PDFJSDev !== "undefined" && PDFJSDev.test("MOZCENTRAL")) { ); } -if (isNodeJS) { - // eslint-disable-next-line no-var - var packageCapability = Promise.withResolvers(); - // eslint-disable-next-line no-var - var packageMap = null; - - const loadPackages = async () => { - // Native packages. - const fs = await __non_webpack_import__("fs"), - http = await __non_webpack_import__("http"), - https = await __non_webpack_import__("https"), - url = await __non_webpack_import__("url"); - - // Optional, third-party, packages. - let canvas, path2d; - if (typeof PDFJSDev !== "undefined" && !PDFJSDev.test("SKIP_BABEL")) { - try { - canvas = await __non_webpack_import__("canvas"); - } catch {} - try { - path2d = await __non_webpack_import__("path2d"); - } catch {} - } - - return new Map(Object.entries({ fs, http, https, url, canvas, path2d })); - }; - - loadPackages().then( - map => { - packageMap = map; - packageCapability.resolve(); - - if (typeof PDFJSDev === "undefined" || PDFJSDev.test("SKIP_BABEL")) { - return; - } - if (!globalThis.DOMMatrix) { - const DOMMatrix = map.get("canvas")?.DOMMatrix; - - if (DOMMatrix) { - globalThis.DOMMatrix = DOMMatrix; - } else { - warn("Cannot polyfill `DOMMatrix`, rendering may be broken."); - } - } - if (!globalThis.Path2D) { - const CanvasRenderingContext2D = - map.get("canvas")?.CanvasRenderingContext2D; - const applyPath2DToCanvasRenderingContext = - map.get("path2d")?.applyPath2DToCanvasRenderingContext; - const Path2D = map.get("path2d")?.Path2D; - - if ( - CanvasRenderingContext2D && - applyPath2DToCanvasRenderingContext && - Path2D - ) { - applyPath2DToCanvasRenderingContext(CanvasRenderingContext2D); - globalThis.Path2D = Path2D; - } else { - warn("Cannot polyfill `Path2D`, rendering may be broken."); - } - } - }, - reason => { - warn(`loadPackages: ${reason}`); - - packageMap = new Map(); - packageCapability.resolve(); +if ( + typeof PDFJSDev !== "undefined" && + !PDFJSDev.test("SKIP_BABEL") && + isNodeJS +) { + let canvas, path2d; + try { + const require = process + .getBuiltinModule("module") + .createRequire(import.meta.url); + + canvas = require("canvas"); + path2d = require("path2d"); + } catch {} + + if (!globalThis.DOMMatrix) { + const DOMMatrix = canvas?.DOMMatrix; + + if (DOMMatrix) { + globalThis.DOMMatrix = DOMMatrix; + } else { + warn("Cannot polyfill `DOMMatrix`, rendering may be broken."); } - ); -} - -class NodePackages { - static get promise() { - return packageCapability.promise; } - - static get(name) { - return packageMap?.get(name); + if (!globalThis.Path2D) { + const CanvasRenderingContext2D = canvas?.CanvasRenderingContext2D; + const applyPath2DToCanvasRenderingContext = + path2d?.applyPath2DToCanvasRenderingContext; + const Path2D = path2d?.Path2D; + + if ( + CanvasRenderingContext2D && + applyPath2DToCanvasRenderingContext && + Path2D + ) { + applyPath2DToCanvasRenderingContext(CanvasRenderingContext2D); + globalThis.Path2D = Path2D; + } else { + warn("Cannot polyfill `Path2D`, rendering may be broken."); + } } } const fetchData = function (url) { - const fs = NodePackages.get("fs"); + const fs = process.getBuiltinModule("fs"); return fs.promises.readFile(url).then(data => new Uint8Array(data)); }; @@ -121,7 +83,10 @@ class NodeCanvasFactory extends BaseCanvasFactory { * @ignore */ _createCanvas(width, height) { - const canvas = NodePackages.get("canvas"); + const require = process + .getBuiltinModule("module") + .createRequire(import.meta.url); + const canvas = require("canvas"); return canvas.createCanvas(width, height); } } @@ -148,6 +113,5 @@ export { NodeCanvasFactory, NodeCMapReaderFactory, NodeFilterFactory, - NodePackages, NodeStandardFontDataFactory, }; diff --git a/src/display/stubs.js b/src/display/stubs.js index 0b3b9a0bbb788f..271f0d70015bcc 100644 --- a/src/display/stubs.js +++ b/src/display/stubs.js @@ -16,7 +16,6 @@ const NodeCanvasFactory = null; const NodeCMapReaderFactory = null; const NodeFilterFactory = null; -const NodePackages = null; const NodeStandardFontDataFactory = null; const PDFFetchStream = null; const PDFNetworkStream = null; @@ -26,7 +25,6 @@ export { NodeCanvasFactory, NodeCMapReaderFactory, NodeFilterFactory, - NodePackages, NodeStandardFontDataFactory, PDFFetchStream, PDFNetworkStream, diff --git a/test/types/tsconfig.json b/test/types/tsconfig.json index 5cd8ef4507e206..772a382c71e4d4 100644 --- a/test/types/tsconfig.json +++ b/test/types/tsconfig.json @@ -10,7 +10,7 @@ "module": "ESNext", "baseUrl": "./", "strict": true, - "types": [], + "types": ["node"], "lib": ["ESNext", "DOM"], "paths": { "pdfjs-dist": ["../../build/typestest"], diff --git a/test/unit/clitests_helper.js b/test/unit/clitests_helper.js index a5a1c788e2de04..d86eeacbbc2d3f 100644 --- a/test/unit/clitests_helper.js +++ b/test/unit/clitests_helper.js @@ -18,7 +18,6 @@ import { setVerbosityLevel, VerbosityLevel, } from "../../src/shared/util.js"; -import { NodePackages } from "../../src/display/node_utils.js"; // Sets longer timeout, similar to `jasmine-boot.js`. jasmine.DEFAULT_TIMEOUT_INTERVAL = 30000; @@ -30,9 +29,6 @@ if (!isNodeJS) { ); } -// Ensure that all Node.js packages/polyfills have loaded. -await NodePackages.promise; - // Reduce the amount of console "spam", by ignoring `info`/`warn` calls, // when running the unit-tests in Node.js/Travis. setVerbosityLevel(VerbosityLevel.ERRORS); diff --git a/test/unit/node_stream_spec.js b/test/unit/node_stream_spec.js index f3a2daa754545f..226523db1d2a43 100644 --- a/test/unit/node_stream_spec.js +++ b/test/unit/node_stream_spec.js @@ -24,12 +24,11 @@ if (!isNodeJS) { ); } -const path = await __non_webpack_import__("path"); -const url = await __non_webpack_import__("url"); - describe("node_stream", function () { let tempServer = null; + const path = process.getBuiltinModule("path"), + url = process.getBuiltinModule("url"); const pdf = url.parse( encodeURI( "file://" + path.join(process.cwd(), "./test/pdfs/tracemonkey.pdf") diff --git a/test/unit/test_utils.js b/test/unit/test_utils.js index bc1d839d5eadde..a7c3224830b623 100644 --- a/test/unit/test_utils.js +++ b/test/unit/test_utils.js @@ -18,13 +18,6 @@ import { NullStream, StringStream } from "../../src/core/stream.js"; import { Page, PDFDocument } from "../../src/core/document.js"; import { Ref } from "../../src/core/primitives.js"; -let fs, http; -if (isNodeJS) { - // Native packages. - fs = await __non_webpack_import__("fs"); - http = await __non_webpack_import__("http"); -} - const TEST_PDFS_PATH = isNodeJS ? "./test/pdfs/" : "../pdfs/"; const CMAP_URL = isNodeJS ? "./external/bcmaps/" : "../../external/bcmaps/"; @@ -46,6 +39,7 @@ class DOMFileReaderFactory { class NodeFileReaderFactory { static async fetch(params) { return new Promise((resolve, reject) => { + const fs = process.getBuiltinModule("fs"); fs.readFile(params.path, (error, data) => { if (error || !data) { reject(error || new Error(`Empty file for: ${params.path}`)); @@ -148,6 +142,8 @@ function createIdFactory(pageIndex) { function createTemporaryNodeServer() { assert(isNodeJS, "Should only be used in Node.js environments."); + const fs = process.getBuiltinModule("fs"), + http = process.getBuiltinModule("http"); // Create http server to serve pdf data for tests. const server = http .createServer((request, response) => {