ThinkPHP 的 nginx 配置踩坑

lmaster發表於2019-11-11

今天在用一個以 tp 為基礎的快速開發框架時遇到一些問題:

nginx 報錯截圖

thinkphp 的 nginx 配置踩坑
為了方便說明進行手動換行

// 處理時重寫或內部重定向迴圈
2019/11/11 11:16:06 [error] 15164#15432: *1 rewrite or internal redirection cycle while processing 
    "/index.php/index.php/index.php/index.php/index.php/index.php/index.php/index.php/index/user/index.html", 
    client: 127.0.0.1, 
    server: xxxxx, 
    request: "GET /index/user/index.html HTTP/1.1", 
    host: "xxxxx", 
    referrer: "xxxxx"

錯誤配置

參考 larvael 配置

server {
    .
    .
    .

    location / {
        try_files $uri $uri/ /index.php?$query_string;
    }
    .
    .
    .
    location ~ \.php$ {
        fastcgi_pass127.0.0.1:9000;
        fastcgi_index index.php;
        fastcgi_param SCRIPT_FILENAME $realpath_root$fastcgi_script_name;
        include fastcgi_params;
    }
    .
    .
    .

}

發現所有路徑都一樣,都是首頁效果

初步判斷 nginx 重寫規則有問題

# 路徑 / 開頭之後都走這個匹配
# 如 /index /index/user 
location / {
    # $uri 本地有就返回,或者$uri/ 本地有目錄就返回,或者走後面的重寫
    # 本地是指在網站根目錄下,如 當 $uri=index 就是看根目錄下面有 index 檔案或者 index/ 目錄
    try_files $uri $uri/ /index.php?$query_string;
}

開始報錯

解決問題

網上查詢後 tp5 的配置應為

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

改後,發現問題沒解決;對比配置發現

    # location ~ \.php$ 改成  location ~ \.php(.*)$
    location ~ \.php(.*)$ {
        fastcgi_pass   127.0.0.1:9000;
        fastcgi_index index.php;
        fastcgi_split_path_info  ^((?U).+\.php)(/?.+)$;
        fastcgi_param  SCRIPT_FILENAME  $document_root$fastcgi_script_name;
        fastcgi_param  PATH_INFO  $fastcgi_path_info;
        fastcgi_param  PATH_TRANSLATED  $document_root$fastcgi_path_info;
        include fastcgi_params;
    }

解決,完整配置

server {
    listen       80;
    server_name  xxxxxxx ;
    root  www;

    add_header X-Frame-Options "SAMEORIGIN";
    add_header X-XSS-Protection "1; mode=block";
    add_header X-Content-Type-Options "nosniff";

    charset utf-8;

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

    location = /favicon.ico { access_log off; log_not_found off; }
    location = /robots.txt  { access_log off; log_not_found off; }

    error_page 404 /index.php;


    location ~ \.php(.*)$ {
        fastcgi_pass   127.0.0.1:9000;
        fastcgi_index index.php;
        fastcgi_split_path_info  ^((?U).+\.php)(/?.+)$;
        fastcgi_param  SCRIPT_FILENAME  $document_root$fastcgi_script_name;
        fastcgi_param  PATH_INFO  $fastcgi_path_info;
        fastcgi_param  PATH_TRANSLATED  $document_root$fastcgi_path_info;
        include fastcgi_params;
    }

    location ~ /\.(?!well-known).* {
        deny all;
    }
}

一些說明

在偽靜態配置用try_files而非rewrite,是因為rewrite會對全部匹配(這裡是 /)請求都做一次匹配,如:host.com/a.png,這會會浪費一些資源(伺服器的算力等);同時也參考了laravel的官方配置加入一些安全設定。

如果在用到上線專案時最好加入靜態資源快取配置。

    location ~* ^.+\.(css|js|ico|gif|jpg|jpeg|png)$ {
        expires 30d;
    }

修復

2020-08-20 發現get沒有引數,原因是沒有QUERY_STRING為空,解決:

    location / {
        # 帶上 query_string
        try_files $uri $uri/ /index.php$uri?$query_string;
    }
本作品採用《CC 協議》,轉載必須註明作者和本文連結

相關文章