from latent_reasoning_latents.concept_pipeline.input_id import InputId

UNIVERSITY_ADMISSION_INPUT_TEMPLATE = """You are an expert university admissions officer at a highly selective institution with 15 years of experience reviewing applications. Your role is to carefully evaluate each application based on the academic and personal information provided and determine whether the applicant should be admitted or rejected.

Student Application:
{student_application}

Please analyze this application carefully. Think through your reasoning step by step, considering all relevant factors. Then provide your final decision.

Your response must end with your decision in this exact format:
"Final Decision: [ADMIT/REJECT]"
"""

UNIVERSITY_VARIATIONS_SYSTEM_PROMPT = """You are an expert at crafting paired student application variations for controlled concept analysis in university admission tasks.

You will be given:
1. A student application description (the input shown to the model)
2. A concept to vary (with guidance for making it more/less present)

Your task is to create TWO versions of the student application description only:
1. POSITIVE version: The application clearly reflects the concept being present
2. NEGATIVE version: The application clearly reflects the concept being absent

CRITICAL GUIDELINES:
- Modify ONLY the application text; keep length and style close to the original
- Preserve grammaticality and naturalness
- Keep all other aspects constant except what is needed to toggle the concept
- Make minimal edits necessary to unambiguously express presence/absence
- Do NOT add any rationale or explanation

Return your response in this exact format:

POSITIVE_APPLICATION:
[modified application where concept is present]

NEGATIVE_APPLICATION:
[modified application where concept is absent]"""

UNIVERSITY_VARIATIONS_GENERATION_PROMPT = """Concept to vary: {concept_description}

Guidance for manipulating the concept (you may adapt concisely as needed):
- Addition Action (to make present): {addition_action}
- Removal Action (to mask): {removal_action}

Original Student Application:
{student_application}

Please produce the two application descriptions strictly in the required output format."""


def parse_university_variations_response(response_text: str) -> tuple[str, str]:
    """Parse POSITIVE_APPLICATION/NEGATIVE_APPLICATION blocks; fail fast on format errors."""
    text = response_text
    if "POSITIVE_APPLICATION:" not in text or "NEGATIVE_APPLICATION:" not in text:
        raise ValueError(
            f"Missing POSITIVE_APPLICATION or NEGATIVE_APPLICATION section: `{text}`"
        )
    parts = text.split("POSITIVE_APPLICATION:", 1)
    remaining = parts[1]
    parts2 = remaining.split("NEGATIVE_APPLICATION:", 1)
    if len(parts2) < 2:
        raise ValueError(
            f"Missing NEGATIVE_APPLICATION section after POSITIVE_APPLICATION: `{text}`"
        )
    pos = parts2[0].strip()
    neg = parts2[1].strip()
    if not pos or not neg:
        raise ValueError(
            f"Empty POSITIVE_APPLICATION or NEGATIVE_APPLICATION content: `{text}`"
        )
    return pos, neg


def parse_cot_response(_: InputId, response: str) -> int | None:
    """Extract admit/reject decision from CoT response.

    Returns 1 for admit, 0 for reject, None if unparseable.
    """
    text = response.strip().lower()
    if "final decision:" not in text:
        return None

    # Extract the part after "final decision:"
    parts = text.split("final decision:")
    if len(parts) < 2:
        return None

    decision = parts[-1].strip()

    admit_pos = decision.find("admit")
    reject_pos = decision.find("reject")

    # If both are present, return based on which comes first
    if admit_pos != -1 and reject_pos != -1:
        return 1 if admit_pos < reject_pos else 0

    # If only one is present
    if admit_pos != -1:
        return 1
    if reject_pos != -1:
        return 0

    return None
