Spaces:
Sleeping
Sleeping
| """Judge system for evaluating browser-use agent execution traces.""" | |
| import base64 | |
| import logging | |
| from pathlib import Path | |
| from browser_use.llm.messages import ( | |
| BaseMessage, | |
| ContentPartImageParam, | |
| ContentPartTextParam, | |
| ImageURL, | |
| SystemMessage, | |
| UserMessage, | |
| ) | |
| logger = logging.getLogger(__name__) | |
| def _encode_image(image_path: str) -> str | None: | |
| """Encode image to base64 string.""" | |
| try: | |
| path = Path(image_path) | |
| if not path.exists(): | |
| return None | |
| with open(path, 'rb') as f: | |
| return base64.b64encode(f.read()).decode('utf-8') | |
| except Exception as e: | |
| logger.warning(f'Failed to encode image {image_path}: {e}') | |
| return None | |
| def _truncate_text(text: str, max_length: int, from_beginning: bool = False) -> str: | |
| """Truncate text to maximum length with eval system indicator.""" | |
| if len(text) <= max_length: | |
| return text | |
| if from_beginning: | |
| return '...[text truncated]' + text[-max_length + 23 :] | |
| else: | |
| return text[: max_length - 23] + '...[text truncated]...' | |
| def construct_judge_messages( | |
| task: str, | |
| final_result: str, | |
| agent_steps: list[str], | |
| screenshot_paths: list[str], | |
| max_images: int = 10, | |
| ) -> list[BaseMessage]: | |
| """ | |
| Construct messages for judge evaluation of agent trace. | |
| Args: | |
| task: The original task description | |
| final_result: The final result returned to the user | |
| agent_steps: List of formatted agent step descriptions | |
| screenshot_paths: List of screenshot file paths | |
| max_images: Maximum number of screenshots to include | |
| Returns: | |
| List of messages for LLM judge evaluation | |
| """ | |
| task_truncated = _truncate_text(task, 40000) | |
| final_result_truncated = _truncate_text(final_result, 40000) | |
| steps_text = '\n'.join(agent_steps) | |
| steps_text_truncated = _truncate_text(steps_text, 40000) | |
| # Select last N screenshots | |
| selected_screenshots = screenshot_paths[-max_images:] if len(screenshot_paths) > max_images else screenshot_paths | |
| # Encode screenshots | |
| encoded_images: list[ContentPartImageParam] = [] | |
| for img_path in selected_screenshots: | |
| encoded = _encode_image(img_path) | |
| if encoded: | |
| encoded_images.append( | |
| ContentPartImageParam( | |
| image_url=ImageURL( | |
| url=f'data:image/png;base64,{encoded}', | |
| media_type='image/png', | |
| ) | |
| ) | |
| ) | |
| # System prompt for judge | |
| system_prompt = """You are an expert judge evaluating browser automation agent performance. | |
| <evaluation_framework> | |
| **PRIMARY EVALUATION CRITERIA (in order of importance):** | |
| 1. **Task Satisfaction (Most Important)**: Did the agent accomplish what the user asked for? Break down the task into the key criteria and evaluate if the agent all of them. Focus on user intent and final outcome. | |
| 2. **Output Quality**: Is the final result in the correct format and complete? Does it match exactly what was requested? | |
| 3. **Tool Effectiveness**: Did the browser interactions work as expected? Were tools used appropriately? How many % of the tools failed? | |
| 4. **Agent Reasoning**: Quality of decision-making, planning, and problem-solving throughout the trajectory. | |
| 5. **Browser Handling**: Navigation stability, error recovery, and technical execution. If the browser crashes, does not load or a captcha blocks the task, the score must be very low. | |
| **VERDICT GUIDELINES:** | |
| - true: Task completed as requested, human-like execution, all of the users criteria were met and the agent did not make up any information. | |
| - false: Task not completed, or only partially completed. | |
| **Examples of task completion verdict:** | |
| - If task asks for 10 items and agent finds 4 items correctly: false | |
| - If task completed to full user requirements but with some errors to improve in the trajectory: true | |
| - If task impossible due to captcha/login requirements: false | |
| - If the trajectory is ideal and the output is perfect: true | |
| - If the task asks to search all headphones in amazon under $100 but the agent searches all headphones and the lowest price is $150: false | |
| - If the task asks to research a property and create a google doc with the result but the agents only returns the results in text: false | |
| - If the task asks to complete an action on the page, and the agent reports that the action is completed but the screenshot or page shows the action is not actually complete: false | |
| - If the task asks to use a certain tool or site to complete the task but the agent completes the task without using it: false | |
| - If the task asks to look for a section of a page that does not exist: false | |
| - If the agent concludes the task is impossible but it is not: false | |
| - If the agent concludes the task is impossible and it truly is impossible: false | |
| - If the agent is unable to complete the task because no login information was provided and it is truly needed to complete the task: false | |
| **FAILURE CONDITIONS (automatically set verdict to false):** | |
| - Blocked by captcha or missing authentication | |
| - Output format completely wrong or missing | |
| - Infinite loops or severe technical failures | |
| - Critical user requirements ignored | |
| - Page not loaded | |
| - Browser crashed | |
| - Agent could not interact with required UI elements | |
| - The agent moved on from a important step in the task without completing it | |
| - The agent made up content that is not in the screenshot or the page state | |
| - The agent calls done action before completing all key points of the task | |
| **IMPORTANT EVALUATION NOTES:** | |
| - **evaluate for action** - For each key step of the trace, double check whether the action that the agent tried to performed actually happened. If the required action did not actually occur, the verdict should be false. | |
| - **screenshot is not entire content** - The agent has the entire DOM content, but the screenshot is only part of the content. If the agent extracts information from the page, but you do not see it in the screenshot, you can assume this information is there. | |
| - **Penalize poor tool usage** - Wrong tools, inefficient approaches, ignoring available information. | |
| - **ignore unexpected dates and times** - These agent traces are from varying dates, you can assume the dates the agent uses for search or filtering are correct. | |
| - **IMPORTANT**: be very picky about the user's request - Have very high standard for the agent completing the task exactly to the user's request. | |
| - **IMPORTANT**: be initially doubtful of the agent's self reported success, be sure to verify that its methods are valid and fulfill the user's desires to a tee. | |
| </evaluation_framework> | |
| <response_format> | |
| Respond with EXACTLY this JSON structure (no additional text before or after): | |
| {{ | |
| "reasoning": "Breakdown of user task into key points. Detailed analysis covering: what went well, what didn't work, trajectory quality assessment, tool usage evaluation, output quality review, and overall user satisfaction prediction", | |
| "verdict": true or false, | |
| "failure_reason": "If verdict is false, provide the key reason why the task was not completed successfully. If verdict is true, use an empty string." | |
| }} | |
| </response_format> | |
| """ | |
| user_prompt = f""" | |
| <task> | |
| {task_truncated or 'No task provided'} | |
| </task> | |
| <agent_trajectory> | |
| {steps_text_truncated or 'No agent trajectory provided'} | |
| </agent_trajectory> | |
| <final_result> | |
| {final_result_truncated or 'No final result provided'} | |
| </final_result> | |
| {len(encoded_images)} screenshots from execution are attached. | |
| Evaluate this agent execution given the criteria and respond with the exact JSON structure requested.""" | |
| # Build messages with screenshots | |
| content_parts: list[ContentPartTextParam | ContentPartImageParam] = [ContentPartTextParam(text=user_prompt)] | |
| content_parts.extend(encoded_images) | |
| return [ | |
| SystemMessage(content=system_prompt), | |
| UserMessage(content=content_parts), | |
| ] | |