mirror of
https://github.com/renovatebot/renovate.git
synced 2025-01-15 09:06:25 +00:00
137 lines
4.3 KiB
TypeScript
137 lines
4.3 KiB
TypeScript
import semver from 'semver';
|
|
import { logger } from '../../../logger';
|
|
import { newlineRegex, regEx } from '../../../util/regex';
|
|
import { GoDatasource } from '../../datasource/go';
|
|
import { GolangVersionDatasource } from '../../datasource/golang-version';
|
|
import { isVersion } from '../../versioning/semver';
|
|
import type { PackageDependency, PackageFile } from '../types';
|
|
import type { MultiLineParseResult } from './types';
|
|
|
|
function getDep(
|
|
lineNumber: number,
|
|
match: RegExpMatchArray,
|
|
type: string
|
|
): PackageDependency {
|
|
const [, , currentValue] = match;
|
|
let [, depName] = match;
|
|
depName = depName.replace(regEx(/"/g), '');
|
|
const dep: PackageDependency = {
|
|
managerData: {
|
|
lineNumber,
|
|
},
|
|
depName,
|
|
depType: type,
|
|
currentValue,
|
|
};
|
|
if (isVersion(currentValue)) {
|
|
dep.datasource = GoDatasource.id;
|
|
} else {
|
|
dep.skipReason = 'unsupported-version';
|
|
}
|
|
const digestMatch = regEx(/v0\.0.0-\d{14}-([a-f0-9]{12})/).exec(currentValue);
|
|
if (digestMatch) {
|
|
[, dep.currentDigest] = digestMatch;
|
|
dep.digestOneAndOnly = true;
|
|
}
|
|
return dep;
|
|
}
|
|
|
|
function getGoDep(lineNumber: number, goVer: string): PackageDependency {
|
|
return {
|
|
managerData: {
|
|
lineNumber,
|
|
},
|
|
depName: 'go',
|
|
depType: 'golang',
|
|
currentValue: goVer,
|
|
datasource: GolangVersionDatasource.id,
|
|
versioning: 'npm',
|
|
rangeStrategy: 'replace',
|
|
};
|
|
}
|
|
|
|
export function extractPackageFile(content: string): PackageFile | null {
|
|
logger.trace({ content }, 'gomod.extractPackageFile()');
|
|
const constraints: Record<string, any> = {};
|
|
const deps: PackageDependency[] = [];
|
|
try {
|
|
const lines = content.split(newlineRegex);
|
|
for (let lineNumber = 0; lineNumber < lines.length; lineNumber += 1) {
|
|
const line = lines[lineNumber];
|
|
const goVer = line.startsWith('go ') ? line.replace('go ', '') : null;
|
|
if (goVer && semver.validRange(goVer)) {
|
|
const dep = getGoDep(lineNumber, goVer);
|
|
deps.push(dep);
|
|
constraints.go = line.replace('go ', '^');
|
|
}
|
|
const replaceMatch = regEx(
|
|
/^replace\s+[^\s]+[\s]+[=][>]\s+([^\s]+)\s+([^\s]+)/
|
|
).exec(line);
|
|
if (replaceMatch) {
|
|
const dep = getDep(lineNumber, replaceMatch, 'replace');
|
|
deps.push(dep);
|
|
}
|
|
const requireMatch = regEx(/^require\s+([^\s]+)\s+([^\s]+)/).exec(line);
|
|
if (requireMatch && !line.endsWith('// indirect')) {
|
|
logger.trace({ lineNumber }, `require line: "${line}"`);
|
|
const dep = getDep(lineNumber, requireMatch, 'require');
|
|
deps.push(dep);
|
|
}
|
|
if (line.trim() === 'require (') {
|
|
logger.trace(`Matched multi-line require on line ${lineNumber}`);
|
|
const matcher = regEx(/^\s+([^\s]+)\s+([^\s]+)/);
|
|
const { reachedLine, detectedDeps } = parseMultiLine(
|
|
lineNumber,
|
|
lines,
|
|
matcher,
|
|
'require'
|
|
);
|
|
lineNumber = reachedLine;
|
|
deps.push(...detectedDeps);
|
|
} else if (line.trim() === 'replace (') {
|
|
logger.trace(`Matched multi-line replace on line ${lineNumber}`);
|
|
const matcher = regEx(/^\s+[^\s]+[\s]+[=][>]\s+([^\s]+)\s+([^\s]+)/);
|
|
const { reachedLine, detectedDeps } = parseMultiLine(
|
|
lineNumber,
|
|
lines,
|
|
matcher,
|
|
'replace'
|
|
);
|
|
lineNumber = reachedLine;
|
|
deps.push(...detectedDeps);
|
|
}
|
|
}
|
|
} catch (err) /* istanbul ignore next */ {
|
|
logger.warn({ err }, 'Error extracting go modules');
|
|
}
|
|
if (!deps.length) {
|
|
return null;
|
|
}
|
|
return { constraints, deps };
|
|
}
|
|
|
|
function parseMultiLine(
|
|
startingLine: number,
|
|
lines: string[],
|
|
matchRegex: RegExp,
|
|
blockType: 'require' | 'replace'
|
|
): MultiLineParseResult {
|
|
const deps: PackageDependency[] = [];
|
|
let lineNumber = startingLine;
|
|
let line = '';
|
|
do {
|
|
lineNumber += 1;
|
|
line = lines[lineNumber];
|
|
const multiMatch = matchRegex.exec(line);
|
|
logger.trace(`${blockType}: "${line}"`);
|
|
if (multiMatch && !line.endsWith('// indirect')) {
|
|
logger.trace({ lineNumber }, `${blockType} line: "${line}"`);
|
|
const dep = getDep(lineNumber, multiMatch, blockType);
|
|
dep.managerData!.multiLine = true;
|
|
deps.push(dep);
|
|
} else if (line.trim() !== ')') {
|
|
logger.trace(`No multi-line match: ${line}`);
|
|
}
|
|
} while (line.trim() !== ')');
|
|
return { reachedLine: lineNumber, detectedDeps: deps };
|
|
}
|