"""
Small compatibility helpers for HelioX monitoring.

Why this exists:
- Some heliox checkouts do not ship the higher-level Python wrappers (heliox_sim.*).
- The low-level pybind API (`add_monitor_with_array`, `get_monitor_handle_with_array`, `get_monitor_data`)
  is stable enough to build a minimal wrapper here.

This wrapper also tries to use the CoreNEURON/HELIOX "row" index encoded in NEURON
data handles (e.g. str(seg._ref_v) contains 'row=<int>') which is often a better
index than `segment.node_index()` for exported CoreNEURON models.
"""

from __future__ import annotations

import re
import os
from dataclasses import dataclass


def try_parse_row_from_ref(ref) -> int | None:
    """Best-effort parse of `row=<int>` from NEURON `_ref_*` handle string."""
    try:
        match = re.search(r"row=(\d+)", str(ref))
        if match:
            return int(match.group(1))
    except Exception:
        return None
    return None


@dataclass
class HelioXMonitor:
    """
    Minimal monitor wrapper that can survive pending (-1) handles.

    The handle is considered "pending" until after `client.load_model()`, at which
    point `get_monitor_handle_with_array(...)` should resolve it.
    """

    client: object
    mech_name: str
    var_name: str
    node_or_mech_idx: int
    array_index: int = 0
    monitor_id: int = -1

    def register(self) -> int:
        self.monitor_id = int(
            self.client.add_monitor_with_array(
                self.mech_name, self.var_name, int(self.node_or_mech_idx), int(self.array_index)
            )
        )
        return self.monitor_id

    def resolve_handle(self) -> int:
        if self.monitor_id != -1:
            return int(self.monitor_id)
        real_handle = int(
            self.client.get_monitor_handle_with_array(
                self.mech_name, self.var_name, int(self.node_or_mech_idx), int(self.array_index)
            )
        )
        if real_handle != -1:
            self.monitor_id = real_handle
        return int(self.monitor_id)

    def get_data(self):
        handle = self.resolve_handle()
        if handle == -1:
            raise RuntimeError(
                f"HelioX monitor handle is still pending (-1): mech={self.mech_name} var={self.var_name} "
                f"idx={self.node_or_mech_idx} array_index={self.array_index}. "
                "Call client.load_model() before reading monitor data."
            )
        return self.client.get_monitor_data(handle)


def make_segment_monitor(segment, var_name: str, client, array_index: int = 0) -> HelioXMonitor:
    """
    Create a HelioX monitor for a NEURON Segment variable (e.g. soma(0.5).v).

    For segment variables, HelioX expects a "row" index when available; fall back
    to `node_index()` for older NEURON builds/handles.
    """
    ref_name = f"_ref_{var_name}"
    row = None
    debug = os.environ.get("HELIOX_MONITOR_DEBUG", "0").strip().lower() in {"1", "true", "yes", "on"}
    try:
        ref = getattr(segment, ref_name)
        if debug:
            print(f"HelioX monitor: {ref_name} handle str={str(ref)}")
        row = try_parse_row_from_ref(ref)
    except Exception:
        row = None

    if row is None:
        row = int(segment.node_index())
        if debug:
            print(f"HelioX monitor: row parse failed, fallback node_index={row}")
    elif debug:
        print(f"HelioX monitor: parsed row={row} (node_index={int(segment.node_index())})")

    mon = HelioXMonitor(
        client=client,
        mech_name="global",
        var_name=var_name,
        node_or_mech_idx=int(row),
        array_index=int(array_index),
    )
    mon.register()
    return mon
