renovate/lib/modules/manager/cargo/artifacts.ts

224 lines
6.6 KiB
TypeScript

import { quote } from 'shlex';
import type { UserEnv } from '../../../config/types';
import { TEMPORARY_ERROR } from '../../../constants/error-messages';
import { logger } from '../../../logger';
import { coerceArray } from '../../../util/array';
import { exec } from '../../../util/exec';
import type { ExecOptions } from '../../../util/exec/types';
import {
findLocalSiblingOrParent,
readLocalFile,
writeLocalFile,
} from '../../../util/fs';
import { getGitEnvironmentVariables } from '../../../util/git/auth';
import { regEx } from '../../../util/regex';
import type { UpdateArtifact, UpdateArtifactsResult, Upgrade } from '../types';
import { extractLockFileContentVersions } from './locked-version';
async function cargoUpdate(
manifestPath: string,
isLockFileMaintenance: boolean,
userConfiguredEnv: UserEnv,
constraint: string | undefined,
): Promise<void> {
let cmd = `cargo update --config net.git-fetch-with-cli=true --manifest-path ${quote(
manifestPath,
)}`;
// If we're updating a specific crate, `cargo-update` requires `--workspace`
// for more information, see: https://github.com/renovatebot/renovate/issues/12332
if (!isLockFileMaintenance) {
cmd += ` --workspace`;
}
const execOptions: ExecOptions = {
userConfiguredEnv,
extraEnv: { ...getGitEnvironmentVariables(['cargo']) },
docker: {},
toolConstraints: [{ toolName: 'rust', constraint }],
};
await exec(cmd, execOptions);
}
async function cargoUpdatePrecise(
manifestPath: string,
updatedDeps: Upgrade[],
userConfiguredEnv: UserEnv,
constraint: string | undefined,
): Promise<void> {
// First update all dependencies that have been bumped in `Cargo.toml`.
const cmds = [
'cargo update --config net.git-fetch-with-cli=true' +
` --manifest-path ${quote(manifestPath)} --workspace`,
];
// Update individual dependencies to their `newVersion`. Necessary when
// using the `update-lockfile` rangeStrategy which doesn't touch Cargo.toml.
for (const dep of updatedDeps) {
cmds.push(
`cargo update --config net.git-fetch-with-cli=true` +
` --manifest-path ${quote(manifestPath)}` +
` --package ${quote(`${dep.packageName}@${dep.lockedVersion}`)}` +
` --precise ${quote(dep.newVersion!)}`,
);
}
const execOptions: ExecOptions = {
userConfiguredEnv,
extraEnv: { ...getGitEnvironmentVariables(['cargo']) },
docker: {},
toolConstraints: [{ toolName: 'rust', constraint }],
};
await exec(cmds, execOptions);
}
export async function updateArtifacts(
updateArtifact: UpdateArtifact,
): Promise<UpdateArtifactsResult[] | null> {
return await updateArtifactsImpl(updateArtifact);
}
async function updateArtifactsImpl(
{
packageFileName,
updatedDeps,
newPackageFileContent,
config,
}: UpdateArtifact,
recursionLimit = 10,
): Promise<UpdateArtifactsResult[] | null> {
logger.debug(`cargo.updateArtifacts(${packageFileName})`);
// For standalone package crates, the `Cargo.lock` will be in the same
// directory as `Cargo.toml` (ie. a sibling). For cargo workspaces, it
// will be further up.
const lockFileName = await findLocalSiblingOrParent(
packageFileName,
'Cargo.lock',
);
const existingLockFileContent = lockFileName
? await readLocalFile(lockFileName)
: null;
if (!existingLockFileContent || !lockFileName) {
logger.debug('No Cargo.lock found');
return null;
}
const isLockFileMaintenance = config.updateType === 'lockFileMaintenance';
if (!isLockFileMaintenance && !updatedDeps?.length) {
logger.debug('No more dependencies to update');
return [
{
file: {
type: 'addition',
path: lockFileName,
contents: existingLockFileContent,
},
},
];
}
try {
await writeLocalFile(packageFileName, newPackageFileContent);
logger.debug('Updating ' + lockFileName);
if (isLockFileMaintenance) {
await cargoUpdate(
packageFileName,
true,
config.env ?? {},
config.constraints?.rust,
);
} else {
const missingDep = updatedDeps.find((dep) => !dep.lockedVersion);
if (missingDep) {
// If there is a dependency without a locked version then log a warning
// and perform a regular workspace lockfile update.
logger.warn(
`Missing locked version for dependency \`${missingDep.depName}\``,
);
await cargoUpdate(
packageFileName,
false,
config.env ?? {},
config.constraints?.rust,
);
} else {
// If all dependencies have locked versions then update them precisely.
await cargoUpdatePrecise(
packageFileName,
updatedDeps,
config.env ?? {},
config.constraints?.rust,
);
}
}
logger.debug('Returning updated Cargo.lock');
const newCargoLockContent = await readLocalFile(lockFileName);
if (existingLockFileContent === newCargoLockContent) {
logger.debug('Cargo.lock is unchanged');
return null;
}
return [
{
file: {
type: 'addition',
path: lockFileName,
contents: newCargoLockContent,
},
},
];
} catch (err) {
// istanbul ignore if
if (err.message === TEMPORARY_ERROR) {
throw err;
}
// Sometimes `cargo update` will fail when a preceding dependency update
// causes another dependency to update. In this case we can no longer
// reference the dependency by its old version, so we filter it out
// and retry recursively.
const newCargoLockContent = await readLocalFile(lockFileName, 'utf8');
if (
recursionLimit > 0 &&
newCargoLockContent &&
regEx(/error: package ID specification/).test(err.stderr)
) {
const versions = extractLockFileContentVersions(newCargoLockContent);
const newUpdatedDeps = updatedDeps.filter(
(dep) =>
!coerceArray(versions?.get(dep.packageName!)).includes(
dep.newVersion!,
),
);
if (newUpdatedDeps.length < updatedDeps.length) {
logger.debug(
'Dependency already up to date - reattempting recursively',
);
return updateArtifactsImpl(
{
packageFileName,
updatedDeps: newUpdatedDeps,
newPackageFileContent,
config,
},
recursionLimit - 1,
);
}
}
logger.debug({ err }, 'Failed to update Cargo lock file');
return [
{
artifactError: {
lockFile: lockFileName,
stderr: err.message,
},
},
];
}
}