我的 PHP 应用程序死于 fastcgi:意外的文件结束

My PHP application dies with fastcgi: unexpected end-of-file

我在我的本地主机上更新到 PHP 7,但从那时起,任何时候我想在我的 nette 应用程序中从一个页面重定向到另一个页面,我都会收到错误:500 - 内部服务器错误.

我在搜索堆栈溢出时发现了一个与我的问题非常相似的问题:How to solve "mod_fastcgi.c.2566 unexpected end-of-file (perhaps the fastcgi process died)" when calling .php that takes long time to execute?。但是,我不处理大文件,我的连接会立即中断。

我的/var/log/lighttpd/error.log

2016-03-06 10:54:11: (server.c.1456) [note] graceful shutdown started 
2016-03-06 10:54:11: (server.c.1572) server stopped by UID = 0 PID = 351 
2016-03-06 11:03:48: (log.c.194) server started 
2016-03-06 11:07:17: (mod_fastcgi.c.2390) unexpected end-of-file (perhaps the fastcgi process died): pid: 21725 socket: unix:/run/lighttpd/php-fastcgi.sock-3 
2016-03-06 11:07:17: (mod_fastcgi.c.3171) response not received, request sent: 1029 on socket: unix:/run/lighttpd/php-fastcgi.sock-3 for /~rost/lp/web/www/index.php?, closing connection 
2016-03-06 11:09:01: (mod_fastcgi.c.2390) unexpected end-of-file (perhaps the fastcgi process died): pid: 21725 socket: unix:/run/lighttpd/php-fastcgi.sock-3 
2016-03-06 11:09:01: (mod_fastcgi.c.3171) response not received, request sent: 1061 on socket: unix:/run/lighttpd/php-fastcgi.sock-3 for /~rost/lp/web/www/index.php?action=list&presenter=Campaign, closing connection 
2016-03-06 11:09:06: (mod_fastcgi.c.2390) unexpected end-of-file (perhaps the fastcgi process died): pid: 21725 socket: unix:/run/lighttpd/php-fastcgi.sock-3 
2016-03-06 11:09:06: (mod_fastcgi.c.3171) response not received, request sent: 942 on socket: unix:/run/lighttpd/php-fastcgi.sock-3 for /~rost/lp/web/www/index.php?, closing connection 
2016-03-06 11:09:14: (mod_fastcgi.c.2390) unexpected end-of-file (perhaps the fastcgi process died): pid: 21725 socket: unix:/run/lighttpd/php-fastcgi.sock-3 
2016-03-06 11:09:14: (mod_fastcgi.c.3171) response not received, request sent: 1051 on socket: unix:/run/lighttpd/php-fastcgi.sock-3 for /~rost/lp/web/www/index.php?action=out&presenter=Sign, closing connection 

我的/etc/lighttpd/lighttpd.conf

server.modules       = ( "mod_userdir", 
                         "mod_access", 
                         "mod_accesslog", 
                         "mod_fastcgi", 
                         "mod_rewrite", 
                         "mod_auth" 
                       )
server.port          = 80
server.username      = "http"
server.groupname     = "http"
server.document-root = "/srv/http"
server.errorlog      = "/var/log/lighttpd/error.log"
dir-listing.activate = "enable"
index-file.names     = ( "index.html" )

# Rewrite URL without dots to index.php
#url.rewrite-once     = ( "/^[^.?]*$/" => "/index.php" )
mimetype.assign      = ( ".html" => "text/html", 
                         ".htm" => "text/html", 
                         ".txt" => "text/plain", 
                         ".properties" => "text/plain", 
                         ".jpg" => "image/jpeg", 
                         ".png" => "image/png",
                         ".svg" => "image/svg+xml", 
                         ".gif" => "image/gif",  
                         ".css" => "text/css", 
                         ".js" => "application/x-javascript",
                         "" => "application/octet-stream" 
                       )
userdir.path         = "public_html"

# Fast CGI
include "conf.d/fastcgi.conf"

我的/etc/lighttpd/conf.d/fastcgi.conf

server.modules += ( "mod_fastcgi" )

#server.indexfiles += ( "index.php" ) #this is deprecated
index-file.names += ( "index.php" )

fastcgi.server = (
    ".php" => (
        "localhost" => ( 
        "bin-path" => "/usr/bin/php-cgi",
        "socket" => "/run/lighttpd/php-fastcgi.sock",
        "max-procs" => 4, # default value
        "bin-environment" => (
          "PHP_FCGI_CHILDREN" => "1", # default value
        ),
        "broken-scriptfilename" => "enable"
      ))
 )

变量来自 /etc/php/php.ini

cat /etc/php/php.ini | grep max_execution_time
max_execution_time = 30

cat /etc/php/php.ini | grep default_socket_timeout
default_socket_timeout = 60

更新 7.3.2016

我从 php fast cgi 切换到 php-fpm,有趣的是问题仍然存在,但并不常见。有时重定向跳转到 500,有时不会。并再次记录错误:

2016-03-07 22:23:32: (mod_fastcgi.c.2390) unexpected end-of-file (perhaps the fastcgi process died): pid: 0 socket: unix:/run/php-fpm/php-fpm.sock 
2016-03-07 22:23:32: (mod_fastcgi.c.3171) response not received, request sent: 1084 on socket: unix:/run/php-fpm/php-fpm.sock for /~rost/lp/web/www/index.php?action=out&presenter=Sign, closing connection 

另外,先尝试删除 Nette 缓存。

我终于找到了解决办法。这可能是 Nette / Cassandra 相关的问题。

由于对象 Nette\Security\Identity 而出现错误,在我将用户数据分配给它之后:

public function authenticate(array $credentials) {

  // Retrieve username and password
  list($email, $passwd) = $credentials;

  // Select user with given email from database
  $usr = $this->daoManager->getDao("AppUser")->loadByEmail($email);
  if ($usr == null || count($usr) == 0) {
    $msg = 'The email is incorrect.';
    $arg = self::IDENTITY_NOT_FOUND;
    throw new Nette\Security\AuthenticationException($msg, $arg);
  }

  // TODO Check user password
  // TODO Check verification      

  // Create identity - THE PROBLEM WAS HERE
  return new Identity($email, $usr['role'], $usr);
}

这是由于 $usr 数组中 Cassandra\Timestamp 的值 'registered' 引起的。从那时起,几乎所有重定向都因上述错误而崩溃。

以下代码修复了问题:

return new Identity($email, $usr['role'], $this->fixUserArray($usr));

其中:

protected function fixUserArray(array $user) {
  $result = array();
  foreach ($user as $key => $val) {
    if ($key === "registered") {
      $result[$key] = $val->time();
    } else {
      $result[$key] = $val;
    }
  } 
  return $result;
}