123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304 |
- <?php
- namespace loyalsoft;
- /**
- * 错误级别
- */
- class enum_LogLevel extends Enum {
- const All = 0;
- const Info = 1;
- const Warn = 2;
- const Err = 4;
- }
- /**
- * 日志工具类_高性能(High Performance)
- * @author gwang (mail@wanggangzero.cn)
- * @version <br/>
- * 3.0.2 2018年3月9日 完善稳定性, GetDir()方法添加了自动检查目录是否存在的逻辑. -gwang <br/>
- * 3.0.1 2018年1月9日 整理, online情况下paylog->mysql, errlog->redis, log和info信息仍然在文件(online默认关闭log和info的落地操作). -gwang <br/>
- * 2.0.1 2017年8月8日 将paylog和giftlog和output也保留了.以前向兼容就代码. -gwang <br/>
- * 2.0 2017年8月8日 整理, 将redis部分暂时去掉, 保留缓存模式, 在处理的最后集中flush到文件中. -gwang <br/>
- * 1.0 2016年7月1日 创建, 思路: 建立高性能的日志模块,消除写文件.暂时不包含支付日志. -gwang <br/>
- */
- class CLog {
- /**
- * 日志缓存
- * @var array
- */
- private static $Warn = array();
- private static $Info = array();
- private static $Err = array();
- private static $logLevel = enum_LogLevel::Err;
- /**
- * @var int 日志写入redis时保存的记录条数
- */
- const redis_log_trim_max = 5000;
- /**
- * @var int 日志写入redis时每次trim的记录条数
- */
- const redis_log_trim_once = 100;
- /**
- * @return string 日志目录
- */
- private static function GetDir() {
- // 线上版的时候会是linux系统, 单独指定目录, 测试的时候就放到代码旁边得了
- static $dir = null;
- if ($dir) {
- return $dir;
- }
- $dir = (GAME_ONLINE ? "/data/" : ROOTDIR . "/../" ) . "logs/" . PROJECTNAME . "/";
- if (is_dir($dir) || mkdir($dir, 0777, true)) {
- return $dir;
- }
- exit("can not access log directory!");
- }
- /**
- * 设置日志级别
- * @param int $level 参考常量enum_LogLevel :: 0 => info, 1 => log, 2 => err
- */
- public static function SetLogLevel($level) {
- if ($level >= enum_LogLevel::All && $level <= enum_LogLevel::Err) {
- self::$logLevel = $level;
- } else {
- // 无效值,忽视
- }
- }
- /**
- * 断言,
- * @param bool $condition 条件/条件表达式
- * @param string $errmsg 条件不成立时输出到日志中的信息, 日志级别err
- * @param string $okmsg 【可选】默认(null), 当条件成立时不记录任何消息, 除非特别指定了消息内容.日志级别info
- */
- public static function Assert($tag, $condition, $errmsg, $okmsg = null) {
- if (!$condition) {
- self::err($errmsg, $tag);
- } else if ($okmsg) {
- self::info($okmsg, $tag);
- }
- }
- /**
- * 记录日志
- * @param string $msg
- * @param string $tag 标签
- */
- public static function warn($msg, $tag = 'log') {
- if (self::$logLevel <= enum_LogLevel::Warn) {
- self::$Warn[] = self::makeLogMsg($msg, $tag);
- }
- }
- /**
- * @param string $msg
- * @param string $tag 标签
- */
- public static function info($msg, $tag = 'info') {
- if (self::$logLevel <= enum_LogLevel::Info) {
- self::$Info[] = self::makeLogMsg($msg, $tag);
- }
- }
- /**
- * 记录异常日志
- * @param string $msg
- * @param string $tag 标签
- */
- public static function err($msg, $tag = "err") {
- if (self::$logLevel <= enum_LogLevel::Err) {
- self::$Err[] = self::makeLogMsg($msg, $tag);
- }
- }
- /**
- * 将缓存中的数据写入存储设备并清除缓存
- */
- public static function flush() {
- if (GAME_ONLINE) { # 外网错误日志写入redis
- self::flush2Redis(enum_LogLevel::Info);
- self::flush2Redis(enum_LogLevel::Warn);
- self::flush2Redis(enum_LogLevel::Err);
- } else {
- // self::flush2MySQL(enum_LogLevel::Info); # 内网写入MySQL
- // self::flush2MySQL(enum_LogLevel::Warn);
- // self::flush2MySQL(enum_LogLevel::Err);
- self::flush2Redis(enum_LogLevel::Info);
- self::flush2Redis(enum_LogLevel::Warn);
- self::flush2Redis(enum_LogLevel::Err);
- }
- }
- /**
- * 将日志写入文件
- * @global type $zoneid
- * @param enum_LogLevel $type
- */
- private static function flush2File($type) {
- $typename = strval(new enum_LogLevel($type));
- $fileName = self::GetDir() . $typename . "-" . date('Ymd') . ".log"; # 日志文件名(按天分割)
- $fd = fopen($fileName, "a");
- if (false === $fd) { # 打开文件失败
- throw new \Exception("打开 $fileName 失败");
- }
- $arr = self::$$typename; # 取对应类型的数组
- if (count($arr) > 0) {
- foreach ($arr as $msg) {
- $n = fputs($fd, $msg . PHP_EOL);
- if (false === $n) { # 写入时失败
- throw new \Exception("写入 $fileName 时失败");
- }
- // todo: 其实写入日志也是有失败几率的, 当写入日志也失败的时候,
- // 抛出异常, 让nginx来记录下些东西, 但是如果是在errorhandler
- // 中写日志的时候出错呢......
- // -wg 2017年8月5日 08:56:29
- }
- self::$$typename = array(); # 清空数据
- }
- fclose($fd); # 关闭文件
- }
- /**
- * 将缓存中的数据写入存储设备(redis)并清除缓存
- * @type int/enum_LogLevel 日志类型
- */
- private static function flush2Redis($type) {
- $typename = strval(new enum_LogLevel($type));
- $key = "log-" . $typename;
- $arr = self::$$typename; # 取对应类型的数组
- if (count($arr) > 0) {
- gMem()->lpush($key, $arr);
- if (gMem()->llen($key) > self::redis_log_trim_max + self::redis_log_trim_once) {# 达到清理条件
- gMem()->ltrim($key, 0, -self::redis_log_trim_once); # 缩减记录
- }
- self::$$typename = array(); # 清空数据
- }
- }
- /**
- * 将缓存中的数据写入存储设备(mysql)并清除缓存
- * @type int/enum_LogLevel 日志类型
- */
- private static function flush2MySQL($type) {
- $typename = strval(new enum_LogLevel($type));
- $key = "log-" . $typename;
- $arr = self::$$typename; # 取对应类型的数组
- if (count($arr) > 0) {
- // 创建表
- $sql = "create table if not exists `$key` (`row` INT(11) NOT NULL AUTO_INCREMENT COMMENT '自增行号', `msg` TEXT NULL COMMENT '错误消息内容', PRIMARY KEY (`row`)) COLLATE='utf8_general_ci' ENGINE=InnoDB;";
- daoInst()->exec($sql);
- // 插入数据
- $data = array();
- array_map(function($v) use(&$data) {
- $s = daoInst()->quote($v);
- var_dump($s);
- $data[] = "(" . $s . ")";
- }, $arr);
- $sql_insert = "insert into `$key` (`msg`) values " . implode(',', $data) . " ;";
- daoInst()->exec($sql_insert);
- // 检查表大小
- $n = daoInst()->select()->from("`$key`")->count(); # 查一下当前表大小
- $max = 150000; # 最多保留15万条记录
- $del_once = 10000; # 超过15万,清理一次,一次删除1万条
- if ($n > $max) {
- $sql = "delete from `$key` order by `row` limit $del_once ;"; # 一次清理一批
- daoInst()->exec($sql);
- }
- // self::$$typename = array(); # 清空数据
- }
- }
- //
- // 避免异常导致缓存丢失, 付费模块的日志直接写入文件
- // <editor-fold defaultstate="collapsed" desc=" 付费相关日志, 无缓存 ">
- /**
- * 输出日志 到pay.log
- * @param mixed $msg 参数是字符串则直接输出,否则调用json_encode
- */
- public static function pay($msg) {
- if (GAME_ONLINE) {
- self::Paylog2Mysql($msg);
- } else {
- $fileName = self::GetDir() . "pay.log";
- self::Log2File($fileName, $msg);
- }
- }
- /**
- * 向文件写入日志
- * @global int $zoneid
- * @param string $fileName
- * @param string $msg
- */
- private static function Log2File($fileName, $msg) {
- $fd = fopen($fileName, "a");
- if ($fd === FALSE) { # 打开文件失败
- throw new \Exception('open file failed! ' . $fileName);
- }
- // todo: 文件写入也是有机会出错的, 此处出错应予监控,
- // 并抛出异常到nginx/phpfpm层,记录日志.
- // -wg 2017年8月5日 09:10:16
- $str = self::makeLogMsg($msg);
- fputs($fd, $str . PHP_EOL);
- fclose($fd); # 关闭文件
- }
- /**
- * 生成日志,格式: [x区][时间][文件][tag]: msg
- * @global \loyalsoft\type $zoneid
- * @param type $msg
- * @param type $tag
- * @return string
- */
- private static function makeLogMsg($msg, $tag = null) {
- global $zoneid;
- $array = debug_backtrace();
- while (isset($array[0]) && isset($array[0]['file']) && $array[0]['file'] == __FILE__) {
- array_shift($array);
- }
- $row = $array[0];
- $server_addr = isset($_SERVER['SERVER_ADDR']) ? $_SERVER['SERVER_ADDR'] : "-";
- $client_addr = isset($_SERVER['REMOTE_ADDR']) ? $_SERVER['REMOTE_ADDR'] : "-";
- return "[" . $server_addr . "][$zoneid 区][" . date('Y-m-d H:i:s') # # 主机地址、分区、时间
- . "][" . $client_addr . "][" # # 客户端地址
- . (isset($row['file']) ? CommUtil::str2UTF8(basename($row['file'])) : '-') # 所在文件名
- . ":" . (isset($row['line']) ? $row['line'] : "???")# # 所在行
- . "]" . (isset($tag) ? "[$tag]" : "") . "=> " . var_export($msg, true)# # 标签、内容
- . PHP_EOL;
- // . "req: " . JsonUtil::encode($req);
- }
- /**
- * 将pay日志写入mysql
- * @param type $msg
- */
- public static function Paylog2Mysql($msg) {
- $str = self::makeLogMsg($msg); # 统一日志格式字符串
- $t = "tab_paylog_" . date('Ym');
- $sql = "create table if not exists $t like tpl_paylog_tab_ym;"
- . " insert into $t (msg) values ('$str');";
- $n = daoInst()->exec($sql);
- if ($n < 0) {
- self::err("写入数据库失败." . $sql);
- }
- }
- // </editor-fold>
- //
- }
- if (!GAME_ONLINE) { # 除了现网
- CLog::SetLogLevel(enum_LogLevel::All); # 开启所有级别的日志输出
- }
|