Reduce ICU book build size
This commit is contained in:
parent
ca141a26e2
commit
b22aaaec7e
4 changed files with 52 additions and 17 deletions
4
.github/workflows/icu-book.yml
vendored
4
.github/workflows/icu-book.yml
vendored
|
|
@ -49,7 +49,7 @@ jobs:
|
||||||
- run: ./node_modules/.bin/playwright install chromium
|
- run: ./node_modules/.bin/playwright install chromium
|
||||||
- run: ./node_modules/.bin/run-p --race test:storybook:serve test:storybook:test
|
- run: ./node_modules/.bin/run-p --race test:storybook:serve test:storybook:test
|
||||||
env:
|
env:
|
||||||
ARTIFACTS_DIR: stories/data
|
ARTIFACTS_DIR: stories
|
||||||
- run: pnpm run build:esbuild
|
- run: pnpm run build:esbuild
|
||||||
- run: node ts/scripts/compile-stories-icu-lookup.js stories
|
- run: node ts/scripts/compile-stories-icu-lookup.js stories
|
||||||
|
|
||||||
|
|
@ -59,6 +59,7 @@ jobs:
|
||||||
with:
|
with:
|
||||||
name: desktop-test-icu
|
name: desktop-test-icu
|
||||||
path: stories
|
path: stories
|
||||||
|
compression-level: 9
|
||||||
|
|
||||||
- name: Upload release artifacts
|
- name: Upload release artifacts
|
||||||
if: github.event_name != 'workflow_dispatch'
|
if: github.event_name != 'workflow_dispatch'
|
||||||
|
|
@ -66,3 +67,4 @@ jobs:
|
||||||
with:
|
with:
|
||||||
name: desktop-${{ github.ref_name }}-icu
|
name: desktop-${{ github.ref_name }}-icu
|
||||||
path: stories
|
path: stories
|
||||||
|
compression-level: 9
|
||||||
|
|
|
||||||
|
|
@ -47,7 +47,7 @@
|
||||||
summary.textContent = `${key}: "${message}"`;
|
summary.textContent = `${key}: "${message}"`;
|
||||||
details.appendChild(summary);
|
details.appendChild(summary);
|
||||||
|
|
||||||
for (const storyId of stories) {
|
for (const [storyId, image] of stories) {
|
||||||
const story = document.createElement('details');
|
const story = document.createElement('details');
|
||||||
details.appendChild(story);
|
details.appendChild(story);
|
||||||
|
|
||||||
|
|
@ -56,8 +56,7 @@
|
||||||
story.appendChild(title);
|
story.appendChild(title);
|
||||||
|
|
||||||
const img = document.createElement('img');
|
const img = document.createElement('img');
|
||||||
img.src = `data/${encodeURIComponent(storyId)}/` +
|
img.src = `images/${image}`;
|
||||||
`${encodeURIComponent(key.replace(/^icu:/, ''))}.jpg`;
|
|
||||||
img.loading = 'lazy';
|
img.loading = 'lazy';
|
||||||
story.appendChild(img);
|
story.appendChild(img);
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -1,8 +1,9 @@
|
||||||
// Copyright 2025 Signal Messenger, LLC
|
// Copyright 2025 Signal Messenger, LLC
|
||||||
// SPDX-License-Identifier: AGPL-3.0-only
|
// SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
|
||||||
import { mkdir, writeFile } from 'node:fs/promises';
|
import { mkdir, writeFile, symlink } from 'node:fs/promises';
|
||||||
import { join } from 'node:path';
|
import { join } from 'node:path';
|
||||||
|
import { createHash } from 'node:crypto';
|
||||||
import {
|
import {
|
||||||
type TestRunnerConfig,
|
type TestRunnerConfig,
|
||||||
waitForPageReady,
|
waitForPageReady,
|
||||||
|
|
@ -40,9 +41,13 @@ const config: TestRunnerConfig = {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
const dir = join(ARTIFACTS_DIR, context.id);
|
const storeDir = join(ARTIFACTS_DIR, 'images');
|
||||||
await mkdir(dir, { recursive: true });
|
await mkdir(storeDir, { recursive: true });
|
||||||
|
|
||||||
|
const componentDir = join(ARTIFACTS_DIR, 'components', context.id);
|
||||||
|
await mkdir(componentDir, { recursive: true });
|
||||||
|
|
||||||
|
const saves = new Array<Promise<void>>();
|
||||||
for (const [key, value] of result) {
|
for (const [key, value] of result) {
|
||||||
const locator = page
|
const locator = page
|
||||||
.getByText(value)
|
.getByText(value)
|
||||||
|
|
@ -76,8 +81,28 @@ const config: TestRunnerConfig = {
|
||||||
quality: 95,
|
quality: 95,
|
||||||
});
|
});
|
||||||
|
|
||||||
await writeFile(join(dir, `${key.replace(/^icu:/, '')}.jpg`), image);
|
const digest = createHash('sha256').update(image).digest('hex');
|
||||||
|
const storeFile = join(storeDir, `${digest}.jpg`);
|
||||||
|
const targetFile = join(componentDir, `${key.replace(/^icu:/, '')}.jpg`);
|
||||||
|
|
||||||
|
saves.push(
|
||||||
|
(async () => {
|
||||||
|
try {
|
||||||
|
await writeFile(storeFile, image, {
|
||||||
|
// Fail if exists
|
||||||
|
flags: 'wx',
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
if (error.code !== 'EEXIST') {
|
||||||
|
throw error;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
await symlink(storeFile, targetFile);
|
||||||
|
})()
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
await Promise.all(saves);
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
export default config;
|
export default config;
|
||||||
|
|
|
||||||
|
|
@ -1,10 +1,9 @@
|
||||||
// Copyright 2025 Signal Messenger, LLC
|
// Copyright 2025 Signal Messenger, LLC
|
||||||
// SPDX-License-Identifier: AGPL-3.0-only
|
// SPDX-License-Identifier: AGPL-3.0-only
|
||||||
|
|
||||||
import { readFile, writeFile } from 'node:fs/promises';
|
import { readFile, writeFile, readdir, readlink } from 'node:fs/promises';
|
||||||
import { join, basename } from 'node:path';
|
import { join, basename } from 'node:path';
|
||||||
import pMap from 'p-map';
|
import pMap from 'p-map';
|
||||||
import fastGlob from 'fast-glob';
|
|
||||||
|
|
||||||
import { drop } from '../util/drop';
|
import { drop } from '../util/drop';
|
||||||
|
|
||||||
|
|
@ -14,9 +13,9 @@ async function main(): Promise<void> {
|
||||||
throw new Error('Missing required source directory argument');
|
throw new Error('Missing required source directory argument');
|
||||||
}
|
}
|
||||||
|
|
||||||
const dirEntries = await fastGlob('*/*.jpg', {
|
const dirEntries = await readdir(join(source, 'components'), {
|
||||||
cwd: join(source, 'data'),
|
withFileTypes: true,
|
||||||
onlyFiles: true,
|
recursive: true,
|
||||||
});
|
});
|
||||||
|
|
||||||
const enMessages = JSON.parse(
|
const enMessages = JSON.parse(
|
||||||
|
|
@ -26,21 +25,31 @@ async function main(): Promise<void> {
|
||||||
)
|
)
|
||||||
);
|
);
|
||||||
|
|
||||||
const icuToStory: Record<string, Array<string>> = Object.create(null);
|
const icuToStory: Record<string, Array<[string, string]>> = Object.create(
|
||||||
|
null
|
||||||
|
);
|
||||||
|
|
||||||
await pMap(
|
await pMap(
|
||||||
dirEntries,
|
dirEntries,
|
||||||
async entry => {
|
async entry => {
|
||||||
const [storyId, imageFile] = entry.split('/', 2);
|
if (!entry.isSymbolicLink()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
const icuId = `icu:${basename(imageFile, '.jpg')}`;
|
const fullPath = join(entry.parentPath, entry.name);
|
||||||
|
const image = basename(await readlink(fullPath));
|
||||||
|
|
||||||
|
const storyId = basename(entry.parentPath);
|
||||||
|
const linkFile = entry.name;
|
||||||
|
|
||||||
|
const icuId = `icu:${basename(linkFile, '.jpg')}`;
|
||||||
|
|
||||||
let list = icuToStory[icuId];
|
let list = icuToStory[icuId];
|
||||||
if (list == null) {
|
if (list == null) {
|
||||||
list = [];
|
list = [];
|
||||||
icuToStory[icuId] = list;
|
icuToStory[icuId] = list;
|
||||||
}
|
}
|
||||||
list.push(storyId);
|
list.push([storyId, image]);
|
||||||
},
|
},
|
||||||
{ concurrency: 20 }
|
{ concurrency: 20 }
|
||||||
);
|
);
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue