Introduce a GitHub Actions workflow for running the font tests

This commit migrates the font tests away from the bots. Not only are the
font tests broken on the Windows bot since some time, they also run on
Python 2 (end of life since January 2020) and `ttx` 3.19.0 (released in
November 2017). The latter is installed via a submodule, which requires
more complicated logic for finding and running `ttx`.

We solve the issues by implementing a modern workflow that installs the
most recent stable Python and `ttx` (`fonttools` package) versions. This
simplifies the `ttx` driver code as well because it can now assume `ttx`
is available on the path (just like we do for e.g. `node` invocations).
GitHub Actions takes care of creating a virtual environment with
`fonttools` in it so that the `ttx`  entrypoint is available. Locally
the font tests can be run in a similar way by creating and sourcing a
virtual environment with `fonttools` in it before running the font
tests, and a README file is included with instructions for doing so.
This commit is contained in:
Tim van der Meij 2023-10-29 19:43:32 +01:00
parent 69452bb60e
commit 8157f39c62
No known key found for this signature in database
GPG Key ID: 8C3FD2925A5F2762
7 changed files with 125 additions and 54 deletions

64
.github/workflows/font_tests.yml vendored Normal file
View File

@ -0,0 +1,64 @@
name: Font tests
on:
push:
paths:
- 'gulpfile.mjs'
- 'src/**'
- 'test/test.mjs'
- 'test/font/**'
- '.github/workflows/font_tests.yml'
branches:
- master
pull_request:
paths:
- 'gulpfile.mjs'
- 'src/**'
- 'test/test.mjs'
- 'test/font/**'
- '.github/workflows/font_tests.yml'
branches:
- master
workflow_dispatch:
permissions:
contents: read
jobs:
test:
name: Test
strategy:
fail-fast: false
matrix:
node-version: [lts/*]
os: [windows-latest, ubuntu-latest]
runs-on: ${{ matrix.os }}
steps:
- name: Checkout repository
uses: actions/checkout@v4
with:
fetch-depth: 0
- name: Use Node.js ${{ matrix.node-version }}
uses: actions/setup-node@v4
with:
node-version: ${{ matrix.node-version }}
- name: Install Gulp
run: npm install -g gulp-cli
- name: Install other dependencies
run: npm install
- name: Use Python 3.12
uses: actions/setup-python@v4
with:
python-version: '3.12'
cache: 'pip'
- name: Install Fonttools
run: pip install fonttools
- name: Run font tests
run: gulp fonttest --headless

3
.gitmodules vendored
View File

@ -1,3 +0,0 @@
[submodule "test/ttx/fonttools-code"]
path = test/ttx/fonttools-code
url = https://github.com/behdad/fonttools.git

View File

@ -1749,7 +1749,6 @@ gulp.task(
return streamqueue( return streamqueue(
{ objectMode: true }, { objectMode: true },
createTestSource("unit", { bot: true }), createTestSource("unit", { bot: true }),
createTestSource("font", { bot: true }),
createTestSource("browser", { bot: true }), createTestSource("browser", { bot: true }),
createTestSource("integration") createTestSource("integration")
); );
@ -1774,7 +1773,6 @@ gulp.task(
return streamqueue( return streamqueue(
{ objectMode: true }, { objectMode: true },
createTestSource("unit", { bot: true }), createTestSource("unit", { bot: true }),
createTestSource("font", { bot: true }),
createTestSource("browser", { bot: true, xfaOnly: true }), createTestSource("browser", { bot: true, xfaOnly: true }),
createTestSource("integration") createTestSource("integration")
); );

36
test/font/README.md Normal file
View File

@ -0,0 +1,36 @@
# Font tests
The font tests check if PDF.js can read font data correctly. For validation
the `ttx` tool (from the Python `fonttools` library) is used that can convert
font data to an XML format that we can easily use for assertions in the tests.
In the font tests we let PDF.js read font data and pass the PDF.js-interpreted
font data through `ttx` to check its correctness. The font tests are successful
if PDF.js can successfully read the font data and `ttx` can successfully read
the PDF.js-interpreted font data back, proving that PDF.js does not apply any
transformations that break the font data.
## Running the font tests
The font tests are run on GitHub Actions using the workflow defined in
`.github/workflows/font_tests.yml`, but it is also possible to run the font
tests locally. The current stable versions of the following dependencies are
required to be installed on the system:
- Python 3
- `fonttools` (see https://pypi.org/project/fonttools and https://github.com/fonttools/fonttools)
The recommended way of installing `fonttools` is using `pip` in a virtual
environment because it avoids having to do a system-wide installation and
therefore improves isolation, but any other way of installing `fonttools`
that makes `ttx` available in the `PATH` environment variable also works.
Using the virtual environment approach the font tests can be run locally by
creating and sourcing a virtual environment with `fonttools` installed in
it before running the font tests:
```
python3 -m venv venv
source venv/bin/activate
pip install fonttools
gulp fonttest
```

View File

@ -14,65 +14,43 @@
* limitations under the License. * limitations under the License.
*/ */
import { fileURLToPath } from "url";
import fs from "fs"; import fs from "fs";
import os from "os";
import path from "path"; import path from "path";
import { spawn } from "child_process"; import { spawn } from "child_process";
const __dirname = path.dirname(fileURLToPath(import.meta.url)); let ttxTaskId = Date.now();
const ttxResourcesHome = path.join(__dirname, "..", "ttx"); function runTtx(fontPath, registerOnCancel, callback) {
const ttx = spawn("ttx", [fontPath], { stdio: "ignore" });
let nextTTXTaskId = Date.now(); let ttxRunError;
registerOnCancel(function (reason) {
function runTtx(ttxResourcesHomePath, fontPath, registerOnCancel, callback) { ttxRunError = reason;
fs.realpath(ttxResourcesHomePath, function (error, realTtxResourcesHomePath) { callback(reason);
const fontToolsHome = path.join(realTtxResourcesHomePath, "fonttools-code"); ttx.kill();
fs.realpath(fontPath, function (errorFontPath, realFontPath) { });
const ttxPath = path.join("Lib", "fontTools", "ttx.py"); ttx.on("error", function (errorTtx) {
if (!fs.existsSync(path.join(fontToolsHome, ttxPath))) { ttxRunError = errorTtx;
callback("TTX was not found, please checkout PDF.js submodules"); callback(
return; "Unable to execute `ttx`; make sure the `fonttools` dependency is installed"
} );
const ttxEnv = { });
PYTHONPATH: path.join(fontToolsHome, "Lib"), ttx.on("close", function (code) {
PYTHONDONTWRITEBYTECODE: true, if (ttxRunError) {
}; return;
const ttxStdioMode = "ignore"; }
const python = process.platform !== "win32" ? "python2" : "python"; callback();
const ttx = spawn(python, [ttxPath, realFontPath], {
cwd: fontToolsHome,
stdio: ttxStdioMode,
env: ttxEnv,
});
let ttxRunError;
registerOnCancel(function (reason) {
ttxRunError = reason;
callback(reason);
ttx.kill();
});
ttx.on("error", function (errorTtx) {
ttxRunError = errorTtx;
callback("Unable to execute ttx");
});
ttx.on("close", function (code) {
if (ttxRunError) {
return;
}
callback();
});
});
}); });
} }
function translateFont(content, registerOnCancel, callback) { function translateFont(content, registerOnCancel, callback) {
const buffer = Buffer.from(content, "base64"); const buffer = Buffer.from(content, "base64");
const taskId = (nextTTXTaskId++).toString(); const taskId = (ttxTaskId++).toString();
const fontPath = path.join(ttxResourcesHome, taskId + ".otf"); const fontPath = path.join(os.tmpdir(), `pdfjs-font-test-${taskId}.otf`);
const resultPath = path.join(ttxResourcesHome, taskId + ".ttx"); const resultPath = path.join(os.tmpdir(), `pdfjs-font-test-${taskId}.ttx`);
fs.writeFileSync(fontPath, buffer); fs.writeFileSync(fontPath, buffer);
runTtx(ttxResourcesHome, fontPath, registerOnCancel, function (err) { runTtx(fontPath, registerOnCancel, function (err) {
fs.unlinkSync(fontPath); fs.unlinkSync(fontPath);
if (err) { if (err) {
console.error(err); console.error(err);

View File

@ -1 +0,0 @@
If `git clone --recursive` was not used, please run `git submodule init; git submodule update` to pull fonttools code.

@ -1 +0,0 @@
Subproject commit d8170131a3458ffbc19089cf33249777bde390e7