"""
Reasoning Module for Robot Ontology

Implements:
A) Capability inference (deterministic + optional LLM)
B) Skill substitution inference (equivalence + decomposition)
C) Object mapping inference for scene understanding
"""

import json
import re
import difflib
from typing import Dict, List, Any, Optional, Set, Tuple
from pathlib import Path
from collections import defaultdict

from ontology import (
    TripleGraph, Predicates,
    robot_id, skill_id, cap_id, obj_id
)


# Standard capability tags
CAPABILITIES = {
    "MotionPlanning": "Can plan collision-free paths",
    "Grasping": "Can grasp objects with end effector",
    "MoveEE": "Can move end effector to target pose",
    "PickPlace": "Can pick and place objects",
    "CartesianControl": "Can control end effector in Cartesian space",
    "JointControl": "Can control individual joints",
    "ForceControl": "Can control interaction forces",
    "VisualServoing": "Can use visual feedback for control",
    "DualArm": "Has two or more arms",
    "MobileBase": "Has mobile base for navigation"
}


def predicate_to_string(pred_item) -> str:
    """Convert predicate item (dict or string) to string representation."""
    if isinstance(pred_item, str):
        return pred_item
    if isinstance(pred_item, dict):
        predicate = pred_item.get('predicate', '')
        args = pred_item.get('args', [])
        positive = pred_item.get('positive', True)
        pred_str = f"{predicate}({', '.join(args)})"
        if not positive:
            pred_str = f"not({pred_str})"
        return pred_str
    return str(pred_item)


def normalize_predicate(pred) -> str:
    """Normalize a predicate (string or dict) for comparison."""
    # Convert dict to string if needed
    pred_str = predicate_to_string(pred)
    # Remove variable placeholders and normalize
    normalized = pred_str.lower()
    normalized = re.sub(r'\?\w+', '?VAR', normalized)
    normalized = re.sub(r'\s+', ' ', normalized).strip()
    return normalized


def extract_predicate_signature(pred) -> Tuple[str, int]:
    """Extract predicate name and arity from a predicate (string or dict)."""
    # Handle dict format
    if isinstance(pred, dict):
        name = pred.get('predicate', '').lower()
        args = pred.get('args', [])
        return (name, len(args))

    # Handle string format: e.g., "holding(?obj)" -> ("holding", 1)
    pred_str = str(pred)
    match = re.match(r'(\w+)\s*\(([^)]*)\)', pred_str)
    if match:
        name = match.group(1).lower()
        args = [a.strip() for a in match.group(2).split(',') if a.strip()]
        return (name, len(args))
    return (pred_str.lower().strip(), 0)


def compute_effect_similarity(effects1: List[str], effects2: List[str]) -> float:
    """Return similarity between predicate lists using relaxed matching."""
    if not effects1 or not effects2:
        return 0.0

    # Normalize and extract signatures
    sigs1 = [extract_predicate_signature(e) for e in effects1]
    sigs2 = [extract_predicate_signature(e) for e in effects2]

    # Count matching signatures
    matched = 0
    used = set()

    for sig1 in sigs1:
        best_match = -1
        best_score = 0.0

        for i, sig2 in enumerate(sigs2):
            if i in used:
                continue

            # Same predicate name and arity
            if sig1[0] == sig2[0] and sig1[1] == sig2[1]:
                score = 1.0
            else:
                # Partial match using string similarity
                score = difflib.SequenceMatcher(None, sig1[0], sig2[0]).ratio()
                if sig1[1] != sig2[1]:
                    score *= 0.5

            if score > best_score:
                best_score = score
                best_match = i

        if best_match >= 0 and best_score > 0.5:
            matched += best_score
            used.add(best_match)

    # Jaccard-like score
    total = len(sigs1) + len(sigs2) - matched
    return matched / total if total > 0 else 0.0


def compute_capability_overlap(caps1: List[str], caps2: List[str]) -> float:
    """Compute Jaccard overlap between required capability lists."""
    set1 = {c.lower() for c in caps1 or []}
    set2 = {c.lower() for c in caps2 or []}
    if not set1 and not set2:
        return 0.0

    intersection = set1 & set2
    union = set1 | set2
    return len(intersection) / len(union) if union else 0.0


