The core protocol of WoopChain
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
woop/internal/utils/singleton.go

171 lines
4.1 KiB

/* This module keeps all struct used as singleton */
package utils
import (
"fmt"
"io"
"os"
"path"
"sync"
"sync/atomic"
"time"
"github.com/ethereum/go-ethereum/log"
"github.com/natefinch/lumberjack"
"github.com/rs/zerolog"
)
var (
// Validator ID
validatorIDInstance *UniqueValidatorID
onceForUniqueValidatorID sync.Once
// Global port and ip for logging.
port string
ip string
// Logging
logInstance log.Logger
glogger *log.GlogHandler // top-level handler
logHandlers []log.Handler // sub handlers of glogger
logVerbosity log.Lvl
onceForLog sync.Once
// ZeroLog
zeroLogger *zerolog.Logger
zeroLoggerLevel = zerolog.Disabled
)
// SetLogContext used to print out loggings of node with port and ip.
// Every instance (node, etc..) needs to set this for logging.
func SetLogContext(_port, _ip string) {
port = _port
ip = _ip
setZeroLogContext(_port, _ip)
}
// SetLogVerbosity specifies the verbosity of global logger
func SetLogVerbosity(verbosity log.Lvl) {
logVerbosity = verbosity
if glogger != nil {
glogger.Verbosity(logVerbosity)
}
updateZeroLogLevel(int(verbosity))
}
// AddLogFile creates a StreamHandler that outputs JSON logs
// into rotating files with specified max file size
func AddLogFile(filepath string, maxSize int) {
AddLogHandler(log.StreamHandler(&lumberjack.Logger{
Filename: filepath,
MaxSize: maxSize,
Compress: true,
}, log.JSONFormat()))
setZeroLoggerFileOutput(filepath, maxSize)
}
// AddLogHandler add a log handler
func AddLogHandler(handler log.Handler) {
logHandlers = append(logHandlers, handler)
if glogger != nil {
multiHandler := log.MultiHandler(logHandlers...)
glogger.SetHandler(multiHandler)
}
}
// UniqueValidatorID defines the structure of unique validator ID
type UniqueValidatorID struct {
uniqueID uint32
}
// GetUniqueValidatorIDInstance returns a singleton instance
func GetUniqueValidatorIDInstance() *UniqueValidatorID {
onceForUniqueValidatorID.Do(func() {
validatorIDInstance = &UniqueValidatorID{
uniqueID: 0,
}
})
return validatorIDInstance
}
// GetUniqueID returns a unique ID and increment the internal variable
func (s *UniqueValidatorID) GetUniqueID() uint32 {
return atomic.AddUint32(&s.uniqueID, 1)
}
// GetLogInstance returns logging singleton.
func GetLogInstance() log.Logger {
onceForLog.Do(func() {
ostream := log.StreamHandler(io.Writer(os.Stdout), log.TerminalFormat(false))
logHandlers = append(logHandlers, ostream)
multiHandler := log.MultiHandler(logHandlers...)
glogger = log.NewGlogHandler(multiHandler)
glogger.Verbosity(logVerbosity)
logInstance = log.New("port", port, "ip", ip)
logInstance.SetHandler(glogger)
log.Root().SetHandler(glogger)
})
return logInstance
}
// ZeroLog
func setZeroLogContext(port string, ip string) {
childLogger := Logger().
With().
Str("port", port).
Str("ip", ip).
Logger()
zeroLogger = &childLogger
}
// SetZeroLoggerFileOutput sets zeroLogger's output stream
// to destinated filepath with log file rotation.
func setZeroLoggerFileOutput(filepath string, maxSize int) error {
dir := path.Dir(filepath)
filename := path.Base(filepath)
// Initialize ZeroLogger if it hasn't been already
// TODO: zerolog filename prefix can be removed once all loggers
// has been replaced
childLogger := Logger().Output(&lumberjack.Logger{
Filename: fmt.Sprintf("%s/zerolog-%s", dir, filename),
MaxSize: maxSize,
Compress: true,
})
zeroLogger = &childLogger
return nil
}
// Logger returns a zerolog.Logger singleton
func Logger() *zerolog.Logger {
if zeroLogger == nil {
zerolog.TimeFieldFormat = time.RFC3339Nano
logger := zerolog.New(zerolog.ConsoleWriter{Out: os.Stderr}).
Level(zeroLoggerLevel).
With().
Caller().
Timestamp().
Logger()
zeroLogger = &logger
}
return zeroLogger
}
func updateZeroLogLevel(level int) {
switch level {
case 0:
zeroLoggerLevel = zerolog.Disabled
case 1:
zeroLoggerLevel = zerolog.ErrorLevel
case 2:
zeroLoggerLevel = zerolog.WarnLevel
case 3:
zeroLoggerLevel = zerolog.InfoLevel
default:
zeroLoggerLevel = zerolog.DebugLevel
}
childLogger := Logger().Level(zeroLoggerLevel)
zeroLogger = &childLogger
}