防止 PHP-FPM 在輕負載下崩潰

防止 PHP-FPM 在輕負載下崩潰

我在我託管的 WordPress 網站上遭到 DoS 攻擊。

173.192.109.118 - - [30/Sep/2015:22:31:36 +0000] "POST /xmlrpc.php HTTP/1.0" 499 0 "-" "Mozilla/4.0 (compatible: MSIE 7.0; Windows NT 6.0)"

我在訪問日誌中收到大約 140 個這樣的日誌nginx(花了大約 10 秒,所以每秒大約 14 個請求),然後它們切換到 502:

173.192.109.118 - - [30/Sep/2015:22:31:46 +0000] "POST /xmlrpc.php HTTP/1.0" 502 537 "-" "Mozilla/4.0 (compatible: MSIE 7.0; Windows NT 6.0)"

此時,PHP-FPM必須重新啟動才能恢復站點。

所以,我的問題是:我能做些什麼來防止一個單獨的攻擊者崩潰嗎PHP-FPM

我的大部分(有限)經驗都是與 Apache 相關的,所以任何建議都是大大地讚賞。

我試著對一切設定合理的限制。伺服器在負載下有足夠的 RAM,因此這似乎不是問題。我剛剛從以下教程中添加了速率限制器:https://www.howtoforge.com/rate-limiting-with-nginx,雖然這似乎延遲了痛苦,但最終還是崩潰了PHP-FPM

/var/log/php5-fpm.log除了當我忘記在設定檔中添加前導 / 時引入的幾個錯誤以及重新啟動後的一堆成功行之外,似乎沒有顯示任何有趣或有用的內容:

[30-Sep-2015 23:03:51] ERROR: Unable to create or open slowlog(/usr/log/www.log.slow): No such file or directory (2)
[30-Sep-2015 23:03:51] ERROR: failed to post process the configuration
[30-Sep-2015 23:03:51] ERROR: FPM initialization failed
[30-Sep-2015 23:05:47] NOTICE: configuration file /etc/php5/fpm/php-fpm.conf test is successful

/etc/php5/fpm/pool.d/www.conf

[www]
user = www-data
group = www-data
listen = /var/run/php5-fpm.sock
listen.owner = www-data
listen.group = www-data
pm = dynamic
pm.max_children = 5
pm.start_servers = 2
pm.min_spare_servers = 1
pm.max_spare_servers = 3
pm.status_path = /status
ping.path = /ping
ping.response = pong
slowlog = /var/log/php-fpm_$pool.slow.log
request_slowlog_timeout = 30
request_terminate_timeout = 30
chdir = /

/etc/nginx/nginx.conf

user www-data;
worker_processes 4;
pid /run/nginx.pid;
events {
    worker_connections 768;
}
http {
    sendfile on;
    tcp_nopush on;
    tcp_nodelay on;
    keepalive_timeout 65;
    types_hash_max_size 2048;
    include /etc/nginx/mime.types;
    default_type application/octet-stream;
    access_log /var/log/nginx/access.log;
    error_log /var/log/nginx/error.log;
    gzip on;
    gzip_disable "msie6";
    gzip_vary on;
    gzip_proxied any;
    gzip_comp_level 6;
    gzip_buffers 16 8k;
    gzip_http_version 1.1;
    gzip_types text/plain text/css application/json application/x-javascript text/xml application/xml application/xml+rss text/javascript;

    limit_req_zone  $binary_remote_addr  zone=one:10m   rate=1r/s;

    include /etc/nginx/conf.d/*.conf;
    include /etc/nginx/sites-enabled/*;
}

/etc/nginx/sites-enabled/example.com

server {
  server_name localhost www.example.com;
  return 301 http://example.com$request_uri;
}
server {
    listen 80 default_server;
    listen [::]:80 default_server ipv6only=on;
    root /var/www/html;
    index index.php index.html index.htm;
    server_name example.com;

    client_max_body_size 500M;

    location / {
        try_files $uri $uri/ /index.php?q=$uri&$args;
    }

    error_page 404 /404.html;

    error_page 500 502 503 504 /50x.html;
    location = /50x.html {
        root /var/www/html;
    }

    location ~*  \.(jpg|jpeg|png|gif|ico|css|js|woff)$ {
            expires 365d;
    }
        location ~ \.php$ {
            try_files $uri =404;
            fastcgi_split_path_info ^(.+\.php)(/.+)$;
            fastcgi_pass unix:/var/run/php5-fpm.sock;
            fastcgi_index index.php;
            fastcgi_param SCRIPT_FILENAME $document_root$fastcgi_script_name;
            include fastcgi_params;
            limit_req zone=one burst=5;
        }

    location /status {
        fastcgi_pass php;
    }

    location /ping {
        fastcgi_pass php;
    }

    location ~ /\. {
        deny all;
    }
}

**更新**

我更新了標題以更好地反映我的問題,希望能吸引一些有關PHP-FPM調優的高品質討論。

作為第二個問題,可能比我的第一個問題更重要,我想知道: 如何調整/強化 PHP-FPM 以利用所有可用的伺服器資源而不先崩潰。

Apache / PHP 可能沒有那麼高效,但它並沒有停止處理請求,直到伺服器癱瘓,然後當攻擊結束時,網站又恢復了。必須手動重新啟動一個稍微過度勞累的服務似乎相當不愉快。 (14 請求/秒真的沒什麼)

我同意用來減輕 DoS 攻擊的想法fail2ban,但我真正擔心的是如果/當常規的流量達到 15 請求/秒?

答案1

基本上你有以下選擇:

  • 使用資料包過濾器阻止
  • 使用 nginx 阻塞就像

location / { deny xx.xx.xx.xx; allow all; }

  • 增加到pm.max_children一個相等的數CPU 核心 x 2, 5 太低了 - 可能增加後每秒只能處理 14 個請求,實際上並不是那麼大的數字。此外,您正在使用 nginxlimit_req指令來限制請求速率,我建議您添加另一個區域並將其配置為較低的突發大小或nodelay.

相關內容