goon/app/models/bug_report.py
goon-foss ad0284585b Initial commit
Goon — self-hosted aggregator for adult-content scene metadata.

Indexes scenes from TPDB, StashDB, and 30+ public adult tube sites.
Cross-source deduplication via perceptual hash + Levenshtein distance.
FastAPI backend + APScheduler worker + React Native (Expo) mobile client.

FOSS, ad-free, donation-funded. See README for details.
2026-05-20 10:10:22 +02:00

50 lines
2.2 KiB
Python

"""In-app bug reports — composed wewnątrz mobile, wysłane przez POST /bug-reports.
Powód: użytkownik nie może łatwo zgłaszać bugów bo Android FLAG_SECURE blokuje
screenshoty (NSFW content). Przepisywanie tytułów ręcznie z telefonu na Google
Keep jest powolne. Tu mobile sam screen capture'uje (przez react-native-view-shot
omija FLAG_SECURE) i wysyła z metadata.
"""
from __future__ import annotations
import uuid
from datetime import datetime
from sqlalchemy import DateTime, ForeignKey, String, Text, func
from sqlalchemy.dialects.postgresql import UUID
from sqlalchemy.orm import Mapped, mapped_column
from app.models.base import Base
class BugReport(Base):
__tablename__ = "bug_reports"
id: Mapped[uuid.UUID] = mapped_column(
UUID(as_uuid=True), primary_key=True, default=uuid.uuid4
)
created_at: Mapped[datetime] = mapped_column(
DateTime(timezone=True), server_default=func.now(), nullable=False
)
# Free-form context: {"screen": "SceneDetail", "build_version": "0.1.2", ...}
screen_name: Mapped[str | None] = mapped_column(String(64), nullable=True)
app_version: Mapped[str | None] = mapped_column(String(32), nullable=True)
# Nullable scene/movie FK — bug może być na liście, na ekranie favorites itd.
# Mobile w Player ekranie używa tego samego param `sceneId` dla movies (legacy
# progress tracking hack), więc backend smart-routes po lookup'ie: payload
# `scene_id` próbujemy najpierw jako Scene, jeśli nie ma — jako Movie.
scene_id: Mapped[uuid.UUID | None] = mapped_column(
UUID(as_uuid=True),
ForeignKey("scenes.id", ondelete="SET NULL"),
nullable=True,
)
movie_id: Mapped[uuid.UUID | None] = mapped_column(
UUID(as_uuid=True),
ForeignKey("movies.id", ondelete="SET NULL"),
nullable=True,
)
message: Mapped[str] = mapped_column(Text, nullable=False)
# PNG/JPEG bytes z react-native-view-shot, base64. Limit 1MB w API (per-request).
screenshot_b64: Mapped[str | None] = mapped_column(Text, nullable=True)
# Po obejrzeniu/naprawieniu: True. Brak osobnej tabeli statusów — single-user app.
resolved: Mapped[bool] = mapped_column(default=False, nullable=False)