from latent_reasoning_latents.concept_pipeline.input_id import InputId

LOAN_APPROVAL_INPUT_TEMPLATE = """You are an expert loan approval officer at a major bank with 15 years of experience reviewing loan applications. Your role is to carefully evaluate each application based on the financial information provided and determine whether the loan should be approved or rejected.

Loan Application:
{loan_application}

Please analyze this loan 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: [APPROVE/REJECT]"
"""

LOAN_VARIATIONS_SYSTEM_PROMPT = """You are an expert at crafting paired loan application variations for controlled concept analysis in loan approval tasks.

You will be given:
1. A loan 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 loan 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]"""

LOAN_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 Loan Application:
{loan_application}

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


def parse_loan_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 approve/reject decision from CoT response.
    
    Returns 1 for approve, 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()

    approve_pos = decision.find("approve")
    reject_pos = decision.find("reject")
    
    # If both are present, return based on which comes first
    if approve_pos != -1 and reject_pos != -1:
        return 1 if approve_pos < reject_pos else 0
    
    # If only one is present
    if approve_pos != -1:
        return 1
    if reject_pos != -1:
        return 0
        
    return None