# syntax=docker/dockerfile:1.10 FROM python:3.12-slim AS metadata-stage WORKDIR /tmp ENV DEBIAN_FRONTEND=noninteractive \ PYTHONUNBUFFERED=1 \ TZ=Asia/Shanghai RUN apt-get update && \ apt-get install -y --no-install-recommends \ curl \ python3-pip \ git \ && pip install toml \ && apt-get clean \ && rm -rf /var/lib/apt/lists/* COPY ./pyproject.toml ./ RUN echo "πŸ“‹ Collecting metadata..." && \ export CREATED_DATE="$(date -u '+%Y-%m-%d %H:%M:%S UTC')" && \ export PYPROJECT_NAME=$(python3 -c 'import toml; print(toml.load("pyproject.toml")["project"]["name"])') && \ export DESCRIPTION=$(python3 -c 'import toml; print(toml.load("pyproject.toml")["project"]["description"])') && \ export AUTHOR=$(python3 -c 'import toml; print(toml.load("pyproject.toml")["project"]["authors"][0]["name"])') && \ echo "⏰ Created Date: $CREATED_DATE" && \ echo "πŸ“ Project Name: $PYPROJECT_NAME" && \ echo "πŸ“„ Description: $DESCRIPTION" && \ echo "πŸ‘€ Author: $AUTHOR" && \ echo "$CREATED_DATE" > /tmp/CREATED_DATE && \ echo "$PYPROJECT_NAME" > /tmp/PYPROJECT_NAME && \ echo "$DESCRIPTION" > /tmp/DESCRIPTION && \ echo "$AUTHOR" > /tmp/AUTHOR RUN --mount=type=bind,source=./.git/,target=/tmp/.git/ \ echo "🏷️ Checking version information..." && \ VERSION=$(git describe --tags --exact-match 2>/dev/null || git rev-parse --short HEAD || echo "unknown") && \ echo "πŸ“Œ Version: $VERSION" && \ echo "$VERSION" > /tmp/VERSION LABEL org.opencontainers.image.created="$(cat /tmp/CREATED_DATE)" \ org.opencontainers.image.authors="$(cat /tmp/AUTHOR)" \ org.opencontainers.image.description="$(cat /tmp/DESCRIPTION)" \ org.opencontainers.image.name="$(cat /tmp/PYPROJECT_NAME)" \ org.opencontainers.image.version="$(cat /tmp/VERSION)" FROM node:20-slim AS frontend-stage WORKDIR /tmp/admin-frontend ENV TZ=Asia/Shanghai \ PNPM_HOME="/pnpm" \ PATH="$PNPM_HOME:$PATH" COPY ./admin-frontend/package.json ./admin-frontend/pnpm-lock.yaml ./ RUN corepack enable && \ corepack prepare pnpm@9.15.1 --activate && \ echo "πŸ”” Verifying Node.js installation..." && \ if NODE_VERSION=$(node --version 2>&1); then \ echo "βœ… Node.js version ${NODE_VERSION} installed successfully"; \ else \ echo "❌ Node.js installation failed" && exit 1; \ fi && \ echo "πŸ”” Verifying pnpm installation..." && \ if PNPM_VERSION=$(pnpm --version 2>&1); then \ echo "βœ… pnpm version ${PNPM_VERSION} installed successfully"; \ else \ echo "❌ pnpm installation failed" && exit 1; \ fi RUN pnpm install --frozen-lockfile COPY ./admin-frontend . RUN mkdir -p ../nonebot_bison/admin_page/dist RUN echo "πŸ—οΈ Starting frontend build..." && \ pnpm run build && \ echo "βœ… Frontend build completed" FROM python:3.12-slim AS production WORKDIR /app #ENV UV_INDEX_URL=https://mirrors.aliyun.com/pypi/simple/ ENV DEBIAN_FRONTEND=noninteractive \ UV_COMPILE_BYTECODE=1 \ UV_SYSTEM_PYTHON=true \ PYTHONUNBUFFERED=1 \ UV_LINK_MODE=copy \ TZ=Asia/Shanghai \ SHELL="/bin/bash" \ PNPM_HOME="/pnpm" \ PATH="/pnpm:${PATH}" RUN echo "πŸ“¦ Installing system dependencies..." && \ apt-get update && \ apt-get install -y --no-install-recommends \ curl \ xvfb \ fonts-noto-color-emoji \ fonts-unifont \ libfontconfig1 \ libfreetype6 \ xfonts-scalable \ fonts-liberation \ fonts-ipafont-gothic \ fonts-wqy-zenhei \ fonts-tlwg-loma-otf && \ mkdir -p /etc/apt/keyrings && \ curl -fsSL https://deb.nodesource.com/setup_20.x | bash - && \ apt-get install -y nodejs && \ echo "πŸ”” Verifying Node.js installation..." && \ if NODE_VERSION=$(node --version 2>&1); then \ echo "βœ… Node.js version ${NODE_VERSION} installed successfully"; \ else \ echo "❌ Node.js installation failed" && exit 1; \ fi && \ mkdir -p "${PNPM_HOME}" && \ npm install -g pnpm@9.15.1 && \ # ι’„δΈ‹θ½½ pnpm εŒ…εˆ°ε…¨ε±€ε­˜ε‚¨ curl -fsSL https://registry.npmjs.org/pnpm/-/pnpm-9.15.1.tgz -o /tmp/pnpm.tgz && \ mkdir -p /root/.local/share/pnpm/store/v3 && \ pnpm store add /tmp/pnpm.tgz && \ rm /tmp/pnpm.tgz && \ echo "πŸ”” Verifying pnpm installation..." && \ if PNPM_VERSION=$(pnpm --version 2>&1); then \ echo "βœ… pnpm version ${PNPM_VERSION} installed successfully"; \ else \ echo "❌ pnpm installation failed" && exit 1; \ fi && \ fc-cache -fv && \ apt-get clean && \ rm -rf /var/lib/apt/lists/* && \ echo "βœ… System dependencies installed successfully" COPY --from=ghcr.io/astral-sh/uv:latest /uv /uvx /bin/ RUN echo "πŸ”” Verifying uv installation..." && \ if UV_VERSION=$(uv --version 2>&1); then \ echo "βœ… uv version ${UV_VERSION} installed successfully"; \ else \ echo "❌ uv installation failed" && exit 1; \ fi COPY ./pyproject.toml ./uv.lock ./ RUN echo "πŸ“¦ Installing bison dependencies..." && \ uv pip install --extra=docker --requirement pyproject.toml --no-cache && \ echo "βœ… Dependencies installed successfully" && \ echo "πŸ”” Cleaning up uv..." && \ rm -rf /bin/uv /bin/uvx && \ rm -rf /root/.cache/ && \ rm -rf /root/.uv && \ echo "βœ… UV cleanup completed" COPY --from=metadata-stage /tmp/VERSION /app/VERSION COPY . . COPY --from=frontend-stage /tmp/nonebot_bison/admin_page/dist ./dist COPY --from=frontend-stage /tmp/admin-frontend/package.json ./ COPY --from=frontend-stage /tmp/admin-frontend/pnpm-lock.yaml ./ #RUN echo "πŸ“¦ Installing frontend dependencies..." && \ # cd /app/admin-frontend && \ # SHELL="/bin/bash" pnpm install --offline --prod --frozen-lockfile && \ # rm -rf /root/.cache/ && \ # echo "βœ… Frontend dependencies installed successfully" RUN echo "πŸ“¦ Installing frontend dependencies..." && \ cd /app/admin-frontend && \ SHELL="/bin/bash" pnpm install --prod --frozen-lockfile --no-verify-store-integrity && \ rm -rf /root/.cache/ /root/.local/share/pnpm/store/v3/* /root/.pnpm-store && \ echo "βœ… Frontend dependencies installed successfully" RUN echo "🎭 Installing Playwright..." && \ playwright install --with-deps chromium && \ rm -rf /var/lib/apt/lists/* /tmp/* && \ mv docker.env.prod .env.prod && \ echo "βœ… Playwright installed successfully" EXPOSE 3000 8080 RUN echo '#!/bin/bash' > /app/start.sh && \ echo 'echo "πŸ“Œ Current bison Version: $(cat /app/VERSION)"' >> /app/start.sh && \ echo 'echo "πŸš€ Starting backend service..."' >> /app/start.sh && \ echo 'nb run &' >> /app/start.sh && \ echo 'echo "πŸš€ Starting frontend service..."' >> /app/start.sh && \ echo 'cd /app/admin-frontend && SHELL=/bin/bash exec pnpm start' >> /app/start.sh && \ chmod +x /app/start.sh CMD ["/app/start.sh"]