Add STO booking and maintenance automation
This commit is contained in:
@@ -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"
|
||||
|
||||
|
||||
Reference in New Issue
Block a user