blob: a49fbc041584b307cebccda14bfd815b862a82be [file] [log] [blame]
Rob Pike806d00f2009-01-26 09:28:07 -08001// Copyright 2009 The Go Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style
3// license that can be found in the LICENSE file.
4
5// Rudimentary logging package. Defines a type, Logger, with simple
6// methods for formatting output to one or two destinations. Also has
7// predefined Loggers accessible through helper functions Stdout[f],
Rob Pike26cb4df2009-03-04 22:46:44 -08008// Stderr[f], Exit[f], and Crash[f], which are easier to use than creating
9// a Logger manually.
Rob Pike806d00f2009-01-26 09:28:07 -080010// Exit exits when written to.
11// Crash causes a crash when written to.
Rob Pike806d00f2009-01-26 09:28:07 -080012package log
13
14import (
15 "fmt";
16 "io";
Russ Cox918afd942009-05-08 15:21:41 -070017 "runtime";
Rob Pike806d00f2009-01-26 09:28:07 -080018 "os";
19 "time";
20)
21
Rob Pike26cb4df2009-03-04 22:46:44 -080022// These flags define the properties of the Logger and the output they produce.
Rob Pike806d00f2009-01-26 09:28:07 -080023const (
Rob Pike646b3b52009-01-26 15:34:16 -080024 // Flags
Russ Coxc2ec9582009-10-06 19:41:51 -070025 Lok = iota;
Rob Pike806d00f2009-01-26 09:28:07 -080026 Lexit; // terminate execution when written
27 Lcrash; // crash (panic) when written
Rob Pike646b3b52009-01-26 15:34:16 -080028 // Bits or'ed together to control what's printed. There is no control over the
29 // order they appear (the order listed here) or the format they present (as
30 // described in the comments). A colon appears after these items:
31 // 2009/0123 01:23:23.123123 /a/b/c/d.go:23: message
Robert Griesemer3bb00322009-11-09 21:23:52 -080032 Ldate = 1 << iota; // the date: 2009/0123
Russ Coxc2ec9582009-10-06 19:41:51 -070033 Ltime; // the time: 01:23:23
Rob Pike646b3b52009-01-26 15:34:16 -080034 Lmicroseconds; // microsecond resolution: 01:23:23.123123. assumes Ltime.
35 Llongfile; // full file name and line number: /a/b/c/d.go:23
36 Lshortfile; // final file name element and line number: d.go:23. overrides Llongfile
Russ Coxc2ec9582009-10-06 19:41:51 -070037 lAllBits = Ldate | Ltime | Lmicroseconds | Llongfile | Lshortfile;
Rob Pike806d00f2009-01-26 09:28:07 -080038)
39
Rob Pike26cb4df2009-03-04 22:46:44 -080040// Logger represents an active logging object.
Rob Pike806d00f2009-01-26 09:28:07 -080041type Logger struct {
Rob Pikec8b47c62009-05-08 11:22:57 -070042 out0 io.Writer; // first destination for output
43 out1 io.Writer; // second destination for output; may be nil
Russ Coxc2ec9582009-10-06 19:41:51 -070044 prefix string; // prefix to write at beginning of each line
45 flag int; // properties
Rob Pike806d00f2009-01-26 09:28:07 -080046}
47
Rob Pikec54699c2009-05-15 17:22:30 -070048// New creates a new Logger. The out0 and out1 variables set the
Rob Pike26cb4df2009-03-04 22:46:44 -080049// destinations to which log data will be written; out1 may be nil.
50// The prefix appears at the beginning of each generated log line.
51// The flag argument defines the logging properties.
Rob Pikec54699c2009-05-15 17:22:30 -070052func New(out0, out1 io.Writer, prefix string, flag int) *Logger {
Robert Griesemer40621d52009-11-09 12:07:39 -080053 return &Logger{out0, out1, prefix, flag}
Rob Pike806d00f2009-01-26 09:28:07 -080054}
55
56var (
Russ Coxc2ec9582009-10-06 19:41:51 -070057 stdout = New(os.Stdout, nil, "", Lok|Ldate|Ltime);
58 stderr = New(os.Stderr, nil, "", Lok|Ldate|Ltime);
59 exit = New(os.Stderr, nil, "", Lexit|Ldate|Ltime);
60 crash = New(os.Stderr, nil, "", Lcrash|Ldate|Ltime);
Rob Pike806d00f2009-01-26 09:28:07 -080061)
62
Russ Coxc2ec9582009-10-06 19:41:51 -070063var shortnames = make(map[string]string) // cache of short names to avoid allocation.
Rob Pike806d00f2009-01-26 09:28:07 -080064
Rob Pike646b3b52009-01-26 15:34:16 -080065// Cheap integer to fixed-width decimal ASCII. Use a negative width to avoid zero-padding
66func itoa(i int, wid int) string {
67 var u uint = uint(i);
68 if u == 0 && wid <= 1 {
Robert Griesemer40621d52009-11-09 12:07:39 -080069 return "0"
Rob Pike646b3b52009-01-26 15:34:16 -080070 }
71
72 // Assemble decimal in reverse order.
73 var b [32]byte;
74 bp := len(b);
75 for ; u > 0 || wid > 0; u /= 10 {
76 bp--;
77 wid--;
Robert Griesemer3bb00322009-11-09 21:23:52 -080078 b[bp] = byte(u%10) + '0';
Rob Pike646b3b52009-01-26 15:34:16 -080079 }
80
Russ Cox9ac44492009-11-20 11:45:05 -080081 return string(b[bp:]);
Rob Pike646b3b52009-01-26 15:34:16 -080082}
83
84func (l *Logger) formatHeader(ns int64, calldepth int) string {
85 h := l.prefix;
Robert Griesemer3bb00322009-11-09 21:23:52 -080086 if l.flag&(Ldate|Ltime|Lmicroseconds) != 0 {
87 t := time.SecondsToLocalTime(ns / 1e9);
88 if l.flag&(Ldate) != 0 {
Robert Griesemer40621d52009-11-09 12:07:39 -080089 h += itoa(int(t.Year), 4) + "/" + itoa(t.Month, 2) + "/" + itoa(t.Day, 2) + " "
Rob Pike646b3b52009-01-26 15:34:16 -080090 }
Robert Griesemer3bb00322009-11-09 21:23:52 -080091 if l.flag&(Ltime|Lmicroseconds) != 0 {
Rob Pike646b3b52009-01-26 15:34:16 -080092 h += itoa(t.Hour, 2) + ":" + itoa(t.Minute, 2) + ":" + itoa(t.Second, 2);
Robert Griesemer3bb00322009-11-09 21:23:52 -080093 if l.flag&Lmicroseconds != 0 {
Robert Griesemer40621d52009-11-09 12:07:39 -080094 h += "." + itoa(int(ns%1e9)/1e3, 6)
Rob Pike646b3b52009-01-26 15:34:16 -080095 }
96 h += " ";
97 }
98 }
Robert Griesemer3bb00322009-11-09 21:23:52 -080099 if l.flag&(Lshortfile|Llongfile) != 0 {
Russ Coxca6a0fe2009-09-15 09:41:59 -0700100 _, file, line, ok := runtime.Caller(calldepth);
Rob Pike646b3b52009-01-26 15:34:16 -0800101 if ok {
Robert Griesemer3bb00322009-11-09 21:23:52 -0800102 if l.flag&Lshortfile != 0 {
Rob Pike646b3b52009-01-26 15:34:16 -0800103 short, ok := shortnames[file];
104 if !ok {
105 short = file;
Robert Griesemer3bb00322009-11-09 21:23:52 -0800106 for i := len(file) - 1; i > 0; i-- {
Rob Pike646b3b52009-01-26 15:34:16 -0800107 if file[i] == '/' {
Russ Cox9ac44492009-11-20 11:45:05 -0800108 short = file[i+1:];
Rob Pike646b3b52009-01-26 15:34:16 -0800109 break;
110 }
111 }
112 shortnames[file] = short;
113 }
114 file = short;
115 }
116 } else {
117 file = "???";
118 line = 0;
119 }
120 h += file + ":" + itoa(line, -1) + ": ";
121 }
122 return h;
123}
124
Rob Pike26cb4df2009-03-04 22:46:44 -0800125// Output writes the output for a logging event. The string s contains the text to print after
126// the time stamp; calldepth is used to recover the PC. It is provided for generality, although
127// at the moment on all pre-defined paths it will be 2.
Rob Pike646b3b52009-01-26 15:34:16 -0800128func (l *Logger) Output(calldepth int, s string) {
Rob Pike806d00f2009-01-26 09:28:07 -0800129 now := time.Nanoseconds(); // get this early.
130 newline := "\n";
131 if len(s) > 0 && s[len(s)-1] == '\n' {
Robert Griesemer40621d52009-11-09 12:07:39 -0800132 newline = ""
Rob Pike806d00f2009-01-26 09:28:07 -0800133 }
Robert Griesemer3bb00322009-11-09 21:23:52 -0800134 s = l.formatHeader(now, calldepth+1) + s + newline;
Rob Pike806d00f2009-01-26 09:28:07 -0800135 io.WriteString(l.out0, s);
136 if l.out1 != nil {
Robert Griesemer40621d52009-11-09 12:07:39 -0800137 io.WriteString(l.out1, s)
Rob Pike806d00f2009-01-26 09:28:07 -0800138 }
Rob Pike646b3b52009-01-26 15:34:16 -0800139 switch l.flag & ^lAllBits {
Rob Pike806d00f2009-01-26 09:28:07 -0800140 case Lcrash:
Robert Griesemer40621d52009-11-09 12:07:39 -0800141 panic("log: fatal error")
Rob Pike806d00f2009-01-26 09:28:07 -0800142 case Lexit:
Robert Griesemer40621d52009-11-09 12:07:39 -0800143 os.Exit(1)
Rob Pike806d00f2009-01-26 09:28:07 -0800144 }
145}
146
Rob Pike26cb4df2009-03-04 22:46:44 -0800147// Logf is analogous to Printf() for a Logger.
Robert Griesemer368f8cb2009-11-06 14:24:38 -0800148func (l *Logger) Logf(format string, v ...) { l.Output(2, fmt.Sprintf(format, v)) }
Rob Pike806d00f2009-01-26 09:28:07 -0800149
David Symonds6ea866c2009-11-06 18:43:57 -0800150// Log is analogous to Print() for a Logger.
Robert Griesemer368f8cb2009-11-06 14:24:38 -0800151func (l *Logger) Log(v ...) { l.Output(2, fmt.Sprintln(v)) }
Rob Pike806d00f2009-01-26 09:28:07 -0800152
Rob Pike26cb4df2009-03-04 22:46:44 -0800153// Stdout is a helper function for easy logging to stdout. It is analogous to Print().
Robert Griesemer368f8cb2009-11-06 14:24:38 -0800154func Stdout(v ...) { stdout.Output(2, fmt.Sprint(v)) }
Rob Pike806d00f2009-01-26 09:28:07 -0800155
Rob Pike26cb4df2009-03-04 22:46:44 -0800156// Stderr is a helper function for easy logging to stderr. It is analogous to Fprint(os.Stderr).
Robert Griesemer368f8cb2009-11-06 14:24:38 -0800157func Stderr(v ...) { stderr.Output(2, fmt.Sprintln(v)) }
Rob Pike806d00f2009-01-26 09:28:07 -0800158
Rob Pike26cb4df2009-03-04 22:46:44 -0800159// Stdoutf is a helper functions for easy formatted logging to stdout. It is analogous to Printf().
Robert Griesemer368f8cb2009-11-06 14:24:38 -0800160func Stdoutf(format string, v ...) { stdout.Output(2, fmt.Sprintf(format, v)) }
Rob Pike806d00f2009-01-26 09:28:07 -0800161
Rob Pike26cb4df2009-03-04 22:46:44 -0800162// Stderrf is a helper function for easy formatted logging to stderr. It is analogous to Fprintf(os.Stderr).
Robert Griesemer368f8cb2009-11-06 14:24:38 -0800163func Stderrf(format string, v ...) { stderr.Output(2, fmt.Sprintf(format, v)) }
Rob Pike806d00f2009-01-26 09:28:07 -0800164
Russ Cox918afd942009-05-08 15:21:41 -0700165// Exit is equivalent to Stderr() followed by a call to os.Exit(1).
Robert Griesemer368f8cb2009-11-06 14:24:38 -0800166func Exit(v ...) { exit.Output(2, fmt.Sprintln(v)) }
Rob Pike806d00f2009-01-26 09:28:07 -0800167
Russ Cox918afd942009-05-08 15:21:41 -0700168// Exitf is equivalent to Stderrf() followed by a call to os.Exit(1).
Robert Griesemer368f8cb2009-11-06 14:24:38 -0800169func Exitf(format string, v ...) { exit.Output(2, fmt.Sprintf(format, v)) }
Rob Pike806d00f2009-01-26 09:28:07 -0800170
Kai Backman8149a8c2009-08-13 15:14:41 -0700171// Crash is equivalent to Stderr() followed by a call to panic().
Robert Griesemer368f8cb2009-11-06 14:24:38 -0800172func Crash(v ...) { crash.Output(2, fmt.Sprintln(v)) }
Rob Pike806d00f2009-01-26 09:28:07 -0800173
Rob Pike26cb4df2009-03-04 22:46:44 -0800174// Crashf is equivalent to Stderrf() followed by a call to panic().
Robert Griesemer368f8cb2009-11-06 14:24:38 -0800175func Crashf(format string, v ...) { crash.Output(2, fmt.Sprintf(format, v)) }