123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332 |
- <?php
- namespace Grav\Plugin;
- use Grav\Common\Cache;
- use Grav\Common\Plugin;
- use Grav\Common\Uri;
- class ProblemsPlugin extends Plugin
- {
- protected $results = array();
- protected $check;
- /**
- * @return array
- */
- public static function getSubscribedEvents()
- {
- return [
- 'onPluginsInitialized' => ['onPluginsInitialized', 100001],
- 'onFatalException' => ['onFatalException', 0]
- ];
- }
- public function onFatalException()
- {
- if ($this->isAdmin()) {
- $this->active = false;
- return;
- }
- // Run through potential issues
- if ($this->problemChecker()) {
- $this->renderProblems();
- }
- }
- public function onPluginsInitialized()
- {
- if ($this->isAdmin()) {
- $this->active = false;
- return;
- }
- /** @var Cache $cache */
- $cache = $this->grav['cache'];
- $validated_prefix = 'problem-check-';
- $this->check = CACHE_DIR . $validated_prefix . $cache->getKey();
- if (!file_exists($this->check)) {
- // If no issues remain, save a state file in the cache
- if (!$this->problemChecker()) {
- // delete any existing validated files
- foreach (new \GlobIterator(CACHE_DIR . $validated_prefix . '*') as $fileInfo) {
- @unlink($fileInfo->getPathname());
- }
- // create a file in the cache dir so it only runs on cache changes
- touch($this->check);
- } else {
- $this->renderProblems();
- }
- }
- }
- protected function renderProblems()
- {
- $theme = 'antimatter';
- /** @var Uri $uri */
- $uri = $this->grav['uri'];
- $baseUrlRelative = $uri->rootUrl(false);
- $themeUrl = $baseUrlRelative . '/' . USER_PATH . basename(THEMES_DIR) . '/' . $theme;
- $problemsUrl = $baseUrlRelative . '/user/plugins/problems';
- $html = file_get_contents(__DIR__ . '/html/problems.html');
- /**
- * Process the results, ignore the statuses passed as $ignore_status
- *
- * @param $results
- * @param $ignore_status
- */
- $processResults = function ($results, $ignore_status) {
- $problems = '';
- foreach ($results as $key => $result) {
- if ($key == 'files' || $key == 'apache' || $key == 'execute') {
- foreach ($result as $key_text => $value_text) {
- foreach ($value_text as $status => $text) {
- if ($status == $ignore_status) continue;
- $problems .= $this->getListRow($status, '<b>' . $key_text . '</b> ' . $text);
- }
- }
- } else {
- foreach ($result as $status => $text) {
- if ($status == $ignore_status) continue;
- $problems .= $this->getListRow($status, $text);
- }
- }
- }
- return $problems;
- };
- // First render the errors
- $problems = $processResults($this->results, 'success');
- // Then render the successful checks
- $problems .= $processResults($this->results, 'error');
- $html = str_replace('%%BASE_URL%%', $baseUrlRelative, $html);
- $html = str_replace('%%THEME_URL%%', $themeUrl, $html);
- $html = str_replace('%%PROBLEMS_URL%%', $problemsUrl, $html);
- $html = str_replace('%%PROBLEMS%%', $problems, $html);
- echo $html;
- http_response_code(500);
- exit();
- }
- protected function getListRow($status, $text)
- {
- if ($status == 'error') {
- $icon = 'fa-times';
- } elseif ($status == 'info') {
- $icon = 'fa-info';
- } else {
- $icon = 'fa-check';
- }
- $output = "\n";
- $output .= '<li class="' . $status . ' clearfix"><i class="fa fa-fw '. $icon . '"></i><p>'. $text . '</p></li>';
- return $output;
- }
- protected function problemChecker()
- {
- $min_php_version = defined('GRAV_PHP_MIN') ? GRAV_PHP_MIN : '5.4.0';
- $problems_found = false;
- $essential_files = [
- 'cache' => true,
- 'logs' => true,
- 'images' => true,
- 'assets' => true,
- 'system' => false,
- 'user/data' => true,
- 'user/pages' => false,
- 'user/config' => false,
- 'user/plugins/error' => false,
- 'user/plugins' => false,
- 'user/themes' => false,
- 'vendor' => false
- ];
- if (version_compare(GRAV_VERSION, '0.9.27', ">=")) {
- $essential_files['backup'] = true;
- $backup_folder = ROOT_DIR . 'backup';
- // try to create backup folder if missing
- if (!file_exists($backup_folder)) {
- @mkdir($backup_folder, 0770);
- }
- }
- if (version_compare(GRAV_VERSION, '1.1.4', ">=")) {
- $essential_files['tmp'] = true;
- $tmp_folder = ROOT_DIR . 'tmp';
- // try to create tmp folder if missing
- if (!file_exists($tmp_folder)) {
- @mkdir($tmp_folder, 0770);
- }
- }
- // Perform some Apache checks
- if (strpos(php_sapi_name(), 'apache') !== false) {
- $require_apache_modules = ['mod_rewrite'];
- $apache_modules = apache_get_modules();
- $apache_status = [];
- foreach ($require_apache_modules as $module) {
- if (in_array($module, $apache_modules)) {
- $apache_module_adjective = ' Apache module is enabled';
- $apache_module_status = 'success';
- } else {
- $problems_found = true;
- $apache_module_adjective = ' Apache module is not installed or enabled';
- $apache_module_status = 'error';
- }
- $apache_status[$module] = [$apache_module_status => $apache_module_adjective];
- }
- if (sizeof($apache_status) > 0) {
- $this->results['apache'] = $apache_status;
- }
- }
- // Check PHP version
- if (version_compare(phpversion(), $min_php_version, '<')) {
- $problems_found = true;
- $php_version_adjective = 'lower';
- $php_version_status = 'error';
- } else {
- $php_version_adjective = 'greater';
- $php_version_status = 'success';
- }
- $this->results['php'] = [$php_version_status => 'Your PHP version (' . phpversion() . ') is '. $php_version_adjective . ' than the minimum required: <b>' . $min_php_version . '</b> - <a href="http://getgrav.org/blog/changing-php-requirements-to-5.5">Additional Information</a>'];
- // Check for GD library
- if (defined('GD_VERSION') && function_exists('gd_info')) {
- $gd_adjective = '';
- $gd_status = 'success';
- } else {
- $problems_found = true;
- $gd_adjective = 'not ';
- $gd_status = 'error';
- }
- $this->results['gd'] = [$gd_status => 'PHP GD (Image Manipulation Library) is '. $gd_adjective . 'installed'];
- // Check for PHP CURL library
- if (function_exists('curl_version')) {
- $curl_adjective = '';
- $curl_status = 'success';
- } else {
- $problems_found = true;
- $curl_adjective = 'not ';
- $curl_status = 'error';
- }
- $this->results['curl'] = [$curl_status => 'PHP Curl (Data Transfer Library) is '. $curl_adjective . 'installed'];
- // Check for PHP Open SSL library
- if (extension_loaded('openssl') && defined('OPENSSL_VERSION_TEXT')) {
- $ssl_adjective = '';
- $ssl_status = 'success';
- } else {
- $problems_found = true;
- $ssl_adjective = 'not ';
- $ssl_status = 'error';
- }
- $this->results['ssl'] = [$ssl_status => 'PHP OpenSSL (Secure Sockets Library) is '. $ssl_adjective . 'installed'];
- // Check for PHP XML library
- if (extension_loaded('xml')) {
- $xml_adjective = '';
- $xml_status = 'success';
- } else {
- $problems_found = true;
- $xml_adjective = 'not ';
- $xml_status = 'error';
- }
- $this->results['xml'] = [$xml_status => 'PHP XML Library is '. $xml_adjective . 'installed'];
- // Check for PHP MbString library
- if (extension_loaded('mbstring')) {
- $mbstring_adjective = '';
- $mbstring_status = 'success';
- } else {
- $problems_found = true;
- $mbstring_adjective = 'not ';
- $mbstring_status = 'error';
- }
- $this->results['mbstring'] = [$mbstring_status => 'PHP Mbstring (Multibyte String Library) is '. $mbstring_adjective . 'installed'];
- // Check Exif if enabled
- if ($this->grav['config']->get('system.media.auto_metadata_exif')) {
- if(extension_loaded('exif')) {
- $exif_adjective = '';
- $exif_status = 'success';
- } else {
- $problems_found = true;
- $exif_adjective = 'not ';
- $exif_status = 'error';
- }
- $this->results['exif'] = [$exif_status => 'PHP Exif (Exchangeable Image File Format) is '. $exif_adjective . 'installed'];
- }
- // Check for PHP Zip library
- if (extension_loaded('zip')) {
- $zip_adjective = '';
- $zip_status = 'success';
- } else {
- $problems_found = true;
- $zip_adjective = 'not ';
- $zip_status = 'error';
- }
- $this->results['zip'] = [$zip_status => 'PHP Zip extension is '. $zip_adjective . 'installed'];
- // Check for essential files & perms
- $file_problems = [];
- foreach ($essential_files as $file => $check_writable) {
- $file_path = ROOT_DIR . $file;
- $is_dir = false;
- if (!file_exists($file_path)) {
- $problems_found = true;
- $file_status = 'error';
- $file_adjective = 'does not exist';
- } else {
- $file_status = 'success';
- $file_adjective = 'exists';
- $is_writeable = is_writable($file_path);
- $is_dir = is_dir($file_path);
- if ($check_writable) {
- if (!$is_writeable) {
- $file_status = 'error';
- $problems_found = true;
- $file_adjective .= ' but is <b class="underline">not writeable</b>';
- } else {
- $file_adjective .= ' and <b class="underline">is writeable</b>';
- }
- }
- }
- $file_problems[$file_path] = [$file_status => $file_adjective];
- }
- if (sizeof($file_problems) > 0) {
- $this->results['files'] = $file_problems;
- }
- return $problems_found;
- }
- }
|