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.

250 lines
5.4 KiB

5 years ago
package loggy
import "samhofi.us/x/keybase"
import "time"
import "fmt"
import "os"
// Level of importance for a log message
type LogLevel int
const (
// Info level logging
Info LogLevel = 4
5 years ago
// Debugging output
Debug LogLevel = 5
5 years ago
// Will show if logger is set to warning
Warnings LogLevel = 3
// Errors will show by default
Errors LogLevel = 2
// Critical will show but can be silenced
Critical LogLevel = 1
// Special level for only showing via stdout
StdoutOnly LogLevel = 0
)
// A basic Log struct with type and message
type Log struct {
Level LogLevel
Msg string
}
// Logging options 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
5 years ago
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
5 years ago
UseStdout bool
}
5 years ago
// A basic Logger with options for logging to file, keybase or stdout.
// More functionality could be added within the internal handleLog() func.
type Logger struct {
5 years ago
opts LogOpts
k *keybase.Keybase
team keybase.Channel
}
5 years ago
// Generate string from type Log with severity prepended
5 years ago
func (msg Log) String() string {
levels := [...]string{
"StdoutOnly",
"Critical",
"Error",
"Warning",
"Info",
"Debug"}
5 years ago
return fmt.Sprintf("%s: %s", levels[msg.Level], msg.Msg)
}
5 years ago
// Generate a timestamp for non-Keybase logs
5 years ago
func timeStamp() string {
now := time.Now()
return now.Format("02Jan06 15:04:05.9999")
}
5 years ago
// Write log to file from LogOpts
func (l Logger) toFile(msg Log) {
5 years ago
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())
5 years ago
chat := l.k.NewChat(l.team)
chat.Send(output)
}
// Write log to Stdout
func (l Logger) toStdout(msg Log) {
5 years ago
output := fmt.Sprintf("[%s] %s",
timeStamp(), msg.String())
fmt.Println(output)
}
// Log Info shortcut from string
func (l Logger) LogInfo(msg string) {
5 years ago
var logMsg Log
logMsg.Level = Info
logMsg.Msg = msg
go handleLog(l, logMsg)
5 years ago
}
// Log Debug shortcut from string
func (l Logger) LogDebug(msg string) {
5 years ago
var logMsg Log
logMsg.Level = Debug
logMsg.Msg = msg
go handleLog(l, logMsg)
5 years ago
}
// Log Warning shortcut from string
func (l Logger) LogWarn(msg string) {
5 years ago
var logMsg Log
logMsg.Level = Warnings
logMsg.Msg = msg
go handleLog(l, logMsg)
5 years ago
}
// Log Error shortcut from string - Will notify Keybase users
func (l Logger) LogError(msg string) {
5 years ago
var logMsg Log
logMsg.Level = Errors
logMsg.Msg = msg
go handleLog(l, logMsg)
5 years ago
}
// Log Critical shortcut from string - Will notifiy Keybase users
func (l Logger) LogCritical(msg string) {
5 years ago
var logMsg Log
logMsg.Level = Critical
logMsg.Msg = msg
go handleLog(l, logMsg)
}
// Log Critical shortcut that terminates program
func (l Logger) LogPanic(msg string) {
var logMsg Log
logMsg.Level = Critical
logMsg.Msg = msg
5 years ago
handleLog(l, logMsg)
5 years ago
os.Exit(-1)
5 years ago
}
// Log error type for compatibility - Will notify keybase users
func (l Logger) LogErrorType(e error) {
5 years ago
var logMsg Log
5 years ago
// Will set Level to Critical without terminating program
5 years ago
logMsg.Level = Critical
logMsg.Msg = e.Error()
go handleLog(l, logMsg)
5 years ago
}
// Func to hack to add other logging functionality
func handleLog(l Logger, logMsg Log) {
5 years ago
if logMsg.Level > l.opts.Level && logMsg.Level != 0 {
return
}
if logMsg.Level == 0 {
go l.toStdout(logMsg)
5 years ago
return
}
if l.opts.toKeybase {
go l.toKeybase(logMsg)
5 years ago
}
if l.opts.toFile {
go l.toFile(logMsg)
5 years ago
}
if l.opts.toStdout {
go l.toStdout(logMsg)
5 years ago
}
}
// Log func, takes LogLevel and string and passes to internal handler.
func (l Logger) Log(level LogLevel, msg string) {
5 years ago
var logMsg Log
logMsg.Level = level
logMsg.Msg = msg
go handleLog(l, logMsg)
5 years ago
}
// LogMsg takes a type Log and passes it to internal handler.
func (l Logger) LogMsg(msg Log) {
go handleLog(l, msg)
5 years ago
}
// 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))
}
}
5 years ago
// Create a new logger instance and pass it
func NewLogger(opts LogOpts) Logger {
5 years ago
if opts.Level == 0 {
opts.Level = 2
}
var l Logger
5 years ago
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
}