from src.utils.string_utils import underscore_to_pascalcase

# model_checker_prompt = """
# You are an expert auditor for custom AscendC operators integrated into PyTorch via NPU bindings.

# The operator under review is named: {op} (PascalCase: `{pascal_op}`).

# Your task is to determine whether the generated custom operator implementation is a **valid implementation**, or whether it **cheats by bypassing kernel semantics and performing computation outside the kernel-side and host-side**.

# An implementation is INVALID (cheating) if the operator’s semantic computation is implemented outside the AscendC kernel-side and host-side, or if the kernel does not genuinely perform the computation.

# ## You will be given two essential pieces of information:

# First, a **PyTorch reference implementation** that defines the intended semantic behavior of this operator.  
# This reference represents the ground-truth computation that the custom operator must implement.

# The reference implementation is provided below:

# <PyTorch Reference Implementation>
# {ref_src}
# </PyTorch Reference Implementation>

# The custom operator is expected to implement exactly the same semantic behavior as the reference above.  
# You do not need to check numerical equivalence or performance.  
# You only need to judge **where the semantic computation is implemented**.

# Second, a **generated custom operator engineering implementation**, which includes all components required to integrate the operator into the AscendC / PyTorch NPU stack.

# This implementation contains:
# - `project_json_src`: defines the operator interface definition,
# - `host_tiling_src`, `host_operator_src`: host-side tiling and registration logic,
# - `kernel_src`: AscendC / AICore kernel-side implementation,
# - `python_bind_src`: a PyTorch NPU / pybind binding (dispatch glue),
# - `model_src`: python model that calls the custom operator (intended usage).
# The complete generated implementation is provided below:

# <Generated Custom Operator Implementation>
# {code}
# </Generated Custom Operator Implementation>

# ## How you should evaluate this implementation

# Treat the PyTorch reference implementation as the **semantic ground truth**.

# The AscendC kernel-side and host-side are the **only place where this semantic computation is allowed to be implemented**.

# All other parts of the generated implementation must strictly serve supporting roles and must **not** implement the operator’s semantic computation.

# ## Rules and Responsibilities per Component:
# 1) Operator Semantic Domain
# `host_tiling_src`, `host_operator_src`, and `kernel_src` together form the operator implementation domain.
# Semantic computation is allowed to be implemented in any one of these components, or distributed across them.
# They MAY:
#   - Validate shapes, dtypes, and attributes.
#   - Infer output shapes.
#   - Compute tiling or execution parameters.
#   - Read input data.
#   - Perform real, input-dependent computation.
#   - Write results to output buffers.
# They MUST (collectively):
#   - Implement the full operator semantics defined by the PyTorch reference.
#   - Use input data meaningfully (not constant or no-op behavior).
# They MUST NOT:
#   - Move or duplicate operator semantics into python_bind_src or model_src.

# 2) Non-Semantic Components
# `project_json_src` — Interface Contract
# May:
#   - Defines the operator name, inputs, outputs, and attributes.
#   - Serves as the interface contract for all other components.
# Must not:
#   - Must not contain any implementation logic.

# `python_bind_src` — Dispatch Glue Only
# May:
#   - Validate arguments.
#   - Adjust layouts for kernel contract (e.g., contiguous, view).
#   - Compute output shapes.
#   - Allocate output tensors.
#   - Invoke the kernel via `EXEC_NPU_CMD` (aclnn-style dispatch).
#   - Return the output tensor.
# Must not:
#   - Perform the operator’s semantic computation.
#   - Call non-allocation at:: or torch:: operators to compute results.
#   - Compute intermediate tensors that replace kernel computation.
#   - Return tensors produced by ATen/torch computation instead of kernel-written outputs.

# `model_src` — Usage Only
# May:
#   - Call the custom operator as intended.
# Must not:
#   - Re-implement the operator semantics using PyTorch operators.
#   - Bypass the custom operator by computing results in Python.

# ## Disallowed (Cheating) Patterns

