mirror of
https://github.com/renovatebot/renovate.git
synced 2025-01-15 09:06:25 +00:00
136 lines
3.7 KiB
TypeScript
136 lines
3.7 KiB
TypeScript
import is from '@sindresorhus/is';
|
|
import toml from 'toml';
|
|
import { RANGE_PATTERN } from '@renovate/pep440/lib/specifier';
|
|
import { logger } from '../../logger';
|
|
import { PackageFile, PackageDependency } from '../common';
|
|
|
|
// based on https://www.python.org/dev/peps/pep-0508/#names
|
|
const packageRegex = /^([A-Z0-9]|[A-Z0-9][A-Z0-9._-]*[A-Z0-9])$/i;
|
|
const rangePattern = RANGE_PATTERN;
|
|
|
|
const specifierPartPattern = `\\s*${rangePattern.replace(
|
|
/\?<\w+>/g,
|
|
'?:'
|
|
)}\\s*`;
|
|
const specifierPattern = `${specifierPartPattern}(?:,${specifierPartPattern})*`;
|
|
interface PipSource {
|
|
name: string;
|
|
url: string;
|
|
}
|
|
|
|
interface PipFile {
|
|
source: PipSource[];
|
|
|
|
packages?: Record<string, PipRequirement>;
|
|
'dev-packages'?: Record<string, PipRequirement>;
|
|
}
|
|
|
|
interface PipRequirement {
|
|
index?: string;
|
|
version?: string;
|
|
path?: string;
|
|
file?: string;
|
|
git?: string;
|
|
}
|
|
|
|
function extractFromSection(
|
|
pipfile: PipFile,
|
|
section: 'packages' | 'dev-packages'
|
|
): PackageDependency[] {
|
|
if (!(section in pipfile)) {
|
|
return [];
|
|
}
|
|
const specifierRegex = new RegExp(`^${specifierPattern}$`);
|
|
const pipfileSection = pipfile[section];
|
|
|
|
const deps = Object.entries(pipfileSection)
|
|
.map(x => {
|
|
const [depName, requirements] = x;
|
|
let currentValue: string;
|
|
let nestedVersion: boolean;
|
|
let skipReason: string;
|
|
if (requirements.git) {
|
|
skipReason = 'git-dependency';
|
|
} else if (requirements.file) {
|
|
skipReason = 'file-dependency';
|
|
} else if (requirements.path) {
|
|
skipReason = 'local-dependency';
|
|
} else if (requirements.version) {
|
|
currentValue = requirements.version;
|
|
nestedVersion = true;
|
|
} else if (is.object(requirements)) {
|
|
skipReason = 'any-version';
|
|
} else {
|
|
currentValue = requirements;
|
|
}
|
|
if (currentValue === '*') {
|
|
skipReason = 'any-version';
|
|
}
|
|
if (!skipReason) {
|
|
const packageMatches = packageRegex.exec(depName);
|
|
if (!packageMatches) {
|
|
logger.info(
|
|
`Skipping dependency with malformed package name "${depName}".`
|
|
);
|
|
skipReason = 'invalid-name';
|
|
}
|
|
const specifierMatches = specifierRegex.exec(currentValue);
|
|
if (!specifierMatches) {
|
|
logger.debug(
|
|
`Skipping dependency with malformed version specifier "${currentValue}".`
|
|
);
|
|
skipReason = 'invalid-version';
|
|
}
|
|
}
|
|
const dep: PackageDependency = {
|
|
depType: section,
|
|
depName,
|
|
managerData: {},
|
|
};
|
|
if (currentValue) dep.currentValue = currentValue;
|
|
if (skipReason) {
|
|
dep.skipReason = skipReason;
|
|
} else {
|
|
dep.datasource = 'pypi';
|
|
}
|
|
if (nestedVersion) dep.managerData.nestedVersion = nestedVersion;
|
|
if (requirements.index) {
|
|
if (is.array(pipfile.source)) {
|
|
const source = pipfile.source.find(
|
|
item => item.name === requirements.index
|
|
);
|
|
if (source) {
|
|
dep.registryUrls = [source.url];
|
|
}
|
|
}
|
|
}
|
|
return dep;
|
|
})
|
|
.filter(Boolean);
|
|
return deps;
|
|
}
|
|
|
|
export function extractPackageFile(content: string): PackageFile | null {
|
|
logger.debug('pipenv.extractPackageFile()');
|
|
|
|
let pipfile: PipFile;
|
|
try {
|
|
pipfile = toml.parse(content);
|
|
} catch (err) {
|
|
logger.debug({ err }, 'Error parsing Pipfile');
|
|
return null;
|
|
}
|
|
const res: PackageFile = { deps: [] };
|
|
if (pipfile.source) {
|
|
res.registryUrls = pipfile.source.map(source => source.url);
|
|
}
|
|
|
|
res.deps = [
|
|
...extractFromSection(pipfile, 'packages'),
|
|
...extractFromSection(pipfile, 'dev-packages'),
|
|
];
|
|
if (res.deps.length) {
|
|
return res;
|
|
}
|
|
return null;
|
|
}
|