run(); */ class EnvironmentCheckSuite extends SS_Object { /** * Name of this suite. * * @var string */ protected $name; /** * @var array */ protected $checks = array(); /** * Associative array of named checks registered via the config system. Each check should specify: * - definition (e.g. 'MyHealthCheck("my param")') * - title (e.g. 'Is my feature working?') * - state (setting this to 'disabled' will cause suites to skip this check entirely. * * @var array */ private static $registered_checks = array(); /** * Associative array of named suites registered via the config system. Each suite should enumerate * named checks that have been configured in 'registered_checks'. * * @var array */ private static $registered_suites = array(); /** * Load checks for this suite from the configuration system. This is an alternative to the * EnvironmentCheckSuite::register - both can be used, checks will be appended to the suite. * * @param string $suiteName The name of this suite. */ public function __construct($suiteName) { parent::__construct(); if (empty($this->config()->registered_suites[$suiteName])) { // Not registered via config system, but it still may be configured later via self::register. return; } foreach ($this->config()->registered_suites[$suiteName] as $checkName) { if (empty($this->config()->registered_checks[$checkName])) { throw new InvalidArgumentException( "Bad EnvironmentCheck: '$checkName' - the named check has not been registered." ); } $check = $this->config()->registered_checks[$checkName]; // Existing named checks can be disabled by setting their 'state' to 'disabled'. // This is handy for disabling checks mandated by modules. if (!empty($check['state']) && $check['state']==='disabled') { continue; } // Add the check to this suite. $this->push($check['definition'], $check['title']); } } /** * Run this test suite and return the result code of the worst result. * * @return int */ public function run() { $result = new EnvironmentCheckSuiteResult(); foreach ($this->checkInstances() as $check) { list($checkClass, $checkTitle) = $check; try { list($status, $message) = $checkClass->check(); // If the check fails, register that as an error } catch (Exception $e) { $status = EnvironmentCheck::ERROR; $message = $e->getMessage(); } $result->addResult($status, $message, $checkTitle); } return $result; } /** * Get instances of all the environment checks. * * @return array */ protected function checkInstances() { $output = array(); foreach ($this->checks as $check) { list($checkClass, $checkTitle) = $check; if (is_string($checkClass)) { $checkInst = SS_Object::create_from_string($checkClass); if ($checkInst instanceof EnvironmentCheck) { $output[] = array($checkInst, $checkTitle); } else { throw new InvalidArgumentException("Bad EnvironmentCheck: '$checkClass' - the named class doesn't implement EnvironmentCheck"); } } elseif ($checkClass instanceof EnvironmentCheck) { $output[] = array($checkClass, $checkTitle); } else { throw new InvalidArgumentException("Bad EnvironmentCheck: " . var_export($check, true)); } } return $output; } /** * Add a check to this suite. * * @param mixed $check * @param string $title */ public function push($check, $title = null) { if (!$title) { $title = is_string($check) ? $check : get_class($check); } $this->checks[] = array($check, $title); } ///////////////////////////////////////////////////////////////////////////////////////////// /** * @var array */ protected static $instances = array(); /** * Return a named instance of EnvironmentCheckSuite. * * @param string $name * * @return EnvironmentCheckSuite */ public static function inst($name) { if (!isset(self::$instances[$name])) { self::$instances[$name] = new EnvironmentCheckSuite($name); } return self::$instances[$name]; } /** * Register a check against the named check suite. * * @param string|array $names * @param EnvironmentCheck $check * @param string|array */ public static function register($names, $check, $title = null) { if (!is_array($names)) { $names = array($names); } foreach ($names as $name) { self::inst($name)->push($check, $title); } } /** * Unregisters all checks. */ public static function reset() { self::$instances = array(); } } /** * A single set of results from running an EnvironmentCheckSuite */ class EnvironmentCheckSuiteResult extends ViewableData { /** * @var ArrayList */ protected $details; /** * @var int */ protected $worst = 0; public function __construct() { parent::__construct(); $this->details = new ArrayList(); } /** * @param int $status * @param string $message * @param string $checkIdentifier */ public function addResult($status, $message, $checkIdentifier) { $this->details->push(new ArrayData(array( 'Check' => $checkIdentifier, 'Status' => $this->statusText($status), 'StatusCode' => $status, 'Message' => $message, ))); $this->worst = max($this->worst, $status); } /** * Returns true if there are no errors. * * @return bool */ public function ShouldPass() { return $this->worst <= EnvironmentCheck::WARNING; } /** * Returns overall (i.e. worst) status as a string. * * @return string */ public function Status() { return $this->statusText($this->worst); } /** * Returns detailed status information about each check. * * @return ArrayList */ public function Details() { return $this->details; } /** * Convert the final result status and details to JSON. * * @return string */ public function toJSON() { $result = array( 'Status' => $this->Status(), 'ShouldPass' => $this->ShouldPass(), 'Checks' => array() ); foreach ($this->details as $detail) { $result['Checks'][] = $detail->toMap(); } return json_encode($result); } /** * Return a text version of a status code. * * @return string */ protected function statusText($status) { switch ($status) { case EnvironmentCheck::ERROR: return "ERROR"; case EnvironmentCheck::WARNING: return "WARNING"; case EnvironmentCheck::OK: return "OK"; case 0: return "NO CHECKS"; default: throw new InvalidArgumentException("Bad environment check status '$status'"); } } }