-
Notifications
You must be signed in to change notification settings - Fork 6
/
Copy pathpico_lint.py
156 lines (131 loc) · 8.08 KB
/
pico_lint.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
from utils import *
from pico_tokenize import CommentHint, is_identifier
from pico_parse import VarKind, NodeType
from pico_parse import is_assign_target, is_function_target, is_any_assign_target, is_root_global_or_builtin_local
k_lint_used_prefix = "used::"
def lint_code(ctxt, root, lint_opts):
errors = []
builtin_globals = ctxt.builtins
lint_undefined = lint_opts.get("undefined", True)
lint_unused = lint_opts.get("unused", True)
lint_unused_global = lint_opts.get("unused-global", True)
lint_duplicate = lint_opts.get("duplicate", True)
lint_duplicate_global = lint_opts.get("duplicate-global", True)
custom_globals = set(lint_opts.get("globals", ()))
def add_error(msg, node):
errors.append(Error(msg, node))
# find global assignment, and check for uses
used_globals = set()
used_locals = set()
used_labels = set()
assigned_globals = set()
assigned_locals = set()
# (builtin locals will end up both as used globals and used locals)
def preprocess_tokens(token):
if token.children:
for comment in token.children:
if comment.hint == CommentHint.lint:
for value in comment.hintdata:
if "::" not in value:
custom_globals.add(value)
elif value.startswith(k_lint_used_prefix):
used_globals.add(value[len(k_lint_used_prefix):])
def preprocess_vars(node):
if node.type == NodeType.var:
if is_root_global_or_builtin_local(node):
if node.name not in custom_globals:
assign = False
if is_assign_target(node):
func = node.find_parent(NodeType.function)
assign = True
elif is_function_target(node):
func = node.parent.find_parent(NodeType.function)
assign = True
if assign and (func == None or func.kind == "_init" or (func.kind is None and func.name == "_init" and func.find_parent(NodeType.function) == None)):
custom_globals.add(node.name)
if is_any_assign_target(node):
assigned_globals.add(node.name)
else:
used_globals.add(node.name)
if node.kind == VarKind.local and not node.new:
if is_any_assign_target(node):
assigned_locals.add(node.var)
else:
used_locals.add(node.var)
if node.kind == VarKind.label and not node.new:
used_labels.add(node.var)
elif node.type == NodeType.sublang:
for glob in node.sublang.get_defined_globals():
if glob not in custom_globals and is_identifier(glob, ctxt.lang):
custom_globals.add(glob)
assigned_globals.add(glob)
for glob in node.sublang.get_used_globals():
used_globals.add(glob)
root.traverse_nodes(preprocess_vars, tokens=preprocess_tokens, extra=True)
# check for issues
def lint_pre(node):
if node.type == NodeType.var:
if node.kind == VarKind.local and node.new:
if lint_duplicate and node.name not in ('_', '_ENV'):
prev_var = node.scope.parent.find(node.name)
if prev_var is None or prev_var.builtin:
if lint_duplicate_global:
if node.name in custom_globals:
add_error(f"Local '{node.name}' has the same name as a global", node)
elif node.name in used_globals or node.name in assigned_globals:
add_error(f"Local '{node.name}' has the same name as a built-in global used elsewhere in the cart", node)
elif prev_var.scope.funcdepth < node.var.scope.funcdepth:
if prev_var.scope.funcdepth < 0:
pass # local builtin
elif prev_var.scope.funcdepth == 0:
add_error(f"Local '{node.name}' has the same name as a local declared at the top level", node)
else:
add_error(f"Local '{node.name}' has the same name as a local declared in a parent function", node)
elif prev_var.scope.depth < node.var.scope.depth:
add_error(f"Local '{node.name}' has the same name as a local declared in a parent scope", node)
else:
add_error(f"Local '{node.name}' has the same name as a local declared in the same scope", node)
if lint_unused and node.var not in used_locals and not node.name.startswith("_"):
if node.var in assigned_locals:
add_error(f"Local '{node.name}' is only ever assigned to, never used", node)
elif not (node.parent.type == NodeType.function and node in node.parent.params and
(node != node.parent.params[-1] or node not in node.parent.children)): # don't warn for non-last or implicit params
add_error(f"Local '{node.name}' isn't used", node)
elif node.kind == VarKind.label and node.new:
if lint_duplicate and node.name != '_':
prev_var = node.scope.parent.find(node.name, crossfunc=True)
if prev_var != None:
if prev_var.scope.funcdepth < node.var.scope.funcdepth:
if prev_var.scope.funcdepth <= 0:
add_error(f"Label '{node.name}' has the same name as a label declared at the top level", node)
else:
add_error(f"Label '{node.name}' has the same name as a label declared in a parent function", node)
else:
add_error(f"Label '{node.name}' has the same name as a label declared in a parent scope", node)
if lint_unused and node.var not in used_labels and not node.name.startswith("_"):
add_error(f"Label '{node.name}' isn't used", node)
elif is_root_global_or_builtin_local(node):
if lint_undefined and node.name not in custom_globals:
if node.name in builtin_globals:
if is_assign_target(node):
add_error(f"Built-in global '{node.name}' assigned outside _init - did you mean to use 'local'?", node)
elif is_function_target(node):
add_error(f"Built-in global '{node.name}' assigned outside _init - did you mean to use 'local function'?", node)
else:
if is_assign_target(node):
add_error(f"Identifier '{node.name}' not found - did you mean to use 'local' to define it?", node)
elif is_function_target(node):
add_error(f"Identifier '{node.name}' not found - did you mean to use 'local function' to define it?", node)
else:
add_error(f"Identifier '{node.name}' not found", node)
if lint_unused and lint_unused_global and node.name not in used_globals and not node.name.startswith("_"):
if node.name in assigned_globals:
add_error(f"Global '{node.name}' is only ever assigned to, never used", node)
else:
add_error(f"Global '{node.name}' isn't used", node)
elif node.type == NodeType.sublang:
add_lang_error = lambda msg: add_error(f"{node.name}: {msg}", node)
node.sublang.lint(on_error=add_lang_error, builtins=builtin_globals, globals=custom_globals)
root.traverse_nodes(lint_pre, extra=True)
return errors
from pico_process import Error