2020-09-02 16:07:11 +08:00
|
|
|
import * as fs from 'fs';
|
2023-09-08 21:28:08 +08:00
|
|
|
import * as path from 'path';
|
2020-09-02 16:07:11 +08:00
|
|
|
import * as stateHelper from './state-helper';
|
2020-08-16 06:36:41 +08:00
|
|
|
import * as core from '@actions/core';
|
2023-02-20 18:11:15 +08:00
|
|
|
import * as actionsToolkit from '@docker/actions-toolkit';
|
2024-04-26 17:20:49 +08:00
|
|
|
|
2024-05-02 19:49:01 +08:00
|
|
|
import {Buildx} from '@docker/actions-toolkit/lib/buildx/buildx';
|
|
|
|
import {History as BuildxHistory} from '@docker/actions-toolkit/lib/buildx/history';
|
2023-02-20 18:11:15 +08:00
|
|
|
import {Context} from '@docker/actions-toolkit/lib/context';
|
2023-03-13 06:34:24 +08:00
|
|
|
import {Docker} from '@docker/actions-toolkit/lib/docker/docker';
|
2023-02-20 18:11:15 +08:00
|
|
|
import {Exec} from '@docker/actions-toolkit/lib/exec';
|
|
|
|
import {GitHub} from '@docker/actions-toolkit/lib/github';
|
|
|
|
import {Toolkit} from '@docker/actions-toolkit/lib/toolkit';
|
2024-05-02 19:49:01 +08:00
|
|
|
import {Util} from '@docker/actions-toolkit/lib/util';
|
2024-04-26 17:20:49 +08:00
|
|
|
|
2024-06-27 17:26:35 +08:00
|
|
|
import {BuilderInfo} from '@docker/actions-toolkit/lib/types/buildx/builder';
|
2024-05-27 18:45:25 +08:00
|
|
|
import {ConfigFile} from '@docker/actions-toolkit/lib/types/docker/docker';
|
2024-07-03 00:07:54 +08:00
|
|
|
import {UploadArtifactResponse} from '@docker/actions-toolkit/lib/types/github';
|
2020-08-16 06:36:41 +08:00
|
|
|
|
2023-02-20 18:11:15 +08:00
|
|
|
import * as context from './context';
|
2022-04-28 15:31:47 +08:00
|
|
|
|
2023-02-20 18:11:15 +08:00
|
|
|
actionsToolkit.run(
|
|
|
|
// main
|
|
|
|
async () => {
|
2024-05-02 19:49:01 +08:00
|
|
|
const startedTime = new Date();
|
2023-04-17 07:32:21 +08:00
|
|
|
const inputs: context.Inputs = await context.getInputs();
|
2024-03-06 21:20:33 +08:00
|
|
|
core.debug(`inputs: ${JSON.stringify(inputs)}`);
|
2024-05-02 19:49:01 +08:00
|
|
|
stateHelper.setInputs(inputs);
|
2024-03-06 21:20:33 +08:00
|
|
|
|
2023-02-20 18:11:15 +08:00
|
|
|
const toolkit = new Toolkit();
|
2022-04-28 15:31:47 +08:00
|
|
|
|
2023-02-20 18:11:15 +08:00
|
|
|
await core.group(`GitHub Actions runtime token ACs`, async () => {
|
|
|
|
try {
|
|
|
|
await GitHub.printActionsRuntimeTokenACs();
|
|
|
|
} catch (e) {
|
|
|
|
core.warning(e.message);
|
2023-01-13 02:44:15 +08:00
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2023-02-20 18:11:15 +08:00
|
|
|
await core.group(`Docker info`, async () => {
|
|
|
|
try {
|
|
|
|
await Docker.printVersion();
|
|
|
|
await Docker.printInfo();
|
|
|
|
} catch (e) {
|
|
|
|
core.info(e.message);
|
|
|
|
}
|
|
|
|
});
|
2020-08-16 06:36:41 +08:00
|
|
|
|
2023-09-08 21:28:08 +08:00
|
|
|
await core.group(`Proxy configuration`, async () => {
|
|
|
|
let dockerConfig: ConfigFile | undefined;
|
|
|
|
let dockerConfigMalformed = false;
|
|
|
|
try {
|
|
|
|
dockerConfig = await Docker.configFile();
|
|
|
|
} catch (e) {
|
|
|
|
dockerConfigMalformed = true;
|
|
|
|
core.warning(`Unable to parse config file ${path.join(Docker.configDir, 'config.json')}: ${e}`);
|
|
|
|
}
|
|
|
|
if (dockerConfig && dockerConfig.proxies) {
|
2023-06-13 17:46:07 +08:00
|
|
|
for (const host in dockerConfig.proxies) {
|
|
|
|
let prefix = '';
|
2023-09-07 18:21:01 +08:00
|
|
|
if (Object.keys(dockerConfig.proxies).length > 1) {
|
2023-06-13 17:46:07 +08:00
|
|
|
prefix = ' ';
|
|
|
|
core.info(host);
|
|
|
|
}
|
|
|
|
for (const key in dockerConfig.proxies[host]) {
|
|
|
|
core.info(`${prefix}${key}: ${dockerConfig.proxies[host][key]}`);
|
|
|
|
}
|
|
|
|
}
|
2023-09-08 21:28:08 +08:00
|
|
|
} else if (!dockerConfigMalformed) {
|
|
|
|
core.info('No proxy configuration found');
|
|
|
|
}
|
|
|
|
});
|
2023-06-13 17:46:07 +08:00
|
|
|
|
2023-02-20 18:11:15 +08:00
|
|
|
if (!(await toolkit.buildx.isAvailable())) {
|
2021-04-27 22:16:22 +08:00
|
|
|
core.setFailed(`Docker buildx is required. See https://github.com/docker/setup-buildx-action to set up buildx.`);
|
|
|
|
return;
|
2020-08-16 06:36:41 +08:00
|
|
|
}
|
|
|
|
|
2023-02-20 18:11:15 +08:00
|
|
|
stateHelper.setTmpDir(Context.tmpDir());
|
|
|
|
|
2022-04-28 15:31:47 +08:00
|
|
|
await core.group(`Buildx version`, async () => {
|
2023-02-20 18:11:15 +08:00
|
|
|
await toolkit.buildx.printVersion();
|
2022-04-28 15:31:47 +08:00
|
|
|
});
|
2020-08-17 04:31:37 +08:00
|
|
|
|
2024-06-27 17:26:35 +08:00
|
|
|
let builder: BuilderInfo;
|
2024-05-29 18:52:50 +08:00
|
|
|
await core.group(`Builder info`, async () => {
|
2024-06-27 17:26:35 +08:00
|
|
|
builder = await toolkit.builder.inspect(inputs.builder);
|
2024-05-29 18:52:50 +08:00
|
|
|
core.info(JSON.stringify(builder, null, 2));
|
|
|
|
});
|
|
|
|
|
2023-02-20 18:11:15 +08:00
|
|
|
const args: string[] = await context.getArgs(inputs, toolkit);
|
2024-03-06 21:20:33 +08:00
|
|
|
core.debug(`context.getArgs: ${JSON.stringify(args)}`);
|
|
|
|
|
2023-02-20 18:11:15 +08:00
|
|
|
const buildCmd = await toolkit.buildx.getCommand(args);
|
2024-03-06 21:20:33 +08:00
|
|
|
core.debug(`buildCmd.command: ${buildCmd.command}`);
|
|
|
|
core.debug(`buildCmd.args: ${JSON.stringify(buildCmd.args)}`);
|
|
|
|
|
2024-05-02 19:49:01 +08:00
|
|
|
let err: Error | undefined;
|
2023-02-20 18:11:15 +08:00
|
|
|
await Exec.getExecOutput(buildCmd.command, buildCmd.args, {
|
|
|
|
ignoreReturnCode: true
|
|
|
|
}).then(res => {
|
|
|
|
if (res.stderr.length > 0 && res.exitCode != 0) {
|
2024-05-02 19:49:01 +08:00
|
|
|
err = Error(`buildx failed with: ${res.stderr.match(/(.*)\s*$/)?.[0]?.trim() ?? 'unknown error'}`);
|
2023-02-20 18:11:15 +08:00
|
|
|
}
|
|
|
|
});
|
2020-08-18 04:18:15 +08:00
|
|
|
|
2024-05-14 20:11:32 +08:00
|
|
|
const imageID = toolkit.buildxBuild.resolveImageID();
|
|
|
|
const metadata = toolkit.buildxBuild.resolveMetadata();
|
|
|
|
const digest = toolkit.buildxBuild.resolveDigest();
|
2022-02-09 18:32:35 +08:00
|
|
|
if (imageID) {
|
2022-03-15 02:30:50 +08:00
|
|
|
await core.group(`ImageID`, async () => {
|
2022-02-09 18:32:35 +08:00
|
|
|
core.info(imageID);
|
2022-10-12 12:56:31 +08:00
|
|
|
core.setOutput('imageid', imageID);
|
2022-03-15 02:30:50 +08:00
|
|
|
});
|
|
|
|
}
|
|
|
|
if (digest) {
|
|
|
|
await core.group(`Digest`, async () => {
|
|
|
|
core.info(digest);
|
2022-10-12 12:56:31 +08:00
|
|
|
core.setOutput('digest', digest);
|
2022-02-09 18:32:35 +08:00
|
|
|
});
|
|
|
|
}
|
|
|
|
if (metadata) {
|
2022-03-15 02:30:50 +08:00
|
|
|
await core.group(`Metadata`, async () => {
|
2024-04-26 17:20:49 +08:00
|
|
|
const metadatadt = JSON.stringify(metadata, null, 2);
|
|
|
|
core.info(metadatadt);
|
|
|
|
core.setOutput('metadata', metadatadt);
|
2022-02-09 18:32:35 +08:00
|
|
|
});
|
|
|
|
}
|
2024-06-27 17:26:35 +08:00
|
|
|
|
|
|
|
let ref: string;
|
2024-05-02 19:49:01 +08:00
|
|
|
await core.group(`Reference`, async () => {
|
2024-06-27 17:26:35 +08:00
|
|
|
ref = await buildRef(toolkit, startedTime, inputs.builder);
|
2024-05-02 19:49:01 +08:00
|
|
|
if (ref) {
|
|
|
|
core.info(ref);
|
|
|
|
stateHelper.setBuildRef(ref);
|
|
|
|
} else {
|
2024-06-27 17:26:35 +08:00
|
|
|
core.info('No build reference found');
|
2024-05-02 19:49:01 +08:00
|
|
|
}
|
|
|
|
});
|
2024-06-27 17:26:35 +08:00
|
|
|
|
|
|
|
await core.group(`Check build summary support`, async () => {
|
2024-07-02 23:38:24 +08:00
|
|
|
if (!buildSummaryEnabled()) {
|
2024-06-27 17:26:35 +08:00
|
|
|
core.info('Build summary disabled');
|
|
|
|
} else if (GitHub.isGHES) {
|
|
|
|
core.warning('Build summary is not yet supported on GHES');
|
|
|
|
} else if (!(await toolkit.buildx.versionSatisfies('>=0.13.0'))) {
|
|
|
|
core.warning('Build summary requires Buildx >= 0.13.0');
|
|
|
|
} else if (builder && builder.driver === 'cloud') {
|
|
|
|
core.warning('Build summary is not yet supported with Docker Build Cloud');
|
|
|
|
} else if (!ref) {
|
|
|
|
core.warning('Build summary requires a build reference');
|
|
|
|
} else {
|
|
|
|
core.info('Build summary supported!');
|
|
|
|
stateHelper.setSummarySupported();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2024-05-02 19:49:01 +08:00
|
|
|
if (err) {
|
|
|
|
throw err;
|
|
|
|
}
|
2023-02-20 18:11:15 +08:00
|
|
|
},
|
|
|
|
// post
|
|
|
|
async () => {
|
2024-06-27 17:26:35 +08:00
|
|
|
if (stateHelper.isSummarySupported) {
|
2024-05-02 19:49:01 +08:00
|
|
|
await core.group(`Generating build summary`, async () => {
|
2024-05-02 19:49:01 +08:00
|
|
|
try {
|
2024-07-03 00:07:54 +08:00
|
|
|
const recordUploadEnabled = buildRecordUploadEnabled();
|
|
|
|
let exportRetentionDays: number | undefined;
|
|
|
|
if (recordUploadEnabled) {
|
|
|
|
exportRetentionDays = buildExportRetentionDays();
|
|
|
|
}
|
|
|
|
|
2024-05-02 19:49:01 +08:00
|
|
|
const buildxHistory = new BuildxHistory();
|
|
|
|
const exportRes = await buildxHistory.export({
|
2024-06-27 17:26:35 +08:00
|
|
|
refs: stateHelper.buildRef ? [stateHelper.buildRef] : []
|
2024-05-02 19:49:01 +08:00
|
|
|
});
|
2024-07-03 00:07:54 +08:00
|
|
|
core.info(`Build record written to ${exportRes.dockerbuildFilename} (${Util.formatFileSize(exportRes.dockerbuildSize)})`);
|
|
|
|
|
|
|
|
let uploadRes: UploadArtifactResponse | undefined;
|
|
|
|
if (recordUploadEnabled) {
|
|
|
|
uploadRes = await GitHub.uploadArtifact({
|
|
|
|
filename: exportRes.dockerbuildFilename,
|
|
|
|
mimeType: 'application/gzip',
|
|
|
|
retentionDays: exportRetentionDays
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2024-05-02 19:49:01 +08:00
|
|
|
await GitHub.writeBuildSummary({
|
|
|
|
exportRes: exportRes,
|
|
|
|
uploadRes: uploadRes,
|
|
|
|
inputs: stateHelper.inputs
|
|
|
|
});
|
2024-05-02 19:49:01 +08:00
|
|
|
} catch (e) {
|
|
|
|
core.warning(e.message);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
2023-02-20 18:11:15 +08:00
|
|
|
if (stateHelper.tmpDir.length > 0) {
|
|
|
|
await core.group(`Removing temp folder ${stateHelper.tmpDir}`, async () => {
|
|
|
|
fs.rmSync(stateHelper.tmpDir, {recursive: true});
|
|
|
|
});
|
|
|
|
}
|
2020-09-02 16:07:11 +08:00
|
|
|
}
|
2023-02-20 18:11:15 +08:00
|
|
|
);
|
2024-05-02 19:49:01 +08:00
|
|
|
|
|
|
|
async function buildRef(toolkit: Toolkit, since: Date, builder?: string): Promise<string> {
|
|
|
|
// get ref from metadata file
|
|
|
|
const ref = toolkit.buildxBuild.resolveRef();
|
|
|
|
if (ref) {
|
|
|
|
return ref;
|
|
|
|
}
|
|
|
|
// otherwise, look for the very first build ref since the build has started
|
|
|
|
if (!builder) {
|
|
|
|
const currentBuilder = await toolkit.builder.inspect();
|
|
|
|
builder = currentBuilder.name;
|
|
|
|
}
|
|
|
|
const refs = Buildx.refs({
|
|
|
|
dir: Buildx.refsDir,
|
|
|
|
builderName: builder,
|
|
|
|
since: since
|
|
|
|
});
|
|
|
|
return Object.keys(refs).length > 0 ? Object.keys(refs)[0] : '';
|
|
|
|
}
|
2024-06-24 16:14:14 +08:00
|
|
|
|
2024-07-02 23:38:24 +08:00
|
|
|
function buildSummaryEnabled(): boolean {
|
2024-07-02 20:49:31 +08:00
|
|
|
if (process.env.DOCKER_BUILD_NO_SUMMARY) {
|
2024-07-02 23:38:24 +08:00
|
|
|
core.warning('DOCKER_BUILD_NO_SUMMARY is deprecated. Set DOCKER_BUILD_SUMMARY to false instead.');
|
|
|
|
return !Util.parseBool(process.env.DOCKER_BUILD_NO_SUMMARY);
|
|
|
|
} else if (process.env.DOCKER_BUILD_SUMMARY) {
|
|
|
|
return Util.parseBool(process.env.DOCKER_BUILD_SUMMARY);
|
2024-07-02 20:49:31 +08:00
|
|
|
}
|
2024-07-02 23:38:24 +08:00
|
|
|
return true;
|
2024-07-02 20:49:31 +08:00
|
|
|
}
|
|
|
|
|
2024-07-03 00:07:54 +08:00
|
|
|
function buildRecordUploadEnabled(): boolean {
|
|
|
|
if (process.env.DOCKER_BUILD_RECORD_UPLOAD) {
|
|
|
|
return Util.parseBool(process.env.DOCKER_BUILD_RECORD_UPLOAD);
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2024-06-24 16:14:14 +08:00
|
|
|
function buildExportRetentionDays(): number | undefined {
|
|
|
|
if (process.env.DOCKER_BUILD_EXPORT_RETENTION_DAYS) {
|
|
|
|
const res = parseInt(process.env.DOCKER_BUILD_EXPORT_RETENTION_DAYS);
|
|
|
|
if (isNaN(res)) {
|
|
|
|
throw Error(`Invalid build export retention days: ${process.env.DOCKER_BUILD_EXPORT_RETENTION_DAYS}`);
|
|
|
|
}
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
}
|