mirror of
https://github.com/redhat-actions/buildah-build.git
synced 2025-04-19 00:41:23 +00:00
313 lines
11 KiB
TypeScript
313 lines
11 KiB
TypeScript
/***************************************************************************************************
|
|
* Copyright (c) Red Hat, Inc. All rights reserved.
|
|
* Licensed under the MIT License. See LICENSE file in the project root for license information.
|
|
**************************************************************************************************/
|
|
|
|
import * as core from "@actions/core";
|
|
import * as io from "@actions/io";
|
|
import * as path from "path";
|
|
import { Inputs, Outputs } from "./generated/inputs-outputs";
|
|
import { BuildahCli, BuildahConfigSettings } from "./buildah";
|
|
import {
|
|
getArch, getPlatform, getContainerfiles, getInputList, splitByNewline,
|
|
isFullImageName, getFullImageName, removeIllegalCharacters,
|
|
} from "./utils";
|
|
|
|
export async function run(): Promise<void> {
|
|
if (process.env.RUNNER_OS !== "Linux") {
|
|
throw new Error("buildah, and therefore this action, only works on Linux. Please use a Linux runner.");
|
|
}
|
|
|
|
// get buildah cli
|
|
const self_hosted_runner = core.getInput(Inputs.SELF_HOSTED_RUNNER_ROOT)
|
|
const buildahPath = await io.which("buildah", true);
|
|
const buildahExec = self_hosted_runner === "true" ? `sudo ${buildahPath}` : buildahPath;
|
|
const cli: BuildahCli = new BuildahCli(buildahExec);
|
|
|
|
// print buildah version
|
|
await cli.execute([ "version" ], { group: true });
|
|
|
|
// Check if fuse-overlayfs exists and find the storage driver
|
|
await cli.setStorageOptsEnv();
|
|
|
|
const DEFAULT_TAG = "latest";
|
|
const workspace = process.env.GITHUB_WORKSPACE || process.cwd();
|
|
const containerFiles = getContainerfiles();
|
|
const image = core.getInput(Inputs.IMAGE);
|
|
const tags = core.getInput(Inputs.TAGS);
|
|
const tagsList: string[] = tags.trim().split(/\s+/);
|
|
const labels = core.getInput(Inputs.LABELS);
|
|
const labelsList: string[] = labels ? splitByNewline(labels) : [];
|
|
|
|
const normalizedTagsList: string[] = [];
|
|
let isNormalized = false;
|
|
for (const tag of tagsList) {
|
|
normalizedTagsList.push(tag.toLowerCase());
|
|
if (tag.toLowerCase() !== tag) {
|
|
isNormalized = true;
|
|
}
|
|
}
|
|
const normalizedImage = image.toLowerCase();
|
|
if (isNormalized || image !== normalizedImage) {
|
|
core.warning(`Reference to image and/or tag must be lowercase.`
|
|
+ ` Reference has been converted to be compliant with standard.`);
|
|
}
|
|
|
|
// info message if user doesn't provides any tag
|
|
if (tagsList.length === 0) {
|
|
core.info(`Input "${Inputs.TAGS}" is not provided, using default tag "${DEFAULT_TAG}"`);
|
|
tagsList.push(DEFAULT_TAG);
|
|
}
|
|
|
|
const inputExtraArgsStr = core.getInput(Inputs.EXTRA_ARGS);
|
|
let buildahExtraArgs: string[] = [];
|
|
if (inputExtraArgsStr) {
|
|
// transform the array of lines into an array of arguments
|
|
// by splitting over lines, then over spaces, then trimming.
|
|
const lines = splitByNewline(inputExtraArgsStr);
|
|
buildahExtraArgs = lines.flatMap((line) => line.split(" ")).map((arg) => arg.trim());
|
|
}
|
|
|
|
// check if all tags provided are in `image:tag` format
|
|
const isFullImageNameTag = isFullImageName(normalizedTagsList[0]);
|
|
if (normalizedTagsList.some((tag) => isFullImageName(tag) !== isFullImageNameTag)) {
|
|
throw new Error(`Input "${Inputs.TAGS}" cannot have a mix of full name and non full name tags. Refer to https://github.com/redhat-actions/buildah-build#image-tag-inputs`);
|
|
}
|
|
if (!isFullImageNameTag && !normalizedImage) {
|
|
throw new Error(`Input "${Inputs.IMAGE}" must be provided when not using full image name tags. Refer to https://github.com/redhat-actions/buildah-build#image-tag-inputs`);
|
|
}
|
|
|
|
const newImage = getFullImageName(normalizedImage, normalizedTagsList[0]);
|
|
const useOCI = core.getInput(Inputs.OCI) === "true";
|
|
|
|
const archs = getArch();
|
|
const platforms = getPlatform();
|
|
|
|
if ((archs.length > 0) && (platforms.length > 0)) {
|
|
throw new Error("The --platform option may not be used in combination with the --arch option.");
|
|
}
|
|
|
|
const builtImage = [];
|
|
if (containerFiles.length !== 0) {
|
|
builtImage.push(...await doBuildUsingContainerFiles(
|
|
cli,
|
|
newImage,
|
|
workspace,
|
|
containerFiles,
|
|
useOCI,
|
|
archs,
|
|
platforms,
|
|
labelsList,
|
|
buildahExtraArgs
|
|
));
|
|
}
|
|
else {
|
|
if (platforms.length > 0) {
|
|
throw new Error("The --platform option is not supported for builds without containerfiles.");
|
|
}
|
|
builtImage.push(...await doBuildFromScratch(cli, newImage, useOCI, archs, labelsList, buildahExtraArgs));
|
|
}
|
|
|
|
if ((archs.length > 1) || (platforms.length > 1)) {
|
|
core.info(`Creating manifest with tag${normalizedTagsList.length !== 1 ? "s" : ""} `
|
|
+ `"${normalizedTagsList.join(", ")}"`);
|
|
const builtManifest = [];
|
|
for (const tag of normalizedTagsList) {
|
|
const manifestName = getFullImageName(normalizedImage, tag);
|
|
// Force-remove existing manifest to prevent errors on recurring build on the same machine
|
|
await cli.manifestRm(manifestName);
|
|
await cli.manifestCreate(manifestName);
|
|
builtManifest.push(manifestName);
|
|
|
|
for (const arch of archs) {
|
|
const tagSuffix = removeIllegalCharacters(arch);
|
|
await cli.manifestAdd(manifestName, `${newImage}-${tagSuffix}`);
|
|
}
|
|
|
|
for (const platform of platforms) {
|
|
const tagSuffix = removeIllegalCharacters(platform);
|
|
await cli.manifestAdd(manifestName, `${newImage}-${tagSuffix}`);
|
|
}
|
|
}
|
|
|
|
core.info(`✅ Successfully built image${builtImage.length !== 1 ? "s" : ""} "${builtImage.join(", ")}" `
|
|
+ `and manifest${builtManifest.length !== 1 ? "s" : ""} "${builtManifest.join(", ")}"`);
|
|
}
|
|
else if (normalizedTagsList.length > 1) {
|
|
await cli.tag(normalizedImage, normalizedTagsList);
|
|
}
|
|
else if (normalizedTagsList.length === 1) {
|
|
core.info(`✅ Successfully built image "${getFullImageName(normalizedImage, normalizedTagsList[0])}"`);
|
|
}
|
|
|
|
core.setOutput(Outputs.IMAGE, normalizedImage);
|
|
core.setOutput(Outputs.TAGS, tags);
|
|
core.setOutput(Outputs.IMAGE_WITH_TAG, newImage);
|
|
}
|
|
|
|
async function doBuildUsingContainerFiles(
|
|
cli: BuildahCli,
|
|
newImage: string,
|
|
workspace: string,
|
|
containerFiles: string[],
|
|
useOCI: boolean,
|
|
archs: string[],
|
|
platforms: string[],
|
|
labels: string[],
|
|
extraArgs: string[]
|
|
): Promise<string[]> {
|
|
if (containerFiles.length === 1) {
|
|
core.info(`Performing build from Containerfile`);
|
|
}
|
|
else {
|
|
core.info(`Performing build from ${containerFiles.length} Containerfiles`);
|
|
}
|
|
|
|
const context = path.join(workspace, core.getInput(Inputs.CONTEXT));
|
|
const buildArgs = getInputList(Inputs.BUILD_ARGS);
|
|
const containerFileAbsPaths = containerFiles.map((file) => path.join(workspace, file));
|
|
const layers = core.getInput(Inputs.LAYERS);
|
|
const tlsVerify = core.getInput(Inputs.TLS_VERIFY) === "true";
|
|
|
|
const builtImage = [];
|
|
// since multi arch image can not have same tag
|
|
// therefore, appending arch/platform in the tag
|
|
if (archs.length > 0 || platforms.length > 0) {
|
|
for (const arch of archs) {
|
|
// handling it seperately as, there is no need of
|
|
// tagSuffix if only one image has to be built
|
|
let tagSuffix = "";
|
|
if (archs.length > 1) {
|
|
tagSuffix = `-${removeIllegalCharacters(arch)}`;
|
|
}
|
|
await cli.buildUsingDocker(
|
|
`${newImage}${tagSuffix}`,
|
|
context,
|
|
containerFileAbsPaths,
|
|
buildArgs,
|
|
useOCI,
|
|
labels,
|
|
layers,
|
|
extraArgs,
|
|
tlsVerify,
|
|
arch
|
|
);
|
|
builtImage.push(`${newImage}${tagSuffix}`);
|
|
}
|
|
|
|
for (const platform of platforms) {
|
|
let tagSuffix = "";
|
|
if (platforms.length > 1) {
|
|
tagSuffix = `-${removeIllegalCharacters(platform)}`;
|
|
}
|
|
await cli.buildUsingDocker(
|
|
`${newImage}${tagSuffix}`,
|
|
context,
|
|
containerFileAbsPaths,
|
|
buildArgs,
|
|
useOCI,
|
|
labels,
|
|
layers,
|
|
extraArgs,
|
|
tlsVerify,
|
|
undefined,
|
|
platform
|
|
);
|
|
builtImage.push(`${newImage}${tagSuffix}`);
|
|
}
|
|
}
|
|
|
|
else if (archs.length === 1 || platforms.length === 1) {
|
|
await cli.buildUsingDocker(
|
|
newImage,
|
|
context,
|
|
containerFileAbsPaths,
|
|
buildArgs,
|
|
useOCI,
|
|
labels,
|
|
layers,
|
|
extraArgs,
|
|
tlsVerify,
|
|
archs[0],
|
|
platforms[0]
|
|
);
|
|
builtImage.push(newImage);
|
|
}
|
|
else {
|
|
await cli.buildUsingDocker(
|
|
newImage,
|
|
context,
|
|
containerFileAbsPaths,
|
|
buildArgs,
|
|
useOCI,
|
|
labels,
|
|
layers,
|
|
extraArgs,
|
|
tlsVerify
|
|
);
|
|
builtImage.push(newImage);
|
|
}
|
|
|
|
return builtImage;
|
|
}
|
|
|
|
async function doBuildFromScratch(
|
|
cli: BuildahCli,
|
|
newImage: string,
|
|
useOCI: boolean,
|
|
archs: string[],
|
|
labels: string[],
|
|
extraArgs: string[]
|
|
): Promise<string[]> {
|
|
core.info(`Performing build from scratch`);
|
|
|
|
const baseImage = core.getInput(Inputs.BASE_IMAGE, { required: true });
|
|
const content = getInputList(Inputs.CONTENT);
|
|
const entrypoint = getInputList(Inputs.ENTRYPOINT);
|
|
const port = core.getInput(Inputs.PORT);
|
|
const workingDir = core.getInput(Inputs.WORKDIR);
|
|
const envs = getInputList(Inputs.ENVS);
|
|
const tlsVerify = core.getInput(Inputs.TLS_VERIFY) === "true";
|
|
|
|
const container = await cli.from(baseImage, tlsVerify, extraArgs);
|
|
const containerId = container.output.replace("\n", "");
|
|
|
|
const builtImage = [];
|
|
if (archs.length > 0) {
|
|
for (const arch of archs) {
|
|
let tagSuffix = "";
|
|
if (archs.length > 1) {
|
|
tagSuffix = `-${removeIllegalCharacters(arch)}`;
|
|
}
|
|
const newImageConfig: BuildahConfigSettings = {
|
|
entrypoint,
|
|
port,
|
|
workingdir: workingDir,
|
|
envs,
|
|
arch,
|
|
labels,
|
|
};
|
|
await cli.config(containerId, newImageConfig);
|
|
await cli.copy(containerId, content);
|
|
await cli.commit(containerId, `${newImage}${tagSuffix}`, useOCI);
|
|
builtImage.push(`${newImage}${tagSuffix}`);
|
|
}
|
|
}
|
|
else {
|
|
const newImageConfig: BuildahConfigSettings = {
|
|
entrypoint,
|
|
port,
|
|
workingdir: workingDir,
|
|
envs,
|
|
labels,
|
|
};
|
|
await cli.config(containerId, newImageConfig);
|
|
await cli.copy(containerId, content);
|
|
await cli.commit(containerId, newImage, useOCI);
|
|
builtImage.push(newImage);
|
|
}
|
|
|
|
return builtImage;
|
|
}
|
|
|
|
run().catch(core.setFailed);
|