def infer_capabilities(
    graph: TripleGraph,
    robot_id_str: str,
    robot_summary: Dict[str, Any],
    skills_json: List[Dict[str, Any]],
    llm: Optional[Any] = None,
    output_dir: str = None
) -> Dict[str, Any]:
    """
    Infer robot capabilities using deterministic rules and optional LLM.

    Deterministic rules:
    - hasEndEffector OR hasGroup => MotionPlanning
    - skills require Grasping OR gripper in end effector name => Grasping
    - skills include 'move' => MoveEE
    - skills include 'pick'/'place' => PickPlace

    LLM (optional):
    - Propose additional capabilities with confidence
    - Never overwrites deterministic results

    Returns:
    {
        "robot": str,
        "capabilities_deterministic": [...],
        "capabilities_llm_proposed": [...],
        "capabilities_final": [...],
        "llm_notes": str
    }
    """
    capabilities_det = set()

    # Check for groups/end effectors
    has_group = len(graph.query(subject=robot_id_str, predicate=Predicates.HAS_GROUP)) > 0
    has_ee = len(graph.query(subject=robot_id_str, predicate=Predicates.HAS_END_EFFECTOR)) > 0

    if has_group or has_ee:
        capabilities_det.add("MotionPlanning")

    # Check for gripper-like end effector
    gripper_keywords = ['gripper', 'hand', 'finger', 'grasp']
    ee_triples = graph.query(subject=robot_id_str, predicate=Predicates.HAS_END_EFFECTOR)
    for _, _, ee_id in ee_triples:
        if any(kw in ee_id.lower() for kw in gripper_keywords):
            capabilities_det.add("Grasping")
            break

    # Check skills
    skill_names = [s.get('name', '').lower() for s in skills_json]
    skill_required_caps = set()
    for skill in skills_json:
        for cap in skill.get('required_capabilities', []):
            skill_required_caps.add(cap)

    # Grasping from skills
    if 'Grasping' in skill_required_caps:
        capabilities_det.add("Grasping")
    if any('grasp' in name or 'grip' in name for name in skill_names):
        capabilities_det.add("Grasping")

    # MoveEE from skills
    if any('move' in name for name in skill_names):
        capabilities_det.add("MoveEE")

    # PickPlace from skills
    if any('pick' in name or 'place' in name for name in skill_names):
        capabilities_det.add("PickPlace")

    # CartesianControl if move_ee or cartesian in skills
    if any('cartesian' in name or 'move_ee' in name for name in skill_names):
        capabilities_det.add("CartesianControl")

    # JointControl if joint in skills or has joints
    dof_triples = graph.query(subject=robot_id_str, predicate=Predicates.HAS_DOF)
    if dof_triples:
        dof = int(dof_triples[0][2])
        if dof > 0:
            capabilities_det.add("JointControl")

    # Add all required capabilities from skills
    for cap in skill_required_caps:
        if cap in CAPABILITIES:
            capabilities_det.add(cap)

    # LLM step (optional)
    llm_proposed = []
    llm_notes = "LLM not used"

    if llm is not None:
        # Prepare context for LLM
        context = {
            "robot": robot_id_str,
            "end_effectors": [t[2] for t in ee_triples],
            "skills": skill_names,
            "dof": dof_triples[0][2] if dof_triples else "unknown",
            "known_capabilities": list(capabilities_det)
        }

        system_prompt = """You are a robot capability analyzer. Given robot info, propose additional capabilities.
Return ONLY valid JSON with format: {"proposed": [{"capability": str, "confidence": float, "rationale": str}]}
Only propose capabilities from: """ + ", ".join(CAPABILITIES.keys())

        user_prompt = f"Analyze this robot and propose additional capabilities:\n{json.dumps(context, indent=2)}"

        try:
            result = llm.generate_json(system_prompt, user_prompt)
            if "proposed" in result:
                for prop in result["proposed"]:
                    cap = prop.get("capability", "")
                    conf = prop.get("confidence", 0)

                    # Validate: must be known capability and not already determined
                    if cap in CAPABILITIES and cap not in capabilities_det and conf > 0.5:
                        # Basic validation: check for evidence in robot summary
                        evidence_found = False
                        cap_lower = cap.lower()

                        # Check in skill descriptions
                        for skill in skills_json:
                            desc = skill.get('description', '').lower()
                            if cap_lower in desc or any(kw in desc for kw in cap_lower.split()):
                                evidence_found = True
                                break

                        if evidence_found:
                            llm_proposed.append({
                                "capability": cap,
                                "confidence": conf,
                                "rationale": prop.get("rationale", ""),
                                "validated": True
                            })
                        else:
                            llm_proposed.append({
                                "capability": cap,
                                "confidence": conf,
                                "rationale": prop.get("rationale", ""),
                                "validated": False,
                                "rejection_reason": "No evidence found"
                            })

            llm_notes = "LLM proposals processed"
        except Exception as e:
            llm_notes = f"LLM error: {str(e)}"

    # Final capabilities
    capabilities_final = list(capabilities_det)
    for prop in llm_proposed:
        if prop.get("validated") and prop["capability"] not in capabilities_final:
            capabilities_final.append(prop["capability"])

    # Add to graph
    for cap in capabilities_final:
        graph.add(robot_id_str, Predicates.HAS_CAPABILITY, cap_id(cap))

    result = {
        "robot": robot_id_str,
        "capabilities_deterministic": sorted(list(capabilities_det)),
        "capabilities_llm_proposed": llm_proposed,
        "capabilities_final": sorted(capabilities_final),
        "llm_notes": llm_notes
    }

    if output_dir:
        with open(Path(output_dir) / "capabilities.json", 'w') as f:
            json.dump(result, f, indent=2)

    return result


