Files
sihnon-php-lib/source/lib/SihnonFramework/Main.class.php
Ben Roberts 1d8609573b Add option to mask known PHP warnings by pattern matching
Useful to mask things like 'mysql has gone away' and method signature overrides
2012-01-12 00:25:05 +00:00

507 lines
17 KiB
PHP

<?php
class SihnonFramework_Main {
protected static $instance;
protected static $autoload_classes = array(
array(
'base' => 'SihnonFramework',
'base_dir_prefix' => SihnonFramework_Lib,
'subclass' => 'Sihnon',
'subclass_dir_prefix' => Sihnon_Lib,
),
);
protected $config;
protected $database;
protected $log;
protected $cache;
protected $session;
protected $auth;
protected $base_uri;
protected function __construct() {
$dirname = dirname($_SERVER['SCRIPT_NAME']);
$this->base_uri = $dirname == '/' ? '/' : $dirname . '/';
}
protected function init() {
if (Sihnon_DatabaseSupport) {
$this->database = new Sihnon_Database(Sihnon_DBConfig);
}
$this->config = new Sihnon_Config(Sihnon_ConfigPlugin, array(
'database' => $this->database,
'table' => Sihnon_ConfigTable,
'filename' => Sihnon_ConfigFile)
);
$this->log = new Sihnon_Log($this->config);
$this->cache = new Sihnon_Cache($this->config);
$this->session = new Sihnon_Session($this->config);
$this->auth = new Sihnon_Auth($this->config, $this->session);
}
/**
*
* @return SihnonFramework_Main
*/
public static function instance() {
if (!self::$instance) {
$called_class = get_called_class();
self::$instance = new $called_class();
self::$instance->init();
}
return self::$instance;
}
/**
*
* @return SihnonFramework_Config
*/
public function config() {
return $this->config;
}
/**
*
* @return SihnonFramework_Database
*/
public function database() {
return $this->database;
}
/**
*
* @return SihnonFramework_Log
*/
public function log() {
return $this->log;
}
/**
*
* @return SihnonFramework_Cache
*/
public function cache() {
return $this->cache;
}
/**
*
* @return SihnonFramework_Session
*/
public function session() {
return $this->session;
}
/**
*
* @return SihnonFramework_Auth
*/
public function auth() {
return $this->auth;
}
public function baseUri() {
return $this->base_uri;
}
public function absoluteUrl($relative_url) {
$secure = isset($_SERVER['SECURE']) || (isset($_SERVER['HTTPS']) && $_SERVER['HTTPS'] == 'on');
$port = $_SERVER['SERVER_PORT'];
return 'http' . ($secure ? 's' : '') . '://'
. $_SERVER['HTTP_HOST'] . (($port == 80 || ($secure && $port == 443)) ? '' : ':' . $port)
. $this->base_uri . $relative_url;
}
public static function initialise() {
// Provide a means to load framework classes autonomously
spl_autoload_register(array('SihnonFramework_Main','autoload'));
// Handle error messages using custom error handler
set_error_handler(array(get_called_class(), 'errorHandler'));
}
public static function errorHandler($errno, $errstr, $errfile = '', $errline = 0, $errcontext = array()) {
$ignore_patterns = array(
/* Really don't consider this to be a bug, and make extensive use of this in the plugin architecture */
'/^Declaration of .* should be compatible with that of .*/',
/* This error is already handled in Database, but the warning is still triggered by PHP */
'/^PDO::__construct.*: MySQL server has gone away/',
);
foreach ($ignore_patterns as $pattern) {
if (preg_match($pattern, $errstr)) {
if (defined('Sihnon_MaskKnownErrors') && Sihnon_MaskKnownErrors) {
return false;
}
}
}
$severity = '';
switch ($errno) {
case E_NOTICE:
case E_USER_NOTICE:
$severity = SihnonFramework_Log::LEVEL_INFO;
break;
case E_WARNING:
case E_USER_WARNING:
$severity = SihnonFramework_Log::LEVEL_WARNING;
break;
case E_ERROR:
case E_USER_ERROR:
$severity = SihnonFramework_Log::LEVEL_ERROR;
break;
default:
$severity = SihnonFramework_Log::LEVEL_INFO;
break;
}
SihnonFramework_LogEntry::logInternal(static::instance()->log(), $severity, $errfile, $errline, $errstr, SihnonFramework_Log::CATEGORY_DEFAULT);
// If dev mode is enabled, fail here to enable the normal PHP error handling
return ! Sihnon_Dev;
}
public static function autoload($classname) {
// Ensure the classname contains only valid class name characters
if (!preg_match('/^[A-Z][a-zA-Z0-9_]*$/', $classname)) {
throw new Exception('Illegal characters in classname');
}
foreach (self::$autoload_classes as $class) {
// Ensure the class to load begins with our prefix
if (preg_match("/^{$class['base']}_/", $classname)) {
// Special case: all related exceptions are grouped into a single file
if (preg_match("/^({$class['base']}_(?:.*?_)?)Exception/", $classname, $matches)) {
$exceptions_filename = /*$class['base_dir_prefix'] .*/ preg_replace('/_/', '/', $matches[1]) . 'Exceptions.class.php';
if (stream_resolve_include_path($exceptions_filename)) {
require_once($exceptions_filename);
}
return;
}
// Replace any underscores with directory separators
$filename = /*$class['base_dir_prefix'] .*/ preg_replace('/_/', '/', $classname) . '.class.php';
// If this file exists, load it
if (stream_resolve_include_path($filename)) {
require_once($filename);
return;
}
// Try again without the .class suffix
$filename = /*$class['base_dir_prefix'] .*/ preg_replace('/_/', '/', $classname) . '.php';
// If this file exists, load it
if (stream_resolve_include_path($filename)) {
require_once($filename);
return;
}
} elseif ($class['subclass'] && preg_match("/^{$class['subclass']}_/", $classname)) {
// Sihnon_ classes subclass the SihnonFramework_ classes.
// If a subclass doesn't exist, create it on the fly
// Special case: all related exceptions are grouped into a single file
if (preg_match("/^({$class['subclass']}_(?:.*?_)?)Exception/", $classname, $matches)) {
$exceptions_filename = /*$class['subclass_dir_prefix'] .*/ preg_replace('/_/', '/', $matches[1]) . 'Exceptions.class.php';
if (stream_resolve_include_path($exceptions_filename)) {
require_once($exceptions_filename);
// If that found the class, break here, otherwise look upstream
if (class_exists($classname, false)) {
return;
}
} else {
// Create this class to extend the Framework parent
$parent_classname = preg_replace("/^{$class['subclass']}_/", "{$class['base']}_", $classname);
class_alias($parent_classname, $classname);
return;
}
}
// Replace any underscores with directory separators
$filename = /*$class['subclass_dir_prefix'] .*/ preg_replace('/_/', '/', $classname) . '.class.php';
// If this file exists, load it
if (stream_resolve_include_path($filename)) {
require_once($filename);
return;
} else {
// Create this class to extend the Framework parent
$parent_classname = preg_replace("/^{$class['subclass']}_/", "{$class['base']}_", $classname);
class_alias($parent_classname, $classname);
return;
}
}
}
}
/**
* Adds additional class names to the autoloader
*
* The base name is the prefix of all classes in the base tree. The subclass name is the prefix of all classes from
* another tree in which the base classes can be extended. For example, the SihnonFramework and Sihnon class trees.
*
* For the base and subclasses, the dir prefix is the top-level directory which contains all the class files.
*
* If there are no subclasses, the latter two parameters can be left as null, or unspecified.
*
* @param unknown_type $base
* @param unknown_type $base_dir_prefix
* @param unknown_type $subclass
* @param unknown_type $subclass_dir_prefix
*/
public static function registerAutoloadClasses($base, $base_dir_prefix, $subclass = null, $subclass_dir_prefix = null) {
$canonical_base_dir_prefix = static::makeAbsolutePath($base_dir_prefix);
if ( ! $canonical_base_dir_prefix) {
throw new SihnonFramework_Exception_FileNotFound($base_dir_prefix);
}
// The paths must end with a trailing slash
if ($canonical_base_dir_prefix && $canonical_base_dir_prefix[strlen($canonical_base_dir_prefix) - 1] != DIRECTORY_SEPARATOR) {
$canonical_base_dir_prefix .= DIRECTORY_SEPARATOR;
}
$include_path_prefix = $canonical_base_dir_prefix . PATH_SEPARATOR;
$canonical_subclass_dir_prefix = null;
if ($subclass_dir_prefix) {
$canonical_subclass_dir_prefix = static::makeAbsolutePath($subclass_dir_prefix);
if ( ! $canonical_subclass_dir_prefix) {
throw new SihnonFramework_Exception_FileNotFound($subclass_dir_prefix);
}
if ($canonical_subclass_dir_prefix[strlen($canonical_subclass_dir_prefix) - 1] != DIRECTORY_SEPARATOR) {
$canonical_subclass_dir_prefix .= DIRECTORY_SEPARATOR;
}
$include_path_prefix = $canonical_subclass_dir_prefix . PATH_SEPARATOR . $include_path_prefix;
}
set_include_path($include_path_prefix . get_include_path());
array_unshift(self::$autoload_classes, array(
'base' => $base,
'base_dir_prefix' => $canonical_base_dir_prefix,
'subclass' => $subclass,
'subclass_dir_prefix' => $canonical_subclass_dir_prefix,
));
}
/**
* Throws an exception if the requested name has not been defined.
*
* @param string $name Name of the definition to check for the existence of
* @throws Sihnon_Exception_MissingDefinition
*/
public static function ensureDefined($name) {
if (! defined($name)) {
throw new Sihnon_Exception_MissingDefinition($name);
}
}
/**
* Returns the canonical form of the given path, made absolute if relative
*
* @param string $relative_path
* @return string
*/
public static function makeAbsolutePath($relative_path) {
if (preg_match('#^/#', $relative_path)) {
// This path is already absolute, just canonicalise it
return realpath($relative_path);
}
// Use the base path in order of precedence
$prefix = '';
if (isset($_SERVER['SihnonFramework_Base'])) {
$prefix = $_SERVER['SihnonFramework_Base'];
} elseif (defined('SihnonFramework_Base')) {
$prefix = SihnonFramework_Base;
} else {
$prefix = getcwd() . DIRECTORY_SEPARATOR;
}
$absolute_path = $prefix . $relative_path;
return realpath($absolute_path);
}
public static function mkdir_recursive($directory, $permissions=0777) {
$parts = explode('/', $directory);
$path = '';
for ($i=1,$l=count($parts); $i<=$l; $i++) {
$iPath = $parts;
$path = join('/', array_slice($iPath, 0, $i));
if (empty($path)) continue;
if (!file_exists($path)) {
if (!mkdir($path)) return false;
if (!chmod($path, $permissions)) return false;
}
}
return true;
}
public static function rmdir_recursive($dir) {
if (is_dir($dir)) {
$objects = scandir($dir);
foreach ($objects as $object) {
if ($object != "." && $object != "..") {
if (filetype($dir."/".$object) == "dir") self::rmdir_recursive($dir."/".$object); else unlink($dir."/".$object);
}
}
reset($objects);
rmdir($dir);
}
return true;
}
public static function recursiveFilesize($file) {
$size = 0;
if (is_dir($file)) {
$objects = scandir($file);
foreach ($objects as $object) {
if ($object != "." && $object != "..") {
$size += static::recursiveFilesize($file . '/' . $object);
}
}
} else {
$size += filesize($file);
}
return $size;
}
public static function issetelse($var, $default = null) {
if (isset($var)) {
return $var;
}
if (is_string($default) && preg_match('/^Sihnon(Framework)?_Exception/', $default) && class_exists($default) && is_subclass_of($default, 'SihnonFramework_Exception')) {
throw new $default();
}
return $default;
}
public static function formatDuration($seconds, $fuzziness = 0) {
if (is_null($seconds)) {
return 'indeterminate time';
}
$labels = array('second', 'minute', 'hour', 'day', 'week', 'month', 'year');
$pluralLabels = array('seconds', 'minutes', 'hours', 'days', 'weeks', 'months', 'years');
$limits = array(1, 60, 3600, 86400, 604800, 2592000, 31556926, PHP_INT_MAX);
$components = array(0, 0, 0, 0, 0, 0, 0);
$workingTime = $seconds;
$result = "";
$ptr = count($labels) - 1;
while ($ptr >= 0 && $workingTime < $limits[$ptr]) {
--$ptr;
}
$mostSignificantPtr = $ptr;
// Convert the value into components using the remaining labels
while ($ptr >= 0) {
$unitTime = floor($workingTime / $limits[$ptr]);
$workingTime -= $unitTime * $limits[$ptr];
$components[$ptr] = $unitTime;
--$ptr;
}
$componentsUsed = 0;
$approximate = false;
$lastComponent = false;
$ptr = $mostSignificantPtr;
while ($ptr >= 0) {
if ($fuzziness && $componentsUsed >= $fuzziness) {
break;
} elseif ($ptr == 0 || ($fuzziness && $componentsUsed == $fuzziness-1)) {
$lastComponent = true;
}
$component = $components[$ptr];
if ($component) {
// If we're going to hide the next value, take its component into account here
if ($lastComponent && $ptr > 0) {
$component += round($components[$ptr-1] / $limits[$ptr]);
$approximate = true;
}
if ($lastComponent && $ptr < $mostSignificantPtr) {
$result .= ' and';
}
$result .= ' ' . $component . ' ' . ($component == 1 ? $labels[$ptr] : $pluralLabels[$ptr]);
}
// Increment even if we've hidden this component because it's zero
// Then we don't end up with overly precise times like '2 years and 1 second'
++$componentsUsed;
--$ptr;
}
if ($approximate) {
$result = 'approximately ' . $result;
}
return $result;
}
public static function formatFilesize($bytes) {
if (is_null($bytes)) {
return 'unknown';
}
$labels = array('B', 'KB', 'MB', 'GB', 'TB');
$limits = array(1, 1024, 1024*1024, 1024*1024*1024, 1024*1024*1024*1024);
$size = $bytes;
$ptr = count($labels) - 1;
while ($ptr > 0 && $bytes < $limits[$ptr]) {
--$ptr;
}
$size = round($bytes / $limits[$ptr], 2) . ' ' . $labels[$ptr];
return $size;
}
public static function isClassConstantValue($class, $prefix, $value) {
try {
$reflected = new ReflectionClass($class);
} catch (ReflectionException $e) {
throw new Sihnon_Exception_ClassNotFound($class);
}
$constants = $reflected->getConstants();
foreach ($constants as $const_name => $const_value) {
if (preg_match("/{$prefix}/", $const_name)) {
if ($value == $const_value) {
return true;
}
}
}
return false;
}
}
SihnonFramework_Main::initialise();
?>