import sys
from pathlib import Path

# Add project root directory to path
sys.path.insert(0, str(Path(__file__).parent.parent))

from src.core.Plotter import RealWorldPlotter
import json
from src.core.NumericalCheck import NumericalCheck
from src.core.LLMPlotter import LLMPlotter
from src.core.VLImageQuality import VLImageQuality
problem = """
The figure contains a circle with center O. There are four points A, B, C, and D on the circle. The segments AB, BC, CD, and DA are drawn, forming a cyclic quadrilateral ABCD. In addition, two diagonals AC and BD are drawn. Segment BD passes through the center O. In the figure, segment AB is labeled with length 6, segment BC is labeled with length 6, and angle BAC is labeled as 30 degrees.
"""

if False:
    VLImageQuality = VLImageQuality('None','models/Qwen3-VL-8B-Thinking', 'http://localhost:8002/v1')
    plotting_data = {"points": {"A": [12, 0], "B": [0, 0], "C": [6, 9], "D": [6, 15], "E": [0, 9], "F": [3, 3], "G": [0, 3], "H": [3, 0], "I": [3, 6]}, "segments": [["A", "B"], ["B", "E"], ["A", "E"], ["C", "D"], ["C", "E"], ["G", "F"], ["F", "I"], ["H", "F"], ["b", "g"], ["a", "h"]], "circles": [], "annotations": {"right_angles": [["D", "C", "E"], ["A", "B", "E"], ["B", "G", "F"], ["A", "H", "F"], ["G", "F", "I"]], "length_of_line": [[["C", "D"], "6"], [["C", "E"], "6"]], "measure_of_angle": []}, "segment_chains": [["A", "H", "B"], ["B", "G", "E"], ["A", "E"], ["C", "D"], ["C", "E"], ["G", "F"], ["H", "F", "I"]], "annotation_summary": {"length_of_line": {"applied": [[["C", "D"], "6"], [["C", "E"], "6"]], "skipped": [{"entry": [["A", "B"], "12"], "reason": "non_atomic_segment"}, {"entry": [["B", "E"], "9"], "reason": "non_atomic_segment"}]}, "measure_of_angle": {"applied": [], "skipped": []}, "right_angles": {"applied": [["D", "C", "E"], ["A", "B", "E"], ["B", "G", "F"], ["A", "H", "F"], ["G", "F", "I"]], "skipped": []}}}
    VLImageQuality.check_quality_and_generate_caption('data/figures/figure_9.png', plotting_data, index=91)
    plotter = LLMPlotter(
            api_key="your-api-key",
            model="models/gpt-oss-120b/snapshots/8b193b0ef83bd41b40eb71fee8f1432315e02a3e/",
            base_url="http://localhost:8001/v1",
        )

    result = plotter.plot(problem, index=0)

    if result.get("success"):
        plotting_data = result.get("plotting_data", {})
        print("\n" + "="*60)
        print("Success! plotting_data:")
        print("="*60)
        print(json.dumps(plotting_data, ensure_ascii=False))

