- auth: first registered user becomes superadmin (active immediately) - entrypoint: no longer seeds demo data in prod (opt-in via RUN_SEED=1) - config: refuse to boot in prod with weak/placeholder SECRET_KEY (<32 chars) - main: restrict CORS to FRONTEND_URL only in prod (localhost dev-only) - seed_db: block prod seeding, read passwords from env, stop printing them - login: remove demo account credentials from UI Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
85 lines
3.5 KiB
Python
85 lines
3.5 KiB
Python
"""FastAPI application entry point."""
|
|
import os
|
|
|
|
from fastapi import FastAPI
|
|
from fastapi.middleware.cors import CORSMiddleware
|
|
from fastapi.staticfiles import StaticFiles
|
|
|
|
from app.api.attachments import router as attachments_router
|
|
from app.api.audit_log import router as audit_log_router
|
|
from app.api.auth import router as auth_router
|
|
from app.api.booking_templates import router as booking_templates_router
|
|
from app.api.bookings import admin_router as bookings_admin_router
|
|
from app.api.bookings import bookings_router
|
|
from app.api.bookings import router as spaces_bookings_router
|
|
from app.api.google_calendar import router as google_calendar_router
|
|
from app.api.notifications import router as notifications_router
|
|
from app.api.organizations import admin_router as organizations_admin_router
|
|
from app.api.organizations import router as organizations_router
|
|
from app.api.properties import admin_router as properties_admin_router
|
|
from app.api.properties import manager_router as properties_manager_router
|
|
from app.api.properties import router as properties_router
|
|
from app.api.public import router as public_router
|
|
from app.api.reports import router as reports_router
|
|
from app.api.settings import router as settings_router
|
|
from app.api.spaces import admin_router as spaces_admin_router
|
|
from app.api.spaces import router as spaces_router
|
|
from app.api.users import admin_router as users_admin_router
|
|
from app.api.users import router as users_router
|
|
from app.core.config import settings
|
|
from app.db.session import Base, engine
|
|
|
|
# Create database tables
|
|
Base.metadata.create_all(bind=engine)
|
|
|
|
app = FastAPI(title=settings.app_name)
|
|
|
|
# CORS middleware
|
|
# In production only the configured frontend is allowed; localhost is dev-only.
|
|
_allowed_origins = [settings.frontend_url]
|
|
if settings.debug:
|
|
_allowed_origins.append("http://localhost:5173")
|
|
|
|
app.add_middleware(
|
|
CORSMiddleware,
|
|
allow_origins=_allowed_origins,
|
|
allow_credentials=True,
|
|
allow_methods=["*"],
|
|
allow_headers=["*"],
|
|
)
|
|
|
|
# Include routers
|
|
app.include_router(auth_router, prefix="/api")
|
|
app.include_router(users_router, prefix="/api")
|
|
app.include_router(users_admin_router, prefix="/api")
|
|
app.include_router(spaces_router, prefix="/api")
|
|
app.include_router(spaces_admin_router, prefix="/api")
|
|
app.include_router(spaces_bookings_router, prefix="/api")
|
|
app.include_router(bookings_router, prefix="/api")
|
|
app.include_router(bookings_admin_router, prefix="/api")
|
|
app.include_router(booking_templates_router, prefix="/api")
|
|
app.include_router(settings_router, prefix="/api")
|
|
app.include_router(notifications_router, prefix="/api")
|
|
app.include_router(audit_log_router, prefix="/api", tags=["audit-log"])
|
|
app.include_router(attachments_router, prefix="/api", tags=["attachments"])
|
|
app.include_router(reports_router, prefix="/api", tags=["reports"])
|
|
app.include_router(google_calendar_router, prefix="/api", tags=["google-calendar"])
|
|
app.include_router(properties_router, prefix="/api")
|
|
app.include_router(properties_manager_router, prefix="/api")
|
|
app.include_router(properties_admin_router, prefix="/api")
|
|
app.include_router(organizations_router, prefix="/api")
|
|
app.include_router(organizations_admin_router, prefix="/api")
|
|
app.include_router(public_router, prefix="/api")
|
|
|
|
|
|
@app.get("/health")
|
|
def health() -> dict[str, str]:
|
|
"""Health check endpoint."""
|
|
return {"status": "ok"}
|
|
|
|
|
|
# Serve Vue.js frontend in production (when /app/dist exists)
|
|
_dist_dir = os.path.join(os.path.dirname(__file__), "..", "..", "dist")
|
|
if os.path.isdir(_dist_dir):
|
|
app.mount("/", StaticFiles(directory=_dist_dir, html=True), name="static")
|