forked from snychka/python-static-site-generator
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathconftest.py
133 lines (108 loc) · 4.34 KB
/
conftest.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
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
import re
import ast
import parso
import pytest
from pathlib import Path
from redbaron import RedBaron
from redbaron.utils import indent
class SourceCode:
def __init__(self, exists, code):
self.exists = exists
self.code = code
class Parser:
def __init__(self, filename):
self.code = ""
self.message = ""
error_message = ""
error_start_pos = ""
if filename == "ssg":
file_path = Path.cwd() / "ssg.py"
else:
file_path = Path.cwd() / "ssg" / "{}.py".format(filename)
grammar = parso.load_grammar()
module = grammar.parse(path=file_path.resolve())
self.success = len(grammar.iter_errors(module)) == 0
if self.success:
with open(file_path.resolve(), "r") as source_code:
self.code = RedBaron(source_code.read())
else:
error_message = grammar.iter_errors(module)[0].message
error_start_pos = grammar.iter_errors(module)[0].start_pos[0]
self.message = "{} on or around line {} in `{}`.".format(
error_message, error_start_pos, file_path.name
)
def get_by_name(self, type, name, code=None):
if code is None:
item = self.code.find_all(type, lambda node: node.name == name)
else:
item = code.find_all(type, lambda node: node.name == name)
return SourceCode(True, item[0]) if len(item) > 0 else SourceCode(False, [])
def get_call(self, value, code):
call = code.find("call", lambda node: node.previous.value == value)
return SourceCode(True, call) if call is not None and len(call) > 0 else SourceCode(False, [])
def get_args(self, code):
return list(
code.find_all("call_argument").map(
lambda node: str(node.target) + ":" + str(node.value).replace("'", '"')
)
)
def get_by_value(self, type, value, code=None):
if code is None:
item = self.code.find_all(type, lambda node: str(node.target) == value)
else:
item = code.find_all(type, lambda node: str(node.target) == value)
return SourceCode(True, item[0]) if len(item) > 0 else SourceCode(False, [])
def get_imports(self):
imports = []
self.code.find_all(
"import",
lambda node: node.find_all(
"dotted_as_name", lambda node: imports.append(str(node))
),
)
return imports
def get_from_import(self, value):
imports = self.code.find_all(
"from_import",
lambda node: "".join(list(node.value.node_list.map(lambda node: str(node))))
== value,
).find_all("name_as_name")
return list(imports.map(lambda node: node.value))
def flatten(self, dictionary):
def _flatten(node):
trimmed = re.sub(r"\"|'", "", node.key.value)
flattened = []
if node.value.type is "list":
for item in node.value.node_list:
if item.type is not "comma":
flattened.append("{}:{}".format(trimmed, str(item)))
else:
flattened.append("{}:{}".format(trimmed, node.value.value))
return flattened
items = list(dictionary.find_all("dictitem").map(lambda node: _flatten(node)))
return [item for sublist in items for item in sublist]
def get_conditional(self, values, type, nested=False):
def flat(node):
if node.type == "comparison":
return "{}:{}:{}".format(
str(node.first).replace("'", '"'),
str(node.value).replace(" ", ":"),
str(node.second).replace("'", '"'),
)
elif node.type == "unitary_operator":
return "{}:{}".format(
str(node.value), str(node.target).replace("'", '"')
)
nodes = self.code.value if nested else self.code
for value in values:
final_node = nodes.find_all(type).find(
["comparison", "unitary_operator"], lambda node: flat(node) == value
)
if final_node is not None:
return final_node
return None
@pytest.fixture
def parse():
def _parse(filename):
return Parser(filename)
return _parse