import re

from mathruler.grader import grade_answer, extract_boxed_content
from math_verify import parse, verify
from sympy import symbols, pi

def _inject_implicit_mixed_number(step: str):
    """
    Automatically make a mixed number evalable
    e.g. 7 3/4 => 7+3/4
    """
    p1 = re.compile("([0-9]) +([0-9])")
    step = p1.sub("\\1+\\2", step)  ## implicit mults
    return step


def fix_frac(expr: str) -> str:
    # frac{xxx}{xxx} -> \frac{xxx}{xxx}
    expr = re.sub(r'(?<!\\)frac', r'\\frac', expr)
    # \fracab, \frac{a}b, \fraca{b}, \frac(a)b, \fraca(b), \frac(a)(b) -> \frac{a}{b}
    expr = re.sub(r'\\frac([^{\s])([^{\s])', r'\\frac{\1}{\2}', expr)
    expr = re.sub(r'\\frac(\{[^{}]+\})([^{\s])', r'\\frac\1{\2}', expr)
    expr = re.sub(r'\\frac([^{\s])(\{[^{}]+\})', r'\\frac{\1}\2', expr)
    expr = re.sub(r'\\frac\(([^()]+)\)\(([^()]+)\)', r'\\frac{\1}{\2}', expr)
    expr = re.sub(r'\\frac([^{\s])\(([^()]+)\)', r'\\frac{\1}{\2}', expr)
    expr = re.sub(r'\\frac\(([^()]+)\)([^{\s])', r'\\frac{\1}{\2}', expr)
    return expr


def fix_sqrt(expr: str) -> str:
    # sqrt{xxx} -> \sqrt{xxx}
    expr = re.sub(r'(?<!\\)sqrt', r'\\sqrt', expr)
    # \sqrt(xxx) -> \sqrt{xxx}
    expr = re.sub(r"\\sqrt\((.*?)\)", r'\\sqrt{\1}', expr)
    # \sqrtxxxx -> \sqrt{x}xxx
    expr = re.sub(r'\\sqrt(?!\{)(.)', r'\\sqrt{\1}', expr)
    return expr


def replace_circled_numbers(text: str) -> str:
    def circled_to_digit(match):
        char = match.group(0)
        return str(ord(char) - 0x2460 + 1)

    return re.sub(r'[\u2460-\u2473]', circled_to_digit, text)


def normalize(expr: str) -> str:
    """Normalize answer expressions."""
    if expr is None:
        return None

    # Remove enclosing `\text{}`.
    # m = re.search(r"^\\text\{(?P<text>.+?)\}$", expr)
    # if m is not None:
    #     expr = m.group("text")
    # Remove enclosing `\text{}`. Execute twice to account for two levels of nesting.
    expr = re.sub(r"\\text\{(.*?)\}", r'\1', expr)
    expr = re.sub(r"\\text\{(.*?)\}", r'\1', expr)


    expr = expr.replace("\\%", "%")
    expr = expr.replace("\\$", "$")
    expr = expr.replace("$", "")
    expr = expr.replace(" or ", " , ")
    expr = expr.replace(" and ", " , ")

    expr = expr.replace("million", "*10^6")
    expr = expr.replace("billion", "*10^9")
    expr = expr.replace("trillion", "*10^12")

    for unit in [
        "degree",
        "cm",
        "centimeter",
        "dm",
        "meter",
        "mile",
        "gram",
        "kilo",
        "kilogram",
        "kg",
        "liter",
        "second",
        "minute",
        "hour",
        "day",
        "week",
        "month",
        "year",
        "foot",
        "feet",
        "inch",
        "in",
        "yard",
        "square",
        "cell",
        "unit",
        "yuan",
        "米",
        "厘米",
        "克",
        "千克",
        "公斤",
        "升",
        "秒",
        "分钟",
        "分",
        "小时",
        "天",
        "周",
        "月",
        "年",
        "元",
    ]:
        # end by es/s, ^d or unicode superscript
        expr = re.sub(rf"{unit}(es)?(s)? *(\^({{*)[0-9]+(}}*))?([\u00B2\u00B3\u2070-\u2079]+)?", "", expr)
    
    # delete \cric or ^\cric or ^{\cric} or unicode format degree
    expr = re.sub(r"\^ *\\circ", "", expr)
    expr = re.sub(r"\^ *{\\circ}", "", expr)
    expr = re.sub(r"\\circ", "", expr)
    expr = re.sub(r"°", "", expr)

    if len(expr) > 0 and expr[0] == "{" and expr[-1] == "}":
        expr = expr[1:-1]

    expr = re.sub(",\\\\! *", "", expr)
    # if _is_float(expr) and _is_int(float(expr)):
    #     expr = str(int(round(float(expr))))
    # if "\\" in expr:
    #     try:
    #         expr = _parse_latex(expr)
    #     except:
    #         pass

    # edge case with mixed numbers and negative signs
    expr = re.sub("- *", "-", expr)

    expr = _inject_implicit_mixed_number(expr)
    expr = expr.replace(" ", "")

    # if we somehow still have latex braces here, just drop them
    expr = expr.replace("{", "")
    expr = expr.replace("}", "")

    # if we somehow still have blank (), drop them 
    expr = expr.replace("()", "")

    expr = expr.replace("√", "\\sqrt")
    expr = fix_frac(expr)
    expr = fix_sqrt(expr)

    # don't be case sensitive for text answers
    expr = expr.lower()

    # if _str_is_int(expr):
    #     expr = str(_str_to_int(expr))
    # Geometry
    expr = expr.replace("\\parallel", "//")
    expr = expr.replace("平行", "//")
    expr = expr.replace("⊥", "\\perp")
    expr = expr.replace("△", "")
    expr = expr.replace("∠", "\\angle")
    expr = expr.replace("角", "\\angle")
    expr = expr.replace("平面", "plane")
    expr = expr.replace("且", "and")
    expr = expr.replace("\\times", "*")
    expr = expr.replace("正确", "correct")
    expr = expr.replace("错误", "incorrect")
    expr = expr.replace("notlessthan", "\\leq")
    expr = expr.replace("notlessthan", "\\leq")
    expr = replace_circled_numbers(expr)
    expr = expr.replace(",", "")
    if "不够" in expr:
        expr = "no"
    elif "够" in expr:
        expr = "yes"
    if "notenough" in expr:
        expr = "no"
    elif "enough" in expr:
        expr = "yes"

    return expr


