vision-agent 0.2.193__py3-none-any.whl → 0.2.196__py3-none-any.whl
Sign up to get free protection for your applications and to get access to all the features.
- vision_agent/.sim_tools/df.csv +640 -0
- vision_agent/.sim_tools/embs.npy +0 -0
- vision_agent/agent/__init__.py +2 -0
- vision_agent/agent/agent_utils.py +211 -3
- vision_agent/agent/vision_agent_coder.py +5 -113
- vision_agent/agent/vision_agent_coder_prompts_v2.py +119 -0
- vision_agent/agent/vision_agent_coder_v2.py +341 -0
- vision_agent/agent/vision_agent_planner.py +2 -2
- vision_agent/agent/vision_agent_planner_prompts.py +1 -1
- vision_agent/agent/vision_agent_planner_prompts_v2.py +748 -0
- vision_agent/agent/vision_agent_planner_v2.py +432 -0
- vision_agent/lmm/lmm.py +4 -0
- vision_agent/tools/__init__.py +2 -1
- vision_agent/tools/planner_tools.py +246 -0
- vision_agent/tools/tool_utils.py +65 -1
- vision_agent/tools/tools.py +76 -22
- vision_agent/utils/image_utils.py +12 -6
- vision_agent/utils/sim.py +65 -14
- {vision_agent-0.2.193.dist-info → vision_agent-0.2.196.dist-info}/METADATA +2 -1
- vision_agent-0.2.196.dist-info/RECORD +42 -0
- vision_agent-0.2.193.dist-info/RECORD +0 -35
- {vision_agent-0.2.193.dist-info → vision_agent-0.2.196.dist-info}/LICENSE +0 -0
- {vision_agent-0.2.193.dist-info → vision_agent-0.2.196.dist-info}/WHEEL +0 -0
@@ -0,0 +1,748 @@
|
|
1
|
+
PLAN = """
|
2
|
+
**Role**: You are an expert planning agent that can understand the user request and search for a plan to accomplish it.
|
3
|
+
|
4
|
+
**Task**: As a planning agent you are required to understand the user's request and search for a plan to accomplish it. Use Chain-of-Thought approach to break down the problem, create a plan, and then provide a response. Esnure your response is clear, concise, andhelpful. You can use an interactive Pyton (Jupyter Notebok) environment, executing code with <execute_python>, each execution is a new cell so old code and outputs are saved.
|
5
|
+
|
6
|
+
**Documentation**: this is the documentation for the functions you can use to accomplish the task:
|
7
|
+
{tool_desc}
|
8
|
+
|
9
|
+
**Example Planning**: Here are some examples of how you can search for a plan, in the examples the user output is denoted by USER, your output is denoted by AGENT and the observations after your code execution are denoted by OBSERVATION:
|
10
|
+
{examples}
|
11
|
+
|
12
|
+
**Current Planning**:
|
13
|
+
--- START PLANNING ---
|
14
|
+
{planning}
|
15
|
+
--- END PLANNING ---
|
16
|
+
|
17
|
+
**Instructions**:
|
18
|
+
1. Read over the user request and context provided and output <thinking> tags to indicate your thought process. You can <count> number of turns to complete the user's request.
|
19
|
+
2. You can execute python code in the ipython notebook using <execute_python> tags. Only output one <execute_python> tag at a time.
|
20
|
+
3. Only output <finalize_plan> when you are done planning and want to end the planning process. DO NOT output <finalize_plan> with <execute_python> tags, only after OBSERVATION's.
|
21
|
+
4. Only load/save files from {media_list} unless you specifically saved the previously.
|
22
|
+
5. Ensure you always call `suggestion` initially and `get_tool_for_task` to get the right tool for the subtask.
|
23
|
+
6. Calling `plt.imshow` or `save_image` will display the image to you, use this to visually check your results.
|
24
|
+
7. DO NOT hard code the answer into your code, it should be dynamic and work for any similar request.
|
25
|
+
8. DO NOT over index on claude35_vqa, if tool output is close to claude35_vqa's output you do not need to improve the tool.
|
26
|
+
9. You can only respond in the following format with a single <thinking>, <execute_python> or <finalize_plan> tag:
|
27
|
+
|
28
|
+
<thinking>Your thought process...</thinking>
|
29
|
+
<execute_python>Your code here</execute_python>
|
30
|
+
<finalize_plan>Your final recommended plan</finalize_plan>
|
31
|
+
"""
|
32
|
+
|
33
|
+
EXAMPLE_PLAN1 = """
|
34
|
+
--- EXAMPLE1 ---
|
35
|
+
USER: Count the number of pedestrians in the image.
|
36
|
+
<count>10</count>
|
37
|
+
|
38
|
+
AGENT: <thinking>I need to gather more information, I will ask for a description of the image and a solution to the problem to work towards as well as a suggestion on how best to solve it.</thinking>
|
39
|
+
<execute_python>
|
40
|
+
image = load_image('drone.jpg')
|
41
|
+
claude35_vqa('Can you describe this image? How many pedestrians do you count in the image?', [image])
|
42
|
+
suggestion('How can I count the number of pedestrians in the image?', [image])
|
43
|
+
</execute_python>
|
44
|
+
|
45
|
+
OBSERVATION: [claude35_vqa output]
|
46
|
+
From this aerial view of a busy urban street, it's difficult to clearly see or count individual pedestrians. The image shows a bird's eye view of a city intersection with multiple lanes of traffic, parked cars, sidewalks, and some green spaces. While there may be people in cars or on the sidewalks, they are not distinctly visible from this height and perspective. The focus of the image is more on the urban infrastructure, traffic patterns, and overall city layout rather than on individuals.
|
47
|
+
[end of claude35_vqa output]
|
48
|
+
|
49
|
+
[suggestions]
|
50
|
+
[suggestion 0]
|
51
|
+
The image is very large and the items you need to detect are small.
|
52
|
+
|
53
|
+
Step 1: You should start by splitting the image into sections and runing the detection algorithm on each section:
|
54
|
+
|
55
|
+
def subdivide_image(image):
|
56
|
+
height, width, _ = image.shape
|
57
|
+
mid_height = height // 2
|
58
|
+
mid_width = width // 2
|
59
|
+
overlap_height = int(mid_height * 0.1)
|
60
|
+
overlap_width = int(mid_width * 0.1)
|
61
|
+
top_left = image[:mid_height + overlap_height, :mid_width + overlap_width, :]
|
62
|
+
top_right = image[:mid_height + overlap_height, mid_width - overlap_width:, :]
|
63
|
+
bottom_left = image[mid_height - overlap_height:, :mid_width + overlap_width, :]
|
64
|
+
bottom_right = image[mid_height - overlap_height:, mid_width - overlap_width:, :]
|
65
|
+
return [top_left, top_right, bottom_left, bottom_right]
|
66
|
+
|
67
|
+
get_tool_for_task('<your prompt here>', subdivide_image(image))
|
68
|
+
|
69
|
+
Step 2: Once you have the detections from each subdivided image, you will need to merge them back together to remove overlapping predictions:
|
70
|
+
|
71
|
+
def translate_ofset(bbox, offset_x, offset_y):
|
72
|
+
return (bbox[0] + offset_x, bbox[1] + offset_y, bbox[2] + offset_x, bbox[3] + offset_y)
|
73
|
+
|
74
|
+
def bounding_boxes_overlap(bbox1, bbox2):
|
75
|
+
if bbox1[2] < bbox2[0] or bbox2[0] > bbox1[2]:
|
76
|
+
return False
|
77
|
+
if bbox1[3] < bbox2[1] or bbox2[3] > bbox1[3]:
|
78
|
+
return False
|
79
|
+
return True
|
80
|
+
|
81
|
+
def merge_bounding_boxes(bbox1, bbox2):
|
82
|
+
x_min = min(bbox1[0], bbox2[0])
|
83
|
+
y_min = min(bbox1[1], bbox2[1])
|
84
|
+
x_max = max(bbox1[2], bbox2[2])
|
85
|
+
y_max = max(bbox1[3], bbox2[3])
|
86
|
+
return (x_min, y_min, x_max, y_max)
|
87
|
+
|
88
|
+
def merge_bounding_box_list(bboxes):
|
89
|
+
merged_bboxes = []
|
90
|
+
while bboxes:
|
91
|
+
bbox = bboxes.pop()
|
92
|
+
overlap_found = False
|
93
|
+
for i, other_bbox in enumerate(merged_bboxes):
|
94
|
+
if bounding_boxes_overlap(bbox, other_bbox):
|
95
|
+
merged_bboxes[i] = merge_bounding_boxes(bbox, other_bbox)
|
96
|
+
overlap_found = True
|
97
|
+
break
|
98
|
+
if not overlap_found:
|
99
|
+
p
|
100
|
+
merged_bboxes.append(bbox)
|
101
|
+
return merged_bboxes
|
102
|
+
|
103
|
+
detection = merge_bounding_box_list(detection_from_subdivided_images)
|
104
|
+
[end of suggestion 0]
|
105
|
+
[end of suggestion]
|
106
|
+
<count>9</count>
|
107
|
+
|
108
|
+
AGENT: <thinking>Following the suggested plan, I will subdivide the image into smaller parts and find the right tool on the smaller parts. I will also re-check with claude35_vqa on the subdivided images.</thinking>
|
109
|
+
<execute_python>
|
110
|
+
def subdivide_image(image):
|
111
|
+
height, width, _ = image.shape
|
112
|
+
mid_height = height // 2
|
113
|
+
mid_width = width // 2
|
114
|
+
overlap_height = int(mid_height * 0.1)
|
115
|
+
overlap_width = int(mid_width * 0.1)
|
116
|
+
top_left = image[:mid_height + overlap_height, :mid_width + overlap_width, :]
|
117
|
+
top_right = image[:mid_height + overlap_height, mid_width - overlap_width:, :]
|
118
|
+
bottom_left = image[mid_height - overlap_height:, :mid_width + overlap_width, :]
|
119
|
+
bottom_right = image[mid_height - overlap_height:, mid_width - overlap_width:, :]
|
120
|
+
return [top_left, top_right, bottom_left, bottom_right]
|
121
|
+
|
122
|
+
images = subdivide_image(image)
|
123
|
+
claude35_vqa('How many pedestrians do you see in these images?', images)
|
124
|
+
get_tool_for_task('Count the number of pedestrians in the images', images)
|
125
|
+
</execute_python>
|
126
|
+
|
127
|
+
OBSERVATION:
|
128
|
+
[claude35_vqa_output]
|
129
|
+
In these aerial images, I can see approximately 5-6 pedestrians walking in various locations, including near the retail signage are and along the sidewalks.
|
130
|
+
[end of claude35_vqa_output]
|
131
|
+
|
132
|
+
[get_tool_for_task output]
|
133
|
+
After examining the image, I can see it's an aerial view of a busy urban intersection with multiple lanes of traffic. There are numerous cars visible, each likely containing at least one person (the driver). While it's difficult to count exact individuals, I can estimate based on the number of vehicles. The countgd_counting tool returned an empty list, which is incorrect given the visible evidence in the image. This suggests the tool may have failed to detect people inside vehicles or had difficulty with the aerial perspective. Despite this tool being specifically requested, its output is clearly inaccurate for this task.
|
134
|
+
|
135
|
+
Tool Documentation:
|
136
|
+
countgd_counting(prompt: str, image: numpy.ndarray, box_threshold: float = 0.23) -> List[Dict[str, Any]]:
|
137
|
+
'countgd_counting' is a tool that can precisely count multiple instances of an
|
138
|
+
object given a text prompt. It returns a list of bounding boxes with normalized
|
139
|
+
coordinates, label names and associated confidence scores.
|
140
|
+
|
141
|
+
Parameters:
|
142
|
+
prompt (str): The object that needs to be counted.
|
143
|
+
image (np.ndarray): The image that contains multiple instances of the object.
|
144
|
+
box_threshold (float, optional): The threshold for detection. Defaults
|
145
|
+
to 0.23.
|
146
|
+
|
147
|
+
Returns:
|
148
|
+
List[Dict[str, Any]]: A list of dictionaries containing the score, label, and
|
149
|
+
bounding box of the detected objects with normalized coordinates between 0
|
150
|
+
and 1 (xmin, ymin, xmax, ymax). xmin and ymin are the coordinates of the
|
151
|
+
top-left and xmax and ymax are the coordinates of the bottom-right of the
|
152
|
+
bounding box.
|
153
|
+
|
154
|
+
Example
|
155
|
+
-------
|
156
|
+
>>> countgd_counting("flower", image)
|
157
|
+
[
|
158
|
+
{'score': 0.49, 'label': 'flower', 'bbox': [0.1, 0.11, 0.35, 0.4]},
|
159
|
+
]
|
160
|
+
[end of get_tool_for_task output]
|
161
|
+
<count>8</count>
|
162
|
+
|
163
|
+
AGENT: <thinking>I need to now merge the boxes from all regions and count the unique pedestrians.</thinking>
|
164
|
+
<execute_python>
|
165
|
+
def translate_ofset(bbox, offset_x, offset_y):
|
166
|
+
return (bbox[0] + offset_x, bbox[1] + offset_y, bbox[2] + offset_x, bbox[3] + offset_y)
|
167
|
+
|
168
|
+
def bounding_boxes_overlap(bbox1, bbox2):
|
169
|
+
if bbox1[2] < bbox2[0] or bbox2[0] > bbox1[2]:
|
170
|
+
return False
|
171
|
+
if bbox1[3] < bbox2[1] or bbox2[3] > bbox1[3]:
|
172
|
+
return False
|
173
|
+
return True
|
174
|
+
|
175
|
+
def merge_bounding_boxes(bbox1, bbox2):
|
176
|
+
x_min = min(bbox1[0], bbox2[0])
|
177
|
+
y_min = min(bbox1[1], bbox2[1])
|
178
|
+
x_max = max(bbox1[2], bbox2[2])
|
179
|
+
y_max = max(bbox1[3], bbox2[3])
|
180
|
+
return (x_min, y_min, x_max, y_max)
|
181
|
+
|
182
|
+
def merge_bounding_box_list(bboxes):
|
183
|
+
merged_bboxes = []
|
184
|
+
while bboxes:
|
185
|
+
bbox = bboxes.pop()
|
186
|
+
overlap_found = False
|
187
|
+
for i, other_bbox in enumerate(merged_bboxes):
|
188
|
+
if bounding_boxes_overlap(bbox, other_bbox):
|
189
|
+
merged_bboxes[i] = merge_bounding_boxes(bbox, other_bbox)
|
190
|
+
overlap_found = True
|
191
|
+
break
|
192
|
+
if not overlap_found:
|
193
|
+
merged_bboxes.append(bbox)
|
194
|
+
return merged_bboxes
|
195
|
+
|
196
|
+
detections = []
|
197
|
+
for region in subdivide_image(image):
|
198
|
+
detections.extend(countgd_counting("pedestrian", region))
|
199
|
+
detections = merge_bounding_box_list(detections)
|
200
|
+
print(len(detections))
|
201
|
+
</execute_python>
|
202
|
+
|
203
|
+
OBSERVATION: 24
|
204
|
+
<count>7</count>
|
205
|
+
|
206
|
+
AGENT: <thinking>countgd_counting detected 24 pedestrians in the image by subdividing the image into smaller regions, this differs from claude35_vqa's answer of 5-6 pedestrians. I will use countgd_counting's answer since it is specifically designed to count objects.</thinking>
|
207
|
+
<finalize_plan>This plan successfully detected 24 pedestrians in the image by subdividing the image into smaller regions and using the countgd_counting tool on each region. The bounding boxes were then merged to count the unique pedestrians.</finalize_plan>
|
208
|
+
--- END EXAMPLE1 ---
|
209
|
+
"""
|
210
|
+
|
211
|
+
EXAMPLE_PLAN2 = """
|
212
|
+
--- EXAMPLE2 ---
|
213
|
+
USER: Can you identify which boxes have tape on them in this video?
|
214
|
+
<count>10</count>
|
215
|
+
|
216
|
+
AGENT: <thinking>I need to gather more information, I will ask for a description of the video and a solution to the problem to work towards as well as a suggestion on how best to solve it.</thinking>
|
217
|
+
<execute_python>
|
218
|
+
frames = extract_frames_and_timestamps('video.mp4', 1)
|
219
|
+
frames = [f["frame"] for f in frames]
|
220
|
+
claude35_vqa('Can you describe this video? How many boxes are there and do they have tape on them?', frames[:5])
|
221
|
+
suggestion('How can I identify boxes with tape on them in the video?', frames[:5])
|
222
|
+
</execute_python>
|
223
|
+
|
224
|
+
OBSERVATION: [claude35_vqa output]
|
225
|
+
In this video, there appear to be 2 cardboard boxes moving along a roller conveyor system. The boxes are brown/kraft colored and appear to have tape sealing them. You can see clear packing tape on the top and sides of the boxes.
|
226
|
+
[end of claude35_vqa output]
|
227
|
+
|
228
|
+
[suggestions]
|
229
|
+
[suggestion 0]
|
230
|
+
You are trying to track objects in a video and identify features on those objects. You need to first get a tool that can track objects in a video, and then for each object find another tool to identify the features on the object. You can use the following code to help with this task:
|
231
|
+
|
232
|
+
track_predictions = <object tracking tool>("object", video_frames)
|
233
|
+
|
234
|
+
|
235
|
+
# Step 1: go through each frame and each prediction and extract the predicted bounding boxes as crops
|
236
|
+
obj_to_info = {{}}
|
237
|
+
for frame, frame_predictions in zip(video_frames, track_predictions):
|
238
|
+
for obj in frame_predictions:
|
239
|
+
if obj["label"] not in obj_to_info:
|
240
|
+
obj_to_info[obj["label"]] = []
|
241
|
+
height, width = frame.shape[:2]
|
242
|
+
crop = frame[
|
243
|
+
int(obj["bbox"][1] * height) : int(obj["bbox"][3] * height),
|
244
|
+
int(obj["bbox"][0] * width) : int(obj["bbox"][2] * width),
|
245
|
+
:,
|
246
|
+
]
|
247
|
+
# For each crop use an object detection tool, VQA tool or classification tool to identify if the object contains the features you want
|
248
|
+
detections = <object detection tool to find features>("<your feature here>", crop)
|
249
|
+
obj_to_info[obj["label"]].extend(detections)
|
250
|
+
|
251
|
+
print(f"{len(obj_to_info)} objects tracked")
|
252
|
+
|
253
|
+
objects_with_info = set()
|
254
|
+
for infos in obj_to_info:
|
255
|
+
for info in info:
|
256
|
+
if info["label"] == "<your feature here>":
|
257
|
+
objects_with_info.add(info)
|
258
|
+
break
|
259
|
+
|
260
|
+
print(f"{len(objects_with_info)} objects with features found")
|
261
|
+
[end of suggestion 0]
|
262
|
+
[end of suggestions]
|
263
|
+
<count>9</count>
|
264
|
+
|
265
|
+
AGENT: <thinking>Following the suggested plan, I first need to identify a tool for tracking.</thinking>
|
266
|
+
<execute_python>
|
267
|
+
get_tool_for_task('Identify and track the boxes in the video', frames[:5])
|
268
|
+
</execute_python>
|
269
|
+
|
270
|
+
OBSERVATION:
|
271
|
+
[get_tool_for_task output]
|
272
|
+
For tracking boxes moving on a conveyor belt, we need a tool that can consistently track the same box across frames without losing it or double counting. Looking at the outputs: florence2_sam2_video_tracking successfully tracks the single box across all 5 frames, maintaining consistent tracking IDs and showing the box's movement along the conveyor.
|
273
|
+
|
274
|
+
'florence2_sam2_video_tracking' is a tool that can segment and track multiple
|
275
|
+
entities in a video given a text prompt such as category names or referring
|
276
|
+
expressions. You can optionally separate the categories in the text with commas. It
|
277
|
+
can find new objects every 'chunk_length' frames and is useful for tracking and
|
278
|
+
counting without duplicating counts and always outputs scores of 1.0.
|
279
|
+
|
280
|
+
Parameters:
|
281
|
+
prompt (str): The prompt to ground to the video.
|
282
|
+
frames (List[np.ndarray]): The list of frames to ground the prompt to.
|
283
|
+
chunk_length (Optional[int]): The number of frames to re-run florence2 to find
|
284
|
+
new objects.
|
285
|
+
|
286
|
+
Returns:
|
287
|
+
List[List[Dict[str, Any]]]: A list of list of dictionaries containing the
|
288
|
+
label,segment mask and bounding boxes. The outer list represents each frame and
|
289
|
+
the inner list is the entities per frame. The label contains the object ID
|
290
|
+
followed by the label name. The objects are only identified in the first framed
|
291
|
+
and tracked throughout the video.
|
292
|
+
|
293
|
+
Example
|
294
|
+
-------
|
295
|
+
>>> florence2_sam2_video("car, dinosaur", frames)
|
296
|
+
[
|
297
|
+
[
|
298
|
+
{
|
299
|
+
'label': '0: dinosaur',
|
300
|
+
'bbox': [0.1, 0.11, 0.35, 0.4],
|
301
|
+
'mask': array([[0, 0, 0, ..., 0, 0, 0],
|
302
|
+
...,
|
303
|
+
[0, 0, 0, ..., 0, 0, 0]], dtype=uint8),
|
304
|
+
},
|
305
|
+
],
|
306
|
+
...
|
307
|
+
]
|
308
|
+
[end of get_tool_for_task output]
|
309
|
+
<count>8</count>
|
310
|
+
|
311
|
+
AGENT: <thinking>Now that I have the tool for tracking I need to find a tool to identify if there is tape on each of the boxes.</thinking>
|
312
|
+
<execute_python>
|
313
|
+
track_predictions = florence2_sam2_video_tracking("box", frames)
|
314
|
+
|
315
|
+
# only use first 5 frames to reduce processing time
|
316
|
+
crops = []
|
317
|
+
for frame, frame_predictions in zip(frames[:5], track_predictions[:5]):
|
318
|
+
for obj in frame_predictions:
|
319
|
+
height, width = frame.shape[:2]
|
320
|
+
crop = frame[
|
321
|
+
int(obj["bbox"][1] * height) : int(obj["bbox"][3] * height),
|
322
|
+
int(obj["bbox"][0] * width) : int(obj["bbox"][2] * width),
|
323
|
+
:,
|
324
|
+
]
|
325
|
+
crops.append(crop)
|
326
|
+
|
327
|
+
# only use first 5 crops to reduce processing time
|
328
|
+
get_tool_for_task('Identify if there is tape on the boxes', crops[:3])
|
329
|
+
|
330
|
+
OBSERVATION:
|
331
|
+
[get_tool_for_task output]
|
332
|
+
owl_v2_image performed best as it specifically detected multiple instances of tape with localized bounding boxes, which matches what's visible in the images.
|
333
|
+
|
334
|
+
'owl_v2_image' is a tool that can detect and count multiple objects given a text
|
335
|
+
prompt such as category names or referring expressions on images. The categories in
|
336
|
+
text prompt are separated by commas. It returns a list of bounding boxes with
|
337
|
+
normalized coordinates, label names and associated probability scores.
|
338
|
+
|
339
|
+
Parameters:
|
340
|
+
prompt (str): The prompt to ground to the image.
|
341
|
+
image (np.ndarray): The image to ground the prompt to.
|
342
|
+
box_threshold (float, optional): The threshold for the box detection. Defaults
|
343
|
+
to 0.10.
|
344
|
+
fine_tune_id (Optional[str]): If you have a fine-tuned model, you can pass the
|
345
|
+
fine-tuned model ID here to use it.
|
346
|
+
|
347
|
+
Returns:
|
348
|
+
List[Dict[str, Any]]: A list of dictionaries containing the score, label, and
|
349
|
+
bounding box of the detected objects with normalized coordinates between 0
|
350
|
+
and 1 (xmin, ymin, xmax, ymax). xmin and ymin are the coordinates of the
|
351
|
+
top-left and xmax and ymax are the coordinates of the bottom-right of the
|
352
|
+
bounding box.
|
353
|
+
|
354
|
+
Example
|
355
|
+
-------
|
356
|
+
>>> owl_v2_image("car, dinosaur", image)
|
357
|
+
[
|
358
|
+
{'score': 0.99, 'label': 'dinosaur', 'bbox': [0.1, 0.11, 0.35, 0.4]},
|
359
|
+
{'score': 0.98, 'label': 'car', 'bbox': [0.2, 0.21, 0.45, 0.5},
|
360
|
+
]
|
361
|
+
[end of get_tool_for_task output]
|
362
|
+
<count>7</count>
|
363
|
+
|
364
|
+
AGENT: <thinking>Now that I have the tools for tracking and identifying tape on the boxes, I will use them to accomplish the user's request.</thinking>
|
365
|
+
<execute_python>
|
366
|
+
obj_to_info = {{}}
|
367
|
+
for frame, frame_predictions in zip(frames, track_predictions):
|
368
|
+
for obj in frame_predictions:
|
369
|
+
if obj["label"] not in obj_to_info:
|
370
|
+
obj_to_info[obj["label"]] = []
|
371
|
+
height, width = frame.shape[:2]
|
372
|
+
crop = frame[
|
373
|
+
int(obj["bbox"][1] * height) : int(obj["bbox"][3] * height),
|
374
|
+
int(obj["bbox"][0] * width) : int(obj["bbox"][2] * width),
|
375
|
+
:,
|
376
|
+
]
|
377
|
+
detections = owl_v2_image("tape", crop)
|
378
|
+
obj_to_info[obj["label"]].extend(detections)
|
379
|
+
|
380
|
+
|
381
|
+
print(f"{len(obj_to_info)} boxes tracked")
|
382
|
+
objects_with_tape = set()
|
383
|
+
for infos in obj_to_info:
|
384
|
+
for info in info:
|
385
|
+
if info["label"] == "tape":
|
386
|
+
objects_with_tape.add(info)
|
387
|
+
break
|
388
|
+
|
389
|
+
print(f"{len(objects_with_tape)} boxes with tape found")
|
390
|
+
</execute_python>
|
391
|
+
|
392
|
+
OBJERVATION:
|
393
|
+
3 boxes were tracked
|
394
|
+
2 boxes with tape found
|
395
|
+
<count>6</count>
|
396
|
+
|
397
|
+
AGENT: <thinking>After tracking the boxes and identifying tape on them, I found 3 boxes in the video and 2 of them had tape on them. I will finalize this plan.</thinking>
|
398
|
+
<finalize_plan>This plan successfully tracked the boxes in the video and identified tape on them. It found 3 boxes in the video and 2 of them had tape on them.</finalize_plan>
|
399
|
+
--- END EXAMPLE2 ---
|
400
|
+
"""
|
401
|
+
|
402
|
+
PICK_PLAN = """
|
403
|
+
**Role**: You are an expert evaluator that can understand user requests and evaluate different plans to determine the best one.
|
404
|
+
|
405
|
+
**Task**: You are given a planning chains by a planning agent complete with thoughts, python executions and observations. You are also given 3 different responses to add to the chain. Your task is to evaluate the different plans and determine the best one for the user request.
|
406
|
+
|
407
|
+
**Current Planning**:
|
408
|
+
--- START PLANNING ---
|
409
|
+
{planning}
|
410
|
+
--- END PLANNING ---
|
411
|
+
|
412
|
+
**Responses**:
|
413
|
+
--- RESPONSE1 ---
|
414
|
+
{response1}
|
415
|
+
--- END RESPONSE1 ---
|
416
|
+
--- RESPONSE2 ---
|
417
|
+
{response2}
|
418
|
+
--- END RESPONSE2 ---
|
419
|
+
--- RESPONSE3 ---
|
420
|
+
{response3}
|
421
|
+
--- END RESPONSE3 ---
|
422
|
+
|
423
|
+
**Instructions**:
|
424
|
+
1. Read over the planning chain and responses.
|
425
|
+
2. The agent can output <thinking>, <execute_python> or <finalize_plan> tags. Any code under OBSERVATION: has not been executed by the agent, only code under <execute_python> tags has been executed.
|
426
|
+
3. Rate each instruction on a scale of 1 to 10 based on how well it solves the user request and give your thoughts on why you rated it that way.
|
427
|
+
4. Return your results in the following JSON format inside <json> tags:
|
428
|
+
|
429
|
+
<json>
|
430
|
+
{{
|
431
|
+
"response1": # int: rating for response1,
|
432
|
+
"response1_thoughts": # str: your thoughts on response1,
|
433
|
+
"response2": # int: rating for response2,
|
434
|
+
"response2_thoughts": # str: your thoughts on response2,
|
435
|
+
"response3": # int: rating for response3,
|
436
|
+
"response3_thoughts": # str: your thoughts on response3
|
437
|
+
}}
|
438
|
+
</json>
|
439
|
+
"""
|
440
|
+
|
441
|
+
CATEGORIZE_TOOL_REQUEST = """
|
442
|
+
You are given a task: {task} from the user. Your task is to extract the type of category this task belongs to, it can be one or more of the following:
|
443
|
+
- "object detection and counting" - detecting objects or counting objects from a text prompt in an image or video.
|
444
|
+
- "classification" - classifying objects in an image given a text prompt.
|
445
|
+
- "segmentation" - segmenting objects in an image or video given a text prompt.
|
446
|
+
- "OCR" - extracting text from an image.
|
447
|
+
- "VQA" - answering questions about an image or video, can also be used for text extraction.
|
448
|
+
- "video object tracking" - tracking objects in a video.
|
449
|
+
- "depth and pose estimation" - estimating the depth or pose of objects in an image.
|
450
|
+
|
451
|
+
Return the category inside tags <category># your categories here</category>.
|
452
|
+
"""
|
453
|
+
|
454
|
+
TEST_TOOLS = """
|
455
|
+
**Role**: You are an expert software programming that specializes in tool testing.
|
456
|
+
|
457
|
+
**Task**: You are responsible for testing different tools on a set of images to determine the best tool for the user request.
|
458
|
+
|
459
|
+
**Tools**: This is the documentation for the functions you have access to. You may call any of these functions to help you complete the task. They are available through importing `from vision_agent.tools import *`.
|
460
|
+
{tool_docs}
|
461
|
+
|
462
|
+
**Previous Attempts**: You previously tried to execute this code and got the following error or no output (if empty that means you have not tried to previously execute code, if it is 'EMPTY' that means you have tried but got no output):
|
463
|
+
{previous_attempts}
|
464
|
+
|
465
|
+
**User Request**:
|
466
|
+
{user_request}
|
467
|
+
|
468
|
+
**Media**:
|
469
|
+
{media}
|
470
|
+
|
471
|
+
**Examples**:
|
472
|
+
{examples}
|
473
|
+
|
474
|
+
**Instructions**:
|
475
|
+
1. List all the tools under **Tools** and the user request. Write a program to load the media and call every tool in parallel and print it's output along with other relevant information.
|
476
|
+
2. Create a dictionary where the keys are the tool name and the values are the tool outputs. Remove numpy arrays from the printed dictionary.
|
477
|
+
3. Your test case MUST run only on the given images which are {media}
|
478
|
+
4. Print this final dictionary.
|
479
|
+
5. Output your code in the following format wrapped in <code> tags:
|
480
|
+
<code>
|
481
|
+
# Your code here
|
482
|
+
</code>
|
483
|
+
"""
|
484
|
+
|
485
|
+
TEST_TOOLS_EXAMPLE1 = """
|
486
|
+
--- EXAMPLE1 ---
|
487
|
+
**User Request**:
|
488
|
+
Count the number of pedestrians across all the images.
|
489
|
+
|
490
|
+
**Media**:
|
491
|
+
["image1.jpg", "image2.jpg", "image3.jpg", "image4.jpg"]
|
492
|
+
|
493
|
+
<code>
|
494
|
+
from concurrent.futures import ThreadPoolExecutor, as_completed
|
495
|
+
from vision_agent.tools import load_image, owl_v2_image, florence2_phrase_grounding, countgd_counting
|
496
|
+
|
497
|
+
# process functions in a try catch so that if it fails it doesn't cause `as_completed` to hang
|
498
|
+
def process_owl_v2(image_paths):
|
499
|
+
try:
|
500
|
+
results = []
|
501
|
+
for image_path in image_paths:
|
502
|
+
image = load_image(image_path)
|
503
|
+
results.extend(owl_v2_image("person", image))
|
504
|
+
except Exception as e:
|
505
|
+
results = f"Encountered error when executing process_owl_v2: {str(e)}"
|
506
|
+
return results
|
507
|
+
|
508
|
+
def process_florence2(image_paths):
|
509
|
+
try:
|
510
|
+
results = []
|
511
|
+
for image_path in image_paths:
|
512
|
+
image = load_image(image_path)
|
513
|
+
results.extend(florence2_phrase_grounding("person", image))
|
514
|
+
except Exception as e:
|
515
|
+
results = f"Encountered error when executing process_florence2: {str(e)}"
|
516
|
+
return results
|
517
|
+
|
518
|
+
def process_countgd(image_paths):
|
519
|
+
try:
|
520
|
+
results = []
|
521
|
+
for image_path in image_paths:
|
522
|
+
image = load_image(image_path)
|
523
|
+
results.extend(countgd_counting("person", image))
|
524
|
+
except Exception as e:
|
525
|
+
results = f"Encountered error when executing process_countgd: {str(e)}"
|
526
|
+
return results
|
527
|
+
|
528
|
+
image_paths = ["image1.jpg", "image2.jpg", "image3.jpg", "image4.jpg"]
|
529
|
+
|
530
|
+
with ThreadPoolExecutor() as executor:
|
531
|
+
futures = {{
|
532
|
+
executor.submit(process_owl_v2, image_paths): "owl_v2_image",
|
533
|
+
executor.submit(process_florence2, image_paths): "florence2_phrase_grounding",
|
534
|
+
executor.submit(process_countgd, image_paths): "countgd_counting",
|
535
|
+
}}
|
536
|
+
|
537
|
+
final_results = {{}}
|
538
|
+
for future in as_completed(futures):
|
539
|
+
tool_name = futures[future]
|
540
|
+
final_results[tool_name] = future.result()
|
541
|
+
|
542
|
+
print(final_results)
|
543
|
+
</code>
|
544
|
+
--- END EXAMPLE1 ---
|
545
|
+
"""
|
546
|
+
|
547
|
+
TEST_TOOLS_EXAMPLE2 = """
|
548
|
+
--- EXAMPLE2 ---
|
549
|
+
**User Request**:
|
550
|
+
Count the number of people in the video.
|
551
|
+
|
552
|
+
**Media**:
|
553
|
+
["video.mp4"]
|
554
|
+
|
555
|
+
<code>
|
556
|
+
import numpy as np
|
557
|
+
from concurrent.futures import ThreadPoolExecutor, as_completed
|
558
|
+
from vision_agent.tools import extract_frames_and_timestamps, owl_v2_video, florence2_sam2_video_tracking
|
559
|
+
|
560
|
+
# sample at 1 FPS and use the first 10 frames to reduce processing time
|
561
|
+
frames = extract_frames_and_timestamps("video.mp4", 1)
|
562
|
+
frames = [f["frame"] for f in frames][:10]
|
563
|
+
|
564
|
+
# strip arrays from the output to make it easier to read
|
565
|
+
def remove_arrays(o):
|
566
|
+
if isinstance(o, list):
|
567
|
+
return [remove_arrays(e) for e in o]
|
568
|
+
elif isinstance(o, dict):
|
569
|
+
return {{k: remove_arrays(v) for k, v in o.items()}}
|
570
|
+
elif isinstance(o, np.ndarray):
|
571
|
+
return "array: " + str(o.shape)
|
572
|
+
else:
|
573
|
+
return o
|
574
|
+
|
575
|
+
def process_owl_v2_video(frames):
|
576
|
+
try:
|
577
|
+
results = owl_v2_video("person", frames)
|
578
|
+
except Exception as e:
|
579
|
+
results = f"Encountered error when executing process_owl_v2_video: {str(e)}"
|
580
|
+
return results
|
581
|
+
|
582
|
+
def process_florence2_sam2(frames):
|
583
|
+
try:
|
584
|
+
results = florence2_sam2_video_tracking("person", frames)
|
585
|
+
except Exception as e:
|
586
|
+
results = f"Encountered error when executing process_florence2_sam2: {str(e)}"
|
587
|
+
return results
|
588
|
+
|
589
|
+
|
590
|
+
with ThreadPoolExecutor() as executor:
|
591
|
+
futures = {{
|
592
|
+
executor.submit(process_owl_v2_video, frames): "owl_v2_video",
|
593
|
+
executor.submit(process_florence2_sam2, frames): "florence2_sam2_video_tracking",
|
594
|
+
}}
|
595
|
+
final_results = {{}}
|
596
|
+
for future in as_completed(futures):
|
597
|
+
tool_name = futures[future]
|
598
|
+
final_results[tool_name] = remove_arrays(future.result())
|
599
|
+
|
600
|
+
print(final_results)
|
601
|
+
</code>
|
602
|
+
--- END EXAMPLE2 ---
|
603
|
+
"""
|
604
|
+
|
605
|
+
PICK_TOOL = """
|
606
|
+
**Role**: You are an expert evaluator that can understand user requests and evaluate the output of different tools.
|
607
|
+
|
608
|
+
**Task**: You are given the output of different tools for a user request along with the image. You must evaluate the output and determine the best tool for the user request.
|
609
|
+
|
610
|
+
**User Request**:
|
611
|
+
{user_request}
|
612
|
+
|
613
|
+
**Tools**: This is the documentation of all the functions that were tested.
|
614
|
+
{tool_docs}
|
615
|
+
|
616
|
+
**Testing Code and Tool Output**:
|
617
|
+
{context}
|
618
|
+
|
619
|
+
**Previous Attempt**: This is the code and output of the previous attempt, if it is empty then there was no previous attempt.
|
620
|
+
{previous_attempts}
|
621
|
+
|
622
|
+
**Instructions**:
|
623
|
+
1. Re-read the user request, plans, tool outputs and examine the image.
|
624
|
+
2. Given the user request, try to solve the problem yourself.
|
625
|
+
3. Pick which tool output best matches your solution and the user request, DO NOT focus on other factors.
|
626
|
+
4. DO NOT modify confidence thresholds unless the tool output is completely wrong.
|
627
|
+
5. Remember for videos that in order to count objects a video some sort of tracking is needed, or else you will overcount the objects.
|
628
|
+
7. Return the following JSON format inside <json> tags using the exact tool name for best_tool:
|
629
|
+
<json>
|
630
|
+
{{
|
631
|
+
"predicted_answer": str # the answer you would expect from the best plan
|
632
|
+
"thoughts": str # your thought process for choosing the best tool over other tools and any modifications you madeas well as the prompt you used with the tool.
|
633
|
+
"best_tool": str # the name of the best tool
|
634
|
+
}}
|
635
|
+
</json>
|
636
|
+
"""
|
637
|
+
|
638
|
+
PICK_TOOL_V2 = """
|
639
|
+
**Role**: You are an expert evaluator that can understand user requests and evaluate the output of different tools.
|
640
|
+
|
641
|
+
**Task**: You are given the output of different tools for a user request along with the image. You must evaluate the output and determine the best tool for the user request.
|
642
|
+
|
643
|
+
**User Request**:
|
644
|
+
{user_request}
|
645
|
+
|
646
|
+
**Tools**: This is the documentation of all the functions that were tested.
|
647
|
+
{tool_docs}
|
648
|
+
|
649
|
+
**Testing Code and Tool Output**:
|
650
|
+
{context}
|
651
|
+
|
652
|
+
**Previous Attempt**: This is the code and output of the previous attempt, if it is empty then there was no previous attempt.
|
653
|
+
{previous_attempts}
|
654
|
+
|
655
|
+
**Instructions**:
|
656
|
+
1. Re-read the user request, plans, tool outputs and examine the image.
|
657
|
+
2. Given the user request, try to solve the problem yourself.
|
658
|
+
3. Pick which tool output best matches your solution first and the user request, then consider other factors like box size, etc. DO NOT worry about low confidence scores if the output is correct.
|
659
|
+
4. DO NOT modify confidence thresholds unless the tool output is completely wrong.
|
660
|
+
5. Remember for videos that in order to count objects a video some sort of tracking is needed, or else you will overcount the objects.
|
661
|
+
6. Assign each tool a score from 0 to 10 based on how well it solves the user request. A score of 8+ means the tool output matches your solution and the tool is the best choice, 5-7 means the tool output is okay but needs some modifications, less than 5 means the tool output is bad and the tool should not be used. Return the the following JSON format inside <json> tags using the exact tool name as the key and the score as the value:
|
662
|
+
<json>
|
663
|
+
{{
|
664
|
+
"predicted_answer": str # the answer you would expect from the best plan
|
665
|
+
"thoughts": str # your thought process for choosing the best tool over other tools and any modifications you madeas well as the prompt you used with the tool.
|
666
|
+
"first tool": int # the score for the first tool
|
667
|
+
"second tool": int # the score for the second tool
|
668
|
+
...
|
669
|
+
}}
|
670
|
+
</json>
|
671
|
+
"""
|
672
|
+
|
673
|
+
FINALIZE_PLAN = """
|
674
|
+
**Role**: You are an expert AI model that can understand the user request and construct plans to accomplish it.
|
675
|
+
|
676
|
+
**Task**: You are given a chain of thoughts, python executions and observations from a planning agent as it tries to construct a plan to solve a user request. Your task is to summarize the plan it found so that another programming agnet to write a program to accomplish the user request.
|
677
|
+
|
678
|
+
**Planning**: Here is chain of thoughts, executions and observations from the planning agent:
|
679
|
+
{planning}
|
680
|
+
|
681
|
+
**Instructions**:
|
682
|
+
1. Read the chain of thoughts and python executions.
|
683
|
+
2. Summarize the plan that the planning agent found.
|
684
|
+
3. Include ALL relevant python code in your plan to accomplish the user request.
|
685
|
+
4. Specifically call out the tools used and the order in which they were used. Only include tools obtained from calling `get_tool_for_task`.
|
686
|
+
5. Do not include {excluded_tools} tools in your instructions.
|
687
|
+
6. Respond in the following format with JSON surrounded by <json> tags and code surrounded by <code> tags:
|
688
|
+
|
689
|
+
<json>
|
690
|
+
{{
|
691
|
+
"plan": str # the plan you have summarized
|
692
|
+
"instructions": [
|
693
|
+
str # the instructions for each step in the plan with either specific code or a specific tool name
|
694
|
+
]
|
695
|
+
}}
|
696
|
+
</json>
|
697
|
+
|
698
|
+
<code>
|
699
|
+
# Code snippets here
|
700
|
+
</code>
|
701
|
+
"""
|
702
|
+
|
703
|
+
FIX_BUG = """
|
704
|
+
**Role**: You are a software programmer responsible for fixing bugs in code.
|
705
|
+
|
706
|
+
**Task**: Your responsibility is to fix the bug in the code and provide the correct output.
|
707
|
+
|
708
|
+
**Instructions**: This is the previous chat history:
|
709
|
+
--- CHAT HISTORY ---
|
710
|
+
{chat_history}
|
711
|
+
--- END CHAT HISTORY ---
|
712
|
+
|
713
|
+
This is the code that was previously run:
|
714
|
+
<python>
|
715
|
+
{code}
|
716
|
+
</python>
|
717
|
+
|
718
|
+
And this is the error message that was received:
|
719
|
+
<error>
|
720
|
+
{error}
|
721
|
+
</error>
|
722
|
+
|
723
|
+
Please fix the code by correcting the error. Do not wrap failures in try-except statements and instead solve the root cause. Output your code in the following format wrapped in <code> tags:
|
724
|
+
|
725
|
+
<code>
|
726
|
+
# Your fixed code here
|
727
|
+
</code>
|
728
|
+
"""
|
729
|
+
|
730
|
+
CRITIQUE_PLAN = """
|
731
|
+
**Role**: You are an expert evaluator that can understand user requests and evaluate different plans to determine the best one.
|
732
|
+
|
733
|
+
**Task**: You are given a planning chain by a planning agent complete with thoughts, python executions and observations. The agent is trying to use tools to accomplish a vision task requested by a user. Your job is to evaluate the plan so far given the user request and image.
|
734
|
+
|
735
|
+
|
736
|
+
**Current Planning**:
|
737
|
+
--- START PLANNING ---
|
738
|
+
{planning}
|
739
|
+
--- END PLANNING ---
|
740
|
+
|
741
|
+
**Instructions**:
|
742
|
+
1. Read over the planning chain and examine the image and user request. Try to solve the user problem yourself and compare it against the solution the agent has come up with so far.
|
743
|
+
2. Rate the plan on a scale of 1 to 10 based on how well it solves the user request. A score of 8+ means the plan is good and the agent should continue on the same path, 5-7 means the plan is good but needs some minor modifications, less than 5 means the plan is bad and the agent should consider a different approach.
|
744
|
+
3. Provide your score and thoughts in the following format:
|
745
|
+
|
746
|
+
<score>Your score here</score>
|
747
|
+
<thoughts>Your thoughts here</thoughts>
|
748
|
+
"""
|