if True:
    #plotting_data =  {"points": {"A": [0, 0], "B": [13, 0], "C": [7, 12], "D": [7, 4], "E": [10, 5], "F": [3, 6], "G": [7, 0], "H": [3, 2]}, "segments": [["A", "B"], ["B", "C"], ["C", "A"], ["D", "E"], ["D", "F"], ["D", "G"], ["E", "H"], ["D", "H"]], "circles": [["C1", "D", 4.0]], "annotations": {"right_angles": [], "length_of_line": [[["B", "C"], "14"]], "measure_of_angle": []}, "segment_chains": [["A", "G", "B"], ["B", "C"], ["C", "F", "A"], ["H", "D", "E"], ["D", "F"], ["D", "G"]]}

    #plotter = RealWorldPlotter()
    plotting_data = """
    {"points": {"A": [0.0, 0.0], "B": [8, 0.0], "C": [4.0, 6.928203230275509], "D": [8, 8.0], "E": [0.0, 8.0]}, "segments": [["A", "B"], ["B", "C"], ["A", "C"], ["B", "D"], ["A", "D"], ["C", "D"], ["D", "E"], ["E", "C"], ["A", "E"]], "circles": [], "quantities": ["area(C, D, E)"], "annotations": {"right_angles": [["A", "B", "D"]], "length_of_line": [[["A", "B"], "8"]], "measure_of_angle": []}}
    """

    plotting_data = json.loads(plotting_data)

    answers = ["45^{\\circ}","60^{\\circ}","4/5","45^{\\circ}","\\arctan\\frac{3}{4}","-\\frac{7}{25}","-\\frac{7}{25}","13.9^{\\circ}","\\arccos\\frac{4}{5}","60^{\\circ}","12^{\\circ}","30^{\\circ}","30^{\\circ}","2","45^{\\circ}","60","50^{\\circ}","45^{\\circ}","45^{\\circ}","45^{\\circ}","30^{\\circ}","\\frac{5}{12}","\\arctan\\frac{3}{2}","\\frac{3}{2}","2","22.5^{\\circ}","26.565^{\\circ}","30^{\\circ}", "45^\circ", "\\frac{35\sqrt{2}}{31}"]
    answers = [
  "\\arccos\\frac{13}{5\\sqrt{17}}",
  "\\frac{\\sqrt{193}}{10}\\left(\\frac{3\\sqrt{193}-30}{8}+\\frac{1}{2}\\sqrt{-37+20\\sqrt{193}}\\right)",
  "9\\left(1+\\sqrt{3}+\\sqrt{7}-\\sqrt{21}\\right)",
  "\\arccos\\left(\\frac{-3(2\\sqrt{13}-5)+2\\sqrt{80\\sqrt{13}-58}}{25}\\right)",
  "\\arccos\\frac{9}{5\\sqrt{10}}",
  "\\arccos\\!\\left(\\frac{1-\\sqrt2+\\sqrt5}{2\\sqrt{6-\\sqrt{10}+2\\sqrt5-2\\sqrt2}}\\right)",
  "\\sqrt{\\frac{5}{2}\\left(11+\\sqrt{117}\\right)}",
  "\\frac{\\sqrt{274}}{8}",
  "\\arccos\\frac{41}{13\\sqrt{10}}",
  "\\frac{\\sqrt{10}\\,(19-\\sqrt{109})}{10}",
  "\\sqrt{24+6\\left(\\sqrt{6}-\\sqrt{2}\\right)}",
  "\\arccos\\frac{2}{\\sqrt{5}}",
  "\\frac{5\\sqrt{2}\\,(1+\\sqrt{3})}{3}",
  "\\frac{27}{7}\\left(\\sqrt{6}-\\sqrt{2}\\right)",
  "\\arccos\\frac{63}{\\sqrt{3990}}",
  "\\frac{\\sqrt{2}\\,(\\sqrt{3}-1)\\bigl(\\sqrt{217+96\\sqrt{3}}-5\\bigr)}{4}",
  "\\frac{2\\sqrt{29}}{3}",
  "36\\pi",
  "\\arccos\\left(\\sqrt{\\frac{3}{28}}\\right)",
  "\\frac{\\sqrt{3}\\,(7-\\sqrt{13})}{6}",
  "\\arccos\\frac{17+\\sqrt5}{\\sqrt{650+130\\sqrt5}}",
  "6\\left(2-\\sqrt{2}\\right)",
  "\\arccos\\frac{7\\sqrt{2}}{10}",
  "12\\sin 10^{\\circ}",
  "3\\left(\\sqrt{6}+\\sqrt{2}\\right)",
  "\\frac{3\\sqrt{13}}{2}",
  "\\sqrt{\\frac{2\\bigl(11-\\sqrt{21}\\bigr)}{5}}",
  "\\frac{6}{5}\\left(\\sqrt{21}-3\\right)",
  "\\arccos\\left(-\\frac{4}{5}\\right)",
  "\\sqrt{101}",
  "\\frac{3(\\sqrt{13}+1)\\sqrt{2\\sqrt{13}-5}}{2\\sqrt[4]{13}}",
  "\\arccos\\frac{5}{\\sqrt{29}}",
  "\\frac{32\\sqrt{207}-190}{7\\sqrt{2}\\,\\sqrt{17+\\sqrt{207}}}",
  "\\frac{5}{2}\\left(2-\\sqrt{2}\\right)",
  "2\\sqrt{2}\\,(5-\\sqrt{13})",
  "\\frac{4\\sqrt{5}\\,(7-2\\sqrt{2})}{5\\sqrt{34-11\\sqrt{2}}}",
  "\\arctan\\left(\\frac{3\\sqrt{2}}{4}-1\\right)",
  "3\\bigl(3\\sqrt2-\\sqrt6\\bigr)",
  "\\arccos\\frac{5}{\\sqrt{34}}",
  "\\arccos\\left(-\\frac{9}{40}\\right)"
]
   
    output = Path("test.png")
    print(output)
    #print(plotter.render_image(plotting_data, output))

if True:
    answer = "676"
    numerical_check = NumericalCheck()
    for answer in answers:
        result = numerical_check.check(answer=answer, meta_data=plotting_data)
        print(answer, result)

