Fix: parameter change from version: str to vinfo: VersionInfo

Using `version: str` meant that version.parse_version_info implicitly
used the "{pycalver}" pattern. The changed parameter vinfo: VersionInfo
now requires the caller (with access to cfg.version_pattern) to provide
the correctly parsed vinfo.
This commit is contained in:
Manuel Barkhau 2019-03-28 23:49:21 +01:00
parent e4a1ea64f0
commit 6d3dc6dfcd
3 changed files with 33 additions and 23 deletions

View file

@ -223,7 +223,8 @@ def _bump(cfg: config.Config, new_version: str, allow_dirty: bool = False) -> No
_assert_not_dirty(_vcs, filepaths, allow_dirty) _assert_not_dirty(_vcs, filepaths, allow_dirty)
try: try:
rewrite.rewrite(new_version, cfg.file_patterns) new_vinfo = version.parse_version_info(new_version, cfg.version_pattern)
rewrite.rewrite(new_vinfo, cfg.file_patterns)
except Exception as ex: except Exception as ex:
log.error(str(ex)) log.error(str(ex))
sys.exit(1) sys.exit(1)
@ -244,7 +245,8 @@ def _bump(cfg: config.Config, new_version: str, allow_dirty: bool = False) -> No
def _print_diff(cfg: config.Config, new_version: str) -> None: def _print_diff(cfg: config.Config, new_version: str) -> None:
diff: str = rewrite.diff(new_version, cfg.file_patterns) new_vinfo = version.parse_version_info(new_version, cfg.version_pattern)
diff: str = rewrite.diff(new_vinfo, cfg.file_patterns)
if sys.stdout.isatty(): if sys.stdout.isatty():
for line in diff.splitlines(): for line in diff.splitlines():

View file

