- .env.example: полный шаблон, защита секретов - .gitignore: явное исключение .env.* и секретов - layout.tsx: XSS — заменён dangerouslySetInnerHTML на next/script для SW - ESLint: no-console error (allow warn/error), ignore scripts/ - scripts/remove-console-logs.js: очистка console.log без glob - backend/routes/modules: README с планом рефакторинга крупных файлов - SECURITY.md: гид по секретам, XSS, CORS, auth, линту - .husky/pre-commit: запуск npm run lint + прочие правки приложения и бэкенда Co-authored-by: Cursor <cursoragent@cursor.com>
80 lines
5.1 KiB
Python
80 lines
5.1 KiB
Python
from datetime import datetime, date
|
||
"""
|
||
Модели для управления лётной годностью согласно требованиям ИКАО Annex 8.
|
||
|
||
Соответствует требованиям:
|
||
- ИКАО Annex 8 (Airworthiness of Aircraft)
|
||
- EASA Part M (Continuing Airworthiness)
|
||
- Лучшие практики индустрии
|
||
"""
|
||
|
||
from sqlalchemy import String, ForeignKey, DateTime, Text, Boolean
|
||
from sqlalchemy.orm import Mapped, mapped_column, relationship
|
||
|
||
from app.db.base import Base
|
||
from app.models.common import TimestampMixin, uuid4_str
|
||
|
||
|
||
class AirworthinessCertificate(Base, TimestampMixin):
|
||
"""Документ лётной годности (ДЛГ).
|
||
|
||
Соответствует требованиям ИКАО Annex 8: каждый ВС должен иметь действующий
|
||
сертификат лётной годности.
|
||
"""
|
||
__tablename__ = "airworthiness_certificates"
|
||
|
||
id: Mapped[str] = mapped_column(String(36), primary_key=True, default=uuid4_str)
|
||
aircraft_id: Mapped[str] = mapped_column(String(36), ForeignKey("aircraft.id"), nullable=False, index=True)
|
||
aircraft = relationship("Aircraft", foreign_keys=[aircraft_id])
|
||
|
||
certificate_number: Mapped[str] = mapped_column(String(64), unique=True, index=True, nullable=False, doc="Номер сертификата")
|
||
certificate_type: Mapped[str] = mapped_column(String(32), nullable=False, doc="Тип сертификата (standard, export, special и т.д.)")
|
||
|
||
issue_date: Mapped[DateTime] = mapped_column(DateTime(timezone=True), nullable=False, doc="Дата выдачи")
|
||
expiry_date: Mapped[DateTime | None] = mapped_column(DateTime(timezone=True), nullable=True, doc="Дата истечения (если применимо)")
|
||
|
||
issuing_authority: Mapped[str] = mapped_column(String(128), nullable=False, doc="Орган, выдавший сертификат")
|
||
issued_by_user_id: Mapped[str | None] = mapped_column(String(36), ForeignKey("users.id"), nullable=True, doc="Пользователь, выдавший сертификат")
|
||
issued_by = relationship("User", foreign_keys=[issued_by_user_id])
|
||
|
||
status: Mapped[str] = mapped_column(String(32), nullable=False, default="valid", doc="Статус (valid, expired, suspended, revoked)")
|
||
|
||
conditions: Mapped[str | None] = mapped_column(Text, nullable=True, doc="Условия действия сертификата")
|
||
limitations: Mapped[str | None] = mapped_column(Text, nullable=True, doc="Ограничения")
|
||
remarks: Mapped[str | None] = mapped_column(Text, nullable=True, doc="Примечания")
|
||
|
||
is_active: Mapped[bool] = mapped_column(Boolean, nullable=False, default=True, doc="Активен ли сертификат")
|
||
|
||
|
||
class AircraftHistory(Base, TimestampMixin):
|
||
"""История событий воздушного судна.
|
||
|
||
Отслеживает все значимые события в жизненном цикле ВС:
|
||
обслуживания, модификации, инспекции, инциденты.
|
||
"""
|
||
__tablename__ = "aircraft_history"
|
||
|
||
id: Mapped[str] = mapped_column(String(36), primary_key=True, default=uuid4_str)
|
||
aircraft_id: Mapped[str] = mapped_column(String(36), ForeignKey("aircraft.id"), nullable=False, index=True)
|
||
aircraft = relationship("Aircraft", foreign_keys=[aircraft_id])
|
||
|
||
event_type: Mapped[str] = mapped_column(String(32), nullable=False, index=True, doc="Тип события (maintenance, inspection, modification, incident, status_change)")
|
||
event_date: Mapped[DateTime] = mapped_column(DateTime(timezone=True), nullable=False, index=True, doc="Дата события")
|
||
|
||
description: Mapped[str] = mapped_column(Text, nullable=False, doc="Описание события")
|
||
|
||
performed_by_org_id: Mapped[str | None] = mapped_column(String(36), ForeignKey("organizations.id"), nullable=True, doc="Организация, выполнившая работу")
|
||
performed_by_org = relationship("Organization", foreign_keys=[performed_by_org_id])
|
||
|
||
performed_by_user_id: Mapped[str | None] = mapped_column(String(36), ForeignKey("users.id"), nullable=True, doc="Специалист, выполнивший работу")
|
||
performed_by_user = relationship("User", foreign_keys=[performed_by_user_id])
|
||
|
||
# Счетчики на момент события
|
||
hours_at_event: Mapped[str | None] = mapped_column(String(32), nullable=True, doc="Налет на момент события (TTSN)")
|
||
cycles_at_event: Mapped[str | None] = mapped_column(String(32), nullable=True, doc="Циклы на момент события (TCSN)")
|
||
|
||
compliance_status: Mapped[str | None] = mapped_column(String(32), nullable=True, doc="Статус соответствия требованиям")
|
||
reference_documents: Mapped[str | None] = mapped_column(Text, nullable=True, doc="Ссылки на документы (через запятую или JSON)")
|
||
|
||
notes: Mapped[str | None] = mapped_column(Text, nullable=True, doc="Дополнительные примечания")
|