如何配置Laravel+Nginx_Laravel+Nginx站点以优化性能?

2026-04-29 03:072阅读0评论SEO基础
  • 内容介绍
  • 文章标签
  • 相关推荐

本文共计1011个文字,预计阅读时间需要5分钟。

如何配置Laravel+Nginx_Laravel+Nginx站点以优化性能?

很多人将 `root` 设成 Laravel 项目的根目录(例如 `/var/www/myapp`),导致访问时直接暴露 `app/`、`vendor/` 等敏感目录,甚至出现 `Index of /` 列表。Laravel 的入口只有 `public/index.php`,Nginx 必须从这里开始接管请求。

正确做法是把 root 指向 public/ 子目录:

server { listen 80; server_name example.com; root /var/www/myapp/public; # ← 关键:必须是 public/ index index.php; <pre class='brush:php;toolbar:false;'>location / { try_files $uri $uri/ /index.php?$query_string; } location ~ \.php$ { fastcgi_pass unix:/var/run/php/php8.2-fpm.sock; fastcgi_index index.php; fastcgi_param SCRIPT_FILENAME $realpath_root$fastcgi_script_name; include fastcgi_params; }

}

  • $realpath_root$document_root 更可靠,能避免符号链接导致的路径解析错误
  • 别漏掉 try_files —— 没它,静态资源(如 /css/app.css)会 404
  • 如果用的是 Docker 或非标准 PHP-FPM socket 路径,fastcgi_pass 必须对应上,常见错配:写成 127.0.0.1:9000 却没开 TCP 监听

rewrite 规则不能少,否则 Laravel 路由 404

Nginx 默认不解析 Laravel 的路由,所有非静态资源请求都得转发给 index.php,靠它来分发。只靠 location / + try_files 不够,某些 URL(比如带点号或斜杠嵌套的)可能被提前截断。

稳妥写法是显式重写所有非静态文件请求:

location / { try_files $uri $uri/ @laravel; } <p>location @laravel { rewrite ^(.*)$ /index.php?$query_string last; }

  • 不用 return 301proxy_pass —— 这是 PHP-FPM 场景,不是反代
  • 如果项目用了多语言路由(如 /zh/about),确保 $query_string 被完整传递,否则 app()->getLocale() 可能失效
  • 部分人误加 location ~ /\.ht 类规则,Nginx 根本不读 .htaccess,这类配置纯属冗余

PHP-FPM 权限和 PATH_INFO 配置不到位,502/500 随机出现

502 Bad Gateway 多半是 PHP-FPM 进程没响应,但更隐蔽的问题是 500 错误里看不到日志——因为 SCRIPT_FILENAME 解析错了,或者 PATH_INFO 为空导致 Laravel 认不出当前路由。

关键两行必须存在且顺序正确:

fastcgi_param SCRIPT_FILENAME $realpath_root$fastcgi_script_name; fastcgi_param PATH_INFO $fastcgi_path_info;

  • PATH_INFO:Laravel 的 Request::path() 返回空,所有路由匹配失败
  • $document_root 替代 $realpath_root:在启用 symlink 的部署(如 Laravel Envoyer、Capistrano)中,$document_root 可能指向软链本身,而非真实路径,导致 file_exists() 检查失败
  • PHP-FPM pool 用户(如 www-data)必须对 storage/bootstrap/cache/ 有写权限,否则 php artisan config:cache 后直接报错

HTTPS 和 www 重定向容易写死,后续难调试

上线后加 HTTPS 或强制 www,很多人直接在 server 块里硬编码 return 301 https://...,结果本地开发时也跳转,或者 CDN 回源失败。

更安全的做法是依赖 $scheme$host 动态判断:

