mirror of
https://github.com/silverstripe/silverstripe-environmentcheck
synced 2024-10-22 15:05:40 +00:00
81c7f2ba27
Gives more granular control over monitoring failures
279 lines
6.8 KiB
PHP
279 lines
6.8 KiB
PHP
<?php
|
|
|
|
/**
|
|
* Represents a suite of environment checks.
|
|
* Specific checks can be registered against a named instance of EnvironmentCheckSuite.
|
|
*
|
|
* Usage #1 - _config.php
|
|
* EnvironmentCheckSuite::register('health', 'MyHealthCheck("my param")', 'Title of my health check');
|
|
*
|
|
* Usage #2 - config.yml
|
|
* EnvironmentCheckSuite:
|
|
* registered_checks:
|
|
* mycheck:
|
|
* definition: 'MyHealthCheck("my param")'
|
|
* title: 'Title of my health check'
|
|
* registered_suites:
|
|
* health:
|
|
* - mycheck
|
|
*
|
|
* $result = EnvironmentCheckSuite::inst('health')->run();
|
|
*/
|
|
class EnvironmentCheckSuite extends 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 = 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");
|
|
}
|
|
} else if($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
|
|
*/
|
|
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
|
|
*/
|
|
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.
|
|
*/
|
|
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;
|
|
|
|
function __construct() {
|
|
parent::__construct();
|
|
$this->details = new ArrayList();
|
|
}
|
|
|
|
/**
|
|
* @param int $status
|
|
* @param string $message
|
|
* @param string $checkIdentifier
|
|
*/
|
|
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
|
|
*/
|
|
function Status() {
|
|
return $this->statusText($this->worst);
|
|
}
|
|
|
|
/**
|
|
* Returns detailed status information about each check.
|
|
*
|
|
* @return ArrayList
|
|
*/
|
|
function Details() {
|
|
return $this->details;
|
|
}
|
|
|
|
/**
|
|
* Convert the final result status and details to JSON.
|
|
*
|
|
* @return string
|
|
*/
|
|
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'");
|
|
}
|
|
}
|
|
}
|