import re
import click
import pandas as pd
from eval.util import (
    # load_model_and_tokenizer,
    # batched_generate,
    parse_number,
    format_example,
    prep_incontext_examples,
    write_results,
)
from eval.eval_hacks import load_sampler, batched_generate
from utils import read_json, seed_all
from tqdm.auto import tqdm as tqdm
from pathlib import Path
import json
seed_all(42)


def evaluate_gsm(sampler_factory, test_df, batch_size, num_incontext_examples, qa_format):
    test_df = test_df.reset_index(drop=True)
    incontext_indices = prep_incontext_examples(test_df, num_incontext_examples)

    def format_answer(answer):
        answer = re.sub(r"<<.*?>>", "", answer)
        final_answer = answer.split("####")[-1].strip()
        sentences = answer.split("####")[0].strip().split("\n")
        sentences = [s + "." if not s.endswith(".") else s for s in sentences]
        return " ".join(sentences) + f"\n#### {final_answer}"

    prompts = []
    for i, row in test_df.iterrows():
        prompt = ""
        for j in incontext_indices[i]:
            ic_row = test_df.iloc[j]
            prompt += (
                format_example(ic_row["question"], answer=format_answer(ic_row["answer"]), qa_format=qa_format)
                + "\n\n"
            )

        prompt += format_example(row["question"], qa_format=qa_format)
        prompts.append(prompt)

    print(f"--- GSM8K example prompt ---\n{prompts[0]}\n----------------------")
    outputs = batched_generate(
        prompts=prompts,
        bsf=sampler_factory,
        do_sample=False,
        max_new_bytes=1000,
        batch_size=batch_size,
        stop_strings=["\n\nQuestion:"]
    )
    results = []
    for prompt, output, answer in zip(prompts, outputs, test_df.answer):
        output = output.split("\n\n")[0]
        pred = parse_number(output.split("####")[-1])
        short_answer = parse_number(answer.split("####")[-1])
        results.append(
            {
                "prompt": prompt,
                "output": output,
                "answer": answer,
                "correct": pred == short_answer,
            }
        )
    return results


@click.command()
@click.option("--model_name_or_path", type=str, default="pile-npt25k")
@click.option("--output_dir", type=str)
@click.option("--num_incontext_examples", type=int, default=5)
@click.option("--eval_batch_size", type=int, default=2)
@click.option("--qa_format", type=str, default="qa")
@click.option("--start", type=int)
@click.option("--end", type=int)
def main(
    model_name_or_path: str,
    output_dir: str,
    num_incontext_examples: int,
    eval_batch_size: int,
    qa_format: str,
    start: int,
    end: int
):
    out_dir = Path(output_dir)
    out_dir.mkdir(exist_ok=True)
    out_file = out_dir/ f'out_{start}_{end}.json'
    print("out:", out_file)
    if out_file.exists():
        return 
        
    sampler_factory = load_sampler(model_name_or_path)
    test_df = pd.read_json("olmo_data/eval/gsm8k/test.jsonl", lines=True)

    global results
    results = evaluate_gsm(
       sampler_factory,
        test_df[start:end],
        batch_size=eval_batch_size,
        num_incontext_examples=num_incontext_examples,
        qa_format=qa_format,
    )
    with open(out_file, 'w') as f:
        json.dump(results, f)

if __name__ == "__main__":
    main()
# main("meta-llama/Llama-3.2-1B", None, 5, 10, 8, 'qa')