golang中有很多优秀的日志库,例如zerolog,see-log,blog4go等。 这些日志首先具备强大的高并发能力,其次具备丰富的接口,输入格式,索引接口。 把外壳去掉,日志库的设计原理本质上是一样的。
我个人认为一个日志系统需要满足的特点:
- 1 内容不确定(非格式化数据)。为能够方便索引,因此需要额外引入一些namespace和类型,时间戳等的字符串。
- 2 要求并发量大,写入快,也就是日志需要有实时性,不能对现有业务造成阻塞。 因此日志需要设置成带缓冲的功能、多协程。 在日志的生成、粗加工、传递、归类、输出的过程中,减少传递、减少cpu锁、减少拷贝次数/内容 方面下功夫,持久化优化,能提升日志库的性能。
- 3 日志需要一定的格式,因此在输出时需要顺序输出到终端、或日志文件。 落盘需要是一个任务去执行。
- 4 日志文件不能太大,需要定期清理掉无效的日志。 因此日志按天、周为单位,每个单位写入对应以时间戳命名的文件中。
- 5 参照linux系统自带的系统日志,对于历史日志(例如一周之前的日志),需要压缩成压缩文档。删掉非压缩的文档。
- 6 输出目标可以灵活指定,可以是终端,也可以是日志目录,也可以是网络流。 输出格式可以是raw原始内容、也可以是json格式。
- 7 考虑小白用户。例如一些运维人员,vim用的也许不是那么的熟悉,还有一些c端的小白用户估计linux系统都不晓得是啥,因此日志系统如果能带一个简单的web界面,日志索引提供输入框返回结果,这虽然不一定用的上,但是会显得比较完善。
补充:本文比较仓促,是以学习原理为目的(我写的文章侧重原理分析 和 实现。 当然,生产环境是另外一套另当别论,我优先考虑用别人写好的组件)。 因此上述7个特点并未在本文全部实现。
直接上源码:
package main
import (
"context"
"fmt"
"io"
"os"
"sync"
"time"
)
var (
colorOff = []byte("\033[0m")
colorRed = []byte("\033[0;31m")
colorGreen = []byte("\033[0;32m")
colorOrange = []byte("\033[0;33m")
colorBlue = []byte("\033[0;34m")
colorPurple = []byte("\033[0;35m")
colorCyan = []byte("\033[0;36m")
colorGray = []byte("\033[0;37m")
)
var (
plainFatal = []byte("[FATAL] ")
plainError = []byte("[ERROR] ")
plainWarn = []byte("[WARN] ")
plainInfo = []byte("[INFO] ")
plainDebug = []byte("[DEBUG] ")
plainTrace = []byte("[TRACE] ")
)
var (
MAX_INJECTION = 20000
)
// 着色器
func mixer(data []byte, color []byte) []byte {
var result []byte
return append(append(append(result, color...), data...), colorOff...)
}
// Red 将 data 染成红色
func Red(data []byte) []byte {
return mixer(data, colorRed)
}
func Green(data []byte) []byte {
return mixer(data, colorGreen)
}
func Orange(data []byte) []byte {
return mixer(data, colorOrange)
}
func Blue(data []byte) []byte {
return mixer(data, colorBlue)
}
func Purple(data []byte) []byte {
return mixer(data, colorPurple)
}
func Cyan(data []byte) []byte {
return mixer(data, colorCyan)
}
func Gray(data []byte) []byte {
return mixer(data, colorGray)
}
type event struct {
Plain []byte
Color []byte
Data []byte
}
type Logger struct {
ctx context.Context
start time.Time
mu sync.Mutex
prefix string
cnt int64
fsig chan struct{}
ch chan *event
pevents *[]*event
events [2][]*event
out io.Writer
rootdir string
namespace string
oj bool // output as json format
}
func NewLogger(ctx context.Context, out io.Writer, prefix string, cnt int64, dir string, ns string) *Logger {
log := &Logger{
out: out,
prefix: prefix,
cnt: cnt,
fsig: make(chan struct{}, 1),
ch: make(chan *event, MAX_INJECTION),
ctx: ctx,
rootdir: dir,
namespace: ns,
}
log.pevents = &log.events[0]
go log.Run()
return log
}
// SetOutput sets the output destination for the logger.
func (l *Logger) SetOutput(w io.Writer) {
l.mu.Lock()
defer l.mu.Unlock()
l.out = w
}
func (l *Logger) Run() {
fsync := func() {
l.mu.Lock()
if len(*l.pevents) == 0 {
l.mu.Unlock()
return
}
pcurevents := l.pevents
for i, _ := range l.events {
if pcurevents != &l.events[i] {
l.pevents = &l.events[i]
break
}
}
l.mu.Unlock()
for _, v := range *pcurevents {
if l.out == os.Stderr || l.out == os.Stdout {
l.out.Write(v.Color)
} else {
l.out.Write(v.Plain)
}
l.out.Write([]byte(l.prefix))
l.out.Write([]byte(fmt.Sprint(time.Now())))
l.out.Write(v.Data)
}
*pcurevents = (*pcurevents)[0:0]
}
for i := 0; i < 10; i++ {
go func() {
for {
select {
case <-l.ctx.Done():
return
case req := <-l.ch:
l.mu.Lock()
*l.pevents = append(*l.pevents, req)
l.cnt++
l.mu.Unlock()
}
}
}()
}
go func() {
defer func() {
fsync()
}()
for {
select {
case <-l.ctx.Done():
return
case <-time.After(time.Second * 5):
fsync()
case <-l.fsig:
fsync()
}
}
}()
}
func (l *Logger) Output(e *event) {
if len(l.ch)+1 >= MAX_INJECTION {
l.fsig <- struct{}{}
}
l.ch <- e
}
func (l *Logger) Debug(v ...interface{}) {
s := fmt.Sprint(v...)
s += "\r\n"
l.Output(&event{
Plain: plainDebug,
Color: Purple(plainDebug),
Data: []byte(s),
})
}
func (l *Logger) Info(v ...interface{}) {
s := fmt.Sprint(v...)
s += "\r\n"
l.Output(&event{
Plain: plainInfo,
Color: Green(plainInfo),
Data: []byte(s),
})
}
func (l *Logger) Warn(v ...interface{}) {
s := fmt.Sprint(v...)
s += "\r\n"
l.Output(&event{
Plain: plainWarn,
Color: Orange(plainWarn),
Data: []byte(s),
})
}
func (l *Logger) Error(v ...interface{}) {
s := fmt.Sprint(v...)
s += "\r\n"
l.Output(&event{
Plain: plainError,
Color: Red(plainError),
Data: []byte(s),
})
}
var std = NewLogger(context.Background(), os.Stderr, "mylog->", 0, "", "misc")
func Default() *Logger { return std }
func main() {
// 功能测试
std.Info("这里是一条Info日志, a")
std.Info("这里是一条Info日志, b")
std.Info("这里是一条Info日志, c")
std.Info("这里是一条Info日志, d")
std.Info("这里是一条Info日志, e")
std.Debug("这里是一条Debuf日志, m")
std.Error("这里是一条Error日志, n")
std.Warn("这里是一条Warn日志, n")
outstr := ""
tf := func() {
std.start = time.Now()
wg := &sync.WaitGroup{}
for i := 0; i < 200; i++ {
wg.Add(1)
go func() {
for j := 0; j < 100000; j++ {
std.Info("-----------------------------------------------------info 日志")
}
wg.Done()
}()
}
time.Sleep(time.Microsecond * 200)
wg.Wait()
diff := time.Now().Sub(std.start).Seconds()
outstr = fmt.Sprintf("logger is exit now, total %+v item, used %+v second, ", std.cnt, diff)
}
//fmt.Println("输出到终端,性能测试")
//std.SetOutput(std.out)
//tf()
fmt.Println("输出到文件,性能测试")
///*
fd, err := os.OpenFile("./test.log", os.O_CREATE|os.O_APPEND|os.O_RDWR, 0666)
if err != nil {
panic(err)
}
std.SetOutput(fd)
tf()
fd.Sync()
//*/
fmt.Println(outstr)
}
硬件:
root@jack-VirtualBox:~/comment/logger# lscpu
Architecture: x86_64
CPU op-mode(s): 32-bit, 64-bit
Byte Order: Little Endian
Address sizes: 39 bits physical, 48 bits virtual
CPU(s): 4
On-line CPU(s) list: 0-3
Thread(s) per core: 1
Core(s) per socket: 4
Socket(s): 1
NUMA node(s): 1
Vendor ID: GenuineIntel
CPU family: 6
Model: 165
Model name: Intel(R) Core(TM) i3-10100 CPU @ 3.60GHz
Stepping: 3
CPU MHz: 3600.006
BogoMIPS: 7200.01
Hypervisor vendor: KVM
Virtualization type: full
L1d cache: 128 KiB
L1i cache: 128 KiB
L2 cache: 1 MiB
L3 cache: 24 MiB
NUMA node0 CPU(s): 0-3
Vulnerability Itlb multihit: KVM: Mitigation: VMX unsupported
Vulnerability L1tf: Not affected
Vulnerability Mds: Not affected
Vulnerability Meltdown: Not affected
Vulnerability Spec store bypass: Vulnerable
Vulnerability Spectre v1: Mitigation; usercopy/swapgs barriers and __user pointer sanitization
Vulnerability Spectre v2: Mitigation; Full generic retpoline, STIBP disabled, RSB filling
Vulnerability Srbds: Not affected
Vulnerability Tsx async abort: Not affected
Flags: fpu vme de pse tsc msr pae mce cx8 apic sep mtrr pge mca cmov pat pse36 clflush mmx fxsr sse sse2 ht syscall nx rdtscp lm constant_tsc rep_good nopl xtopology nonstop_tsc cpuid ts
c_known_freq pni pclmulqdq ssse3 cx16 pcid sse4_1 sse4_2 x2apic movbe popcnt aes xsave avx rdrand hypervisor lahf_lm abm 3dnowprefetch invpcid_single fsgsbase avx2 invpcid rdseed
clflushopt md_clear flush_l1d arch_capabilities
root@jack-VirtualBox:~/comment/logger#
root@jack-VirtualBox:~/comment/logger# free -h
total used free shared buff/cache available
Mem: 7.8Gi 2.1Gi 1.8Gi 1.0Mi 3.9Gi 5.4Gi
Swap: 1.6Gi 1.0Mi 1.6Gi
root@jack-VirtualBox:~/comment/logger#
功能测试:
压力测试:
参考值20w条日志/秒。 磁盘,显卡,cpu等的性能等有对此时结果有影响。
同样的硬件配置,golang系统自带的l"log"包的速度是3.7w条日志/秒。
输出到日志文件:
func main() {
file, err := os.Create("test.log")
if err != nil {
log.Fatalln("fail to create test.log file!")
}
logger := log.New(file, "", log.Llongfile)
// 写入文件log格式:/Users/zhou/go/src/zhouTest/test.go:22: 2.Println log with log.LstdFlags ...
logger.Println("2.Println log with log.LstdFlags ...")
logger.SetFlags(log.LstdFlags) // 设置写入文件的log日志的格式
// 写入log文件格式: 2018/07/31 17:28:21 4.Println log without log.LstdFlags ...
logger.Println("4.Println log without log.LstdFlags ...")
fmt.Println("打印")
logger.Fatal("9.Fatal log without log.LstdFlags ...")
fmt.Println("Fatal终止了程序,这句不执行!")
}