import os
import sys
project_root = os.path.dirname(
    os.path.dirname(os.path.abspath(__file__)))
sys.path.insert(0, project_root)

from synthesizer.fact import AttributeFact, RelationFact
from synthesizer.rule import Rule
from synthesizer.expression import ConstantExpression, IdentityExpression, LinearExpression, BinaryExpression
import copy


class ConflictError(Exception):
    """Custom exception to indicate a conflict in triggering rules or facts."""

    def __init__(self, message):
        super().__init__(message)
        self.message = message


def trigger_all_rules(facts, rules, values, relations, debug_info=None):
    """
    Trigger all rules based on the given facts, values, and relations.
    If there is conflict, it will return None.
    If there is no conflict, it will return triggered (values, relations).
    """
    values = copy.deepcopy(values)
    relations = copy.deepcopy(relations)
    for fact in facts:
        try:
            trigger_fact(fact, values, relations)
        except ConflictError as e:
            print(f"Conflict while triggering fact {fact}: {e}")
            # if debug_info is not None:
            #     print(f"Debug info:\n\n{debug_info}")
            #     exit(1)
            return None
    while True:
        triggered = False
        for rule in rules:
            try:
                triggered = trigger_rule(rule, values, relations) or triggered
            except ConflictError as e:
                print(f"Conflict while triggering rule {rule}: {e}")
                # if debug_info is not None:
                #     print(f"Debug info:\n\n{debug_info}")
                #     exit(1)
                return None
        if not triggered:
            break
    return values, relations


def trigger_rule(rule, values, relations):
    """
    Trigger a single rule based on the given values and relations and,
    Update values and relations in this function.
    If there is conflict, it will raise an ConflictError.
    If an update is made, it will return True.
    If no update is made, it will return False.
    """
    def solve():
        def validate_expression_dependency(expression, solution):
            if isinstance(expression, ConstantExpression):
                return True
            elif isinstance(expression, IdentityExpression):
                return values[solution[expression.entity]][expression.attribute] is not None
            elif isinstance(expression, LinearExpression):
                return values[solution[expression.entity]][expression.attribute] is not None
            elif isinstance(expression, BinaryExpression):
                return (validate_expression_dependency(expression.expr1, solution) and
                        validate_expression_dependency(expression.expr2, solution))
            else:
                raise ValueError(
                    f"Unknown expression type: {type(expression)}")
        entity_list = set()
        partial_solutions = []
        for condition in rule.conditions:
            if isinstance(condition, AttributeFact):
                entity_list.add(condition.entity)
                partial_solution = []
                assert isinstance(condition.expression, ConstantExpression), \
                    f"Expected ConstantExpression, got {type(condition.expression)}"
                for real_entity in values.keys():
                    if values[real_entity][condition.attribute] == condition.expression.value:
                        partial_solution.append(
                            {condition.entity: real_entity})
                partial_solutions.append(partial_solution)
            elif isinstance(condition, RelationFact):
                entity_list.add(condition.entity1)
                entity_list.add(condition.entity2)
                partial_solution = []
                for real_entity1 in relations.keys():
                    for real_entity2 in relations[real_entity1].keys():
                        if condition.relation in relations[real_entity1][real_entity2]:
                            partial_solution.append(
                                {condition.entity1: real_entity1, condition.entity2: real_entity2})
                partial_solutions.append(partial_solution)
            else:
                raise ValueError(f"Unknown condition type: {type(condition)}")
        entity_list = list(entity_list)
        full_solutions = []
        for conditional_partial_solutions in partial_solutions:
            if not full_solutions:
                full_solutions = conditional_partial_solutions
            else:
                new_full_solutions = []
                for full_solution in full_solutions:
                    for partial_solution in conditional_partial_solutions:
                        new_solution = copy.deepcopy(full_solution)
                        conflict = False
                        for partial_entity in partial_solution:
                            if partial_entity in new_solution and new_solution[partial_entity] != partial_solution[partial_entity]:
                                conflict = True
                                break
                        new_solution.update(partial_solution)
                        if not conflict:
                            new_full_solutions.append(new_solution)
                full_solutions = new_full_solutions
            if not full_solutions:
                break
        for full_solution in full_solutions:
            for entity in entity_list:
                assert entity in full_solution, \
                    f"Entity {entity} not found in full solution: {full_solution}"
        if isinstance(rule.conclusion, AttributeFact):
            full_solutions = list(
                filter(
                    lambda solution: validate_expression_dependency(
                        rule.conclusion.expression, solution),
                    full_solutions
                )
            )
        return full_solutions

    def apply(solution):
        """
        Return True if a new update is made, False otherwise.
        Solution's validity is already checked in solve().
        """
        update = False
        conclusion = rule.conclusion
        if isinstance(conclusion, RelationFact):
            true_entity1, true_entity2 = solution[conclusion.entity1], solution[conclusion.entity2]
            if conclusion.relation not in relations[true_entity1][true_entity2]:
                relations[true_entity1][true_entity2].append(
                    conclusion.relation)
                update = True
        elif isinstance(conclusion, AttributeFact):
            true_entity = solution[conclusion.entity]
            true_expression = conclusion.expression.substitute_entity(solution)
            compute_args = true_expression.parse_compute_args(values)
            value = conclusion.expression.compute(**compute_args)
            if values[true_entity][conclusion.attribute] is None:
                values[true_entity][conclusion.attribute] = value
                update = True
            elif values[true_entity][conclusion.attribute] != value:
                raise ConflictError(
                    f"Conflict: {true_entity}.{conclusion.attribute} already has value {values[true_entity][conclusion.attribute]}, "
                    f"but trying to set it to {value}"
                )
            elif values[true_entity][conclusion.attribute] == value:
                pass
            else:
                raise ValueError(
                    f"Unexpected case: {true_entity}.{conclusion.attribute} has value {values[true_entity][conclusion.attribute]}, "
                    f"but trying to set it to {value}"
                )
        else:
            raise ValueError(f"Unknown conclusion type: {type(conclusion)}")
        return update

    global_update = False
    while True:
        solutions = solve()
        if not solutions:
            break
        local_update = False
        for solution in solutions:
            try:
                local_update = apply(solution) or local_update
            except ConflictError as e:
                print(
                    f"Conflict while applying solution {solution}: {e} on rule {rule}")
                raise e
        global_update = global_update or local_update
        if not local_update:
            break
    return global_update


