| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207 |
- // Author: NiuJiuRu
- // Email: niujiuru@qq.com
- package baseapp
- import "C"
- import (
- "fmt"
- "io"
- "os"
- "path/filepath"
- "strings"
- "github.com/sirupsen/logrus"
- "gopkg.in/ini.v1"
- "gopkg.in/natefinch/lumberjack.v2"
- )
- // 日志级别
- type level uint32
- const (
- LEVEL_TRACE level = level(logrus.TraceLevel) // trace
- LEVEL_DEBUG = level(logrus.DebugLevel) // debug
- LEVEL_INFO = level(logrus.InfoLevel) // info
- LEVEL_WARN = level(logrus.WarnLevel) // warning
- LEVEL_ERROR = level(logrus.ErrorLevel) // error
- LEVEL_FATAL = level(logrus.FatalLevel) // fatal, 会结束程序运行, 会自动调用os.Exit(1)
- LEVEL_PANIC = level(logrus.PanicLevel) // panic, 会触发程序崩溃, 不处理的话, 程序退出
- )
- func (l *level) fromString(s string) {
- switch strings.ToLower(s) {
- case "trace":
- *l = LEVEL_TRACE
- case "debug":
- *l = LEVEL_DEBUG
- case "warn":
- *l = LEVEL_WARN
- case "error":
- *l = LEVEL_ERROR
- case "fatal":
- *l = LEVEL_FATAL
- case "panic":
- *l = LEVEL_PANIC
- default:
- *l = LEVEL_INFO
- }
- }
- // 输出目标
- type target uint32
- const (
- TARGET_NONE target = iota // 不输出日志
- TARGET_CONSOLE target = 1 << iota // 控制台输出
- TARGET_FILE // 输出到文件
- )
- func (t *target) fromString(s string) {
- switch strings.ToLower(s) {
- case "console":
- *t = TARGET_CONSOLE
- case "file":
- *t = TARGET_FILE
- case "all":
- *t = (TARGET_CONSOLE | TARGET_FILE)
- default:
- *t = TARGET_NONE
- }
- }
- // 输出格式
- type formatter struct{}
- func (f *formatter) Format(entry *logrus.Entry) ([]byte, error) {
- msg := ""
- pid := os.Getpid()
- if entry.Caller != nil {
- file := filepath.Base(entry.Caller.File)
- line := entry.Caller.Line
- msg = fmt.Sprintf("[%s, PID: %d] [%s] (%s:%d): %s\n", entry.Time.Format("2006-01-02 15:04:05"), pid, entry.Level.String(), file, line, entry.Message)
- } else {
- msg = fmt.Sprintf("[%s, PID: %d] [%s]: %s\n", entry.Time.Format("2006-01-02 15:04:05"), pid, entry.Level.String(), entry.Message)
- }
- return []byte(msg), nil
- }
- // 日志配置
- type config struct {
- // 日志输出级别
- Level level
- // 日志输出目标
- Target target
- // 每个日志文件的最大大下, 以"MB"为单位
- MaxSize int `ini:"MaxFileSize"`
- // 保留日志文件的最大天数, 以"天"为单位
- MaxAge int `ini:"MaxReserveDays"`
- // 保留的最大备份文件数量, 以"个"为单位
- MaxBackups int `ini:"MaxBackupFileCnts"`
- // 轮转文件是否压缩(gzip)减少占用的空间
- Compress bool `ini:"IsCompress"`
- }
- var (
- Logger *logrus.Logger
- logFileWriter *lumberjack.Logger
- cfgLog = &config{
- Level: LEVEL_TRACE,
- Target: TARGET_CONSOLE,
- MaxSize: 0,
- MaxAge: 0,
- MaxBackups: 0,
- Compress: true,
- }
- )
- func InitLogger() {
- cfgFile := filepath.Join(CFG_DIR, "config.ini")
- cfgIni, err := ini.Load(cfgFile) // 从配置文件中加载相关配置项覆盖默认值
- if err == nil && cfgIni.HasSection("Log") {
- tmpCfgLog := *cfgLog
- err = cfgIni.Section("Log").MapTo(&tmpCfgLog)
- if err != nil {
- fmt.Printf("Failed to load the \"[Log]\" section data from the \"%s\" file: %v\n", cfgFile, err)
- } else {
- *cfgLog = tmpCfgLog
- }
- if cfgIni.Section("Log").HasKey("Level") {
- cfgLog.Level.fromString(cfgIni.Section("Log").Key("Level").String())
- }
- if cfgIni.Section("Log").HasKey("Target") {
- cfgLog.Target.fromString(cfgIni.Section("Log").Key("Target").String())
- }
- }
- Logger = logrus.New()
- Logger.SetLevel(logrus.Level(cfgLog.Level))
- if cfgLog.Level == LEVEL_TRACE || cfgLog.Level == LEVEL_DEBUG {
- Logger.SetReportCaller(true) // 使得输出日志时能够获取到文件名和行号
- }
- Logger.SetFormatter(new(formatter))
- writers := make([]io.Writer, 0)
- if (cfgLog.Target & TARGET_CONSOLE) != 0 {
- writers = append(writers, os.Stdout)
- }
- if (cfgLog.Target & TARGET_FILE) != 0 {
- logFile := filepath.Join(LOG_DIR, EXEC_FILENAME+".log")
- logFileWriter = &lumberjack.Logger{Filename: logFile, MaxSize: cfgLog.MaxSize, MaxAge: cfgLog.MaxAge,
- MaxBackups: cfgLog.MaxBackups, Compress: cfgLog.Compress}
- writers = append(writers, logFileWriter)
- }
- if len(writers) > 0 {
- multiWriter := io.MultiWriter(writers...)
- Logger.SetOutput(multiWriter)
- } else {
- Logger.SetOutput(io.Discard) // 不输出
- }
- }
- func ExitLogger() {
- if Logger != nil {
- Logger.Out = io.Discard
- }
- if logFileWriter != nil {
- logFileWriter.Close() // 关闭日志文件, 只是将日志提交到内核缓存区, 如果要强制落盘, 避免日志丢失,
- // 需要修改开源库代码, 在关闭函数中调用"l.file.Sync()"方法并等待一会儿, 20251021 by niujiuru.
- logFileWriter = nil
- }
- }
- //export LogFromGo
- func LogFromGo(level *C.char, file *C.char, line C.int, message *C.char) {
- goLevel := C.GoString(level)
- goMessage := C.GoString(message)
- if (cfgLog.Level == LEVEL_TRACE || cfgLog.Level == LEVEL_DEBUG) && line > 0 {
- gofile := filepath.Base(C.GoString(file))
- goline := int(line)
- goMessage = fmt.Sprintf("(from %s:%d): %s", gofile, goline, goMessage)
- }
- switch goLevel {
- case "trace":
- Logger.Tracef("%s", goMessage)
- case "debug":
- Logger.Debugf("%s", goMessage)
- case "info":
- Logger.Infof("%s", goMessage)
- case "warn":
- Logger.Warnf("%s", goMessage)
- case "error":
- Logger.Errorf("%s", goMessage)
- case "fatal":
- Logger.Fatalf("%s", goMessage)
- }
- }
|