Commit 4228eb79 authored by John Graham-Cumming's avatar John Graham-Cumming Committed by Russ Cox

log/syslog: correct message format

The syslog implementation was not correctly implementing the
traditional syslog format because it had a confused notion of
'priority'.  syslog priority is not a single number but is, in
fact, the combination of a facility number and a severity. The
previous Go syslog implementation had a single Priority that
appeared to be the syslog severity and no way of setting the
facility.  That meant that all syslog messages from Go
programs appeared to have a facility of 0 (LOG_KERN) which
meant they all appeared to come from the kernel.

Also, the 'prefix' was in fact the syslog tag (changed the
internal name for clarity as the term tag is more widely used)
and the timestamp and hostname values were missing from
messages.

With this change syslog messages are generated in the correct
format with facility and severity combined into a priority,
the timestamp in RFC3339 format, the hostname, the tag (with
the PID in [] appened) and the message.

The format is now:

   <PRI>1 TIMESTAMP HOSTNAME TAG[PID]: MSG

The TIMESTAMP, HOSTNAME and PID fields are filled in
automatically by the package. The TAG and the MSG are supplied
by the user. This is what rsyslogd calls TraditionalFormat and
should be compatible with multiple systems.

R=rsc, jgc, 0xjnml, mikioh.mikioh, bradfitz
CC=golang-dev
https://golang.org/cl/6782118
parent 908e1b5e
...@@ -4,9 +4,9 @@ ...@@ -4,9 +4,9 @@
// +build !windows,!plan9 // +build !windows,!plan9
// Package syslog provides a simple interface to the system log service. It // Package syslog provides a simple interface to the system log
// can send messages to the syslog daemon using UNIX domain sockets, UDP, or // service. It can send messages to the syslog daemon using UNIX
// TCP connections. // domain sockets, UDP, or TCP connections.
package syslog package syslog
import ( import (
...@@ -15,11 +15,21 @@ import ( ...@@ -15,11 +15,21 @@ import (
"log" "log"
"net" "net"
"os" "os"
"time"
) )
// The Priority is a combination of the syslog facility and
// severity. For example, LOG_ALERT | LOG_FTP sends an alert severity
// message from the FTP facility. The default severity is LOG_EMERG;
// the default facility is LOG_KERN.
type Priority int type Priority int
const severityMask = 0x07
const facilityMask = 0xf8
const ( const (
// Severity.
// From /usr/include/sys/syslog.h. // From /usr/include/sys/syslog.h.
// These are the same on Linux, BSD, and OS X. // These are the same on Linux, BSD, and OS X.
LOG_EMERG Priority = iota LOG_EMERG Priority = iota
...@@ -32,16 +42,47 @@ const ( ...@@ -32,16 +42,47 @@ const (
LOG_DEBUG LOG_DEBUG
) )
const (
// Facility.
// From /usr/include/sys/syslog.h.
// These are the same up to LOG_FTP on Linux, BSD, and OS X.
LOG_KERN Priority = iota << 3
LOG_USER
LOG_MAIL
LOG_DAEMON
LOG_AUTH
LOG_SYSLOG
LOG_LPR
LOG_NEWS
LOG_UUCP
LOG_CRON
LOG_AUTHPRIV
LOG_FTP
_ // unused
_ // unused
_ // unused
_ // unused
LOG_LOCAL0
LOG_LOCAL1
LOG_LOCAL2
LOG_LOCAL3
LOG_LOCAL4
LOG_LOCAL5
LOG_LOCAL6
LOG_LOCAL7
)
// A Writer is a connection to a syslog server. // A Writer is a connection to a syslog server.
type Writer struct { type Writer struct {
priority Priority priority Priority
prefix string tag string
hostname string
conn serverConn conn serverConn
} }
type serverConn interface { type serverConn interface {
writeBytes(p Priority, prefix string, b []byte) (int, error) writeString(p Priority, hostname, tag, s string) (int, error)
writeString(p Priority, prefix string, s string) (int, error)
close() error close() error
} }
...@@ -49,116 +90,130 @@ type netConn struct { ...@@ -49,116 +90,130 @@ type netConn struct {
conn net.Conn conn net.Conn
} }
// New establishes a new connection to the system log daemon. // New establishes a new connection to the system log daemon. Each
// Each write to the returned writer sends a log message with // write to the returned writer sends a log message with the given
// the given priority and prefix. // priority and prefix.
func New(priority Priority, prefix string) (w *Writer, err error) { func New(priority Priority, tag string) (w *Writer, err error) {
return Dial("", "", priority, prefix) return Dial("", "", priority, tag)
} }
// Dial establishes a connection to a log daemon by connecting // Dial establishes a connection to a log daemon by connecting to
// to address raddr on the network net. // address raddr on the network net. Each write to the returned
// Each write to the returned writer sends a log message with // writer sends a log message with the given facility, severity and
// the given priority and prefix. // tag.
func Dial(network, raddr string, priority Priority, prefix string) (w *Writer, err error) { func Dial(network, raddr string, priority Priority, tag string) (w *Writer, err error) {
if prefix == "" { if priority < 0 || priority > LOG_LOCAL7|LOG_DEBUG {
prefix = os.Args[0] return nil, errors.New("log/syslog: invalid priority")
} }
if tag == "" {
tag = os.Args[0]
}
hostname, _ := os.Hostname()
var conn serverConn var conn serverConn
if network == "" { if network == "" {
conn, err = unixSyslog() conn, err = unixSyslog()
if hostname == "" {
hostname = "localhost"
}
} else { } else {
var c net.Conn var c net.Conn
c, err = net.Dial(network, raddr) c, err = net.Dial(network, raddr)
conn = netConn{c} conn = netConn{c}
if hostname == "" {
hostname = c.LocalAddr().String()
}
}
if err != nil {
return nil, err
} }
return &Writer{priority, prefix, conn}, err
return &Writer{priority: priority, tag: tag, hostname: hostname, conn: conn}, nil
} }
// Write sends a log message to the syslog daemon. // Write sends a log message to the syslog daemon.
func (w *Writer) Write(b []byte) (int, error) { func (w *Writer) Write(b []byte) (int, error) {
if w.priority > LOG_DEBUG || w.priority < LOG_EMERG { return w.writeString(w.priority, string(b))
return 0, errors.New("log/syslog: invalid priority")
}
return w.conn.writeBytes(w.priority, w.prefix, b)
}
func (w *Writer) writeString(p Priority, s string) (int, error) {
return w.conn.writeString(p, w.prefix, s)
} }
func (w *Writer) Close() error { return w.conn.close() } func (w *Writer) Close() error { return w.conn.close() }
// Emerg logs a message using the LOG_EMERG priority. // Emerg logs a message with severity LOG_EMERG, ignoring the severity
// passed to New.
func (w *Writer) Emerg(m string) (err error) { func (w *Writer) Emerg(m string) (err error) {
_, err = w.writeString(LOG_EMERG, m) _, err = w.writeString(LOG_EMERG, m)
return err return err
} }
// Alert logs a message using the LOG_ALERT priority. // Alert logs a message with severity LOG_ALERT, ignoring the severity
// passed to New.
func (w *Writer) Alert(m string) (err error) { func (w *Writer) Alert(m string) (err error) {
_, err = w.writeString(LOG_ALERT, m) _, err = w.writeString(LOG_ALERT, m)
return err return err
} }
// Crit logs a message using the LOG_CRIT priority. // Crit logs a message with severity LOG_CRIT, ignoring the severity
// passed to New.
func (w *Writer) Crit(m string) (err error) { func (w *Writer) Crit(m string) (err error) {
_, err = w.writeString(LOG_CRIT, m) _, err = w.writeString(LOG_CRIT, m)
return err return err
} }
// Err logs a message using the LOG_ERR priority. // Err logs a message with severity LOG_ERR, ignoring the severity
// passed to New.
func (w *Writer) Err(m string) (err error) { func (w *Writer) Err(m string) (err error) {
_, err = w.writeString(LOG_ERR, m) _, err = w.writeString(LOG_ERR, m)
return err return err
} }
// Warning logs a message using the LOG_WARNING priority. // Wanring logs a message with severity LOG_WARNING, ignoring the
// severity passed to New.
func (w *Writer) Warning(m string) (err error) { func (w *Writer) Warning(m string) (err error) {
_, err = w.writeString(LOG_WARNING, m) _, err = w.writeString(LOG_WARNING, m)
return err return err
} }
// Notice logs a message using the LOG_NOTICE priority. // Notice logs a message with severity LOG_NOTICE, ignoring the
// severity passed to New.
func (w *Writer) Notice(m string) (err error) { func (w *Writer) Notice(m string) (err error) {
_, err = w.writeString(LOG_NOTICE, m) _, err = w.writeString(LOG_NOTICE, m)
return err return err
} }
// Info logs a message using the LOG_INFO priority. // Info logs a message with severity LOG_INFO, ignoring the severity
// passed to New.
func (w *Writer) Info(m string) (err error) { func (w *Writer) Info(m string) (err error) {
_, err = w.writeString(LOG_INFO, m) _, err = w.writeString(LOG_INFO, m)
return err return err
} }
// Debug logs a message using the LOG_DEBUG priority. // Debug logs a message with severity LOG_DEBUG, ignoring the severity
// passed to New.
func (w *Writer) Debug(m string) (err error) { func (w *Writer) Debug(m string) (err error) {
_, err = w.writeString(LOG_DEBUG, m) _, err = w.writeString(LOG_DEBUG, m)
return err return err
} }
func (n netConn) writeBytes(p Priority, prefix string, b []byte) (int, error) { func (w *Writer) writeString(p Priority, s string) (int, error) {
nl := "" return w.conn.writeString((w.priority&facilityMask)|(p&severityMask),
if len(b) == 0 || b[len(b)-1] != '\n' { w.hostname, w.tag, s)
nl = "\n"
}
_, err := fmt.Fprintf(n.conn, "<%d>%s: %s%s", p, prefix, b, nl)
if err != nil {
return 0, err
}
return len(b), nil
} }
func (n netConn) writeString(p Priority, prefix string, s string) (int, error) { // writeString: generates and writes a syslog formatted string. The
// format is as follows: <PRI>1 TIMESTAMP HOSTNAME TAG[PID]: MSG
func (n netConn) writeString(p Priority, hostname, tag, msg string) (int, error) {
nl := "" nl := ""
if len(s) == 0 || s[len(s)-1] != '\n' { if len(msg) == 0 || msg[len(msg)-1] != '\n' {
nl = "\n" nl = "\n"
} }
_, err := fmt.Fprintf(n.conn, "<%d>%s: %s%s", p, prefix, s, nl) timestamp := time.Now().Format(time.RFC3339)
if err != nil { if _, err := fmt.Fprintf(n.conn, "<%d>1 %s %s %s[%d]: %s%s", p, timestamp, hostname,
tag, os.Getpid(), msg, nl); err != nil {
return 0, err return 0, err
} }
return len(s), nil return len(msg), nil
} }
func (n netConn) close() error { func (n netConn) close() error {
......
...@@ -7,9 +7,11 @@ ...@@ -7,9 +7,11 @@
package syslog package syslog
import ( import (
"fmt"
"io" "io"
"log" "log"
"net" "net"
"os"
"testing" "testing"
"time" "time"
) )
...@@ -49,10 +51,14 @@ func skipNetTest(t *testing.T) bool { ...@@ -49,10 +51,14 @@ func skipNetTest(t *testing.T) bool {
} }
func TestNew(t *testing.T) { func TestNew(t *testing.T) {
if LOG_LOCAL7 != 23<<3 {
t.Fatalf("LOG_LOCAL7 has wrong value")
}
if skipNetTest(t) { if skipNetTest(t) {
return return
} }
s, err := New(LOG_INFO, "")
s, err := New(LOG_INFO|LOG_USER, "")
if err != nil { if err != nil {
t.Fatalf("New() failed: %s", err) t.Fatalf("New() failed: %s", err)
} }
...@@ -64,7 +70,7 @@ func TestNewLogger(t *testing.T) { ...@@ -64,7 +70,7 @@ func TestNewLogger(t *testing.T) {
if skipNetTest(t) { if skipNetTest(t) {
return return
} }
f, err := NewLogger(LOG_INFO, 0) f, err := NewLogger(LOG_USER|LOG_INFO, 0)
if f == nil { if f == nil {
t.Error(err) t.Error(err)
} }
...@@ -74,7 +80,15 @@ func TestDial(t *testing.T) { ...@@ -74,7 +80,15 @@ func TestDial(t *testing.T) {
if skipNetTest(t) { if skipNetTest(t) {
return return
} }
l, err := Dial("", "", LOG_ERR, "syslog_test") f, err := Dial("", "", (LOG_LOCAL7|LOG_DEBUG)+1, "syslog_test")
if f != nil {
t.Fatalf("Should have trapped bad priority")
}
f, err = Dial("", "", -1, "syslog_test")
if f != nil {
t.Fatalf("Should have trapped bad priority")
}
l, err := Dial("", "", LOG_USER|LOG_ERR, "syslog_test")
if err != nil { if err != nil {
t.Fatalf("Dial() failed: %s", err) t.Fatalf("Dial() failed: %s", err)
} }
...@@ -84,16 +98,23 @@ func TestDial(t *testing.T) { ...@@ -84,16 +98,23 @@ func TestDial(t *testing.T) {
func TestUDPDial(t *testing.T) { func TestUDPDial(t *testing.T) {
done := make(chan string) done := make(chan string)
startServer(done) startServer(done)
l, err := Dial("udp", serverAddr, LOG_INFO, "syslog_test") l, err := Dial("udp", serverAddr, LOG_USER|LOG_INFO, "syslog_test")
if err != nil { if err != nil {
t.Fatalf("syslog.Dial() failed: %s", err) t.Fatalf("syslog.Dial() failed: %s", err)
} }
msg := "udp test" msg := "udp test"
l.Info(msg) l.Info(msg)
expected := "<6>syslog_test: udp test\n" expected := fmt.Sprintf("<%d>1 ", LOG_USER+LOG_INFO) + "%s %s syslog_test[%d]: udp test\n"
rcvd := <-done rcvd := <-done
if rcvd != expected { var parsedHostname, timestamp string
t.Fatalf("s.Info() = '%q', but wanted '%q'", rcvd, expected) var pid int
if hostname, err := os.Hostname(); err != nil {
t.Fatalf("Error retrieving hostname")
} else {
if n, err := fmt.Sscanf(rcvd, expected, &timestamp, &parsedHostname, &pid); n != 3 ||
err != nil || hostname != parsedHostname {
t.Fatalf("s.Info() = '%q', didn't match '%q'", rcvd, expected)
}
} }
} }
...@@ -104,26 +125,34 @@ func TestWrite(t *testing.T) { ...@@ -104,26 +125,34 @@ func TestWrite(t *testing.T) {
msg string msg string
exp string exp string
}{ }{
{LOG_ERR, "syslog_test", "", "<3>syslog_test: \n"}, {LOG_USER | LOG_ERR, "syslog_test", "", "%s %s syslog_test[%d]: \n"},
{LOG_ERR, "syslog_test", "write test", "<3>syslog_test: write test\n"}, {LOG_USER | LOG_ERR, "syslog_test", "write test", "%s %s syslog_test[%d]: write test\n"},
// Write should not add \n if there already is one // Write should not add \n if there already is one
{LOG_ERR, "syslog_test", "write test 2\n", "<3>syslog_test: write test 2\n"}, {LOG_USER | LOG_ERR, "syslog_test", "write test 2\n", "%s %s syslog_test[%d]: write test 2\n"},
} }
for _, test := range tests { if hostname, err := os.Hostname(); err != nil {
done := make(chan string) t.Fatalf("Error retrieving hostname")
startServer(done) } else {
l, err := Dial("udp", serverAddr, test.pri, test.pre) for _, test := range tests {
if err != nil { done := make(chan string)
t.Fatalf("syslog.Dial() failed: %s", err) startServer(done)
} l, err := Dial("udp", serverAddr, test.pri, test.pre)
_, err = io.WriteString(l, test.msg) if err != nil {
if err != nil { t.Fatalf("syslog.Dial() failed: %s", err)
t.Fatalf("WriteString() failed: %s", err) }
} _, err = io.WriteString(l, test.msg)
rcvd := <-done if err != nil {
if rcvd != test.exp { t.Fatalf("WriteString() failed: %s", err)
t.Fatalf("s.Info() = '%q', but wanted '%q'", rcvd, test.exp) }
rcvd := <-done
test.exp = fmt.Sprintf("<%d>1 ", test.pri) + test.exp
var parsedHostname, timestamp string
var pid int
if n, err := fmt.Sscanf(rcvd, test.exp, &timestamp, &parsedHostname, &pid); n != 3 ||
err != nil || hostname != parsedHostname {
t.Fatalf("s.Info() = '%q', didn't match '%q'", rcvd, test.exp)
}
} }
} }
} }
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment