#!/usr/bin/env python3
"""
Standalone run script for autocomp search.

Copy this file and edit the configuration below to start a new run
without modifying search.py defaults.

Usage:
    cd /path/to/autocomp
    python run.py
"""
import pathlib
import random

from autocomp.search.search import (
    BeamSearchStrategy,
    ExhaustiveSearchStrategy,
    create_backend_and_agents,
    load_initial_code,
)
from autocomp.search.prob import Prob
from autocomp.hw_config import TpuHardwareConfig
from autocomp.common import logger


def main():
    # ── Backend & agent ──────────────────────────────────────────────────
    backend_name = "jaxbench"
    agent_name = "built:tpu-v6e"
    simulator = None
    hw_config = TpuHardwareConfig("v6e-1")

    # ── LLM models ──────────────────────────────────────────────────────
    models = [
        "aws::us.anthropic.claude-opus-4-5-20251101-v1:0",
        "openai::gpt-5.4",
        "aws::zai.glm-5",
        "aws::minimax.minimax-m2.5",
        "gcp::gemini-3-flash-preview",
    ]
    code_models = None

    # ── Problem ─────────────────────────────────────────────────────────
    prob_type = "jaxbench-priority"
    prob_id = "mla_attention"

    # ── Search ──────────────────────────────────────────────────────────
    metric = "latency"
    search_strategy = "beam"
    iterations = 24

    # Beam search
    num_plan_candidates = 5
    num_code_candidates = 2
    beam_size = 4

    # Translation
    translate_iters = 0
    translate_perf_threshold = 15
    translate_drop_original = True
    translate_score = True

    # Resume from a previous run (set to None to start fresh)
    resume_from = "output/built:tpu-v6e_jaxbench-priority_mla_attention_beam_iters6_TPU_v6e-1_us-4-5-20251101-v1:0_openai::gpt-5.4_aws::zai.glm-4.7_aws::deepseek.v3.2_mini-3-flash-preview_do0.25_p5_c2_b4_tr6_10_trdrop_tscore_score1_ms1_fgisa1_ex0.25"

    # Edit mode (True = JSON edits, False = full code rewrite)
    use_edits = True

    # ── Agent tuning ────────────────────────────────────────────────────
    menu_strategy = "one-shot"
    fine_grained_isa = True
    example_rate = 0.25
    dropout_menu_options = 0.25
    give_score_feedback = 1
    give_util_feedback = 0
    give_hw_feedback = 0
    include_ancestors = False
    plan_icl_examples = False
    code_icl_examples = False

    # ── Misc ────────────────────────────────────────────────────────────
    reimplement_failed = False
    early_stop_iters = 0
    early_stop_threshold = 1.0
    num_analyses = 0
    num_pairs_to_combine = 0
    num_gen_per_combine = 0
    trigger_exhaustive_threshold = 1
    trigger_exhaustive_iters = 20
    start_exhaustive_iters = 0
    prevent_duplicate_level = 0
    random.seed(1111)

    # ── Build output directory name ─────────────────────────────────────
    built_menu_strategy_enum = {None: 0, "one-shot": 1}
    for i in range(len(models)):
        models[i] = models[i].replace("/", "_")
    if code_models is not None:
        for i in range(len(code_models)):
            code_models[i] = code_models[i].replace("/", "_")

    clean_agent_name = pathlib.Path(agent_name).name if "/" in agent_name else agent_name
    output_str = f"{clean_agent_name}"
    output_str += f"_{prob_type}_{prob_id}_{search_strategy}_iters{iterations}"
    if simulator is not None:
        output_str += f"_{simulator}"
    hw_desc = hw_config.get_hw_description().replace(" ", "").replace("(", "_").replace(")", "").replace(",", "_")
    output_str += f"_{hw_desc}"
    for model in models:
        output_str += f"_{model[-20:]}"
    if code_models is not None:
        output_str += "_code"
        for model in code_models:
            output_str += f"_{model[-20:]}"
    if dropout_menu_options:
        output_str += f"_do{dropout_menu_options}"
    if search_strategy == "beam":
        if num_analyses:
            output_str += f"_an{num_analyses}"
        output_str += f"_p{num_plan_candidates}_c{num_code_candidates}_b{beam_size}"
    if translate_iters > 0:
        output_str += f"_tr{translate_iters}_{translate_perf_threshold}"
        if translate_drop_original:
            output_str += "_trdrop"
        if translate_score:
            output_str += "_tscore"
    if give_score_feedback:
        output_str += f"_score{give_score_feedback}"
    if give_util_feedback:
        output_str += f"_util{give_util_feedback}"
    if give_hw_feedback:
        output_str += f"_hwfb{give_hw_feedback}"
    if include_ancestors:
        output_str += "_anc1"
    if prevent_duplicate_level:
        output_str += f"_pd{prevent_duplicate_level}"
    if plan_icl_examples:
        output_str += "_picl1"
    if code_icl_examples:
        output_str += "_cicl1"
    if reimplement_failed:
        output_str += "_reimpl1"
    if early_stop_iters > 0:
        output_str += f"_es{early_stop_iters}_{early_stop_threshold}"
    if menu_strategy:
        output_str += f"_ms{built_menu_strategy_enum[menu_strategy]}"
    if fine_grained_isa:
        output_str += "_fgisa1"
    if example_rate > 0:
        output_str += f"_ex{example_rate}"
    if resume_from:
        output_str += "_resumed"
    if use_edits:
        output_str += "_edits"
    output_dir = pathlib.Path("output/" + output_str)

    output_dir.mkdir(parents=True, exist_ok=True)
    import autocomp.common.my_logging
    autocomp.common.my_logging.move_log(output_dir)

    logger.info(f"Output directory: {output_dir}")

    # ── Initialize and run ──────────────────────────────────────────────
    prob = Prob(prob_type, prob_id)
    initial_code = load_initial_code(backend_name, prob)

    eval_backend, agent, code_agent = create_backend_and_agents(
        backend_name, agent_name, hw_config, prob, models, code_models,
        menu_strategy=menu_strategy, fine_grained_isa=fine_grained_isa,
        example_rate=example_rate, cache_dir=output_dir,
    )

    if search_strategy == "exhaustive":
        optimizer = ExhaustiveSearchStrategy(
            output_dir, eval_backend, agent, initial_code, prob, metric, simulator,
            give_score_feedback, give_util_feedback, give_hw_feedback, include_ancestors,
            plan_icl_examples, code_icl_examples, dropout_menu_options, prevent_duplicate_level,
            translate_iters, translate_perf_threshold, translate_drop_original, translate_score,
            code_agent=code_agent, early_stop_iters=early_stop_iters,
            early_stop_threshold=early_stop_threshold, resume_from=resume_from,
            use_edits=use_edits,
        )
    elif search_strategy == "beam":
        optimizer = BeamSearchStrategy(
            output_dir, eval_backend, agent, initial_code, prob, metric, simulator,
            give_score_feedback, give_util_feedback, give_hw_feedback, include_ancestors,
            plan_icl_examples, code_icl_examples,
            num_analyses=num_analyses, num_plan_candidates=num_plan_candidates,
            num_code_candidates=num_code_candidates, beam_size=beam_size,
            num_pairs_to_combine=num_pairs_to_combine, num_gen_per_combine=num_gen_per_combine,
            dropout_menu_options=dropout_menu_options,
            trigger_exhaustive_threshold=trigger_exhaustive_threshold,
            trigger_exhaustive_iters=trigger_exhaustive_iters,
            start_exhaustive_iters=start_exhaustive_iters,
            prevent_duplicate_level=prevent_duplicate_level,
            reimplement_failed=reimplement_failed,
            translate_iters=translate_iters, translate_perf_threshold=translate_perf_threshold,
            translate_drop_original=translate_drop_original, translate_score=translate_score,
            code_agent=code_agent, early_stop_iters=early_stop_iters,
            early_stop_threshold=early_stop_threshold, resume_from=resume_from,
            use_edits=use_edits,
        )

    optimizer.optimize(iterations)


if __name__ == "__main__":
    main()