def is_semantically_opposite_by_name(nameA: str, nameB: str) -> bool:
    """
    Check if two skill names represent semantically opposite actions by name.
    E.g., open_gripper vs close_gripper, open_ur5_ee vs close_ur5_ee
    """
    nameA_lower = nameA.lower()
    nameB_lower = nameB.lower()

    # Define opposite action pairs
    opposite_keywords = [
        ('open', 'close'),
        ('grasp', 'release'),
        ('pick', 'place'),
    ]

    for kw1, kw2 in opposite_keywords:
        # Check if one name contains kw1 and the other contains kw2
        a_has_kw1 = kw1 in nameA_lower
        a_has_kw2 = kw2 in nameA_lower
        b_has_kw1 = kw1 in nameB_lower
        b_has_kw2 = kw2 in nameB_lower

        # Opposite if A has kw1 and B has kw2, or vice versa
        if (a_has_kw1 and b_has_kw2) or (a_has_kw2 and b_has_kw1):
            # But make sure they're not the SAME keyword in both
            if not (a_has_kw1 and b_has_kw1) and not (a_has_kw2 and b_has_kw2):
                return True

    return False


def has_opposite_gripper_effects(effectsA: List[Dict], effectsB: List[Dict]) -> bool:
    """
    Check if two skills have opposite gripper state effects.
    E.g., one has gripper_closed=true and the other has gripper_open=true.
    """
    def get_gripper_state(effects: List[Dict]) -> Optional[str]:
        for e in effects:
            pred = e.get('predicate', '')
            positive = e.get('positive', True)
            if 'gripper_closed' in pred and positive:
                return 'closed'
            if 'gripper_open' in pred and positive:
                return 'open'
        return None

    stateA = get_gripper_state(effectsA)
    stateB = get_gripper_state(effectsB)

    # If both have gripper effects and they're opposite
    if stateA and stateB and stateA != stateB:
        return True
    return False


def is_semantically_opposite(nameA: str, nameB: str,
                              effectsA: List[Dict] = None,
                              effectsB: List[Dict] = None) -> bool:
    """
    Check if two skills are semantically opposite by name OR by effects.

    Args:
        nameA: Name of first skill
        nameB: Name of second skill
        effectsA: Effects of first skill (optional, for effect-based check)
        effectsB: Effects of second skill (optional, for effect-based check)

    Returns:
        True if skills are semantically opposite
    """
    # Check by name first
    if is_semantically_opposite_by_name(nameA, nameB):
        return True

    # Check by effects if provided
    if effectsA is not None and effectsB is not None:
        if has_opposite_gripper_effects(effectsA, effectsB):
            return True

    return False


def collect_unmapped_primitives(
    skillsA: List[Dict[str, Any]],
    skillsB: List[Dict[str, Any]],
    equivalences: List[Dict[str, Any]],
    robotA: str,
    robotB: str
) -> Tuple[List[Dict[str, Any]], List[Dict[str, Any]]]:
    """
    Collect primitives that were not mapped by deterministic matching.

    Args:
        skillsA: Skills from robot A
        skillsB: Skills from robot B
        equivalences: Already found equivalences
        robotA: Robot A name
        robotB: Robot B name

    Returns:
        (unmapped_a, unmapped_b): Lists of unmapped skills from each robot
    """
    # Extract mapped skill names
    mapped_a = set()
    mapped_b = set()

    for eq in equivalences:
        # Format: "robotA:skill_name"
        skill_a = eq['skill_a'].split(':')[-1]
        skill_b = eq['skill_b'].split(':')[-1]
        mapped_a.add(skill_a)
        mapped_b.add(skill_b)

    unmapped_a = [s for s in skillsA if s['name'] not in mapped_a]
    unmapped_b = [s for s in skillsB if s['name'] not in mapped_b]

    return unmapped_a, unmapped_b


