Add support for sections in requirements-fixer

This commit is contained in:
Michael Aquilina 2024-08-16 14:41:47 +01:00
parent 429455474b
commit cc5d4504d0
No known key found for this signature in database
GPG key ID: 636066730B056BD1
2 changed files with 58 additions and 28 deletions

View file

@ -66,10 +66,7 @@ class Requirement:
def fix_requirements(f: IO[bytes]) -> int:
requirements: list[Requirement] = []
before = list(f)
after: list[bytes] = []
before_string = b''.join(before)
# adds new line in case one is missing
@ -81,17 +78,21 @@ def fix_requirements(f: IO[bytes]) -> int:
if before_string.strip() == b'':
return PASS
# split requirements into sections based on newlines
sections: list[list[Requirement]] = [[]]
for line in before:
requirements = sections[-1]
# If the most recent requirement object has a value, then it's
# time to start building the next requirement object.
if not len(requirements) or requirements[-1].is_complete():
requirements.append(Requirement())
requirement = requirements[-1]
# If we see a newline before any requirements, then this is a
# top of file comment.
# top of section comment.
if len(requirements) == 1 and line.strip() == b'':
if (
len(requirement.comments) and
@ -100,38 +101,48 @@ def fix_requirements(f: IO[bytes]) -> int:
requirement.value = b'\n'
else:
requirement.comments.append(line)
elif line.lstrip() == b'':
sections.append([])
elif line.lstrip().startswith(b'#') or line.strip() == b'':
requirement.comments.append(line)
else:
requirement.append_value(line)
# if a file ends in a comment, preserve it at the end
if requirements[-1].value is None:
rest = requirements.pop().comments
else:
rest = []
after_string = b''
# find and remove pkg-resources==0.0.0
# which is automatically added by broken pip package under Debian
requirements = [
req for req in requirements
if req.value not in [
b'pkg-resources==0.0.0\n',
b'pkg_resources==0.0.0\n',
for requirements in sections:
after: list[bytes] = []
if len(after_string) > 0:
after_string += b'\n'
# if a section ends in a comment, preserve it at the end
if requirements[-1].value is None:
rest = requirements.pop().comments
else:
rest = []
# find and remove pkg-resources==0.0.0
# which is automatically added by broken pip package under Debian
requirements = [
req for req in requirements
if req.value not in [
b'pkg-resources==0.0.0\n',
b'pkg_resources==0.0.0\n',
]
]
]
# sort the requirements and remove duplicates
prev = None
for requirement in sorted(requirements):
after.extend(requirement.comments)
assert requirement.value, requirement.value
if prev is None or requirement.value != prev.value:
after.append(requirement.value)
prev = requirement
after.extend(rest)
# sort the requirements and remove duplicates
prev = None
for requirement in sorted(requirements):
after.extend(requirement.comments)
assert requirement.value, requirement.value
if prev is None or requirement.value != prev.value:
after.append(requirement.value)
prev = requirement
after.extend(rest)
after_string = b''.join(after)
after_string += b''.join(after)
if before_string == after_string:
return PASS

View file

@ -107,6 +107,25 @@ from pre_commit_hooks.requirements_txt_fixer import Requirement
PASS,
b'a=2.0.0 \\\n --hash=sha256:abcd\nb==1.0.0\n',
),
(
(
b'-c external.txt\n'
b'\n'
b'Peach\n'
b'apple\n'
b'banana\n'
b'pear\n'
),
FAIL,
(
b'-c external.txt\n'
b'\n'
b'apple\n'
b'banana\n'
b'Peach\n'
b'pear\n'
),
),
),
)
def test_integration(input_s, expected_retval, output, tmpdir):