phpversion(), 2 => CRM_Upgrade_Incremental_General::MIN_RECOMMENDED_PHP_VER, )), ts('PHP Out-of-Date'), \Psr\Log\LogLevel::NOTICE, 'fa-server' ); } else { $messages[] = new CRM_Utils_Check_Message( __FUNCTION__, ts('This system uses PHP version %1 which meets or exceeds the minimum recommendation of %2.', array( 1 => phpversion(), 2 => CRM_Upgrade_Incremental_General::MIN_RECOMMENDED_PHP_VER, )), ts('PHP Up-to-Date'), \Psr\Log\LogLevel::INFO, 'fa-server' ); } return $messages; } /** * @return array */ public function checkPhpMysqli() { $messages = array(); if (!extension_loaded('mysqli')) { $messages[] = new CRM_Utils_Check_Message( __FUNCTION__, ts('Future versions of CiviCRM may require the PHP extension "%2". To ensure that your system will be compatible, please install it in advance. For more explanation, see the announcement.', array( 1 => 'https://civicrm.org/blog/totten/psa-please-verify-php-extension-mysqli', 2 => 'mysqli', )), ts('Forward Compatibility: Enable "mysqli"'), \Psr\Log\LogLevel::WARNING, 'fa-server' ); } return $messages; } /** * Check that the MySQL time settings match the PHP time settings. * * @return array an empty array, or a list of warnings */ public function checkMysqlTime() { $messages = array(); $phpNow = date('Y-m-d H:i'); $sqlNow = CRM_Core_DAO::singleValueQuery("SELECT date_format(now(), '%Y-%m-%d %H:%i')"); if (!CRM_Utils_Time::isEqual($phpNow, $sqlNow, 2.5 * 60)) { $messages[] = new CRM_Utils_Check_Message( __FUNCTION__, ts('Timestamps reported by MySQL (eg "%2") and PHP (eg "%3" ) are mismatched.
Read more about this warning', array( 1 => CRM_Utils_System::getWikiBaseURL() . 'checkMysqlTime', 2 => $sqlNow, 3 => $phpNow, )), ts('Timestamp Mismatch'), \Psr\Log\LogLevel::ERROR, 'fa-server' ); } return $messages; } /** * @return array */ public function checkDebug() { $messages = array(); $config = CRM_Core_Config::singleton(); if ($config->debug) { $messages[] = new CRM_Utils_Check_Message( __FUNCTION__, ts('Warning: Debug is enabled in system settings. This should not be enabled on production servers.', array(1 => CRM_Utils_System::url('civicrm/admin/setting/debug', 'reset=1'))), ts('Debug Mode Enabled'), \Psr\Log\LogLevel::WARNING, 'fa-bug' ); } return $messages; } /** * @return array */ public function checkOutboundMail() { $messages = array(); $mailingInfo = Civi::settings()->get('mailing_backend'); if (($mailingInfo['outBound_option'] == CRM_Mailing_Config::OUTBOUND_OPTION_REDIRECT_TO_DB || (defined('CIVICRM_MAIL_LOG') && CIVICRM_MAIL_LOG) || $mailingInfo['outBound_option'] == CRM_Mailing_Config::OUTBOUND_OPTION_DISABLED || $mailingInfo['outBound_option'] == CRM_Mailing_Config::OUTBOUND_OPTION_MOCK) ) { $messages[] = new CRM_Utils_Check_Message( __FUNCTION__, ts('Warning: Outbound email is disabled in system settings. Proper settings should be enabled on production servers.', array(1 => CRM_Utils_System::url('civicrm/admin/setting/smtp', 'reset=1'))), ts('Outbound Email Disabled'), \Psr\Log\LogLevel::WARNING, 'fa-envelope' ); } return $messages; } /** * Check that domain email and org name are set * @return array */ public function checkDomainNameEmail() { $messages = array(); list($domainEmailName, $domainEmailAddress) = CRM_Core_BAO_Domain::getNameAndEmail(TRUE); $domain = CRM_Core_BAO_Domain::getDomain(); $domainName = $domain->name; $fixEmailUrl = CRM_Utils_System::url("civicrm/admin/domain", "action=update&reset=1"); if (!$domainEmailAddress || $domainEmailAddress == 'info@EXAMPLE.ORG') { if (!$domainName || $domainName == 'Default Domain Name') { $msg = ts("Please enter your organization's name, primary address, and default FROM Email Address (for system-generated emails).", array(1 => $fixEmailUrl)); } else { $msg = ts('Please enter a default FROM Email Address (for system-generated emails).', array(1 => $fixEmailUrl)); } } elseif (!$domainName || $domainName == 'Default Domain Name') { $msg = ts("Please enter your organization's name and primary address.", array(1 => $fixEmailUrl)); } if (!empty($msg)) { $messages[] = new CRM_Utils_Check_Message( __FUNCTION__, $msg, ts('Complete Setup'), \Psr\Log\LogLevel::WARNING, 'fa-check-square-o' ); } return $messages; } /** * Checks if a default bounce handling mailbox is set up * @return array */ public function checkDefaultMailbox() { $messages = array(); $config = CRM_Core_Config::singleton(); if (in_array('CiviMail', $config->enableComponents) && CRM_Core_BAO_MailSettings::defaultDomain() == "EXAMPLE.ORG" ) { $message = new CRM_Utils_Check_Message( __FUNCTION__, ts('Please configure a default mailbox for CiviMail.', array(1 => CRM_Utils_System::url('civicrm/admin/mailSettings', "reset=1"))), ts('Configure Default Mailbox'), \Psr\Log\LogLevel::WARNING, 'fa-envelope' ); $docUrl = 'target="_blank" href="' . CRM_Utils_System::docURL(array('page' => 'user/advanced-configuration/email-system-configuration/', 'URLonly' => TRUE)) . '""'; $message->addHelp( ts('A default mailbox must be configured for email bounce processing.') . '
' . ts("Learn more in the online documentation.", array(1 => $docUrl)) ); $messages[] = $message; } return $messages; } /** * Checks if cron has run in a reasonable amount of time * @return array */ public function checkLastCron() { $messages = array(); $statusPreference = new CRM_Core_DAO_StatusPreference(); $statusPreference->domain_id = CRM_Core_Config::domainID(); $statusPreference->name = 'checkLastCron'; if ($statusPreference->find(TRUE) && !empty($statusPreference->check_info)) { $lastCron = $statusPreference->check_info; $msg = ts('Last cron run at %1.', array(1 => CRM_Utils_Date::customFormat(date('c', $lastCron)))); } else { $lastCron = 0; $msg = ts('No cron runs have been recorded.'); } if ($lastCron > gmdate('U') - 3600) { $messages[] = new CRM_Utils_Check_Message( __FUNCTION__, $msg, ts('Cron Running OK'), \Psr\Log\LogLevel::INFO, 'fa-clock-o' ); } else { $message = new CRM_Utils_Check_Message( __FUNCTION__, $msg, ts('Cron Not Running'), ($lastCron > gmdate('U') - 86400) ? \Psr\Log\LogLevel::WARNING : \Psr\Log\LogLevel::ERROR, 'fa-clock-o' ); $docUrl = 'target="_blank" href="' . CRM_Utils_System::docURL(array('resource' => 'wiki', 'page' => 'Managing Scheduled Jobs', 'URLonly' => TRUE)) . '""'; $message->addHelp( ts('Configuring cron on your server is necessary for running scheduled jobs such as sending mail and scheduled reminders.') . '
' . ts("Learn more in the online documentation.", array(1 => $docUrl)) ); $messages[] = $message; } return $messages; } /** * Checks if new versions are available * @return array */ public function checkVersion() { $messages = array(); try { $vc = new CRM_Utils_VersionCheck(); $vc->initialize(); } catch (Exception $e) { $messages[] = new CRM_Utils_Check_Message( 'checkVersionError', ts('Directory %1 is not writable. Please change your file permissions.', array(1 => dirname($vc->cacheFile))), ts('Directory not writable'), \Psr\Log\LogLevel::ERROR, 'fa-times-circle-o' ); return $messages; } // Show a notice if the version_check job is disabled if (empty($vc->cronJob['is_active'])) { $args = empty($vc->cronJob['id']) ? array('reset' => 1) : array('reset' => 1, 'action' => 'update', 'id' => $vc->cronJob['id']); $messages[] = new CRM_Utils_Check_Message( 'checkVersionDisabled', ts('The check for new versions of CiviCRM has been disabled. Re-enable the scheduled job to receive important security update notifications.', array(1 => 'href="' . CRM_Utils_System::url('civicrm/admin/job', $args) . '"')), ts('Update Check Disabled'), \Psr\Log\LogLevel::NOTICE, 'fa-times-circle-o' ); } if ($vc->isInfoAvailable) { $newerVersion = $vc->isNewerVersionAvailable(); if ($newerVersion['version']) { $vInfo = array( 1 => $newerVersion['version'], 2 => $vc->localVersion, ); // LTS = long-term support version if ($newerVersion['status'] == 'lts') { $vInfo[1] .= ' ' . ts('(long-term support)'); } if ($newerVersion['upgrade'] == 'security') { // Security $severity = \Psr\Log\LogLevel::CRITICAL; $title = ts('CiviCRM Security Update Required'); $message = ts('New security release %1 is available. The site is currently running %2.', $vInfo); } elseif ($newerVersion['status'] == 'eol') { // Warn about EOL $severity = \Psr\Log\LogLevel::WARNING; $title = ts('CiviCRM Update Needed'); $message = ts('New version %1 is available. The site is currently running %2, which has reached its end of life.', $vInfo); } else { // For most new versions, just make them notice $severity = \Psr\Log\LogLevel::NOTICE; $title = ts('CiviCRM Update Available'); $message = ts('New version %1 is available. The site is currently running %2.', $vInfo); } } elseif (!empty($vc->cronJob['is_active'])) { $vNum = $vc->localVersion; // LTS = long-term support version if ($newerVersion['status'] == 'lts') { $vNum .= ' ' . ts('(long-term support)'); } $severity = \Psr\Log\LogLevel::INFO; $title = ts('CiviCRM Up-to-Date'); $message = ts('CiviCRM version %1 is up-to-date.', array(1 => $vNum)); } if (!empty($message)) { $messages[] = new CRM_Utils_Check_Message( __FUNCTION__, $message, $title, $severity, 'fa-cloud-upload' ); } } return $messages; } /** * Checks if extensions are set up properly * @return array */ public function checkExtensions() { $messages = array(); $extensionSystem = CRM_Extension_System::singleton(); $mapper = $extensionSystem->getMapper(); $manager = $extensionSystem->getManager(); if ($extensionSystem->getDefaultContainer()) { $basedir = $extensionSystem->getDefaultContainer()->baseDir; } if (empty($basedir)) { // no extension directory $messages[] = new CRM_Utils_Check_Message( __FUNCTION__, ts('Your extensions directory is not set. Click here to set the extensions directory.', array(1 => CRM_Utils_System::url('civicrm/admin/setting/path', 'reset=1'))), ts('Directory not writable'), \Psr\Log\LogLevel::NOTICE, 'fa-plug' ); return $messages; } if (!is_dir($basedir)) { $messages[] = new CRM_Utils_Check_Message( __FUNCTION__, ts('Your extensions directory path points to %1, which is not a directory. Please check your file system.', array(1 => $basedir)), ts('Extensions directory incorrect'), \Psr\Log\LogLevel::ERROR, 'fa-plug' ); return $messages; } elseif (!is_writable($basedir)) { $messages[] = new CRM_Utils_Check_Message( __FUNCTION__, ts('Directory %1 is not writable. Please change your file permissions.', array(1 => $basedir)), ts('Directory not writable'), \Psr\Log\LogLevel::ERROR, 'fa-plug' ); return $messages; } if (empty($extensionSystem->getDefaultContainer()->baseUrl)) { $messages[] = new CRM_Utils_Check_Message( __FUNCTION__, ts('The extensions URL is not properly set. Please go to the URL setting page and correct it.', array(1 => CRM_Utils_System::url('civicrm/admin/setting/url', 'reset=1'))), ts('Extensions url missing'), \Psr\Log\LogLevel::ERROR, 'fa-plug' ); return $messages; } if (!$extensionSystem->getBrowser()->isEnabled()) { $messages[] = new CRM_Utils_Check_Message( __FUNCTION__, ts('Not checking remote URL for extensions since ext_repo_url is set to false.'), ts('Extensions check disabled'), \Psr\Log\LogLevel::NOTICE, 'fa-plug' ); return $messages; } try { $remotes = $extensionSystem->getBrowser()->getExtensions(); } catch (CRM_Extension_Exception $e) { $messages[] = new CRM_Utils_Check_Message( __FUNCTION__, $e->getMessage(), ts('Extension download error'), \Psr\Log\LogLevel::ERROR, 'fa-plug' ); return $messages; } if (!$remotes) { // CRM-13141 There may not be any compatible extensions available for the requested CiviCRM version + CMS. If so, $extdir is empty so just return a notice. $messages[] = new CRM_Utils_Check_Message( __FUNCTION__, ts('There are currently no extensions on the CiviCRM public extension directory which are compatible with version %1. If you want to install an extension which is not marked as compatible, you may be able to download and install extensions manually (depending on access to your web server).', array( 1 => CRM_Utils_System::majorVersion(), 2 => 'href="http://wiki.civicrm.org/confluence/display/CRMDOC/Extensions"', )), ts('No Extensions Available for this Version'), \Psr\Log\LogLevel::NOTICE, 'fa-plug' ); return $messages; } $keys = array_keys($manager->getStatuses()); sort($keys); $updates = $errors = $okextensions = array(); foreach ($keys as $key) { try { $obj = $mapper->keyToInfo($key); } catch (CRM_Extension_Exception $ex) { $errors[] = ts('Failed to read extension (%1). Please refresh the extension list.', array(1 => $key)); continue; } $row = CRM_Admin_Page_Extensions::createExtendedInfo($obj); switch ($row['status']) { case CRM_Extension_Manager::STATUS_INSTALLED_MISSING: $errors[] = ts('%1 extension (%2) is installed but missing files.', array(1 => CRM_Utils_Array::value('label', $row), 2 => $key)); break; case CRM_Extension_Manager::STATUS_INSTALLED: if (!empty($remotes[$key]) && version_compare($row['version'], $remotes[$key]->version, '<')) { $updates[] = ts('%1 (%2) version %3 is installed. Upgrade to version %5.', array( 1 => CRM_Utils_Array::value('label', $row), 2 => $key, 3 => $row['version'], 4 => 'href="' . CRM_Utils_System::url('civicrm/admin/extensions', "action=update&id=$key&key=$key") . '"', 5 => $remotes[$key]->version, )); } else { if (empty($row['label'])) { $okextensions[] = $key; } else { $okextensions[] = ts('%1 (%2) version %3', array( 1 => $row['label'], 2 => $key, 3 => $row['version'], )); } } break; } } if (!$okextensions && !$updates && !$errors) { $messages[] = new CRM_Utils_Check_Message( 'extensionsOk', ts('No extensions installed. Browse available extensions.', array( 1 => 'href="' . CRM_Utils_System::url('civicrm/admin/extensions', 'reset=1') . '"', )), ts('Extensions'), \Psr\Log\LogLevel::INFO, 'fa-plug' ); } if ($errors) { $messages[] = new CRM_Utils_Check_Message( __FUNCTION__, '', ts('Extension Error'), \Psr\Log\LogLevel::ERROR, 'fa-plug' ); } if ($updates) { $messages[] = new CRM_Utils_Check_Message( 'extensionUpdates', '', ts('Extension Update Available', array('plural' => '%count Extension Updates Available', 'count' => count($updates))), \Psr\Log\LogLevel::WARNING, 'fa-plug' ); } if ($okextensions) { if ($updates || $errors) { $message = ts('1 extension is up-to-date:', array('plural' => '%count extensions are up-to-date:', 'count' => count($okextensions))); } else { $message = ts('All extensions are up-to-date:'); } $messages[] = new CRM_Utils_Check_Message( 'extensionsOk', $message . '', ts('Extensions'), \Psr\Log\LogLevel::INFO, 'fa-plug' ); } return $messages; } /** * Checks if extensions are set up properly * @return array */ public function checkExtensionUpgrades() { $messages = array(); if (CRM_Extension_Upgrades::hasPending()) { $messages[] = new CRM_Utils_Check_Message( __FUNCTION__, ts('Extension upgrades are pending. Please visit the upgrade page to run them.', array(1 => CRM_Utils_System::url('civicrm/admin/extensions/upgrade', 'reset=1'))), ts('Run Extension Upgrades'), \Psr\Log\LogLevel::ERROR, 'fa-plug' ); } return $messages; } /** * Checks if CiviCRM database version is up-to-date * @return array */ public function checkDbVersion() { $messages = array(); $dbVersion = CRM_Core_BAO_Domain::version(); $upgradeUrl = CRM_Utils_System::url("civicrm/upgrade", "reset=1"); if (!$dbVersion) { // if db.ver missing $messages[] = new CRM_Utils_Check_Message( __FUNCTION__, ts('Version information found to be missing in database. You will need to determine the correct version corresponding to your current database state.'), ts('Database Version Missing'), \Psr\Log\LogLevel::ERROR, 'fa-database' ); } elseif (!CRM_Utils_System::isVersionFormatValid($dbVersion)) { $messages[] = new CRM_Utils_Check_Message( __FUNCTION__, ts('Database is marked with invalid version format. You may want to investigate this before you proceed further.'), ts('Database Version Invalid'), \Psr\Log\LogLevel::ERROR, 'fa-database' ); } elseif (stripos($dbVersion, 'upgrade')) { // if db.ver indicates a partially upgraded db $messages[] = new CRM_Utils_Check_Message( __FUNCTION__, ts('Database check failed - the database looks to have been partially upgraded. You must reload the database with the backup and try the upgrade process again.', array(1 => $upgradeUrl)), ts('Database Partially Upgraded'), \Psr\Log\LogLevel::ALERT, 'fa-database' ); } else { $codeVersion = CRM_Utils_System::version(); // if db.ver < code.ver, time to upgrade if (version_compare($dbVersion, $codeVersion) < 0) { $messages[] = new CRM_Utils_Check_Message( __FUNCTION__, ts('New codebase version detected. You must visit upgrade screen to upgrade the database.', array(1 => $upgradeUrl)), ts('Database Upgrade Required'), \Psr\Log\LogLevel::ALERT, 'fa-database' ); } // if db.ver > code.ver, sth really wrong if (version_compare($dbVersion, $codeVersion) > 0) { $messages[] = new CRM_Utils_Check_Message( __FUNCTION__, ts('Your database is marked with an unexpected version number: %1. The v%2 codebase may not be compatible with your database state. You will need to determine the correct version corresponding to your current database state. You may want to revert to the codebase you were using until you resolve this problem.
OR if this is a manual install from git, you might want to fix civicrm-version.php file.', array(1 => $dbVersion, 2 => $codeVersion) ), ts('Database In Unexpected Version'), \Psr\Log\LogLevel::ERROR, 'fa-database' ); } } return $messages; } /** * ensure that all CiviCRM tables are InnoDB * @return array */ public function checkDbEngine() { $messages = array(); if (CRM_Core_DAO::isDBMyISAM(150)) { $messages[] = new CRM_Utils_Check_Message( __FUNCTION__, ts('Your database is configured to use the MyISAM database engine. CiviCRM requires InnoDB. You will need to convert any MyISAM tables in your database to InnoDB. Using MyISAM tables will result in data integrity issues.'), ts('MyISAM Database Engine'), \Psr\Log\LogLevel::ERROR, 'fa-database' ); } return $messages; } }