def ask_llm_unmapped_mapping(
    unmapped_a: List[Dict[str, Any]],
    candidates_b: List[Dict[str, Any]],
    robotA: str,
    robotB: str,
    llm: Any
) -> List[Dict[str, Any]]:
    """
    Ask LLM to find mappings for unmapped primitives from robot A.

    Supports N:1 mapping - multiple skills from A can map to the same skill in B.
    This handles cases where robot B has fewer but more general skills.

    Args:
        unmapped_a: Unmapped skills from robot A
        candidates_b: ALL skills from robot B (candidates for mapping)
        robotA: Robot A name
        robotB: Robot B name
        llm: LLM instance

    Returns:
        List of proposed equivalences from LLM
    """
    if not unmapped_a or not candidates_b:
        return []

    # Format skills for prompt
    def format_skill(skill: Dict) -> Dict:
        return {
            "name": skill['name'],
            "description": skill.get('description', ''),
            "parameters": [
                {"name": p['name'], "type": p.get('type', ''), "semantic": p.get('semantic', '')}
                for p in skill.get('parameters', [])
            ],
            "effects": [predicate_to_string(e) for e in skill.get('effects', [])],
            "preconditions": [predicate_to_string(p) for p in skill.get('preconditions', [])]
        }

    context = {
        "robot_a": robotA,
        "robot_b": robotB,
        "unmapped_skills_a": [format_skill(s) for s in unmapped_a],
        "candidate_skills_b": [format_skill(s) for s in candidates_b]
    }

    system_prompt = """You are a robot skill mapping expert. Your task is to find functionally equivalent skills between two robots.

Two skills are functionally equivalent if they achieve the same goal, even if:
- They have different parameter interfaces (e.g., low-level axis specification vs high-level direction names)
- They use different abstractions (e.g., mathematical vs semantic)

Focus on the FUNCTIONAL PURPOSE, not the parameter structure.

Return ONLY valid JSON with format:
{
  "equivalences": [
    {
      "skill_a": "skill_name_from_robot_a",
      "skill_b": "skill_name_from_robot_b",
      "confidence": 0.0-1.0,
      "rationale": "explanation of why these are functionally equivalent"
    }
  ]
}

Only propose mappings you are confident about (confidence >= 0.7)."""

    user_prompt = f"""The following skills from {robotA} were NOT matched by deterministic methods.
Find functionally equivalent skills from {robotB}'s skill set (N:1 mapping is allowed):

{json.dumps(context, indent=2)}"""

    try:
        print(f"[ask_llm_unmapped_mapping] Querying LLM for {len(unmapped_a)} unmapped skills...")
        result = llm.generate_json(system_prompt, user_prompt, max_new_tokens=1024)
        print(f"[ask_llm_unmapped_mapping] LLM response: {result}")
        equivalences = result.get("equivalences", [])
        print(f"[ask_llm_unmapped_mapping] Found {len(equivalences)} proposed equivalences")
        return equivalences
    except Exception as e:
        print(f"[ask_llm_unmapped_mapping] LLM error: {e}")
        return []


