Spaces:
Paused
Paused
| # Stage 1: Frontend Builder | |
| FROM node:20-slim AS frontend-builder | |
| WORKDIR /app/frontend | |
| # Copy dependency files first to leverage cache | |
| COPY frontend/package.json frontend/package-lock.json ./ | |
| ARG NPM_REGISTRY=https://registry.npmjs.org/ | |
| RUN npm config set registry ${NPM_REGISTRY} \ | |
| && npm config set fetch-retries 5 \ | |
| && npm config set fetch-retry-mintimeout 20000 \ | |
| && npm config set fetch-retry-maxtimeout 120000 | |
| # Retry npm ci to survive transient registry ECONNRESETs, which are common on | |
| # the QEMU-emulated arm64 leg of the multi-arch build. | |
| RUN i=0; until npm ci; do \ | |
| i=$((i+1)); \ | |
| if [ "$i" -ge 5 ]; then echo "npm ci failed after $i attempts"; exit 1; fi; \ | |
| echo "npm ci failed (attempt $i); retrying in 15s"; sleep 15; \ | |
| done | |
| # Copy the rest of the frontend source | |
| COPY frontend/ ./ | |
| # Build the frontend | |
| RUN npm run build | |
| # Stage 2: SurrealDB binary (pinned to v2 to match docker-compose.yml) | |
| FROM surrealdb/surrealdb:v2 AS surreal-binary | |
| # Stage 4: Backend Builder | |
| FROM python:3.12-slim-bookworm AS backend-builder | |
| # Install build dependencies | |
| RUN apt-get update && apt-get upgrade -y && apt-get install -y --no-install-recommends build-essential && rm -rf /var/lib/apt/lists/* | |
| # Install uv | |
| COPY --from=ghcr.io/astral-sh/uv:latest /uv /uvx /bin/ | |
| WORKDIR /app | |
| # Set build optimization environment variables | |
| ENV UV_HTTP_TIMEOUT=120 | |
| # Copy dependency files first | |
| COPY pyproject.toml uv.lock ./ | |
| COPY open_notebook/__init__.py ./open_notebook/__init__.py | |
| # Install dependencies | |
| RUN uv sync --frozen --no-dev | |
| # Pre-download tiktoken encoding so the app works offline (issue #264). | |
| # /app/tiktoken-cache is intentionally outside /app/data/ so that volume mounts | |
| # of /app/data (for user data persistence) do not hide the pre-baked encoding. | |
| # config.py reads TIKTOKEN_CACHE_DIR from the environment to pick up this path. | |
| ENV TIKTOKEN_CACHE_DIR=/app/tiktoken-cache | |
| RUN mkdir -p /app/tiktoken-cache && \ | |
| .venv/bin/python -c "import tiktoken; tiktoken.get_encoding('o200k_base')" | |
| # Stage 5: Runtime | |
| FROM python:3.12-slim-bookworm AS runtime | |
| # Install runtime dependencies | |
| RUN apt-get update && apt-get upgrade -y && apt-get install -y \ | |
| ffmpeg \ | |
| supervisor \ | |
| curl \ | |
| && curl -fsSL https://deb.nodesource.com/setup_20.x | bash - \ | |
| && apt-get install -y nodejs \ | |
| && rm -rf /var/lib/apt/lists/* | |
| # Install SurrealDB (copied from pinned v2 image to match docker-compose.yml) | |
| COPY --from=surreal-binary /surreal /usr/local/bin/surreal | |
| # Install uv (optional but helpful for some scripts) | |
| COPY --from=ghcr.io/astral-sh/uv:latest /uv /uvx /bin/ | |
| WORKDIR /app | |
| # Copy backend virtualenv and source code | |
| COPY --from=backend-builder /app/.venv /app/.venv | |
| COPY . /app/ | |
| # Copy pre-downloaded tiktoken encoding from builder (outside /data/ — volume-mount safe) | |
| COPY --from=backend-builder /app/tiktoken-cache /app/tiktoken-cache | |
| # Copy built frontend from standalone output | |
| COPY --from=frontend-builder /app/frontend/.next/standalone /app/frontend/ | |
| COPY --from=frontend-builder /app/frontend/.next/static /app/frontend/.next/static | |
| COPY --from=frontend-builder /app/frontend/public /app/frontend/public | |
| # Bind Next.js to all interfaces (required for Docker networking and reverse proxies) | |
| ENV HOSTNAME=0.0.0.0 | |
| # Point the app at the pre-baked tiktoken encoding (see open_notebook/config.py) | |
| ENV TIKTOKEN_CACHE_DIR=/app/tiktoken-cache | |
| # Setup directories and permissions | |
| RUN mkdir -p /app/data /mydata | |
| # Ensure wait-for-api script is executable | |
| RUN chmod +x /app/scripts/wait-for-api.sh | |
| # Copy supervisord configuration | |
| COPY supervisord.single.conf /etc/supervisor/conf.d/supervisord.conf | |
| # Create log directories | |
| RUN mkdir -p /var/log/supervisor | |
| # Expose ports | |
| EXPOSE 8502 5055 | |
| # Set startup command | |
| CMD ["/usr/bin/supervisord", "-c", "/etc/supervisor/conf.d/supervisord.conf"] |