mirror of
https://github.com/docker/build-push-action.git
synced 2025-10-24 02:52:18 +00:00
*: refactor methods to support mocking
Additionally, write some tests to ensure the driver method `startBlacksmithBuilder` handles all exceptions correctly in both nofallback=true and nofallback=false configurations.
This commit is contained in:
parent
15e5beff2d
commit
c71ad2dbef
15 changed files with 712 additions and 1380 deletions
|
@ -1,24 +1,9 @@
|
||||||
{
|
{
|
||||||
"env": {
|
"root": true,
|
||||||
"node": true,
|
"parser": "@typescript-eslint/parser",
|
||||||
"es6": true,
|
"plugins": ["@typescript-eslint"],
|
||||||
"jest": true
|
|
||||||
},
|
|
||||||
"extends": [
|
"extends": [
|
||||||
"eslint:recommended",
|
"eslint:recommended",
|
||||||
"plugin:@typescript-eslint/eslint-recommended",
|
"plugin:@typescript-eslint/recommended"
|
||||||
"plugin:@typescript-eslint/recommended",
|
|
||||||
"plugin:jest/recommended",
|
|
||||||
"plugin:prettier/recommended"
|
|
||||||
],
|
|
||||||
"parser": "@typescript-eslint/parser",
|
|
||||||
"parserOptions": {
|
|
||||||
"ecmaVersion": 2023,
|
|
||||||
"sourceType": "module"
|
|
||||||
},
|
|
||||||
"plugins": [
|
|
||||||
"@typescript-eslint",
|
|
||||||
"jest",
|
|
||||||
"prettier"
|
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
|
|
13
.github/workflows/build.yml
vendored
Normal file
13
.github/workflows/build.yml
vendored
Normal file
|
@ -0,0 +1,13 @@
|
||||||
|
name: Build and Test
|
||||||
|
on: [push, pull_request]
|
||||||
|
jobs:
|
||||||
|
build:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v3
|
||||||
|
- uses: actions/setup-node@v3
|
||||||
|
with:
|
||||||
|
node-version: '16'
|
||||||
|
- run: npm ci
|
||||||
|
- run: npm run build
|
||||||
|
- run: npm test
|
|
@ -1,826 +0,0 @@
|
||||||
import {beforeEach, describe, expect, jest, test} from '@jest/globals';
|
|
||||||
import * as fs from 'fs';
|
|
||||||
import * as path from 'path';
|
|
||||||
|
|
||||||
import {Builder} from '@docker/actions-toolkit/lib/buildx/builder';
|
|
||||||
import {Buildx} from '@docker/actions-toolkit/lib/buildx/buildx';
|
|
||||||
import {Build} from '@docker/actions-toolkit/lib/buildx/build';
|
|
||||||
import {Context} from '@docker/actions-toolkit/lib/context';
|
|
||||||
import {Docker} from '@docker/actions-toolkit/lib/docker/docker';
|
|
||||||
import {GitHub} from '@docker/actions-toolkit/lib/github';
|
|
||||||
import {Toolkit} from '@docker/actions-toolkit/lib/toolkit';
|
|
||||||
|
|
||||||
import {BuilderInfo} from '@docker/actions-toolkit/lib/types/buildx/builder';
|
|
||||||
import {GitHubRepo} from '@docker/actions-toolkit/lib/types/github';
|
|
||||||
|
|
||||||
import * as context from '../src/context';
|
|
||||||
|
|
||||||
const tmpDir = path.join('/tmp', '.docker-build-push-jest');
|
|
||||||
const tmpName = path.join(tmpDir, '.tmpname-jest');
|
|
||||||
|
|
||||||
import repoFixture from './fixtures/github-repo.json';
|
|
||||||
jest.spyOn(GitHub.prototype, 'repoData').mockImplementation((): Promise<GitHubRepo> => {
|
|
||||||
return <Promise<GitHubRepo>>(repoFixture as unknown);
|
|
||||||
});
|
|
||||||
|
|
||||||
jest.spyOn(Context, 'tmpDir').mockImplementation((): string => {
|
|
||||||
if (!fs.existsSync(tmpDir)) {
|
|
||||||
fs.mkdirSync(tmpDir, {recursive: true});
|
|
||||||
}
|
|
||||||
return tmpDir;
|
|
||||||
});
|
|
||||||
|
|
||||||
jest.spyOn(Context, 'tmpName').mockImplementation((): string => {
|
|
||||||
return tmpName;
|
|
||||||
});
|
|
||||||
|
|
||||||
jest.spyOn(Docker, 'isAvailable').mockImplementation(async (): Promise<boolean> => {
|
|
||||||
return true;
|
|
||||||
});
|
|
||||||
|
|
||||||
const metadataJson = path.join(tmpDir, 'metadata.json');
|
|
||||||
jest.spyOn(Build.prototype, 'getMetadataFilePath').mockImplementation((): string => {
|
|
||||||
return metadataJson;
|
|
||||||
});
|
|
||||||
|
|
||||||
const imageIDFilePath = path.join(tmpDir, 'iidfile.txt');
|
|
||||||
jest.spyOn(Build.prototype, 'getImageIDFilePath').mockImplementation((): string => {
|
|
||||||
return imageIDFilePath;
|
|
||||||
});
|
|
||||||
|
|
||||||
jest.spyOn(Builder.prototype, 'inspect').mockImplementation(async (): Promise<BuilderInfo> => {
|
|
||||||
return {
|
|
||||||
name: 'builder2',
|
|
||||||
driver: 'docker-container',
|
|
||||||
lastActivity: new Date('2023-01-16 09:45:23 +0000 UTC'),
|
|
||||||
nodes: [
|
|
||||||
{
|
|
||||||
buildkit: 'v0.11.0',
|
|
||||||
'buildkitd-flags': '--debug --allow-insecure-entitlement security.insecure --allow-insecure-entitlement network.host',
|
|
||||||
'driver-opts': ['BUILDKIT_STEP_LOG_MAX_SIZE=10485760', 'BUILDKIT_STEP_LOG_MAX_SPEED=10485760', 'JAEGER_TRACE=localhost:6831', 'image=moby/buildkit:latest', 'network=host'],
|
|
||||||
endpoint: 'unix:///var/run/docker.sock',
|
|
||||||
name: 'builder20',
|
|
||||||
platforms: 'linux/amd64,linux/amd64/v2,linux/amd64/v3,linux/arm64,linux/riscv64,linux/ppc64le,linux/s390x,linux/386,linux/mips64le,linux/mips64,linux/arm/v7,linux/arm/v6',
|
|
||||||
status: 'running'
|
|
||||||
}
|
|
||||||
]
|
|
||||||
};
|
|
||||||
});
|
|
||||||
|
|
||||||
describe('getArgs', () => {
|
|
||||||
beforeEach(() => {
|
|
||||||
process.env = Object.keys(process.env).reduce((object, key) => {
|
|
||||||
if (!key.startsWith('INPUT_')) {
|
|
||||||
object[key] = process.env[key];
|
|
||||||
}
|
|
||||||
return object;
|
|
||||||
}, {});
|
|
||||||
});
|
|
||||||
|
|
||||||
// prettier-ignore
|
|
||||||
test.each([
|
|
||||||
[
|
|
||||||
0,
|
|
||||||
'0.4.1',
|
|
||||||
new Map<string, string>([
|
|
||||||
['context', '.'],
|
|
||||||
['load', 'false'],
|
|
||||||
['no-cache', 'false'],
|
|
||||||
['push', 'false'],
|
|
||||||
['pull', 'false'],
|
|
||||||
]),
|
|
||||||
[
|
|
||||||
'build',
|
|
||||||
'--iidfile', imageIDFilePath,
|
|
||||||
'.'
|
|
||||||
]
|
|
||||||
],
|
|
||||||
[
|
|
||||||
1,
|
|
||||||
'0.4.2',
|
|
||||||
new Map<string, string>([
|
|
||||||
['build-args', `MY_ARG=val1,val2,val3
|
|
||||||
ARG=val
|
|
||||||
"MULTILINE=aaaa
|
|
||||||
bbbb
|
|
||||||
ccc"`],
|
|
||||||
['load', 'false'],
|
|
||||||
['no-cache', 'false'],
|
|
||||||
['push', 'false'],
|
|
||||||
['pull', 'false'],
|
|
||||||
]),
|
|
||||||
[
|
|
||||||
'build',
|
|
||||||
'--build-arg', 'MY_ARG=val1,val2,val3',
|
|
||||||
'--build-arg', 'ARG=val',
|
|
||||||
'--build-arg', `MULTILINE=aaaa\nbbbb\nccc`,
|
|
||||||
'--iidfile', imageIDFilePath,
|
|
||||||
'https://github.com/docker/build-push-action.git#refs/heads/master'
|
|
||||||
]
|
|
||||||
],
|
|
||||||
[
|
|
||||||
2,
|
|
||||||
'0.4.2',
|
|
||||||
new Map<string, string>([
|
|
||||||
['tags', 'name/app:7.4, name/app:latest'],
|
|
||||||
['load', 'false'],
|
|
||||||
['no-cache', 'false'],
|
|
||||||
['push', 'false'],
|
|
||||||
['pull', 'false'],
|
|
||||||
]),
|
|
||||||
[
|
|
||||||
'build',
|
|
||||||
'--iidfile', imageIDFilePath,
|
|
||||||
'--tag', 'name/app:7.4',
|
|
||||||
'--tag', 'name/app:latest',
|
|
||||||
'https://github.com/docker/build-push-action.git#refs/heads/master'
|
|
||||||
]
|
|
||||||
],
|
|
||||||
[
|
|
||||||
3,
|
|
||||||
'0.4.2',
|
|
||||||
new Map<string, string>([
|
|
||||||
['context', '.'],
|
|
||||||
['labels', 'org.opencontainers.image.title=buildkit\norg.opencontainers.image.description=concurrent, cache-efficient, and Dockerfile-agnostic builder toolkit'],
|
|
||||||
['outputs', 'type=local,dest=./release-out'],
|
|
||||||
['load', 'false'],
|
|
||||||
['no-cache', 'false'],
|
|
||||||
['push', 'false'],
|
|
||||||
['pull', 'false'],
|
|
||||||
]),
|
|
||||||
[
|
|
||||||
'build',
|
|
||||||
'--label', 'org.opencontainers.image.title=buildkit',
|
|
||||||
'--label', 'org.opencontainers.image.description=concurrent, cache-efficient, and Dockerfile-agnostic builder toolkit',
|
|
||||||
'--output', 'type=local,dest=./release-out',
|
|
||||||
'.'
|
|
||||||
]
|
|
||||||
],
|
|
||||||
[
|
|
||||||
4,
|
|
||||||
'0.4.1',
|
|
||||||
new Map<string, string>([
|
|
||||||
['context', '.'],
|
|
||||||
['platforms', 'linux/amd64,linux/arm64'],
|
|
||||||
['load', 'false'],
|
|
||||||
['no-cache', 'false'],
|
|
||||||
['push', 'false'],
|
|
||||||
['pull', 'false'],
|
|
||||||
]),
|
|
||||||
[
|
|
||||||
'build',
|
|
||||||
'--platform', 'linux/amd64,linux/arm64',
|
|
||||||
'.'
|
|
||||||
]
|
|
||||||
],
|
|
||||||
[
|
|
||||||
5,
|
|
||||||
'0.4.1',
|
|
||||||
new Map<string, string>([
|
|
||||||
['context', '.'],
|
|
||||||
['load', 'false'],
|
|
||||||
['no-cache', 'false'],
|
|
||||||
['push', 'false'],
|
|
||||||
['pull', 'false'],
|
|
||||||
]),
|
|
||||||
[
|
|
||||||
'build',
|
|
||||||
'--iidfile', imageIDFilePath,
|
|
||||||
'.'
|
|
||||||
]
|
|
||||||
],
|
|
||||||
[
|
|
||||||
6,
|
|
||||||
'0.4.2',
|
|
||||||
new Map<string, string>([
|
|
||||||
['context', '.'],
|
|
||||||
['secrets', 'GIT_AUTH_TOKEN=abcdefghijklmno=0123456789'],
|
|
||||||
['load', 'false'],
|
|
||||||
['no-cache', 'false'],
|
|
||||||
['push', 'false'],
|
|
||||||
['pull', 'false'],
|
|
||||||
]),
|
|
||||||
[
|
|
||||||
'build',
|
|
||||||
'--iidfile', imageIDFilePath,
|
|
||||||
'--secret', `id=GIT_AUTH_TOKEN,src=${tmpName}`,
|
|
||||||
'.'
|
|
||||||
]
|
|
||||||
],
|
|
||||||
[
|
|
||||||
7,
|
|
||||||
'0.4.2',
|
|
||||||
new Map<string, string>([
|
|
||||||
['github-token', 'abcdefghijklmno0123456789'],
|
|
||||||
['outputs', '.'],
|
|
||||||
['load', 'false'],
|
|
||||||
['no-cache', 'false'],
|
|
||||||
['push', 'false'],
|
|
||||||
['pull', 'false'],
|
|
||||||
]),
|
|
||||||
[
|
|
||||||
'build',
|
|
||||||
'--output', '.',
|
|
||||||
'--secret', `id=GIT_AUTH_TOKEN,src=${tmpName}`,
|
|
||||||
'https://github.com/docker/build-push-action.git#refs/heads/master'
|
|
||||||
]
|
|
||||||
],
|
|
||||||
[
|
|
||||||
8,
|
|
||||||
'0.4.2',
|
|
||||||
new Map<string, string>([
|
|
||||||
['context', 'https://github.com/docker/build-push-action.git#refs/heads/master'],
|
|
||||||
['tag', 'localhost:5000/name/app:latest'],
|
|
||||||
['platforms', 'linux/amd64,linux/arm64'],
|
|
||||||
['secrets', 'GIT_AUTH_TOKEN=abcdefghijklmno=0123456789'],
|
|
||||||
['file', './test/Dockerfile'],
|
|
||||||
['builder', 'builder-git-context-2'],
|
|
||||||
['load', 'false'],
|
|
||||||
['no-cache', 'false'],
|
|
||||||
['push', 'true'],
|
|
||||||
['pull', 'false'],
|
|
||||||
]),
|
|
||||||
[
|
|
||||||
'build',
|
|
||||||
'--file', './test/Dockerfile',
|
|
||||||
'--iidfile', imageIDFilePath,
|
|
||||||
'--platform', 'linux/amd64,linux/arm64',
|
|
||||||
'--secret', `id=GIT_AUTH_TOKEN,src=${tmpName}`,
|
|
||||||
'--builder', 'builder-git-context-2',
|
|
||||||
'--push',
|
|
||||||
'https://github.com/docker/build-push-action.git#refs/heads/master'
|
|
||||||
]
|
|
||||||
],
|
|
||||||
[
|
|
||||||
9,
|
|
||||||
'0.4.2',
|
|
||||||
new Map<string, string>([
|
|
||||||
['context', 'https://github.com/docker/build-push-action.git#refs/heads/master'],
|
|
||||||
['tag', 'localhost:5000/name/app:latest'],
|
|
||||||
['platforms', 'linux/amd64,linux/arm64'],
|
|
||||||
['secrets', `GIT_AUTH_TOKEN=abcdefghi,jklmno=0123456789
|
|
||||||
"MYSECRET=aaaaaaaa
|
|
||||||
bbbbbbb
|
|
||||||
ccccccccc"
|
|
||||||
FOO=bar
|
|
||||||
"EMPTYLINE=aaaa
|
|
||||||
|
|
||||||
bbbb
|
|
||||||
ccc"`],
|
|
||||||
['file', './test/Dockerfile'],
|
|
||||||
['builder', 'builder-git-context-2'],
|
|
||||||
['load', 'false'],
|
|
||||||
['no-cache', 'false'],
|
|
||||||
['push', 'true'],
|
|
||||||
['pull', 'false'],
|
|
||||||
]),
|
|
||||||
[
|
|
||||||
'build',
|
|
||||||
'--file', './test/Dockerfile',
|
|
||||||
'--iidfile', imageIDFilePath,
|
|
||||||
'--platform', 'linux/amd64,linux/arm64',
|
|
||||||
'--secret', `id=GIT_AUTH_TOKEN,src=${tmpName}`,
|
|
||||||
'--secret', `id=MYSECRET,src=${tmpName}`,
|
|
||||||
'--secret', `id=FOO,src=${tmpName}`,
|
|
||||||
'--secret', `id=EMPTYLINE,src=${tmpName}`,
|
|
||||||
'--builder', 'builder-git-context-2',
|
|
||||||
'--push',
|
|
||||||
'https://github.com/docker/build-push-action.git#refs/heads/master'
|
|
||||||
]
|
|
||||||
],
|
|
||||||
[
|
|
||||||
10,
|
|
||||||
'0.4.2',
|
|
||||||
new Map<string, string>([
|
|
||||||
['context', 'https://github.com/docker/build-push-action.git#refs/heads/master'],
|
|
||||||
['tag', 'localhost:5000/name/app:latest'],
|
|
||||||
['platforms', 'linux/amd64,linux/arm64'],
|
|
||||||
['secrets', `GIT_AUTH_TOKEN=abcdefghi,jklmno=0123456789
|
|
||||||
MYSECRET=aaaaaaaa
|
|
||||||
bbbbbbb
|
|
||||||
ccccccccc
|
|
||||||
FOO=bar
|
|
||||||
EMPTYLINE=aaaa
|
|
||||||
|
|
||||||
bbbb
|
|
||||||
ccc`],
|
|
||||||
['file', './test/Dockerfile'],
|
|
||||||
['builder', 'builder-git-context-2'],
|
|
||||||
['load', 'false'],
|
|
||||||
['no-cache', 'false'],
|
|
||||||
['push', 'true'],
|
|
||||||
['pull', 'false'],
|
|
||||||
]),
|
|
||||||
[
|
|
||||||
'build',
|
|
||||||
'--file', './test/Dockerfile',
|
|
||||||
'--iidfile', imageIDFilePath,
|
|
||||||
'--platform', 'linux/amd64,linux/arm64',
|
|
||||||
'--secret', `id=GIT_AUTH_TOKEN,src=${tmpName}`,
|
|
||||||
'--secret', `id=MYSECRET,src=${tmpName}`,
|
|
||||||
'--secret', `id=FOO,src=${tmpName}`,
|
|
||||||
'--secret', `id=EMPTYLINE,src=${tmpName}`,
|
|
||||||
'--builder', 'builder-git-context-2',
|
|
||||||
'--push',
|
|
||||||
'https://github.com/docker/build-push-action.git#refs/heads/master'
|
|
||||||
]
|
|
||||||
],
|
|
||||||
[
|
|
||||||
11,
|
|
||||||
'0.5.1',
|
|
||||||
new Map<string, string>([
|
|
||||||
['context', 'https://github.com/docker/build-push-action.git#refs/heads/master'],
|
|
||||||
['tag', 'localhost:5000/name/app:latest'],
|
|
||||||
['secret-files', `MY_SECRET=${path.join(__dirname, 'fixtures', 'secret.txt')}`],
|
|
||||||
['file', './test/Dockerfile'],
|
|
||||||
['builder', 'builder-git-context-2'],
|
|
||||||
['network', 'host'],
|
|
||||||
['load', 'false'],
|
|
||||||
['no-cache', 'false'],
|
|
||||||
['push', 'true'],
|
|
||||||
['pull', 'false'],
|
|
||||||
]),
|
|
||||||
[
|
|
||||||
'build',
|
|
||||||
'--file', './test/Dockerfile',
|
|
||||||
'--iidfile', imageIDFilePath,
|
|
||||||
'--secret', `id=MY_SECRET,src=${tmpName}`,
|
|
||||||
'--builder', 'builder-git-context-2',
|
|
||||||
'--network', 'host',
|
|
||||||
'--push',
|
|
||||||
'https://github.com/docker/build-push-action.git#refs/heads/master'
|
|
||||||
]
|
|
||||||
],
|
|
||||||
[
|
|
||||||
12,
|
|
||||||
'0.4.2',
|
|
||||||
new Map<string, string>([
|
|
||||||
['context', '.'],
|
|
||||||
['labels', 'org.opencontainers.image.title=filter_results_top_n\norg.opencontainers.image.description=Reference implementation of operation "filter results (top-n)"'],
|
|
||||||
['outputs', 'type=local,dest=./release-out'],
|
|
||||||
['load', 'false'],
|
|
||||||
['no-cache', 'false'],
|
|
||||||
['push', 'false'],
|
|
||||||
['pull', 'false'],
|
|
||||||
]),
|
|
||||||
[
|
|
||||||
'build',
|
|
||||||
'--label', 'org.opencontainers.image.title=filter_results_top_n',
|
|
||||||
'--label', 'org.opencontainers.image.description=Reference implementation of operation "filter results (top-n)"',
|
|
||||||
'--output', 'type=local,dest=./release-out',
|
|
||||||
'.'
|
|
||||||
]
|
|
||||||
],
|
|
||||||
[
|
|
||||||
13,
|
|
||||||
'0.6.0',
|
|
||||||
new Map<string, string>([
|
|
||||||
['context', '.'],
|
|
||||||
['tag', 'localhost:5000/name/app:latest'],
|
|
||||||
['file', './test/Dockerfile'],
|
|
||||||
['add-hosts', 'docker:10.180.0.1,foo:10.0.0.1'],
|
|
||||||
['network', 'host'],
|
|
||||||
['load', 'false'],
|
|
||||||
['no-cache', 'false'],
|
|
||||||
['push', 'true'],
|
|
||||||
['pull', 'false'],
|
|
||||||
]),
|
|
||||||
[
|
|
||||||
'build',
|
|
||||||
'--add-host', 'docker:10.180.0.1',
|
|
||||||
'--add-host', 'foo:10.0.0.1',
|
|
||||||
'--file', './test/Dockerfile',
|
|
||||||
'--iidfile', imageIDFilePath,
|
|
||||||
'--metadata-file', metadataJson,
|
|
||||||
'--network', 'host',
|
|
||||||
'--push',
|
|
||||||
'.'
|
|
||||||
]
|
|
||||||
],
|
|
||||||
[
|
|
||||||
14,
|
|
||||||
'0.7.0',
|
|
||||||
new Map<string, string>([
|
|
||||||
['context', '.'],
|
|
||||||
['file', './test/Dockerfile'],
|
|
||||||
['add-hosts', 'docker:10.180.0.1\nfoo:10.0.0.1'],
|
|
||||||
['cgroup-parent', 'foo'],
|
|
||||||
['shm-size', '2g'],
|
|
||||||
['ulimit', `nofile=1024:1024
|
|
||||||
nproc=3`],
|
|
||||||
['load', 'false'],
|
|
||||||
['no-cache', 'false'],
|
|
||||||
['push', 'false'],
|
|
||||||
['pull', 'false'],
|
|
||||||
]),
|
|
||||||
[
|
|
||||||
'build',
|
|
||||||
'--add-host', 'docker:10.180.0.1',
|
|
||||||
'--add-host', 'foo:10.0.0.1',
|
|
||||||
'--cgroup-parent', 'foo',
|
|
||||||
'--file', './test/Dockerfile',
|
|
||||||
'--iidfile', imageIDFilePath,
|
|
||||||
'--shm-size', '2g',
|
|
||||||
'--ulimit', 'nofile=1024:1024',
|
|
||||||
'--ulimit', 'nproc=3',
|
|
||||||
'--metadata-file', metadataJson,
|
|
||||||
'.'
|
|
||||||
]
|
|
||||||
],
|
|
||||||
[
|
|
||||||
15,
|
|
||||||
'0.7.0',
|
|
||||||
new Map<string, string>([
|
|
||||||
['context', '{{defaultContext}}:docker'],
|
|
||||||
['load', 'false'],
|
|
||||||
['no-cache', 'false'],
|
|
||||||
['push', 'false'],
|
|
||||||
['pull', 'false'],
|
|
||||||
]),
|
|
||||||
[
|
|
||||||
'build',
|
|
||||||
'--iidfile', imageIDFilePath,
|
|
||||||
'--metadata-file', metadataJson,
|
|
||||||
'https://github.com/docker/build-push-action.git#refs/heads/master:docker'
|
|
||||||
]
|
|
||||||
],
|
|
||||||
[
|
|
||||||
16,
|
|
||||||
'0.8.2',
|
|
||||||
new Map<string, string>([
|
|
||||||
['github-token', 'abcdefghijklmno0123456789'],
|
|
||||||
['context', '{{defaultContext}}:subdir'],
|
|
||||||
['load', 'false'],
|
|
||||||
['no-cache', 'false'],
|
|
||||||
['push', 'false'],
|
|
||||||
['pull', 'false'],
|
|
||||||
]),
|
|
||||||
[
|
|
||||||
'build',
|
|
||||||
'--iidfile', imageIDFilePath,
|
|
||||||
'--secret', `id=GIT_AUTH_TOKEN,src=${tmpName}`,
|
|
||||||
'--metadata-file', metadataJson,
|
|
||||||
'https://github.com/docker/build-push-action.git#refs/heads/master:subdir'
|
|
||||||
]
|
|
||||||
],
|
|
||||||
[
|
|
||||||
17,
|
|
||||||
'0.8.2',
|
|
||||||
new Map<string, string>([
|
|
||||||
['context', '.'],
|
|
||||||
['load', 'false'],
|
|
||||||
['no-cache', 'false'],
|
|
||||||
['push', 'false'],
|
|
||||||
['pull', 'false'],
|
|
||||||
['provenance', 'true'],
|
|
||||||
]),
|
|
||||||
[
|
|
||||||
'build',
|
|
||||||
'--iidfile', imageIDFilePath,
|
|
||||||
'--metadata-file', metadataJson,
|
|
||||||
'.'
|
|
||||||
]
|
|
||||||
],
|
|
||||||
[
|
|
||||||
18,
|
|
||||||
'0.10.0',
|
|
||||||
new Map<string, string>([
|
|
||||||
['context', '.'],
|
|
||||||
['load', 'false'],
|
|
||||||
['no-cache', 'false'],
|
|
||||||
['push', 'false'],
|
|
||||||
['pull', 'false'],
|
|
||||||
]),
|
|
||||||
[
|
|
||||||
'build',
|
|
||||||
'--iidfile', imageIDFilePath,
|
|
||||||
'--attest', `type=provenance,mode=min,inline-only=true,builder-id=https://github.com/docker/build-push-action/actions/runs/123456789/attempts/1`,
|
|
||||||
'--metadata-file', metadataJson,
|
|
||||||
'.'
|
|
||||||
]
|
|
||||||
],
|
|
||||||
[
|
|
||||||
19,
|
|
||||||
'0.10.0',
|
|
||||||
new Map<string, string>([
|
|
||||||
['context', '.'],
|
|
||||||
['load', 'false'],
|
|
||||||
['no-cache', 'false'],
|
|
||||||
['push', 'false'],
|
|
||||||
['pull', 'false'],
|
|
||||||
['provenance', 'true'],
|
|
||||||
]),
|
|
||||||
[
|
|
||||||
'build',
|
|
||||||
'--iidfile', imageIDFilePath,
|
|
||||||
'--attest', `type=provenance,builder-id=https://github.com/docker/build-push-action/actions/runs/123456789/attempts/1`,
|
|
||||||
'--metadata-file', metadataJson,
|
|
||||||
'.'
|
|
||||||
]
|
|
||||||
],
|
|
||||||
[
|
|
||||||
20,
|
|
||||||
'0.10.0',
|
|
||||||
new Map<string, string>([
|
|
||||||
['context', '.'],
|
|
||||||
['load', 'false'],
|
|
||||||
['no-cache', 'false'],
|
|
||||||
['push', 'false'],
|
|
||||||
['pull', 'false'],
|
|
||||||
['provenance', 'mode=max'],
|
|
||||||
]),
|
|
||||||
[
|
|
||||||
'build',
|
|
||||||
'--iidfile', imageIDFilePath,
|
|
||||||
'--attest', `type=provenance,mode=max,builder-id=https://github.com/docker/build-push-action/actions/runs/123456789/attempts/1`,
|
|
||||||
'--metadata-file', metadataJson,
|
|
||||||
'.'
|
|
||||||
]
|
|
||||||
],
|
|
||||||
[
|
|
||||||
21,
|
|
||||||
'0.10.0',
|
|
||||||
new Map<string, string>([
|
|
||||||
['context', '.'],
|
|
||||||
['load', 'false'],
|
|
||||||
['no-cache', 'false'],
|
|
||||||
['push', 'false'],
|
|
||||||
['pull', 'false'],
|
|
||||||
['provenance', 'false'],
|
|
||||||
]),
|
|
||||||
[
|
|
||||||
'build',
|
|
||||||
'--iidfile', imageIDFilePath,
|
|
||||||
'--attest', 'type=provenance,disabled=true',
|
|
||||||
'--metadata-file', metadataJson,
|
|
||||||
'.'
|
|
||||||
]
|
|
||||||
],
|
|
||||||
[
|
|
||||||
22,
|
|
||||||
'0.10.0',
|
|
||||||
new Map<string, string>([
|
|
||||||
['context', '.'],
|
|
||||||
['load', 'false'],
|
|
||||||
['no-cache', 'false'],
|
|
||||||
['push', 'false'],
|
|
||||||
['pull', 'false'],
|
|
||||||
['provenance', 'builder-id=foo'],
|
|
||||||
]),
|
|
||||||
[
|
|
||||||
'build',
|
|
||||||
'--iidfile', imageIDFilePath,
|
|
||||||
'--attest', 'type=provenance,builder-id=foo',
|
|
||||||
'--metadata-file', metadataJson,
|
|
||||||
'.'
|
|
||||||
]
|
|
||||||
],
|
|
||||||
[
|
|
||||||
23,
|
|
||||||
'0.10.0',
|
|
||||||
new Map<string, string>([
|
|
||||||
['context', '.'],
|
|
||||||
['load', 'false'],
|
|
||||||
['no-cache', 'false'],
|
|
||||||
['push', 'false'],
|
|
||||||
['pull', 'false'],
|
|
||||||
['outputs', 'type=docker'],
|
|
||||||
]),
|
|
||||||
[
|
|
||||||
'build',
|
|
||||||
'--iidfile', imageIDFilePath,
|
|
||||||
"--output", 'type=docker',
|
|
||||||
'--metadata-file', metadataJson,
|
|
||||||
'.'
|
|
||||||
]
|
|
||||||
],
|
|
||||||
[
|
|
||||||
24,
|
|
||||||
'0.10.0',
|
|
||||||
new Map<string, string>([
|
|
||||||
['context', '.'],
|
|
||||||
['load', 'true'],
|
|
||||||
['no-cache', 'false'],
|
|
||||||
['push', 'false'],
|
|
||||||
['pull', 'false'],
|
|
||||||
]),
|
|
||||||
[
|
|
||||||
'build',
|
|
||||||
'--iidfile', imageIDFilePath,
|
|
||||||
'--load',
|
|
||||||
'--metadata-file', metadataJson,
|
|
||||||
'.'
|
|
||||||
]
|
|
||||||
],
|
|
||||||
[
|
|
||||||
25,
|
|
||||||
'0.10.0',
|
|
||||||
new Map<string, string>([
|
|
||||||
['context', '.'],
|
|
||||||
['build-args', `FOO=bar#baz`],
|
|
||||||
['load', 'true'],
|
|
||||||
['no-cache', 'false'],
|
|
||||||
['push', 'false'],
|
|
||||||
['pull', 'false'],
|
|
||||||
]),
|
|
||||||
[
|
|
||||||
'build',
|
|
||||||
'--build-arg', 'FOO=bar#baz',
|
|
||||||
'--iidfile', imageIDFilePath,
|
|
||||||
'--load',
|
|
||||||
'--metadata-file', metadataJson,
|
|
||||||
'.'
|
|
||||||
]
|
|
||||||
],
|
|
||||||
[
|
|
||||||
26,
|
|
||||||
'0.10.0',
|
|
||||||
new Map<string, string>([
|
|
||||||
['context', '.'],
|
|
||||||
['no-cache', 'false'],
|
|
||||||
['load', 'true'],
|
|
||||||
['push', 'false'],
|
|
||||||
['pull', 'false'],
|
|
||||||
['secret-envs', `MY_SECRET=MY_SECRET_ENV
|
|
||||||
ANOTHER_SECRET=ANOTHER_SECRET_ENV`]
|
|
||||||
]),
|
|
||||||
[
|
|
||||||
'build',
|
|
||||||
'--secret', 'id=MY_SECRET,env=MY_SECRET_ENV',
|
|
||||||
'--secret', 'id=ANOTHER_SECRET,env=ANOTHER_SECRET_ENV',
|
|
||||||
'--iidfile', imageIDFilePath,
|
|
||||||
'--load',
|
|
||||||
'--metadata-file', metadataJson,
|
|
||||||
'.'
|
|
||||||
]
|
|
||||||
],
|
|
||||||
[
|
|
||||||
27,
|
|
||||||
'0.10.0',
|
|
||||||
new Map<string, string>([
|
|
||||||
['context', '.'],
|
|
||||||
['no-cache', 'false'],
|
|
||||||
['load', 'true'],
|
|
||||||
['push', 'false'],
|
|
||||||
['pull', 'false'],
|
|
||||||
['secret-envs', 'MY_SECRET=MY_SECRET_ENV,ANOTHER_SECRET=ANOTHER_SECRET_ENV']
|
|
||||||
]),
|
|
||||||
[
|
|
||||||
'build',
|
|
||||||
'--secret', 'id=MY_SECRET,env=MY_SECRET_ENV',
|
|
||||||
'--secret', 'id=ANOTHER_SECRET,env=ANOTHER_SECRET_ENV',
|
|
||||||
'--iidfile', imageIDFilePath,
|
|
||||||
'--load',
|
|
||||||
'--metadata-file', metadataJson,
|
|
||||||
'.'
|
|
||||||
]
|
|
||||||
],
|
|
||||||
[
|
|
||||||
28,
|
|
||||||
'0.11.0',
|
|
||||||
new Map<string, string>([
|
|
||||||
['context', '.'],
|
|
||||||
['annotations', 'example1=www\nindex:example2=xxx\nmanifest:example3=yyy\nmanifest-descriptor[linux/amd64]:example4=zzz'],
|
|
||||||
['outputs', 'type=local,dest=./release-out'],
|
|
||||||
['load', 'false'],
|
|
||||||
['no-cache', 'false'],
|
|
||||||
['push', 'false'],
|
|
||||||
['pull', 'false'],
|
|
||||||
]),
|
|
||||||
[
|
|
||||||
'build',
|
|
||||||
'--output', 'type=local,dest=./release-out',
|
|
||||||
'--attest', `type=provenance,mode=min,inline-only=true,builder-id=https://github.com/docker/build-push-action/actions/runs/123456789/attempts/1`,
|
|
||||||
'--metadata-file', metadataJson,
|
|
||||||
'.'
|
|
||||||
]
|
|
||||||
],
|
|
||||||
[
|
|
||||||
29,
|
|
||||||
'0.12.0',
|
|
||||||
new Map<string, string>([
|
|
||||||
['context', '.'],
|
|
||||||
['annotations', 'example1=www\nindex:example2=xxx\nmanifest:example3=yyy\nmanifest-descriptor[linux/amd64]:example4=zzz'],
|
|
||||||
['outputs', 'type=local,dest=./release-out'],
|
|
||||||
['load', 'false'],
|
|
||||||
['no-cache', 'false'],
|
|
||||||
['push', 'false'],
|
|
||||||
['pull', 'false'],
|
|
||||||
]),
|
|
||||||
[
|
|
||||||
'build',
|
|
||||||
'--annotation', 'example1=www',
|
|
||||||
'--annotation', 'index:example2=xxx',
|
|
||||||
'--annotation', 'manifest:example3=yyy',
|
|
||||||
'--annotation', 'manifest-descriptor[linux/amd64]:example4=zzz',
|
|
||||||
'--output', 'type=local,dest=./release-out',
|
|
||||||
'--attest', `type=provenance,mode=min,inline-only=true,builder-id=https://github.com/docker/build-push-action/actions/runs/123456789/attempts/1`,
|
|
||||||
'--metadata-file', metadataJson,
|
|
||||||
'.'
|
|
||||||
]
|
|
||||||
],
|
|
||||||
[
|
|
||||||
30,
|
|
||||||
'0.12.0',
|
|
||||||
new Map<string, string>([
|
|
||||||
['context', '.'],
|
|
||||||
['outputs', `type=image,"name=localhost:5000/name/app:latest,localhost:5000/name/app:foo",push-by-digest=true,name-canonical=true,push=true`],
|
|
||||||
['load', 'false'],
|
|
||||||
['no-cache', 'false'],
|
|
||||||
['push', 'false'],
|
|
||||||
['pull', 'false'],
|
|
||||||
]),
|
|
||||||
[
|
|
||||||
'build',
|
|
||||||
'--iidfile', imageIDFilePath,
|
|
||||||
"--output", `type=image,"name=localhost:5000/name/app:latest,localhost:5000/name/app:foo",push-by-digest=true,name-canonical=true,push=true`,
|
|
||||||
'--attest', `type=provenance,mode=min,inline-only=true,builder-id=https://github.com/docker/build-push-action/actions/runs/123456789/attempts/1`,
|
|
||||||
'--metadata-file', metadataJson,
|
|
||||||
'.'
|
|
||||||
]
|
|
||||||
],
|
|
||||||
[
|
|
||||||
31,
|
|
||||||
'0.13.1',
|
|
||||||
new Map<string, string>([
|
|
||||||
['context', '.'],
|
|
||||||
['load', 'false'],
|
|
||||||
['no-cache', 'false'],
|
|
||||||
['push', 'false'],
|
|
||||||
['pull', 'false'],
|
|
||||||
['provenance', 'mode=max'],
|
|
||||||
['sbom', 'true'],
|
|
||||||
]),
|
|
||||||
[
|
|
||||||
'build',
|
|
||||||
'--iidfile', imageIDFilePath,
|
|
||||||
'--attest', `type=provenance,mode=max,builder-id=https://github.com/docker/build-push-action/actions/runs/123456789/attempts/1`,
|
|
||||||
'--attest', `type=sbom,disabled=false`,
|
|
||||||
'--metadata-file', metadataJson,
|
|
||||||
'.'
|
|
||||||
]
|
|
||||||
],
|
|
||||||
[
|
|
||||||
32,
|
|
||||||
'0.13.1',
|
|
||||||
new Map<string, string>([
|
|
||||||
['context', '.'],
|
|
||||||
['load', 'false'],
|
|
||||||
['no-cache', 'false'],
|
|
||||||
['push', 'false'],
|
|
||||||
['pull', 'false'],
|
|
||||||
['attests', 'type=provenance,mode=min'],
|
|
||||||
['provenance', 'mode=max'],
|
|
||||||
]),
|
|
||||||
[
|
|
||||||
'build',
|
|
||||||
'--iidfile', imageIDFilePath,
|
|
||||||
'--attest', `type=provenance,mode=max,builder-id=https://github.com/docker/build-push-action/actions/runs/123456789/attempts/1`,
|
|
||||||
'--metadata-file', metadataJson,
|
|
||||||
'.'
|
|
||||||
]
|
|
||||||
],
|
|
||||||
[
|
|
||||||
33,
|
|
||||||
'0.13.1',
|
|
||||||
new Map<string, string>([
|
|
||||||
['context', '.'],
|
|
||||||
['load', 'false'],
|
|
||||||
['no-cache', 'false'],
|
|
||||||
['push', 'false'],
|
|
||||||
['pull', 'false'],
|
|
||||||
['attests', 'type=provenance,mode=min'],
|
|
||||||
]),
|
|
||||||
[
|
|
||||||
'build',
|
|
||||||
'--iidfile', imageIDFilePath,
|
|
||||||
'--attest', `type=provenance,mode=min,builder-id=https://github.com/docker/build-push-action/actions/runs/123456789/attempts/1`,
|
|
||||||
'--metadata-file', metadataJson,
|
|
||||||
'.'
|
|
||||||
]
|
|
||||||
],
|
|
||||||
])(
|
|
||||||
'[%d] given %p with %p as inputs, returns %p',
|
|
||||||
async (num: number, buildxVersion: string, inputs: Map<string, string>, expected: Array<string>) => {
|
|
||||||
inputs.forEach((value: string, name: string) => {
|
|
||||||
setInput(name, value);
|
|
||||||
});
|
|
||||||
const toolkit = new Toolkit();
|
|
||||||
jest.spyOn(Buildx.prototype, 'version').mockImplementation(async (): Promise<string> => {
|
|
||||||
return buildxVersion;
|
|
||||||
});
|
|
||||||
const inp = await context.getInputs();
|
|
||||||
const res = await context.getArgs(inp, toolkit);
|
|
||||||
expect(res).toEqual(expected);
|
|
||||||
}
|
|
||||||
);
|
|
||||||
});
|
|
||||||
|
|
||||||
// See: https://github.com/actions/toolkit/blob/a1b068ec31a042ff1e10a522d8fdf0b8869d53ca/packages/core/src/core.ts#L89
|
|
||||||
function getInputName(name: string): string {
|
|
||||||
return `INPUT_${name.replace(/ /g, '_').toUpperCase()}`;
|
|
||||||
}
|
|
||||||
|
|
||||||
function setInput(name: string, value: string): void {
|
|
||||||
process.env[getInputName(name)] = value;
|
|
||||||
}
|
|
20
dist/index.js
generated
vendored
20
dist/index.js
generated
vendored
File diff suppressed because one or more lines are too long
2
dist/index.js.map
generated
vendored
2
dist/index.js.map
generated
vendored
File diff suppressed because one or more lines are too long
|
@ -1,30 +1,20 @@
|
||||||
import fs from 'fs';
|
import type {Config} from '@jest/types';
|
||||||
import os from 'os';
|
|
||||||
import path from 'path';
|
|
||||||
|
|
||||||
const tmpDir = fs.mkdtempSync(path.join(os.tmpdir(), 'docker-build-push-action-'));
|
const config: Config.InitialOptions = {
|
||||||
|
clearMocks: true,
|
||||||
process.env = Object.assign({}, process.env, {
|
|
||||||
TEMP: tmpDir,
|
|
||||||
GITHUB_REPOSITORY: 'docker/build-push-action',
|
|
||||||
RUNNER_TEMP: path.join(tmpDir, 'runner-temp'),
|
|
||||||
RUNNER_TOOL_CACHE: path.join(tmpDir, 'runner-tool-cache')
|
|
||||||
}) as {
|
|
||||||
[key: string]: string;
|
|
||||||
};
|
|
||||||
|
|
||||||
module.exports = {
|
|
||||||
clearMocks: false,
|
|
||||||
testEnvironment: 'node',
|
|
||||||
moduleFileExtensions: ['js', 'ts'],
|
moduleFileExtensions: ['js', 'ts'],
|
||||||
|
testEnvironment: 'node',
|
||||||
testMatch: ['**/*.test.ts'],
|
testMatch: ['**/*.test.ts'],
|
||||||
transform: {
|
transform: {
|
||||||
'^.+\\.ts$': 'ts-jest'
|
'^.+\\.ts$': 'ts-jest'
|
||||||
},
|
},
|
||||||
moduleNameMapper: {
|
verbose: true,
|
||||||
'^csv-parse/sync': '<rootDir>/node_modules/csv-parse/dist/cjs/sync.cjs'
|
collectCoverage: true,
|
||||||
},
|
collectCoverageFrom: [
|
||||||
collectCoverageFrom: ['src/**/{!(main.ts),}.ts'],
|
'src/**/*.ts',
|
||||||
coveragePathIgnorePatterns: ['lib/', 'node_modules/', '__mocks__/', '__tests__/'],
|
'!src/**/*.d.ts',
|
||||||
verbose: true
|
'!src/**/__tests__/**'
|
||||||
|
]
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export default config;
|
||||||
|
|
13
package-lock.json
generated
13
package-lock.json
generated
|
@ -15,6 +15,7 @@
|
||||||
"portfinder": "^1.0.32"
|
"portfinder": "^1.0.32"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
|
"@types/jest": "^29.5.14",
|
||||||
"@types/node": "^20.12.12",
|
"@types/node": "^20.12.12",
|
||||||
"@typescript-eslint/eslint-plugin": "^7.9.0",
|
"@typescript-eslint/eslint-plugin": "^7.9.0",
|
||||||
"@typescript-eslint/parser": "^7.9.0",
|
"@typescript-eslint/parser": "^7.9.0",
|
||||||
|
@ -25,7 +26,7 @@
|
||||||
"eslint-plugin-prettier": "^5.1.3",
|
"eslint-plugin-prettier": "^5.1.3",
|
||||||
"jest": "^29.7.0",
|
"jest": "^29.7.0",
|
||||||
"prettier": "^3.2.5",
|
"prettier": "^3.2.5",
|
||||||
"ts-jest": "^29.1.2",
|
"ts-jest": "^29.2.5",
|
||||||
"ts-node": "^10.9.2",
|
"ts-node": "^10.9.2",
|
||||||
"typescript": "^5.4.5"
|
"typescript": "^5.4.5"
|
||||||
}
|
}
|
||||||
|
@ -2208,6 +2209,16 @@
|
||||||
"@types/istanbul-lib-report": "*"
|
"@types/istanbul-lib-report": "*"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/@types/jest": {
|
||||||
|
"version": "29.5.14",
|
||||||
|
"resolved": "https://registry.npmjs.org/@types/jest/-/jest-29.5.14.tgz",
|
||||||
|
"integrity": "sha512-ZN+4sdnLUbo8EVvVc2ao0GFW6oVrQRPn4K2lglySj7APvSrgzxHiNNK99us4WDMi57xxA2yggblIAMNhXOotLQ==",
|
||||||
|
"dev": true,
|
||||||
|
"dependencies": {
|
||||||
|
"expect": "^29.0.0",
|
||||||
|
"pretty-format": "^29.0.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/@types/node": {
|
"node_modules/@types/node": {
|
||||||
"version": "20.16.5",
|
"version": "20.16.5",
|
||||||
"resolved": "https://registry.npmjs.org/@types/node/-/node-20.16.5.tgz",
|
"resolved": "https://registry.npmjs.org/@types/node/-/node-20.16.5.tgz",
|
||||||
|
|
|
@ -34,6 +34,7 @@
|
||||||
"portfinder": "^1.0.32"
|
"portfinder": "^1.0.32"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
|
"@types/jest": "^29.5.14",
|
||||||
"@types/node": "^20.12.12",
|
"@types/node": "^20.12.12",
|
||||||
"@typescript-eslint/eslint-plugin": "^7.9.0",
|
"@typescript-eslint/eslint-plugin": "^7.9.0",
|
||||||
"@typescript-eslint/parser": "^7.9.0",
|
"@typescript-eslint/parser": "^7.9.0",
|
||||||
|
@ -44,7 +45,7 @@
|
||||||
"eslint-plugin-prettier": "^5.1.3",
|
"eslint-plugin-prettier": "^5.1.3",
|
||||||
"jest": "^29.7.0",
|
"jest": "^29.7.0",
|
||||||
"prettier": "^3.2.5",
|
"prettier": "^3.2.5",
|
||||||
"ts-jest": "^29.1.2",
|
"ts-jest": "^29.2.5",
|
||||||
"ts-node": "^10.9.2",
|
"ts-node": "^10.9.2",
|
||||||
"typescript": "^5.4.5"
|
"typescript": "^5.4.5"
|
||||||
}
|
}
|
||||||
|
|
80
src/__tests__/blacksmith-builder.test.ts
Normal file
80
src/__tests__/blacksmith-builder.test.ts
Normal file
|
@ -0,0 +1,80 @@
|
||||||
|
import * as core from '@actions/core';
|
||||||
|
import * as main from '../main';
|
||||||
|
import * as reporter from '../reporter';
|
||||||
|
import {getDockerfilePath} from '../context';
|
||||||
|
import { getBuilderAddr } from '../setup_builder';
|
||||||
|
|
||||||
|
jest.mock('@actions/core', () => ({
|
||||||
|
debug: jest.fn(),
|
||||||
|
warning: jest.fn(),
|
||||||
|
info: jest.fn(),
|
||||||
|
saveState: jest.fn(),
|
||||||
|
getState: jest.fn(),
|
||||||
|
setOutput: jest.fn(),
|
||||||
|
setFailed: jest.fn(),
|
||||||
|
error: jest.fn()
|
||||||
|
}));
|
||||||
|
|
||||||
|
jest.mock('../context', () => ({
|
||||||
|
getDockerfilePath: jest.fn(),
|
||||||
|
Inputs: jest.fn()
|
||||||
|
}));
|
||||||
|
|
||||||
|
jest.mock('../reporter', () => ({
|
||||||
|
reportBuilderCreationFailed: jest.fn().mockResolvedValue(undefined)
|
||||||
|
}));
|
||||||
|
|
||||||
|
jest.mock('../setup_builder', () => ({
|
||||||
|
getBuilderAddr: jest.fn()
|
||||||
|
}));
|
||||||
|
|
||||||
|
describe('startBlacksmithBuilder', () => {
|
||||||
|
let mockInputs;
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
jest.clearAllMocks();
|
||||||
|
mockInputs = {nofallback: false};
|
||||||
|
});
|
||||||
|
|
||||||
|
test('should handle missing dockerfile path with nofallback=false', async () => {
|
||||||
|
(getDockerfilePath as jest.Mock).mockReturnValue(null);
|
||||||
|
|
||||||
|
const result = await main.startBlacksmithBuilder(mockInputs);
|
||||||
|
|
||||||
|
expect(result).toEqual({addr: null, buildId: null, exposeId: ''});
|
||||||
|
expect(core.warning).toHaveBeenCalledWith('Error during Blacksmith builder setup: Failed to resolve dockerfile path. Falling back to a local build.');
|
||||||
|
expect(reporter.reportBuilderCreationFailed).toHaveBeenCalledWith(new Error('Failed to resolve dockerfile path'));
|
||||||
|
});
|
||||||
|
|
||||||
|
test('should handle missing dockerfile path with nofallback=true', async () => {
|
||||||
|
(getDockerfilePath as jest.Mock).mockReturnValue(null);
|
||||||
|
mockInputs.nofallback = true;
|
||||||
|
|
||||||
|
await expect(main.startBlacksmithBuilder(mockInputs)).rejects.toThrow('Failed to resolve dockerfile path');
|
||||||
|
expect(core.warning).not.toHaveBeenCalled();
|
||||||
|
expect(reporter.reportBuilderCreationFailed).toHaveBeenCalledWith(new Error('Failed to resolve dockerfile path'));
|
||||||
|
});
|
||||||
|
|
||||||
|
test('should handle error in getBuilderAddr with nofallback=false', async () => {
|
||||||
|
(getDockerfilePath as jest.Mock).mockReturnValue('/path/to/Dockerfile');
|
||||||
|
(getBuilderAddr as jest.Mock).mockRejectedValue(new Error('Failed to obtain Blacksmith builder'));
|
||||||
|
|
||||||
|
mockInputs.nofallback = false;
|
||||||
|
const result = await main.startBlacksmithBuilder(mockInputs);
|
||||||
|
|
||||||
|
expect(result).toEqual({addr: null, buildId: null, exposeId: ''});
|
||||||
|
expect(core.warning).toHaveBeenCalledWith('Error during Blacksmith builder setup: Failed to obtain Blacksmith builder. Falling back to a local build.');
|
||||||
|
expect(reporter.reportBuilderCreationFailed).toHaveBeenCalledWith(new Error('Failed to obtain Blacksmith builder'));
|
||||||
|
});
|
||||||
|
|
||||||
|
test('should handle error in getBuilderAddr with nofallback=true', async () => {
|
||||||
|
(getDockerfilePath as jest.Mock).mockReturnValue('/path/to/Dockerfile');
|
||||||
|
const error = new Error('Failed to obtain Blacksmith builder');
|
||||||
|
(getBuilderAddr as jest.Mock).mockRejectedValue(error);
|
||||||
|
mockInputs.nofallback = true;
|
||||||
|
|
||||||
|
await expect(main.startBlacksmithBuilder(mockInputs)).rejects.toThrow(error);
|
||||||
|
expect(core.warning).not.toHaveBeenCalled();
|
||||||
|
expect(reporter.reportBuilderCreationFailed).toHaveBeenCalledWith(error);
|
||||||
|
});
|
||||||
|
});
|
540
src/main.ts
540
src/main.ts
|
@ -6,7 +6,6 @@ import * as actionsToolkit from '@docker/actions-toolkit';
|
||||||
|
|
||||||
import {Buildx} from '@docker/actions-toolkit/lib/buildx/buildx';
|
import {Buildx} from '@docker/actions-toolkit/lib/buildx/buildx';
|
||||||
import {History as BuildxHistory} from '@docker/actions-toolkit/lib/buildx/history';
|
import {History as BuildxHistory} from '@docker/actions-toolkit/lib/buildx/history';
|
||||||
import {ExportRecordResponse} from '@docker/actions-toolkit/lib/types/buildx/history';
|
|
||||||
import {Context} from '@docker/actions-toolkit/lib/context';
|
import {Context} from '@docker/actions-toolkit/lib/context';
|
||||||
import {Docker} from '@docker/actions-toolkit/lib/docker/docker';
|
import {Docker} from '@docker/actions-toolkit/lib/docker/docker';
|
||||||
import {Exec} from '@docker/actions-toolkit/lib/exec';
|
import {Exec} from '@docker/actions-toolkit/lib/exec';
|
||||||
|
@ -16,481 +15,17 @@ import {Util} from '@docker/actions-toolkit/lib/util';
|
||||||
|
|
||||||
import {BuilderInfo} from '@docker/actions-toolkit/lib/types/buildx/builder';
|
import {BuilderInfo} from '@docker/actions-toolkit/lib/types/buildx/builder';
|
||||||
import {ConfigFile} from '@docker/actions-toolkit/lib/types/docker/docker';
|
import {ConfigFile} from '@docker/actions-toolkit/lib/types/docker/docker';
|
||||||
import axios, {AxiosError, AxiosInstance, AxiosResponse} from 'axios';
|
|
||||||
|
|
||||||
import * as context from './context';
|
import * as context from './context';
|
||||||
import {promisify} from 'util';
|
import {promisify} from 'util';
|
||||||
import {exec} from 'child_process';
|
import {exec} from 'child_process';
|
||||||
import * as TOML from '@iarna/toml';
|
import * as reporter from './reporter';
|
||||||
|
import {getBuilderAddr} from './setup_builder';
|
||||||
|
|
||||||
const buildxVersion = 'v0.17.0';
|
const buildxVersion = 'v0.17.0';
|
||||||
const mountPoint = '/var/lib/buildkit';
|
const mountPoint = '/var/lib/buildkit';
|
||||||
const execAsync = promisify(exec);
|
const execAsync = promisify(exec);
|
||||||
|
|
||||||
// Returns a client for the sticky disk manager on the agent on this host
|
|
||||||
async function getBlacksmithAgentClient(): Promise<AxiosInstance> {
|
|
||||||
const stickyDiskMgrUrl = 'http://192.168.127.1:5556';
|
|
||||||
return axios.create({
|
|
||||||
baseURL: stickyDiskMgrUrl
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
// Reports a successful build to the local sticky disk manager
|
|
||||||
async function reportBuildCompleted(exportRes?: ExportRecordResponse, blacksmithDockerBuildId?: string | null, buildRef?: string, dockerBuildDurationSeconds?: string, exposeId?: string) {
|
|
||||||
if (!blacksmithDockerBuildId) {
|
|
||||||
core.warning('No docker build ID found, skipping build completion report');
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
try {
|
|
||||||
const client = await getBlacksmithAgentClient();
|
|
||||||
const formData = new FormData();
|
|
||||||
formData.append('shouldCommit', 'true');
|
|
||||||
formData.append('vmID', process.env.VM_ID || '');
|
|
||||||
formData.append('exposeID', exposeId || '');
|
|
||||||
formData.append('stickyDiskKey', process.env.GITHUB_REPO_NAME || '');
|
|
||||||
const retryCondition = (error: AxiosError) => {
|
|
||||||
return error.response?.status ? error.response.status > 500 : false;
|
|
||||||
};
|
|
||||||
|
|
||||||
await postWithRetry(client, '/stickydisks', formData, retryCondition);
|
|
||||||
|
|
||||||
// Report success to Blacksmith API
|
|
||||||
const requestOptions = {
|
|
||||||
docker_build_id: blacksmithDockerBuildId,
|
|
||||||
conclusion: 'successful',
|
|
||||||
runtime_seconds: dockerBuildDurationSeconds
|
|
||||||
};
|
|
||||||
|
|
||||||
if (exportRes) {
|
|
||||||
let buildRefSummary;
|
|
||||||
// Extract just the ref ID from the full buildRef path
|
|
||||||
const refId = buildRef?.split('/').pop();
|
|
||||||
core.info(`Using buildRef ID: ${refId}`);
|
|
||||||
if (refId && exportRes.summaries[refId]) {
|
|
||||||
buildRefSummary = exportRes.summaries[refId];
|
|
||||||
} else {
|
|
||||||
// Take first summary if buildRef not found
|
|
||||||
const summaryKeys = Object.keys(exportRes.summaries);
|
|
||||||
if (summaryKeys.length > 0) {
|
|
||||||
buildRefSummary = exportRes.summaries[summaryKeys[0]];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (buildRefSummary) {
|
|
||||||
const cachedRatio = buildRefSummary.numCachedSteps / buildRefSummary.numTotalSteps;
|
|
||||||
requestOptions['cached_steps_ratio'] = cachedRatio;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
await postWithRetryToBlacksmithAPI(`/stickydisks/dockerbuilds/${blacksmithDockerBuildId}`, requestOptions, retryCondition);
|
|
||||||
return;
|
|
||||||
} catch (error) {
|
|
||||||
core.warning('Error reporting build completed:', error);
|
|
||||||
// We don't want to fail the build if this fails so we swallow the error
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Reports a failed build to both the local sticky disk manager and the Blacksmith API
|
|
||||||
async function reportBuildFailed(dockerBuildId: string | null, dockerBuildDurationSeconds?: string, exposeId?: string | null) {
|
|
||||||
if (!dockerBuildId) {
|
|
||||||
core.warning('No docker build ID found, skipping build completion report');
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
try {
|
|
||||||
const client = await getBlacksmithAgentClient();
|
|
||||||
const formData = new FormData();
|
|
||||||
formData.append('shouldCommit', 'false');
|
|
||||||
formData.append('vmID', process.env.VM_ID || '');
|
|
||||||
formData.append('exposeID', exposeId || '');
|
|
||||||
formData.append('stickyDiskKey', process.env.GITHUB_REPO_NAME || '');
|
|
||||||
const retryCondition = (error: AxiosError) => {
|
|
||||||
return error.response?.status ? error.response.status > 500 : false;
|
|
||||||
};
|
|
||||||
|
|
||||||
await postWithRetry(client, '/stickydisks', formData, retryCondition);
|
|
||||||
|
|
||||||
// Report failure to Blacksmith API
|
|
||||||
const requestOptions = {
|
|
||||||
docker_build_id: dockerBuildId,
|
|
||||||
conclusion: 'failed',
|
|
||||||
runtime_seconds: dockerBuildDurationSeconds
|
|
||||||
};
|
|
||||||
|
|
||||||
await postWithRetryToBlacksmithAPI(`/stickydisks/dockerbuilds/${dockerBuildId}`, requestOptions, retryCondition);
|
|
||||||
return;
|
|
||||||
} catch (error) {
|
|
||||||
core.warning('Error reporting build failed:', error);
|
|
||||||
// We don't want to fail the build if this fails so we swallow the error
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async function postWithRetryToBlacksmithAPI(url: string, requestBody: unknown, retryCondition: (error: AxiosError) => boolean): Promise<AxiosResponse> {
|
|
||||||
const maxRetries = 5;
|
|
||||||
const retryDelay = 100;
|
|
||||||
const apiUrl = process.env.BLACKSMITH_ENV?.includes('staging') ? 'https://stagingapi.blacksmith.sh' : 'https://api.blacksmith.sh';
|
|
||||||
|
|
||||||
for (let attempt = 1; attempt <= maxRetries; attempt++) {
|
|
||||||
try {
|
|
||||||
core.debug(`Request headers: Authorization: Bearer ${process.env.BLACKSMITH_STICKYDISK_TOKEN}, X-Github-Repo-Name: ${process.env.GITHUB_REPO_NAME || ''}`);
|
|
||||||
|
|
||||||
const fullUrl = `${apiUrl}${url}`;
|
|
||||||
core.debug(`Making request to full URL: ${fullUrl}`);
|
|
||||||
|
|
||||||
return await axios.post(fullUrl, requestBody, {
|
|
||||||
headers: {
|
|
||||||
Authorization: `Bearer ${process.env.BLACKSMITH_STICKYDISK_TOKEN}`,
|
|
||||||
'X-Github-Repo-Name': process.env.GITHUB_REPO_NAME || '',
|
|
||||||
'Content-Type': 'application/json'
|
|
||||||
}
|
|
||||||
});
|
|
||||||
} catch (error) {
|
|
||||||
if (attempt === maxRetries || !retryCondition(error as AxiosError)) {
|
|
||||||
throw error;
|
|
||||||
}
|
|
||||||
core.warning(`Request failed, retrying (${attempt}/${maxRetries})...`);
|
|
||||||
await new Promise(resolve => setTimeout(resolve, retryDelay));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
throw new Error('Max retries reached');
|
|
||||||
}
|
|
||||||
|
|
||||||
async function postWithRetry(client: AxiosInstance, url: string, formData: FormData, retryCondition: (error: AxiosError) => boolean): Promise<AxiosResponse> {
|
|
||||||
const maxRetries = 5;
|
|
||||||
const retryDelay = 100;
|
|
||||||
|
|
||||||
for (let attempt = 1; attempt <= maxRetries; attempt++) {
|
|
||||||
try {
|
|
||||||
return await client.post(url, formData, {
|
|
||||||
headers: {
|
|
||||||
Authorization: `Bearer ${process.env.BLACKSMITH_STICKYDISK_TOKEN}`,
|
|
||||||
'X-Github-Repo-Name': process.env.GITHUB_REPO_NAME || '',
|
|
||||||
'Content-Type': 'multipart/form-data'
|
|
||||||
}
|
|
||||||
});
|
|
||||||
} catch (error) {
|
|
||||||
if (attempt === maxRetries || !retryCondition(error as AxiosError)) {
|
|
||||||
throw error;
|
|
||||||
}
|
|
||||||
core.warning(`Request failed, retrying (${attempt}/${maxRetries})...`);
|
|
||||||
await new Promise(resolve => setTimeout(resolve, retryDelay));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
throw new Error('Max retries reached');
|
|
||||||
}
|
|
||||||
|
|
||||||
async function getWithRetry(client: AxiosInstance, url: string, formData: FormData | null, retryCondition: (error: AxiosError) => boolean, options?: {signal?: AbortSignal}): Promise<AxiosResponse> {
|
|
||||||
const maxRetries = 5;
|
|
||||||
const retryDelay = 100;
|
|
||||||
|
|
||||||
for (let attempt = 1; attempt <= maxRetries; attempt++) {
|
|
||||||
try {
|
|
||||||
if (formData) {
|
|
||||||
return await client.get(url, {
|
|
||||||
data: formData,
|
|
||||||
headers: {
|
|
||||||
Authorization: `Bearer ${process.env.BLACKSMITH_STICKYDISK_TOKEN}`,
|
|
||||||
'X-Github-Repo-Name': process.env.GITHUB_REPO_NAME || '',
|
|
||||||
'Content-Type': 'multipart/form-data'
|
|
||||||
},
|
|
||||||
signal: options?.signal
|
|
||||||
});
|
|
||||||
}
|
|
||||||
return await client.get(url, {signal: options?.signal});
|
|
||||||
} catch (error) {
|
|
||||||
if (attempt === maxRetries || !retryCondition(error as AxiosError)) {
|
|
||||||
throw error;
|
|
||||||
}
|
|
||||||
core.warning(`Request failed, retrying (${attempt}/${maxRetries})...`);
|
|
||||||
await new Promise(resolve => setTimeout(resolve, retryDelay));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
throw new Error('Max retries reached');
|
|
||||||
}
|
|
||||||
|
|
||||||
async function getStickyDisk(retryCondition: (error: AxiosError) => boolean, options?: {signal?: AbortSignal}): Promise<{expose_id: string; device: string}> {
|
|
||||||
const client = await getBlacksmithAgentClient();
|
|
||||||
const formData = new FormData();
|
|
||||||
// TODO(adityamaru): Support a stickydisk-per-build flag that will namespace the stickydisks by Dockerfile.
|
|
||||||
// For now, we'll use the repo name as the stickydisk key.
|
|
||||||
const repoName = process.env.GITHUB_REPO_NAME || '';
|
|
||||||
if (repoName === '') {
|
|
||||||
throw new Error('GITHUB_REPO_NAME is not set');
|
|
||||||
}
|
|
||||||
formData.append('stickyDiskKey', repoName);
|
|
||||||
formData.append('region', process.env.BLACKSMITH_REGION || 'eu-central');
|
|
||||||
formData.append('installationModelID', process.env.BLACKSMITH_INSTALLATION_MODEL_ID || '');
|
|
||||||
formData.append('vmID', process.env.VM_ID || '');
|
|
||||||
core.debug(`Getting sticky disk for ${repoName}`);
|
|
||||||
core.debug('FormData contents:');
|
|
||||||
for (const pair of formData.entries()) {
|
|
||||||
core.debug(`${pair[0]}: ${pair[1]}`);
|
|
||||||
}
|
|
||||||
const response = await getWithRetry(client, '/stickydisks', formData, retryCondition, options);
|
|
||||||
// For backward compatibility, if expose_id is set, return it
|
|
||||||
if (response.data?.expose_id && response.data?.disk_identifier) {
|
|
||||||
return {expose_id: response.data.expose_id, device: response.data.disk_identifier};
|
|
||||||
}
|
|
||||||
return {expose_id: '', device: ''};
|
|
||||||
}
|
|
||||||
|
|
||||||
async function getDiskSize(device: string): Promise<number> {
|
|
||||||
try {
|
|
||||||
const {stdout} = await execAsync(`sudo lsblk -b -n -o SIZE ${device}`);
|
|
||||||
const sizeInBytes = parseInt(stdout.trim(), 10);
|
|
||||||
if (isNaN(sizeInBytes)) {
|
|
||||||
throw new Error('Failed to parse disk size');
|
|
||||||
}
|
|
||||||
return sizeInBytes;
|
|
||||||
} catch (error) {
|
|
||||||
console.error(`Error getting disk size: ${error.message}`);
|
|
||||||
throw error;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async function writeBuildkitdTomlFile(parallelism: number, device: string): Promise<void> {
|
|
||||||
const diskSize = await getDiskSize(device);
|
|
||||||
const jsonConfig: TOML.JsonMap = {
|
|
||||||
root: '/var/lib/buildkit',
|
|
||||||
grpc: {
|
|
||||||
address: ['unix:///run/buildkit/buildkitd.sock']
|
|
||||||
},
|
|
||||||
registry: {
|
|
||||||
'docker.io': {
|
|
||||||
mirrors: ['http://192.168.127.1:5000'],
|
|
||||||
http: true,
|
|
||||||
insecure: true
|
|
||||||
},
|
|
||||||
'192.168.127.1:5000': {
|
|
||||||
http: true,
|
|
||||||
insecure: true
|
|
||||||
}
|
|
||||||
},
|
|
||||||
worker: {
|
|
||||||
oci: {
|
|
||||||
enabled: true,
|
|
||||||
gc: true,
|
|
||||||
gckeepstorage: diskSize.toString(),
|
|
||||||
'max-parallelism': parallelism,
|
|
||||||
snapshotter: 'overlayfs',
|
|
||||||
gcpolicy: [
|
|
||||||
{
|
|
||||||
all: true,
|
|
||||||
keepDuration: 1209600
|
|
||||||
},
|
|
||||||
{
|
|
||||||
all: true,
|
|
||||||
keepBytes: diskSize.toString()
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
containerd: {
|
|
||||||
enabled: false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const tomlString = TOML.stringify(jsonConfig);
|
|
||||||
|
|
||||||
try {
|
|
||||||
await fs.promises.writeFile('buildkitd.toml', tomlString);
|
|
||||||
core.debug(`TOML configuration is ${tomlString}`);
|
|
||||||
} catch (err) {
|
|
||||||
core.warning('error writing TOML configuration:', err);
|
|
||||||
throw err;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async function startBuildkitd(parallelism: number, device: string): Promise<string> {
|
|
||||||
try {
|
|
||||||
await writeBuildkitdTomlFile(parallelism, device);
|
|
||||||
await execAsync('sudo mkdir -p /run/buildkit');
|
|
||||||
await execAsync('sudo chmod 755 /run/buildkit');
|
|
||||||
const addr = 'unix:///run/buildkit/buildkitd.sock';
|
|
||||||
const {stdout: startStdout, stderr: startStderr} = await execAsync(
|
|
||||||
`sudo nohup buildkitd --debug --addr ${addr} --allow-insecure-entitlement security.insecure --config=buildkitd.toml --allow-insecure-entitlement network.host > buildkitd.log 2>&1 &`
|
|
||||||
);
|
|
||||||
|
|
||||||
if (startStderr) {
|
|
||||||
throw new Error(`error starting buildkitd service: ${startStderr}`);
|
|
||||||
}
|
|
||||||
core.debug(`buildkitd daemon started successfully ${startStdout}`);
|
|
||||||
|
|
||||||
const {stderr} = await execAsync(`pgrep -f buildkitd`);
|
|
||||||
if (stderr) {
|
|
||||||
throw new Error(`error finding buildkitd PID: ${stderr}`);
|
|
||||||
}
|
|
||||||
return addr;
|
|
||||||
} catch (error) {
|
|
||||||
core.error('failed to start buildkitd daemon:', error);
|
|
||||||
throw error;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Function to gracefully shut down the buildkitd process
|
|
||||||
async function shutdownBuildkitd(): Promise<void> {
|
|
||||||
try {
|
|
||||||
await execAsync(`sudo pkill -TERM buildkitd`);
|
|
||||||
} catch (error) {
|
|
||||||
core.error('error shutting down buildkitd process:', error);
|
|
||||||
throw error;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Function to get the number of available CPUs
|
|
||||||
async function getNumCPUs(): Promise<number> {
|
|
||||||
try {
|
|
||||||
const {stdout} = await execAsync('sudo nproc');
|
|
||||||
return parseInt(stdout.trim());
|
|
||||||
} catch (error) {
|
|
||||||
core.warning('Failed to get CPU count, defaulting to 1:', error);
|
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
async function maybeFormatBlockDevice(device: string): Promise<string> {
|
|
||||||
try {
|
|
||||||
// Check if device is formatted with ext4
|
|
||||||
try {
|
|
||||||
const {stdout} = await execAsync(`sudo blkid -o value -s TYPE ${device}`);
|
|
||||||
if (stdout.trim() === 'ext4') {
|
|
||||||
core.debug(`Device ${device} is already formatted with ext4`);
|
|
||||||
try {
|
|
||||||
// Run resize2fs to ensure filesystem uses full block device
|
|
||||||
await execAsync(`sudo resize2fs -f ${device}`);
|
|
||||||
core.debug(`Resized ext4 filesystem on ${device}`);
|
|
||||||
} catch (error) {
|
|
||||||
core.warning(`Error resizing ext4 filesystem on ${device}: ${error}`);
|
|
||||||
}
|
|
||||||
return device;
|
|
||||||
}
|
|
||||||
} catch (error) {
|
|
||||||
// blkid returns non-zero if no filesystem found, which is fine
|
|
||||||
core.debug(`No filesystem found on ${device}, will format it`);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Format device with ext4
|
|
||||||
core.debug(`Formatting device ${device} with ext4`);
|
|
||||||
await execAsync(`sudo mkfs.ext4 -m0 -Enodiscard,lazy_itable_init=1,lazy_journal_init=1 -F ${device}`);
|
|
||||||
core.debug(`Successfully formatted ${device} with ext4`);
|
|
||||||
return device;
|
|
||||||
} catch (error) {
|
|
||||||
core.error(`Failed to format device ${device}:`, error);
|
|
||||||
throw error;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// reportBuild reports the build to the Blacksmith API and returns the build ID
|
|
||||||
async function reportBuild(dockerfilePath: string) {
|
|
||||||
try {
|
|
||||||
const requestBody = {
|
|
||||||
dockerfile_path: dockerfilePath,
|
|
||||||
repo_name: process.env.GITHUB_REPO_NAME || '',
|
|
||||||
region: process.env.BLACKSMITH_REGION || 'eu-central',
|
|
||||||
arch: process.env.BLACKSMITH_ENV?.includes('arm') ? 'arm64' : 'amd64',
|
|
||||||
git_sha: process.env.GITHUB_SHA || '',
|
|
||||||
vm_id: process.env.VM_ID || '',
|
|
||||||
git_branch: process.env.GITHUB_REF_NAME || ''
|
|
||||||
};
|
|
||||||
core.debug(`Reporting build with options: ${JSON.stringify(requestBody, null, 2)}`);
|
|
||||||
const retryCondition = (error: AxiosError) => {
|
|
||||||
return error.response?.status ? error.response.status > 500 : false;
|
|
||||||
};
|
|
||||||
const response = await postWithRetryToBlacksmithAPI('/stickydisks/dockerbuilds', requestBody, retryCondition);
|
|
||||||
stateHelper.setBlacksmithDockerBuildId(response.data.docker_build_id);
|
|
||||||
return response.data;
|
|
||||||
} catch (error) {
|
|
||||||
const statusCode = (error as AxiosError)?.response?.status;
|
|
||||||
core.warning(`Error reporting build to Blacksmith API (status: ${statusCode || 'unknown'}):`);
|
|
||||||
core.warning(error);
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async function reportBuilderCreationFailed() {
|
|
||||||
const requestOptions = {
|
|
||||||
stickydisk_key: process.env.GITHUB_REPO_NAME || '',
|
|
||||||
repo_name: process.env.GITHUB_REPO_NAME || '',
|
|
||||||
region: process.env.BLACKSMITH_REGION || 'eu-central',
|
|
||||||
arch: process.env.BLACKSMITH_ENV?.includes('arm') ? 'arm64' : 'amd64',
|
|
||||||
vm_id: process.env.VM_ID || '',
|
|
||||||
petname: process.env.PETNAME || ''
|
|
||||||
};
|
|
||||||
const retryCondition = (error: AxiosError) => {
|
|
||||||
return error.response?.status ? error.response.status > 500 : false;
|
|
||||||
};
|
|
||||||
const response = await postWithRetryToBlacksmithAPI('/stickydisks/report-failed', requestOptions, retryCondition);
|
|
||||||
return response.data;
|
|
||||||
}
|
|
||||||
|
|
||||||
// getBuilderAddr mounts a sticky disk for the entity, sets up buildkitd on top of it
|
|
||||||
// and returns the address to the builder.
|
|
||||||
// If it is unable to do so because of a timeout or an error it returns null.
|
|
||||||
async function getBuilderAddr(inputs: context.Inputs, dockerfilePath: string): Promise<{addr: string | null; buildId?: string | null; exposeId: string}> {
|
|
||||||
try {
|
|
||||||
const retryCondition = (error: AxiosError) => (error.response?.status ? error.response.status >= 500 : error.code === 'ECONNRESET');
|
|
||||||
const controller = new AbortController();
|
|
||||||
const timeoutId = setTimeout(() => controller.abort(), 10000);
|
|
||||||
|
|
||||||
let buildResponse: {docker_build_id: string} | null = null;
|
|
||||||
let exposeId: string = '';
|
|
||||||
let device: string = '';
|
|
||||||
try {
|
|
||||||
const stickyDiskResponse = await getStickyDisk(retryCondition, {signal: controller.signal});
|
|
||||||
exposeId = stickyDiskResponse.expose_id;
|
|
||||||
device = stickyDiskResponse.device;
|
|
||||||
if (device === '') {
|
|
||||||
// TODO(adityamaru): Remove this once all of our VM agents are returning the device in the stickydisk response.
|
|
||||||
device = '/dev/vdb';
|
|
||||||
}
|
|
||||||
clearTimeout(timeoutId);
|
|
||||||
await maybeFormatBlockDevice(device);
|
|
||||||
buildResponse = await reportBuild(dockerfilePath);
|
|
||||||
await execAsync(`sudo mkdir -p ${mountPoint}`);
|
|
||||||
await execAsync(`sudo mount ${device} ${mountPoint}`);
|
|
||||||
core.debug(`${device} has been mounted to ${mountPoint}`);
|
|
||||||
} catch (error) {
|
|
||||||
if (error.name === 'AbortError') {
|
|
||||||
return {addr: null, exposeId: ''};
|
|
||||||
}
|
|
||||||
throw error;
|
|
||||||
}
|
|
||||||
|
|
||||||
core.debug('Successfully obtained sticky disk, proceeding to start buildkitd');
|
|
||||||
|
|
||||||
// Start buildkitd.
|
|
||||||
const parallelism = await getNumCPUs();
|
|
||||||
const buildkitdAddr = await startBuildkitd(parallelism, device);
|
|
||||||
core.debug(`buildkitd daemon started at addr ${buildkitdAddr}`);
|
|
||||||
// Change permissions on the buildkitd socket to allow non-root access
|
|
||||||
const startTime = Date.now();
|
|
||||||
const timeout = 5000; // 5 seconds in milliseconds
|
|
||||||
|
|
||||||
while (Date.now() - startTime < timeout) {
|
|
||||||
if (fs.existsSync('/run/buildkit/buildkitd.sock')) {
|
|
||||||
// Change permissions on the buildkitd socket to allow non-root access
|
|
||||||
await execAsync(`sudo chmod 666 /run/buildkit/buildkitd.sock`);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
await new Promise(resolve => setTimeout(resolve, 100)); // Poll every 100ms
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!fs.existsSync('/run/buildkit/buildkitd.sock')) {
|
|
||||||
throw new Error('buildkitd socket not found after 5s timeout');
|
|
||||||
}
|
|
||||||
return {addr: buildkitdAddr, buildId: buildResponse?.docker_build_id, exposeId: exposeId};
|
|
||||||
} catch (error) {
|
|
||||||
if ((error as AxiosError).response && (error as AxiosError).response!.status === 404) {
|
|
||||||
if (!inputs.nofallback) {
|
|
||||||
core.warning('No builder instances were available, falling back to a local build');
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
core.warning(`Error in getBuildkitdAddr: ${(error as Error).message}`);
|
|
||||||
}
|
|
||||||
return {addr: null, exposeId: ''};
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async function setupBuildx(version: string, toolkit: Toolkit): Promise<void> {
|
async function setupBuildx(version: string, toolkit: Toolkit): Promise<void> {
|
||||||
let toolPath;
|
let toolPath;
|
||||||
const standalone = await toolkit.buildx.isStandalone();
|
const standalone = await toolkit.buildx.isStandalone();
|
||||||
|
@ -516,6 +51,36 @@ async function setupBuildx(version: string, toolkit: Toolkit): Promise<void> {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Core logic for starting a Blacksmith builder
|
||||||
|
export async function startBlacksmithBuilder(inputs: context.Inputs): Promise<{addr: string | null; buildId: string | null; exposeId: string}> {
|
||||||
|
try {
|
||||||
|
const dockerfilePath = context.getDockerfilePath(inputs);
|
||||||
|
if (!dockerfilePath) {
|
||||||
|
throw new Error('Failed to resolve dockerfile path');
|
||||||
|
}
|
||||||
|
|
||||||
|
if (dockerfilePath && dockerfilePath.length > 0) {
|
||||||
|
core.debug(`Using dockerfile path: ${dockerfilePath}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
const {addr, buildId, exposeId} = await getBuilderAddr(inputs, dockerfilePath);
|
||||||
|
if (!addr) {
|
||||||
|
throw Error('Failed to obtain Blacksmith builder. Failing the build');
|
||||||
|
}
|
||||||
|
|
||||||
|
return {addr: addr || null, buildId: buildId || null, exposeId};
|
||||||
|
} catch (error) {
|
||||||
|
await reporter.reportBuilderCreationFailed(error);
|
||||||
|
if (inputs.nofallback) {
|
||||||
|
throw error;
|
||||||
|
} else {
|
||||||
|
console.log('coming to no fallback false');
|
||||||
|
core.warning(`Error during Blacksmith builder setup: ${error.message}. Falling back to a local build.`);
|
||||||
|
return {addr: null, buildId: null, exposeId: ''};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
actionsToolkit.run(
|
actionsToolkit.run(
|
||||||
// main
|
// main
|
||||||
async () => {
|
async () => {
|
||||||
|
@ -557,33 +122,7 @@ actionsToolkit.run(
|
||||||
exposeId: '' as string
|
exposeId: '' as string
|
||||||
};
|
};
|
||||||
await core.group(`Starting Blacksmith builder`, async () => {
|
await core.group(`Starting Blacksmith builder`, async () => {
|
||||||
const dockerfilePath = context.getDockerfilePath(inputs);
|
builderInfo = await startBlacksmithBuilder(inputs);
|
||||||
if (!dockerfilePath) {
|
|
||||||
if (inputs.nofallback) {
|
|
||||||
await reportBuilderCreationFailed();
|
|
||||||
throw Error('Failed to resolve dockerfile path, and fallback is disabled');
|
|
||||||
} else {
|
|
||||||
core.warning('Failed to resolve dockerfile path, and fallback is enabled. Falling back to a local build.');
|
|
||||||
}
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if (dockerfilePath && dockerfilePath.length > 0) {
|
|
||||||
core.debug(`Using dockerfile path: ${dockerfilePath}`);
|
|
||||||
}
|
|
||||||
const {addr, buildId, exposeId} = await getBuilderAddr(inputs, dockerfilePath);
|
|
||||||
builderInfo = {
|
|
||||||
addr: addr || null,
|
|
||||||
buildId: buildId || null,
|
|
||||||
exposeId: exposeId
|
|
||||||
};
|
|
||||||
if (!builderInfo.addr) {
|
|
||||||
await reportBuilderCreationFailed();
|
|
||||||
if (inputs.nofallback) {
|
|
||||||
throw Error('Failed to obtain Blacksmith builder. Failing the build');
|
|
||||||
} else {
|
|
||||||
core.warning('Failed to obtain Blacksmith builder address. Falling back to a local build.');
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
});
|
||||||
|
|
||||||
let buildError: Error | undefined;
|
let buildError: Error | undefined;
|
||||||
|
@ -778,7 +317,7 @@ actionsToolkit.run(
|
||||||
}
|
}
|
||||||
core.info('Unmounted device');
|
core.info('Unmounted device');
|
||||||
if (!buildError) {
|
if (!buildError) {
|
||||||
await reportBuildCompleted(exportRes, builderInfo.buildId, ref, buildDurationSeconds, builderInfo.exposeId);
|
await reporter.reportBuildCompleted(exportRes, builderInfo.buildId, ref, buildDurationSeconds, builderInfo.exposeId);
|
||||||
} else {
|
} else {
|
||||||
try {
|
try {
|
||||||
const buildkitdLog = fs.readFileSync('buildkitd.log', 'utf8');
|
const buildkitdLog = fs.readFileSync('buildkitd.log', 'utf8');
|
||||||
|
@ -787,7 +326,7 @@ actionsToolkit.run(
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
core.warning(`Failed to read buildkitd.log: ${error.message}`);
|
core.warning(`Failed to read buildkitd.log: ${error.message}`);
|
||||||
}
|
}
|
||||||
await reportBuildFailed(builderInfo.buildId, buildDurationSeconds, builderInfo.exposeId);
|
await reporter.reportBuildFailed(builderInfo.buildId, buildDurationSeconds, builderInfo.exposeId);
|
||||||
}
|
}
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
core.warning(`Error during Blacksmith builder shutdown: ${error.message}`);
|
core.warning(`Error during Blacksmith builder shutdown: ${error.message}`);
|
||||||
|
@ -882,3 +421,12 @@ function buildSummaryEnabled(): boolean {
|
||||||
}
|
}
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export async function shutdownBuildkitd(): Promise<void> {
|
||||||
|
try {
|
||||||
|
await execAsync(`sudo pkill -TERM buildkitd`);
|
||||||
|
} catch (error) {
|
||||||
|
core.error('error shutting down buildkitd process:', error);
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
218
src/reporter.ts
Normal file
218
src/reporter.ts
Normal file
|
@ -0,0 +1,218 @@
|
||||||
|
import * as core from '@actions/core';
|
||||||
|
import axios, {AxiosError, AxiosInstance, AxiosResponse} from 'axios';
|
||||||
|
import {ExportRecordResponse} from '@docker/actions-toolkit/lib/types/buildx/history';
|
||||||
|
import * as utils from './utils';
|
||||||
|
|
||||||
|
export async function reportBuilderCreationFailed(error?: Error) {
|
||||||
|
const requestOptions = {
|
||||||
|
stickydisk_key: process.env.GITHUB_REPO_NAME || '',
|
||||||
|
repo_name: process.env.GITHUB_REPO_NAME || '',
|
||||||
|
region: process.env.BLACKSMITH_REGION || 'eu-central',
|
||||||
|
arch: process.env.BLACKSMITH_ENV?.includes('arm') ? 'arm64' : 'amd64',
|
||||||
|
vm_id: process.env.VM_ID || '',
|
||||||
|
petname: process.env.PETNAME || ''
|
||||||
|
};
|
||||||
|
const retryCondition = (error: AxiosError) => {
|
||||||
|
return error.response?.status ? error.response.status > 500 : false;
|
||||||
|
};
|
||||||
|
const response = await postWithRetryToBlacksmithAPI('/stickydisks/report-failed', requestOptions, retryCondition);
|
||||||
|
return response.data;
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function reportBuildCompleted(exportRes?: ExportRecordResponse, blacksmithDockerBuildId?: string | null, buildRef?: string, dockerBuildDurationSeconds?: string, exposeId?: string): Promise<void> {
|
||||||
|
if (!blacksmithDockerBuildId) {
|
||||||
|
core.warning('No docker build ID found, skipping build completion report');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
const client = await utils.getBlacksmithAgentClient();
|
||||||
|
const formData = new FormData();
|
||||||
|
formData.append('shouldCommit', 'true');
|
||||||
|
formData.append('vmID', process.env.VM_ID || '');
|
||||||
|
formData.append('exposeID', exposeId || '');
|
||||||
|
formData.append('stickyDiskKey', process.env.GITHUB_REPO_NAME || '');
|
||||||
|
const retryCondition = (error: AxiosError) => {
|
||||||
|
return error.response?.status ? error.response.status > 500 : false;
|
||||||
|
};
|
||||||
|
|
||||||
|
await postWithRetry(client, '/stickydisks', formData, retryCondition);
|
||||||
|
|
||||||
|
// Report success to Blacksmith API
|
||||||
|
const requestOptions = {
|
||||||
|
docker_build_id: blacksmithDockerBuildId,
|
||||||
|
conclusion: 'successful',
|
||||||
|
runtime_seconds: dockerBuildDurationSeconds
|
||||||
|
};
|
||||||
|
|
||||||
|
if (exportRes) {
|
||||||
|
let buildRefSummary;
|
||||||
|
// Extract just the ref ID from the full buildRef path
|
||||||
|
const refId = buildRef?.split('/').pop();
|
||||||
|
core.info(`Using buildRef ID: ${refId}`);
|
||||||
|
if (refId && exportRes.summaries[refId]) {
|
||||||
|
buildRefSummary = exportRes.summaries[refId];
|
||||||
|
} else {
|
||||||
|
// Take first summary if buildRef not found
|
||||||
|
const summaryKeys = Object.keys(exportRes.summaries);
|
||||||
|
if (summaryKeys.length > 0) {
|
||||||
|
buildRefSummary = exportRes.summaries[summaryKeys[0]];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (buildRefSummary) {
|
||||||
|
const cachedRatio = buildRefSummary.numCachedSteps / buildRefSummary.numTotalSteps;
|
||||||
|
requestOptions['cached_steps_ratio'] = cachedRatio;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
await postWithRetryToBlacksmithAPI(`/stickydisks/dockerbuilds/${blacksmithDockerBuildId}`, requestOptions, retryCondition);
|
||||||
|
return;
|
||||||
|
} catch (error) {
|
||||||
|
core.warning('Error reporting build completed:', error);
|
||||||
|
// We don't want to fail the build if this fails so we swallow the error
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function reportBuildFailed(dockerBuildId: string | null, dockerBuildDurationSeconds?: string, exposeId?: string | null): Promise<void> {
|
||||||
|
if (!dockerBuildId) {
|
||||||
|
core.warning('No docker build ID found, skipping build completion report');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
const client = await utils.getBlacksmithAgentClient();
|
||||||
|
const formData = new FormData();
|
||||||
|
formData.append('shouldCommit', 'false');
|
||||||
|
formData.append('vmID', process.env.VM_ID || '');
|
||||||
|
formData.append('exposeID', exposeId || '');
|
||||||
|
formData.append('stickyDiskKey', process.env.GITHUB_REPO_NAME || '');
|
||||||
|
const retryCondition = (error: AxiosError) => {
|
||||||
|
return error.response?.status ? error.response.status > 500 : false;
|
||||||
|
};
|
||||||
|
|
||||||
|
await postWithRetry(client, '/stickydisks', formData, retryCondition);
|
||||||
|
|
||||||
|
// Report failure to Blacksmith API
|
||||||
|
const requestOptions = {
|
||||||
|
docker_build_id: dockerBuildId,
|
||||||
|
conclusion: 'failed',
|
||||||
|
runtime_seconds: dockerBuildDurationSeconds
|
||||||
|
};
|
||||||
|
|
||||||
|
await postWithRetryToBlacksmithAPI(`/stickydisks/dockerbuilds/${dockerBuildId}`, requestOptions, retryCondition);
|
||||||
|
return;
|
||||||
|
} catch (error) {
|
||||||
|
core.warning('Error reporting build failed:', error);
|
||||||
|
// We don't want to fail the build if this fails so we swallow the error
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// reportBuild reports the build to the Blacksmith API and returns the build ID
|
||||||
|
export async function reportBuild(dockerfilePath: string) {
|
||||||
|
try {
|
||||||
|
const requestBody = {
|
||||||
|
dockerfile_path: dockerfilePath,
|
||||||
|
repo_name: process.env.GITHUB_REPO_NAME || '',
|
||||||
|
region: process.env.BLACKSMITH_REGION || 'eu-central',
|
||||||
|
arch: process.env.BLACKSMITH_ENV?.includes('arm') ? 'arm64' : 'amd64',
|
||||||
|
git_sha: process.env.GITHUB_SHA || '',
|
||||||
|
vm_id: process.env.VM_ID || '',
|
||||||
|
git_branch: process.env.GITHUB_REF_NAME || ''
|
||||||
|
};
|
||||||
|
core.debug(`Reporting build with options: ${JSON.stringify(requestBody, null, 2)}`);
|
||||||
|
const retryCondition = (error: AxiosError) => {
|
||||||
|
return error.response?.status ? error.response.status > 500 : false;
|
||||||
|
};
|
||||||
|
const response = await postWithRetryToBlacksmithAPI('/stickydisks/dockerbuilds', requestBody, retryCondition);
|
||||||
|
return response.data;
|
||||||
|
} catch (error) {
|
||||||
|
const statusCode = (error as AxiosError)?.response?.status;
|
||||||
|
core.warning(`Error reporting build to Blacksmith API (status: ${statusCode || 'unknown'}):`);
|
||||||
|
core.warning(error);
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function postWithRetryToBlacksmithAPI(url: string, requestBody: unknown, retryCondition: (error: AxiosError) => boolean): Promise<AxiosResponse> {
|
||||||
|
const maxRetries = 5;
|
||||||
|
const retryDelay = 100;
|
||||||
|
const apiUrl = process.env.BLACKSMITH_ENV?.includes('staging') ? 'https://stagingapi.blacksmith.sh' : 'https://api.blacksmith.sh';
|
||||||
|
|
||||||
|
for (let attempt = 1; attempt <= maxRetries; attempt++) {
|
||||||
|
try {
|
||||||
|
core.debug(`Request headers: Authorization: Bearer ${process.env.BLACKSMITH_STICKYDISK_TOKEN}, X-Github-Repo-Name: ${process.env.GITHUB_REPO_NAME || ''}`);
|
||||||
|
|
||||||
|
const fullUrl = `${apiUrl}${url}`;
|
||||||
|
core.debug(`Making request to full URL: ${fullUrl}`);
|
||||||
|
|
||||||
|
return await axios.post(fullUrl, requestBody, {
|
||||||
|
headers: {
|
||||||
|
Authorization: `Bearer ${process.env.BLACKSMITH_STICKYDISK_TOKEN}`,
|
||||||
|
'X-Github-Repo-Name': process.env.GITHUB_REPO_NAME || '',
|
||||||
|
'Content-Type': 'application/json'
|
||||||
|
}
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
if (attempt === maxRetries || !retryCondition(error as AxiosError)) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
core.warning(`Request failed, retrying (${attempt}/${maxRetries})...`);
|
||||||
|
await new Promise(resolve => setTimeout(resolve, retryDelay));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
throw new Error('Max retries reached');
|
||||||
|
}
|
||||||
|
|
||||||
|
async function postWithRetry(client: AxiosInstance, url: string, formData: FormData, retryCondition: (error: AxiosError) => boolean): Promise<AxiosResponse> {
|
||||||
|
const maxRetries = 5;
|
||||||
|
const retryDelay = 100;
|
||||||
|
|
||||||
|
for (let attempt = 1; attempt <= maxRetries; attempt++) {
|
||||||
|
try {
|
||||||
|
return await client.post(url, formData, {
|
||||||
|
headers: {
|
||||||
|
Authorization: `Bearer ${process.env.BLACKSMITH_STICKYDISK_TOKEN}`,
|
||||||
|
'X-Github-Repo-Name': process.env.GITHUB_REPO_NAME || '',
|
||||||
|
'Content-Type': 'multipart/form-data'
|
||||||
|
}
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
if (attempt === maxRetries || !retryCondition(error as AxiosError)) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
core.warning(`Request failed, retrying (${attempt}/${maxRetries})...`);
|
||||||
|
await new Promise(resolve => setTimeout(resolve, retryDelay));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
throw new Error('Max retries reached');
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function getWithRetry(client: AxiosInstance, url: string, formData: FormData | null, retryCondition: (error: AxiosError) => boolean, options?: {signal?: AbortSignal}): Promise<AxiosResponse> {
|
||||||
|
const maxRetries = 5;
|
||||||
|
const retryDelay = 100;
|
||||||
|
|
||||||
|
for (let attempt = 1; attempt <= maxRetries; attempt++) {
|
||||||
|
try {
|
||||||
|
if (formData) {
|
||||||
|
return await client.get(url, {
|
||||||
|
data: formData,
|
||||||
|
headers: {
|
||||||
|
Authorization: `Bearer ${process.env.BLACKSMITH_STICKYDISK_TOKEN}`,
|
||||||
|
'X-Github-Repo-Name': process.env.GITHUB_REPO_NAME || '',
|
||||||
|
'Content-Type': 'multipart/form-data'
|
||||||
|
},
|
||||||
|
signal: options?.signal
|
||||||
|
});
|
||||||
|
}
|
||||||
|
return await client.get(url, {signal: options?.signal});
|
||||||
|
} catch (error) {
|
||||||
|
if (attempt === maxRetries || !retryCondition(error as AxiosError)) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
core.warning(`Request failed, retrying (${attempt}/${maxRetries})...`);
|
||||||
|
await new Promise(resolve => setTimeout(resolve, retryDelay));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
throw new Error('Max retries reached');
|
||||||
|
}
|
242
src/setup_builder.ts
Normal file
242
src/setup_builder.ts
Normal file
|
@ -0,0 +1,242 @@
|
||||||
|
import * as fs from 'fs';
|
||||||
|
import * as core from '@actions/core';
|
||||||
|
import {AxiosError} from 'axios';
|
||||||
|
import {exec} from 'child_process';
|
||||||
|
import {promisify} from 'util';
|
||||||
|
import * as TOML from '@iarna/toml';
|
||||||
|
import {Inputs} from './context';
|
||||||
|
import * as reporter from './reporter';
|
||||||
|
import * as utils from './utils';
|
||||||
|
|
||||||
|
const mountPoint = '/var/lib/buildkit';
|
||||||
|
const execAsync = promisify(exec);
|
||||||
|
|
||||||
|
async function maybeFormatBlockDevice(device: string): Promise<string> {
|
||||||
|
try {
|
||||||
|
// Check if device is formatted with ext4
|
||||||
|
try {
|
||||||
|
const {stdout} = await execAsync(`sudo blkid -o value -s TYPE ${device}`);
|
||||||
|
if (stdout.trim() === 'ext4') {
|
||||||
|
core.debug(`Device ${device} is already formatted with ext4`);
|
||||||
|
try {
|
||||||
|
// Run resize2fs to ensure filesystem uses full block device
|
||||||
|
await execAsync(`sudo resize2fs -f ${device}`);
|
||||||
|
core.debug(`Resized ext4 filesystem on ${device}`);
|
||||||
|
} catch (error) {
|
||||||
|
core.warning(`Error resizing ext4 filesystem on ${device}: ${error}`);
|
||||||
|
}
|
||||||
|
return device;
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
// blkid returns non-zero if no filesystem found, which is fine
|
||||||
|
core.debug(`No filesystem found on ${device}, will format it`);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Format device with ext4
|
||||||
|
core.debug(`Formatting device ${device} with ext4`);
|
||||||
|
await execAsync(`sudo mkfs.ext4 -m0 -Enodiscard,lazy_itable_init=1,lazy_journal_init=1 -F ${device}`);
|
||||||
|
core.debug(`Successfully formatted ${device} with ext4`);
|
||||||
|
return device;
|
||||||
|
} catch (error) {
|
||||||
|
core.error(`Failed to format device ${device}:`, error);
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function getNumCPUs(): Promise<number> {
|
||||||
|
try {
|
||||||
|
const {stdout} = await execAsync('sudo nproc');
|
||||||
|
return parseInt(stdout.trim());
|
||||||
|
} catch (error) {
|
||||||
|
core.warning('Failed to get CPU count, defaulting to 1:', error);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function writeBuildkitdTomlFile(parallelism: number, device: string): Promise<void> {
|
||||||
|
const diskSize = await getDiskSize(device);
|
||||||
|
const jsonConfig: TOML.JsonMap = {
|
||||||
|
root: '/var/lib/buildkit',
|
||||||
|
grpc: {
|
||||||
|
address: ['unix:///run/buildkit/buildkitd.sock']
|
||||||
|
},
|
||||||
|
registry: {
|
||||||
|
'docker.io': {
|
||||||
|
mirrors: ['http://192.168.127.1:5000'],
|
||||||
|
http: true,
|
||||||
|
insecure: true
|
||||||
|
},
|
||||||
|
'192.168.127.1:5000': {
|
||||||
|
http: true,
|
||||||
|
insecure: true
|
||||||
|
}
|
||||||
|
},
|
||||||
|
worker: {
|
||||||
|
oci: {
|
||||||
|
enabled: true,
|
||||||
|
gc: true,
|
||||||
|
gckeepstorage: diskSize.toString(),
|
||||||
|
'max-parallelism': parallelism,
|
||||||
|
snapshotter: 'overlayfs',
|
||||||
|
gcpolicy: [
|
||||||
|
{
|
||||||
|
all: true,
|
||||||
|
keepDuration: 1209600
|
||||||
|
},
|
||||||
|
{
|
||||||
|
all: true,
|
||||||
|
keepBytes: diskSize.toString()
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
containerd: {
|
||||||
|
enabled: false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const tomlString = TOML.stringify(jsonConfig);
|
||||||
|
|
||||||
|
try {
|
||||||
|
await fs.promises.writeFile('buildkitd.toml', tomlString);
|
||||||
|
core.debug(`TOML configuration is ${tomlString}`);
|
||||||
|
} catch (err) {
|
||||||
|
core.warning('error writing TOML configuration:', err);
|
||||||
|
throw err;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function startBuildkitd(parallelism: number, device: string): Promise<string> {
|
||||||
|
try {
|
||||||
|
await writeBuildkitdTomlFile(parallelism, device);
|
||||||
|
await execAsync('sudo mkdir -p /run/buildkit');
|
||||||
|
await execAsync('sudo chmod 755 /run/buildkit');
|
||||||
|
const addr = 'unix:///run/buildkit/buildkitd.sock';
|
||||||
|
const {stdout: startStdout, stderr: startStderr} = await execAsync(
|
||||||
|
`sudo nohup buildkitd --debug --addr ${addr} --allow-insecure-entitlement security.insecure --config=buildkitd.toml --allow-insecure-entitlement network.host > buildkitd.log 2>&1 &`
|
||||||
|
);
|
||||||
|
|
||||||
|
if (startStderr) {
|
||||||
|
throw new Error(`error starting buildkitd service: ${startStderr}`);
|
||||||
|
}
|
||||||
|
core.debug(`buildkitd daemon started successfully ${startStdout}`);
|
||||||
|
|
||||||
|
const {stderr} = await execAsync(`pgrep -f buildkitd`);
|
||||||
|
if (stderr) {
|
||||||
|
throw new Error(`error finding buildkitd PID: ${stderr}`);
|
||||||
|
}
|
||||||
|
return addr;
|
||||||
|
} catch (error) {
|
||||||
|
core.error('failed to start buildkitd daemon:', error);
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function getDiskSize(device: string): Promise<number> {
|
||||||
|
try {
|
||||||
|
const {stdout} = await execAsync(`sudo lsblk -b -n -o SIZE ${device}`);
|
||||||
|
const sizeInBytes = parseInt(stdout.trim(), 10);
|
||||||
|
if (isNaN(sizeInBytes)) {
|
||||||
|
throw new Error('Failed to parse disk size');
|
||||||
|
}
|
||||||
|
return sizeInBytes;
|
||||||
|
} catch (error) {
|
||||||
|
console.error(`Error getting disk size: ${error.message}`);
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function getStickyDisk(retryCondition: (error: AxiosError) => boolean, options?: {signal?: AbortSignal}): Promise<{expose_id: string; device: string}> {
|
||||||
|
const client = await utils.getBlacksmithAgentClient();
|
||||||
|
const formData = new FormData();
|
||||||
|
// TODO(adityamaru): Support a stickydisk-per-build flag that will namespace the stickydisks by Dockerfile.
|
||||||
|
// For now, we'll use the repo name as the stickydisk key.
|
||||||
|
const repoName = process.env.GITHUB_REPO_NAME || '';
|
||||||
|
if (repoName === '') {
|
||||||
|
throw new Error('GITHUB_REPO_NAME is not set');
|
||||||
|
}
|
||||||
|
formData.append('stickyDiskKey', repoName);
|
||||||
|
formData.append('region', process.env.BLACKSMITH_REGION || 'eu-central');
|
||||||
|
formData.append('installationModelID', process.env.BLACKSMITH_INSTALLATION_MODEL_ID || '');
|
||||||
|
formData.append('vmID', process.env.VM_ID || '');
|
||||||
|
core.debug(`Getting sticky disk for ${repoName}`);
|
||||||
|
core.debug('FormData contents:');
|
||||||
|
for (const pair of formData.entries()) {
|
||||||
|
core.debug(`${pair[0]}: ${pair[1]}`);
|
||||||
|
}
|
||||||
|
const response = await reporter.getWithRetry(client, '/stickydisks', formData, retryCondition, options);
|
||||||
|
// For backward compatibility, if expose_id is set, return it
|
||||||
|
if (response.data?.expose_id && response.data?.disk_identifier) {
|
||||||
|
return {expose_id: response.data.expose_id, device: response.data.disk_identifier};
|
||||||
|
}
|
||||||
|
return {expose_id: '', device: ''};
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// getBuilderAddr mounts a sticky disk for the entity, sets up buildkitd on top of it
|
||||||
|
// and returns the address to the builder.
|
||||||
|
// If it is unable to do so because of a timeout or an error it returns null.
|
||||||
|
export async function getBuilderAddr(inputs: Inputs, dockerfilePath: string): Promise<{addr: string | null; buildId?: string | null; exposeId: string}> {
|
||||||
|
try {
|
||||||
|
const retryCondition = (error: AxiosError) => (error.response?.status ? error.response.status >= 500 : error.code === 'ECONNRESET');
|
||||||
|
const controller = new AbortController();
|
||||||
|
const timeoutId = setTimeout(() => controller.abort(), 10000);
|
||||||
|
|
||||||
|
let buildResponse: {docker_build_id: string} | null = null;
|
||||||
|
let exposeId: string = '';
|
||||||
|
let device: string = '';
|
||||||
|
try {
|
||||||
|
const stickyDiskResponse = await getStickyDisk(retryCondition, {signal: controller.signal});
|
||||||
|
exposeId = stickyDiskResponse.expose_id;
|
||||||
|
device = stickyDiskResponse.device;
|
||||||
|
if (device === '') {
|
||||||
|
// TODO(adityamaru): Remove this once all of our VM agents are returning the device in the stickydisk response.
|
||||||
|
device = '/dev/vdb';
|
||||||
|
}
|
||||||
|
clearTimeout(timeoutId);
|
||||||
|
await maybeFormatBlockDevice(device);
|
||||||
|
buildResponse = await reporter.reportBuild(dockerfilePath);
|
||||||
|
await execAsync(`sudo mkdir -p ${mountPoint}`);
|
||||||
|
await execAsync(`sudo mount ${device} ${mountPoint}`);
|
||||||
|
core.debug(`${device} has been mounted to ${mountPoint}`);
|
||||||
|
} catch (error) {
|
||||||
|
if (error.name === 'AbortError') {
|
||||||
|
return {addr: null, exposeId: ''};
|
||||||
|
}
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
|
||||||
|
core.debug('Successfully obtained sticky disk, proceeding to start buildkitd');
|
||||||
|
|
||||||
|
// Start buildkitd.
|
||||||
|
const parallelism = await getNumCPUs();
|
||||||
|
const buildkitdAddr = await startBuildkitd(parallelism, device);
|
||||||
|
core.debug(`buildkitd daemon started at addr ${buildkitdAddr}`);
|
||||||
|
// Change permissions on the buildkitd socket to allow non-root access
|
||||||
|
const startTime = Date.now();
|
||||||
|
const timeout = 5000; // 5 seconds in milliseconds
|
||||||
|
|
||||||
|
while (Date.now() - startTime < timeout) {
|
||||||
|
if (fs.existsSync('/run/buildkit/buildkitd.sock')) {
|
||||||
|
// Change permissions on the buildkitd socket to allow non-root access
|
||||||
|
await execAsync(`sudo chmod 666 /run/buildkit/buildkitd.sock`);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
await new Promise(resolve => setTimeout(resolve, 100)); // Poll every 100ms
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!fs.existsSync('/run/buildkit/buildkitd.sock')) {
|
||||||
|
throw new Error('buildkitd socket not found after 5s timeout');
|
||||||
|
}
|
||||||
|
return {addr: buildkitdAddr, buildId: buildResponse?.docker_build_id, exposeId: exposeId};
|
||||||
|
} catch (error) {
|
||||||
|
if ((error as AxiosError).response && (error as AxiosError).response!.status === 404) {
|
||||||
|
if (!inputs.nofallback) {
|
||||||
|
core.warning('No builder instances were available, falling back to a local build');
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
core.warning(`Error in getBuildkitdAddr: ${(error as Error).message}`);
|
||||||
|
}
|
||||||
|
return {addr: null, exposeId: ''};
|
||||||
|
}
|
||||||
|
}
|
59
src/utils.ts
Normal file
59
src/utils.ts
Normal file
|
@ -0,0 +1,59 @@
|
||||||
|
import axios, {AxiosError, AxiosInstance, AxiosResponse} from 'axios';
|
||||||
|
import * as core from '@actions/core';
|
||||||
|
|
||||||
|
export async function getBlacksmithAgentClient(): Promise<AxiosInstance> {
|
||||||
|
const stickyDiskMgrUrl = 'http://192.168.127.1:5556';
|
||||||
|
return axios.create({
|
||||||
|
baseURL: stickyDiskMgrUrl
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function postWithRetry(client: AxiosInstance, url: string, formData: FormData, retryCondition: (error: AxiosError) => boolean): Promise<AxiosResponse> {
|
||||||
|
const maxRetries = 5;
|
||||||
|
const retryDelay = 100;
|
||||||
|
|
||||||
|
for (let attempt = 1; attempt <= maxRetries; attempt++) {
|
||||||
|
try {
|
||||||
|
return await client.post(url, formData, {
|
||||||
|
headers: {
|
||||||
|
Authorization: `Bearer ${process.env.BLACKSMITH_STICKYDISK_TOKEN}`,
|
||||||
|
'X-Github-Repo-Name': process.env.GITHUB_REPO_NAME || '',
|
||||||
|
'Content-Type': 'multipart/form-data'
|
||||||
|
}
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
if (attempt === maxRetries || !retryCondition(error as AxiosError)) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
core.warning(`Request failed, retrying (${attempt}/${maxRetries})...`);
|
||||||
|
await new Promise(resolve => setTimeout(resolve, retryDelay));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
throw new Error('Max retries reached');
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function postWithRetryToBlacksmithAPI(url: string, requestBody: unknown, retryCondition: (error: AxiosError) => boolean): Promise<AxiosResponse> {
|
||||||
|
const maxRetries = 5;
|
||||||
|
const retryDelay = 100;
|
||||||
|
const apiUrl = process.env.BLACKSMITH_ENV?.includes('staging') ? 'https://stagingapi.blacksmith.sh' : 'https://api.blacksmith.sh';
|
||||||
|
|
||||||
|
for (let attempt = 1; attempt <= maxRetries; attempt++) {
|
||||||
|
try {
|
||||||
|
const fullUrl = `${apiUrl}${url}`;
|
||||||
|
return await axios.post(fullUrl, requestBody, {
|
||||||
|
headers: {
|
||||||
|
Authorization: `Bearer ${process.env.BLACKSMITH_STICKYDISK_TOKEN}`,
|
||||||
|
'X-Github-Repo-Name': process.env.GITHUB_REPO_NAME || '',
|
||||||
|
'Content-Type': 'application/json'
|
||||||
|
}
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
if (attempt === maxRetries || !retryCondition(error as AxiosError)) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
core.warning(`Request failed, retrying (${attempt}/${maxRetries})...`);
|
||||||
|
await new Promise(resolve => setTimeout(resolve, retryDelay));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
throw new Error('Max retries reached');
|
||||||
|
}
|
|
@ -12,6 +12,9 @@
|
||||||
"resolveJsonModule": true,
|
"resolveJsonModule": true,
|
||||||
"useUnknownInCatchVariables": false,
|
"useUnknownInCatchVariables": false,
|
||||||
},
|
},
|
||||||
|
"include": [
|
||||||
|
"src/**/*"
|
||||||
|
],
|
||||||
"exclude": [
|
"exclude": [
|
||||||
"./__mocks__/**/*",
|
"./__mocks__/**/*",
|
||||||
"./__tests__/**/*",
|
"./__tests__/**/*",
|
||||||
|
|
14
yarn.lock
14
yarn.lock
|
@ -1254,6 +1254,14 @@
|
||||||
dependencies:
|
dependencies:
|
||||||
"@types/istanbul-lib-report" "*"
|
"@types/istanbul-lib-report" "*"
|
||||||
|
|
||||||
|
"@types/jest@^29.5.14":
|
||||||
|
version "29.5.14"
|
||||||
|
resolved "https://registry.npmjs.org/@types/jest/-/jest-29.5.14.tgz"
|
||||||
|
integrity sha512-ZN+4sdnLUbo8EVvVc2ao0GFW6oVrQRPn4K2lglySj7APvSrgzxHiNNK99us4WDMi57xxA2yggblIAMNhXOotLQ==
|
||||||
|
dependencies:
|
||||||
|
expect "^29.0.0"
|
||||||
|
pretty-format "^29.0.0"
|
||||||
|
|
||||||
"@types/node@*", "@types/node@^20.12.12":
|
"@types/node@*", "@types/node@^20.12.12":
|
||||||
version "20.16.5"
|
version "20.16.5"
|
||||||
resolved "https://registry.npmjs.org/@types/node/-/node-20.16.5.tgz"
|
resolved "https://registry.npmjs.org/@types/node/-/node-20.16.5.tgz"
|
||||||
|
@ -2212,7 +2220,7 @@ exit@^0.1.2:
|
||||||
resolved "https://registry.npmjs.org/exit/-/exit-0.1.2.tgz"
|
resolved "https://registry.npmjs.org/exit/-/exit-0.1.2.tgz"
|
||||||
integrity sha512-Zk/eNKV2zbjpKzrsQ+n1G6poVbErQxJ0LBOJXaKZ1EViLzH+hrLu9cdXI4zw9dBQJslwBEpbQ2P1oS7nDxs6jQ==
|
integrity sha512-Zk/eNKV2zbjpKzrsQ+n1G6poVbErQxJ0LBOJXaKZ1EViLzH+hrLu9cdXI4zw9dBQJslwBEpbQ2P1oS7nDxs6jQ==
|
||||||
|
|
||||||
expect@^29.7.0:
|
expect@^29.0.0, expect@^29.7.0:
|
||||||
version "29.7.0"
|
version "29.7.0"
|
||||||
resolved "https://registry.npmjs.org/expect/-/expect-29.7.0.tgz"
|
resolved "https://registry.npmjs.org/expect/-/expect-29.7.0.tgz"
|
||||||
integrity sha512-2Zks0hf1VLFYI1kbh0I5jP3KHHyCHpkfyHBzsSXRFgl/Bg9mWYfMW8oD+PdMPlEwy5HNsR9JutYy6pMeOh61nw==
|
integrity sha512-2Zks0hf1VLFYI1kbh0I5jP3KHHyCHpkfyHBzsSXRFgl/Bg9mWYfMW8oD+PdMPlEwy5HNsR9JutYy6pMeOh61nw==
|
||||||
|
@ -3611,7 +3619,7 @@ prettier@^3.2.5, prettier@>=3.0.0:
|
||||||
resolved "https://registry.npmjs.org/prettier/-/prettier-3.3.3.tgz"
|
resolved "https://registry.npmjs.org/prettier/-/prettier-3.3.3.tgz"
|
||||||
integrity sha512-i2tDNA0O5IrMO757lfrdQZCc2jPNDVntV0m/+4whiDfWaTKfMNgR7Qz0NAeGz/nRqF4m5/6CLzbP4/liHt12Ew==
|
integrity sha512-i2tDNA0O5IrMO757lfrdQZCc2jPNDVntV0m/+4whiDfWaTKfMNgR7Qz0NAeGz/nRqF4m5/6CLzbP4/liHt12Ew==
|
||||||
|
|
||||||
pretty-format@^29.7.0:
|
pretty-format@^29.0.0, pretty-format@^29.7.0:
|
||||||
version "29.7.0"
|
version "29.7.0"
|
||||||
resolved "https://registry.npmjs.org/pretty-format/-/pretty-format-29.7.0.tgz"
|
resolved "https://registry.npmjs.org/pretty-format/-/pretty-format-29.7.0.tgz"
|
||||||
integrity sha512-Pdlw/oPxN+aXdmM9R00JVC9WVFoCLTKJvDVLgmJ+qAffBMxsV85l/Lu7sNx4zSzPyoL2euImuEwHhOXdEgNFZQ==
|
integrity sha512-Pdlw/oPxN+aXdmM9R00JVC9WVFoCLTKJvDVLgmJ+qAffBMxsV85l/Lu7sNx4zSzPyoL2euImuEwHhOXdEgNFZQ==
|
||||||
|
@ -4123,7 +4131,7 @@ ts-api-utils@^1.3.0:
|
||||||
resolved "https://registry.npmjs.org/ts-api-utils/-/ts-api-utils-1.3.0.tgz"
|
resolved "https://registry.npmjs.org/ts-api-utils/-/ts-api-utils-1.3.0.tgz"
|
||||||
integrity sha512-UQMIo7pb8WRomKR1/+MFVLTroIvDVtMX3K6OUir8ynLyzB8Jeriont2bTAtmNPa1ekAgN7YPDyf6V+ygrdU+eQ==
|
integrity sha512-UQMIo7pb8WRomKR1/+MFVLTroIvDVtMX3K6OUir8ynLyzB8Jeriont2bTAtmNPa1ekAgN7YPDyf6V+ygrdU+eQ==
|
||||||
|
|
||||||
ts-jest@^29.1.2:
|
ts-jest@^29.2.5:
|
||||||
version "29.2.5"
|
version "29.2.5"
|
||||||
resolved "https://registry.npmjs.org/ts-jest/-/ts-jest-29.2.5.tgz"
|
resolved "https://registry.npmjs.org/ts-jest/-/ts-jest-29.2.5.tgz"
|
||||||
integrity sha512-KD8zB2aAZrcKIdGk4OwpJggeLcH1FgrICqDSROWqlnJXGCXK4Mn6FcdK2B6670Xr73lHMG1kHw8R87A0ecZ+vA==
|
integrity sha512-KD8zB2aAZrcKIdGk4OwpJggeLcH1FgrICqDSROWqlnJXGCXK4Mn6FcdK2B6670Xr73lHMG1kHw8R87A0ecZ+vA==
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue