7.18 實戰多階段建立 Laravel 映象

本節適用於 PHP 開發者閱讀。Laravel 基於 8.x 版本,各個版本的檔案結構可能會有差異,請根據實際自行修改。

7.18.1 準備

新建一個 Laravel 專案或在已有的 Laravel 專案根目錄下新建 Dockerfile .dockerignore laravel.conf 檔案。

.dockerignore 檔案中寫入以下內容。

.idea/
.git/

vendor/

node_modules/

public/js/
public/css/
public/mix-manifest.json

yarn-error.log

bootstrap/cache/*
storage/

## 自行新增其他需要排除的檔案,例如 .env.* 檔案

...

laravel.conf 檔案中寫入 nginx 設定。

server {
  listen 80 default_server;
  root /app/laravel/public;
  index index.php index.html;

  location / {
      try_files $uri $uri/ /index.php?$query_string;
  }

  location ~ .*\.php(\/.*)*$ {
    fastcgi_pass   laravel:9000;
    include        fastcgi.conf;

    # fastcgi_connect_timeout 300;

    # fastcgi_send_timeout 300;

    # fastcgi_read_timeout 300;

  }
}

7.18.2 前端建立

第一階段進行前端建立。

# 注:node 映象推薦使用具體的版本標籤(如 node:22-alpine)
FROM node:22-alpine as frontend

COPY package.json /app/

RUN set -x ; cd /app \
      && npm install --registry=https://registry.npmmirror.com

COPY webpack.mix.js webpack.config.js tailwind.config.js /app/
COPY resources/ /app/resources/

RUN set -x ; cd /app \
      && touch artisan \
      && mkdir -p public \
      && npm run production

7.18.3 安裝 Composer 依賴

第二階段安裝 Composer 依賴。

# 注:composer 映象推薦使用具體的版本標籤(如 composer:2.x)
FROM composer:2 as composer

COPY database/ /app/database/
COPY composer.json composer.lock /app/

RUN set -x ; cd /app \
      && composer config -g repo.packagist composer https://mirrors.aliyun.com/composer/ \
      && composer install \
           --ignore-platform-reqs \
           --no-interaction \
           --no-plugins \
           --no-scripts \
           --prefer-dist

7.18.4 整合以上階段所生成的檔案

第三階段對以上階段生成的檔案進行整合。

# 注:php 映象版本號已包含 major.minor(8.3),生產環境可根據需要調整
FROM php:8.3-fpm-alpine as laravel

ARG LARAVEL_PATH=/app/laravel

COPY --from=composer /app/vendor/ ${LARAVEL_PATH}/vendor/
COPY . ${LARAVEL_PATH}
COPY --from=frontend /app/public/js/ ${LARAVEL_PATH}/public/js/
COPY --from=frontend /app/public/css/ ${LARAVEL_PATH}/public/css/
COPY --from=frontend /app/public/mix-manifest.json ${LARAVEL_PATH}/public/mix-manifest.json

RUN set -x ; cd ${LARAVEL_PATH} \
      && mkdir -p storage \
      && mkdir -p storage/framework/cache \
      && mkdir -p storage/framework/sessions \
      && mkdir -p storage/framework/testing \
      && mkdir -p storage/framework/views \
      && mkdir -p storage/logs \
      && chmod -R 777 storage \
      && php artisan package:discover

7.18.5 最後一個階段建立 NGINX 映象

# 注:nginx 映象推薦使用具體的版本標籤(如 nginx:1.28-alpine)
FROM nginx:1.28-alpine as nginx

ARG LARAVEL_PATH=/app/laravel

COPY laravel.conf /etc/nginx/conf.d/
COPY --from=laravel ${LARAVEL_PATH}/public ${LARAVEL_PATH}/public

7.18.6 建立 Laravel 及 Nginx 映象

使用 docker build 指令建立映象。

$ docker build -t my/laravel --target=laravel .

$ docker build -t my/nginx --target=nginx .

7.18.7 啟動容器並測試

新建 Docker 網路

$ docker network create laravel

啟動 laravel 容器,--name=laravel 引數設定的名字必須與 nginx 設定檔案中的 fastcgi_pass laravel:9000; 一致

$ docker run -dit --rm --name=laravel --network=laravel my/laravel

啟動 nginx 容器

$ docker run -dit --rm --network=laravel -p 8080:80 my/nginx

瀏覽器訪問 127.0.0.1:8080 可以看到 Laravel 專案首頁。

也許 Laravel 專案依賴其他外部服務,例如 redis、MySQL,請自行啟動這些服務之後再進行測試,本小節不再贅述。

7.18.8 生產環境最佳化

本小節內容為了方便測試,將設定檔案直接放到了映象中,實際在使用時 建議 將設定檔案作為 configsecret 掛載到容器中,請讀者自行學習 Kubernetes 的相關內容。

由於篇幅所限本小節只是簡單列出,更多內容可以參考 khs1994-docker/laravel-demo 專案。

7.18.9 附錄

完整的 Dockerfile 檔案如下。

# 注:生產環境推薦使用具體的版本標籤,如 node:22-alpine、composer:2.x、php:8.3-fpm-alpine、nginx:1.28-alpine
FROM node:22-alpine as frontend

COPY package.json /app/

RUN set -x ; cd /app \
      && npm install --registry=https://registry.npmmirror.com

COPY webpack.mix.js webpack.config.js tailwind.config.js /app/
COPY resources/ /app/resources/

RUN set -x ; cd /app \
      && touch artisan \
      && mkdir -p public \
      && npm run production

FROM composer:2 as composer

COPY database/ /app/database/
COPY composer.json composer.lock /app/

RUN set -x ; cd /app \
      && composer config -g repo.packagist composer https://mirrors.aliyun.com/composer/ \
      && composer install \
           --ignore-platform-reqs \
           --no-interaction \
           --no-plugins \
           --no-scripts \
           --prefer-dist

FROM php:8.3-fpm-alpine as laravel

ARG LARAVEL_PATH=/app/laravel

COPY --from=composer /app/vendor/ ${LARAVEL_PATH}/vendor/
COPY . ${LARAVEL_PATH}
COPY --from=frontend /app/public/js/ ${LARAVEL_PATH}/public/js/
COPY --from=frontend /app/public/css/ ${LARAVEL_PATH}/public/css/
COPY --from=frontend /app/public/mix-manifest.json ${LARAVEL_PATH}/public/mix-manifest.json

RUN set -x ; cd ${LARAVEL_PATH} \
      && mkdir -p storage \
      && mkdir -p storage/framework/cache \
      && mkdir -p storage/framework/sessions \
      && mkdir -p storage/framework/testing \
      && mkdir -p storage/framework/views \
      && mkdir -p storage/logs \
      && chmod -R 777 storage \
      && php artisan package:discover

FROM nginx:1.28-alpine as nginx

ARG LARAVEL_PATH=/app/laravel

COPY laravel.conf /etc/nginx/conf.d/
COPY --from=laravel ${LARAVEL_PATH}/public ${LARAVEL_PATH}/public
第 67 页,共 196 页
使用 mdPress 构建