mirror of
https://github.com/github/codeql-action.git
synced 2026-01-01 04:00:24 +08:00
This refactoring commit records the name of the tar program in the new TarVersion.name field and makes extractTarZst() use the new field instead of the hardcoded name "tar". Code behavior remains unchanged because currently TarVersion.name is always "tar". This is the first step toward supporting a tar program under a different executable name.
232 lines
6.3 KiB
TypeScript
232 lines
6.3 KiB
TypeScript
import { spawn } from "child_process";
|
|
import * as fs from "fs";
|
|
import * as stream from "stream";
|
|
|
|
import { ToolRunner } from "@actions/exec/lib/toolrunner";
|
|
import * as io from "@actions/io";
|
|
import * as toolcache from "@actions/tool-cache";
|
|
import * as semver from "semver";
|
|
|
|
import { CommandInvocationError } from "./actions-util";
|
|
import { Logger } from "./logging";
|
|
import { assertNever, cleanUpGlob, isBinaryAccessible } from "./util";
|
|
|
|
const MIN_REQUIRED_BSD_TAR_VERSION = "3.4.3";
|
|
const MIN_REQUIRED_GNU_TAR_VERSION = "1.31";
|
|
|
|
export type TarVersion = {
|
|
name: string;
|
|
type: "gnu" | "bsd";
|
|
version: string;
|
|
};
|
|
|
|
async function getTarVersion(): Promise<TarVersion> {
|
|
const tar = await io.which("tar", true);
|
|
let stdout = "";
|
|
const exitCode = await new ToolRunner(tar, ["--version"], {
|
|
listeners: {
|
|
stdout: (data: Buffer) => {
|
|
stdout += data.toString();
|
|
},
|
|
},
|
|
}).exec();
|
|
if (exitCode !== 0) {
|
|
throw new Error("Failed to call tar --version");
|
|
}
|
|
// Return whether this is GNU tar or BSD tar, and the version number
|
|
if (stdout.includes("GNU tar")) {
|
|
const match = stdout.match(/tar \(GNU tar\) ([0-9.]+)/);
|
|
if (!match || !match[1]) {
|
|
throw new Error("Failed to parse output of tar --version.");
|
|
}
|
|
|
|
return { name: "tar", type: "gnu", version: match[1] };
|
|
} else if (stdout.includes("bsdtar")) {
|
|
const match = stdout.match(/bsdtar ([0-9.]+)/);
|
|
if (!match || !match[1]) {
|
|
throw new Error("Failed to parse output of tar --version.");
|
|
}
|
|
|
|
return { name: "tar", type: "bsd", version: match[1] };
|
|
} else {
|
|
throw new Error("Unknown tar version");
|
|
}
|
|
}
|
|
|
|
export interface ZstdAvailability {
|
|
available: boolean;
|
|
foundZstdBinary: boolean;
|
|
version?: TarVersion;
|
|
}
|
|
|
|
export async function isZstdAvailable(
|
|
logger: Logger,
|
|
): Promise<ZstdAvailability> {
|
|
const foundZstdBinary = await isBinaryAccessible("zstd", logger);
|
|
try {
|
|
const tarVersion = await getTarVersion();
|
|
const { type, version } = tarVersion;
|
|
logger.info(`Found ${type} tar version ${version}.`);
|
|
switch (type) {
|
|
case "gnu":
|
|
return {
|
|
available:
|
|
foundZstdBinary &&
|
|
// GNU tar only uses major and minor version numbers
|
|
semver.gte(
|
|
semver.coerce(version)!,
|
|
semver.coerce(MIN_REQUIRED_GNU_TAR_VERSION)!,
|
|
),
|
|
foundZstdBinary,
|
|
version: tarVersion,
|
|
};
|
|
case "bsd":
|
|
return {
|
|
available:
|
|
foundZstdBinary &&
|
|
// Do a loose comparison since these version numbers don't contain
|
|
// a patch version number.
|
|
semver.gte(version, MIN_REQUIRED_BSD_TAR_VERSION),
|
|
foundZstdBinary,
|
|
version: tarVersion,
|
|
};
|
|
default:
|
|
assertNever(type);
|
|
}
|
|
} catch (e) {
|
|
logger.warning(
|
|
"Failed to determine tar version, therefore will assume zstd is not available. " +
|
|
`The underlying error was: ${e}`,
|
|
);
|
|
return { available: false, foundZstdBinary };
|
|
}
|
|
}
|
|
|
|
export type CompressionMethod = "gzip" | "zstd";
|
|
|
|
export async function extract(
|
|
tarPath: string,
|
|
dest: string,
|
|
compressionMethod: CompressionMethod,
|
|
tarVersion: TarVersion | undefined,
|
|
logger: Logger,
|
|
): Promise<string> {
|
|
// Ensure destination exists
|
|
fs.mkdirSync(dest, { recursive: true });
|
|
|
|
switch (compressionMethod) {
|
|
case "gzip":
|
|
// Defensively continue to call the toolcache API as requesting a gzipped
|
|
// bundle may be a fallback option.
|
|
return await toolcache.extractTar(tarPath, dest);
|
|
case "zstd": {
|
|
if (!tarVersion) {
|
|
throw new Error(
|
|
"Could not determine tar version, which is required to extract a Zstandard archive.",
|
|
);
|
|
}
|
|
await extractTarZst(tarPath, dest, tarVersion, logger);
|
|
return dest;
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Extract a compressed tar archive
|
|
*
|
|
* @param tar tar stream, or path to the tar
|
|
* @param dest destination directory
|
|
*/
|
|
export async function extractTarZst(
|
|
tar: stream.Readable | string,
|
|
dest: string,
|
|
tarVersion: TarVersion,
|
|
logger: Logger,
|
|
): Promise<void> {
|
|
logger.debug(
|
|
`Extracting to ${dest}.${
|
|
tar instanceof stream.Readable
|
|
? ` Input stream has high water mark ${tar.readableHighWaterMark}.`
|
|
: ""
|
|
}`,
|
|
);
|
|
|
|
try {
|
|
// Initialize args
|
|
const args = ["-x", "--zstd"];
|
|
|
|
if (tarVersion.type === "gnu") {
|
|
// Suppress warnings when using GNU tar to extract archives created by BSD tar
|
|
args.push("--warning=no-unknown-keyword");
|
|
args.push("--overwrite");
|
|
}
|
|
|
|
args.push("-f", tar instanceof stream.Readable ? "-" : tar, "-C", dest);
|
|
|
|
process.stdout.write(`[command]${tarVersion.name} ${args.join(" ")}\n`);
|
|
|
|
await new Promise<void>((resolve, reject) => {
|
|
const tarProcess = spawn(tarVersion.name, args, { stdio: "pipe" });
|
|
|
|
let stdout = "";
|
|
tarProcess.stdout?.on("data", (data: Buffer) => {
|
|
stdout += data.toString();
|
|
process.stdout.write(data);
|
|
});
|
|
|
|
let stderr = "";
|
|
tarProcess.stderr?.on("data", (data: Buffer) => {
|
|
stderr += data.toString();
|
|
// Mimic the standard behavior of the toolrunner by writing stderr to stdout
|
|
process.stdout.write(data);
|
|
});
|
|
|
|
tarProcess.on("error", (err) => {
|
|
reject(new Error(`Error while extracting tar: ${err}`));
|
|
});
|
|
|
|
if (tar instanceof stream.Readable) {
|
|
tar.pipe(tarProcess.stdin).on("error", (err) => {
|
|
reject(
|
|
new Error(`Error while downloading and extracting tar: ${err}`),
|
|
);
|
|
});
|
|
}
|
|
|
|
tarProcess.on("exit", (code) => {
|
|
if (code !== 0) {
|
|
reject(
|
|
new CommandInvocationError(
|
|
tarVersion.name,
|
|
args,
|
|
code ?? undefined,
|
|
stdout,
|
|
stderr,
|
|
),
|
|
);
|
|
}
|
|
resolve();
|
|
});
|
|
});
|
|
} catch (e) {
|
|
await cleanUpGlob(dest, "extraction destination directory", logger);
|
|
throw e;
|
|
}
|
|
}
|
|
|
|
const KNOWN_EXTENSIONS: Record<string, CompressionMethod> = {
|
|
"tar.gz": "gzip",
|
|
"tar.zst": "zstd",
|
|
};
|
|
|
|
export function inferCompressionMethod(
|
|
tarPath: string,
|
|
): CompressionMethod | undefined {
|
|
for (const [ext, method] of Object.entries(KNOWN_EXTENSIONS)) {
|
|
if (tarPath.endsWith(`.${ext}`)) {
|
|
return method;
|
|
}
|
|
}
|
|
return undefined;
|
|
}
|