2020-09-12 11:34:59 +02:00
|
|
|
package accesslog
|
|
|
|
|
|
|
|
import (
|
2020-09-13 01:56:22 +02:00
|
|
|
"bytes"
|
2020-09-12 11:34:59 +02:00
|
|
|
"io"
|
|
|
|
"text/template"
|
|
|
|
)
|
|
|
|
|
|
|
|
// Template is a Formatter.
|
|
|
|
// It's used to print the Log in a text/template way.
|
|
|
|
// The caller has full control over the printable result;
|
|
|
|
// certain fields can be ignored, change the display order and e.t.c.
|
2020-09-13 01:56:22 +02:00
|
|
|
//
|
|
|
|
// For faster execution you can create a custom Formatter
|
|
|
|
// and compile your own quicktemplate: https://github.com/valyala/quicktemplate
|
|
|
|
//
|
|
|
|
// This one uses the standard text/template syntax.
|
2020-09-12 11:34:59 +02:00
|
|
|
type Template struct {
|
|
|
|
// Custom template source.
|
|
|
|
// Use this or `Tmpl/TmplName` fields.
|
|
|
|
Text string
|
|
|
|
// Custom template funcs to used when `Text` is not empty.
|
|
|
|
Funcs template.FuncMap
|
|
|
|
|
|
|
|
// Custom template to use, overrides the `Text` and `Funcs` fields.
|
|
|
|
Tmpl *template.Template
|
|
|
|
// If not empty then this named template/block renders the log line.
|
|
|
|
TmplName string
|
|
|
|
|
2020-09-13 01:56:22 +02:00
|
|
|
ac *AccessLog
|
2020-09-12 11:34:59 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// SetOutput creates the default template if missing
|
|
|
|
// when this formatter is registered.
|
|
|
|
func (f *Template) SetOutput(dest io.Writer) {
|
2020-09-13 01:56:22 +02:00
|
|
|
f.ac, _ = dest.(*AccessLog)
|
|
|
|
|
2020-09-12 11:34:59 +02:00
|
|
|
if f.Tmpl == nil {
|
|
|
|
tmpl := template.New("")
|
|
|
|
|
|
|
|
text := f.Text
|
|
|
|
if text != "" {
|
|
|
|
tmpl.Funcs(f.Funcs)
|
|
|
|
} else {
|
|
|
|
text = defaultTmplText
|
|
|
|
}
|
|
|
|
|
|
|
|
f.Tmpl = template.Must(tmpl.Parse(text))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const defaultTmplText = "{{.Now.Format .TimeFormat}}|{{.Latency}}|{{.Code}}|{{.Method}}|{{.Path}}|{{.IP}}|{{.RequestValuesLine}}|{{.BytesReceivedLine}}|{{.BytesSentLine}}|{{.Request}}|{{.Response}}|\n"
|
|
|
|
|
2022-02-18 21:19:33 +01:00
|
|
|
func (f *Template) LogText(log *Log) (string, error) {
|
|
|
|
var err error
|
|
|
|
|
|
|
|
// A template may be executed safely in parallel, although if parallel
|
|
|
|
// executions share a Writer the output may be interleaved.
|
|
|
|
// We solve that using a buffer pool, no locks when template is executing (x2 performance boost).
|
|
|
|
temp := f.ac.bufPool.Get().(*bytes.Buffer)
|
|
|
|
|
|
|
|
if f.TmplName != "" {
|
|
|
|
err = f.Tmpl.ExecuteTemplate(temp, f.TmplName, log)
|
|
|
|
} else {
|
|
|
|
err = f.Tmpl.Execute(temp, log)
|
|
|
|
}
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
f.ac.bufPool.Put(temp)
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
|
|
|
text := temp.String()
|
|
|
|
temp.Reset()
|
|
|
|
f.ac.bufPool.Put(temp)
|
|
|
|
|
|
|
|
return text, nil
|
|
|
|
}
|
|
|
|
|
2020-09-12 11:34:59 +02:00
|
|
|
// Format prints the logs in text/template format.
|
|
|
|
func (f *Template) Format(log *Log) (bool, error) {
|
|
|
|
var err error
|
|
|
|
|
|
|
|
// A template may be executed safely in parallel, although if parallel
|
|
|
|
// executions share a Writer the output may be interleaved.
|
2020-09-13 01:56:22 +02:00
|
|
|
// We solve that using a buffer pool, no locks when template is executing (x2 performance boost).
|
|
|
|
temp := f.ac.bufPool.Get().(*bytes.Buffer)
|
|
|
|
|
2020-09-12 11:34:59 +02:00
|
|
|
if f.TmplName != "" {
|
2020-09-13 01:56:22 +02:00
|
|
|
err = f.Tmpl.ExecuteTemplate(temp, f.TmplName, log)
|
2020-09-12 11:34:59 +02:00
|
|
|
} else {
|
2020-09-13 01:56:22 +02:00
|
|
|
err = f.Tmpl.Execute(temp, log)
|
|
|
|
}
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
f.ac.bufPool.Put(temp)
|
|
|
|
return true, err
|
2020-09-12 11:34:59 +02:00
|
|
|
}
|
|
|
|
|
2020-09-13 01:56:22 +02:00
|
|
|
f.ac.Write(temp.Bytes())
|
|
|
|
temp.Reset()
|
|
|
|
f.ac.bufPool.Put(temp)
|
|
|
|
return true, nil
|
2020-09-12 11:34:59 +02:00
|
|
|
}
|