renovate/lib/modules/manager/gomod/update.ts

165 lines
5.3 KiB
TypeScript

// TODO: types (#22198)
import { logger } from '../../../logger';
import { newlineRegex, regEx } from '../../../util/regex';
import type { UpdateDependencyConfig } from '../types';
function getNameWithNoVersion(name: string): string {
let nameNoVersion = name.split('/').slice(0, 3).join('/');
if (nameNoVersion.startsWith('gopkg.in')) {
nameNoVersion = nameNoVersion.replace(regEx(/\.v\d+$/), '');
}
return nameNoVersion;
}
export function updateDependency({
fileContent,
upgrade,
}: UpdateDependencyConfig): string | null {
try {
logger.debug(`gomod.updateDependency: ${upgrade.newValue}`);
const { depType, updateType } = upgrade;
const currentName = upgrade.depName;
if (updateType === 'replacement') {
logger.warn('gomod manager does not support replacement updates yet');
return null;
}
// istanbul ignore if: should never happen
if (!currentName || !upgrade.managerData) {
return null;
}
const currentNameNoVersion = getNameWithNoVersion(currentName);
const lines = fileContent.split(newlineRegex);
// istanbul ignore if: hard to test
if (lines.length <= upgrade.managerData.lineNumber) {
logger.warn('go.mod current line no longer exists after update');
return null;
}
const lineToChange = lines[upgrade.managerData.lineNumber];
logger.trace({ upgrade, lineToChange }, 'go.mod current line');
if (
!lineToChange.includes(currentNameNoVersion) &&
!lineToChange.includes('rethinkdb/rethinkdb-go.v5')
) {
logger.debug(
{ lineToChange, depName: currentName },
"go.mod current line doesn't contain dependency",
);
return null;
}
let updateLineExp: RegExp | undefined;
if (depType === 'golang' || depType === 'toolchain') {
updateLineExp = regEx(
/(?<depPart>(?:toolchain )?go)(?<divider>\s*)([^\s]+|[\w]+)/,
);
}
if (depType === 'replace') {
if (upgrade.managerData.multiLine) {
updateLineExp = regEx(
/^(?<depPart>\s+[^\s]+[\s]+[=][>]+\s+)(?<divider>[^\s]+\s+)[^\s]+/,
);
} else {
updateLineExp = regEx(
/^(?<depPart>replace\s+[^\s]+[\s]+[=][>]+\s+)(?<divider>[^\s]+\s+)[^\s]+/,
);
}
} else if (depType === 'require' || depType === 'indirect') {
if (upgrade.managerData.multiLine) {
updateLineExp = regEx(/^(?<depPart>\s+[^\s]+)(?<divider>\s+)[^\s]+/);
} else {
updateLineExp = regEx(
/^(?<depPart>require\s+[^\s]+)(?<divider>\s+)[^\s]+/,
);
}
}
if (updateLineExp && !updateLineExp.test(lineToChange)) {
logger.debug('No line found to update');
return null;
}
let newLine: string;
if (upgrade.updateType === 'digest') {
const newDigestRightSized = upgrade.newDigest!.substring(
0,
upgrade.currentDigest!.length,
);
if (lineToChange.includes(newDigestRightSized)) {
return fileContent;
}
logger.debug(
{ depName: currentName, lineToChange, newDigestRightSized },
'gomod: need to update digest',
);
newLine = lineToChange.replace(
// TODO: can be undefined? (#22198)
updateLineExp!,
`$<depPart>$<divider>${newDigestRightSized}`,
);
} else {
newLine = lineToChange.replace(
// TODO: can be undefined? (#22198)
updateLineExp!,
`$<depPart>$<divider>${upgrade.newValue}`,
);
}
if (upgrade.updateType === 'major') {
logger.debug(`gomod: major update for ${currentName}`);
if (currentName.startsWith('gopkg.in/')) {
const oldV = currentName.split('.').pop();
newLine = newLine.replace(`.${oldV}`, `.v${upgrade.newMajor}`);
// Package renames - I couldn't think of a better place to do this
newLine = newLine.replace(
'gorethink/gorethink.v5',
'rethinkdb/rethinkdb-go.v5',
);
} else if (
upgrade.newMajor! > 1 &&
!newLine.includes(`/v${upgrade.newMajor}`) &&
!upgrade.newValue!.endsWith('+incompatible')
) {
if (currentName === currentNameNoVersion) {
// If package currently has no version, pin to latest one.
newLine = newLine.replace(
currentName,
`${currentName}/v${upgrade.newMajor}`,
);
} else {
// Replace version
const [oldV] = upgrade.currentValue!.split('.');
newLine = newLine.replace(
regEx(`/${oldV}(\\s+)`, undefined, false),
`/v${upgrade.newMajor}$1`,
);
}
}
}
if (
lineToChange.endsWith('+incompatible') &&
!upgrade.newValue?.endsWith('+incompatible')
) {
let toAdd = '+incompatible';
if (upgrade.updateType === 'major' && upgrade.newMajor! >= 2) {
toAdd = '';
}
newLine += toAdd;
}
if (newLine === lineToChange) {
logger.debug('No changes necessary');
return fileContent;
}
if (depType === 'indirect') {
newLine = newLine.replace(
regEx(/\s*(?:\/\/\s*indirect(?:\s*;)?\s*)*$/),
' // indirect',
);
}
lines[upgrade.managerData.lineNumber] = newLine;
return lines.join('\n');
} catch (err) {
logger.debug({ err }, 'Error setting new go.mod version');
return null;
}
}