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[list[list[tuple[str, bool]]] | None, str]]:
for line in value.split("\n"):
factors: list[list[tuple[str, bool]]] | None = None
marker_search = re.search(r":(\s|$)", line)
marker_at, content = marker_search.start() if marker_search else -1, line
if marker_at != -1:
try:
factors = list(find_factor_groups(line[:marker_at].strip()))
except ValueError:
pass # when cannot extract factors keep the entire line
else:
content = line[marker_at + 1 :].strip()
yield factors, 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
_FACTOR_RE = re.compile(r"!?[\w._][\w._-]*")
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 = [[i.strip() for i in elem.split(",")] for elem in elements]
for variant in product(*parts):
variant_str = "".join(variant)
if not all(_FACTOR_RE.fullmatch(i) for i in variant_str.split("-")):
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",
)
|