Skip to content

Commit 238d5b9

Browse files
committed
ci: add architectures update script
1 parent 0230503 commit 238d5b9

File tree

2 files changed

+193
-0
lines changed

2 files changed

+193
-0
lines changed
+47
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,47 @@
1+
name: update-architectures
2+
3+
on:
4+
# Convert to schedule when done or whatever is preferred
5+
push:
6+
pull_request:
7+
8+
jobs:
9+
update-architectures:
10+
name: update-architectures
11+
runs-on: ubuntu-latest
12+
13+
steps:
14+
- name: Checkout the docker-node repo
15+
uses: actions/checkout@v2
16+
with:
17+
path: docker-node
18+
19+
- name: Checkout the official-images repo
20+
uses: actions/checkout@v2
21+
with:
22+
path: official-images
23+
repository: docker-library/official-images
24+
25+
- name: Download bashbrew
26+
run: |
27+
mkdir -p ${GITHUB_WORKSPACE}/bin
28+
wget --no-verbose -O ${GITHUB_WORKSPACE}/bin/bashbrew https://doi-janky.infosiftr.net/job/bashbrew/job/master/lastSuccessfulBuild/artifact/bashbrew-amd64
29+
sudo chmod +x ${GITHUB_WORKSPACE}/bin/bashbrew
30+
echo "::add-path::${GITHUB_WORKSPACE}/bin"
31+
32+
- name: Update architectures
33+
uses: actions/github-script@v3
34+
id: arch-updater
35+
env:
36+
BASHBREW_LIBRARY: "${{ github.workspace }}/official-images/library"
37+
with:
38+
script: |
39+
const script = require(`${process.env.GITHUB_WORKSPACE}/docker-node/updateArches.js`)
40+
return script();
41+
42+
- name: Open a PR
43+
if: steps.arch-updater.outputs.result == 'true'
44+
# TODO: open a PR
45+
run: |
46+
cd docker-node
47+
git diff --exit-code

updateArches.js