def infer_skill_substitutions(
    graph: TripleGraph,
    robotA: str,
    robotB: str,
    skillsA: List[Dict[str, Any]],
    skillsB: List[Dict[str, Any]],
    llm: Optional[Any] = None,
    output_dir: str = None
) -> Dict[str, Any]:
    """
    Infer skill equivalences and decompositions between two robots.

    Deterministic:
    - Compare effects/preconditions with string similarity
    - Effects match > 0.85 => equivalent
    - Decomposition: find sequence of skills that covers target effects

    LLM (optional):
    - First: propose candidates, validated by deterministic scorer
    - Then: find mappings for unmapped primitives (functional equivalence)

    Returns dict with equivalences, decompositions, accepted/rejected lists
    """
    equivalences = []
    decompositions = []
    rejected = []

    # Build skill lookup
    skills_by_robot = {
        robotA: {s['name']: s for s in skillsA},
        robotB: {s['name']: s for s in skillsB}
    }

    # Deterministic equivalence detection
    for skillA in skillsA:
        nameA = skillA['name']
        effectsA = skillA.get('effects', [])
        precondsA = skillA.get('preconditions', [])
        capsA = skillA.get('required_capabilities', [])

        for skillB in skillsB:
            nameB = skillB['name']
            effectsB = skillB.get('effects', [])
            precondsB = skillB.get('preconditions', [])
            capsB = skillB.get('required_capabilities', [])

            # Skip semantically opposite skills (e.g., open_gripper vs close_ur5_ee, pick vs release)
            if is_semantically_opposite(nameA, nameB, effectsA, effectsB):
                continue

            # Compare effects
            effect_sim = compute_effect_similarity(effectsA, effectsB)

            # Compare preconditions
            precond_sim = compute_effect_similarity(precondsA, precondsB)

            # Compare required capabilities
            cap_sim = compute_capability_overlap(capsA, capsB)

            # Combined score
            combined_score = (
                effect_sim * 0.5 +
                precond_sim * 0.3 +
                cap_sim * 0.2
            )

            # Require some structure overlap to avoid false positives
            has_structure_match = (
                effect_sim >= 0.4 or
                precond_sim >= 0.4
            )

            if combined_score >= 0.65 and has_structure_match:
                equivalences.append({
                    "skill_a": f"{robotA}:{nameA}",
                    "skill_b": f"{robotB}:{nameB}",
                    "score": round(combined_score, 4),
                    "method": "deterministic",
                    "effect_similarity": round(effect_sim, 4),
                    "precond_similarity": round(precond_sim, 4),
                    "capability_overlap": round(cap_sim, 4)
                })

                # Add to graph
                graph.add(
                    skill_id(robotA, nameA),
                    Predicates.EQUIV_SKILL,
                    skill_id(robotB, nameB)
                )

    # Deterministic decomposition detection
    # Try to decompose complex skills (pick, place) into primitives

    # Define common decomposition patterns
    decomp_patterns = {
        'pick': [
            ['move', 'grasp_object'],
            ['move', 'close_gripper'],
            ['move_ee', 'close_gripper']
        ],
        'place': [
            ['move', 'open_gripper'],
            ['place_object'],
            ['move_ee', 'open_gripper']
        ],
        'pick_and_place': [
            ['pick', 'move', 'place'],
            ['grasp_object', 'move', 'place_object']
        ],
        'push': [
            ['close_gripper', 'move'],
            ['close_gripper', 'move_ee']
        ]
    }

    for robot, skills_list in [(robotA, skillsA), (robotB, skillsB)]:
        skill_names_set = set(s['name'].lower() for s in skills_list)
        skill_by_name = {s['name'].lower(): s for s in skills_list}

        for target_skill in skills_list:
            target_name = target_skill['name'].lower()
            target_effects = target_skill.get('effects', [])

            # Check if any decomposition pattern matches
            for pattern_name, patterns in decomp_patterns.items():
                if pattern_name not in target_name:
                    continue

                for pattern in patterns:
                    # Check if all skills in pattern exist
                    if all(p in skill_names_set for p in pattern):
                        # Verify effect coverage
                        combined_effects = []
                        for p in pattern:
                            combined_effects.extend(skill_by_name[p].get('effects', []))

                        coverage = compute_effect_similarity(combined_effects, target_effects)

                        if coverage >= 0.5:
                            decompositions.append({
                                "target_skill": f"{robot}:{target_skill['name']}",
                                "decomposition": [f"{robot}:{p}" for p in pattern],
                                "coverage_score": round(coverage, 4),
                                "method": "deterministic"
                            })

                            # Add to graph
                            graph.add(
                                skill_id(robot, target_skill['name']),
                                Predicates.DECOMP_SKILL,
                                json.dumps(pattern)
                            )
                            break

    # LLM step (optional)
    llm_proposals = []

    if llm is not None:
        # Prepare context
        context = {
            "robot_a": robotA,
            "robot_b": robotB,
            "skills_a": [{"name": s['name'], "effects": s.get('effects', [])} for s in skillsA],
            "skills_b": [{"name": s['name'], "effects": s.get('effects', [])} for s in skillsB],
            "found_equivalences": len(equivalences),
            "found_decompositions": len(decompositions)
        }

        system_prompt = """You are a skill substitution analyzer. Propose skill equivalences or decompositions.
Return ONLY valid JSON with format:
{
  "equivalences": [{"skill_a": str, "skill_b": str, "rationale": str}],
  "decompositions": [{"target": str, "sequence": [str], "rationale": str}]
}"""

        user_prompt = f"Find skill substitutions:\n{json.dumps(context, indent=2)}"

        try:
            result = llm.generate_json(system_prompt, user_prompt)

            # Validate LLM equivalences
            for eq in result.get("equivalences", []):
                skill_a = eq.get("skill_a", "").split(":")[-1]
                skill_b = eq.get("skill_b", "").split(":")[-1]

                # Verify skills exist
                if skill_a in skills_by_robot.get(robotA, {}) and skill_b in skills_by_robot.get(robotB, {}):
                    # Compute actual similarity
                    sa = skills_by_robot[robotA][skill_a]
                    sb = skills_by_robot[robotB][skill_b]
                    sim = compute_effect_similarity(sa.get('effects', []), sb.get('effects', []))

                    if sim >= 0.5:
                        llm_proposals.append({
                            "type": "equivalence",
                            "proposal": eq,
                            "validated": True,
                            "computed_score": round(sim, 4)
                        })
                        # Add if not already found
                        if not any(e['skill_a'].endswith(skill_a) and e['skill_b'].endswith(skill_b) for e in equivalences):
                            equivalences.append({
                                "skill_a": f"{robotA}:{skill_a}",
                                "skill_b": f"{robotB}:{skill_b}",
                                "score": round(sim, 4),
                                "method": "llm_validated"
                            })
                    else:
                        llm_proposals.append({
                            "type": "equivalence",
                            "proposal": eq,
                            "validated": False,
                            "rejection_reason": f"Low similarity: {sim:.2f}"
                        })
                        rejected.append(eq)

        except Exception as e:
            llm_proposals.append({"error": str(e)})

    # LLM step 2: Find mappings for unmapped primitives
    llm_unmapped_proposals = []

    if llm is not None:
        # Collect unmapped primitives
        unmapped_a, unmapped_b = collect_unmapped_primitives(
            skillsA, skillsB, equivalences, robotA, robotB
        )

        # Only need unmapped_a to have items - we'll search ALL of skillsB for matches
        # This allows N:1 mapping (multiple A skills -> same B skill)
        if unmapped_a:
            print(f"[infer_skill_substitutions] Found {len(unmapped_a)} unmapped skills from {robotA}")
            print(f"[infer_skill_substitutions] Unmapped from {robotA}: {[s['name'] for s in unmapped_a]}")

            # Ask LLM for functional equivalences - provide ALL skillsB as candidates
            llm_equiv_proposals = ask_llm_unmapped_mapping(
                unmapped_a, skillsB, robotA, robotB, llm
            )

            for proposal in llm_equiv_proposals:
                skill_a = proposal.get("skill_a", "")
                skill_b = proposal.get("skill_b", "")
                confidence = proposal.get("confidence", 0)
                rationale = proposal.get("rationale", "")

                # Validate: skill_a must be unmapped, skill_b must exist in skillsB
                skill_a_exists = any(s['name'] == skill_a for s in unmapped_a)
                skill_b_exists = any(s['name'] == skill_b for s in skillsB)

                if skill_a_exists and skill_b_exists and confidence >= 0.7:
                    # Accept the proposal
                    equivalences.append({
                        "skill_a": f"{robotA}:{skill_a}",
                        "skill_b": f"{robotB}:{skill_b}",
                        "score": round(confidence, 4),
                        "method": "llm_unmapped",
                        "rationale": rationale
                    })

                    # Add to graph
                    graph.add(
                        skill_id(robotA, skill_a),
                        Predicates.EQUIV_SKILL,
                        skill_id(robotB, skill_b)
                    )

                    llm_unmapped_proposals.append({
                        "type": "unmapped_equivalence",
                        "proposal": proposal,
                        "validated": True
                    })

                    print(f"[infer_skill_substitutions] LLM found unmapped equivalence: "
                          f"{skill_a} <=> {skill_b} (confidence: {confidence})")
                else:
                    reason = []
                    if not skill_a_exists:
                        reason.append(f"skill_a '{skill_a}' not in unmapped list")
                    if not skill_b_exists:
                        reason.append(f"skill_b '{skill_b}' not in unmapped list")
                    if confidence < 0.7:
                        reason.append(f"low confidence: {confidence}")

                    llm_unmapped_proposals.append({
                        "type": "unmapped_equivalence",
                        "proposal": proposal,
                        "validated": False,
                        "rejection_reason": "; ".join(reason)
                    })
                    rejected.append(proposal)

    result = {
        "equivalences": equivalences,
        "decompositions": decompositions,
        "llm_proposals": llm_proposals,
        "llm_unmapped_proposals": llm_unmapped_proposals,
        "rejected": rejected,
        "summary": {
            "total_equivalences": len(equivalences),
            "total_decompositions": len(decompositions),
            "deterministic_equivalences": len([e for e in equivalences if e.get('method') == 'deterministic']),
            "llm_validated": len([p for p in llm_proposals if p.get("validated")]),
            "llm_unmapped_found": len([p for p in llm_unmapped_proposals if p.get("validated")]),
            "llm_rejected": len(rejected)
        }
    }

    if output_dir:
        with open(Path(output_dir) / "skill_substitutions.json", 'w') as f:
            json.dump(result, f, indent=2)

    return result


