forked from aptos-labs/aptos-core
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathdeterminator.py
111 lines (83 loc) · 2.66 KB
/
determinator.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
104
105
106
107
108
109
110
111
from __future__ import annotations
import re
from dataclasses import dataclass
from typing import Generic, Optional, Sequence, Tuple, TypeVar, TypedDict
import click
@dataclass
class Verdict:
verdict: bool
reason: str
subverdicts: Sequence[Verdict]
def format(self, indent=0) -> str:
return "\n".join([
(indent * " ") + ("PASS" if self.verdict else "FAIL") + f"ED because {self.reason}",
] + [
verdict.format(indent+1) for verdict in self.subverdicts
])
TEvaluationContext = TypeVar("TEvaluationContext")
class Predicate(Generic[TEvaluationContext]):
def evaluate(self, context: TEvaluationContext) -> Verdict:
raise NotImplementedError()
@dataclass
class OrPredicate(Predicate[TEvaluationContext]):
left: Predicate[TEvaluationContext]
right: Predicate[TEvaluationContext]
def evaluate(self, context: TEvaluationContext) -> Verdict:
leftval = self.left.evaluate(context)
rightval = self.right.evaluate(context)
return Verdict(
leftval.verdict or rightval.verdict,
f"({leftval.reason}) or ({rightval.reason})",
[leftval, rightval],
)
class ChangedFilesContext(TypedDict):
changed_files: Sequence[str]
@dataclass
class ChangedFilesPredicate(Predicate[ChangedFilesContext]):
patterns: Sequence[str]
def evaluate(self, context: ChangedFilesContext) -> Verdict:
matches = []
for path in context["changed_files"]:
for pattern in self.patterns:
if re.match(pattern, path):
matches.append(path)
return Verdict(
bool(matches),
f"Matched files: {matches}",
[],
)
@click.group()
def main() -> None:
pass
@dataclass
class GithubOutput:
key: str
value: str
def format(self) -> str:
return f"::set-output name={self.key}::{self.value}"
@main.command()
@click.option("--pattern", "patterns", multiple=True)
@click.option("--github-output-key")
@click.argument("changed_files", nargs=-1)
def changed_files(
patterns: Tuple[str],
github_output_key: Optional[str],
changed_files: Tuple[str],
) -> None:
context: ChangedFilesContext = {
"changed_files": changed_files,
}
predicate = ChangedFilesPredicate(patterns)
verdict = predicate.evaluate(context)
print(verdict.format())
if github_output_key:
output = GithubOutput(
github_output_key,
"true" if verdict.verdict else "false"
)
print(output.format())
else:
if not verdict.verdict:
raise SystemExit(1)
if __name__ == "__main__":
main()