renovate/lib/manager/bundler/extract.ts
2021-03-02 21:44:55 +01:00

204 lines
6.6 KiB
TypeScript

import * as datasourceRubygems from '../../datasource/rubygems';
import { logger } from '../../logger';
import { SkipReason } from '../../types';
import { readLocalFile } from '../../util/fs';
import { regEx } from '../../util/regex';
import type { PackageDependency, PackageFile } from '../types';
import { extractLockFileEntries } from './locked-version';
export async function extractPackageFile(
content: string,
fileName?: string
): Promise<PackageFile | null> {
const res: PackageFile = {
registryUrls: [],
deps: [],
};
const lines = content.split('\n');
const delimiters = ['"', "'"];
for (let lineNumber = 0; lineNumber < lines.length; lineNumber += 1) {
const line = lines[lineNumber];
let sourceMatch: RegExpMatchArray;
for (const delimiter of delimiters) {
sourceMatch =
sourceMatch ||
regEx(`^source ${delimiter}([^${delimiter}]+)${delimiter}\\s*$`).exec(
line
);
}
if (sourceMatch) {
res.registryUrls.push(sourceMatch[1]);
}
let rubyMatch: RegExpMatchArray;
for (const delimiter of delimiters) {
rubyMatch =
rubyMatch ||
regEx(`^ruby ${delimiter}([^${delimiter}]+)${delimiter}`).exec(line);
}
if (rubyMatch) {
res.constraints = { ruby: rubyMatch[1] };
}
const gemMatchRegex = /^\s*gem\s+(['"])(?<depName>[^'"]+)\1(\s*,\s*(?<currentValue>(['"])[^'"]+\5(\s*,\s*\5[^'"]+\5)?))?/;
const gemMatch = gemMatchRegex.exec(line);
if (gemMatch) {
const dep: PackageDependency = {
depName: gemMatch.groups.depName,
managerData: { lineNumber },
};
if (gemMatch.groups.currentValue) {
const currentValue = gemMatch.groups.currentValue;
dep.currentValue = /\s*,\s*/.test(currentValue)
? currentValue
: currentValue.slice(1, -1);
} else {
dep.skipReason = SkipReason.NoVersion;
}
if (!dep.skipReason) {
dep.datasource = datasourceRubygems.id;
}
res.deps.push(dep);
}
const groupMatch = /^group\s+(.*?)\s+do/.exec(line);
if (groupMatch) {
const depTypes = groupMatch[1]
.split(',')
.map((group) => group.trim())
.map((group) => group.replace(/^:/, ''));
const groupLineNumber = lineNumber;
let groupContent = '';
let groupLine = '';
while (lineNumber < lines.length && groupLine !== 'end') {
lineNumber += 1;
groupLine = lines[lineNumber];
if (groupLine !== 'end') {
groupContent += (groupLine || '').replace(/^ {2}/, '') + '\n';
}
}
const groupRes = await extractPackageFile(groupContent);
if (groupRes) {
res.deps = res.deps.concat(
groupRes.deps.map((dep) => ({
...dep,
depTypes,
managerData: {
lineNumber:
Number(dep.managerData.lineNumber) + groupLineNumber + 1,
},
}))
);
}
}
for (const delimiter of delimiters) {
const sourceBlockMatch = regEx(
`^source\\s+${delimiter}(.*?)${delimiter}\\s+do`
).exec(line);
if (sourceBlockMatch) {
const repositoryUrl = sourceBlockMatch[1];
const sourceLineNumber = lineNumber;
let sourceContent = '';
let sourceLine = '';
while (lineNumber < lines.length && sourceLine.trim() !== 'end') {
lineNumber += 1;
sourceLine = lines[lineNumber];
// istanbul ignore if
if (sourceLine === null || sourceLine === undefined) {
logger.info({ content, fileName }, 'Undefined sourceLine');
sourceLine = 'end';
}
if (sourceLine !== 'end') {
sourceContent += sourceLine.replace(/^ {2}/, '') + '\n';
}
}
const sourceRes = await extractPackageFile(sourceContent);
if (sourceRes) {
res.deps = res.deps.concat(
sourceRes.deps.map((dep) => ({
...dep,
registryUrls: [repositoryUrl],
managerData: {
lineNumber:
Number(dep.managerData.lineNumber) + sourceLineNumber + 1,
},
}))
);
}
}
}
const platformsMatch = /^platforms\s+(.*?)\s+do/.test(line);
if (platformsMatch) {
const platformsLineNumber = lineNumber;
let platformsContent = '';
let platformsLine = '';
while (lineNumber < lines.length && platformsLine !== 'end') {
lineNumber += 1;
platformsLine = lines[lineNumber];
if (platformsLine !== 'end') {
platformsContent += platformsLine.replace(/^ {2}/, '') + '\n';
}
}
const platformsRes = await extractPackageFile(platformsContent);
if (platformsRes) {
res.deps = res.deps.concat(
// eslint-disable-next-line no-loop-func
platformsRes.deps.map((dep) => ({
...dep,
managerData: {
lineNumber:
Number(dep.managerData.lineNumber) + platformsLineNumber + 1,
},
}))
);
}
}
const ifMatch = /^if\s+(.*?)/.test(line);
if (ifMatch) {
const ifLineNumber = lineNumber;
let ifContent = '';
let ifLine = '';
while (lineNumber < lines.length && ifLine !== 'end') {
lineNumber += 1;
ifLine = lines[lineNumber];
if (ifLine !== 'end') {
ifContent += ifLine.replace(/^ {2}/, '') + '\n';
}
}
const ifRes = await extractPackageFile(ifContent);
if (ifRes) {
res.deps = res.deps.concat(
// eslint-disable-next-line no-loop-func
ifRes.deps.map((dep) => ({
...dep,
managerData: {
lineNumber: Number(dep.managerData.lineNumber) + ifLineNumber + 1,
},
}))
);
}
}
}
if (!res.deps.length && !res.registryUrls.length) {
return null;
}
if (fileName) {
const gemfileLock = fileName + '.lock';
const lockContent = await readLocalFile(gemfileLock, 'utf8');
if (lockContent) {
logger.debug({ packageFile: fileName }, 'Found Gemfile.lock file');
res.lockFiles = [gemfileLock];
const lockedEntries = extractLockFileEntries(lockContent);
for (const dep of res.deps) {
const lockedDepValue = lockedEntries.get(dep.depName);
if (lockedDepValue) {
dep.lockedVersion = lockedDepValue;
}
}
const bundledWith = /\nBUNDLED WITH\n\s+(.*?)(\n|$)/.exec(lockContent);
if (bundledWith) {
res.constraints = res.constraints || {};
res.constraints.bundler = bundledWith[1];
}
}
}
return res;
}