| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332 | <?phpnamespace 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;    }}
 |