renovate/lib/util/exec/docker/index.ts

287 lines
8.3 KiB
TypeScript

import is from '@sindresorhus/is';
import { GlobalConfig } from '../../../config/global';
import { SYSTEM_INSUFFICIENT_MEMORY } from '../../../constants/error-messages';
import { logger } from '../../../logger';
import { getPkgReleases } from '../../../modules/datasource';
import * as allVersioning from '../../../modules/versioning';
import { newlineRegex, regEx } from '../../regex';
import { uniq } from '../../uniq';
import { rawExec } from '../common';
import type { DockerOptions, Opt, VolumeOption, VolumesPair } from '../types';
const prefetchedImages = new Map<string, string>();
const digestRegex = regEx('Digest: (.*?)\n');
export const sideCarImage = 'sidecar';
export async function prefetchDockerImage(taggedImage: string): Promise<void> {
if (prefetchedImages.has(taggedImage)) {
logger.debug(
`Docker image is already prefetched: ${taggedImage}@${prefetchedImages.get(
taggedImage,
)!}`,
);
} else {
logger.debug(`Fetching Docker image: ${taggedImage}`);
const res = await rawExec(`docker pull ${taggedImage}`, {
encoding: 'utf-8',
});
const imageDigest = digestRegex.exec(res?.stdout)?.[1] ?? 'unknown';
logger.debug(
`Finished fetching Docker image ${taggedImage}@${imageDigest}`,
);
prefetchedImages.set(taggedImage, imageDigest);
}
}
export function resetPrefetchedImages(): void {
prefetchedImages.clear();
}
function expandVolumeOption(x: VolumeOption): VolumesPair | null {
if (is.nonEmptyString(x)) {
return [x, x];
}
if (Array.isArray(x) && x.length === 2) {
const [from, to] = x;
if (is.nonEmptyString(from) && is.nonEmptyString(to)) {
return [from, to];
}
}
return null;
}
function volumesEql(x: VolumesPair, y: VolumesPair): boolean {
const [xFrom, xTo] = x;
const [yFrom, yTo] = y;
return xFrom === yFrom && xTo === yTo;
}
function prepareVolumes(volumes: VolumeOption[]): string[] {
const expanded: (VolumesPair | null)[] = volumes.map(expandVolumeOption);
const filtered: VolumesPair[] = expanded.filter(
(vol): vol is VolumesPair => vol !== null,
);
const unique: VolumesPair[] = uniq<VolumesPair>(filtered, volumesEql);
return unique.map(([from, to]) => `-v "${from}":"${to}"`);
}
function prepareCommands(commands: Opt<string>[]): string[] {
return commands.filter<string>((command): command is string =>
is.string(command),
);
}
export async function getDockerTag(
packageName: string,
constraint: string,
versioning: string,
): Promise<string> {
const versioningApi = allVersioning.get(versioning);
if (!versioningApi.isValid(constraint)) {
logger.warn(
{ versioning, constraint },
`Invalid Docker image version constraint`,
);
return 'latest';
}
logger.debug(
{ packageName, versioning, constraint },
`Found version constraint - checking for a compatible image to use`,
);
const imageReleases = await getPkgReleases({
datasource: 'docker',
packageName,
versioning,
});
if (imageReleases?.releases) {
let versions = imageReleases.releases.map((release) => release.version);
versions = versions.filter(
(version) =>
versioningApi.isVersion(version) &&
versioningApi.matches(version, constraint),
);
// Prefer stable versions over unstable, even if the range satisfies both types
if (!versions.every((version) => versioningApi.isStable(version))) {
logger.debug('Filtering out unstable versions');
versions = versions.filter((version) => versioningApi.isStable(version));
}
const version = versions
.sort(versioningApi.sortVersions.bind(versioningApi))
.pop();
if (version) {
logger.debug(
{ packageName, versioning, constraint, version },
`Found compatible image version`,
);
return version;
}
} else {
logger.error({ packageName }, `Docker exec: no releases found`);
return 'latest';
}
logger.warn(
{ packageName, constraint, versioning },
'Failed to find a tag satisfying constraint, using "latest" tag instead',
);
return 'latest';
}
function getContainerName(image: string, prefix?: string): string {
return `${prefix ?? 'renovate_'}${image}`.replace(regEx(/\//g), '_');
}
function getContainerLabel(prefix: string | undefined): string {
return `${prefix ?? 'renovate_'}child`;
}
export async function removeDockerContainer(
image: string,
prefix: string,
): Promise<void> {
const containerName = getContainerName(image, prefix);
let cmd = `docker ps --filter name=${containerName} -aq`;
try {
const res = await rawExec(cmd, {
encoding: 'utf-8',
});
const containerId = res?.stdout?.trim() || '';
if (containerId.length) {
logger.debug(`Removing container with ID: ${containerId}`);
cmd = `docker rm -f ${containerId}`;
await rawExec(cmd, {
encoding: 'utf-8',
});
} else {
logger.trace({ image, containerName }, 'No running containers to remove');
}
} catch (err) {
logger.warn(
{ image, containerName, cmd, err },
'Could not remove Docker container',
);
}
}
export async function removeDanglingContainers(): Promise<void> {
if (GlobalConfig.get('binarySource') !== 'docker') {
return;
}
try {
const containerLabel = getContainerLabel(
GlobalConfig.get('dockerChildPrefix'),
);
logger.debug(
`Removing dangling child containers with label ${containerLabel}`,
);
const res = await rawExec(
`docker ps --filter label=${containerLabel} -aq`,
{
encoding: 'utf-8',
},
);
if (res?.stdout?.trim().length) {
const containerIds = res.stdout
.trim()
.split(newlineRegex)
.map((container) => container.trim())
.filter(Boolean);
logger.debug({ containerIds }, 'Removing dangling child containers');
await rawExec(`docker rm -f ${containerIds.join(' ')}`, {
encoding: 'utf-8',
});
} else {
logger.debug('No dangling containers to remove');
}
} catch (err) {
if (err.errno === 'ENOMEM') {
throw new Error(SYSTEM_INSUFFICIENT_MEMORY);
}
if (err.stderr?.includes('Cannot connect to the Docker daemon')) {
logger.info('No docker daemon found');
} else {
logger.warn({ err }, 'Error removing dangling containers');
}
}
}
export async function generateDockerCommand(
commands: string[],
preCommands: string[],
options: DockerOptions,
): Promise<string> {
const { envVars, cwd } = options;
let image = sideCarImage;
const volumes = options.volumes ?? [];
const {
localDir,
cacheDir,
containerbaseDir,
dockerUser,
dockerChildPrefix,
dockerCliOptions,
dockerSidecarImage,
} = GlobalConfig.get();
const result = ['docker run --rm'];
const containerName = getContainerName(image, dockerChildPrefix);
const containerLabel = getContainerLabel(dockerChildPrefix);
result.push(`--name=${containerName}`);
result.push(`--label=${containerLabel}`);
if (dockerUser) {
result.push(`--user=${dockerUser}`);
}
if (dockerCliOptions) {
result.push(dockerCliOptions);
}
const volumeDirs: VolumeOption[] = [localDir, cacheDir];
if (containerbaseDir) {
if (cacheDir && containerbaseDir.startsWith(cacheDir)) {
logger.debug('containerbaseDir is inside cacheDir');
} else {
logger.debug('containerbaseDir is separate from cacheDir');
volumeDirs.push(containerbaseDir);
}
} else {
logger.debug('containerbaseDir is missing');
}
volumeDirs.push(...volumes);
result.push(...prepareVolumes(volumeDirs));
if (envVars) {
result.push(
...uniq(envVars)
.filter(is.string)
.map((e) => `-e ${e}`),
);
}
if (cwd) {
result.push(`-w "${cwd}"`);
}
// TODO: #22198
image = dockerSidecarImage!;
// TODO: add constraint: const tag = getDockerTag(image, sideCarImageVersion, 'semver');
logger.debug(
{ image /*, tagConstraint: sideCarImageVersion, tag */ },
'Resolved tag constraint',
);
const taggedImage = image; // TODO: tag ? `${image}:${tag}` : `${image}`;
await prefetchDockerImage(taggedImage);
result.push(taggedImage);
const bashCommand = [...prepareCommands(preCommands), ...commands].join(
' && ',
);
result.push(`bash -l -c "${bashCommand.replace(regEx(/"/g), '\\"')}"`); // lgtm [js/incomplete-sanitization]
return result.join(' ');
}