-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathpov.py
190 lines (164 loc) · 6.9 KB
/
pov.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
183
184
185
186
187
188
189
190
import hyperdiv as hd
from dap_client import *
# Variables to filter out from the display
## These tend to be boilerplate or internal variables that are not interesting to display
VARIABLE_NAMES_TO_FILTER = [
"__builtins__",
"__doc__",
"__loader__",
"__name__",
"__package__",
"__spec__",
"special variables",
"function variables",
"module variables",
"class variables",
"debugpy",
]
VARIABLE_TYPES_TO_FILTER = [
"builtin_function_or_method",
"method-wrapper",
]
def render_tree(variables, title):
with hd.box(border="0px solid blue", padding=0.8):
with hd.tree(indent_guide_width="1px"):
render_variable_tree(variables)
def render_variable_tree(variables):
"""
Renders a list of variables (each may have 'children') in a nested tree format.
"""
print(f"Rendering variable tree with {len(variables)} variables")
for v in variables:
# print(f"Rendering variable tree for: {v}")
name = v["name"]
value = v.get("value", "unknown")
var_type = v.get("type", "unknown")
evaluate_name = v.get("evaluateName", "")
children = v.get("children", [])
# Filter out certain variable names
if name in VARIABLE_NAMES_TO_FILTER:
continue
# Filter out certain variable types
if var_type in VARIABLE_TYPES_TO_FILTER:
continue
# Truncate long values
if len(value) > 100:
value = f"{value[:100]} ..."
bg_color = None
# TODO: Find a better way to color code variables / style them based on type
# col_varient = "-800"
# if var_type == "str":
# bg_color = f"sky{col_varient}"
# if var_type == "int":
# bg_color = f"violet{col_varient}"
# if var_type == "float":
# bg_color = f"purple{col_varient}"
# if var_type == "list":
# bg_color = f"pink{col_varient}"
# if var_type == "dict":
# bg_color = f"rose{col_varient}"
# Render one node for the variable
with hd.scope(v):
# print(f"DEBUG: Rendering variable: {name} with value: {value}")
with hd.tree_item(background_color=bg_color):
# hd.markdown(f"**{name}**")
# hd.markdown(f"{name}")
# hd.markdown(f"`{value}`")
# hd.markdown(f"`{var_type}`")
# hd.markdown(f"`{evaluate_name}`")
# hd.markdown(f"`{v.get('variablesReference', 0)}`")
hd.markdown(f"**{name}**: `{value}` (**Type**: `{var_type}`) ")
if name != evaluate_name and evaluate_name:
hd.markdown(" ")
# hd.markdown(f" **Evaluate Name**: `{evaluate_name}`")
hd.markdown(f" | `{evaluate_name}`")
# hd.markdown(
# f" **Variables Reference**: `{v.get('variablesReference', 0)}`"
# )
# If this variable has child variables, recurse
if children:
render_variable_tree(children)
def pov():
with hd.hbox(gap=1, justify="space-around", border="0px solid red", padding=0.8):
with hd.box(
font_size=1,
gap=0,
justify="space-around",
border="0px solid yellow",
align="center",
):
hd.markdown("## Python Object Viewer")
hd.divider(spacing=0.4, thickness=0)
dap_task = hd.task()
dap_task.run(
lambda: dap_client(depth_limit=2)
) # TODO: make this a dropdown/similar
if dap_task.running:
hd.markdown("### Waiting for variables...")
hd.markdown(" ") # hack to add some space
with hd.hbox(font_size=4, justify="space-around"):
hd.spinner(speed="5s", track_width=0.5)
hd.markdown(" ") # hack to add some space
hd.markdown(
"This can take a couple of minutes, depending on the size of your program."
)
if dap_task.error:
hd.markdown(
"`Error collecting variables - make sure the debugger is running on localhost:5678`"
)
return
if dap_task.done:
print("dap_task is done")
results = dap_task.result # This is the dict returned by dap_client()
# If no frames, nothing to display
frames = results.get("frames", [])
if not frames:
print("No frames returned from dap_client. Re-running.")
dap_task.clear()
dap_task.run(dap_client)
return
# Right now we only get one frame, so we'll just use that
first_frame = frames[0]
dap_scopes = first_frame.get("scopes", {})
print(f"Scopes available: {list(dap_scopes.keys())}")
# Count variables in each scope
for scope_list in dap_scopes.keys():
print(
f"Scope: {scope_list} has {len(dap_scopes[scope_list])} variables"
)
# Create a tab group for all scope names
tabs_dict = {}
with hd.tab_group():
for scope_name in dap_scopes.keys():
with hd.scope(scope_name):
# Create a tab with the title = scope_name.title()
tab_obj = hd.tab(scope_name.title())
# Store the tab object in a dict so we can check if it's active later
tabs_dict[scope_name] = tab_obj
# Now show the variables for whichever tab is active
with hd.hbox(gap=1):
# We'll iterate again to find the active tab
for scope_name, tab_obj in tabs_dict.items():
with hd.scope(tab_obj):
if tab_obj.active:
scope_vars = dap_scopes[scope_name]
render_tree(
scope_vars, title=f"{scope_name.title()} Scope"
)
# Dropdown for selecting the depth limit
# with hd.box(gap=1):
# with hd.select(
# placeholder="Depth Limit",
# ) as select:
# hd.option("1")
# hd.option("2")
# hd.option("3")
# hd.option("4")
# hd.option("5")
# hd.text("Selected:", select.value)
hd.run(
pov,
index_page=hd.index_page(
title="Python Object Viewer",
),
)