博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
php 错误 异常,一个显示效果非常不错的PHP错误、异常处理类
阅读量:5150 次
发布时间:2019-06-13

本文共 6740 字,大约阅读时间需要 22 分钟。

// 自定义异常函数

set_exception_handler('handle_exception');

// 自定义错误函数

set_error_handler('handle_error');

/**

* 异常处理

*

* @param mixed $exception 异常对象

* @author blog.snsgou.com

*/

function handle_exception($exception) {

Error::exceptionError($exception);

}

/**

* 错误处理

*

* @param string $errNo 错误代码

* @param string $errStr 错误信息

* @param string $errFile 出错文件

* @param string $errLine 出错行

* @author blog.snsgou.com

*/

function handle_error($errNo, $errStr, $errFile, $errLine) {

if ($errNo) {

Error::systemError($errStr, false, true, false);

}

}

/**

* 系统错误处理

*

* @author blog.snsgou.com

*/

class Error {

public static function systemError($message, $show = true, $save = true, $halt = true) {

list($showTrace, $logTrace) = self::debugBacktrace();

if ($save) {

$messageSave = '' . $message . 'PHP:' . $logTrace;

self::writeErrorLog($messageSave);

}

if ($show) {

self::showError('system', "

$message", $showTrace, 0);

}

if ($halt) {

exit();

} else {

return $message;

}

}

/**

* 代码执行过程回溯信息

*

* @static

* @access public

*/

public static function debugBacktrace() {

$skipFunc[] = 'Error->debugBacktrace';

$show = $log = '';

$debugBacktrace = debug_backtrace();

ksort($debugBacktrace);

foreach ($debugBacktrace as $k => $error) {

if (!isset($error['file'])) {

// 利用反射API来获取方法/函数所在的文件和行数

try {

if (isset($error['class'])) {

$reflection = new ReflectionMethod($error['class'], $error['function']);

} else {

$reflection = new ReflectionFunction($error['function']);

}

$error['file'] = $reflection->getFileName();

$error['line'] = $reflection->getStartLine();

} catch (Exception $e) {

continue;

}

}

$file = str_replace(SITE_PATH, '', $error['file']);

$func = isset($error['class']) ? $error['class'] : '';

$func .= isset($error['type']) ? $error['type'] : '';

$func .= isset($error['function']) ? $error['function'] : '';

if (in_array($func, $skipFunc)) {

break;

}

$error['line'] = sprintf('%04d', $error['line']);

$show .= '

[Line: ' . $error['line'] . ']' . $file . '(' . $func . ')';

$log .= !empty($log) ? ' -> ' : '';

$log .= $file . ':' . $error['line'];

}

return array($show, $log);

}

/**

* 异常处理

*

* @static

* @access public

* @param mixed $exception

*/

public static function exceptionError($exception) {

if ($exception instanceof DbException) {

$type = 'db';

} else {

$type = 'system';

}

if ($type == 'db') {

$errorMsg = '(' . $exception->getCode() . ') ';

$errorMsg .= self::sqlClear($exception->getMessage(), $exception->getDbConfig());

if ($exception->getSql()) {

$errorMsg .= '

';

$errorMsg .= self::sqlClear($exception->getSql(), $exception->getDbConfig());

$errorMsg .= '

';

}

} else {

$errorMsg = $exception->getMessage();

}

$trace = $exception->getTrace();

krsort($trace);

$trace[] = array('file' => $exception->getFile(), 'line' => $exception->getLine(), 'function' => 'break');

$phpMsg = array();

foreach ($trace as $error) {

if (!empty($error['function'])) {

$fun = '';

if (!empty($error['class'])) {

$fun .= $error['class'] . $error['type'];

}

$fun .= $error['function'] . '(';

if (!empty($error['args'])) {

$mark = '';

foreach ($error['args'] as $arg) {

$fun .= $mark;

if (is_array($arg)) {

$fun .= 'Array';

} elseif (is_bool($arg)) {

$fun .= $arg ? 'true' : 'false';

} elseif (is_int($arg)) {

$fun .= (defined('SITE_DEBUG') && SITE_DEBUG) ? $arg : '%d';

} elseif (is_float($arg)) {

$fun .= (defined('SITE_DEBUG') && SITE_DEBUG) ? $arg : '%f';

} else {

$fun .= (defined('SITE_DEBUG') && SITE_DEBUG) ? '\'' . htmlspecialchars(substr(self::clear($arg), 0, 10)) . (strlen($arg) > 10 ? ' ...' : '') . '\'' : '%s';

}

$mark = ', ';

}

}

$fun .= ')';

$error['function'] = $fun;

}

if (!isset($error['line'])) {

continue;

}

$phpMsg[] = array('file' => str_replace(array(SITE_PATH, '\\'), array('', '/'), $error['file']), 'line' => $error['line'], 'function' => $error['function']);

}

self::showError($type, $errorMsg, $phpMsg);

exit();

}

/**

* 记录错误日志

*

* @static

* @access public

* @param string $message

*/

public static function writeErrorLog($message) {

return false; // 暂时不写入

$message = self::clear($message);

$time = time();

$file = LOG_PATH . '/' . date('Y.m.d') . '_errorlog.php';

$hash = md5($message);

$userId = 0;

$ip = get_client_ip();

$user = 'User: userId=' . intval($userId) . '; IP=' . $ip . '; RIP:' . $_SERVER['REMOTE_ADDR'];

$uri = 'Request: ' . htmlspecialchars(self::clear($_SERVER['REQUEST_URI']));

$message = "<?php exit;?>\t{$time}\t$message\t$hash\t$user $uri\n";

// 判断该$message是否在时间间隔$maxtime内已记录过,有,则不用再记录了

if (is_file($file)) {

$fp = @fopen($file, 'rb');

$lastlen = 50000;  // 读取最后的 $lastlen 长度字节内容

$maxtime = 60 * 10;  // 时间间隔:10分钟

$offset = filesize($file) - $lastlen;

if ($offset > 0) {

fseek($fp, $offset);

}

if ($data = fread($fp, $lastlen)) {

$array = explode("\n", $data);

if (is_array($array))

foreach ($array as $key => $val) {

$row = explode("\t", $val);

if ($row[0] != '<?php exit;?>') {

continue;

}

if ($row[3] == $hash && ($row[1] > $time - $maxtime)) {

return;

}

}

}

}

error_log($message, 3, $file);

}

/**

* 清除文本部分字符

*

* @param string $message

*/

public static function clear($message) {

return str_replace(array("\t", "\r", "\n"), " ", $message);

}

/**

* sql语句字符清理

*

* @static

* @access public

* @param string $message

* @param string $dbConfig

*/

public static function sqlClear($message, $dbConfig) {

$message = self::clear($message);

if (!(defined('SITE_DEBUG') && SITE_DEBUG)) {

$message = str_replace($dbConfig['database'], '***', $message);

//$message = str_replace($dbConfig['prefix'], '***', $message);

$message = str_replace(C('DB_PREFIX'), '***', $message);

}

$message = htmlspecialchars($message);

return $message;

}

/**

* 显示错误

*

* @static

* @access public

* @param string $type 错误类型 db,system

* @param string $errorMsg

* @param string $phpMsg

*/

public static function showError($type, $errorMsg, $phpMsg = '') {

global $_G;

$errorMsg = str_replace(SITE_PATH, '', $errorMsg);

ob_end_clean();

$host = $_SERVER['HTTP_HOST'];

$title = $type == 'db' ? 'Database' : 'System';

echo <<

$host - $title Error

$title Error

$errorMsg

EOT;

if (!empty($phpMsg)) {

echo '

';

echo '

PHP Debug

';

echo '

if (is_array($phpMsg)) {

echo '

No.FileLineCode';

foreach ($phpMsg as $k => $msg) {

$k++;

echo '

';

echo '

' . $k . '';

echo '

' . $msg['file'] . '';

echo '

' . $msg['line'] . '';

echo '

' . $msg['function'] . '';

echo '

';

}

} else {

echo '

  • ' . $phpMsg . '
';

}

echo '

';

}

echo <<

EOT;

exit();

}

}

/**

* DB异常类

*

* @author blog.snsgou.com

*/

class DbException extends Exception {

protected $sql;

protected $dbConfig; // 当前数据库配置信息

public function __construct($message, $code = 0, $sql = '', $dbConfig = array()) {

$this->sql = $sql;

$this->dbConfig = $dbConfig;

parent::__construct($message, $code);

}

public function getSql() {

return $this->sql;

}

public function getDbConfig() {

return $this->dbConfig;

}

}

转载地址:http://kldnv.baihongyu.com/

你可能感兴趣的文章
设计模式《JAVA与模式》之访问者模式
查看>>
Timer-triggered memory-to-memory DMA transfer demonstrator
查看>>
《架构之美》阅读笔记六
查看>>
[BZOJ2667][cqoi2012]模拟工厂
查看>>
java EE设计模式--spring企业级开发最佳实践文摘
查看>>
实验6 数组一
查看>>
课堂作业05
查看>>
HTML基础2
查看>>
冒泡排序的两种实现
查看>>
boa web服务器
查看>>
将博客搬至CSDN
查看>>
AngularJS ng-model在ng-if里面无效
查看>>
今天2019年5月,21点58分
查看>>
JavaScript_几种创建对象(2017-07-04)
查看>>
类的初始化
查看>>
centos 7 install eclipse cdt and use github
查看>>
android自定义键盘光标不显示解决方法
查看>>
第一章 大型网站架构演化
查看>>
java基础<迷你DVD系统>
查看>>
NO.6LINUX基本命令
查看>>