mirror of
https://github.com/renovatebot/renovate.git
synced 2025-01-11 13:48:55 +00:00
134 lines
3.6 KiB
TypeScript
134 lines
3.6 KiB
TypeScript
// based on https://www.python.org/dev/peps/pep-0508/#names
|
|
import { RANGE_PATTERN } from '@renovatebot/pep440';
|
|
import { logger } from '../../../logger';
|
|
import { newlineRegex, regEx } from '../../../util/regex';
|
|
import { PypiDatasource } from '../../datasource/pypi';
|
|
import { normalizePythonDepName } from '../../datasource/pypi/common';
|
|
import type {
|
|
MaybePromise,
|
|
PackageDependency,
|
|
PackageFileContent,
|
|
} from '../types';
|
|
|
|
function getSectionName(str: string): string {
|
|
const [, sectionName] = regEx(/^\[\s*([^\s]+)\s*]\s*$/).exec(str) ?? [];
|
|
return sectionName;
|
|
}
|
|
|
|
function getSectionRecord(str: string): string {
|
|
const [, sectionRecord] = regEx(/^([^\s]+)\s*=/).exec(str) ?? [];
|
|
return sectionRecord;
|
|
}
|
|
|
|
function getDepType(
|
|
section: string | null,
|
|
record: string | null,
|
|
): null | string {
|
|
if (section === 'options') {
|
|
if (record === 'install_requires') {
|
|
return 'install';
|
|
}
|
|
if (record === 'setup_requires') {
|
|
return 'setup';
|
|
}
|
|
if (record === 'tests_require') {
|
|
return 'test';
|
|
}
|
|
}
|
|
if (section === 'options.extras_require') {
|
|
return 'extra';
|
|
}
|
|
return null;
|
|
}
|
|
|
|
function parseDep(
|
|
line: string,
|
|
section: string | null,
|
|
record: string | null,
|
|
): PackageDependency | null {
|
|
const packagePattern = '[a-zA-Z0-9]|[a-zA-Z0-9][a-zA-Z0-9._-]*[a-zA-Z0-9]';
|
|
const extrasPattern = '(?:\\s*\\[[^\\]]+\\])?';
|
|
|
|
const rangePattern: string = RANGE_PATTERN;
|
|
const specifierPartPattern = `\\s*${rangePattern.replace(
|
|
regEx(/\?<\w+>/g),
|
|
'?:',
|
|
)}`;
|
|
const specifierPattern = `${specifierPartPattern}(?:\\s*,${specifierPartPattern})*`;
|
|
const dependencyPattern = `(${packagePattern})(${extrasPattern})(${specifierPattern})`;
|
|
|
|
const pkgRegex = regEx(`^(${packagePattern})$`);
|
|
const pkgValRegex = regEx(`^${dependencyPattern}$`);
|
|
|
|
const depType = getDepType(section, record);
|
|
if (!depType) {
|
|
return null;
|
|
}
|
|
|
|
const [lineNoEnvMarkers] = line.split(';').map((part) => part.trim());
|
|
const packageMatches =
|
|
pkgValRegex.exec(lineNoEnvMarkers) ?? pkgRegex.exec(lineNoEnvMarkers);
|
|
|
|
if (!packageMatches) {
|
|
return null;
|
|
}
|
|
|
|
const [, depName, , currVal] = packageMatches;
|
|
const currentValue = currVal?.trim();
|
|
|
|
const dep: PackageDependency = {
|
|
depName,
|
|
packageName: normalizePythonDepName(depName),
|
|
currentValue,
|
|
datasource: PypiDatasource.id,
|
|
depType,
|
|
};
|
|
|
|
if (currentValue?.startsWith('==')) {
|
|
dep.currentVersion = currentValue.replace(/^==\s*/, '');
|
|
}
|
|
|
|
return dep;
|
|
}
|
|
|
|
export function extractPackageFile(
|
|
content: string,
|
|
): MaybePromise<PackageFileContent | null> {
|
|
logger.trace('setup-cfg.extractPackageFile()');
|
|
|
|
let sectionName: string | null = null;
|
|
let sectionRecord: string | null = null;
|
|
|
|
const deps: PackageDependency[] = [];
|
|
|
|
content
|
|
.split(newlineRegex)
|
|
.map((line) => line.replace(regEx(/#.*$/), '').trimEnd())
|
|
.forEach((rawLine) => {
|
|
let line = rawLine;
|
|
const newSectionName = getSectionName(line);
|
|
const newSectionRecord = getSectionRecord(line);
|
|
if (newSectionName) {
|
|
sectionName = newSectionName;
|
|
}
|
|
if (newSectionRecord) {
|
|
sectionRecord = newSectionRecord;
|
|
// Probably there are also requirements in this line.
|
|
line = rawLine.replace(regEx(/^[^=]*=\s*/), '');
|
|
line.split(';').forEach((part) => {
|
|
const dep = parseDep(part, sectionName, sectionRecord);
|
|
if (dep) {
|
|
deps.push(dep);
|
|
}
|
|
});
|
|
return;
|
|
}
|
|
|
|
const dep = parseDep(line, sectionName, sectionRecord);
|
|
if (dep) {
|
|
deps.push(dep);
|
|
}
|
|
});
|
|
|
|
return deps.length ? { deps } : null;
|
|
}
|