bumpver/src/pycalver/rewrite.py

215 lines
6.8 KiB
Python
Raw Normal View History

2018-09-02 21:48:12 +02:00
# This file is part of the pycalver project
2019-02-22 10:56:47 +01:00
# https://gitlab.com/mbarkhau/pycalver
2018-09-02 21:48:12 +02:00
#
2019-02-22 10:56:47 +01:00
# Copyright (c) 2019 Manuel Barkhau (mbarkhau@gmail.com) - MIT License
2018-09-02 21:48:12 +02:00
# SPDX-License-Identifier: MIT
2018-11-15 22:16:16 +01:00
"""Rewrite files, updating occurences of version strings."""
2018-09-02 21:48:12 +02:00
2018-09-03 22:23:51 +02:00
import io
2019-02-21 15:41:06 +01:00
import glob
2018-09-02 21:48:12 +02:00
import difflib
2018-09-03 22:23:51 +02:00
import logging
import typing as typ
2019-02-21 15:41:06 +01:00
import pathlib2 as pl
2018-09-03 22:23:51 +02:00
from . import parse
2018-11-15 22:16:16 +01:00
from . import config
2018-12-09 14:49:13 +01:00
from . import version
from . import patterns
2018-09-02 21:48:12 +02:00
2018-09-02 21:48:12 +02:00
log = logging.getLogger("pycalver.rewrite")
2018-11-15 22:16:16 +01:00
def detect_line_sep(content: str) -> str:
r"""Parse line separator from content.
>>> detect_line_sep('\r\n')
'\r\n'
>>> detect_line_sep('\r')
'\r'
>>> detect_line_sep('\n')
'\n'
>>> detect_line_sep('')
'\n'
"""
2018-11-11 15:40:16 +01:00
if "\r\n" in content:
return "\r\n"
elif "\r" in content:
return "\r"
else:
return "\n"
2018-11-04 21:11:42 +01:00
def rewrite_lines(
pattern_strs: typ.List[str], new_version: str, old_lines: typ.List[str]
2018-11-04 21:11:42 +01:00
) -> typ.List[str]:
"""Replace occurances of pattern_strs in old_lines with new_version.
2018-11-15 22:16:16 +01:00
>>> pattern_strs = ['__version__ = "{pycalver}"']
>>> rewrite_lines(pattern_strs, "v201811.0123-beta", ['__version__ = "v201809.0002-beta"'])
['__version__ = "v201811.0123-beta"']
>>> pattern_strs = ['__version__ = "{pep440_version}"']
>>> rewrite_lines(pattern_strs, "v201811.0123-beta", ['__version__ = "201809.2b0"'])
['__version__ = "201811.123b0"']
2018-11-15 22:16:16 +01:00
"""
new_version_nfo = version.parse_version_info(new_version)
2018-09-03 22:23:51 +02:00
new_lines = old_lines[:]
found_patterns = set()
2018-11-04 21:11:42 +01:00
for m in parse.iter_matches(old_lines, pattern_strs):
found_patterns.add(m.pattern)
replacement = version.format_version(new_version_nfo, m.pattern)
2018-11-04 21:11:42 +01:00
span_l, span_r = m.span
new_line = m.line[:span_l] + replacement + m.line[span_r:]
new_lines[m.lineno] = new_line
non_matched_patterns = set(pattern_strs) - found_patterns
if non_matched_patterns:
for non_matched_pattern in non_matched_patterns:
log.error(f"No match for pattern '{non_matched_pattern}'")
compiled_pattern = patterns._compile_pattern(non_matched_pattern)
log.error(f"Pattern compiles to regex '{compiled_pattern}'")
2019-02-22 10:40:31 +01:00
raise ValueError("Invalid pattern(s)")
else:
return new_lines
2018-11-04 21:11:42 +01:00
2018-11-15 22:16:16 +01:00
class RewrittenFileData(typ.NamedTuple):
"""Container for line-wise content of rewritten files."""
path : str
line_sep : str
old_lines: typ.List[str]
new_lines: typ.List[str]
def rfd_from_content(
pattern_strs: typ.List[str], new_version: str, content: str
) -> RewrittenFileData:
r"""Rewrite pattern occurrences with version string.
>>> pattern_strs = ['__version__ = "{pycalver}"']
>>> content = '__version__ = "v201809.0001-alpha"'
>>> rfd = rfd_from_content(pattern_strs, "v201809.0123", content)
>>> rfd.new_lines
['__version__ = "v201809.0123"']
"""
line_sep = detect_line_sep(content)
old_lines = content.split(line_sep)
new_lines = rewrite_lines(pattern_strs, new_version, old_lines)
return RewrittenFileData("<path>", line_sep, old_lines, new_lines)
2019-02-21 15:41:06 +01:00
def _iter_file_paths(
file_patterns: config.PatternsByGlob
) -> typ.Iterable[typ.Tuple[pl.Path, config.Patterns]]:
for globstr, pattern_strs in file_patterns.items():
2019-02-21 16:30:27 +01:00
file_paths = glob.glob(globstr)
if not any(file_paths):
errmsg = f"No files found for path/glob '{globstr}'"
raise ValueError(errmsg)
for file_path_str in file_paths:
2019-02-21 15:41:06 +01:00
file_path = pl.Path(file_path_str)
yield (file_path, pattern_strs)
2019-02-21 15:41:06 +01:00
def iter_rewritten(
2019-02-21 15:41:06 +01:00
file_patterns: config.PatternsByGlob, new_version: str
) -> typ.Iterable[RewrittenFileData]:
r'''Iterate over files with version string replaced.
>>> file_patterns = {"src/pycalver/__init__.py": ['__version__ = "{pycalver}"']}
>>> rewritten_datas = iter_rewritten(file_patterns, "v201809.0123")
>>> rfd = list(rewritten_datas)[0]
>>> assert rfd.new_lines == [
... '# This file is part of the pycalver project',
... '# https://gitlab.com/mbarkhau/pycalver',
... '#',
2019-02-15 00:17:46 +01:00
... '# Copyright (c) 2019 Manuel Barkhau (mbarkhau@gmail.com) - MIT License',
... '# SPDX-License-Identifier: MIT',
2019-02-15 00:17:46 +01:00
... '"""PyCalVer: CalVer for Python Packages."""',
... '',
... '__version__ = "v201809.0123"',
... '',
... ]
>>>
'''
2019-01-07 17:30:02 +01:00
fh: typ.IO[str]
for file_path, pattern_strs in _iter_file_paths(file_patterns):
2019-02-21 15:41:06 +01:00
with file_path.open(mode="rt", encoding="utf-8") as fh:
content = fh.read()
rfd = rfd_from_content(pattern_strs, new_version, content)
2019-02-21 15:41:06 +01:00
yield rfd._replace(path=str(file_path))
def diff_lines(rfd: RewrittenFileData) -> typ.List[str]:
r"""Generate unified diff.
>>> rfd = RewrittenFileData(
... path = "<path>",
... line_sep = "\n",
... old_lines = ["foo"],
... new_lines = ["bar"],
... )
>>> diff_lines(rfd)
['--- <path>', '+++ <path>', '@@ -1 +1 @@', '-foo', '+bar']
"""
lines = difflib.unified_diff(
a=rfd.old_lines, b=rfd.new_lines, lineterm="", fromfile=rfd.path, tofile=rfd.path
)
return list(lines)
2018-11-15 22:16:16 +01:00
2019-02-21 15:41:06 +01:00
def diff(new_version: str, file_patterns: config.PatternsByGlob) -> str:
2018-11-15 22:16:16 +01:00
r"""Generate diffs of rewritten files.
>>> file_patterns = {"src/pycalver/__init__.py": ['__version__ = "{pycalver}"']}
>>> diff_str = diff("v201809.0123", file_patterns)
>>> lines = diff_str.split("\n")
>>> lines[:2]
2018-11-15 22:16:16 +01:00
['--- src/pycalver/__init__.py', '+++ src/pycalver/__init__.py']
>>> assert lines[6].startswith('-__version__ = "v2')
>>> assert not lines[6].startswith('-__version__ = "v201809.0123"')
>>> lines[7]
2018-11-15 22:16:16 +01:00
'+__version__ = "v201809.0123"'
"""
full_diff = ""
2019-02-21 15:41:06 +01:00
fh: typ.IO[str]
2019-01-07 17:30:02 +01:00
for file_path, pattern_strs in sorted(_iter_file_paths(file_patterns)):
2019-02-21 15:41:06 +01:00
with file_path.open(mode="rt", encoding="utf-8") as fh:
content = fh.read()
2019-02-22 11:04:53 +01:00
try:
2019-02-22 22:47:44 +01:00
rfd = rfd_from_content(pattern_strs, new_version, content)
2019-02-22 11:04:53 +01:00
except ValueError:
errmsg = f"No patterns matched for '{file_path}'"
raise ValueError(errmsg)
rfd = rfd._replace(path=str(file_path))
2019-02-21 16:30:27 +01:00
lines = diff_lines(rfd)
if len(lines) == 0:
errmsg = f"No patterns matched for '{file_path}'"
raise ValueError(errmsg)
full_diff += "\n".join(lines) + "\n"
2018-11-15 22:16:16 +01:00
full_diff = full_diff.rstrip("\n")
return full_diff
2018-11-15 22:16:16 +01:00
2019-02-21 15:41:06 +01:00
def rewrite(new_version: str, file_patterns: config.PatternsByGlob) -> None:
2018-11-15 22:16:16 +01:00
"""Rewrite project files, updating each with the new version."""
2019-01-07 17:30:02 +01:00
fh: typ.IO[str]
2018-11-15 22:16:16 +01:00
for file_data in iter_rewritten(file_patterns, new_version):
2018-11-15 22:16:16 +01:00
new_content = file_data.line_sep.join(file_data.new_lines)
with io.open(file_data.path, mode="wt", encoding="utf-8") as fh:
2018-09-03 22:23:51 +02:00
fh.write(new_content)