gabriel / musehub public
musehub_collaborator_models.py python
54 lines 2.3 KB
Raw
sha256:0997d6250ae6476362f6fe2025af7789f46d03df3e9f34356d5e8ee79b201923 fix(issues): use issue number as pagination cursor, not cre… Sonnet 4.6 patch 8 days ago
1 """SQLAlchemy ORM model for MuseHub collaborators.
2
3 Collaborators are users granted explicit push/admin access to a repo beyond
4 the owner. Permission levels: read | write | admin (default: write).
5 """
6
7 from datetime import datetime, timezone
8
9 from sqlalchemy import DateTime, ForeignKey, Index, String, UniqueConstraint
10 from sqlalchemy.orm import Mapped, MappedAsDataclass, mapped_column
11
12 from musehub.db.database import Base
13
14 def _utc_now() -> datetime:
15 return datetime.now(tz=timezone.utc)
16
17 class MusehubCollaborator(MappedAsDataclass, Base):
18 """A collaborator record granting a user explicit access to a repo.
19
20 ``permission`` is one of "read" | "write" | "admin"; defaults to "write".
21 ``invited_by_handle`` references the identity handle who extended the invitation (nullable
22 some collaborators may be added programmatically without an inviter).
23 ``accepted_at`` is null until the invited user explicitly accepts.
24
25 ``id`` is genesis-addressed: sha256(repo_id NUL identity_id NUL created_at_iso).
26 """
27
28 __tablename__ = "musehub_collaborators"
29 __table_args__ = (
30 UniqueConstraint("repo_id", "identity_handle", name="uq_musehub_collaborators_repo_identity"),
31 Index("ix_musehub_collaborators_repo_id", "repo_id"),
32 Index("ix_musehub_collaborators_identity_handle", "identity_handle"),
33 )
34
35 # --- Required fields ---
36 id: Mapped[str] = mapped_column(String(128), primary_key=True)
37 repo_id: Mapped[str] = mapped_column(
38 String(128),
39 ForeignKey("musehub_repos.repo_id", ondelete="CASCADE"),
40 nullable=False,
41 )
42 identity_handle: Mapped[str] = mapped_column(String(64), nullable=False)
43
44 # --- Optional fields with Python-side defaults ---
45 # Permission level: "read" | "write" | "admin"
46 permission: Mapped[str] = mapped_column(String(20), nullable=False, default="write", server_default="write")
47 invited_by_handle: Mapped[str | None] = mapped_column(String(64), nullable=True, default=None)
48 invited_at: Mapped[datetime] = mapped_column(
49 DateTime(timezone=True), nullable=False, default_factory=_utc_now
50 )
51 # Null until the invited user accepts the invitation
52 accepted_at: Mapped[datetime | None] = mapped_column(
53 DateTime(timezone=True), nullable=True, default=None
54 )
File History 1 commit
sha256:0997d6250ae6476362f6fe2025af7789f46d03df3e9f34356d5e8ee79b201923 fix(issues): use issue number as pagination cursor, not cre… Sonnet 4.6 patch 8 days ago