"""
Lightweight Triple Graph for Robot Ontology

A minimal triple store that is RDF/OWL-expansion-friendly but avoids
heavyweight dependencies. Uses namespaced string IDs for subjects,
predicates, and objects.
"""

import json
from typing import Optional, List, Set, Tuple, Dict, Any
from collections import defaultdict

# Predicate constants for consistent usage
class Predicates:
    # Robot structure
    HAS_JOINT = "hasJoint"
    HAS_LINK = "hasLink"
    HAS_GROUP = "hasGroup"
    HAS_END_EFFECTOR = "hasEndEffector"
    IN_CHAIN = "inChain"
    HAS_DOF = "hasDoF"
    JOINT_TYPE = "jointType"
    PARENT_LINK = "parentLink"
    CHILD_LINK = "childLink"
    HAS_LIMIT_LOWER = "hasLimitLower"
    HAS_LIMIT_UPPER = "hasLimitUpper"
    HAS_LIMIT_EFFORT = "hasLimitEffort"
    HAS_LIMIT_VELOCITY = "hasLimitVelocity"

    # Skills
    HAS_SKILL = "hasSkill"
    REQUIRES_CAP = "requiresCap"
    HAS_PARAM = "hasParam"
    HAS_PRE = "hasPre"
    HAS_EFF = "hasEff"
    HAS_DESCRIPTION = "hasDescription"

    # Scene/objects
    HAS_OBJECT = "hasObject"
    HAS_POSE = "hasPose"
    HAS_TYPE = "hasType"
    HAS_AFFORDANCE = "hasAffordance"

    # Spatial relations
    REL_ON = "rel:on"
    REL_INSIDE = "rel:inside"
    REL_NEAR = "rel:near"
    REL_SUPPORTED_BY = "rel:supportedBy"

    # Reasoning outputs
    EQUIV_SKILL = "equivSkill"
    DECOMP_SKILL = "decompSkill"
    MAPS_TO_OBJECT = "mapsToObject"
    HAS_CAPABILITY = "hasCapability"

    # Group/chain info
    GROUP_HAS_JOINT = "groupHasJoint"
    GROUP_HAS_LINK = "groupHasLink"
    CHAIN_BASE = "chainBase"
    CHAIN_TIP = "chainTip"

    # End-effector info
    HAS_EE_TYPE = "hasEEType"
    HAS_EE_OFFSET = "hasEEOffset"
    HAS_EE_OFFSET_DOWN = "hasEEOffsetDown"
    HAS_EE_OFFSET_LEFT = "hasEEOffsetLeft"
    HAS_EE_OFFSET_RIGHT = "hasEEOffsetRight"


