-
Notifications
You must be signed in to change notification settings - Fork 355
/
Copy pathconftest.py
182 lines (139 loc) · 5.32 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
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
import ast
import json
import parso
import pytest
from collections import OrderedDict
from types import GeneratorType as generator
from itertools import chain
from pathlib import Path
from objectpath import Tree
from mongoquery import Query
from tests.nodes import convert_node, flatten
from tests.template import Template
class Parser:
def __init__(self, file_name, nodes):
ssg = Path.cwd() / "ssg"
ext = ssg / "extensions"
self.data = {
"success": True,
"full_path": "",
"message": "",
"start_pos": 0,
"nodes": nodes,
}
if file_name is not None:
path = lambda root, fn: root / "{}.py".format(fn)
if file_name == "menu" or file_name == "stats":
full_path = path(ext, file_name)
elif file_name == "ssg":
full_path = path(Path.cwd(), file_name)
else:
full_path = path(ssg, file_name)
grammar = parso.load_grammar()
module = grammar.parse(path=full_path)
self.data["success"] = len(grammar.iter_errors(module)) == 0
if self.data["success"]:
self.data["message"] = "Syntax: valid"
if file_name is not None:
self.data["nodes"] = convert_node(ast.parse(full_path.read_text()))
self.data["code"] = full_path.read_text()
else:
self.data["message"] = grammar.iter_errors(module)[0].message
self.data["start_pos"] = grammar.iter_errors(module)[0].start_pos[0]
@property
def nodes(self):
return self.data["nodes"]
n = nodes
@property
def success(self):
return self.data["success"]
@property
def code(self):
return self.data["code"]
def query(self, pattern):
nodes = Template(pattern).process(self.code)
if isinstance(nodes, list) and len(nodes) == 1:
nodes = nodes[0]
return Parser(None, nodes)
def query_raw(self, pattern):
nodes = Template(pattern).process(self.code, True)
if isinstance(nodes, list) and len(nodes) == 1:
nodes = [flatten(convert_node(node)) for node in nodes[0].body]
return Parser(None, nodes)
def last_line(self):
return flatten(self.nodes["body"][-1])
@property
def message(self):
return "{} on or around line {} in `{}`.".format(
self.data["message"], self.data["start_pos"], self.data["full_path"]
)
def match(self, template):
return Parser(None, list(filter(Query(template).match, self.nodes)))
def execute(self, expr):
result = Tree(self.nodes).execute(expr)
if isinstance(result, (generator, chain, map)):
process = list(result)
return (
Parser(None, process[0]) if len(process) == 1 else Parser(None, process)
)
else:
return Parser(None, result)
ex = execute
def exists(self):
return bool(self.nodes)
def calls(self):
nodes = self.execute("$.body[@.type is 'Expr' and @.value.type is 'Call']").n
node_list = [nodes] if isinstance(nodes, dict) else nodes
return Parser(None, [flatten(node) for node in node_list])
def assign_(self):
return Parser(None, [flatten(self.execute("$.body[@.type is 'Assign']").n)])
def assigns(self):
return Parser(
None,
[flatten(node) for node in self.execute("$.body[@.type is 'Assign']").n],
)
def globals(self, name):
return name in self.execute("$.body[@.type is 'Global'].names").n
def defines(self, name):
return self.execute(
"$.body[@.type is 'FunctionDef' and @.name is '{}'].(name, args, body, decorator_list)".format(
name
)
)
def class_(self, name):
return self.execute(
"$.body[@.type is 'ClassDef' and @.name is '{}'].(name, args, body)".format(
name
)
)
def decorators(self):
return Parser(None, [flatten(self.execute("$.decorator_list").n)])
def returns(self, name):
return name == self.execute("$.body[@.type is 'Return'].value.id").n
def returns_call(self):
return Parser(None, [flatten(self.execute("$.body[@.type is 'Return']").n)])
def method(self, name):
return self.execute(
"$..body[@.type is 'FunctionDef' and @.name is '{}']".format(name)
)
def has_arg(self, name, pos=0):
nodes = self.execute("$.args.args.arg").n
return nodes[pos] if isinstance(nodes, list) else nodes
def imports(self, name):
return name in self.execute("$.body[@.type is 'Import'].names..name").n
def for_(self):
for_body = self.execute("$.body[@.type is 'For'].body").n
iterators = self.execute("$.body[@.type is 'For'].(target, iter)").n
return Parser(None, [flatten(for_body), flatten(iterators)])
def from_imports(self, mod, alias):
nodes = self.execute(
"$.body[@.type is 'ImportFrom' and @.module is '{}'].names..name".format(
mod
)
).n
return alias in (nodes if isinstance(nodes, list) else [nodes])
@pytest.fixture
def parse():
def _parse(file_name):
return Parser(file_name, {})
return _parse