def r1v_format_reward(predict_str: str) -> float:
    pattern = r"<think>.*?</think>\s*<answer>.*?</answer>"
    match = re.fullmatch(pattern, predict_str, re.DOTALL)
    return 1.0 if match else 0.0

'''
def r1v_accuracy_reward(predict_str: str, ground_truth: str) -> float:
    try:
        ground_truth = ground_truth.strip()
        content_match = re.search(r"<answer>(.*?)</answer>", predict_str)
        pred_answer = content_match.group(1).strip() if content_match else predict_str.strip()
        if grade_answer(pred_answer, ground_truth):
            return 1.0
    except Exception:
        pass

    return 0.0
'''

def r1v_accuracy_reward(predict_str: str, ground_truth: str, prompt_str: str) -> float:
    
    if prompt_str != None:
        if 'Here is an image of multiple input output grids.' in prompt_str:
            print('scoring arc-agi, using direct matching')
            try:
                box_response = extract_boxed_content(predict_str)
            except:
                box_response = ''

            if box_response == ground_truth:
                return 1.0
            else:
                return 0.0

    try:
        ground_truth = ground_truth.strip()

        content_match = re.search(r"<answer>(.*?)</answer>", predict_str, re.DOTALL)
        pred_answer = content_match.group(1).strip()# if content_match else predict_str.strip()

        box_response = extract_boxed_content(pred_answer)
        #verify content in the box frist
        #verify in mathruler
        try:
            if grade_answer(box_response, ground_truth):
                return 1.0
        except Exception as e:
            print('grade_answer for box content went wrong!!!!!!')

        #verify in math_verify
        try:
            box_response_math_verify = parse(box_response)
            answer_math_verify = parse(ground_truth)
            if verify(box_response_math_verify, answer_math_verify):
                return 1.0
        except Exception as e:
            print('math_verify for box content went wrong!!!!!!')

        #verify in homemake rule
        try:
            box_response_homemake = normalize(box_response).strip()
            answer_homemake = normalize(ground_truth).strip()
            if box_response_homemake == answer_homemake:
                return 1.0
            if verify(box_response_homemake, answer_homemake):
                return 1.0
        except Exception as e:
            print('homemake rule for box content went wrong!!!!!!')
        '''
        #verify content in <answer>
        content_match = re.search(r"<answer>(.*?)</answer>", predict_str)
        pred_answer = content_match.group(1).strip() if content_match else predict_str.strip()
        if grade_answer(pred_answer, ground_truth):
            return 1.0

        pred_answer = normalize(pred_answer).strip()
        ground_truth = normalize(ground_truth).strip()
        if pred_answer == ground_truth:
            return 1.0
        if grade_answer(pred_answer, ground_truth):
            return 1.0
        if verify(pred_answer, ground_truth):
            return 1.0
        '''

        if pred_answer == ground_truth:
            return 1.0
        pred_answer = normalize(pred_answer).strip()
        ground_truth = normalize(ground_truth).strip()
        if pred_answer == ground_truth:
            return 1.0
        pred_answer = parse(f"\\boxed{{{pred_answer}}}")
        ground_truth = parse(f"\\boxed{{{ground_truth}}}")

        # consider the constant pi
        pred_answer[0] = pred_answer[0].subs(pi, 3.14)
        ground_truth[0] = ground_truth[0].subs(pi, 3.14)

        if content_match is None or pred_answer is None:
            return 0.0
        # print(pred_answer, ground_truth)
        if verify(pred_answer, ground_truth):
            return 1.0

    except Exception as e:
        if isinstance(e, ValueError):
            print(f"Error occurred when computing reward!\n[[predict_str]] {predict_str}\n[[ground_truth]] {ground_truth}")
        

    return 0.0

def r1v_compute_score(predict_str: str, ground_truth: str, valid_response_ids = [], validation: bool = False, prompt_str: str = None) -> float:
    acc_reward = r1v_accuracy_reward(predict_str, ground_truth, prompt_str)
    if validation:
        return acc_reward
    format_reward = r1v_format_reward(predict_str)
    reward = acc_reward + format_reward * 0.5
    #return reward
    return {'all_reward':reward, 'acc_reward':acc_reward, 'format_reward':format_reward}


def r1v_acc_score(predict_str: str, ground_truth: str, valid_response_ids = [], validation: bool = False, prompt_str: str = None) -> float:
    acc_reward = r1v_accuracy_reward(predict_str, ground_truth, prompt_str)
    return {'all_reward':acc_reward, 'acc_reward':acc_reward}