def trigger_fact(fact, values, relations):
    """
    Trigger a single fact based on the given values and relations.
    If there is conflict, it will raise an ConflictError.
    If an update is made, it will return True.
    If no update is made, it will return False.
    """
    if isinstance(fact, AttributeFact):
        assert isinstance(fact.expression, ConstantExpression), \
            f"Expected ConstantExpression, got {type(fact.expression)}"
        assert fact.entity in values, \
            f"Entity {fact.entity} not found in values"
        assert fact.attribute in values[fact.entity], \
            f"Attribute {fact.attribute} not found in entity {fact.entity}"
        if values[fact.entity][fact.attribute] is None:
            values[fact.entity][fact.attribute] = fact.expression.value
            return True
        elif values[fact.entity][fact.attribute] != fact.expression.value:
            raise ConflictError(
                f"Conflict: {fact.entity}.{fact.attribute} already has value {values[fact.entity][fact.attribute]}, "
                f"but trying to set it to {fact.expression.value}"
            )
    elif isinstance(fact, RelationFact):
        assert fact.entity1 in relations, \
            f"Entity {fact.entity1} not found in relations: {relations}"
        assert fact.entity2 in relations[fact.entity1], \
            f"Entity {fact.entity2} not found in relations of {fact.entity1}: {relations}"
        if fact.relation not in relations[fact.entity1][fact.entity2]:
            relations[fact.entity1][fact.entity2].append(fact.relation)
            return True
        else:
            return False
    else:
        raise ValueError(f"Unknown fact type: {type(fact)}")