def infer_object_mappings(
    graph: TripleGraph,
    scene_json: Dict[str, Any],
    llm: Optional[Any] = None,
    output_dir: str = None
) -> Dict[str, Any]:
    """
    Infer object mappings and similarities in a scene.

    Deterministic:
    - Type match + name similarity + affordance overlap + relation context

    LLM (optional):
    - Propose merges/aliases with validation

    Returns mappings with top-3 candidates per object
    """
    objects = scene_json.get('objects', [])

    # Build object features
    obj_features = {}
    for obj in objects:
        obj_id_str = obj['id']
        obj_features[obj_id_str] = {
            'type': obj.get('type', ''),
            'name': obj_id_str,
            'affordances': set(obj.get('affordances', [])),
            'relations': []
        }

    # Extract relations
    for rel in scene_json.get('relations', []):
        sub = rel.get('sub', '')
        pred = rel.get('pred', '')
        obj = rel.get('obj', '')
        if sub in obj_features:
            obj_features[sub]['relations'].append((pred, obj))
        if obj in obj_features:
            obj_features[obj]['relations'].append((f"inv_{pred}", sub))

    # Compute pairwise similarities
    mappings = {}
    obj_ids = list(obj_features.keys())

    for i, obj1 in enumerate(obj_ids):
        candidates = []
        f1 = obj_features[obj1]

        for j, obj2 in enumerate(obj_ids):
            if i == j:
                continue

            f2 = obj_features[obj2]

            # Type similarity
            type_sim = 1.0 if f1['type'] == f2['type'] else (
                0.5 if f1['type'] in f2['type'] or f2['type'] in f1['type'] else 0.0
            )

            # Name similarity
            name_sim = difflib.SequenceMatcher(None, f1['name'], f2['name']).ratio()

            # Affordance overlap
            aff1, aff2 = f1['affordances'], f2['affordances']
            aff_overlap = len(aff1 & aff2) / len(aff1 | aff2) if (aff1 | aff2) else 0.0

            # Relation context (same predicates with similar objects)
            rel_sim = 0.0
            if f1['relations'] and f2['relations']:
                preds1 = set(r[0] for r in f1['relations'])
                preds2 = set(r[0] for r in f2['relations'])
                rel_sim = len(preds1 & preds2) / len(preds1 | preds2) if (preds1 | preds2) else 0.0

            # Combined score
            score = type_sim * 0.4 + name_sim * 0.2 + aff_overlap * 0.25 + rel_sim * 0.15

            if score > 0.1:
                candidates.append({
                    "object": obj2,
                    "score": round(score, 4),
                    "type_match": f1['type'] == f2['type'],
                    "affordance_overlap": round(aff_overlap, 4)
                })

        # Sort and take top-3
        candidates.sort(key=lambda x: x['score'], reverse=True)
        mappings[obj1] = candidates[:3]

    # Add to graph
    for obj1, candidates in mappings.items():
        for cand in candidates:
            if cand['score'] >= 0.5:
                graph.add(
                    obj_id("scene", obj1),
                    Predicates.MAPS_TO_OBJECT,
                    obj_id("scene", cand['object'])
                )

    # LLM step (optional)
    llm_merges = []

    if llm is not None:
        # Prepare context
        context = {
            "objects": [
                {"id": obj['id'], "type": obj.get('type', ''), "affordances": obj.get('affordances', [])}
                for obj in objects
            ],
            "relations": scene_json.get('relations', [])
        }

        system_prompt = """You are an object mapping analyzer. Propose object merges or aliases.
Return ONLY valid JSON with format:
{"merges": [{"objects": [str, str], "rationale": str, "confidence": float}]}
Only merge objects that are clearly the same or aliases."""

        user_prompt = f"Analyze these objects:\n{json.dumps(context, indent=2)}"

        try:
            result = llm.generate_json(system_prompt, user_prompt)

            for merge in result.get("merges", []):
                objs = merge.get("objects", [])
                conf = merge.get("confidence", 0)

                # Validate
                if len(objs) >= 2 and all(o in obj_features for o in objs):
                    # Check type compatibility or high affordance overlap
                    types = [obj_features[o]['type'] for o in objs]
                    affs = [obj_features[o]['affordances'] for o in objs]

                    type_compatible = len(set(types)) == 1
                    aff_overlap = len(set.intersection(*affs)) / len(set.union(*affs)) if any(affs) else 0

                    if type_compatible or aff_overlap >= 0.5:
                        llm_merges.append({
                            "objects": objs,
                            "rationale": merge.get("rationale", ""),
                            "validated": True,
                            "type_compatible": type_compatible,
                            "affordance_overlap": round(aff_overlap, 4)
                        })
                    else:
                        llm_merges.append({
                            "objects": objs,
                            "rationale": merge.get("rationale", ""),
                            "validated": False,
                            "rejection_reason": "Type/affordance mismatch"
                        })

        except Exception as e:
            llm_merges.append({"error": str(e)})

    result = {
        "object_similarities": mappings,
        "llm_merges": llm_merges,
        "summary": {
            "total_objects": len(objects),
            "high_similarity_pairs": sum(1 for m in mappings.values() for c in m if c['score'] >= 0.7),
            "llm_validated_merges": len([m for m in llm_merges if m.get("validated")])
        }
    }

    if output_dir:
        with open(Path(output_dir) / "object_mappings.json", 'w') as f:
            json.dump(result, f, indent=2)

    return result


