Spaces:
Running
Running
File size: 6,097 Bytes
59abb4f 6fa287a 59abb4f 1428c0f 6fa287a 59abb4f 6fa287a 59abb4f | 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 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 | # βββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββ
# ClinicalMatch AI β HuggingFace Spaces Dockerfile
# Single container: Neo4j Community + FastAPI + Next.js + Nginx (supervisord)
# Exposed port: 7860 (HF Spaces default)
# Persistent storage: /data (Neo4j data lives here β survives restarts)
# βββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββ
# ββ Stage 1: Build Next.js ββββββββββββββββββββββββββββββββββββββββββββββββββββ
FROM node:20-slim AS frontend-builder
WORKDIR /build/frontend
COPY frontend/package*.json ./
RUN npm install --legacy-peer-deps --prefer-offline
COPY frontend/ ./
# Build with empty API URL so all requests are relative (Nginx routes them)
ENV NEXT_PUBLIC_API_URL=""
RUN npm run build
# ββ Stage 2: Final runtime image ββββββββββββββββββββββββββββββββββββββββββββββ
FROM ubuntu:22.04
ENV DEBIAN_FRONTEND=noninteractive
ENV LANG=C.UTF-8
# ββ System dependencies ββββββββββββββββββββββββββββββββββββββββββββββββββββββββ
RUN apt-get update && apt-get install -y --no-install-recommends \
# Java for Neo4j
openjdk-17-jre-headless \
# Python
python3.11 python3-pip python3.11-venv \
# Web / infra
nginx \
supervisor \
# Utilities
curl wget ca-certificates gnupg \
&& rm -rf /var/lib/apt/lists/*
# ββ Node.js 20 ββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββ
RUN curl -fsSL https://deb.nodesource.com/setup_20.x | bash - \
&& apt-get install -y --no-install-recommends nodejs \
&& rm -rf /var/lib/apt/lists/*
# ββ Neo4j Community 2026.04.0 βββββββββββββββββββββββββββββββββββββββββββββββββ
ENV NEO4J_VERSION=2026.04.0
ENV NEO4J_HOME=/opt/neo4j
ENV PATH="${NEO4J_HOME}/bin:${PATH}"
RUN wget -q "https://dist.neo4j.org/neo4j-community-${NEO4J_VERSION}-unix.tar.gz" \
&& tar -xzf "neo4j-community-${NEO4J_VERSION}-unix.tar.gz" -C /opt \
&& mv "/opt/neo4j-community-${NEO4J_VERSION}" /opt/neo4j \
&& rm "neo4j-community-${NEO4J_VERSION}-unix.tar.gz" \
&& rm -rf /opt/neo4j/data # will be symlinked to /data at runtime
# Neo4j configuration β listen on all interfaces, use /data for persistence
RUN { \
echo "server.bolt.listen_address=0.0.0.0:7687"; \
echo "server.http.listen_address=0.0.0.0:7474"; \
echo "server.directories.data=/data/neo4j/data"; \
echo "server.directories.logs=/data/neo4j/logs"; \
echo "server.directories.plugins=/opt/neo4j/plugins"; \
echo "dbms.security.auth_enabled=true"; \
echo "server.memory.heap.initial_size=512m"; \
echo "server.memory.heap.max_size=1g"; \
echo "server.memory.pagecache.size=256m"; \
echo "db.transaction.timeout=60s"; \
echo "dbms.logs.query.enabled=OFF"; \
} >> /opt/neo4j/conf/neo4j.conf
# ββ Python backend ββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββ
WORKDIR /app/backend
COPY backend/requirements.txt .
RUN pip3 install --no-cache-dir -r requirements.txt
COPY backend/ .
# ββ Next.js frontend (pre-built) βββββββββββββββββββββββββββββββββββββββββββββββ
WORKDIR /app/frontend
# Copy only what Next.js needs to run (not dev deps)
COPY --from=frontend-builder /build/frontend/.next/standalone ./
COPY --from=frontend-builder /build/frontend/.next/static ./.next/static
COPY --from=frontend-builder /build/frontend/public ./public
# ββ Neo4j seed dump ββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββ
COPY docker/neo4j.dump /app/docker/neo4j.dump
# ββ Config files βββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββββ
COPY docker/nginx.conf /app/docker/nginx.conf
COPY docker/supervisord.conf /app/docker/supervisord.conf
COPY docker/entrypoint.sh /app/docker/entrypoint.sh
RUN chmod +x /app/docker/entrypoint.sh
# ββ Nginx writable dirs (runs without root after init) ββββββββββββββββββββββββ
RUN mkdir -p /tmp/nginx-cache /tmp/nginx-body /tmp/nginx-run \
&& chown -R www-data:www-data /var/log/nginx /var/lib/nginx 2>/dev/null || true
# ββ Expose & environment βββββββββββββββββββββββββββββββββββββββββββββββββββββββ
EXPOSE 7860
# Neo4j β local Community instance (no Aura)
ENV NEO4J_URI=bolt://127.0.0.1:7687
ENV NEO4J_USERNAME=neo4j
ENV NEO4J_PASSWORD=clinicalmatch2024
ENV NEO4J_DATABASE=neo4j
# LLM β OpenAI-compatible (set real values via HF Spaces secrets)
ENV OPENAI_API_KEY=""
ENV OPENAI_BASE_URL=https://ai.aimlapi.com/v1
ENV OPENAI_MODEL=claude-opus-4-7
# Next.js standalone listens on 3000 internally; Nginx routes externally
ENV PORT=3000
ENV HOSTNAME=127.0.0.1
WORKDIR /app
ENTRYPOINT ["/app/docker/entrypoint.sh"]
|