mirror of
https://github.com/silverstripe/silverstripe-environmentcheck
synced 2024-10-22 17:05:40 +02:00
178 lines
4.6 KiB
PHP
178 lines
4.6 KiB
PHP
<?php
|
|
|
|
/**
|
|
* Interface for environment checks
|
|
*
|
|
* An environment check is a test that can be performed on a live environment. They differ from unit
|
|
* tests in that they are designed to check the state of the evnironment / server, rather than the code.
|
|
*
|
|
* Environment checks should *never* alter production data.
|
|
*
|
|
* Some things you might make environment checks for:
|
|
* - Can I access the database?
|
|
* - Are the right PHP modules installed?
|
|
* - Are the file permissions correct?
|
|
*/
|
|
interface EnvironmentCheck {
|
|
|
|
const ERROR = 3;
|
|
const WARNING = 2;
|
|
const OK = 1;
|
|
|
|
/**
|
|
* Perform this check
|
|
*
|
|
* @return 2 element array( $status, $message )
|
|
* $status is EnvironmentCheck::ERROR, EnvironmentCheck::WARNING, or EnvironmentCheck::OK
|
|
*/
|
|
function check();
|
|
|
|
}
|
|
|
|
/**
|
|
* Represents a suite of environment checks.
|
|
* Also has a register for assigning environment checks to named instances of EnvironmentCheckSuite
|
|
*
|
|
* Usage:
|
|
* EnvironmentCheckSuite::register('health', 'MyHealthCheck');
|
|
*
|
|
* $result = EnvironmentCheckSuite::inst('health')->run();
|
|
*/
|
|
class EnvironmentCheckSuite {
|
|
protected $checks = array();
|
|
|
|
/**
|
|
* Run this test suite
|
|
* @return The result code of the worst result.
|
|
*/
|
|
public function run() {
|
|
$worstResult = 0;
|
|
|
|
$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
|
|
*/
|
|
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.
|
|
*
|
|
*/
|
|
public function push($check, $title = null) {
|
|
if(!$title) {
|
|
$title = is_string($check) ? $check : get_class($check);
|
|
}
|
|
$this->checks[] = array($check, $title);
|
|
}
|
|
|
|
/////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
protected static $instances = array();
|
|
|
|
/**
|
|
* Return a named instance of EnvironmentCheckSuite.
|
|
*/
|
|
static function inst($name) {
|
|
if(!isset(self::$instances[$name])) self::$instances[$name] = new EnvironmentCheckSuite();
|
|
return self::$instances[$name];
|
|
}
|
|
|
|
/**
|
|
* Register a check against the named check suite.
|
|
*
|
|
* @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);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* A single set of results from running an EnvironmentCheckSuite
|
|
*/
|
|
class EnvironmentCheckSuiteResult extends ViewableData {
|
|
protected $details, $worst = 0;
|
|
|
|
function __construct() {
|
|
parent::__construct();
|
|
$this->details = new DataObjectSet();
|
|
}
|
|
|
|
function addResult($status, $message, $checkIdentifier) {
|
|
$this->details->push(new ArrayData(array(
|
|
'Check' => $checkIdentifier,
|
|
'Status' => $this->statusText($status),
|
|
'Message' => $message,
|
|
)));
|
|
|
|
$this->worst = max($this->worst, $status);
|
|
}
|
|
|
|
/**
|
|
* Returns true if there are no ERRORs, only WARNINGs or OK
|
|
*/
|
|
function ShouldPass() {
|
|
return $this->worst <= EnvironmentCheck::WARNING;
|
|
}
|
|
|
|
/**
|
|
* Returns overall (i.e. worst) status as a string.
|
|
*/
|
|
function Status() {
|
|
return $this->statusText($this->worst);
|
|
}
|
|
|
|
/**
|
|
* Returns detailed status information about each check
|
|
*/
|
|
function Details() {
|
|
return $this->details;
|
|
}
|
|
|
|
/**
|
|
* Return a text version of a status code
|
|
*/
|
|
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'");
|
|
}
|
|
}
|
|
} |