mirror of
https://github.com/golang/go
synced 2024-11-23 07:00:05 -07:00
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
This commit is contained in:
parent
908e1b5ea1
commit
4228eb7915
@ -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 (
|
|||||||
"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 (
|
|||||||
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 {
|
|||||||
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()
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return &Writer{priority, prefix, conn}, err
|
if err != nil {
|
||||||
|
return nil, 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 @@
|
|||||||
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 {
|
|||||||
}
|
}
|
||||||
|
|
||||||
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) {
|
|||||||
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) {
|
|||||||
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) {
|
|||||||
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, ×tamp, &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) {
|
|||||||
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, ×tamp, &parsedHostname, &pid); n != 3 ||
|
||||||
|
err != nil || hostname != parsedHostname {
|
||||||
|
t.Fatalf("s.Info() = '%q', didn't match '%q'", rcvd, test.exp)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user