Add STO booking and maintenance automation

This commit is contained in:
VPN SaaS Dev
2026-05-15 05:17:54 +09:00
parent 2be7ba2099
commit fec9635079
12 changed files with 2178 additions and 5 deletions

View File

@@ -1,4 +1,4 @@
from datetime import date, datetime
from datetime import date, datetime, time
from decimal import Decimal
from sqlalchemy import (
@@ -11,6 +11,7 @@ from sqlalchemy import (
Numeric,
String,
Text,
Time,
UniqueConstraint,
func,
)
@@ -163,6 +164,14 @@ class ServiceCenter(Base):
employees = relationship("ServiceEmployee", back_populates="service_center", cascade="all, delete-orphan")
visits = relationship("ServiceVisit", back_populates="service_center")
reviews = relationship("ServiceCenterReview", back_populates="service_center", cascade="all, delete-orphan")
booking_settings = relationship(
"ServiceCenterBookingSettings",
back_populates="service_center",
cascade="all, delete-orphan",
uselist=False,
)
holidays = relationship("ServiceCenterHoliday", back_populates="service_center", cascade="all, delete-orphan")
appointments = relationship("ServiceAppointment", back_populates="service_center", cascade="all, delete-orphan")
class CarServiceLink(Base):
@@ -260,6 +269,44 @@ class ServiceEmployee(Base):
service_center = relationship("ServiceCenter", back_populates="employees")
class ServiceCenterBookingSettings(Base):
__tablename__ = "service_center_booking_settings"
id: Mapped[int] = mapped_column(primary_key=True)
service_center_id: Mapped[int] = mapped_column(
ForeignKey("service_centers.id", ondelete="CASCADE"), unique=True, index=True
)
working_days: Mapped[list] = mapped_column(JSON, default=lambda: [0, 1, 2, 3, 4], server_default="[0,1,2,3,4]")
open_time: Mapped[time] = mapped_column(Time, default=time(9, 0), server_default="09:00:00")
close_time: Mapped[time] = mapped_column(Time, default=time(18, 0), server_default="18:00:00")
lunch_break_start: Mapped[time | None] = mapped_column(Time)
lunch_break_end: Mapped[time | None] = mapped_column(Time)
timezone: Mapped[str] = mapped_column(String(64), default="Asia/Seoul", server_default="Asia/Seoul")
slot_duration_minutes: Mapped[int] = mapped_column(Integer, default=30, server_default="30")
booking_buffer_minutes: Mapped[int] = mapped_column(Integer, default=0, server_default="0")
max_parallel_bookings: Mapped[int] = mapped_column(Integer, default=1, server_default="1")
accepts_online_booking: Mapped[bool] = mapped_column(Boolean, default=True, server_default="true", index=True)
created_at: Mapped[datetime] = mapped_column(DateTime(timezone=True), server_default=func.now())
updated_at: Mapped[datetime] = mapped_column(
DateTime(timezone=True), server_default=func.now(), onupdate=func.now()
)
service_center = relationship("ServiceCenter", back_populates="booking_settings")
class ServiceCenterHoliday(Base):
__tablename__ = "service_center_holidays"
__table_args__ = (UniqueConstraint("service_center_id", "holiday_date", name="uq_service_center_holiday"),)
id: Mapped[int] = mapped_column(primary_key=True)
service_center_id: Mapped[int] = mapped_column(ForeignKey("service_centers.id", ondelete="CASCADE"), index=True)
holiday_date: Mapped[date] = mapped_column(Date, index=True)
reason: Mapped[str | None] = mapped_column(String(240))
created_at: Mapped[datetime] = mapped_column(DateTime(timezone=True), server_default=func.now())
service_center = relationship("ServiceCenter", back_populates="holidays")
class ServiceVisit(Base):
__tablename__ = "service_visits"
@@ -283,6 +330,75 @@ class ServiceVisit(Base):
work_items = relationship("ServiceWorkItem", back_populates="visit", cascade="all, delete-orphan")
class MaintenanceRecommendation(Base):
__tablename__ = "maintenance_recommendations"
id: Mapped[int] = mapped_column(primary_key=True)
vehicle_id: Mapped[int] = mapped_column(ForeignKey("cars.id", ondelete="CASCADE"), index=True)
recommendation_type: Mapped[str] = mapped_column(String(64), index=True)
title: Mapped[str] = mapped_column(String(180))
description: Mapped[str | None] = mapped_column(Text)
due_odometer_km: Mapped[int | None] = mapped_column(Integer, index=True)
due_date: Mapped[date | None] = mapped_column(Date, index=True)
priority: Mapped[str] = mapped_column(String(24), default="medium", server_default="medium", index=True)
status: Mapped[str] = mapped_column(String(24), default="active", server_default="active", index=True)
source: Mapped[str] = mapped_column(String(40), default="system_analysis", server_default="system_analysis", index=True)
source_service_center_id: Mapped[int | None] = mapped_column(ForeignKey("service_centers.id", ondelete="SET NULL"), index=True)
source_appointment_id: Mapped[int | None] = mapped_column(Integer, index=True)
created_at: Mapped[datetime] = mapped_column(DateTime(timezone=True), server_default=func.now(), index=True)
updated_at: Mapped[datetime] = mapped_column(
DateTime(timezone=True), server_default=func.now(), onupdate=func.now()
)
class ServiceAppointment(Base):
__tablename__ = "service_appointments"
id: Mapped[int] = mapped_column(primary_key=True)
service_center_id: Mapped[int] = mapped_column(ForeignKey("service_centers.id", ondelete="CASCADE"), index=True)
vehicle_id: Mapped[int] = mapped_column(ForeignKey("cars.id", ondelete="CASCADE"), index=True)
owner_id: Mapped[int] = mapped_column(ForeignKey("users.id", ondelete="CASCADE"), index=True)
created_by: Mapped[int] = mapped_column(ForeignKey("users.id", ondelete="CASCADE"), index=True)
service_type: Mapped[str] = mapped_column(String(64), index=True)
service_name: Mapped[str] = mapped_column(String(180))
requested_start_at: Mapped[datetime] = mapped_column(DateTime(timezone=True), index=True)
requested_end_at: Mapped[datetime] = mapped_column(DateTime(timezone=True), index=True)
confirmed_start_at: Mapped[datetime | None] = mapped_column(DateTime(timezone=True), index=True)
confirmed_end_at: Mapped[datetime | None] = mapped_column(DateTime(timezone=True))
proposed_start_at: Mapped[datetime | None] = mapped_column(DateTime(timezone=True), index=True)
proposed_end_at: Mapped[datetime | None] = mapped_column(DateTime(timezone=True))
estimated_duration_minutes: Mapped[int] = mapped_column(Integer, default=60, server_default="60")
status: Mapped[str] = mapped_column(String(40), default="requested", server_default="requested", index=True)
customer_comment: Mapped[str | None] = mapped_column(Text)
service_center_comment: Mapped[str | None] = mapped_column(Text)
source_recommendation_id: Mapped[int | None] = mapped_column(
ForeignKey("maintenance_recommendations.id", ondelete="SET NULL"), index=True
)
linked_work_order_id: Mapped[int | None] = mapped_column(ForeignKey("service_visits.id", ondelete="SET NULL"), index=True)
cancellation_reason: Mapped[str | None] = mapped_column(Text)
cancelled_at: Mapped[datetime | None] = mapped_column(DateTime(timezone=True))
created_at: Mapped[datetime] = mapped_column(DateTime(timezone=True), server_default=func.now(), index=True)
updated_at: Mapped[datetime] = mapped_column(
DateTime(timezone=True), server_default=func.now(), onupdate=func.now()
)
service_center = relationship("ServiceCenter", back_populates="appointments")
class ServiceNotification(Base):
__tablename__ = "service_notifications"
id: Mapped[int] = mapped_column(primary_key=True)
recipient_user_id: Mapped[int] = mapped_column(ForeignKey("users.id", ondelete="CASCADE"), index=True)
service_center_id: Mapped[int | None] = mapped_column(ForeignKey("service_centers.id", ondelete="SET NULL"), index=True)
appointment_id: Mapped[int | None] = mapped_column(ForeignKey("service_appointments.id", ondelete="SET NULL"), index=True)
notification_type: Mapped[str] = mapped_column(String(80), index=True)
title: Mapped[str] = mapped_column(String(180))
body: Mapped[str | None] = mapped_column(Text)
status: Mapped[str] = mapped_column(String(24), default="unread", server_default="unread", index=True)
created_at: Mapped[datetime] = mapped_column(DateTime(timezone=True), server_default=func.now(), index=True)
class ServiceWorkItem(Base):
__tablename__ = "service_work_items"