2
2
3
3
import asyncio
4
4
from dataclasses import dataclass
5
- from typing import TYPE_CHECKING , Any
5
+ from typing import TYPE_CHECKING , Any , Union , List
6
6
7
7
from openai .types .responses import (
8
8
ResponseComputerToolCall ,
65
65
66
66
67
67
class QueueCompleteSentinel :
68
+ """A sentinel value to indicate that the queue is complete."""
68
69
pass
69
70
70
71
@@ -97,8 +98,7 @@ class ProcessedResponse:
97
98
computer_actions : list [ToolRunComputerAction ]
98
99
99
100
def has_tools_to_run (self ) -> bool :
100
- # Handoffs, functions and computer actions need local processing
101
- # Hosted tools have already run, so there's nothing to do.
101
+ """Check if there are any tools to run."""
102
102
return any (
103
103
[
104
104
self .handoffs ,
@@ -151,6 +151,7 @@ def generated_items(self) -> list[RunItem]:
151
151
def get_model_tracing_impl (
152
152
tracing_disabled : bool , trace_include_sensitive_data : bool
153
153
) -> ModelTracing :
154
+ """Get the model tracing implementation based on the tracing configuration."""
154
155
if tracing_disabled :
155
156
return ModelTracing .DISABLED
156
157
elif trace_include_sensitive_data :
@@ -176,6 +177,7 @@ async def execute_tools_and_side_effects(
176
177
context_wrapper : RunContextWrapper [TContext ],
177
178
run_config : RunConfig ,
178
179
) -> SingleStepResult :
180
+ """Execute tools and side effects for the current step."""
179
181
# Make a copy of the generated items
180
182
pre_step_items = list (pre_step_items )
181
183
@@ -271,6 +273,7 @@ def process_model_response(
271
273
output_schema : AgentOutputSchema | None ,
272
274
handoffs : list [Handoff ],
273
275
) -> ProcessedResponse :
276
+ """Process the model response and extract relevant information."""
274
277
items : list [RunItem ] = []
275
278
276
279
run_handoffs = []
@@ -356,6 +359,7 @@ async def execute_function_tool_calls(
356
359
context_wrapper : RunContextWrapper [TContext ],
357
360
config : RunConfig ,
358
361
) -> list [RunItem ]:
362
+ """Execute function tool calls."""
359
363
async def run_single_tool (
360
364
func_tool : FunctionTool , tool_call : ResponseFunctionToolCall
361
365
) -> str :
@@ -422,6 +426,7 @@ async def execute_computer_actions(
422
426
context_wrapper : RunContextWrapper [TContext ],
423
427
config : RunConfig ,
424
428
) -> list [RunItem ]:
429
+ """Execute computer actions."""
425
430
results : list [RunItem ] = []
426
431
# Need to run these serially, because each action can affect the computer state
427
432
for action in actions :
@@ -451,6 +456,7 @@ async def execute_handoffs(
451
456
context_wrapper : RunContextWrapper [TContext ],
452
457
run_config : RunConfig ,
453
458
) -> SingleStepResult :
459
+ """Execute handoffs."""
454
460
# If there is more than one handoff, add tool responses that reject those handoffs
455
461
if len (run_handoffs ) > 1 :
456
462
output_message = "Multiple handoffs detected, ignoring this one."
@@ -470,9 +476,20 @@ async def execute_handoffs(
470
476
actual_handoff = run_handoffs [0 ]
471
477
with handoff_span (from_agent = agent .name ) as span_handoff :
472
478
handoff = actual_handoff .handoff
473
- new_agent : Agent [Any ] = await handoff .on_invoke_handoff (
474
- context_wrapper , actual_handoff .tool_call .arguments
475
- )
479
+ try :
480
+ new_agent : Agent [Any ] = await handoff .on_invoke_handoff (
481
+ context_wrapper , actual_handoff .tool_call .arguments
482
+ )
483
+ except Exception as e :
484
+ _utils .attach_error_to_span (
485
+ span_handoff ,
486
+ SpanError (
487
+ message = "Error invoking handoff" ,
488
+ data = {"error" : str (e )},
489
+ )
490
+ )
491
+ raise
492
+
476
493
span_handoff .span_data .to_agent = new_agent .name
477
494
478
495
# Append a tool output item for the handoff
@@ -568,6 +585,7 @@ async def execute_final_output(
568
585
hooks : RunHooks [TContext ],
569
586
context_wrapper : RunContextWrapper [TContext ],
570
587
) -> SingleStepResult :
588
+ """Execute final output."""
571
589
# Run the on_end hooks
572
590
await cls .run_final_output_hooks (agent , hooks , context_wrapper , final_output )
573
591
@@ -587,6 +605,7 @@ async def run_final_output_hooks(
587
605
context_wrapper : RunContextWrapper [TContext ],
588
606
final_output : Any ,
589
607
):
608
+ """Run the final output hooks."""
590
609
await asyncio .gather (
591
610
hooks .on_agent_end (context_wrapper , agent , final_output ),
592
611
agent .hooks .on_end (context_wrapper , agent , final_output )
@@ -602,8 +621,19 @@ async def run_single_input_guardrail(
602
621
input : str | list [TResponseInputItem ],
603
622
context : RunContextWrapper [TContext ],
604
623
) -> InputGuardrailResult :
624
+ """Run a single input guardrail."""
605
625
with guardrail_span (guardrail .get_name ()) as span_guardrail :
606
- result = await guardrail .run (agent , input , context )
626
+ try :
627
+ result = await guardrail .run (agent , input , context )
628
+ except Exception as e :
629
+ _utils .attach_error_to_span (
630
+ span_guardrail ,
631
+ SpanError (
632
+ message = "Error running input guardrail" ,
633
+ data = {"error" : str (e )},
634
+ )
635
+ )
636
+ raise
607
637
span_guardrail .span_data .triggered = result .output .tripwire_triggered
608
638
return result
609
639
@@ -615,8 +645,19 @@ async def run_single_output_guardrail(
615
645
agent_output : Any ,
616
646
context : RunContextWrapper [TContext ],
617
647
) -> OutputGuardrailResult :
648
+ """Run a single output guardrail."""
618
649
with guardrail_span (guardrail .get_name ()) as span_guardrail :
619
- result = await guardrail .run (agent = agent , agent_output = agent_output , context = context )
650
+ try :
651
+ result = await guardrail .run (agent = agent , agent_output = agent_output , context = context )
652
+ except Exception as e :
653
+ _utils .attach_error_to_span (
654
+ span_guardrail ,
655
+ SpanError (
656
+ message = "Error running output guardrail" ,
657
+ data = {"error" : str (e )},
658
+ )
659
+ )
660
+ raise
620
661
span_guardrail .span_data .triggered = result .output .tripwire_triggered
621
662
return result
622
663
@@ -626,6 +667,7 @@ def stream_step_result_to_queue(
626
667
step_result : SingleStepResult ,
627
668
queue : asyncio .Queue [StreamEvent | QueueCompleteSentinel ],
628
669
):
670
+ """Stream the step result to the queue."""
629
671
for item in step_result .new_step_items :
630
672
if isinstance (item , MessageOutputItem ):
631
673
event = RunItemStreamEvent (item = item , name = "message_output_created" )
@@ -695,6 +737,7 @@ async def execute(
695
737
context_wrapper : RunContextWrapper [TContext ],
696
738
config : RunConfig ,
697
739
) -> RunItem :
740
+ """Execute a computer action."""
698
741
output_func = (
699
742
cls ._get_screenshot_async (action .computer_tool .computer , action .tool_call )
700
743
if isinstance (action .computer_tool .computer , AsyncComputer )
@@ -741,25 +784,29 @@ async def _get_screenshot_sync(
741
784
computer : Computer ,
742
785
tool_call : ResponseComputerToolCall ,
743
786
) -> str :
787
+ """Get a screenshot synchronously."""
744
788
action = tool_call .action
745
- if isinstance (action , ActionClick ):
746
- computer .click (action .x , action .y , action .button )
747
- elif isinstance (action , ActionDoubleClick ):
748
- computer .double_click (action .x , action .y )
749
- elif isinstance (action , ActionDrag ):
750
- computer .drag ([(p .x , p .y ) for p in action .path ])
751
- elif isinstance (action , ActionKeypress ):
752
- computer .keypress (action .keys )
753
- elif isinstance (action , ActionMove ):
754
- computer .move (action .x , action .y )
755
- elif isinstance (action , ActionScreenshot ):
756
- computer .screenshot ()
757
- elif isinstance (action , ActionScroll ):
758
- computer .scroll (action .x , action .y , action .scroll_x , action .scroll_y )
759
- elif isinstance (action , ActionType ):
760
- computer .type (action .text )
761
- elif isinstance (action , ActionWait ):
762
- computer .wait ()
789
+ try :
790
+ if isinstance (action , ActionClick ):
791
+ computer .click (action .x , action .y , action .button )
792
+ elif isinstance (action , ActionDoubleClick ):
793
+ computer .double_click (action .x , action .y )
794
+ elif isinstance (action , ActionDrag ):
795
+ computer .drag ([(p .x , p .y ) for p in action .path ])
796
+ elif isinstance (action , ActionKeypress ):
797
+ computer .keypress (action .keys )
798
+ elif isinstance (action , ActionMove ):
799
+ computer .move (action .x , action .y )
800
+ elif isinstance (action , ActionScreenshot ):
801
+ computer .screenshot ()
802
+ elif isinstance (action , ActionScroll ):
803
+ computer .scroll (action .x , action .y , action .scroll_x , action .scroll_y )
804
+ elif isinstance (action , ActionType ):
805
+ computer .type (action .text )
806
+ elif isinstance (action , ActionWait ):
807
+ computer .wait ()
808
+ except Exception as e :
809
+ raise ModelBehaviorError (f"Error executing computer action: { e } " )
763
810
764
811
return computer .screenshot ()
765
812
@@ -769,24 +816,28 @@ async def _get_screenshot_async(
769
816
computer : AsyncComputer ,
770
817
tool_call : ResponseComputerToolCall ,
771
818
) -> str :
819
+ """Get a screenshot asynchronously."""
772
820
action = tool_call .action
773
- if isinstance (action , ActionClick ):
774
- await computer .click (action .x , action .y , action .button )
775
- elif isinstance (action , ActionDoubleClick ):
776
- await computer .double_click (action .x , action .y )
777
- elif isinstance (action , ActionDrag ):
778
- await computer .drag ([(p .x , p .y ) for p in action .path ])
779
- elif isinstance (action , ActionKeypress ):
780
- await computer .keypress (action .keys )
781
- elif isinstance (action , ActionMove ):
782
- await computer .move (action .x , action .y )
783
- elif isinstance (action , ActionScreenshot ):
784
- await computer .screenshot ()
785
- elif isinstance (action , ActionScroll ):
786
- await computer .scroll (action .x , action .y , action .scroll_x , action .scroll_y )
787
- elif isinstance (action , ActionType ):
788
- await computer .type (action .text )
789
- elif isinstance (action , ActionWait ):
790
- await computer .wait ()
821
+ try :
822
+ if isinstance (action , ActionClick ):
823
+ await computer .click (action .x , action .y , action .button )
824
+ elif isinstance (action , ActionDoubleClick ):
825
+ await computer .double_click (action .x , action .y )
826
+ elif isinstance (action , ActionDrag ):
827
+ await computer .drag ([(p .x , p .y ) for p in action .path ])
828
+ elif isinstance (action , ActionKeypress ):
829
+ await computer .keypress (action .keys )
830
+ elif isinstance (action , ActionMove ):
831
+ await computer .move (action .x , action .y )
832
+ elif isinstance (action , ActionScreenshot ):
833
+ await computer .screenshot ()
834
+ elif isinstance (action , ActionScroll ):
835
+ await computer .scroll (action .x , action .y , action .scroll_x , action .scroll_y )
836
+ elif isinstance (action , ActionType ):
837
+ await computer .type (action .text )
838
+ elif isinstance (action , ActionWait ):
839
+ await computer .wait ()
840
+ except Exception as e :
841
+ raise ModelBehaviorError (f"Error executing computer action: { e } " )
791
842
792
843
return await computer .screenshot ()
0 commit comments