mirror of https://github.com/Rudi9719/loggy.git
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.
252 lines
5.6 KiB
252 lines
5.6 KiB
package loggy |
|
|
|
import ( |
|
"fmt" |
|
"os" |
|
"time" |
|
|
|
"samhofi.us/x/keybase" |
|
) |
|
|
|
// LogLevel of importance for a log message |
|
type LogLevel int |
|
|
|
const ( |
|
// Debug output and below |
|
Debug LogLevel = 5 |
|
// Info output and below |
|
Info LogLevel = 4 |
|
// Warnings and below |
|
Warnings LogLevel = 3 |
|
// Errors will show by default |
|
Errors LogLevel = 2 |
|
// Critical will show but can be silenced |
|
Critical LogLevel = 1 |
|
// StdoutOnly for only showing via stdout |
|
StdoutOnly LogLevel = 0 |
|
) |
|
|
|
// Log struct with type and message |
|
type Log struct { |
|
Level LogLevel |
|
Msg string |
|
} |
|
|
|
// LogOpts to be passed to NewLogger() |
|
type LogOpts struct { |
|
// Will set to true if OutFile is set |
|
toFile bool |
|
// Will set to true if KBTeam is set |
|
toKeybase bool |
|
// Will set to true if UseStdout is true |
|
toStdout bool |
|
// Output file for logging - Required for file output |
|
OutFile string |
|
// Keybase Team for logging - Required for Keybase output |
|
KBTeam string |
|
// Keybase Channel for logging - Optional for Keybase output |
|
KBChann string |
|
// Log level / verbosity (see LogLevel) |
|
Level LogLevel |
|
// Program name for Keybase logging - Required for Keybase output |
|
ProgName string |
|
// Use stdout - Required to print to stdout |
|
UseStdout bool |
|
} |
|
|
|
// Logger with options for logging to file, keybase or stdout. |
|
// More functionality could be added within the internal handleLog() func. |
|
type Logger struct { |
|
opts LogOpts |
|
k *keybase.Keybase |
|
team keybase.Channel |
|
} |
|
|
|
// Generate string from type Log with severity prepended |
|
func (msg Log) String() string { |
|
levels := [...]string{ |
|
"StdoutOnly", |
|
"Critical", |
|
"Error", |
|
"Warning", |
|
"Info", |
|
"Debug"} |
|
return fmt.Sprintf("%s: %s", levels[msg.Level], msg.Msg) |
|
} |
|
|
|
// Generate a timestamp for non-Keybase logs |
|
func timeStamp() string { |
|
now := time.Now() |
|
return now.Format("02Jan06 15:04:05.9999") |
|
} |
|
|
|
// Write log to file from LogOpts |
|
func (l Logger) toFile(msg Log) { |
|
output := fmt.Sprintf("[%s] %s", |
|
timeStamp(), msg.String()) |
|
|
|
f, err := os.OpenFile(l.opts.OutFile, |
|
os.O_APPEND|os.O_CREATE|os.O_WRONLY, 0644) |
|
if err != nil { |
|
fmt.Println("Unable to open logging file") |
|
} |
|
defer f.Close() |
|
if _, err := f.WriteString(fmt.Sprintf("%s\n", output)); err != nil { |
|
fmt.Println("Error writing output to logging file") |
|
} |
|
|
|
} |
|
|
|
// Send log to Keybase |
|
func (l Logger) toKeybase(msg Log) { |
|
tag := "" |
|
if msg.Level <= 2 { |
|
tag = "@everyone " |
|
} |
|
output := fmt.Sprintf("[%s] %s%s", |
|
l.opts.ProgName, tag, msg.String()) |
|
|
|
chat := l.k.NewChat(l.team) |
|
chat.Send(output) |
|
|
|
} |
|
|
|
// Write log to Stdout |
|
func (l Logger) toStdout(msg Log) { |
|
output := fmt.Sprintf("[%s] %s", |
|
timeStamp(), msg.String()) |
|
fmt.Println(output) |
|
} |
|
|
|
// LogInfo shortcut from string |
|
func (l Logger) LogInfo(msg string, a ...interface{}) { |
|
var logMsg Log |
|
logMsg.Level = Info |
|
logMsg.Msg = fmt.Sprintf(msg, a...) |
|
go handleLog(l, logMsg) |
|
} |
|
|
|
// LogDebug shortcut from string |
|
func (l Logger) LogDebug(msg string, a ...interface{}) { |
|
var logMsg Log |
|
logMsg.Level = Debug |
|
logMsg.Msg = fmt.Sprintf(msg, a...) |
|
|
|
go handleLog(l, logMsg) |
|
} |
|
|
|
// LogWarn shortcut from string |
|
func (l Logger) LogWarn(msg string, a ...interface{}) { |
|
var logMsg Log |
|
logMsg.Level = Warnings |
|
logMsg.Msg = fmt.Sprintf(msg, a...) |
|
go handleLog(l, logMsg) |
|
} |
|
|
|
// LogError shortcut from string - Will notify Keybase users |
|
func (l Logger) LogError(msg string, a ...interface{}) { |
|
var logMsg Log |
|
logMsg.Level = Errors |
|
logMsg.Msg = fmt.Sprintf(msg, a...) |
|
go handleLog(l, logMsg) |
|
} |
|
|
|
// LogCritical shortcut from string - Will notifiy Keybase users |
|
func (l Logger) LogCritical(msg string, a ...interface{}) { |
|
var logMsg Log |
|
logMsg.Level = Critical |
|
logMsg.Msg = fmt.Sprintf(msg, a...) |
|
go handleLog(l, logMsg) |
|
} |
|
|
|
// LogPanic is a LogCritical shortcut that terminates program |
|
func (l Logger) LogPanic(msg string, a ...interface{}) { |
|
var logMsg Log |
|
logMsg.Level = Critical |
|
logMsg.Msg = fmt.Sprintf(msg, a...) |
|
handleLog(l, logMsg) |
|
os.Exit(-1) |
|
} |
|
|
|
// LogErrorType for compatibility - Will notify keybase users |
|
func (l Logger) LogErrorType(e error) { |
|
var logMsg Log |
|
// Will set Level to Critical without terminating program |
|
logMsg.Level = Critical |
|
logMsg.Msg = e.Error() |
|
go handleLog(l, logMsg) |
|
} |
|
|
|
// Func to hack to add other logging functionality |
|
func handleLog(l Logger, logMsg Log) { |
|
|
|
if logMsg.Level > l.opts.Level && logMsg.Level != 0 { |
|
return |
|
} |
|
if logMsg.Level == 0 { |
|
go l.toStdout(logMsg) |
|
return |
|
} |
|
if l.opts.toKeybase { |
|
go l.toKeybase(logMsg) |
|
} |
|
if l.opts.toFile { |
|
go l.toFile(logMsg) |
|
} |
|
if l.opts.toStdout { |
|
go l.toStdout(logMsg) |
|
} |
|
|
|
} |
|
|
|
// Log func, takes LogLevel and string and passes to internal handler. |
|
func (l Logger) Log(level LogLevel, msg string) { |
|
var logMsg Log |
|
logMsg.Level = level |
|
logMsg.Msg = msg |
|
go handleLog(l, logMsg) |
|
} |
|
|
|
// LogMsg takes a type Log and passes it to internal handler. |
|
func (l Logger) LogMsg(msg Log) { |
|
go handleLog(l, msg) |
|
} |
|
|
|
// PanicSafe is a deferrable function to recover from a panic operation. |
|
func (l Logger) PanicSafe() { |
|
if r := recover(); r != nil { |
|
l.LogCritical(fmt.Sprintf("Panic detected: %+v", r)) |
|
} |
|
} |
|
|
|
// NewLogger creates a new logger instance using LogOpts |
|
func NewLogger(opts LogOpts) Logger { |
|
if opts.Level == 0 { |
|
opts.Level = 2 |
|
} |
|
var l Logger |
|
if opts.KBTeam != "" { |
|
l.k = keybase.NewKeybase() |
|
var chann keybase.Channel |
|
if opts.KBChann != "" { |
|
chann.TopicName = opts.KBChann |
|
chann.MembersType = keybase.TEAM |
|
} else { |
|
chann.MembersType = keybase.USER |
|
} |
|
chann.Name = opts.KBTeam |
|
opts.toKeybase = true |
|
if !l.k.LoggedIn { |
|
fmt.Println("Not logged into keybase, but keybase option set.") |
|
os.Exit(-1) |
|
} |
|
l.team = chann |
|
} |
|
if opts.OutFile != "" { |
|
opts.toFile = true |
|
} |
|
opts.toStdout = opts.UseStdout |
|
l.opts = opts |
|
|
|
return l |
|
}
|
|
|