0
0
Fork 0
mirror of https://github.com/renovatebot/renovate.git synced 2025-05-12 23:51:55 +00:00
renovatebot_renovate/lib/modules/manager/mix/artifacts.ts
2025-05-08 11:02:42 +00:00

237 lines
6 KiB
TypeScript

import is from '@sindresorhus/is';
import { quote } from 'shlex';
import { TEMPORARY_ERROR } from '../../../constants/error-messages';
import { logger } from '../../../logger';
import { exec } from '../../../util/exec';
import type { ExecOptions } from '../../../util/exec/types';
import {
deleteLocalFile,
ensureCacheDir,
findLocalSiblingOrParent,
getSiblingFileName,
localPathExists,
readLocalFile,
writeLocalFile,
} from '../../../util/fs';
import * as hostRules from '../../../util/host-rules';
import { regEx } from '../../../util/regex';
import type { UpdateArtifact, UpdateArtifactsResult } from '../types';
const hexRepoUrl = 'https://hex.pm/';
const hexRepoOrgUrlRegex = regEx(
`^https://hex\\.pm/api/repos/(?<organization>[a-z0-9_]+)/$`,
);
export async function updateArtifacts({
packageFileName,
updatedDeps,
newPackageFileContent,
config,
}: UpdateArtifact): Promise<UpdateArtifactsResult[] | null> {
logger.debug(`mix.getArtifacts(${packageFileName})`);
const { isLockFileMaintenance } = config;
if (is.emptyArray(updatedDeps) && !isLockFileMaintenance) {
logger.debug('No updated mix deps');
return null;
}
let lockFileName = getSiblingFileName(packageFileName, 'mix.lock');
let isUmbrella = false;
let existingLockFileContent = await readLocalFile(lockFileName, 'utf8');
if (!existingLockFileContent) {
const lockFileError = await checkLockFileReadError(lockFileName);
if (lockFileError) {
return lockFileError;
}
const parentLockFileName = await findLocalSiblingOrParent(
packageFileName,
'mix.lock',
);
existingLockFileContent =
parentLockFileName && (await readLocalFile(parentLockFileName, 'utf8'));
if (parentLockFileName && existingLockFileContent) {
lockFileName = parentLockFileName;
isUmbrella = true;
} else if (parentLockFileName) {
const lockFileError = await checkLockFileReadError(parentLockFileName);
if (lockFileError) {
return lockFileError;
}
}
}
if (isLockFileMaintenance && isUmbrella) {
logger.debug(
'Cannot use lockFileMaintenance in an umbrella project, see https://docs.renovatebot.com/modules/manager/mix/#lockFileMaintenance',
);
return null;
}
if (isLockFileMaintenance && !existingLockFileContent) {
logger.debug(
'Cannot use lockFileMaintenance when no mix.lock file is present',
);
return null;
}
try {
await writeLocalFile(packageFileName, newPackageFileContent);
if (isLockFileMaintenance) {
await deleteLocalFile(lockFileName);
}
} catch (err) {
logger.warn({ err }, 'mix.exs could not be written');
return [
{
artifactError: {
lockFile: lockFileName,
stderr: err.message,
},
},
];
}
if (!existingLockFileContent) {
logger.debug('No mix.lock found');
return null;
}
const organizations = new Set<string>();
const hexHostRulesWithMatchHost = hostRules
.getAll()
.filter(
(hostRule) =>
!!hostRule.matchHost && hexRepoOrgUrlRegex.test(hostRule.matchHost),
);
for (const { matchHost } of hexHostRulesWithMatchHost) {
if (matchHost) {
const result = hexRepoOrgUrlRegex.exec(matchHost);
if (result?.groups) {
const { organization } = result.groups;
organizations.add(organization);
}
}
}
for (const { packageName } of updatedDeps) {
if (packageName) {
const [, organization] = packageName.split(':');
if (organization) {
organizations.add(organization);
}
}
}
const preCommands = Array.from(organizations).reduce((acc, organization) => {
const url = `${hexRepoUrl}api/repos/${organization}/`;
const { token } = hostRules.find({ url });
if (token) {
logger.debug(`Authenticating to hex organization ${organization}`);
const authCommand = `mix hex.organization auth ${organization} --key ${token}`;
return [...acc, authCommand];
}
return acc;
}, [] as string[]);
const execOptions: ExecOptions = {
extraEnv: {
// https://hexdocs.pm/mix/1.15.0/Mix.Tasks.Archive.html
// TODO: should include a version constraint
MIX_ARCHIVES: await ensureCacheDir('mix_archives'),
},
cwdFile: packageFileName,
docker: {},
toolConstraints: [
{
toolName: 'erlang',
// https://hexdocs.pm/elixir/1.14.5/compatibility-and-deprecations.html#compatibility-between-elixir-and-erlang-otp
constraint: config.constraints?.erlang ?? '^26',
},
{
toolName: 'elixir',
constraint: config.constraints?.elixir,
},
],
preCommands,
};
let command: string;
if (isLockFileMaintenance) {
command = 'mix deps.get';
} else {
command = [
'mix',
'deps.update',
...updatedDeps
.map((dep) => dep.depName)
.filter(is.string)
.map((dep) => quote(dep)),
].join(' ');
}
try {
await exec(command, execOptions);
} catch (err) {
/* v8 ignore next 3 */
if (err.message === TEMPORARY_ERROR) {
throw err;
}
logger.debug(
{ err, message: err.message, command },
'Failed to update Mix lock file',
);
return [
{
artifactError: {
lockFile: lockFileName,
stderr: err.message,
},
},
];
}
const newMixLockContent = await readLocalFile(lockFileName, 'utf8');
if (existingLockFileContent === newMixLockContent) {
logger.debug('mix.lock is unchanged');
return null;
}
logger.debug('Returning updated mix.lock');
return [
{
file: {
type: 'addition',
path: lockFileName,
contents: newMixLockContent,
},
},
];
}
async function checkLockFileReadError(
lockFileName: string,
): Promise<UpdateArtifactsResult[] | null> {
if (await localPathExists(lockFileName)) {
return [
{
artifactError: {
lockFile: lockFileName,
stderr: `Error reading ${lockFileName}`,
},
},
];
}
return null;
}