Skip to content
⚠️ This article was written in 2019. Some content may be outdated.

前端 Docker 化部署方案

前端專案不再只是幾個靜態檔案放到 CDN 上那麼簡單。越來越多的前端應用需要 Node.js 服務端渲染、Nginx 反向代理、環境變數注入等能力。Docker 提供了一致的執行環境,讓前端專案可以在任何地方以相同的方式構建和執行。本文將從零搭建前端專案的 Docker 化部署方案。

為什麼前端需要 Docker

  1. 環境一致性 — 開發、測試、生產環境完全一致,告別"在我機器上沒問題"
  2. 依賴隔離 — 不同專案的 Node.js 版本、系統依賴互不影響
  3. 快速部署 — 映象即產物,部署時不需要重新構建
  4. 彈性伸縮 — 容器化後可以方便地水平擴充套件

基礎 Dockerfile:純靜態站點

dockerfile
# 第一階段:構建
FROM node:12-alpine AS builder

WORKDIR /app

# 先複製 package.json 和 lock 檔案,利用 Docker 快取層
COPY package.json package-lock.json ./
RUN npm ci --registry=https://registry.npm.taobao.org

# 複製原始碼並構建
COPY . .
RUN npm run build

# 第二階段:部署(只保留構建產物)
FROM nginx:1.17-alpine

# 複製構建產物到 nginx 目錄
COPY --from=builder /app/build /usr/share/nginx/html

# 自定義 nginx 配置
COPY nginx.conf /etc/nginx/conf.d/default.conf

EXPOSE 80

CMD ["nginx", "-g", "daemon off;"]

多階段構建說明

多階段構建是 Docker 的重要特性:

  • 第一階段(builder)使用完整的 Node.js 映象構建專案
  • 第二階段使用輕量的 Nginx 映象,只複製構建產物
  • 最終映象不包含 Node.js、npm、原始碼等,體積可以控制在 20MB 以內

Nginx 配置

nginx
# nginx.conf
server {
    listen 80;
    server_name localhost;
    root /usr/share/nginx/html;
    index index.html;

    # gzip 壓縮
    gzip on;
    gzip_min_length 1000;
    gzip_types text/plain text/css application/json application/javascript text/xml;

    # 靜態資源快取
    location ~* \.(js|css|png|jpg|jpeg|gif|ico|svg|woff|woff2|ttf)$ {
        expires 1y;
        add_header Cache-Control "public, immutable";
    }

    # SPA 路由支援(所有路徑都返回 index.html)
    location / {
        try_files $uri $uri/ /index.html;
    }

    # API 反向代理
    location /api/ {
        proxy_pass http://backend:3000;
        proxy_set_header Host $host;
        proxy_set_header X-Real-IP $remote_addr;
        proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
        proxy_set_header X-Forwarded-Proto $scheme;
    }

    # 健康檢查
    location /health {
        return 200 'OK';
        add_header Content-Type text/plain;
    }
}

Docker Compose 編排

前端 + 後端 + 資料庫的完整編排:

yaml
# docker-compose.yml
version: '3.7'

services:
  # 前端
  frontend:
    build:
      context: ./frontend
      dockerfile: Dockerfile
    ports:
      - "80:80"
    depends_on:
      - backend
    environment:
      - NODE_ENV=production
    restart: unless-stopped

  # 後端 API
  backend:
    build:
      context: ./backend
      dockerfile: Dockerfile
    ports:
      - "3000:3000"
    environment:
      - NODE_ENV=production
      - DB_HOST=postgres
      - DB_PORT=5432
      - DB_NAME=myapp
      - REDIS_URL=redis://redis:6379
    depends_on:
      - postgres
      - redis
    restart: unless-stopped

  # 資料庫
  postgres:
    image: postgres:12-alpine
    environment:
      POSTGRES_DB: myapp
      POSTGRES_USER: admin
      POSTGRES_PASSWORD: secret
    volumes:
      - postgres_data:/var/lib/postgresql/data
    restart: unless-stopped

  # 快取
  redis:
    image: redis:5-alpine
    restart: unless-stopped

volumes:
  postgres_data:

環境變數注入

Docker 構建後,環境變數被硬編碼在產物中。執行時注入環境變數有幾種方案:

方案一:執行時替換模板變數

dockerfile
FROM nginx:1.17-alpine

