D5 - 拆解 Docker 環境:快速理解Nginx 反向代理 Nuxt 和 Laravel 的關係

更新於 發佈於 閱讀時間約 15 分鐘

好了,到了這一步,我們已經有了一個基於 Docker 的開發環境,但你可能會想:「這麼多配置,到底是怎麼運作的?」別擔心,今天我們要一起拆解這個 Docker 環境,深入了解 Nginx 如何作為反向代理,協調 Nuxt(前端)和 Laravel(後端)之間的關係。

Docker 環境架構的整體說明

首先,讓我們快速回顧一下整個 Docker 環境的架構:

  • Laravel (backend):後端 API 服務,使用 PHP-FPM 運行。
  • Nuxt (frontend):前端應用,使用 NodeJS 運行,作為 Vue 的服務端渲染框架。
  • Nginx:作為反向代理,處理來自瀏覽器的請求,並將它們分別轉發給 Nuxt 和 Laravel。
  • MariaDB:資料庫服務,用於存儲 Laravel API 的數據。
    我們使用了 Docker Compose 來定義這些容器,讓它們在同一個網路下協同工作。現在,讓我們深入了解每個部分的設定,並探討為什麼要這麼做。

一、docker-compose.yml:容器協調的大腦

docker-compose.yml 是整個 Docker 環境的核心,負責協調各個服務。我們來逐段解析這個檔案,了解每個設定的目的。

version: '3.8'

services:
# Laravel (backend)
backend:
build:
dockerfile: ./.docker-env/api/Dockerfile
container_name: laravel-app
restart: on-failure
volumes:
- ./api:/var/www/html
environment:
- APP_ENV=local
- APP_DEBUG=true
- DB_CONNECTION=mysql
- DB_HOST=db
- DB_PORT=3306
- DB_DATABASE=finance
- DB_USERNAME=root
- DB_PASSWORD=root
command: bash -c "if [ ! -f composer.json ]; then composer create-project --prefer-dist laravel/laravel .; fi && composer install && php-fpm"
depends_on:
- db
networks:
- app-network

為什麼這麼設定?

  • build:指定 Dockerfile 的路徑,構建 Laravel 的容器環境。
  • volumes:將本機的 ./api 目錄掛載到容器內的 /var/www/html,方便我們在本機編輯代碼,容器即時更新。
  • environment:設定 Laravel 環境變數,特別是資料庫的連線資訊,讓 Laravel 知道該連接到哪個資料庫。
  • command:這裡有點技巧,我們在容器啟動時檢查 composer.json 是否存在,沒有的話就創建一個新的 Laravel 專案。接著安裝依賴並啟動 PHP-FPM。
  • depends_on:確保在啟動 backend 服務前,資料庫服務 db 已經啟動。
  • networks:將服務加入同一個網路,方便容器之間的通訊。

小提醒

別忘了:環境變數中的資料庫資訊要與資料庫服務的設定相匹配,否則 Laravel 可是會鬧脾氣的。

二、Dockerfile:打造專屬的容器環境

1. Laravel 的 Dockerfile

位於 .docker-env/api/Dockerfile,這個檔案定義了 Laravel 容器的環境。

# 使用 PHP 基礎映像檔
FROM php:8.2-fpm

# 安裝必要的系統套件和 PHP 擴展
RUN apt-get update && apt-get install -y \
git \
unzip \
libjpeg-dev \
libpng-dev \
libzip-dev \
&& docker-php-ext-install pdo_mysql zip

# 安裝 Composer
COPY --from=composer:latest /usr/bin/composer /usr/bin/composer

# 設定工作目錄
WORKDIR /var/www/html

# 複製專案檔案
COPY ../../api .

# 設置權限
RUN chown -R www-data:www-data /var/www/html

# Expose port and start PHP-FPM
EXPOSE 9000
CMD ["php-fpm"]

為什麼這麼設定?

  • FROM php:8.2-fpm:使用官方的 PHP 8.2 FPM 映像作為基礎,為什麼?因為 Laravel 需要 PHP 環境,而 FPM 能夠與 Nginx 整合。
  • 安裝系統套件和 PHP 擴展:Laravel 依賴於一些 PHP 擴展,如 pdo_mysql、zip 等,沒有它們,某些功能會無法運作。
  • 安裝 Composer:Composer 是 PHP 的套件管理工具,Laravel 的依賴都透過它來管理。
  • 設定工作目錄:將工作目錄設為 /var/www/html,這是 Nginx 預設的網頁根目錄。
  • COPY 專案檔案:將我們的 Laravel 專案代碼複製到容器內。
  • 設置權限:確保網頁服務有權限讀取和寫入必要的檔案。
  • CMD ["php-fpm"]:啟動 PHP-FPM 服務。

