CLog.php 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332
  1. <?php
  2. namespace loyalsoft;
  3. /**
  4. * 错误级别
  5. */
  6. class enum_LogLevel extends Enum {
  7. const All = 0;
  8. const Info = 1;
  9. const Warn = 2;
  10. const Err = 4;
  11. }
  12. /**
  13. * 日志工具类_高性能(High Performance)
  14. * @author gwang (mail@wanggangzero.cn)
  15. * @version <br/>
  16. * 3.0.2 2018年3月9日 完善稳定性, GetDir()方法添加了自动检查目录是否存在的逻辑. -gwang <br/>
  17. * 3.0.1 2018年1月9日 整理, online情况下paylog->mysql, errlog->redis, log和info信息仍然在文件(online默认关闭log和info的落地操作). -gwang <br/>
  18. * 2.0.1 2017年8月8日 将paylog和giftlog和output也保留了.以前向兼容就代码. -gwang <br/>
  19. * 2.0 2017年8月8日 整理, 将redis部分暂时去掉, 保留缓存模式, 在处理的最后集中flush到文件中. -gwang <br/>
  20. * 1.0 2016年7月1日 创建, 思路: 建立高性能的日志模块,消除写文件.暂时不包含支付日志. -gwang <br/>
  21. */
  22. class CLog {
  23. /**
  24. * 日志缓存
  25. * @var array
  26. */
  27. private static $Warn = array();
  28. private static $Info = array();
  29. private static $Err = array();
  30. private static $logLevel = enum_LogLevel::Err;
  31. /**
  32. * @var int 日志写入redis时保存的记录条数
  33. */
  34. const redis_log_trim_max = 5000;
  35. /**
  36. * @var int 日志写入redis时每次trim的记录条数
  37. */
  38. const redis_log_trim_once = 100;
  39. /**
  40. * 设置日志级别
  41. * @param int $level 参考常量enum_LogLevel :: 0 => info, 1 => log, 2 => err
  42. */
  43. public static function SetLogLevel($level) {
  44. if ($level >= enum_LogLevel::All && $level <= enum_LogLevel::Err) {
  45. self::$logLevel = $level;
  46. } else {
  47. // 无效值,忽视
  48. }
  49. }
  50. /**
  51. * 断言日志,注意此断言不是打断程序执行的那种.
  52. * 如果断言失败,则会在日志里面留下一条Err记录,记录内容为errmsg.
  53. * 如果判断成功,且设置了okmsg, 则会在日志里面留下一条info记录记录内容为OKmsg.
  54. * @param bool $condition 条件/条件表达式
  55. * @param string $errmsg 条件不成立时输出到日志中的信息, 日志级别err
  56. * @param string $okmsg 【可选】默认(null), 当条件成立时不记录任何消息, 除非特别指定了消息内容.日志级别info
  57. */
  58. public static function Assert($tag, $condition, $errmsg, $okmsg = null) {
  59. if (!$condition) {
  60. self::err($errmsg, $tag);
  61. } else if ($okmsg) {
  62. self::info($okmsg, $tag);
  63. }
  64. }
  65. /**
  66. * 记录日志
  67. * @param string $msg
  68. * @param string $tag 标签
  69. */
  70. public static function warn($msg, $tag = 'log') {
  71. if (self::$logLevel <= enum_LogLevel::Warn) {
  72. self::$Warn[] = self::makeLogMsg($msg, $tag);
  73. }
  74. }
  75. /**
  76. * @param string $msg
  77. * @param string $tag 标签
  78. */
  79. public static function info($msg, $tag = 'info') {
  80. if (self::$logLevel <= enum_LogLevel::Info) {
  81. self::$Info[] = self::makeLogMsg($msg, $tag);
  82. }
  83. }
  84. /**
  85. * 记录异常日志
  86. * @param string $msg
  87. * @param string $tag 标签
  88. */
  89. public static function err($msg, $tag = "err") {
  90. if (self::$logLevel <= enum_LogLevel::Err) {
  91. self::$Err[] = self::makeLogMsg($msg, $tag);
  92. }
  93. }
  94. /**
  95. * 将缓存中的数据写入存储设备并清除缓存
  96. */
  97. public static function flush() {
  98. self::flush2Redis(enum_LogLevel::Info);
  99. self::flush2Redis(enum_LogLevel::Warn);
  100. self::flush2Redis(enum_LogLevel::Err);
  101. }
  102. // <editor-fold defaultstate="collapsed" desc="辅助方法">
  103. /**
  104. * 生成日志,格式: [x区][时间][文件][tag]: msg
  105. * @global \loyalsoft\type $zoneid
  106. * @param type $msg
  107. * @param type $tag
  108. * @return string
  109. */
  110. private static function makeLogMsg($msg, $tag = null) {
  111. global $zoneid;
  112. $server_addr = isset($_SERVER['SERVER_ADDR']) ? $_SERVER['SERVER_ADDR'] : "-";
  113. $client_addr = isset($_SERVER['REMOTE_ADDR']) ? $_SERVER['REMOTE_ADDR'] : "-";
  114. $uid = empty(req()->uid) ? "-" : req()->uid;
  115. $str = "[" . date('Y-m-d H:i:s') . "][" . PHP_VERSION . "][" . $server_addr . "]<" . $client_addr . ">" # 时间,PHP_ver,主机地址, 客户端来源ip地址
  116. . "[$zoneid 区][$uid]" . PHP_EOL # # 玩家所在分区, uid
  117. . DebugHelper::get_call_stack(3, 8) # # 代码调用堆栈
  118. . (isset($tag) ? "[" . CommUtil::str2UTF8($tag) . "]" : "") # 标签
  119. . "=> " . var_export($msg, true)# # 内容
  120. . PHP_EOL;
  121. return $str;
  122. }
  123. /**
  124. * @return string 日志目录
  125. */
  126. private static function GetDir($subfolder = null) {
  127. # 线上版的时候会是linux系统, 单独指定目录, 测试的时候就放到代码旁边得了
  128. $dir = "/data/logs/" . PROJECTNAME . "/";
  129. if (!empty($subfolder)) { # 如果有子目录, 附加
  130. $dir .= $subfolder . "/";
  131. }
  132. if (is_dir($dir) || mkdir($dir, 0777, true)) { # 如果目录不存在, 创建
  133. return $dir;
  134. }
  135. exit("can not access log directory!");
  136. }
  137. /**
  138. * 将日志写入文件
  139. * @global type $zoneid
  140. * @param enum_LogLevel $type
  141. */
  142. private static function flush2File($type) {
  143. $typename = strval(new enum_LogLevel($type));
  144. $fileName = self::GetDir() . $typename . "-" . date('Ymd') . ".log"; # 日志文件名(按天分割)
  145. $fd = fopen($fileName, "a");
  146. if (false === $fd) { # 打开文件失败
  147. throw new \Exception("打开 $fileName 失败");
  148. }
  149. $arr = self::$$typename; # 取对应类型的数组
  150. if (count($arr) > 0) {
  151. foreach ($arr as $msg) {
  152. $n = fputs($fd, $msg . PHP_EOL);
  153. if (false === $n) { # 写入时失败
  154. throw new \Exception("写入 $fileName 时失败"); # 这种的就得靠查找系统(nginx/php-fpm)日志了
  155. }
  156. }
  157. self::$$typename = array(); # 清空数据
  158. }
  159. fclose($fd); # 关闭文件
  160. }
  161. /**
  162. * 将缓存中的数据写入存储设备(redis)并清除缓存
  163. * @type int/enum_LogLevel 日志类型
  164. */
  165. private static function flush2Redis($type) {
  166. $typename = strval(new enum_LogLevel($type));
  167. $key = "log-" . $typename;
  168. $arr = self::$$typename; # 取对应类型的数组
  169. if (count($arr) > 0) {
  170. if (null != req()) {
  171. array_unshift($arr, date('Y-m-d H:i:s') . " 请求: " . req()); # 如果请求不为空, 附加下本次请求参数.
  172. }
  173. gMem()->lpush($key, $arr);
  174. if (gMem()->llen($key) > self::redis_log_trim_max + self::redis_log_trim_once) {# 达到清理条件
  175. gMem()->ltrim($key, 0, -self::redis_log_trim_once); # 缩减记录
  176. }
  177. self::$$typename = array(); # 清空数据
  178. }
  179. }
  180. /**
  181. * 将缓存中的数据写入存储设备(mysql)并清除缓存
  182. * @type int/enum_LogLevel 日志类型
  183. */
  184. private static function flush2MySQL($type) {
  185. $typename = strval(new enum_LogLevel($type));
  186. $key = "log-" . $typename;
  187. $arr = self::$$typename; # 取对应类型的数组
  188. if (count($arr) > 0) {
  189. $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;";
  190. daoInst()->exec($sql); # 创建表
  191. $data = array();
  192. array_map(function ($v) use (&$data) {
  193. $s = daoInst()->quote($v);
  194. $data[] = "(" . $s . ")";
  195. }, $arr);
  196. $sql_insert = "insert into `$key` (`msg`) values " . implode(',', $data) . " ;";
  197. daoInst()->exec($sql_insert); # 执行插入操作
  198. $n = daoInst()->select()->from("`$key`")->count(); # 查一下当前表大小
  199. $max = 150000; # 最多保留15万条记录
  200. $del_once = 10000; # 超过15万,清理一次,一次删除1万条
  201. if ($n > $max) {
  202. $sql = "delete from `$key` order by `row` limit $del_once ;"; # 一次清理一批
  203. daoInst()->exec($sql);
  204. }
  205. self::$$typename = array(); # 清空数据
  206. }
  207. }
  208. // </editor-fold>
  209. //
  210. // <editor-fold defaultstate="collapsed" desc="优化后的文件日志(引入cache)">
  211. /**
  212. * 写入文件日志
  213. * @global \loyalsoft\type $zoneid
  214. * @param type $type
  215. * @param type $msg
  216. * @return type
  217. */
  218. private static function fileLog($type, $msg) {
  219. $fileName = self::GetDir($type) . PLAT . '-' . date('Ymd') . ".log"; # 日志文件名(按天分割)
  220. $fd = fopen($fileName, "a");
  221. if (is_array($msg) || is_object($msg)) {
  222. $msg = JsonUtil::encode($msg);
  223. }
  224. global $zoneid;
  225. fputs($fd, "[" . self::dtCurrent() . "] $zoneid 区, " . $msg . "\r\n");
  226. fclose($fd);
  227. }
  228. private static $cache = array();
  229. private static function output2($type, $msg) {
  230. if (!isset(self::$cache[$type])) {
  231. self::$cache[$type] = array();
  232. }
  233. if (is_array($msg) || is_object($msg)) {
  234. $msg = JsonUtil::encode($msg);
  235. }
  236. global $zoneid;
  237. self::$cache[$type][] = "[" . self::dtCurrent() . "] $zoneid 区, " . $msg . "\r\n";
  238. }
  239. private static function flushFileCache() {
  240. if (is_array(self::$cache)) {
  241. foreach (self::$cache as $t => $varr) {
  242. $fileName = self::GetDir($t) . PLAT . '-' . date('Ymd') . ".log"; # 日志文件名(按天分割)
  243. $fd = fopen($fileName, "a");
  244. if (is_array($varr)) {
  245. foreach ($varr as $msg) {
  246. fputs($fd, $msg);
  247. }
  248. }
  249. fclose($fd);
  250. }
  251. self::$cache = array();
  252. }
  253. }
  254. /**
  255. * 向文件写入日志
  256. * @global int $zoneid
  257. * @param string $fileName
  258. * @param string $msg
  259. */
  260. private static function Log2File($fileName, $msg) {
  261. $fd = fopen($fileName, "a");
  262. if ($fd === FALSE) { # 打开文件失败
  263. throw new \Exception('open file failed! ' . $fileName);
  264. }
  265. fputs($fd, $msg . PHP_EOL);
  266. fclose($fd); # 关闭文件
  267. }
  268. // </editor-fold>
  269. //
  270. // 避免异常导致缓存丢失, 付费模块的日志直接写入文件
  271. // <editor-fold defaultstate="collapsed" desc=" 付费相关日志, 无缓存 ">
  272. /**
  273. * 输出日志 到pay.log
  274. * @param mixed $msg 参数是字符串则直接输出,否则调用json_encode
  275. */
  276. public static function pay($msg) {
  277. $str = self::makeLogMsg($msg); # 统一日志格式字符串
  278. self::Paylog2Mysql($str);
  279. }
  280. /**
  281. * 将pay日志写入mysql
  282. * @param type $msg
  283. */
  284. private static function Paylog2Mysql($msg) {
  285. $t = "tab_paylog_" . date('Ym');
  286. $sql = "create table if not exists $t like tpl_paylog_tab_ym;" # 组装SQL
  287. . " insert into $t (msg) values ("
  288. . daoInst()->quote($msg) . ");"; # 消息体
  289. $n = daoInst()->exec($sql);
  290. if ($n < 0) {
  291. self::err("写入数据库失败." . $sql);
  292. }
  293. }
  294. // </editor-fold>
  295. //
  296. }
  297. if (defined('DEBUGING') && DEBUGING) { # 除了现网
  298. CLog::SetLogLevel(enum_LogLevel::All); # 开启所有级别的日志输出
  299. }