COPY --from=builder /app/build /usr/share/nginx/html
COPY nginx.conf /etc/nginx/conf.d/default.conf
COPY entrypoint.sh /entrypoint.sh
RUN chmod +x /entrypoint.sh

ENTRYPOINT ["/entrypoint.sh"]
CMD ["nginx", "-g", "daemon off;"]
bash
#!/bin/sh
# entrypoint.sh

# 將環境變數注入到 JS 檔案中
# HTML 中使用 __API_URL__ 佔位符
if [ -n "$API_URL" ]; then
  find /usr/share/nginx/html -name "*.js" -exec \
    sed -i "s|__API_URL__|$API_URL|g" {} \;
fi

if [ -n "$SENTRY_DSN" ]; then
  find /usr/share/nginx/html -name "*.js" -exec \
    sed -i "s|__SENTRY_DSN__|$SENTRY_DSN|g" {} \;
fi

exec "$@"
yaml
services:
  frontend:
    build: ./frontend
    environment:
      - API_URL=https://api.example.com
      - SENTRY_DSN=https://xxx@sentry.io/123

方案二:執行時配置檔案

html
<!-- public/config.js -->
window.__CONFIG__ = {
  API_URL: '__API_URL__',
  SENTRY_DSN: '__SENTRY_DSN__',
  VERSION: '__VERSION__',
};
html
<!-- public/index.html -->
<head>
  <script src="/config.js"></script>
  <script src="/static/js/main.js"></script>
</head>

應用程式碼中使用:

js
const config = window.__CONFIG__ || {};

export const API_URL = config.API_URL || '/api';
export const SENTRY_DSN = config.SENTRY_DSN || '';

這樣 config.js 不會被 Webpack 打包,可以獨立替換。

CI/CD 整合

yaml
# .github/workflows/deploy.yml
name: Build and Deploy

on:
  push:
    branches: [master]

jobs:
  build:
    runs-on: ubuntu-latest
    steps:
      - uses: actions/checkout@v2

      - name: Login to Docker Registry
        run: echo "$&#123;&#123; secrets.DOCKER_PASSWORD &#125;&#125;" | docker login -u "$&#123;&#123; secrets.DOCKER_USERNAME &#125;&#125;" --password-stdin

      - name: Build Docker image
        run: |
          docker build \
            --build-arg NODE_ENV=production \
            -t myapp/frontend:$&#123;&#123; github.sha &#125;&#125; \
            -t myapp/frontend:latest \
            ./frontend

      - name: Push Docker image
        run: |
          docker push myapp/frontend:$&#123;&#123; github.sha &#125;&#125;
          docker push myapp/frontend:latest

      - name: Deploy to production
        run: |
          ssh deploy@server "cd /opt/myapp && \
            docker-compose pull frontend && \
            docker-compose up -d frontend"

生產環境最佳化

.dockerignore

node_modules
.git
.gitignore
*.md
.env.local
.DS_Store
coverage
.nyc_output
*.log

健康檢查

dockerfile
FROM nginx:1.17-alpine

HEALTHCHECK --interval=30s --timeout=3s --start-period=5s --retries=3 \
  CMD wget -qO- http://localhost/health || exit 1

安全配置

dockerfile
# 使用非 root 使用者執行
FROM nginx:1.17-alpine

RUN addgroup -g 1001 -S appgroup && \
    adduser -u 1001 -S appuser -G appgroup

COPY --from=builder --chown=appuser:appgroup /app/build /usr/share/nginx/html
COPY --chown=appuser:appgroup nginx.conf /etc/nginx/conf.d/default.conf

USER appuser

映象體積最佳化

bash
# 檢視映象各層大小
docker history myapp/frontend

# 最終映象對比
# 完整 node 映象:~900MB
# node-alpine + 多階段構建:~20MB
# 去除不需要的檔案:~15MB

小結

  • 使用多階段構建分離構建環境和執行環境,最終映象只包含必要的執行檔案
  • Nginx 作為靜態檔案伺服器 + 反向代理,配置 SPA 路由支援和 gzip 壓縮
  • 環境變數注入可以通過 sed 替換模板變數或獨立的 config.js 檔案實現
  • Docker Compose 編排前端、後端和依賴服務,docker-compose.yml 即架構文件
  • CI/CD 中構建 Docker 映象並推送到 Registry,部署時只需 pull 和 restart
  • 注意 .dockerignore、非 root 使用者、健康檢查等生產環境安全和可靠性配置

MIT Licensed