Implement flexible entity relation system (Phases A–E)
Declarative relation registry via defrelation s-expressions with cardinality enforcement (one-to-one, one-to-many, many-to-many), registry-aware relate/unrelate/can-relate API endpoints, generic container-nav fragment, and relation-driven UI components. Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
This commit is contained in:
@@ -46,6 +46,8 @@ def register() -> Blueprint:
|
||||
child_id=data["child_id"],
|
||||
label=data.get("label"),
|
||||
sort_order=data.get("sort_order"),
|
||||
relation_type=data.get("relation_type"),
|
||||
metadata=data.get("metadata"),
|
||||
)
|
||||
return {
|
||||
"id": rel.id,
|
||||
@@ -54,6 +56,7 @@ def register() -> Blueprint:
|
||||
"child_type": rel.child_type,
|
||||
"child_id": rel.child_id,
|
||||
"sort_order": rel.sort_order,
|
||||
"relation_type": rel.relation_type,
|
||||
}
|
||||
|
||||
_handlers["attach-child"] = _attach_child
|
||||
@@ -70,9 +73,122 @@ def register() -> Blueprint:
|
||||
parent_id=data["parent_id"],
|
||||
child_type=data["child_type"],
|
||||
child_id=data["child_id"],
|
||||
relation_type=data.get("relation_type"),
|
||||
)
|
||||
return {"deleted": deleted}
|
||||
|
||||
_handlers["detach-child"] = _detach_child
|
||||
|
||||
# --- relate (registry-aware) ---
|
||||
async def _relate():
|
||||
"""Create a typed relation with registry validation and cardinality enforcement."""
|
||||
from shared.services.relationships import attach_child, get_children
|
||||
from shared.sexp.relations import get_relation
|
||||
|
||||
data = await request.get_json(force=True)
|
||||
rel_type = data.get("relation_type")
|
||||
if not rel_type:
|
||||
return {"error": "relation_type is required"}, 400
|
||||
|
||||
defn = get_relation(rel_type)
|
||||
if defn is None:
|
||||
return {"error": f"unknown relation_type: {rel_type}"}, 400
|
||||
|
||||
from_id = data["from_id"]
|
||||
to_id = data["to_id"]
|
||||
|
||||
# Cardinality enforcement
|
||||
if defn.cardinality == "one-to-one":
|
||||
existing = await get_children(
|
||||
g.s,
|
||||
parent_type=defn.from_type,
|
||||
parent_id=from_id,
|
||||
child_type=defn.to_type,
|
||||
relation_type=rel_type,
|
||||
)
|
||||
if existing:
|
||||
return {"error": "one-to-one relation already exists", "existing_id": existing[0].child_id}, 409
|
||||
|
||||
rel = await attach_child(
|
||||
g.s,
|
||||
parent_type=defn.from_type,
|
||||
parent_id=from_id,
|
||||
child_type=defn.to_type,
|
||||
child_id=to_id,
|
||||
label=data.get("label"),
|
||||
sort_order=data.get("sort_order"),
|
||||
relation_type=rel_type,
|
||||
metadata=data.get("metadata"),
|
||||
)
|
||||
return {
|
||||
"id": rel.id,
|
||||
"relation_type": rel.relation_type,
|
||||
"parent_type": rel.parent_type,
|
||||
"parent_id": rel.parent_id,
|
||||
"child_type": rel.child_type,
|
||||
"child_id": rel.child_id,
|
||||
"sort_order": rel.sort_order,
|
||||
}
|
||||
|
||||
_handlers["relate"] = _relate
|
||||
|
||||
# --- unrelate (registry-aware) ---
|
||||
async def _unrelate():
|
||||
"""Remove a typed relation with registry validation."""
|
||||
from shared.services.relationships import detach_child
|
||||
from shared.sexp.relations import get_relation
|
||||
|
||||
data = await request.get_json(force=True)
|
||||
rel_type = data.get("relation_type")
|
||||
if not rel_type:
|
||||
return {"error": "relation_type is required"}, 400
|
||||
|
||||
defn = get_relation(rel_type)
|
||||
if defn is None:
|
||||
return {"error": f"unknown relation_type: {rel_type}"}, 400
|
||||
|
||||
deleted = await detach_child(
|
||||
g.s,
|
||||
parent_type=defn.from_type,
|
||||
parent_id=data["from_id"],
|
||||
child_type=defn.to_type,
|
||||
child_id=data["to_id"],
|
||||
relation_type=rel_type,
|
||||
)
|
||||
return {"deleted": deleted}
|
||||
|
||||
_handlers["unrelate"] = _unrelate
|
||||
|
||||
# --- can-relate (pre-flight check) ---
|
||||
async def _can_relate():
|
||||
"""Check if a relation can be created (cardinality, registry validation)."""
|
||||
from shared.services.relationships import get_children
|
||||
from shared.sexp.relations import get_relation
|
||||
|
||||
data = await request.get_json(force=True)
|
||||
rel_type = data.get("relation_type")
|
||||
if not rel_type:
|
||||
return {"error": "relation_type is required"}, 400
|
||||
|
||||
defn = get_relation(rel_type)
|
||||
if defn is None:
|
||||
return {"allowed": False, "reason": f"unknown relation_type: {rel_type}"}
|
||||
|
||||
from_id = data["from_id"]
|
||||
|
||||
if defn.cardinality == "one-to-one":
|
||||
existing = await get_children(
|
||||
g.s,
|
||||
parent_type=defn.from_type,
|
||||
parent_id=from_id,
|
||||
child_type=defn.to_type,
|
||||
relation_type=rel_type,
|
||||
)
|
||||
if existing:
|
||||
return {"allowed": False, "reason": "one-to-one relation already exists"}
|
||||
|
||||
return {"allowed": True}
|
||||
|
||||
_handlers["can-relate"] = _can_relate
|
||||
|
||||
return bp
|
||||
|
||||
Reference in New Issue
Block a user