bumpver/src/pycalver/config.py

211 lines
5.5 KiB
Python
Raw Normal View History

2018-09-02 21:48:12 +02:00
# This file is part of the pycalver project
# https://github.com/mbarkhau/pycalver
#
2018-11-06 21:45:33 +01:00
# Copyright (c) 2018 Manuel Barkhau (@mbarkhau) - MIT License
2018-09-02 21:48:12 +02:00
# SPDX-License-Identifier: MIT
2018-11-11 15:08:46 +01:00
"""Parsing code for setup.cfg or pycalver.cfg"""
2018-09-02 21:48:12 +02:00
import io
import os
import configparser
import pkg_resources
import typing as typ
import datetime as dt
import logging
from .parse import PYCALVER_RE
log = logging.getLogger("pycalver.config")
2018-09-03 22:23:51 +02:00
class Config(typ.NamedTuple):
2018-11-04 21:11:42 +01:00
current_version: str
2018-09-03 22:23:51 +02:00
2018-11-04 21:11:42 +01:00
tag : bool
commit: bool
2018-09-03 22:23:51 +02:00
2018-11-04 21:11:42 +01:00
file_patterns: typ.Dict[str, typ.List[str]]
2018-09-02 21:48:12 +02:00
2018-11-11 15:08:46 +01:00
def _debug_str(self) -> str:
cfg_str_parts = [
f"Config Parsed: Config(",
f"current_version='{self.current_version}'",
f"tag={self.tag}",
f"commit={self.commit}",
f"file_patterns={{",
]
2018-09-03 22:23:51 +02:00
2018-11-11 15:08:46 +01:00
for filename, patterns in self.file_patterns.items():
for pattern in patterns:
cfg_str_parts.append(f"\n '{filename}': '{pattern}'")
2018-09-03 22:23:51 +02:00
2018-11-11 15:08:46 +01:00
cfg_str_parts += ["\n})"]
return ", ".join(cfg_str_parts)
2018-09-03 22:23:51 +02:00
2018-11-11 15:08:46 +01:00
@property
def pep440_version(self) -> str:
return str(pkg_resources.parse_version(self.current_version))
2018-09-02 21:48:12 +02:00
2018-11-11 15:08:46 +01:00
MaybeConfig = typ.Optional[Config]
2018-09-02 21:48:12 +02:00
2018-11-11 15:08:46 +01:00
FilePatterns = typ.Dict[str, typ.List[str]]
2018-09-02 21:48:12 +02:00
2018-11-11 15:08:46 +01:00
def _parse_file_patterns(
cfg_parser: configparser.RawConfigParser, config_filename: str
) -> typ.Optional[FilePatterns]:
2018-09-02 21:48:12 +02:00
2018-11-11 15:08:46 +01:00
file_patterns: FilePatterns = {}
2018-09-02 21:48:12 +02:00
2018-09-03 22:23:51 +02:00
section_name: str
2018-09-02 21:48:12 +02:00
for section_name in cfg_parser.sections():
if not section_name.startswith("pycalver:file:"):
continue
filepath = section_name.split(":", 2)[-1]
if not os.path.exists(filepath):
2018-11-11 15:08:46 +01:00
log.error(f"No such file: {filepath} from {section_name} in {config_filename}")
2018-09-02 21:48:12 +02:00
return None
2018-09-03 22:23:51 +02:00
section: typ.Dict[str, str] = dict(cfg_parser.items(section_name))
patterns = section.get("patterns")
2018-09-02 21:48:12 +02:00
2018-09-03 22:23:51 +02:00
if patterns is None:
file_patterns[filepath] = ["{version}", "{pep440_version}"]
else:
file_patterns[filepath] = [
2018-11-04 21:11:42 +01:00
line.strip() for line in patterns.splitlines() if line.strip()
2018-09-02 21:48:12 +02:00
]
2018-09-03 22:23:51 +02:00
if not file_patterns:
2018-11-11 15:08:46 +01:00
file_patterns[f"{config_filename}"] = ["{version}", "{pep440_version}"]
return file_patterns
def _parse_buffer(cfg_buffer: io.StringIO, config_filename: str = "<pycalver.cfg>") -> MaybeConfig:
cfg_parser = configparser.RawConfigParser()
if hasattr(cfg_parser, 'read_file'):
cfg_parser.read_file(cfg_buffer)
else:
cfg_parser.readfp(cfg_buffer)
if not cfg_parser.has_section("pycalver"):
log.error(f"{config_filename} does not contain a [pycalver] section.")
return None
base_cfg = dict(cfg_parser.items("pycalver"))
if "current_version" not in base_cfg:
log.error(f"{config_filename} does not have 'pycalver.current_version'")
return None
current_version = base_cfg['current_version']
if PYCALVER_RE.match(current_version) is None:
log.error(f"{config_filename} 'pycalver.current_version is invalid")
log.error(f"current_version = {current_version}")
return None
tag = base_cfg.get("tag" , "").lower() in ("yes", "true", "1", "on")
commit = base_cfg.get("commit", "").lower() in ("yes", "true", "1", "on")
file_patterns = _parse_file_patterns(cfg_parser, config_filename)
if file_patterns is None:
return None
2018-11-11 15:15:14 +01:00
if tag and not commit:
log.error(f"Invalid configuration in {config_filename}")
log.error(" pycalver.commit = True required if pycalver.tag = True")
return None
2018-11-11 15:08:46 +01:00
cfg = Config(current_version, tag, commit, file_patterns)
log.debug(cfg._debug_str())
2018-09-02 21:48:12 +02:00
return cfg
2018-11-11 15:08:46 +01:00
def parse(config_filename: str = None) -> MaybeConfig:
if config_filename is None:
if os.path.exists("pycalver.cfg"):
config_filename = "pycalver.cfg"
elif os.path.exists("setup.cfg"):
config_filename = "setup.cfg"
else:
log.error("File not found: pycalver.cfg or setup.cfg")
return None
if not os.path.exists(config_filename):
log.error(f"File not found: {config_filename}")
2018-09-03 22:23:51 +02:00
return None
cfg_buffer = io.StringIO()
2018-11-11 15:08:46 +01:00
with io.open(config_filename, mode="rt", encoding="utf-8") as fh:
2018-09-03 22:23:51 +02:00
cfg_buffer.write(fh.read())
cfg_buffer.seek(0)
2018-11-11 15:08:46 +01:00
return _parse_buffer(cfg_buffer, config_filename)
DEFAULT_CONFIG_BASE_STR = """
[pycalver]
current_version = {initial_version}
commit = True
tag = True
[pycalver:file:setup.cfg]
patterns =
current_version = {{version}}
"""
DEFAULT_CONFIG_SETUP_PY_STR = """
[pycalver:file:setup.py]
patterns =
"{version}"
"{pep440_version}"
"""
DEFAULT_CONFIG_README_RST_STR = """
[pycalver:file:README.rst]
patterns =
{version}
{pep440_version}
"""
DEFAULT_CONFIG_README_MD_STR = """
[pycalver:file:README.md]
patterns =
{version}
{pep440_version}
"""
2018-09-03 22:23:51 +02:00
2018-09-02 21:48:12 +02:00
def default_config_lines() -> typ.List[str]:
initial_version = dt.datetime.now().strftime("v%Y%m.0001-dev")
2018-11-11 15:08:46 +01:00
cfg_str = DEFAULT_CONFIG_BASE_STR.format(initial_version=initial_version)
cfg_lines = cfg_str.splitlines()
2018-09-02 21:48:12 +02:00
if os.path.exists("setup.py"):
2018-11-11 15:08:46 +01:00
cfg_lines.extend(DEFAULT_CONFIG_SETUP_PY_STR.splitlines())
2018-09-02 21:48:12 +02:00
if os.path.exists("README.rst"):
2018-11-11 15:08:46 +01:00
cfg_lines.extend(DEFAULT_CONFIG_README_RST_STR.splitlines())
2018-09-02 21:48:12 +02:00
if os.path.exists("README.md"):
2018-11-11 15:08:46 +01:00
cfg_lines.extend(DEFAULT_CONFIG_README_MD_STR.splitlines())
cfg_lines += [""]
2018-09-02 21:48:12 +02:00
return cfg_lines