import sys
import os

sys.path.append(os.path.abspath(os.path.join(os.path.dirname(__file__), "../")))
import unittest

from StateActionTracker import StateActionStack


class TestStateActionTrack(unittest.TestCase):
    @classmethod
    def setUpClass(cls):
        cls.dfs_tracker = StateActionStack()

    def test_push_state(self):
        state = [0, 2, 6, 4, 5, 2, 5]
        actions = {0, 1, 2, 3}
        self.dfs_tracker.push_state(state, actions)
        self.assertIn(str(state), self.dfs_tracker.stack)
        self.assertEqual(self.dfs_tracker.unexplored[str(state)], actions)

    def test_set_fixing_action(self):
        self.dfs_tracker.set_fixing_action([0, 2, 6, 4, 5, 2, 5], 0)
        self.assertEqual(self.dfs_tracker.get_fixing_action([0, 2, 6, 4, 5, 2, 5]), 0)
        self.dfs_tracker.print_stack_info()

    def test_find_latest_key(self):
        self.dfs_tracker.push_state([1, 2, 3], {0, 1})
        self.dfs_tracker.push_state([2, 3, 4], {0, 1})
        latest_added_key = self.dfs_tracker.find_latest_key([[1, 2, 3], [2, 3, 4]])
        self.assertEqual(latest_added_key, str([2, 3, 4]))

    def test_push_again(self):
        # Push again without change
        self.dfs_tracker.push_state([1, 1, 1], {0, 1})
        self.dfs_tracker.pop_state()
        self.assertFalse(self.dfs_tracker.should_push([1, 1, 1]))

        # Push again with change
        self.dfs_tracker.push_state([1, 1, 1], {0, 1})
        self.dfs_tracker.pop_state()
        self.dfs_tracker.push_state([2, 2, 2], {0, 1})
        self.assertTrue(self.dfs_tracker.should_push([1, 1, 1]))

        # Push - pop - push different
        self.dfs_tracker.pop_specific_key([1, 1, 1])
        self.dfs_tracker.push_state([1, 1, 1], {0, 1})
        self.dfs_tracker.pop_state()
        self.dfs_tracker.push_state([3, 3, 3], {0, 1})
        self.assertTrue(self.dfs_tracker.should_push([3, 3, 3]))

    def test_load_from_stack_info(self):
        info = """
        [STACK INFO] Current State Stack (Top -> Bottom):
          - State [2, 2, 4, 4, 2]
            - Unexplored Actions: {0, 1}
            - Explored Actions: set()
            - Fixing Action: 0
          - State [2, 4, 3, 4, 4]
            - Unexplored Actions: {0, 1}
            - Explored Actions: set()
            - Fixing Action: 1
          - State [2, 2, 4, 2, 4]
            - Unexplored Actions: {0, 1}
            - Explored Actions: set()
            - Fixing Action: 0    
        [END OF STACK INFO]
        """
        self.dfs_tracker.load_from_stack_info(info)
        self.dfs_tracker.print_stack_info()


if __name__ == "__main__":
    suite = unittest.TestSuite()
    suite.addTest(TestStateActionTrack("test_push_state"))
    suite.addTest(TestStateActionTrack("test_set_fixing_action"))
    suite.addTest(TestStateActionTrack("test_find_latest_key"))
    suite.addTest(TestStateActionTrack("test_push_again"))
    suite.addTest(TestStateActionTrack("test_load_from_stack_info"))

    runner = unittest.TextTestRunner(verbosity=2)
    runner.run(suite)