@ -54,26 +54,25 @@ class NoPatternMatch(Exception):
def rewrite_lines( def rewrite_lines(
pattern_strs: typ.List[str], new_version: str, old_lines: typ.List[str] pattern_strs: typ.List[str], new_vinfo: version.VersionInfo, old_lines: typ.List[str]
) -> typ.List[str]: ) -> typ.List[str]:
"""Replace occurances of pattern_strs in old_lines with new_version. """Replace occurances of pattern_strs in old_lines with new_vinfo.
>>> new_vinfo = version.parse_version_info("v201811.0123-beta")
>>> pattern_strs = ['__version__ = "{pycalver}"'] >>> pattern_strs = ['__version__ = "{pycalver}"']
>>> rewrite_lines(pattern_strs, "v201811.0123-beta", ['__version__ = "v201809.0002-beta"']) >>> rewrite_lines(pattern_strs, new_vinfo, ['__version__ = "v201809.0002-beta"'])
['__version__ = "v201811.0123-beta"'] ['__version__ = "v201811.0123-beta"']
>>> pattern_strs = ['__version__ = "{pep440_version}"'] >>> pattern_strs = ['__version__ = "{pep440_version}"']
>>> rewrite_lines(pattern_strs, "v201811.0123-beta", ['__version__ = "201809.2b0"']) >>> rewrite_lines(pattern_strs, new_vinfo, ['__version__ = "201809.2b0"'])
['__version__ = "201811.123b0"'] ['__version__ = "201811.123b0"']
""" """
new_version_nfo = version.parse_version_info(new_version)
new_lines = old_lines[:] new_lines = old_lines[:]
found_patterns = set() found_patterns = set()
for m in parse.iter_matches(old_lines, pattern_strs): for m in parse.iter_matches(old_lines, pattern_strs):
found_patterns.add(m.pattern) found_patterns.add(m.pattern)
replacement = version.format_version(new_version_nfo, m.pattern) replacement = version.format_version(new_vinfo, m.pattern)
span_l, span_r = m.span span_l, span_r = m.span
new_line = m.line[:span_l] + replacement + m.line[span_r:] new_line = m.line[:span_l] + replacement + m.line[span_r:]
new_lines[m.lineno] = new_line new_lines[m.lineno] = new_line
@ -99,19 +98,20 @@ class RewrittenFileData(typ.NamedTuple):
def rfd_from_content( def rfd_from_content(
pattern_strs: typ.List[str], new_version: str, content: str pattern_strs: typ.List[str], new_vinfo: version.VersionInfo, content: str
) -> RewrittenFileData: ) -> RewrittenFileData:
r"""Rewrite pattern occurrences with version string. r"""Rewrite pattern occurrences with version string.
>>> new_vinfo = version.parse_version_info("v201809.0123")
>>> pattern_strs = ['__version__ = "{pycalver}"'] >>> pattern_strs = ['__version__ = "{pycalver}"']
>>> content = '__version__ = "v201809.0001-alpha"' >>> content = '__version__ = "v201809.0001-alpha"'
>>> rfd = rfd_from_content(pattern_strs, "v201809.0123", content) >>> rfd = rfd_from_content(pattern_strs, new_vinfo, content)
>>> rfd.new_lines >>> rfd.new_lines
['__version__ = "v201809.0123"'] ['__version__ = "v201809.0123"']
""" """
line_sep = detect_line_sep(content) line_sep = detect_line_sep(content)
old_lines = content.split(line_sep) old_lines = content.split(line_sep)
new_lines = rewrite_lines(pattern_strs, new_version, old_lines) new_lines = rewrite_lines(pattern_strs, new_vinfo, old_lines)
return RewrittenFileData("<path>", line_sep, old_lines, new_lines) return RewrittenFileData("<path>", line_sep, old_lines, new_lines)
@ -129,12 +129,13 @@ def _iter_file_paths(
def iter_rewritten( def iter_rewritten(
file_patterns: config.PatternsByGlob, new_version: str file_patterns: config.PatternsByGlob, new_vinfo: version.VersionInfo
) -> typ.Iterable[RewrittenFileData]: ) -> typ.Iterable[RewrittenFileData]:
r'''Iterate over files with version string replaced. r'''Iterate over files with version string replaced.
>>> file_patterns = {"src/pycalver/__init__.py": ['__version__ = "{pycalver}"']} >>> file_patterns = {"src/pycalver/__init__.py": ['__version__ = "{pycalver}"']}
>>> rewritten_datas = iter_rewritten(file_patterns, "v201809.0123") >>> new_vinfo = version.parse_version_info("v201809.0123")
>>> rewritten_datas = iter_rewritten(file_patterns, new_vinfo)
>>> rfd = list(rewritten_datas)[0] >>> rfd = list(rewritten_datas)[0]
>>> assert rfd.new_lines == [ >>> assert rfd.new_lines == [
... '# This file is part of the pycalver project', ... '# This file is part of the pycalver project',
@ -149,13 +150,14 @@ def iter_rewritten(
... ] ... ]
>>> >>>
''' '''
fh: typ.IO[str] fh: typ.IO[str]
for file_path, pattern_strs in _iter_file_paths(file_patterns): for file_path, pattern_strs in _iter_file_paths(file_patterns):
with file_path.open(mode="rt", encoding="utf-8") as fh: with file_path.open(mode="rt", encoding="utf-8") as fh:
content = fh.read() content = fh.read()
rfd = rfd_from_content(pattern_strs, new_version, content) rfd = rfd_from_content(pattern_strs, new_vinfo, content)
yield rfd._replace(path=str(file_path)) yield rfd._replace(path=str(file_path))
@ -177,11 +179,12 @@ def diff_lines(rfd: RewrittenFileData) -> typ.List[str]:
return list(lines) return list(lines)
def diff(new_version: str, file_patterns: config.PatternsByGlob) -> str: def diff(new_vinfo: version.VersionInfo, file_patterns: config.PatternsByGlob) -> str:
r"""Generate diffs of rewritten files. r"""Generate diffs of rewritten files.
>>> new_vinfo = version.parse_version_info("v201809.0123")
>>> file_patterns = {"src/pycalver/__init__.py": ['__version__ = "{pycalver}"']} >>> file_patterns = {"src/pycalver/__init__.py": ['__version__ = "{pycalver}"']}
>>> diff_str = diff("v201809.0123", file_patterns) >>> diff_str = diff(new_vinfo, file_patterns)
>>> lines = diff_str.split("\n") >>> lines = diff_str.split("\n")
>>> lines[:2] >>> lines[:2]
['--- src/pycalver/__init__.py', '+++ src/pycalver/__init__.py'] ['--- src/pycalver/__init__.py', '+++ src/pycalver/__init__.py']
@ -199,7 +202,7 @@ def diff(new_version: str, file_patterns: config.PatternsByGlob) -> str:
content = fh.read() content = fh.read()
try: try:
rfd = rfd_from_content(pattern_strs, new_version, content) rfd = rfd_from_content(pattern_strs, new_vinfo, content)
except NoPatternMatch: except NoPatternMatch:
errmsg = f"No patterns matched for '{file_path}'" errmsg = f"No patterns matched for '{file_path}'"
raise NoPatternMatch(errmsg) raise NoPatternMatch(errmsg)
@ -216,11 +219,11 @@ def diff(new_version: str, file_patterns: config.PatternsByGlob) -> str:
return full_diff return full_diff
def rewrite(new_version: str, file_patterns: config.PatternsByGlob) -> None: def rewrite(new_vinfo: version.VersionInfo, file_patterns: config.PatternsByGlob) -> None:
"""Rewrite project files, updating each with the new version.""" """Rewrite project files, updating each with the new version."""
fh: typ.IO[str] fh: typ.IO[str]
for file_data in iter_rewritten(file_patterns, new_version): for file_data in iter_rewritten(file_patterns, new_vinfo):
new_content = file_data.line_sep.join(file_data.new_lines) new_content = file_data.line_sep.join(file_data.new_lines)
with io.open(file_data.path, mode="wt", encoding="utf-8") as fh: with io.open(file_data.path, mode="wt", encoding="utf-8") as fh:
fh.write(new_content) fh.write(new_content)

View file

@ -2,6 +2,7 @@ import copy
from pycalver import config from pycalver import config
from pycalver import rewrite from pycalver import rewrite
from pycalver import version
from . import util from . import util
@ -15,7 +16,8 @@ __version__ = "v201809.0002-beta"
def test_rewrite_lines(): def test_rewrite_lines():
old_lines = REWRITE_FIXTURE.splitlines() old_lines = REWRITE_FIXTURE.splitlines()
patterns = ['__version__ = "{pycalver}"'] patterns = ['__version__ = "{pycalver}"']
new_lines = rewrite.rewrite_lines(patterns, "v201911.0003", old_lines) new_vinfo = version.parse_version_info("v201911.0003")
new_lines = rewrite.rewrite_lines(patterns, new_vinfo, old_lines)
assert len(new_lines) == len(old_lines) assert len(new_lines) == len(old_lines)
assert "v201911.0003" not in "\n".join(old_lines) assert "v201911.0003" not in "\n".join(old_lines)
@ -28,7 +30,8 @@ def test_rewrite_final():
old_lines = REWRITE_FIXTURE.splitlines() old_lines = REWRITE_FIXTURE.splitlines()
patterns = ['__version__ = "v{year}{month}.{build_no}-{release_tag}"'] patterns = ['__version__ = "v{year}{month}.{build_no}-{release_tag}"']
new_lines = rewrite.rewrite_lines(patterns, "v201911.0003", old_lines) new_vinfo = version.parse_version_info("v201911.0003")
new_lines = rewrite.rewrite_lines(patterns, new_vinfo, old_lines)
assert len(new_lines) == len(old_lines) assert len(new_lines) == len(old_lines)
assert "v201911.0003" not in "\n".join(old_lines) assert "v201911.0003" not in "\n".join(old_lines)
@ -92,6 +95,8 @@ def test_error_bad_pattern():
patterns["setup.py"] = patterns["setup.py"][0] + "invalid" patterns["setup.py"] = patterns["setup.py"][0] + "invalid"
try: try:
list(rewrite.diff("v201809.1234", patterns)) new_vinfo = version.parse_version_info("v201809.1234")
list(rewrite.diff(new_vinfo, patterns))
assert False, "expected rewrite.NoPatternMatch"
except rewrite.NoPatternMatch as ex: except rewrite.NoPatternMatch as ex:
assert "setup.py" in str(ex) assert "setup.py" in str(ex)