gorush/logx/log.go

259 lines
5.2 KiB
Go

package logx
import (
"encoding/json"
"errors"
"fmt"
"os"
"strings"
"github.com/appleboy/gorush/core"
"github.com/mattn/go-isatty"
"github.com/sirupsen/logrus"
)
var (
green = string([]byte{27, 91, 57, 55, 59, 52, 50, 109})
yellow = string([]byte{27, 91, 57, 55, 59, 52, 51, 109})
red = string([]byte{27, 91, 57, 55, 59, 52, 49, 109})
blue = string([]byte{27, 91, 57, 55, 59, 52, 52, 109})
reset = string([]byte{27, 91, 48, 109})
)
// LogPushEntry is push response log
type LogPushEntry struct {
ID string `json:"notif_id,omitempty"`
Type string `json:"type"`
Platform string `json:"platform"`
Token string `json:"token"`
Message string `json:"message"`
Error string `json:"error"`
}
var isTerm bool
// nolint
func init() {
isTerm = isatty.IsTerminal(os.Stdout.Fd())
}
var (
// LogAccess is log server request log
LogAccess = logrus.New()
// LogError is log server error log
LogError = logrus.New()
)
// InitLog use for initial log module
func InitLog(accessLevel, accessLog, errorLevel, errorLog string) error {
var err error
if !isTerm {
LogAccess.SetFormatter(&logrus.JSONFormatter{})
LogError.SetFormatter(&logrus.JSONFormatter{})
} else {
LogAccess.Formatter = &logrus.TextFormatter{
TimestampFormat: "2006/01/02 - 15:04:05",
FullTimestamp: true,
}
LogError.Formatter = &logrus.TextFormatter{
TimestampFormat: "2006/01/02 - 15:04:05",
FullTimestamp: true,
}
}
// set logger
if err = SetLogLevel(LogAccess, accessLevel); err != nil {
return errors.New("Set access log level error: " + err.Error())
}
if err = SetLogLevel(LogError, errorLevel); err != nil {
return errors.New("Set error log level error: " + err.Error())
}
if err = SetLogOut(LogAccess, accessLog); err != nil {
return errors.New("Set access log path error: " + err.Error())
}
if err = SetLogOut(LogError, errorLog); err != nil {
return errors.New("Set error log path error: " + err.Error())
}
return nil
}
// SetLogOut provide log stdout and stderr output
func SetLogOut(log *logrus.Logger, outString string) error {
switch outString {
case "stdout":
log.Out = os.Stdout
case "stderr":
log.Out = os.Stderr
default:
f, err := os.OpenFile(outString, os.O_RDWR|os.O_CREATE|os.O_APPEND, 0o600)
if err != nil {
return err
}
log.Out = f
}
return nil
}
// SetLogLevel is define log level what you want
// log level: panic, fatal, error, warn, info and debug
func SetLogLevel(log *logrus.Logger, levelString string) error {
level, err := logrus.ParseLevel(levelString)
if err != nil {
return err
}
log.Level = level
return nil
}
func colorForPlatForm(platform int) string {
switch platform {
case core.PlatFormIos:
return blue
case core.PlatFormAndroid:
return yellow
case core.PlatFormHuawei:
return green
default:
return reset
}
}
func typeForPlatForm(platform int) string {
switch platform {
case core.PlatFormIos:
return "ios"
case core.PlatFormAndroid:
return "android"
case core.PlatFormHuawei:
return "huawei"
default:
return ""
}
}
func hideToken(token string, markLen int) string {
if token == "" {
return ""
}
if len(token) < markLen*2 {
return strings.Repeat("*", len(token))
}
start := token[len(token)-markLen:]
end := token[0:markLen]
result := strings.ReplaceAll(token, start, strings.Repeat("*", markLen))
result = strings.ReplaceAll(result, end, strings.Repeat("*", markLen))
return result
}
// GetLogPushEntry get push data into log structure
func GetLogPushEntry(input *InputLog) LogPushEntry {
var errMsg string
plat := typeForPlatForm(input.Platform)
if input.Error != nil {
errMsg = input.Error.Error()
}
token := input.Token
if input.HideToken {
token = hideToken(input.Token, 10)
}
message := input.Message
if input.HideMessage {
message = "(message redacted)"
}
return LogPushEntry{
ID: input.ID,
Type: input.Status,
Platform: plat,
Token: token,
Message: message,
Error: errMsg,
}
}
// InputLog log request
type InputLog struct {
ID string
Status string
Token string
Message string
Platform int
Error error
HideToken bool
HideMessage bool
Format string
}
// LogPush record user push request and server response.
func LogPush(input *InputLog) LogPushEntry {
var platColor, resetColor, output string
if isTerm {
platColor = colorForPlatForm(input.Platform)
resetColor = reset
}
log := GetLogPushEntry(input)
if input.Format == "json" {
logJSON, _ := json.Marshal(log)
output = string(logJSON)
} else {
var typeColor string
switch input.Status {
case core.SucceededPush:
if isTerm {
typeColor = green
}
output = fmt.Sprintf("|%s %s %s| %s%s%s [%s] %s",
typeColor, log.Type, resetColor,
platColor, log.Platform, resetColor,
log.Token,
log.Message,
)
case core.FailedPush:
if isTerm {
typeColor = red
}
output = fmt.Sprintf("|%s %s %s| %s%s%s [%s] | %s | Error Message: %s",
typeColor, log.Type, resetColor,
platColor, log.Platform, resetColor,
log.Token,
log.Message,
log.Error,
)
}
}
switch input.Status {
case core.SucceededPush:
LogAccess.Info(output)
case core.FailedPush:
LogError.Error(output)
}
return log
}