/
factor.py
103 lines (82 loc) · 3.24 KB
/
factor.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
"""
Expand tox factor expressions to tox environment list.
"""
from __future__ import annotations
import re
from itertools import chain, groupby, product
from typing import Iterator
def filter_for_env(value: str, name: str | None) -> str:
current = (
set(chain.from_iterable([(i for i, _ in a) for a in find_factor_groups(name)])) if name is not None else set()
)
overall = []
for factors, content in expand_factors(value):
if factors is None:
if content:
overall.append(content)
else:
for group in factors:
if all((a_name in current) ^ negate for a_name, negate in group):
overall.append(content)
break # if any match we use it, and then bail
result = "\n".join(overall)
return result
def find_envs(value: str) -> Iterator[str]:
seen = set()
for factors, _ in expand_factors(value):
if factors is not None:
for group in factors:
env = explode_factor(group)
if env not in seen:
yield env
seen.add(env)
def extend_factors(value: str) -> Iterator[str]:
for group in find_factor_groups(value):
yield explode_factor(group)
def explode_factor(group: list[tuple[str, bool]]) -> str:
return "-".join([name for name, _ in group])
def expand_factors(value: str) -> Iterator[tuple[Iterator[list[tuple[str, bool]]] | None, str]]:
for line in value.split("\n"):
marker_at = line.find(":")
if marker_at == -1:
factor_expr, content = None, line
else:
factor_expr, content = line[:marker_at].strip(), line[marker_at + 1 :].lstrip()
if factor_expr:
try:
factors = list(find_factor_groups(factor_expr))
except ValueError:
yield None, line
else:
yield iter(factors), content
else:
yield None, content
def find_factor_groups(value: str) -> Iterator[list[tuple[str, bool]]]:
"""transform '{py,!pi}-{a,b},c' to [{'py', 'a'}, {'py', 'b'}, {'pi', 'a'}, {'pi', 'b'}, {'c'}]"""
for env in expand_env_with_negation(value):
result = [name_with_negate(f) for f in env.split("-")]
yield result
def expand_env_with_negation(value: str) -> Iterator[str]:
"""transform '{py,!pi}-{a,b},c' to ['py-a', 'py-b', '!pi-a', '!pi-b', 'c']"""
for key, group in groupby(re.split(r"((?:{[^}]+})+)|,", value), key=bool):
if key:
group_str = "".join(group).strip()
elements = re.split(r"{([^}]+)}", group_str)
parts = [re.sub(r"\s+", "", elem).split(",") for elem in elements]
for variant in product(*parts):
variant_str = "".join(variant)
if not re.fullmatch(r"[\w!-]+", variant_str):
raise ValueError(variant_str)
yield variant_str
def name_with_negate(factor: str) -> tuple[str, bool]:
negated = is_negated(factor)
result = factor[1:] if negated else factor
return result, negated
def is_negated(factor: str) -> bool:
return factor.startswith("!")
__all__ = (
"filter_for_env",
"find_envs",
"expand_factors",
"extend_factors",
)