class TripleGraph:
    """
    A lightweight in-memory triple store.

    Triples are stored as (subject: str, predicate: str, object: str).
    Supports basic SPARQL-like queries and serialization.
    """

    def __init__(self):
        self._triples: Set[Tuple[str, str, str]] = set()
        # Indexes for fast querying
        self._spo_index: Dict[str, Dict[str, Set[str]]] = defaultdict(lambda: defaultdict(set))
        self._pos_index: Dict[str, Dict[str, Set[str]]] = defaultdict(lambda: defaultdict(set))
        self._osp_index: Dict[str, Dict[str, Set[str]]] = defaultdict(lambda: defaultdict(set))

    def add(self, subject: str, predicate: str, obj: str) -> None:
        """Add a single triple to the graph."""
        triple = (subject, predicate, obj)
        if triple not in self._triples:
            self._triples.add(triple)
            self._spo_index[subject][predicate].add(obj)
            self._pos_index[predicate][obj].add(subject)
            self._osp_index[obj][subject].add(predicate)

    def add_many(self, triples: List[Tuple[str, str, str]]) -> None:
        """Add multiple triples at once."""
        for s, p, o in triples:
            self.add(s, p, o)

    def remove(self, subject: str, predicate: str, obj: str) -> bool:
        """Remove a triple. Returns True if it existed."""
        triple = (subject, predicate, obj)
        if triple in self._triples:
            self._triples.remove(triple)
            self._spo_index[subject][predicate].discard(obj)
            self._pos_index[predicate][obj].discard(subject)
            self._osp_index[obj][subject].discard(predicate)
            return True
        return False

    def query(self,
              subject: Optional[str] = None,
              predicate: Optional[str] = None,
              obj: Optional[str] = None) -> List[Tuple[str, str, str]]:
        """
        Query triples with optional wildcards.

        - query() -> all triples
        - query(s="robot:panda") -> all triples with that subject
        - query(p="hasJoint") -> all triples with that predicate
        - query(s="robot:panda", p="hasJoint") -> specific pattern
        """
        results = []

        if subject is not None and predicate is not None and obj is not None:
            # Exact match
            if (subject, predicate, obj) in self._triples:
                return [(subject, predicate, obj)]
            return []

        if subject is not None and predicate is not None:
            # S-P-? query
            for o in self._spo_index[subject][predicate]:
                results.append((subject, predicate, o))
        elif predicate is not None and obj is not None:
            # ?-P-O query
            for s in self._pos_index[predicate][obj]:
                results.append((s, predicate, obj))
        elif subject is not None and obj is not None:
            # S-?-O query
            for p in self._osp_index[obj][subject]:
                results.append((subject, p, obj))
        elif subject is not None:
            # S-?-? query
            for p, objs in self._spo_index[subject].items():
                for o in objs:
                    results.append((subject, p, o))
        elif predicate is not None:
            # ?-P-? query
            for o, subjects in self._pos_index[predicate].items():
                for s in subjects:
                    results.append((s, predicate, o))
        elif obj is not None:
            # ?-?-O query
            for s, predicates in self._osp_index[obj].items():
                for p in predicates:
                    results.append((s, p, obj))
        else:
            # Return all
            results = list(self._triples)

        return results

    def objects(self, subject: str, predicate: str) -> Set[str]:
        """Get all objects for a given subject and predicate."""
        return set(self._spo_index[subject][predicate])

    def subjects(self, predicate: str, obj: str) -> Set[str]:
        """Get all subjects for a given predicate and object."""
        return set(self._pos_index[predicate][obj])

    def predicates(self, subject: str, obj: str) -> Set[str]:
        """Get all predicates between a subject and object."""
        return set(self._osp_index[obj][subject])

    def has(self, subject: str, predicate: str, obj: str) -> bool:
        """Check if a triple exists."""
        return (subject, predicate, obj) in self._triples

    def __len__(self) -> int:
        return len(self._triples)

    def __iter__(self):
        return iter(self._triples)

    def to_json(self) -> Dict[str, Any]:
        """Serialize graph to JSON-compatible dict."""
        return {
            "triples": [list(t) for t in sorted(self._triples)]
        }

    @classmethod
    def from_json(cls, data: Dict[str, Any]) -> "TripleGraph":
        """Deserialize graph from JSON dict."""
        graph = cls()
        for t in data.get("triples", []):
            if len(t) == 3:
                graph.add(t[0], t[1], t[2])
        return graph

    def save(self, path: str) -> None:
        """Save graph to JSON file."""
        with open(path, 'w', encoding='utf-8') as f:
            json.dump(self.to_json(), f, indent=2, ensure_ascii=False)

    @classmethod
    def load(cls, path: str) -> "TripleGraph":
        """Load graph from JSON file."""
        with open(path, 'r', encoding='utf-8') as f:
            return cls.from_json(json.load(f))

    def export_ttl_like(self, path: str) -> None:
        """
        Export to a simple Turtle-like text format.
        Not strictly valid Turtle, but human-readable and expandable to RDF later.
        """
        # Group by subject
        by_subject: Dict[str, List[Tuple[str, str]]] = defaultdict(list)
        for s, p, o in sorted(self._triples):
            by_subject[s].append((p, o))

        with open(path, 'w', encoding='utf-8') as f:
            f.write("# Lightweight Triple Graph Export\n")
            f.write("# Format: subject predicate object .\n\n")

            for subject in sorted(by_subject.keys()):
                f.write(f"<{subject}>\n")
                for pred, obj in sorted(by_subject[subject]):
                    # Quote strings with spaces
                    if ' ' in obj or ':' not in obj:
                        obj_str = f'"{obj}"'
                    else:
                        obj_str = f"<{obj}>"
                    f.write(f"    <{pred}> {obj_str} ;\n")
                f.write(".\n\n")

    def stats(self) -> Dict[str, int]:
        """Return basic statistics about the graph."""
        subjects = set()
        predicates = set()
        objects = set()
        for s, p, o in self._triples:
            subjects.add(s)
            predicates.add(p)
            objects.add(o)

        return {
            "total_triples": len(self._triples),
            "unique_subjects": len(subjects),
            "unique_predicates": len(predicates),
            "unique_objects": len(objects)
        }

    def merge(self, other: "TripleGraph") -> None:
        """Merge another graph into this one."""
        for triple in other:
            self.add(*triple)


# Helper functions for creating namespaced IDs
def robot_id(name: str) -> str:
    """Create a robot ID like 'robot:panda'"""
    return f"robot:{name}"

def joint_id(robot: str, joint_name: str) -> str:
    """Create a joint ID like 'joint:panda:joint1'"""
    return f"joint:{robot}:{joint_name}"

def link_id(robot: str, link_name: str) -> str:
    """Create a link ID like 'link:panda:link0'"""
    return f"link:{robot}:{link_name}"

def group_id(robot: str, group_name: str) -> str:
    """Create a group ID like 'group:panda:arm'"""
    return f"group:{robot}:{group_name}"

def skill_id(robot: str, skill_name: str) -> str:
    """Create a skill ID like 'skill:panda:pick'"""
    return f"skill:{robot}:{skill_name}"

def obj_id(scene: str, obj_name: str) -> str:
    """Create an object ID like 'obj:scene:cup1'"""
    return f"obj:{scene}:{obj_name}"

def cap_id(capability: str) -> str:
    """Create a capability ID like 'cap:Grasping'"""
    return f"cap:{capability}"
