generated from coulomb/repo-seed
Replace the ad-hoc coordination-domain spine with the Repo Classification Standard: 14 market domains, classification columns on managed_repos, and workplans anchored by repo_id (topic_id optional). - Add Alembic migration d8e9f0a1b2c3 with data backfill and workstream→workplan rename - Add api/classification.py validation and register-from-classification tooling - Expose workplan-first REST/MCP surface with legacy workstream aliases - Add C-24 consistency rule and legacy domain frontmatter mapping - Update dashboard repos page with category/capability/stake filters - Update orientation docs; mark STATE-WP-0065 finished
68 lines
2.2 KiB
Python
68 lines
2.2 KiB
Python
import uuid
|
|
from datetime import datetime
|
|
|
|
from pydantic import BaseModel, ConfigDict, model_validator
|
|
|
|
from api.models.decision import DecisionStatus, DecisionType
|
|
from api.schemas.compat import OptionalWorkplanIdCompatMixin
|
|
from pydantic import AliasChoices, Field
|
|
|
|
|
|
class DecisionCreate(BaseModel):
|
|
topic_id: uuid.UUID | None = None
|
|
workplan_id: uuid.UUID | None = Field(
|
|
default=None,
|
|
validation_alias=AliasChoices("workplan_id", "workstream_id"),
|
|
)
|
|
title: str
|
|
description: str | None = None
|
|
decision_type: DecisionType = DecisionType.pending
|
|
status: DecisionStatus = DecisionStatus.open
|
|
rationale: str | None = None
|
|
decided_by: str | None = None
|
|
decided_at: datetime | None = None
|
|
deadline: datetime | None = None
|
|
escalation_note: str | None = None
|
|
|
|
@model_validator(mode="after")
|
|
def topic_or_workplan_required(self) -> "DecisionCreate":
|
|
if self.topic_id is None and self.workplan_id is None:
|
|
raise ValueError("At least one of topic_id or workplan_id must be set")
|
|
return self
|
|
|
|
|
|
class DecisionResolve(BaseModel):
|
|
rationale: str
|
|
decided_by: str
|
|
write_log: bool = True # append to DECISIONS.md in the registered project directory
|
|
|
|
|
|
class DecisionUpdate(BaseModel):
|
|
title: str | None = None
|
|
description: str | None = None
|
|
decision_type: DecisionType | None = None
|
|
status: DecisionStatus | None = None
|
|
rationale: str | None = None
|
|
decided_by: str | None = None
|
|
decided_at: datetime | None = None
|
|
deadline: datetime | None = None
|
|
escalation_note: str | None = None
|
|
superseded_by: uuid.UUID | None = None
|
|
|
|
|
|
class DecisionRead(OptionalWorkplanIdCompatMixin, BaseModel):
|
|
model_config = ConfigDict(from_attributes=True)
|
|
id: uuid.UUID
|
|
topic_id: uuid.UUID | None = None
|
|
title: str
|
|
description: str | None = None
|
|
decision_type: DecisionType
|
|
status: DecisionStatus
|
|
rationale: str | None = None
|
|
decided_by: str | None = None
|
|
decided_at: datetime | None = None
|
|
deadline: datetime | None = None
|
|
escalation_note: str | None = None
|
|
superseded_by: uuid.UUID | None = None
|
|
created_at: datetime
|
|
updated_at: datetime |