個人經驗分享

小技巧:有時候忘記安裝某個 PHP 擴展,結果功能跑不起來,那種感覺就像買了新電器卻沒電池。所以,提前確認所需的擴展,能省下不少時間。

2. Nuxt 的 Dockerfile

位於 .docker-env/web/Dockerfile,定義了 Nuxt 容器的環境。

# 使用 Node.js 20 映像
FROM node:20-alpine

# 設定工作目錄
WORKDIR /app

# 複製 package.json
COPY package*.json ./

# 暴露端口
EXPOSE 3000

# 啟動命令將在 docker-compose.yml 中定義

為什麼這麼設定?

  • FROM node:20-alpine:使用輕量級的 Node.js 20 Alpine 映像,節省空間和資源。
  • WORKDIR /app:設置工作目錄為 /app。
  • COPY package.json ./*:將 package.json 複製進去,方便後續安裝依賴。
  • EXPOSE 3000:暴露 Nuxt 預設的 3000 端口。

三、Nginx 配置:反向代理的魔法師

Nginx 作為反向代理,負責將不同的請求導向正確的服務。我們需要兩個配置檔案:api.conf 和 web.conf。

1. Laravel 的 Nginx 配置 (api.conf)

server {
listen 80;
server_name api.localhost;
root /var/www/html/public;
index index.php index.html;

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

location ~ \.php$ {
include fastcgi_params;
fastcgi_pass laravel-app:9000;
fastcgi_param SCRIPT_FILENAME $document_root$fastcgi_script_name;
}

location ~ /\.ht {
deny all;
}
}

為什麼這麼設定?

  • server_name api.localhost:指定服務的域名,方便本地開發時區分前後端。
  • root /var/www/html/public:Laravel 的入口位於 public 資料夾。
  • location /:處理所有請求,使用 try_files 嘗試找到對應的檔案,否則轉到 index.php。
  • location ~ .php$:處理 PHP 檔案的請求,透過 FastCGI 傳遞給 PHP-FPM(laravel-app:9000)。

2. Nuxt 的 Nginx 配置 (web.conf)

server {
listen 80;
server_name web.localhost;

location / {
proxy_pass http://nuxt-app:3000;
proxy_http_version 1.1;
proxy_set_header Upgrade $http_upgrade;
proxy_set_header Connection 'upgrade';
proxy_set_header Host $host;
proxy_cache_bypass $http_upgrade;
}
}

為什麼這麼設定?

  • server_name web.localhost:指定前端服務的域名。
  • proxy_pass http://nuxt-app:3000:將所有請求代理到 Nuxt 應用的容器。
  • proxy_set_header:設定必要的頭資訊,確保 WebSocket 等功能正常運作。

小故事時間

曾經有一次,我忘了在 Nginx 配置中設定 proxy_set_header,結果 WebSocket 連線一直失敗。那種抓不到 bug 的感覺,真的是讓人懷疑人生。所以,細節真的很重要!

四、MariaDB:資料庫的心臟

在 docker-compose.yml 中,MariaDB 的配置如下:

# MySQL (MariaDB)
db:
image: mariadb:latest
container_name: mariadb
restart: unless-stopped
environment:
MYSQL_ROOT_PASSWORD: root
MYSQL_DATABASE: finance
MYSQL_USER: root
MYSQL_PASSWORD: root
volumes:
- "./.docker-env/.db_data/mysql:/var/lib/mysql"
networks:
- app-network

為什麼這麼設定?

  • image: mariadb:latest:使用最新的 MariaDB 映像,提供 MySQL 兼容的資料庫服務。
  • environment:設定資料庫的初始用戶名、密碼和資料庫名稱。
  • volumes:將資料庫資料持久化到本機,避免容器重啟時數據丟失。
  • networks:加入同一個網路,讓 Laravel 可以連接到資料庫。

驗證資料庫連線

要確認 Laravel 是否成功連接到資料庫,可以在容器中執行遷移命令:

docker-compose exec backend php artisan migrate

如果看到遷移成功的訊息,表示 Laravel 與資料庫的連線正常。

五、Docker 指令小抄

以下是一些常用的 Docker 指令,方便你在開發過程中操作:

  • 啟動所有服務:
docker-compose up -d --build
  • 查看容器日誌:
docker-compose logs -f [service_name]
  • 進入容器終端:
docker-compose exec [service_name] bash
  • 重新啟動服務:
docker-compose restart [service_name]
  • 停止並移除容器:
docker-compose down

六、為什麼要這樣架構?

  • Docker 的好處:透過容器化,我們可以確保在任何環境下都能有一致的開發和部署環境,再也不用擔心「在我電腦上可以跑啊!」的問題。
  • Nginx 反向代理:統一管理請求,未來要加入 HTTPS、負載均衡等功能都更方便。
  • 服務解耦:前端、後端、資料庫各自獨立,方便維護和升級。

小結

恭喜你!我們已經深入了解了整個 Docker 環境的架構和配置。現在,你不僅知道「怎麼做」,還了解了「為什麼要這麼做」。這種理解將讓你在未來的開發中更加游刃有餘。

最後的叮嚀

別害怕嘗試和錯誤,有時候問題的解答就藏在那些看似不起眼的細節中。就像我常說的,「調整了 N 個配置,只為了解決一個小問題,但這就是成長的過程」。

接下來,我們將進行第一階驗收,看看這一切是否如我們預期的那樣順利運行!

留言
avatar-img
留言分享你的想法!
avatar-img
詹姆士的軟體易開罐
25會員
78內容數
這是一系列以軟體開發為主題的輕鬆分享,內容涵蓋了技術選擇、開發經驗、實戰應用等多方面的議題。無論是如何在眾多框架中做出選擇,還是如何應對技術轉移的挑戰,這裡有幽默、有趣的對話風格,將複雜的技術問題轉化為易懂的故事。
2024/12/22
這是我第一次參加 iThome 鐵人賽,原本並沒有打算參加,但在整理專案時,我忽然想把屬於自己的內容公開分享,而不僅僅藏在雲端裡。於是,我獨自規劃、撰寫並完成了一個完整的開發系列文章。在這段過程中,我體會到獨自開發的挑戰與成就,並希望能將這些經驗分享給每一位正在努力前行的你。
Thumbnail
2024/12/22
這是我第一次參加 iThome 鐵人賽,原本並沒有打算參加,但在整理專案時,我忽然想把屬於自己的內容公開分享,而不僅僅藏在雲端裡。於是,我獨自規劃、撰寫並完成了一個完整的開發系列文章。在這段過程中,我體會到獨自開發的挑戰與成就,並希望能將這些經驗分享給每一位正在努力前行的你。
Thumbnail
2024/12/15
這篇文章介紹建立分類列表頁面以及新增和編輯功能。文中詳細說明瞭頁面結構、資料取得、錯誤處理等重要步驟,並強調了共用元件和資料驗證的一致性。通過這次開發,讀者將獲得關於如何在Nuxt中操作動態路由的深入理解,同時提高使用者體驗。希望本篇能幫助讀者順利完成相關功能建置。
Thumbnail
2024/12/15
這篇文章介紹建立分類列表頁面以及新增和編輯功能。文中詳細說明瞭頁面結構、資料取得、錯誤處理等重要步驟,並強調了共用元件和資料驗證的一致性。通過這次開發,讀者將獲得關於如何在Nuxt中操作動態路由的深入理解,同時提高使用者體驗。希望本篇能幫助讀者順利完成相關功能建置。
Thumbnail
2024/12/14
哈囉,大家好!在前面的文章中,我們已經規劃了前端介面,並盤點了所需的頁面與功能。 現在,是時候開始動手實作了。今天,我們將專注於 銀行帳戶列表頁面(Bank Accounts)以及 新增/編輯銀行帳戶頁面(Add/Edit Bank Account)的開發。 透過這次的實作,我們將學習如何在 N
Thumbnail
2024/12/14
哈囉,大家好!在前面的文章中,我們已經規劃了前端介面,並盤點了所需的頁面與功能。 現在,是時候開始動手實作了。今天,我們將專注於 銀行帳戶列表頁面(Bank Accounts)以及 新增/編輯銀行帳戶頁面(Add/Edit Bank Account)的開發。 透過這次的實作,我們將學習如何在 N
Thumbnail
看更多
你可能也想看
Thumbnail
「欸!這是在哪裡買的?求連結 🥺」 誰叫你太有品味,一發就讓大家跟著剁手手? 讓你回購再回購的生活好物,是時候該介紹出場了吧! 「開箱你的美好生活」現正召喚各路好物的開箱使者 🤩
Thumbnail
「欸!這是在哪裡買的?求連結 🥺」 誰叫你太有品味,一發就讓大家跟著剁手手? 讓你回購再回購的生活好物,是時候該介紹出場了吧! 「開箱你的美好生活」現正召喚各路好物的開箱使者 🤩
Thumbnail
本文介紹了 Docker 的基本概念,包括 Docker Image、Docker Container、Dockerfile、Docker Compose 及其應用情境,如開發與測試環境、微服務架構和持續整合/持續部署。瞭解這些內容能幫助開發人員更有效地利用 Docker 進行應用程式的部署和管理。
Thumbnail
本文介紹了 Docker 的基本概念,包括 Docker Image、Docker Container、Dockerfile、Docker Compose 及其應用情境,如開發與測試環境、微服務架構和持續整合/持續部署。瞭解這些內容能幫助開發人員更有效地利用 Docker 進行應用程式的部署和管理。
Thumbnail
本文詳細介紹了如何使用 Docker 環境構建 Laravel 後端和 Nuxt 前端,並通過 Nginx 進行反向代理來協調它們的互動。從 docker-compose.yml 配置到各個服務的設定,讓開發環境穩定運行,並提供了常用的 Docker 指令以便於操作。
Thumbnail
本文詳細介紹了如何使用 Docker 環境構建 Laravel 後端和 Nuxt 前端,並通過 Nginx 進行反向代理來協調它們的互動。從 docker-compose.yml 配置到各個服務的設定,讓開發環境穩定運行,並提供了常用的 Docker 指令以便於操作。
Thumbnail
※ 必備開發環境: Node.Js環境。 Npm或yarn套件管理工具。 Docker倉庫套件:可以快速建立MySQL的資料庫。 下載 Docker Desktop:Docker Desktop for Windows。 ※ Docker快速建立MySQL資料庫,使用步驟: 準備Dock
Thumbnail
※ 必備開發環境: Node.Js環境。 Npm或yarn套件管理工具。 Docker倉庫套件:可以快速建立MySQL的資料庫。 下載 Docker Desktop:Docker Desktop for Windows。 ※ Docker快速建立MySQL資料庫,使用步驟: 準備Dock
Thumbnail
前言 大家好上次我們教了 Gitlab CI/CD 的 extends 如何使用,今天我們要來實際做一個真正佈署上線我們的服務,我們會分成兩個章節,第一部分先教大家怎麼寫 Docker Compose 佈署計畫書,第二部分教大家撰寫 CI/CD 自動化佈署流程計劃書 什麼是 Docker Com
Thumbnail
前言 大家好上次我們教了 Gitlab CI/CD 的 extends 如何使用,今天我們要來實際做一個真正佈署上線我們的服務,我們會分成兩個章節,第一部分先教大家怎麼寫 Docker Compose 佈署計畫書,第二部分教大家撰寫 CI/CD 自動化佈署流程計劃書 什麼是 Docker Com
Thumbnail
前言 前篇把 Ubuntu 作業系統的安裝跟準備談完了,有需要可以跳回去看。接下來聊容器服務 Docker 的安裝與使用。 Docker 可以應用的場合很多,這次是會用它來模擬 Android 受測裝置
Thumbnail
前言 前篇把 Ubuntu 作業系統的安裝跟準備談完了,有需要可以跳回去看。接下來聊容器服務 Docker 的安裝與使用。 Docker 可以應用的場合很多,這次是會用它來模擬 Android 受測裝置
Thumbnail
簡介 因常會在新機器跑container,藉此紀錄安裝過程,順便寫下腳本安裝懶人包 安裝docker 檢查系統中是否已經安裝了docker 執行:ps -ef |grep docker 如上圖所示,說明系統中並沒有安裝docker 添加docker-ce倉庫 安裝yum倉庫管理工具 配置docker
Thumbnail
簡介 因常會在新機器跑container,藉此紀錄安裝過程,順便寫下腳本安裝懶人包 安裝docker 檢查系統中是否已經安裝了docker 執行:ps -ef |grep docker 如上圖所示,說明系統中並沒有安裝docker 添加docker-ce倉庫 安裝yum倉庫管理工具 配置docker
Thumbnail
Development environment of Laravel. Nginx, php, mysql and centos 7
Thumbnail
Development environment of Laravel. Nginx, php, mysql and centos 7
追蹤感興趣的內容從 Google News 追蹤更多 vocus 的最新精選內容追蹤 Google News