if False:
    row_data = """{"n_premises": 2, "n_points": 4, "fl_problem": "", "nl_problem": "", "n_proof_steps": 1, "llm_input_renamed": "<problem> a : ; b : ; c d : coll a b c [000] coll a b d [001] ? coll b c d </problem>", "llm_output_renamed": "<proof> coll b c d [002] AR [000] [001] ; </proof>", "difficulty_score": 2.25}"""

    from typing import Dict, Any
    import re
    from src.utils.symbol_translator import SymbolTranslator
    translator = SymbolTranslator()

    def extract_problem_and_proof(data: Dict[str, Any]) -> tuple[str, str, str, str]:
            """Extract problem, conclusion, and aux from the data"""
            llm_input = data.get("llm_input_renamed", "")
            llm_output = data.get("llm_output_renamed", "")
            # Extract <problem>...</problem>
            problem_match = re.search(r'<problem>(.*?)</problem>', llm_input, re.DOTALL)
            problem = problem_match.group(1).strip() if problem_match else ""
            
            # Extract conclusion (goal): take the part after ? from problem
            conclusion = ""
            if "?" in problem:
                conclusion = problem.split("?")[-1].strip()
                # Remove the conclusion part from problem and keep only the premises
                problem = problem.split("?")[0].strip()
            
            # Extract <aux>...</aux>
            aux = ""
            aux_match = re.search(r'<aux>(.*?)</aux>', llm_output, re.DOTALL)
            if aux_match:
                aux = aux_match.group(1).strip()
            
            return problem, conclusion, aux
        

    problem, conclusion, aux = extract_problem_and_proof(json.loads(row_data))
    print(problem)
    print(conclusion)
    print(aux)

    # Translate the symbolic language into natural language
    # First try to use translate_problem_origin to directly translate the problem_origin format
    # If problem is in problem_origin format (contains patterns like "a :"), use translate_problem_origin
    if re.search(r'\w+\s*:\s*', problem):
        problem_nl = translator.translate_problem_origin(problem)
        print(problem_nl)
        if not problem_nl:
            # If translation fails, try using translate_problem
            problem_nl = translator.translate_problem(problem, simplify=True)
            if not problem_nl:
                problem_nl = problem

    from src.core.VLImageQuality import VLImageQuality
    from src.utils.config import Config
    from src.utils.model_urls import get_base_url_for_model
    """
    output = Path("test.png")
    plotter.render_image(plotting_data, output)
    vl_image_quality = VLImageQuality(api_key=Config.API_KEY, model="gemini-2.5-flash", base_url=get_base_url_for_model("gemini-2.5-flash"))
    result = vl_image_quality.check_quality_and_generate_caption(image_path=output, plotting_data=plotting_data)
    print(result)
    """

    from src.core.VisualizeQA import VisualizeQA
    caption = "The figure contains a circle with center O. There are four points A, B, C, and D on the circle. The segments AB, BC, CD, and DA are drawn, forming a cyclic quadrilateral ABCD. In addition, two diagonals AC and BD are drawn. Segment BD passes through the center O. In the figure, segment AB is labeled with length 6, segment BC is labeled with length 6, and angle BAC is labeled as 30 degrees."
    visualize_qa = VisualizeQA(api_key=Config.API_KEY, model="gemini-2.5-flash", base_url=get_base_url_for_model("gemini-2.5-flash"))
    problem = "In a cyclic quadrilateral $\\text{ABCD}$, the chord $\\text{BC}$ has length $6$, the chord $\\text{AB}$ has length $6$, and $\\angle\\text{BAC} = 30^\\circ$. Find the measure of $\\angle\\text{ADB}$."
    cot = """
    Step 1: Determine the radius of the circle.\nAccording to the property of the cyclic quadrilateral $\\text{ABCD}$, all vertices lie on the same circle. For the chord $\\text{BC}$ and its subtended inscribed angle $\\angle\\text{BAC}$, by the sine theorem, the ratio of the chord length to the sine of the corresponding inscribed angle equals the diameter $2\\text{R}$ of the circle.\nThus, $\\frac{\\text{BC}}{\\sin(\\angle\\text{BAC})} = 2\\text{R}$.\nSubstitute the known values: $\\frac{6}{\\sin(30^\\circ)} = 2\\text{R}$.\nSince $\\sin(30^\\circ) = \\frac{1}{2}$, we have $\\frac{6}{1/2} = 2\\text{R}$.\nThat is, $12 = 2\\text{R}$.\nTherefore, the radius of the circle is $\\text{R} = 6$.\n\nStep 2: Determine the measure of $\\angle\\text{ADB}$.\nThe chord $\\text{AB}$ subtends the inscribed angle $\\angle\\text{ADB}$ (as well as $\\angle\\text{ACB}$). By the same sine theorem, the ratio of the chord length to the sine of the corresponding inscribed angle equals the diameter $2\\text{R}$ of the circle.\nTherefore, $\\frac{\\text{AB}}{\\sin(\\angle\\text{ADB})} = 2\\text{R}$.\nSubstitute the known values $\\text{AB} = 6$ and $2\\text{R} = 12$ that we just obtained:\n$\\frac{6}{\\sin(\\angle\\text{ADB})} = 12$.\nWe solve to get $\\sin(\\angle\\text{ADB}) = \\frac{6}{12} = \\frac{1}{2}$.\nSince $\\angle\\text{ADB}$ is an inscribed angle, its measure lies between $0^\\circ$ and $180^\\circ$, and for a chord that is not a diameter, the inscribed angle is acute. Thus $\\angle\\text{ADB} = 30^\\circ$.
    """
    #result = visualize_qa.visualize_qa(question=problem, cot=cot, caption=caption)
    #print(result)