def generate_report(
    robot_summaries: List[Dict[str, Any]],
    capabilities: Dict[str, Any],
    skill_subs: Dict[str, Any],
    obj_mappings: Dict[str, Any],
    output_dir: str
) -> str:
    """Generate a markdown report summarizing all reasoning results."""

    lines = [
        "# Robot Ontology Reasoning Report",
        "",
        "## 1. Parsed Robots",
        ""
    ]

    for rs in robot_summaries:
        lines.extend([
            f"### {rs['robot']}",
            f"- URDF: {rs['urdf']['joint_count']} joints, {rs['urdf']['link_count']} links",
            f"- SRDF: {len(rs['srdf']['groups'])} groups, {len(rs['srdf']['end_effectors'])} end effectors",
            f"- Joint mapping: {len(rs['joint_name_mapping'])} matched",
            ""
        ])

        if rs['unmapped_srdf_joints']:
            lines.append(f"- Unmapped SRDF joints: {', '.join(rs['unmapped_srdf_joints'])}")
        if rs.get('notes'):
            for note in rs['notes']:
                lines.append(f"- Note: {note}")
        lines.append("")

    lines.extend([
        "## 2. Inferred Capabilities",
        "",
        f"- Deterministic: {', '.join(capabilities.get('capabilities_deterministic', []))}",
        f"- Final: {', '.join(capabilities.get('capabilities_final', []))}",
        ""
    ])

    if capabilities.get('capabilities_llm_proposed'):
        lines.append("- LLM proposed:")
        for prop in capabilities['capabilities_llm_proposed']:
            status = "validated" if prop.get('validated') else "rejected"
            lines.append(f"  - {prop.get('capability')}: {status}")
        lines.append("")

    lines.extend([
        "## 3. Skill Substitutions",
        "",
        f"- Total equivalences: {skill_subs['summary']['total_equivalences']}",
        f"- Total decompositions: {skill_subs['summary']['total_decompositions']}",
        ""
    ])

    if skill_subs['equivalences']:
        lines.append("### Top Equivalences:")
        for eq in skill_subs['equivalences'][:5]:
            lines.append(f"- {eq['skill_a']} <=> {eq['skill_b']} (score: {eq['score']})")
        lines.append("")

    if skill_subs['decompositions']:
        lines.append("### Decompositions:")
        for dec in skill_subs['decompositions'][:5]:
            lines.append(f"- {dec['target_skill']} -> {' + '.join(dec['decomposition'])}")
        lines.append("")

    lines.extend([
        "## 4. Object Mappings",
        "",
        f"- Total objects: {obj_mappings['summary']['total_objects']}",
        f"- High similarity pairs: {obj_mappings['summary']['high_similarity_pairs']}",
        ""
    ])

    if obj_mappings.get('object_similarities'):
        lines.append("### Top Similar Objects:")
        for obj, candidates in list(obj_mappings['object_similarities'].items())[:3]:
            if candidates:
                top = candidates[0]
                lines.append(f"- {obj} ~ {top['object']} (score: {top['score']})")
        lines.append("")

    lines.extend([
        "---",
        "*Generated by Robot Ontology Reasoning Module*"
    ])

    report_content = "\n".join(lines)

    report_path = Path(output_dir) / "report.md"
    with open(report_path, 'w') as f:
        f.write(report_content)

    return report_content
