File size: 3,135 Bytes
fdbdab0
 
 
 
 
 
 
 
 
 
 
 
 
 
2b0f766
fdbdab0
 
 
2b0f766
698f7fe
 
fdbdab0
 
 
2b0f766
fdbdab0
 
 
 
2b0f766
fdbdab0
 
 
 
 
 
 
3b978bc
 
 
2b0f766
fdbdab0
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
# syntax=docker/dockerfile:1.7
# ---------------------------------------------------------------------------
# Cybersecurity Panel — single-container HuggingFace Spaces image.
#
# Mirrors the structural choices used by the working HF Spaces deployments
# CCAI-Vibe-Demo and CU-Student-AIProject-Helper:
#   * multi-stage Node frontend build → Python+SPA bundle
#   * non-root user uid 1000, port 7860 (HF Spaces convention)
#   * persistence on the HF Storage Bucket mount at /data (SQLite shim)
#   * REACT_APP_API_URL is set to the empty string at build time so the SPA
#     issues relative URLs and shares the FastAPI origin
#
# Build context: repo root (this file).
# ---------------------------------------------------------------------------

# ---- 1. Frontend build (CRA) ----------------------------------------------
FROM node:20-bookworm AS frontend-build
WORKDIR /app/frontend

COPY phd-advisor-frontend/package.json phd-advisor-frontend/package-lock.json* ./
RUN --mount=type=cache,target=/root/.npm \
    npm ci

COPY phd-advisor-frontend/ ./

# Empty REACT_APP_API_URL → CRA inlines '' so every fetch() hits the same
# origin as the SPA (the FastAPI server below).
ENV REACT_APP_API_URL=""
RUN npm run build

# ---- 2. Python runtime + SPA bundle ---------------------------------------
FROM python:3.12-slim-bookworm

# ffmpeg is required by the /api/voice/transcribe endpoint
# (browser WebM/Opus → WAV for Whisper). Kept on the runtime image only,
# not on the build image, so we don't pay the apt cost during incremental
# code rebuilds.
RUN apt-get update && \
    apt-get install -y --no-install-recommends ffmpeg && \
    rm -rf /var/lib/apt/lists/*

# HF Spaces runs the container as uid 1000 and expects the app to do the
# same; with a non-root user we cannot bind privileged ports, but :7860 is
# fine.
RUN useradd -m -u 1000 user
USER user
ENV HOME=/home/user \
    PATH=/home/user/.local/bin:$PATH \
    PYTHONUNBUFFERED=1 \
    CORS_ORIGINS=* \
    DATA_DIR=/data \
    CONFIG_PATH=/home/user/app/cybersecurity_config.yaml

WORKDIR $HOME/app

# ---- Python deps (cached unless requirements.txt changes) -----------------
COPY --chown=user multi_llm_chatbot_backend/requirements.txt ./
RUN --mount=type=cache,target=/home/user/.cache/pip,uid=1000,gid=1000 \
    pip install --no-cache-dir --user -r requirements.txt

# ---- Backend source -------------------------------------------------------
COPY --chown=user multi_llm_chatbot_backend/ ./

# ---- Top-level configuration files (config.yaml + persona definitions) ----
COPY --chown=user cybersecurity_config.yaml ./cybersecurity_config.yaml
COPY --chown=user phd_config.yaml ./phd_config.yaml
COPY --chown=user undergrad_config.yaml ./undergrad_config.yaml
COPY --chown=user personas/ ./personas/

# ---- Frontend bundle ------------------------------------------------------
# main.py mounts $HOME/app/static at "/" so the SPA is served same-origin
# with the API.
COPY --chown=user --from=frontend-build /app/frontend/build ./static

ENV PYTHONPATH=$HOME/app

EXPOSE 7860
CMD ["uvicorn", "app.main:app", "--host", "0.0.0.0", "--port", "7860"]