feat: pip requirements.txt ignore (#2676)

Adds support for “# renovate: ignore” command in requirements.txt files.
This commit is contained in:
Rhys Arkins 2018-10-23 06:09:33 +02:00 committed by GitHub
parent 255f69812a
commit ce8926a003
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
7 changed files with 115 additions and 4 deletions

View file

@ -27,14 +27,28 @@ function extractDependencies(content) {
const regex = new RegExp(`^(${packagePattern})(${specifierPattern})$`, 'g'); const regex = new RegExp(`^(${packagePattern})(${specifierPattern})$`, 'g');
const deps = content const deps = content
.split('\n') .split('\n')
.map((line, lineNumber) => { .map((rawline, lineNumber) => {
let dep = {};
const [line, comment] = rawline.split('#').map(part => part.trim());
if (comment && comment.match(/^(renovate|pyup):/)) {
const command = comment
.split('#')[0]
.split(':')[1]
.trim();
if (command === 'ignore') {
dep.skipReason = 'ignored';
} else {
logger.info('Unknown pip_requirements command: ' + command);
}
}
regex.lastIndex = 0; regex.lastIndex = 0;
const matches = regex.exec(line); const matches = regex.exec(line);
if (!matches) { if (!matches) {
return null; return null;
} }
const [, depName, currentValue] = matches; const [, depName, currentValue] = matches;
const dep = { dep = {
...dep,
depName, depName,
currentValue, currentValue,
lineNumber, lineNumber,

View file

@ -6,7 +6,12 @@ function updateDependency(fileContent, upgrade) {
try { try {
logger.debug(`pip_requirements.updateDependency(): ${upgrade.newValue}`); logger.debug(`pip_requirements.updateDependency(): ${upgrade.newValue}`);
const lines = fileContent.split('\n'); const lines = fileContent.split('\n');
lines[upgrade.lineNumber] = `${upgrade.depName}${upgrade.newValue}`; const oldValue = lines[upgrade.lineNumber];
const newValue = oldValue.replace(
/^.+?(\s.*)?$/,
`${upgrade.depName}${upgrade.newValue}$1`
);
lines[upgrade.lineNumber] = newValue;
return lines.join('\n'); return lines.join('\n');
} catch (err) { } catch (err) {
logger.info({ err }, 'Error setting new package version'); logger.info({ err }, 'Error setting new package version');

View file

@ -0,0 +1,5 @@
Django==1 # renovate
distribute==0.6.27 # renovate: ignore
dj-database-url==0.2 # pyup: nothing
psycopg2==2.4.5 # renovate:
wsgiref==0.1.2 # something else

View file

@ -74,3 +74,44 @@ Array [
}, },
] ]
`; `;
exports[`lib/manager/pip_requirements/extract extractDependencies() handles comments and commands 1`] = `
Array [
Object {
"currentValue": "==1",
"depName": "Django",
"lineNumber": 0,
"purl": "pkg:pypi/Django",
"versionScheme": "pep440",
},
Object {
"currentValue": "==0.6.27",
"depName": "distribute",
"lineNumber": 1,
"purl": "pkg:pypi/distribute",
"skipReason": "ignored",
"versionScheme": "pep440",
},
Object {
"currentValue": "==0.2",
"depName": "dj-database-url",
"lineNumber": 2,
"purl": "pkg:pypi/dj-database-url",
"versionScheme": "pep440",
},
Object {
"currentValue": "==2.4.5",
"depName": "psycopg2",
"lineNumber": 3,
"purl": "pkg:pypi/psycopg2",
"versionScheme": "pep440",
},
Object {
"currentValue": "==0.1.2",
"depName": "wsgiref",
"lineNumber": 4,
"purl": "pkg:pypi/wsgiref",
"versionScheme": "pep440",
},
]
`;

View file

@ -0,0 +1,20 @@
// Jest Snapshot v1, https://goo.gl/fbAQLP
exports[`manager/pip_requirements/update updateDependency replaces existing value 1`] = `
"--index-url http://example.com/private-pypi/
# simple comment
url==1.0.1
some-other-package==1.0.0
not_semver==1.9
"
`;
exports[`manager/pip_requirements/update updateDependency replaces existing value with comment 1`] = `
"Django==1 # renovate
distribute==0.6.27 # renovate: ignore
dj-database-url==0.2 # pyup: nothing
psycopg2==2.4.6 # renovate:
wsgiref==0.1.2 # something else
"
`;

View file

@ -11,6 +11,10 @@ const requirements2 = fs.readFileSync(
'test/_fixtures/pip_requirements/requirements2.txt', 'test/_fixtures/pip_requirements/requirements2.txt',
'utf8' 'utf8'
); );
const requirements3 = fs.readFileSync(
'test/_fixtures/pip_requirements/requirements3.txt',
'utf8'
);
describe('lib/manager/pip_requirements/extract', () => { describe('lib/manager/pip_requirements/extract', () => {
describe('extractDependencies()', () => { describe('extractDependencies()', () => {
@ -31,5 +35,10 @@ describe('lib/manager/pip_requirements/extract', () => {
expect(res).toMatchSnapshot(); expect(res).toMatchSnapshot();
expect(res).toHaveLength(5); expect(res).toHaveLength(5);
}); });
it('handles comments and commands', () => {
const res = extractDependencies(requirements3, config).deps;
expect(res).toMatchSnapshot();
expect(res).toHaveLength(5);
});
}); });
}); });

View file

@ -8,15 +8,21 @@ const requirements = fs.readFileSync(
'utf8' 'utf8'
); );
const requirements3 = fs.readFileSync(
'test/_fixtures/pip_requirements/requirements3.txt',
'utf8'
);
describe('manager/pip_requirements/update', () => { describe('manager/pip_requirements/update', () => {
describe('updateDependency', () => { describe('updateDependency', () => {
it('replaces existing value', () => { it('replaces existing value', () => {
const upgrade = { const upgrade = {
depName: 'url', depName: 'url',
lineNumber: 2, lineNumber: 2,
newValue: '1.0.1', newValue: '==1.0.1',
}; };
const res = updateDependency(requirements, upgrade); const res = updateDependency(requirements, upgrade);
expect(res).toMatchSnapshot();
expect(res).not.toEqual(requirements); expect(res).not.toEqual(requirements);
expect(res.includes(upgrade.newValue)).toBe(true); expect(res.includes(upgrade.newValue)).toBe(true);
}); });
@ -24,5 +30,16 @@ describe('manager/pip_requirements/update', () => {
const res = updateDependency(null, null); const res = updateDependency(null, null);
expect(res).toBe(null); expect(res).toBe(null);
}); });
it('replaces existing value with comment', () => {
const upgrade = {
depName: 'psycopg2',
lineNumber: 3,
newValue: '==2.4.6',
};
const res = updateDependency(requirements3, upgrade);
expect(res).toMatchSnapshot();
expect(res).not.toEqual(requirements3);
expect(res.includes(upgrade.newValue)).toBe(true);
});
}); });
}); });