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 = 5 // Debugging output Debug LogLevel = 4 // 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 toKeybase bool // Will set to true if UseStdout is true toStdout bool // Output file for logging OutFile string // Keybase Team for logging KBTeam string // Keybase Channel for logging KBChann string // Log level / verbosity (see LogLevel) Level LogLevel // Program name for Keybase logging ProgName string // Use stdout UseStdout bool } // 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 { 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", "Debug", "Info"} 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) } // Log Info shortcut from string func (l Logger) LogInfo(msg string) { var logMsg Log logMsg.Level = Info logMsg.Msg = msg go handleLog(l, logMsg) } // Log Debug shortcut from string func (l Logger) LogDebug(msg string) { var logMsg Log logMsg.Level = Debug logMsg.Msg = msg go handleLog(l, logMsg) } // Log Warning shortcut from string func (l Logger) LogWarn(msg string) { var logMsg Log logMsg.Level = Warnings logMsg.Msg = msg go handleLog(l, logMsg) } // Log Error shortcut from string - Will notify Keybase users func (l Logger) LogError(msg string) { var logMsg Log logMsg.Level = Errors logMsg.Msg = msg go handleLog(l, logMsg) } // Log Critical shortcut from string - Will notifiy Keybase users func (l Logger) LogCritical(msg string) { 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 handleLog(l, logMsg) os.Exit(-1) } // Log error type 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) } // Create a new logger instance and pass it 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 }