import unittest
import os
import sys

# Ensure the parent directory is in the system path to find other modules
sys.path.append(os.path.abspath(os.path.join(os.path.dirname(__file__), '..')))

from utility_benchmark.tool_calling_agents import ToolAgent

class TestToolAgent(unittest.TestCase):
    def setUp(self):
        """Set up the agent for testing."""
        self.tools = [
            "notebook", "flights", "attractions", "accommodations",
            "restaurants", "googleDistanceMatrix", "planner", "cities"
        ]
        self.agent = ToolAgent(model_name="gpt-4o", tools_list=self.tools)

    def test_01_load_tools(self):
        """Test that all tools are loaded correctly into the tools_map."""
        # print("\n--- Running test_01_load_tools ---")
        loaded_tools = self.agent.tools_map.keys()
        # print(f"Loaded tools: {list(loaded_tools)}")
        self.assertEqual(len(loaded_tools), len(self.tools), "Should load all specified tools.")
        for tool in self.tools:
            self.assertIn(tool, self.agent.tools_map)
            self.assertIsNotNone(self.agent.tools_map[tool], f"Tool '{tool}' should not be None.")

    def test_02_execute_attraction_search(self):
        """Test executing the AttractionSearch tool to verify lookup and execution."""
        # print("\n--- Running test_02_execute_attraction_search ---")
        tool_name = "AttractionSearch"
        # Using a city present in the database to get a valid response
        tool_args = {"city": "Rockford"}
        result = self.agent._execute_tool(tool_name, tool_args)
        
        # print(f"Execution result for {tool_name}: {result}")
        self.assertIsInstance(result, str)
        self.assertNotIn("Error:", result, "Execution should not return an error message.")
        # Check for expected content if possible
        self.assertIn("Name", result, "Result should contain a 'Name' column for attractions.")

    def test_03_execute_flight_search(self):
        """Test executing the FlightSearch tool."""
        # print("\n--- Running test_03_execute_flight_search ---")
        tool_name = "FlightSearch"
        tool_args = {"origin": "St. Petersburg", "destination": "Rockford", "departure_date": "2022-03-16"}
        result = self.agent._execute_tool(tool_name, tool_args)

        # print(f"Execution result for {tool_name}: {result}")
        self.assertIsInstance(result, str)
        self.assertNotIn("Error:", result, "Execution should not return an error message.")
        self.assertIn("Flight Number", result, "Result should contain a 'Flight Number' column for flights.")

    def test_04_execute_accommodation_search(self):
        """Test executing the AccommodationSearch tool."""
        # print("\n--- Running test_04_execute_accommodation_search ---")
        tool_name = "AccommodationSearch"
        tool_args = {"city": "Rockford"}
        result = self.agent._execute_tool(tool_name, tool_args)

        # print(f"Execution result for {tool_name}: {result}")
        self.assertIsInstance(result, str)
        self.assertNotIn("Error:", result, "Execution should not return an error message.")
        self.assertIn("NAME", result, "Result should contain a 'NAME' column for accommodations.")

    def test_05_execute_restaurant_search(self):
        """Test executing the RestaurantSearch tool."""
        # print("\n--- Running test_05_execute_restaurant_search ---")
        tool_name = "RestaurantSearch"
        tool_args = {"city": "Rockford"}
        result = self.agent._execute_tool(tool_name, tool_args)

        # print(f"Execution result for {tool_name}: {result}")
        self.assertIsInstance(result, str)
        self.assertNotIn("Error:", result, "Execution should not return an error message.")
        self.assertIn("Name", result, "Result should contain a 'Name' column for restaurants.")

    def test_06_execute_google_distance_matrix(self):
        """Test executing the GoogleDistanceMatrix tool."""
        # print("\n--- Running test_06_execute_google_distance_matrix ---")
        tool_name = "GoogleDistanceMatrix"
        tool_args = {"origin": "Chicago", "destination": "Raleigh", "mode": "self-driving"}
        result = self.agent._execute_tool(tool_name, tool_args)

        # print(f"Execution result for {tool_name}: {result}")
        self.assertIsInstance(result, str)
        self.assertNotIn("Error:", result, "Execution should not return an error message.")
        self.assertIn("duration", result, "Result should contain 'duration'.")
        self.assertIn("distance", result, "Result should contain 'distance'.")

    def test_07_execute_city_search(self):
        """Test executing the CitySearch tool."""
        # print("\n--- Running test_07_execute_city_search ---")
        tool_name = "CitySearch"
        # Rockford is in Illinois, so this should be a valid state
        tool_args = {"state": "Illinois"}
        result = self.agent._execute_tool(tool_name, tool_args)

        # print(f"Execution result for {tool_name}: {result}")
        self.assertIsInstance(result, str)
        self.assertNotIn("Error:", result, "Execution should not return an error message.")
        self.assertIn("Rockford", result, "Result for Illinois should contain 'Rockford'.")

    def test_08_execute_notebook_write(self):
        """Test executing the NotebookWrite tool."""
        # print("\n--- Running test_08_execute_notebook_write ---")
        tool_name = "NotebookWrite"
        test_data = "This is a test note."
        test_desc = "test_note"
        tool_args = {"data": test_data, "description": test_desc}
        
        # Execute the tool
        result = self.agent._execute_tool(tool_name, tool_args)
        
        # print(f"Execution result for {tool_name}: {result}")
        self.assertIn("The information has been recorded", result)
        
        # Verify the content was actually written
        notebook_content = self.agent.notebook.list_all()
        self.assertEqual(len(notebook_content), 1, "Notebook should have one entry.")
        self.assertEqual(notebook_content[0]['Short Description'], test_desc)
        self.assertEqual(notebook_content[0]['Content'], test_data)

    def test_09_execute_planner(self):
        """Test executing the Planner tool."""
        # print("\n--- Running test_09_execute_planner ---")
        
        # First, add some information to the notebook for the planner to use
        self.agent.notebook.write("Flight data here", "flights_to_rockford")
        
        tool_name = "Planner"
        tool_args = {"query": "Please plan a trip to Rockford."}
        result = self.agent._execute_tool(tool_name, tool_args)

        # print(f"Execution result for {tool_name}: {result}")
        self.assertIsInstance(result, str)
        self.assertNotIn("Error:", result, "Execution should not return an error message.")

    def test_10_execute_scratchpad(self):
        """Test executing the Scratchpad tool."""
        # print("\n--- Running test_10_execute_scratchpad ---")
        tool_name = "Scratchpad"
        note_content = "This is a test thought."
        tool_args = {"note": note_content}
        result = self.agent._execute_tool(tool_name, tool_args)

        # print(f"Execution result for {tool_name}: {result}")
        self.assertEqual(result, "Note taken.")

if __name__ == '__main__':
    # To run tests from the command line: python agents/test_tool_calling_agents.py
    unittest.main() 