if ($scheme != "https") { return 301 https://$host$request_uri; }

  • 别用 rewrite ^(.*)$ https://... —— rewrite 是字符串替换,$request_uri 包含 query string,rewrite 默认不携带,容易丢参数
  • 如果用了 Cloudflare 或阿里云 SLB,它们可能用 HTTP 回源,此时要检查 $_SERVER['HTTP_X_FORWARDED_PROTO'],并在 Laravel 的 AppServiceProvider 中调用 URL::forceScheme('https')
  • 重定向规则一定要放在主 server 块最前面,否则可能被 location 块拦截,变成 404 而非跳转

Laravel 的 Nginx 配置难点不在语法,而在路径解析、权限边界和环境差异。哪怕一行 $realpath_root 写错,或者少一个分号,都可能让日志里只显示 “recv() failed”,却找不到源头。

标签:LaravelNginx

本文共计1011个文字,预计阅读时间需要5分钟。

如何配置Laravel+Nginx_Laravel+Nginx站点以优化性能?

很多人将 `root` 设成 Laravel 项目的根目录(例如 `/var/www/myapp`),导致访问时直接暴露 `app/`、`vendor/` 等敏感目录,甚至出现 `Index of /` 列表。Laravel 的入口只有 `public/index.php`,Nginx 必须从这里开始接管请求。

正确做法是把 root 指向 public/ 子目录:

server { listen 80; server_name example.com; root /var/www/myapp/public; # ← 关键:必须是 public/ index index.php; <pre class='brush:php;toolbar:false;'>location / { try_files $uri $uri/ /index.php?$query_string; } location ~ \.php$ { fastcgi_pass unix:/var/run/php/php8.2-fpm.sock; fastcgi_index index.php; fastcgi_param SCRIPT_FILENAME $realpath_root$fastcgi_script_name; include fastcgi_params; }

}

  • $realpath_root$document_root 更可靠,能避免符号链接导致的路径解析错误
  • 别漏掉 try_files —— 没它,静态资源(如 /css/app.css)会 404
  • 如果用的是 Docker 或非标准 PHP-FPM socket 路径,fastcgi_pass 必须对应上,常见错配:写成 127.0.0.1:9000 却没开 TCP 监听

rewrite 规则不能少,否则 Laravel 路由 404

Nginx 默认不解析 Laravel 的路由,所有非静态资源请求都得转发给 index.php,靠它来分发。只靠 location / + try_files 不够,某些 URL(比如带点号或斜杠嵌套的)可能被提前截断。

稳妥写法是显式重写所有非静态文件请求:

location / { try_files $uri $uri/ @laravel; } <p>location @laravel { rewrite ^(.*)$ /index.php?$query_string last; }

  • 不用 return 301proxy_pass —— 这是 PHP-FPM 场景,不是反代
  • 如果项目用了多语言路由(如 /zh/about),确保 $query_string 被完整传递,否则 app()->getLocale() 可能失效
  • 部分人误加 location ~ /\.ht 类规则,Nginx 根本不读 .htaccess,这类配置纯属冗余

PHP-FPM 权限和 PATH_INFO 配置不到位,502/500 随机出现

502 Bad Gateway 多半是 PHP-FPM 进程没响应,但更隐蔽的问题是 500 错误里看不到日志——因为 SCRIPT_FILENAME 解析错了,或者 PATH_INFO 为空导致 Laravel 认不出当前路由。

关键两行必须存在且顺序正确:

fastcgi_param SCRIPT_FILENAME $realpath_root$fastcgi_script_name; fastcgi_param PATH_INFO $fastcgi_path_info;

  • PATH_INFO:Laravel 的 Request::path() 返回空,所有路由匹配失败
  • $document_root 替代 $realpath_root:在启用 symlink 的部署(如 Laravel Envoyer、Capistrano)中,$document_root 可能指向软链本身,而非真实路径,导致 file_exists() 检查失败
  • PHP-FPM pool 用户(如 www-data)必须对 storage/bootstrap/cache/ 有写权限,否则 php artisan config:cache 后直接报错

HTTPS 和 www 重定向容易写死,后续难调试

上线后加 HTTPS 或强制 www,很多人直接在 server 块里硬编码 return 301 https://...,结果本地开发时也跳转,或者 CDN 回源失败。

更安全的做法是依赖 $scheme$host 动态判断:

if ($scheme != "https") { return 301 https://$host$request_uri; }

  • 别用 rewrite ^(.*)$ https://... —— rewrite 是字符串替换,$request_uri 包含 query string,rewrite 默认不携带,容易丢参数
  • 如果用了 Cloudflare 或阿里云 SLB,它们可能用 HTTP 回源,此时要检查 $_SERVER['HTTP_X_FORWARDED_PROTO'],并在 Laravel 的 AppServiceProvider 中调用 URL::forceScheme('https')
  • 重定向规则一定要放在主 server 块最前面,否则可能被 location 块拦截,变成 404 而非跳转

Laravel 的 Nginx 配置难点不在语法,而在路径解析、权限边界和环境差异。哪怕一行 $realpath_root 写错,或者少一个分号,都可能让日志里只显示 “recv() failed”,却找不到源头。

标签:LaravelNginx