Spaces:
Build error
Build error
Update agent.py
Browse files
agent.py
CHANGED
|
@@ -81,7 +81,9 @@ def calculator(inputs: dict):
|
|
| 81 |
return a % b
|
| 82 |
else:
|
| 83 |
return "Unknown operation"
|
| 84 |
-
|
|
|
|
|
|
|
| 85 |
|
| 86 |
@tool
|
| 87 |
def wiki_search(query: str) -> str:
|
|
@@ -282,12 +284,16 @@ enabled_tool_names = [
|
|
| 282 |
|
| 283 |
tools = [tool_map[name] for name in enabled_tool_names]
|
| 284 |
|
|
|
|
|
|
|
| 285 |
tools = []
|
| 286 |
for name in enabled_tool_names:
|
| 287 |
-
if name in tool_map:
|
| 288 |
-
|
| 289 |
-
|
| 290 |
-
|
|
|
|
|
|
|
| 291 |
|
| 292 |
# -------------------------------
|
| 293 |
# Step 2: Load the JSON file or tasks (Replace this part if you're loading tasks dynamically)
|
|
@@ -503,57 +509,61 @@ def get_llm(provider: str, config: dict):
|
|
| 503 |
|
| 504 |
|
| 505 |
|
| 506 |
-
|
| 507 |
# ----------------------------------------------------------------
|
| 508 |
# Planning & Execution Logic
|
| 509 |
# ----------------------------------------------------------------
|
| 510 |
-
def planner(question: str, tools: list) ->
|
|
|
|
|
|
|
|
|
|
|
|
|
| 511 |
question = question.lower().strip()
|
| 512 |
|
| 513 |
-
# Define intent-based keywords
|
| 514 |
intent_keywords = {
|
| 515 |
"math": ["calculate", "evaluate", "add", "subtract", "multiply", "divide", "modulus", "plus", "minus", "times"],
|
| 516 |
"wiki_search": ["who is", "what is", "define", "explain", "tell me about", "overview of"],
|
| 517 |
"web_search": ["search", "find", "look up", "google", "latest news", "current info"],
|
| 518 |
-
"
|
| 519 |
-
"
|
| 520 |
"video_analysis": ["analyze video", "summarize video", "video content"],
|
| 521 |
"data_analysis": ["analyze", "plot", "graph", "data", "visualize"],
|
| 522 |
"wikidata_query": ["wikidata", "sparql", "run sparql", "query wikidata"],
|
| 523 |
-
"
|
| 524 |
}
|
| 525 |
|
| 526 |
-
|
| 527 |
-
|
| 528 |
-
# Try to find tools whose description matches intent
|
| 529 |
for intent, keywords in intent_keywords.items():
|
| 530 |
if any(keyword in question for keyword in keywords):
|
| 531 |
-
|
| 532 |
-
|
| 533 |
-
name = getattr(tool, "name", "").lower()
|
| 534 |
-
|
| 535 |
-
# Match based on intent keywords appearing in tool description or name
|
| 536 |
-
if intent in description or intent in name:
|
| 537 |
-
matched_tools.append(tool)
|
| 538 |
-
|
| 539 |
-
# Break after first matching intent — you can remove this if you want to allow multi-intent matching
|
| 540 |
-
if matched_tools:
|
| 541 |
-
break
|
| 542 |
|
| 543 |
-
#
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 544 |
if not matched_tools:
|
| 545 |
matched_tools = [
|
| 546 |
-
tool for tool in tools
|
| 547 |
-
if "default" in getattr(tool, "name", "").lower()
|
| 548 |
or "qa" in getattr(tool, "description", "").lower()
|
| 549 |
]
|
| 550 |
|
| 551 |
-
|
| 552 |
-
return matched_tools if matched_tools else [tools[0]]
|
| 553 |
|
| 554 |
|
| 555 |
|
| 556 |
def task_classifier(question: str) -> str:
|
|
|
|
|
|
|
|
|
|
| 557 |
question = question.lower().strip()
|
| 558 |
|
| 559 |
# Context-aware intent patterns
|
|
@@ -599,6 +609,44 @@ def task_classifier(question: str) -> str:
|
|
| 599 |
|
| 600 |
return "default"
|
| 601 |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 602 |
|
| 603 |
# Function to extract math operation from the question
|
| 604 |
|
|
@@ -640,18 +688,31 @@ def extract_math_from_question(question: str):
|
|
| 640 |
|
| 641 |
# Example tool set (adjust these to match your actual tool names)
|
| 642 |
tools = {
|
| 643 |
-
"math":
|
| 644 |
-
"wiki_search": wiki_tool,
|
| 645 |
-
"
|
| 646 |
-
"
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 647 |
}
|
| 648 |
|
|
|
|
| 649 |
# The task order can also include the tools for each task
|
| 650 |
priority_order = [
|
| 651 |
-
{"task": "math", "tool": "math"},
|
| 652 |
{"task": "wiki_search", "tool": "wiki_search"},
|
|
|
|
|
|
|
|
|
|
| 653 |
{"task": "retriever", "tool": "retriever"},
|
| 654 |
-
{"task": "
|
|
|
|
|
|
|
|
|
|
| 655 |
]
|
| 656 |
|
| 657 |
def decide_task(state: dict) -> str:
|
|
|
|
| 81 |
return a % b
|
| 82 |
else:
|
| 83 |
return "Unknown operation"
|
| 84 |
+
|
| 85 |
+
|
| 86 |
+
|
| 87 |
|
| 88 |
@tool
|
| 89 |
def wiki_search(query: str) -> str:
|
|
|
|
| 284 |
|
| 285 |
tools = [tool_map[name] for name in enabled_tool_names]
|
| 286 |
|
| 287 |
+
|
| 288 |
+
# Safe version
|
| 289 |
tools = []
|
| 290 |
for name in enabled_tool_names:
|
| 291 |
+
if name not in tool_map:
|
| 292 |
+
print(f"❌ Tool not found: {name}")
|
| 293 |
+
continue
|
| 294 |
+
tools.append(tool_map[name])
|
| 295 |
+
|
| 296 |
+
|
| 297 |
|
| 298 |
# -------------------------------
|
| 299 |
# Step 2: Load the JSON file or tasks (Replace this part if you're loading tasks dynamically)
|
|
|
|
| 509 |
|
| 510 |
|
| 511 |
|
|
|
|
| 512 |
# ----------------------------------------------------------------
|
| 513 |
# Planning & Execution Logic
|
| 514 |
# ----------------------------------------------------------------
|
| 515 |
+
def planner(question: str, tools: list) -> tuple:
|
| 516 |
+
"""
|
| 517 |
+
Select the best-matching tool(s) for a question based on keyword-based intent detection and tool metadata.
|
| 518 |
+
Returns the detected intent and matched tools.
|
| 519 |
+
"""
|
| 520 |
question = question.lower().strip()
|
| 521 |
|
| 522 |
+
# Define intent-based keywords
|
| 523 |
intent_keywords = {
|
| 524 |
"math": ["calculate", "evaluate", "add", "subtract", "multiply", "divide", "modulus", "plus", "minus", "times"],
|
| 525 |
"wiki_search": ["who is", "what is", "define", "explain", "tell me about", "overview of"],
|
| 526 |
"web_search": ["search", "find", "look up", "google", "latest news", "current info"],
|
| 527 |
+
"arxiv_search": ["arxiv", "research paper", "scientific paper", "preprint"],
|
| 528 |
+
"get_youtube_transcript": ["youtube", "watch", "play video", "show me a video"],
|
| 529 |
"video_analysis": ["analyze video", "summarize video", "video content"],
|
| 530 |
"data_analysis": ["analyze", "plot", "graph", "data", "visualize"],
|
| 531 |
"wikidata_query": ["wikidata", "sparql", "run sparql", "query wikidata"],
|
| 532 |
+
"default": ["why", "how", "difference between", "compare", "what happens", "reason for", "cause of", "effect of"]
|
| 533 |
}
|
| 534 |
|
| 535 |
+
# Step 1: Identify intent
|
| 536 |
+
detected_intent = None
|
|
|
|
| 537 |
for intent, keywords in intent_keywords.items():
|
| 538 |
if any(keyword in question for keyword in keywords):
|
| 539 |
+
detected_intent = intent
|
| 540 |
+
break
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 541 |
|
| 542 |
+
# Step 2: Match tools by intent
|
| 543 |
+
matched_tools = []
|
| 544 |
+
if detected_intent:
|
| 545 |
+
for tool in tools:
|
| 546 |
+
name = getattr(tool, "name", "").lower()
|
| 547 |
+
description = getattr(tool, "description", "").lower()
|
| 548 |
+
if detected_intent in name or detected_intent in description:
|
| 549 |
+
matched_tools.append(tool)
|
| 550 |
+
|
| 551 |
+
# Step 3: Fallback to general-purpose/default tools if no match found
|
| 552 |
if not matched_tools:
|
| 553 |
matched_tools = [
|
| 554 |
+
tool for tool in tools
|
| 555 |
+
if "default" in getattr(tool, "name", "").lower()
|
| 556 |
or "qa" in getattr(tool, "description", "").lower()
|
| 557 |
]
|
| 558 |
|
| 559 |
+
return detected_intent, matched_tools if matched_tools else [tools[0]]
|
|
|
|
| 560 |
|
| 561 |
|
| 562 |
|
| 563 |
def task_classifier(question: str) -> str:
|
| 564 |
+
"""
|
| 565 |
+
Classifies the question into one of the predefined task categories.
|
| 566 |
+
"""
|
| 567 |
question = question.lower().strip()
|
| 568 |
|
| 569 |
# Context-aware intent patterns
|
|
|
|
| 609 |
|
| 610 |
return "default"
|
| 611 |
|
| 612 |
+
def select_tool_and_run(question: str, tools: list) -> any:
|
| 613 |
+
"""
|
| 614 |
+
Select the best tool based on question intent and run it.
|
| 615 |
+
"""
|
| 616 |
+
# Step 1: Use task_classifier to identify the intent of the question
|
| 617 |
+
intent = task_classifier(question)
|
| 618 |
+
|
| 619 |
+
# Step 2: Use the intent_tool_map to map the intent to the tool
|
| 620 |
+
intent_tool_map = {
|
| 621 |
+
"math": "calculator",
|
| 622 |
+
"wiki_search": "wiki_tool",
|
| 623 |
+
"web_search": "web_search_tool",
|
| 624 |
+
"arxiv": "arxiv_tool",
|
| 625 |
+
"get_youtube_transcript": "youtube_tool",
|
| 626 |
+
"video_analysis": "video_tool",
|
| 627 |
+
"analyze_tool": "analyze_tool",
|
| 628 |
+
"wikidata_query": "wikidata_query",
|
| 629 |
+
"default": "default_tool"
|
| 630 |
+
}
|
| 631 |
+
|
| 632 |
+
tool_name = intent_tool_map.get(intent, "default_tool")
|
| 633 |
+
|
| 634 |
+
# Step 3: Get the selected tool
|
| 635 |
+
tool_func = tools.get(tool_name)
|
| 636 |
+
|
| 637 |
+
if not tool_func:
|
| 638 |
+
print(f"No matching tool found for {intent}")
|
| 639 |
+
return None
|
| 640 |
+
|
| 641 |
+
# Step 4: Call the tool function and return the result
|
| 642 |
+
return tool_func.run(question) # or tool_func(question), depending on how your tools are structured
|
| 643 |
+
|
| 644 |
+
# Run the function to select and execute the tool
|
| 645 |
+
result = select_tool_and_run(question, tools)
|
| 646 |
+
|
| 647 |
+
# Output the result
|
| 648 |
+
print(result)
|
| 649 |
+
|
| 650 |
|
| 651 |
# Function to extract math operation from the question
|
| 652 |
|
|
|
|
| 688 |
|
| 689 |
# Example tool set (adjust these to match your actual tool names)
|
| 690 |
tools = {
|
| 691 |
+
"math": calculator,
|
| 692 |
+
"wiki_search": wiki_tool,
|
| 693 |
+
"web_search": web_search_tool,
|
| 694 |
+
"arvix_search": arvix_tool,
|
| 695 |
+
"retriever": retriever_tool,
|
| 696 |
+
"get_youtube_transcript": youtube_tool,
|
| 697 |
+
"extract_video_id": video_tool,
|
| 698 |
+
"analyze_attachment": analyze_tool,
|
| 699 |
+
"wikidata_query": wikiq_tool,
|
| 700 |
+
"default": default_tool
|
| 701 |
}
|
| 702 |
|
| 703 |
+
|
| 704 |
# The task order can also include the tools for each task
|
| 705 |
priority_order = [
|
| 706 |
+
{"task": "math", "tool": "math"},
|
| 707 |
{"task": "wiki_search", "tool": "wiki_search"},
|
| 708 |
+
{"task": "web_search", "tool": "web_search"},
|
| 709 |
+
{"task": "arvix_search", "tool": "arvix_search"},
|
| 710 |
+
{"task": "wikidata_query", "tool": "wikidata_query"},
|
| 711 |
{"task": "retriever", "tool": "retriever"},
|
| 712 |
+
{"task": "get_youtube_transcript", "tool": "get_youtube_transcript"},
|
| 713 |
+
{"task": "extract_video_id", "tool": "extract_video_id"},
|
| 714 |
+
{"task": "analyze_attachment", "tool": "analyze_attachment"},
|
| 715 |
+
{"task": "default", "tool": "default"} # Fallback
|
| 716 |
]
|
| 717 |
|
| 718 |
def decide_task(state: dict) -> str:
|