log: support multiple log entries under one path scope

fix issue #1044

Signed-off-by: Tw <tw19881113@gmail.com>
This commit is contained in:
Tw 2016-08-22 15:40:14 +08:00
parent 70cbfdc585
commit 5e0f4083c4
4 changed files with 191 additions and 122 deletions

View file

@ -21,7 +21,7 @@ func init() {
// Logger is a basic request logging middleware. // Logger is a basic request logging middleware.
type Logger struct { type Logger struct {
Next httpserver.Handler Next httpserver.Handler
Rules []Rule Rules []*Rule
ErrorFunc func(http.ResponseWriter, *http.Request, int) // failover error handler ErrorFunc func(http.ResponseWriter, *http.Request, int) // failover error handler
} }
@ -52,8 +52,10 @@ func (l Logger) ServeHTTP(w http.ResponseWriter, r *http.Request) (int, error) {
status = 0 status = 0
} }
// Write log entry // Write log entries
rule.Log.Println(rep.Replace(rule.Format)) for _, e := range rule.Entries {
e.Log.Println(rep.Replace(e.Format))
}
return status, err return status, err
} }
@ -61,9 +63,8 @@ func (l Logger) ServeHTTP(w http.ResponseWriter, r *http.Request) (int, error) {
return l.Next.ServeHTTP(w, r) return l.Next.ServeHTTP(w, r)
} }
// Rule configures the logging middleware. // Entry represents a log entry under a path scope
type Rule struct { type Entry struct {
PathScope string
OutputFile string OutputFile string
Format string Format string
Log *log.Logger Log *log.Logger
@ -71,6 +72,12 @@ type Rule struct {
file *os.File // if logging to a file that needs to be closed file *os.File // if logging to a file that needs to be closed
} }
// Rule configures the logging middleware.
type Rule struct {
PathScope string
Entries []*Entry
}
const ( const (
// DefaultLogFilename is the default log filename. // DefaultLogFilename is the default log filename.
DefaultLogFilename = "access.log" DefaultLogFilename = "access.log"

View file

@ -26,12 +26,14 @@ func TestLoggedStatus(t *testing.T) {
var next erroringMiddleware var next erroringMiddleware
rule := Rule{ rule := Rule{
PathScope: "/", PathScope: "/",
Format: DefaultLogFormat + " {testval}", Entries: []*Entry{{
Log: log.New(&f, "", 0), Format: DefaultLogFormat + " {testval}",
Log: log.New(&f, "", 0),
}},
} }
logger := Logger{ logger := Logger{
Rules: []Rule{rule}, Rules: []*Rule{&rule},
Next: next, Next: next,
} }
@ -65,10 +67,12 @@ func TestLoggedStatus(t *testing.T) {
func TestLogRequestBody(t *testing.T) { func TestLogRequestBody(t *testing.T) {
var got bytes.Buffer var got bytes.Buffer
logger := Logger{ logger := Logger{
Rules: []Rule{{ Rules: []*Rule{{
PathScope: "/", PathScope: "/",
Format: "{request_body}", Entries: []*Entry{{
Log: log.New(&got, "", 0), Format: "{request_body}",
Log: log.New(&got, "", 0),
}},
}}, }},
Next: httpserver.HandlerFunc(func(w http.ResponseWriter, r *http.Request) (int, error) { Next: httpserver.HandlerFunc(func(w http.ResponseWriter, r *http.Request) (int, error) {
// drain up body // drain up body

View file

@ -20,39 +20,41 @@ func setup(c *caddy.Controller) error {
// Open the log files for writing when the server starts // Open the log files for writing when the server starts
c.OnStartup(func() error { c.OnStartup(func() error {
for i := 0; i < len(rules); i++ { for _, rule := range rules {
var err error for _, entry := range rule.Entries {
var writer io.Writer var err error
var writer io.Writer
if rules[i].OutputFile == "stdout" { if entry.OutputFile == "stdout" {
writer = os.Stdout writer = os.Stdout
} else if rules[i].OutputFile == "stderr" { } else if entry.OutputFile == "stderr" {
writer = os.Stderr writer = os.Stderr
} else if rules[i].OutputFile == "syslog" { } else if entry.OutputFile == "syslog" {
writer, err = gsyslog.NewLogger(gsyslog.LOG_INFO, "LOCAL0", "caddy") writer, err = gsyslog.NewLogger(gsyslog.LOG_INFO, "LOCAL0", "caddy")
if err != nil { if err != nil {
return err return err
} }
} else {
err := os.MkdirAll(filepath.Dir(rules[i].OutputFile), 0744)
if err != nil {
return err
}
file, err := os.OpenFile(rules[i].OutputFile, os.O_RDWR|os.O_CREATE|os.O_APPEND, 0644)
if err != nil {
return err
}
if rules[i].Roller != nil {
file.Close()
rules[i].Roller.Filename = rules[i].OutputFile
writer = rules[i].Roller.GetLogWriter()
} else { } else {
rules[i].file = file err := os.MkdirAll(filepath.Dir(entry.OutputFile), 0744)
writer = file if err != nil {
return err
}
file, err := os.OpenFile(entry.OutputFile, os.O_RDWR|os.O_CREATE|os.O_APPEND, 0644)
if err != nil {
return err
}
if entry.Roller != nil {
file.Close()
entry.Roller.Filename = entry.OutputFile
writer = entry.Roller.GetLogWriter()
} else {
entry.file = file
writer = file
}
} }
}
rules[i].Log = log.New(writer, "", 0) entry.Log = log.New(writer, "", 0)
}
} }
return nil return nil
@ -61,8 +63,10 @@ func setup(c *caddy.Controller) error {
// When server stops, close any open log files // When server stops, close any open log files
c.OnShutdown(func() error { c.OnShutdown(func() error {
for _, rule := range rules { for _, rule := range rules {
if rule.file != nil { for _, entry := range rule.Entries {
rule.file.Close() if entry.file != nil {
entry.file.Close()
}
} }
} }
return nil return nil
@ -75,8 +79,8 @@ func setup(c *caddy.Controller) error {
return nil return nil
} }
func logParse(c *caddy.Controller) ([]Rule, error) { func logParse(c *caddy.Controller) ([]*Rule, error) {
var rules []Rule var rules []*Rule
for c.Next() { for c.Next() {
args := c.RemainingArgs() args := c.RemainingArgs()
@ -103,16 +107,14 @@ func logParse(c *caddy.Controller) ([]Rule, error) {
} }
if len(args) == 0 { if len(args) == 0 {
// Nothing specified; use defaults // Nothing specified; use defaults
rules = append(rules, Rule{ rules = appendEntry(rules, "/", &Entry{
PathScope: "/",
OutputFile: DefaultLogFilename, OutputFile: DefaultLogFilename,
Format: DefaultLogFormat, Format: DefaultLogFormat,
Roller: logRoller, Roller: logRoller,
}) })
} else if len(args) == 1 { } else if len(args) == 1 {
// Only an output file specified // Only an output file specified
rules = append(rules, Rule{ rules = appendEntry(rules, "/", &Entry{
PathScope: "/",
OutputFile: args[0], OutputFile: args[0],
Format: DefaultLogFormat, Format: DefaultLogFormat,
Roller: logRoller, Roller: logRoller,
@ -133,8 +135,7 @@ func logParse(c *caddy.Controller) ([]Rule, error) {
} }
} }
rules = append(rules, Rule{ rules = appendEntry(rules, args[0], &Entry{
PathScope: args[0],
OutputFile: args[1], OutputFile: args[1],
Format: format, Format: format,
Roller: logRoller, Roller: logRoller,
@ -144,3 +145,19 @@ func logParse(c *caddy.Controller) ([]Rule, error) {
return rules, nil return rules, nil
} }
func appendEntry(rules []*Rule, pathScope string, entry *Entry) []*Rule {
for _, rule := range rules {
if rule.PathScope == pathScope {
rule.Entries = append(rule.Entries, entry)
return rules
}
}
rules = append(rules, &Rule{
PathScope: pathScope,
Entries: []*Entry{entry},
})
return rules
}

View file

@ -29,14 +29,15 @@ func TestSetup(t *testing.T) {
if myHandler.Rules[0].PathScope != "/" { if myHandler.Rules[0].PathScope != "/" {
t.Errorf("Expected / as the default PathScope") t.Errorf("Expected / as the default PathScope")
} }
if myHandler.Rules[0].OutputFile != DefaultLogFilename { if myHandler.Rules[0].Entries[0].OutputFile != DefaultLogFilename {
t.Errorf("Expected %s as the default OutputFile", DefaultLogFilename) t.Errorf("Expected %s as the default OutputFile", DefaultLogFilename)
} }
if myHandler.Rules[0].Format != DefaultLogFormat { if myHandler.Rules[0].Entries[0].Format != DefaultLogFormat {
t.Errorf("Expected %s as the default Log Format", DefaultLogFormat) t.Errorf("Expected %s as the default Log Format", DefaultLogFormat)
} }
if myHandler.Rules[0].Roller != nil { if myHandler.Rules[0].Entries[0].Roller != nil {
t.Errorf("Expected Roller to be nil, got: %v", *myHandler.Rules[0].Roller) t.Errorf("Expected Roller to be nil, got: %v",
*myHandler.Rules[0].Entries[0].Roller)
} }
if !httpserver.SameNext(myHandler.Next, httpserver.EmptyNext) { if !httpserver.SameNext(myHandler.Next, httpserver.EmptyNext) {
t.Error("'Next' field of handler was not set properly") t.Error("'Next' field of handler was not set properly")
@ -51,65 +52,98 @@ func TestLogParse(t *testing.T) {
expectedLogRules []Rule expectedLogRules []Rule
}{ }{
{`log`, false, []Rule{{ {`log`, false, []Rule{{
PathScope: "/", PathScope: "/",
OutputFile: DefaultLogFilename, Entries: []*Entry{{
Format: DefaultLogFormat, OutputFile: DefaultLogFilename,
Format: DefaultLogFormat,
}},
}}}, }}},
{`log log.txt`, false, []Rule{{ {`log log.txt`, false, []Rule{{
PathScope: "/", PathScope: "/",
OutputFile: "log.txt", Entries: []*Entry{{
Format: DefaultLogFormat, OutputFile: "log.txt",
Format: DefaultLogFormat,
}},
}}}, }}},
{`log /api log.txt`, false, []Rule{{ {`log /api log.txt`, false, []Rule{{
PathScope: "/api", PathScope: "/api",
OutputFile: "log.txt", Entries: []*Entry{{
Format: DefaultLogFormat, OutputFile: "log.txt",
Format: DefaultLogFormat,
}},
}}}, }}},
{`log /serve stdout`, false, []Rule{{ {`log /serve stdout`, false, []Rule{{
PathScope: "/serve", PathScope: "/serve",
OutputFile: "stdout", Entries: []*Entry{{
Format: DefaultLogFormat, OutputFile: "stdout",
Format: DefaultLogFormat,
}},
}}}, }}},
{`log /myapi log.txt {common}`, false, []Rule{{ {`log /myapi log.txt {common}`, false, []Rule{{
PathScope: "/myapi", PathScope: "/myapi",
OutputFile: "log.txt", Entries: []*Entry{{
Format: CommonLogFormat, OutputFile: "log.txt",
Format: CommonLogFormat,
}},
}}}, }}},
{`log /test accesslog.txt {combined}`, false, []Rule{{ {`log /test accesslog.txt {combined}`, false, []Rule{{
PathScope: "/test", PathScope: "/test",
OutputFile: "accesslog.txt", Entries: []*Entry{{
Format: CombinedLogFormat, OutputFile: "accesslog.txt",
Format: CombinedLogFormat,
}},
}}}, }}},
{`log /api1 log.txt {`log /api1 log.txt
log /api2 accesslog.txt {combined}`, false, []Rule{{ log /api2 accesslog.txt {combined}`, false, []Rule{{
PathScope: "/api1", PathScope: "/api1",
OutputFile: "log.txt", Entries: []*Entry{{
Format: DefaultLogFormat, OutputFile: "log.txt",
Format: DefaultLogFormat,
}},
}, { }, {
PathScope: "/api2", PathScope: "/api2",
OutputFile: "accesslog.txt", Entries: []*Entry{{
Format: CombinedLogFormat, OutputFile: "accesslog.txt",
Format: CombinedLogFormat,
}},
}}}, }}},
{`log /api3 stdout {host} {`log /api3 stdout {host}
log /api4 log.txt {when}`, false, []Rule{{ log /api4 log.txt {when}`, false, []Rule{{
PathScope: "/api3", PathScope: "/api3",
OutputFile: "stdout", Entries: []*Entry{{
Format: "{host}", OutputFile: "stdout",
Format: "{host}",
}},
}, { }, {
PathScope: "/api4", PathScope: "/api4",
OutputFile: "log.txt", Entries: []*Entry{{
Format: "{when}", OutputFile: "log.txt",
Format: "{when}",
}},
}}}, }}},
{`log access.log { rotate { size 2 age 10 keep 3 } }`, false, []Rule{{ {`log access.log { rotate { size 2 age 10 keep 3 } }`, false, []Rule{{
PathScope: "/", PathScope: "/",
OutputFile: "access.log", Entries: []*Entry{{
Format: DefaultLogFormat, OutputFile: "access.log",
Roller: &httpserver.LogRoller{ Format: DefaultLogFormat,
MaxSize: 2, Roller: &httpserver.LogRoller{
MaxAge: 10, MaxSize: 2,
MaxBackups: 3, MaxAge: 10,
LocalTime: true, MaxBackups: 3,
}, LocalTime: true,
},
}},
}}},
{`log / stdout {host}
log / log.txt {when}`, false, []Rule{{
PathScope: "/",
Entries: []*Entry{{
OutputFile: "stdout",
Format: "{host}",
}, {
OutputFile: "log.txt",
Format: "{when}",
}},
}}}, }}},
} }
for i, test := range tests { for i, test := range tests {
@ -132,39 +166,46 @@ func TestLogParse(t *testing.T) {
i, j, test.expectedLogRules[j].PathScope, actualLogRule.PathScope) i, j, test.expectedLogRules[j].PathScope, actualLogRule.PathScope)
} }
if actualLogRule.OutputFile != test.expectedLogRules[j].OutputFile { if got, expect := len(actualLogRule.Entries), len(test.expectedLogRules[j].Entries); got != expect {
t.Errorf("Test %d expected %dth LogRule OutputFile to be %s , but got %s", t.Fatalf("Test %d expected %dth LogRule with %d no of Log entries, but got %d ",
i, j, test.expectedLogRules[j].OutputFile, actualLogRule.OutputFile) i, j, expect, got)
} }
if actualLogRule.Format != test.expectedLogRules[j].Format { for k, actualEntry := range actualLogRule.Entries {
t.Errorf("Test %d expected %dth LogRule Format to be %s , but got %s", if actualEntry.OutputFile != test.expectedLogRules[j].Entries[k].OutputFile {
i, j, test.expectedLogRules[j].Format, actualLogRule.Format) t.Errorf("Test %d expected %dth LogRule OutputFile to be %s , but got %s",
} i, j, test.expectedLogRules[j].Entries[k].OutputFile, actualEntry.OutputFile)
if actualLogRule.Roller != nil && test.expectedLogRules[j].Roller == nil || actualLogRule.Roller == nil && test.expectedLogRules[j].Roller != nil {
t.Fatalf("Test %d expected %dth LogRule Roller to be %v, but got %v",
i, j, test.expectedLogRules[j].Roller, actualLogRule.Roller)
}
if actualLogRule.Roller != nil && test.expectedLogRules[j].Roller != nil {
if actualLogRule.Roller.Filename != test.expectedLogRules[j].Roller.Filename {
t.Fatalf("Test %d expected %dth LogRule Roller Filename to be %s, but got %s",
i, j, test.expectedLogRules[j].Roller.Filename, actualLogRule.Roller.Filename)
} }
if actualLogRule.Roller.MaxAge != test.expectedLogRules[j].Roller.MaxAge {
t.Fatalf("Test %d expected %dth LogRule Roller MaxAge to be %d, but got %d", if actualEntry.Format != test.expectedLogRules[j].Entries[k].Format {
i, j, test.expectedLogRules[j].Roller.MaxAge, actualLogRule.Roller.MaxAge) t.Errorf("Test %d expected %dth LogRule Format to be %s , but got %s",
i, j, test.expectedLogRules[j].Entries[k].Format, actualEntry.Format)
} }
if actualLogRule.Roller.MaxBackups != test.expectedLogRules[j].Roller.MaxBackups { if actualEntry.Roller != nil && test.expectedLogRules[j].Entries[k].Roller == nil || actualEntry.Roller == nil && test.expectedLogRules[j].Entries[k].Roller != nil {
t.Fatalf("Test %d expected %dth LogRule Roller MaxBackups to be %d, but got %d", t.Fatalf("Test %d expected %dth LogRule Roller to be %v, but got %v",
i, j, test.expectedLogRules[j].Roller.MaxBackups, actualLogRule.Roller.MaxBackups) i, j, test.expectedLogRules[j].Entries[k].Roller, actualEntry.Roller)
} }
if actualLogRule.Roller.MaxSize != test.expectedLogRules[j].Roller.MaxSize { if actualEntry.Roller != nil && test.expectedLogRules[j].Entries[k].Roller != nil {
t.Fatalf("Test %d expected %dth LogRule Roller MaxSize to be %d, but got %d", if actualEntry.Roller.Filename != test.expectedLogRules[j].Entries[k].Roller.Filename {
i, j, test.expectedLogRules[j].Roller.MaxSize, actualLogRule.Roller.MaxSize) t.Fatalf("Test %d expected %dth LogRule Roller Filename to be %s, but got %s",
} i, j, test.expectedLogRules[j].Entries[k].Roller.Filename, actualEntry.Roller.Filename)
if actualLogRule.Roller.LocalTime != test.expectedLogRules[j].Roller.LocalTime { }
t.Fatalf("Test %d expected %dth LogRule Roller LocalTime to be %t, but got %t", if actualEntry.Roller.MaxAge != test.expectedLogRules[j].Entries[k].Roller.MaxAge {
i, j, test.expectedLogRules[j].Roller.LocalTime, actualLogRule.Roller.LocalTime) t.Fatalf("Test %d expected %dth LogRule Roller MaxAge to be %d, but got %d",
i, j, test.expectedLogRules[j].Entries[k].Roller.MaxAge, actualEntry.Roller.MaxAge)
}
if actualEntry.Roller.MaxBackups != test.expectedLogRules[j].Entries[k].Roller.MaxBackups {
t.Fatalf("Test %d expected %dth LogRule Roller MaxBackups to be %d, but got %d",
i, j, test.expectedLogRules[j].Entries[k].Roller.MaxBackups, actualEntry.Roller.MaxBackups)
}
if actualEntry.Roller.MaxSize != test.expectedLogRules[j].Entries[k].Roller.MaxSize {
t.Fatalf("Test %d expected %dth LogRule Roller MaxSize to be %d, but got %d",
i, j, test.expectedLogRules[j].Entries[k].Roller.MaxSize, actualEntry.Roller.MaxSize)
}
if actualEntry.Roller.LocalTime != test.expectedLogRules[j].Entries[k].Roller.LocalTime {
t.Fatalf("Test %d expected %dth LogRule Roller LocalTime to be %t, but got %t",
i, j, test.expectedLogRules[j].Entries[k].Roller.LocalTime, actualEntry.Roller.LocalTime)
}
} }
} }
} }