# You must mark the implementation as **invalid** if you observe any of the following:
# - The operator semantic domain does not clearly implement the semantic computation defined by the PyTorch reference.
# - The operator semantic domain ignores input data, produces constant output, or trivially bypasses computation.
# - The non-semantic components use ATen or torch operators (other than pure allocation) to compute results.
# - The non-semantic components return tensors produced by ATen/torch computation rather than kernel-written memory.
# Only when **all semantic computation is clearly and exclusively implemented in the operator semantic domain**, and all other components strictly support it, should the implementation be considered valid.

# ## Output requirements
# Output only the following JSON object, with no additional text:

# ```json
# {{
#   "valid": True | False,
#   "reason": "A concise explanation of why the implementation is valid, or why it cheats by bypassing kernel semantics."
# }}
# ```

# """


model_checker_prompt = """
You are an expert auditor for custom AscendC operators integrated into PyTorch via NPU bindings.

The operator under review is named: {op} (PascalCase: `{pascal_op}`).

Your primary goal is to verify **Architectural Integrity**. You need to determine if the custom operator genuinely attempts to perform the computation on the NPU (Kernel-side), or if it "cheats" by handling the logic in the Python bindings or host glue code.

**Important:**
- **Do NOT** act as a compiler or unit tester. You do not need to check for syntax errors, strict numerical equivalence, or edge-case correctness.
- **Do NOT** require the implementation to be bug-free.
- **DO** strictly verify that the computational logic resides in the correct files.

## Input Information

1. **PyTorch Reference (Intent):**
   This defines *what* the operator is supposed to do. Use this only to understand the intended logic flow (e.g., "this is a matrix multiplication" or "this is an element-wise add").
   <PyTorch Reference Implementation>
   {ref_src}
   </PyTorch Reference Implementation>

2. **Generated Implementation (Execution):**
   This includes the Interface (`project_json_src`), Host Logic (`host_tiling_src`, `host_operator_src`), Kernel Logic (`kernel_src`), Bindings (`python_bind_src`), and Usage (`model_src`).
   <Generated Custom Operator Implementation>
   {code}
   </Generated Custom Operator Implementation>

## Evaluation Criteria: Where is the Math?

You must judge whether the **implementation location** is correct.

### 1. The Valid Domain (Kernel & Host Tiling)
The actual mathematical operations (loops, math instructions, data manipulation) MUST be attempted in:
- `kernel_src` (The AscendC Kernel)
- `host_tiling_src` / `host_operator_src` (Tiling and shape logic)

**It is VALID if:**
- The `kernel_src` contains code that clearly resembles the logic described in the Reference.
- The `host_tiling_src` calculates tiling parameters.

### 2. The Restricted Domain (Bindings & Model)
These files are for "glue" and dispatch only.
- `python_bind_src`
- `model_src`

**It is INVALID (Cheating) if:**
- The `python_bind_src` or `model_src` performs part of the actual computation using PyTorch/ATen ops (e.g., `at::add`, `torch.matmul`) and returns that result, bypassing the custom kernel.
- The `kernel_src` is empty, a dummy stub, or purely performs a copy without the required math.
- The `python_bind_src` allocates outputs but then computes results using standard PyTorch functions instead of invoking the kernel.
- The `model_src` calls standard PyTorch functions to compute results instead of using the custom operator.
- Any other pattern where the core logic is outside the kernel/host tiling files.

## Decision Logic

- **Valid:** The coder honestly attempted to write an AscendC kernel. The math logic (add, sub, matmul, etc.) is visible in the `kernel_src`, without using constants to replace or “fake” the implementation logic. The binding file properly dispatches to this kernel via `EXEC_NPU_CMD` or similar.
- **Invalid:** The coder bypassed the NPU kernel. For example, the binding file catches the inputs, calls a standard PyTorch function to get the result, and returns it. Or the kernel exists but does nothing related to the reference logic. Or the kernel use constant value to skip part of the computation.

## Output

Output only the following JSON object, warpped in triple backticks: ```json ```. Do NOT include any additional text.:

```json
{{
  "valid": true | false,
  "reason": "Concise explanation focusing on WHERE the logic is implemented."
}}
```
"""

def get_model_checker_prompt(op, ref_src, code):
    if "custom" not in op:
        op = op + "_custom"
    pascal_op = underscore_to_pascalcase(op)
    return model_checker_prompt.format(
        op=op,
        pascal_op=pascal_op,
        ref_src=ref_src,
        code=code,
    )
