import re
import json


def parse_log_pytest(log: str | None) -> dict[str, str]:
    """
    Parser for test logs generated with Sympy framework

    Args:
        log (str): log content
    Returns:
        dict: test case to test status mapping
    """
    if log is None:
        return {}
    test_status_map = {}
    if "short test summary info" not in log:
        return test_status_map
    log = log.split("short test summary info")[1]
    log = log.strip()
    log = log.split("\n")
    for line in log:
        if "PASSED" in line:
            test_name = ".".join(line.split("::")[1:])
            test_status_map[test_name] = "PASSED"
        elif "FAILED" in line:
            test_name = ".".join(line.split("::")[1:]).split(" - ")[0]
            test_status_map[test_name] = "FAILED"
        elif "ERROR" in line:
            try:
                test_name = ".".join(line.split("::")[1:])
            except IndexError:
                test_name = line
            test_name = test_name.split(" - ")[0]
            test_status_map[test_name] = "ERROR"
    return test_status_map


def decolor_dict_keys(key):
    decolor = lambda key: re.sub(r"\u001b\[\d+m", "", key)  # noqa: E731
    return {decolor(k): v for k, v in key.items()}


def get_reward(parse, instance):
    parse = decolor_dict_keys(parse)
    expected_json = instance["expected_output_json"]

    expected: dict = json.loads(expected_json)
    expected = decolor_dict_keys(expected)
    parse = {k.split(" - ")[0]: parse[k] for k in sorted(parse.keys())}
    expected = {k.split(" - ")[0]: expected[k] for k in sorted(expected.keys())}

    # Compare
    if len(parse) != len(expected):
        reward = 0.0
    else:
        # If ANY mismatch, reward = 0.0, else = 1.0
        match = True
        for k in parse.keys():
            if not k:
                continue
            if k not in expected:
                match = False
                break
            if parse[k] != expected[k]:
                match = False
                break
        reward = 1.0 if match else 0.0

    return reward