+146
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,146 @@
1+
const { execFileSync } = require('child_process');
2+
const { readFileSync, readdirSync, writeFileSync } = require('fs');
3+
const path = require('path');
4+
5+
const nodeDirRegex = /^\d+$/;
6+
7+
// Given a name and a tag, this returns an array of architectures that it supports
8+
const fetchImageArches = (repoTag) => execFileSync('bashbrew', [
9+
'cat', repoTag,
10+
], { encoding: 'utf8' }).split('\n')
11+
.find((line) => line.startsWith('Architectures:'))
12+
.split(':')[1]
13+
.trim()
14+
.split(/\s*,\s*/);
15+
16+
// Parses an "architectures" file into an object like:
17+
// {
18+
// arch1: ['variant1', 'variant2'],
19+
// //...
20+
// }
21+
const parseArchitecturesFile = (file) => Object.fromEntries(
22+
[...readFileSync(file, 'utf8').matchAll(/^(?<arch>\S+)\s+(?<variants>\S+)$/mg)]
23+
.slice(1)
24+
.map(({ groups: { arch, variants } }) => [arch, variants.split(',')]),
25+
);
26+
27+
// Takes in an object like:
28+
// {
29+
// arch1: ['variant1', 'variant2'],
30+
// // ...
31+
// }
32+
// and returns an object like
33+
// {
34+
// variant1: ['arch1', 'arch2'],
35+
// // ...
36+
// }
37+
const invertObject = (obj) => Object.entries(obj)
38+
.reduce((acc, [key, vals]) => vals.reduce((valAcc, val) => {
39+
const { [val]: keys, ...rest } = valAcc;
40+
return {
41+
...rest,
42+
[val]: keys
43+
? [...keys, key]
44+
: [key],
45+
};
46+
}, acc), {});
47+
48+
// Returns a list of the child directories in the given path
49+
const getChildDirectories = (parent) => readdirSync(parent, { withFileTypes: true })
50+
.filter((dirent) => dirent.isDirectory())
51+
.map(({ name }) => path.resolve(parent, name));
52+
53+
const getNodeVerionDirs = (base) => getChildDirectories(base)
54+
.filter((childPath) => nodeDirRegex.test(path.basename(childPath)));
55+
56+
// Assume no duplicates
57+
const areArraysEquilivant = (arches1, arches2) => arches1.length === arches2.length
58+
&& arches1.every((arch) => arches2.includes(arch));
59+
60+
// Returns the paths of Dockerfiles that are at: base/*/Dockerfile
61+
const getDockerfilesInChildDirs = (base) => getChildDirectories(base)
62+
.map((childDir) => path.resolve(childDir, 'Dockerfile'));
63+
64+
// Given a path to a Dockerfile like .../14/variant/Dockerfile, this will return "variant"
65+
const getVariantFromPath = (file) => path.dirname(file).split(path.sep).slice(-1);
66+
67+
const getBaseImageFromDockerfile = (file) => readFileSync(file, 'utf8')
68+
.match(/^FROM (\S+)/m)[1];
69+
70+
// Given a dockerfile, this function returns an array like [variant, [arch1, arch2, ...]]
71+
const getVariantAndArches = (dockerfile) => {
72+
const variant = getVariantFromPath(dockerfile);
73+
const baseImage = getBaseImageFromDockerfile(dockerfile);
74+
const arches = fetchImageArches(baseImage);
75+
76+
// TODO: filter by arches node supports
77+
return [variant, arches];
78+
};
79+
80+
const getStoredVariantArches = (file) => {
81+
const storedArchVariants = parseArchitecturesFile(file);
82+
return invertObject(storedArchVariants);
83+
};
84+
85+
const areVariantArchesEquilivant = (current, stored) => Object.keys(current).length
86+
=== Object.keys(stored).length
87+
&& Object.entries(current).every(
88+
([variant, arches]) => stored[variant] && areArraysEquilivant(arches, stored[variant]),
89+
);
90+
91+
const formatEntry = ([arch, variants], variantOffset) => `${arch}${' '.repeat(variantOffset - arch.length)}${variants.join(',')}`;
92+
93+
const sortObjectKeys = (obj) => Object.keys(obj)
94+
.sort()
95+
.reduce((acc, key) => ({
96+
...acc,
97+
[key]: obj[key]
98+
}), {});
99+
100+
const storeArchitectures = (variantArches, architecturesFile) => {
101+
const archVariants = sortObjectKeys(invertObject(variantArches));
102+
const data = {
103+
'bashbrew-arch': ['variants'],
104+
...archVariants,
105+
};
106+
107+
const maxKeyLength = Math.max(...Object.keys(data).map((key) => key.length));
108+
// Variants start 2 spaces after the longest key
109+
const variantOffset = maxKeyLength + 2;
110+
111+
const str = Object.entries(data)
112+
.map((entry) => formatEntry(entry, variantOffset))
113+
.join('\n') + '\n';
114+
115+
writeFileSync(architecturesFile, str);
116+
117+
// Just here for debugging purposes
118+
console.log(str);
119+
console.log('\n\n');
120+
};
121+
122+
const updateNodeDirArches = (nodeDir) => {
123+
const dockerfiles = getDockerfilesInChildDirs(nodeDir);
124+
125+
const currentVariantArches = Object.fromEntries(dockerfiles.map(getVariantAndArches));
126+
const architecturesFile = path.resolve(nodeDir, 'architectures');
127+
const storedVariantArches = getStoredVariantArches(architecturesFile);
128+
129+
if (areVariantArchesEquilivant(currentVariantArches, storedVariantArches)) {
130+
console.log('Architectures up-to-date: ', nodeDir);
131+
return false;
132+
}
133+
134+
console.log('Architectures outdated: ', nodeDir);
135+
storeArchitectures(currentVariantArches, architecturesFile);
136+
137+
return true;
138+
};
139+
140+
const updateArchitectures = () => {
141+
const nodeDirs = getNodeVerionDirs(__dirname);
142+
const dirsUpdated = nodeDirs.map(updateNodeDirArches);
143+
return dirsUpdated.some((updated) => updated);
144+
};
145+
146+
module.exports = updateArchitectures;

0 commit comments

Comments
 (0)