* @author Froxlor team (2010-) * @license GPLv2 http://files.froxlor.org/misc/COPYING.txt * @package Cron * * @link http://www.nutime.de/ * @since 0.9.16 * */ class phpinterface_fpm { /** * Domain-Data array * @var array */ private $_domain = array(); /** * Admin-Date cache array * @var array */ private $_admin_cache = array(); /** * defines what can be used for pool-config from php.ini * Mostly taken from http://php.net/manual/en/ini.list.php * * @var array */ private $_ini = array( 'php_value' => array( 'auto_append_file', 'auto_prepend_file', 'date.timezone', 'default_charset', 'error_reporting', 'include_path', 'log_errors_max_len', 'mail.log', 'max_execution_time', 'session.cookie_domain', 'session.cookie_lifetime', 'session.cookie_path', 'session.name', 'session.serialize_handler', 'upload_max_filesize', 'xmlrpc_error_number', 'session.auto_start', ), 'php_flag' => array( 'asp_tags', 'display_errors', 'display_startup_errors', 'html_errors', 'log_errors', 'magic_quotes_gpc', 'magic_quotes_runtime', 'magic_quotes_sybase', 'mail.add_x_header', 'session.cookie_secure', 'session.use_cookies', 'short_open_tag', 'track_errors', 'xmlrpc_errors' ), 'php_admin_value' => array( 'cgi.redirect_status_env', 'date.timezone', 'disable_classes', 'disable_functions', 'error_log', 'gpc_order', 'max_input_time', 'memory_limit', 'open_basedir', 'output_buffering', 'post_max_size', 'precision', 'sendmail_path', 'session.gc_divisor', 'session.gc_probability', 'variables_order' ), 'php_admin_flag' => array( 'allow_call_time_pass_reference', 'allow_url_fopen', 'allow_url_include', 'auto_detect_line_endings', 'cgi.fix_pathinfo', 'cgi.force_redirect', 'enable_dl', 'expose_php', 'file_uploads', 'ignore_repeated_errors', 'ignore_repeated_source', 'log_errors', 'register_argc_argv', 'report_memleaks' ) ); /** * main constructor */ public function __construct($domain) { $this->_domain = $domain; } /** * create fpm-pool config * * @param array $phpconfig */ public function createConfig($phpconfig) { $fh = @fopen($this->getConfigFile(), 'w'); if ($fh) { $fpm_pm = Settings::Get('phpfpm.pm'); $fpm_children = (int)Settings::Get('phpfpm.max_children'); $fpm_start_servers = (int)Settings::Get('phpfpm.start_servers'); $fpm_min_spare_servers = (int)Settings::Get('phpfpm.min_spare_servers'); $fpm_max_spare_servers = (int)Settings::Get('phpfpm.max_spare_servers'); $fpm_requests = (int)Settings::Get('phpfpm.max_requests'); $fpm_process_idle_timeout = (int)Settings::Get('phpfpm.idle_timeout'); if ($fpm_children == 0) { $fpm_children = 1; } $fpm_config = ';PHP-FPM configuration for "'.$this->_domain['domain'].'" created on ' . date("Y.m.d H:i:s") . "\n"; $fpm_config.= '['.$this->_domain['domain'].']'."\n"; $fpm_config.= 'listen = '.$this->getSocketFile()."\n"; if ($this->_domain['loginname'] == 'froxlor.panel') { $fpm_config.= 'listen.owner = '.$this->_domain['guid']."\n"; $fpm_config.= 'listen.group = '.$this->_domain['guid']."\n"; } else { $fpm_config.= 'listen.owner = '.$this->_domain['loginname']."\n"; $fpm_config.= 'listen.group = '.$this->_domain['loginname']."\n"; } // see #1418 why this is 0660 $fpm_config.= 'listen.mode = 0660'."\n"; if ($this->_domain['loginname'] == 'froxlor.panel') { $fpm_config.= 'user = '.$this->_domain['guid']."\n"; $fpm_config.= 'group = '.$this->_domain['guid']."\n"; } else { $fpm_config.= 'user = '.$this->_domain['loginname']."\n"; $fpm_config.= 'group = '.$this->_domain['loginname']."\n"; } $fpm_config.= 'pm = '.$fpm_pm."\n"; $fpm_config.= 'pm.max_children = '.$fpm_children."\n"; if ($fpm_pm == 'dynamic') { // honor max_children if ($fpm_children < $fpm_min_spare_servers) { $fpm_min_spare_servers = $fpm_children; } if ($fpm_children < $fpm_max_spare_servers) { $fpm_max_spare_servers = $fpm_children; } // failsafe, refs #955 if ($fpm_start_servers < $fpm_min_spare_servers) { $fpm_start_servers = $fpm_min_spare_servers; } if ($fpm_start_servers > $fpm_max_spare_servers) { $fpm_start_servers = $fpm_start_servers - (($fpm_start_servers - $fpm_max_spare_servers) + 1); } $fpm_config.= 'pm.start_servers = '.$fpm_start_servers."\n"; $fpm_config.= 'pm.min_spare_servers = '.$fpm_min_spare_servers."\n"; $fpm_config.= 'pm.max_spare_servers = '.$fpm_max_spare_servers."\n"; } elseif ($fpm_pm == 'ondemand') { $fpm_config.= 'pm.process_idle_timeout = '.$fpm_process_idle_timeout."\n"; } $fpm_config.= 'pm.max_requests = '.$fpm_requests."\n"; // possible slowlog configs if ($phpconfig['fpm_slowlog'] == '1') { $fpm_config.= 'request_terminate_timeout = ' . $phpconfig['fpm_reqterm'] . "\n"; $fpm_config.= 'request_slowlog_timeout = ' . $phpconfig['fpm_reqslow'] . "\n"; $slowlog = makeCorrectFile(Settings::Get('system.logfiles_directory') . '/' . $this->_domain['loginname'] . '-php-slow.log'); $fpm_config.= 'slowlog = ' . $slowlog . "\n"; $fpm_config.= 'catch_workers_output = yes' . "\n"; } $fpm_config.= ';chroot = '.makeCorrectDir($this->_domain['documentroot'])."\n"; $tmpdir = makeCorrectDir(Settings::Get('phpfpm.tmpdir') . '/' . $this->_domain['loginname'] . '/'); if (!is_dir($tmpdir)) { $this->getTempDir(); } $fpm_config.= 'env[TMP] = '.$tmpdir."\n"; $fpm_config.= 'env[TMPDIR] = '.$tmpdir."\n"; $fpm_config.= 'env[TEMP] = '.$tmpdir."\n"; $openbasedir = ''; if ($this->_domain['loginname'] != 'froxlor.panel') { if ($this->_domain['openbasedir'] == '1') { $_phpappendopenbasedir = ''; $_custom_openbasedir = explode(':', Settings::Get('phpfpm.peardir')); foreach ($_custom_openbasedir as $cobd) { $_phpappendopenbasedir .= appendOpenBasedirPath($cobd); } $_custom_openbasedir = explode(':', Settings::Get('system.phpappendopenbasedir')); foreach ($_custom_openbasedir as $cobd) { $_phpappendopenbasedir .= appendOpenBasedirPath($cobd); } if ($this->_domain['openbasedir_path'] == '0' && strstr($this->_domain['documentroot'], ":") === false ) { $openbasedir = appendOpenBasedirPath($this->_domain['documentroot'], true); } else { $openbasedir = appendOpenBasedirPath($this->_domain['customerroot'], true); } $openbasedir .= appendOpenBasedirPath($this->getTempDir()); $openbasedir .= $_phpappendopenbasedir; $openbasedir = explode(':', $openbasedir); $clean_openbasedir = array(); foreach ($openbasedir as $number => $path) { if (trim($path) != '/') { $clean_openbasedir[] = makeCorrectDir($path); } } $openbasedir = implode(':', $clean_openbasedir); } } $fpm_config.= 'php_admin_value[session.save_path] = ' . makeCorrectDir(Settings::Get('phpfpm.tmpdir') . '/' . $this->_domain['loginname'] . '/') . "\n"; $fpm_config.= 'php_admin_value[upload_tmp_dir] = ' . makeCorrectDir(Settings::Get('phpfpm.tmpdir') . '/' . $this->_domain['loginname'] . '/') . "\n"; $admin = $this->_getAdminData($this->_domain['adminid']); $php_ini_variables = array( 'SAFE_MODE' => 'Off', // keep this for compatibility, just in case 'PEAR_DIR' => Settings::Get('phpfpm.peardir'), 'TMP_DIR' => $this->getTempDir(), 'CUSTOMER_EMAIL' => $this->_domain['email'], 'ADMIN_EMAIL' => $admin['email'], 'DOMAIN' => $this->_domain['domain'], 'CUSTOMER' => $this->_domain['loginname'], 'ADMIN' => $admin['loginname'], 'OPEN_BASEDIR' => $openbasedir, 'OPEN_BASEDIR_C' => '' ); $phpini = replace_variables($phpconfig['phpsettings'], $php_ini_variables); $phpini_array = explode("\n", $phpini); $fpm_config.= "\n\n"; foreach ($phpini_array as $inisection) { $is = explode("=", $inisection); foreach ($this->_ini as $sec => $possibles) { if (in_array(trim($is[0]), $possibles)) { // check explictly for open_basedir if (trim($is[0]) == 'open_basedir' && $openbasedir == '') { continue; } $fpm_config.= $sec.'['.trim($is[0]).'] = ' . trim($is[1]) . "\n"; } } } // now check if 'sendmail_path' has not beed set in the custom-php.ini // if not we use our fallback-default as usual if (strpos($fpm_config, 'php_admin_value[sendmail_path]') === false) { $fpm_config.= 'php_admin_value[sendmail_path] = /usr/sbin/sendmail -t -i -f '.$this->_domain['email']."\n"; } fwrite($fh, $fpm_config, strlen($fpm_config)); fclose($fh); } } /** * this is done via createConfig as php-fpm defines * the ini-values/flags in its pool-config * * @param string $phpconfig */ public function createIniFile($phpconfig) { return; } /** * fpm-config file * * @param boolean $createifnotexists create the directory if it does not exist * * @return string the full path to the file */ public function getConfigFile($createifnotexists = true) { $configdir = makeCorrectDir(Settings::Get('phpfpm.configdir')); $config = makeCorrectFile($configdir.'/'.$this->_domain['domain'].'.conf'); if (!is_dir($configdir) && $createifnotexists) { safe_exec('mkdir -p ' . escapeshellarg($configdir)); } return $config; } /** * return path of fpm-socket file * * @param boolean $createifnotexists create the directory if it does not exist * * @return string the full path to the socket */ public function getSocketFile($createifnotexists = true) { $socketdir = makeCorrectDir(Settings::Get('phpfpm.fastcgi_ipcdir')); $socket = makeCorrectFile($socketdir.'/'.$this->_domain['loginname'].'-'.$this->_domain['domain'].'-php-fpm.socket'); if (!is_dir($socketdir) && $createifnotexists) { safe_exec('mkdir -p '.escapeshellarg($socketdir)); safe_exec('chown -R '.Settings::Get('system.httpuser').':'.Settings::Get('system.httpgroup').' '.escapeshellarg($socketdir)); } return $socket; } /** * fpm-temp directory * * @param boolean $createifnotexists create the directory if it does not exist * * @return string the directory */ public function getTempDir($createifnotexists = true) { $tmpdir = makeCorrectDir(Settings::Get('phpfpm.tmpdir') . '/' . $this->_domain['loginname'] . '/'); if (!is_dir($tmpdir) && $createifnotexists) { safe_exec('mkdir -p ' . escapeshellarg($tmpdir)); safe_exec('chown -R ' . $this->_domain['guid'] . ':' . $this->_domain['guid'] . ' ' . escapeshellarg($tmpdir)); safe_exec('chmod 0750 ' . escapeshellarg($tmpdir)); } return $tmpdir; } /** * fastcgi-fakedirectory directory * * @param boolean $createifnotexists create the directory if it does not exist * * @return string the directory */ public function getAliasConfigDir($createifnotexists = true) { // ensure default... if (Settings::Get('phpfpm.aliasconfigdir') == null) { Settings::Set('phpfpm.aliasconfigdir', '/var/www/php-fpm'); } $configdir = makeCorrectDir(Settings::Get('phpfpm.aliasconfigdir') . '/' . $this->_domain['loginname'] . '/' . $this->_domain['domain'] . '/'); if (!is_dir($configdir) && $createifnotexists) { safe_exec('mkdir -p ' . escapeshellarg($configdir)); safe_exec('chown ' . $this->_domain['guid'] . ':' . $this->_domain['guid'] . ' ' . escapeshellarg($configdir)); } return $configdir; } /** * return the admin-data of a specific admin * * @param int $adminid id of the admin-user * * @return array */ private function _getAdminData($adminid) { $adminid = intval($adminid); if (!isset($this->_admin_cache[$adminid])) { $stmt = Database::prepare(" SELECT `email`, `loginname` FROM `" . TABLE_PANEL_ADMINS . "` WHERE `adminid` = :id" ); $this->_admin_cache[$adminid] = Database::pexecute_first($stmt, array('id' => $adminid)); } return $this->_admin_cache[$adminid]; } }