mirror of
https://github.com/Syngnat/GoNavi.git
synced 2026-05-12 07:09:40 +08:00
- 前端改用通用 DB API,避免强制走 MySQL 接口导致 PostgreSQL 等连接异常 - 后端统一各数据源 timeout(Ping 超时 + 连接参数注入) - DSN 生成兼容特殊字符密码(Postgres/Oracle/达梦/金仓) - 增加文件日志与错误链输出,连接失败提示日志路径便于排障
198 lines
3.6 KiB
Go
198 lines
3.6 KiB
Go
package logger
|
||
|
||
import (
|
||
"errors"
|
||
"fmt"
|
||
"io"
|
||
"log"
|
||
"os"
|
||
"path/filepath"
|
||
"sort"
|
||
"strings"
|
||
"sync"
|
||
"time"
|
||
)
|
||
|
||
const (
|
||
envLogDir = "GONAVI_LOG_DIR"
|
||
appDirName = "GoNavi"
|
||
|
||
logFileName = "gonavi.log"
|
||
logRotateMaxBytes = 10 * 1024 * 1024 // 10MB
|
||
logRotateMaxBackups = 10
|
||
)
|
||
|
||
var (
|
||
once sync.Once
|
||
logMu sync.Mutex
|
||
logInst *log.Logger
|
||
logFile *os.File
|
||
logPath string
|
||
)
|
||
|
||
func Init() {
|
||
once.Do(func() {
|
||
path, out := initOutput()
|
||
logMu.Lock()
|
||
defer logMu.Unlock()
|
||
logPath = path
|
||
logInst = log.New(out, "", log.Ldate|log.Ltime|log.Lmicroseconds)
|
||
logInst.Printf("[信息] 日志初始化完成,日志文件:%s", logPath)
|
||
})
|
||
}
|
||
|
||
func Path() string {
|
||
Init()
|
||
logMu.Lock()
|
||
defer logMu.Unlock()
|
||
return logPath
|
||
}
|
||
|
||
func Close() {
|
||
Init()
|
||
logMu.Lock()
|
||
defer logMu.Unlock()
|
||
if logInst != nil {
|
||
logInst.SetOutput(os.Stderr)
|
||
}
|
||
if logFile != nil {
|
||
_ = logFile.Close()
|
||
logFile = nil
|
||
}
|
||
}
|
||
|
||
func Infof(format string, args ...any) {
|
||
printf("信息", format, args...)
|
||
}
|
||
|
||
func Warnf(format string, args ...any) {
|
||
printf("警告", format, args...)
|
||
}
|
||
|
||
func Errorf(format string, args ...any) {
|
||
printf("错误", format, args...)
|
||
}
|
||
|
||
func Error(err error, format string, args ...any) {
|
||
msg := fmt.Sprintf(format, args...)
|
||
if err == nil {
|
||
Errorf("%s", msg)
|
||
return
|
||
}
|
||
Errorf("%s;错误链:%s", msg, ErrorChain(err))
|
||
}
|
||
|
||
func ErrorChain(err error) string {
|
||
if err == nil {
|
||
return ""
|
||
}
|
||
|
||
var parts []string
|
||
seen := map[string]struct{}{}
|
||
cur := err
|
||
truncated := false
|
||
for i := 0; cur != nil && i < 20; i++ {
|
||
s := cur.Error()
|
||
if _, ok := seen[s]; !ok {
|
||
seen[s] = struct{}{}
|
||
parts = append(parts, s)
|
||
}
|
||
cur = errors.Unwrap(cur)
|
||
}
|
||
if cur != nil {
|
||
truncated = true
|
||
}
|
||
|
||
if len(parts) == 0 {
|
||
return err.Error()
|
||
}
|
||
if truncated {
|
||
parts = append(parts, "(错误链过长,已截断)")
|
||
}
|
||
return strings.Join(parts, " -> ")
|
||
}
|
||
|
||
func printf(level string, format string, args ...any) {
|
||
Init()
|
||
logMu.Lock()
|
||
inst := logInst
|
||
logMu.Unlock()
|
||
if inst == nil {
|
||
return
|
||
}
|
||
inst.Printf("[%s] %s", level, fmt.Sprintf(format, args...))
|
||
}
|
||
|
||
func initOutput() (string, io.Writer) {
|
||
dir := strings.TrimSpace(os.Getenv(envLogDir))
|
||
if dir == "" {
|
||
base, err := os.UserConfigDir()
|
||
if err != nil || strings.TrimSpace(base) == "" {
|
||
base = os.TempDir()
|
||
}
|
||
dir = filepath.Join(base, appDirName, "logs")
|
||
}
|
||
|
||
if err := os.MkdirAll(dir, 0o755); err != nil {
|
||
return filepath.Join(dir, logFileName), os.Stderr
|
||
}
|
||
|
||
path := filepath.Join(dir, logFileName)
|
||
rotateIfNeeded(path, dir)
|
||
|
||
f, err := os.OpenFile(path, os.O_CREATE|os.O_APPEND|os.O_WRONLY, 0o644)
|
||
if err != nil {
|
||
return path, os.Stderr
|
||
}
|
||
logFile = f
|
||
return path, f
|
||
}
|
||
|
||
func rotateIfNeeded(path, dir string) {
|
||
fi, err := os.Stat(path)
|
||
if err != nil || fi.IsDir() {
|
||
return
|
||
}
|
||
if fi.Size() < logRotateMaxBytes {
|
||
return
|
||
}
|
||
|
||
ts := time.Now().Format("20060102-150405")
|
||
rotated := filepath.Join(dir, fmt.Sprintf("gonavi-%s.log", ts))
|
||
if err := os.Rename(path, rotated); err != nil {
|
||
return
|
||
}
|
||
cleanupOldLogs(dir)
|
||
}
|
||
|
||
func cleanupOldLogs(dir string) {
|
||
entries, err := os.ReadDir(dir)
|
||
if err != nil {
|
||
return
|
||
}
|
||
|
||
type item struct {
|
||
name string
|
||
path string
|
||
}
|
||
var logs []item
|
||
for _, e := range entries {
|
||
if e.IsDir() {
|
||
continue
|
||
}
|
||
name := e.Name()
|
||
if !strings.HasPrefix(name, "gonavi-") || !strings.HasSuffix(name, ".log") {
|
||
continue
|
||
}
|
||
logs = append(logs, item{name: name, path: filepath.Join(dir, name)})
|
||
}
|
||
|
||
sort.Slice(logs, func(i, j int) bool { return logs[i].name > logs[j].name })
|
||
if len(logs) <= logRotateMaxBackups {
|
||
return
|
||
}
|
||
for _, it := range logs[logRotateMaxBackups:] {
|
||
_ = os.Remove(it.path)
|
||
}
|
||
}
|