2020-09-07 05:12:45 +02:00
|
|
|
package accesslog
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/json"
|
2020-09-07 10:35:27 +02:00
|
|
|
"fmt"
|
2020-09-07 05:12:45 +02:00
|
|
|
"io"
|
2020-09-07 10:35:27 +02:00
|
|
|
"strings"
|
2020-09-07 05:12:45 +02:00
|
|
|
"sync"
|
2020-09-07 10:35:27 +02:00
|
|
|
"text/template"
|
2020-09-07 05:12:45 +02:00
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/kataras/iris/v12/context"
|
|
|
|
"github.com/kataras/iris/v12/core/memstore"
|
|
|
|
)
|
|
|
|
|
|
|
|
// Log represents the log data specifically for the accesslog middleware.
|
|
|
|
type Log struct {
|
|
|
|
// The AccessLog instance this Log was created of.
|
2020-09-07 10:35:27 +02:00
|
|
|
Logger *AccessLog `json:"-" yaml:"-" toml:"-"`
|
2020-09-07 05:12:45 +02:00
|
|
|
|
|
|
|
// The time the log is created.
|
2020-09-07 10:35:27 +02:00
|
|
|
Now time.Time `json:"-" yaml:"-" toml:"-"`
|
|
|
|
// TimeFormat selected to print the Time as string,
|
|
|
|
// useful on Template Formatter.
|
|
|
|
TimeFormat string `json:"-" yaml:"-" toml:"-"`
|
2020-09-07 05:12:45 +02:00
|
|
|
// Timestamp the Now's unix timestamp (seconds).
|
|
|
|
Timestamp int64 `json:"timestamp"`
|
|
|
|
|
|
|
|
// Request-Response latency.
|
|
|
|
Latency time.Duration `json:"latency"`
|
|
|
|
// Init request's Method and Path.
|
|
|
|
Method string `json:"method"`
|
|
|
|
Path string `json:"path"`
|
2020-09-07 10:35:27 +02:00
|
|
|
// The response status code.
|
|
|
|
Code int `json:"code"`
|
2020-09-07 05:12:45 +02:00
|
|
|
// Sorted URL Query arguments.
|
2020-09-09 13:43:26 +02:00
|
|
|
Query []memstore.StringEntry `json:"query,omitempty"`
|
2020-09-07 05:12:45 +02:00
|
|
|
// Dynamic path parameters.
|
2020-09-09 13:43:26 +02:00
|
|
|
PathParams []memstore.Entry `json:"params,omitempty"`
|
2020-09-07 05:12:45 +02:00
|
|
|
// Fields any data information useful to represent this Log.
|
2020-09-07 10:35:27 +02:00
|
|
|
Fields []memstore.Entry `json:"fields,omitempty"`
|
2020-09-07 05:12:45 +02:00
|
|
|
|
2020-09-08 12:44:50 +02:00
|
|
|
// The actual number of bytes received and sent on the network (headers + body).
|
|
|
|
BytesReceived int `json:"bytes_received"`
|
|
|
|
BytesSent int `json:"bytes_sent"`
|
|
|
|
|
2020-09-07 05:12:45 +02:00
|
|
|
// The Request and Response raw bodies.
|
|
|
|
// If they are escaped (e.g. JSON),
|
|
|
|
// A third-party software can read it through:
|
|
|
|
// data, _ := strconv.Unquote(log.Request)
|
|
|
|
// err := json.Unmarshal([]byte(data), &customStruct)
|
|
|
|
Request string `json:"request"`
|
|
|
|
Response string `json:"response"`
|
|
|
|
|
|
|
|
// A copy of the Request's Context when Async is true (safe to use concurrently),
|
|
|
|
// otherwise it's the current Context (not safe for concurrent access).
|
2020-09-07 10:35:27 +02:00
|
|
|
Ctx *context.Context `json:"-" yaml:"-" toml:"-"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// RequestValuesLine returns a string line which
|
|
|
|
// combines the path parameters, query and custom fields.
|
|
|
|
func (l *Log) RequestValuesLine() string {
|
|
|
|
return parseRequestValues(l.Code, l.Ctx.Params(), l.Ctx.URLParamsSorted(), l.Fields)
|
|
|
|
}
|
|
|
|
|
2020-09-08 12:44:50 +02:00
|
|
|
// BytesReceivedLine returns the formatted bytes received length.
|
|
|
|
func (l *Log) BytesReceivedLine() string {
|
|
|
|
return formatBytes(l.BytesReceived)
|
|
|
|
}
|
|
|
|
|
|
|
|
// BytesSentLine returns the formatted bytes sent length.
|
|
|
|
func (l *Log) BytesSentLine() string {
|
|
|
|
return formatBytes(l.BytesSent)
|
|
|
|
}
|
|
|
|
|
|
|
|
func formatBytes(b int) string {
|
|
|
|
if b <= 0 {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
|
|
|
const unit = 1024
|
|
|
|
if b < unit {
|
|
|
|
return fmt.Sprintf("%d B", b)
|
|
|
|
}
|
|
|
|
div, exp := int64(unit), 0
|
|
|
|
for n := b / unit; n >= unit; n /= unit {
|
|
|
|
div *= unit
|
|
|
|
exp++
|
|
|
|
}
|
|
|
|
return fmt.Sprintf("%.1f %cB",
|
|
|
|
float64(b)/float64(div), "KMGTPE"[exp])
|
|
|
|
}
|
|
|
|
|
2020-09-07 10:35:27 +02:00
|
|
|
func parseRequestValues(code int, pathParams *context.RequestParams, query []memstore.StringEntry, fields memstore.Store) (requestValues string) {
|
|
|
|
var buf strings.Builder
|
|
|
|
|
|
|
|
if !context.StatusCodeNotSuccessful(code) {
|
|
|
|
// collect path parameters on a successful request-response only.
|
|
|
|
pathParams.Visit(func(key, value string) {
|
|
|
|
buf.WriteString(key)
|
|
|
|
buf.WriteByte('=')
|
|
|
|
buf.WriteString(value)
|
|
|
|
buf.WriteByte(' ')
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, entry := range query {
|
|
|
|
buf.WriteString(entry.Key)
|
|
|
|
buf.WriteByte('=')
|
|
|
|
buf.WriteString(entry.Value)
|
|
|
|
buf.WriteByte(' ')
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, entry := range fields {
|
|
|
|
buf.WriteString(entry.Key)
|
|
|
|
buf.WriteByte('=')
|
|
|
|
buf.WriteString(fmt.Sprintf("%v", entry.ValueRaw))
|
|
|
|
buf.WriteByte(' ')
|
|
|
|
}
|
|
|
|
|
|
|
|
if n := buf.Len(); n > 1 {
|
|
|
|
requestValues = buf.String()[0 : n-1] // remove last space.
|
|
|
|
}
|
|
|
|
|
|
|
|
return
|
2020-09-07 05:12:45 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Formatter is responsible to print a Log to the accesslog's writer.
|
|
|
|
type Formatter interface {
|
2020-09-09 13:43:26 +02:00
|
|
|
// SetOutput should inject the accesslog's direct output,
|
|
|
|
// if this "dest" is used then the Formatter
|
|
|
|
// should manually control its concurrent use.
|
|
|
|
SetOutput(dest io.Writer)
|
2020-09-07 05:12:45 +02:00
|
|
|
// Format should print the Log.
|
2020-09-07 10:35:27 +02:00
|
|
|
// Returns nil error on handle successfully,
|
|
|
|
// otherwise the log will be printed using the default formatter
|
|
|
|
// and the error will be printed to the Iris Application's error log level.
|
2020-09-09 13:43:26 +02:00
|
|
|
// Should return true if this handled the logging, otherwise false to
|
|
|
|
// continue with the default print format.
|
|
|
|
Format(log *Log) (bool, error)
|
2020-09-07 05:12:45 +02:00
|
|
|
}
|
|
|
|
|
2020-09-07 10:35:27 +02:00
|
|
|
var (
|
|
|
|
_ Formatter = (*JSON)(nil)
|
|
|
|
_ Formatter = (*Template)(nil)
|
|
|
|
)
|
|
|
|
|
2020-09-07 05:12:45 +02:00
|
|
|
// JSON is a Formatter type for JSON logs.
|
|
|
|
type JSON struct {
|
|
|
|
Prefix, Indent string
|
|
|
|
EscapeHTML bool
|
|
|
|
|
|
|
|
enc *json.Encoder
|
|
|
|
mu sync.Mutex
|
|
|
|
}
|
|
|
|
|
|
|
|
// SetOutput creates the json encoder writes to the "dest".
|
|
|
|
// It's called automatically by the middleware when this Formatter is used.
|
|
|
|
func (f *JSON) SetOutput(dest io.Writer) {
|
|
|
|
if dest == nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// All logs share the same accesslog's writer and it cannot change during serve-time.
|
|
|
|
enc := json.NewEncoder(dest)
|
|
|
|
enc.SetEscapeHTML(f.EscapeHTML)
|
|
|
|
enc.SetIndent(f.Prefix, f.Indent)
|
|
|
|
f.enc = enc
|
|
|
|
}
|
|
|
|
|
|
|
|
// Format prints the logs in JSON format.
|
2020-09-07 17:04:35 +02:00
|
|
|
// Writes to the destination directly,
|
|
|
|
// locks on each Format call.
|
2020-09-09 13:43:26 +02:00
|
|
|
func (f *JSON) Format(log *Log) (bool, error) {
|
2020-09-07 05:12:45 +02:00
|
|
|
f.mu.Lock()
|
|
|
|
err := f.enc.Encode(log)
|
|
|
|
f.mu.Unlock()
|
2020-09-07 10:35:27 +02:00
|
|
|
|
2020-09-09 13:43:26 +02:00
|
|
|
return true, err
|
2020-09-07 10:35:27 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// 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.
|
|
|
|
type Template struct {
|
|
|
|
// Custom template source.
|
|
|
|
// Use this or `Tmpl/TmplName` fields.
|
|
|
|
Text string
|
|
|
|
// Custom template to use, overrides the `Text` field if not nil.
|
|
|
|
Tmpl *template.Template
|
|
|
|
// If not empty then this named template/block
|
|
|
|
// is response to hold the log result.
|
|
|
|
TmplName string
|
|
|
|
|
2020-09-07 17:04:35 +02:00
|
|
|
dest io.Writer
|
|
|
|
mu sync.Mutex
|
2020-09-07 10:35:27 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// SetOutput creates the default template if missing
|
|
|
|
// when this formatter is registered.
|
|
|
|
func (f *Template) SetOutput(dest io.Writer) {
|
|
|
|
if f.Tmpl == nil {
|
|
|
|
text := f.Text
|
|
|
|
if f.Text == "" {
|
|
|
|
text = defaultTmplText
|
|
|
|
}
|
|
|
|
|
|
|
|
f.Tmpl = template.Must(template.New("").Parse(text))
|
|
|
|
}
|
2020-09-07 17:04:35 +02:00
|
|
|
|
|
|
|
f.dest = dest
|
2020-09-07 10:35:27 +02:00
|
|
|
}
|
|
|
|
|
2020-09-08 12:44:50 +02:00
|
|
|
const defaultTmplText = "{{.Now.Format .TimeFormat}}|{{.Latency}}|{{.Method}}|{{.Path}}|{{.RequestValuesLine}}|{{.Code}}|{{.BytesReceivedLine}}|{{.BytesSentLine}}|{{.Request}}|{{.Response}}|\n"
|
2020-09-07 10:35:27 +02:00
|
|
|
|
|
|
|
// Format prints the logs in text/template format.
|
2020-09-09 13:43:26 +02:00
|
|
|
func (f *Template) Format(log *Log) (bool, error) {
|
2020-09-07 17:04:35 +02:00
|
|
|
var err error
|
2020-09-07 10:35:27 +02:00
|
|
|
|
|
|
|
// A template may be executed safely in parallel, although if parallel
|
|
|
|
// executions share a Writer the output may be interleaved.
|
|
|
|
f.mu.Lock()
|
|
|
|
if f.TmplName != "" {
|
2020-09-07 17:04:35 +02:00
|
|
|
err = f.Tmpl.ExecuteTemplate(f.dest, f.TmplName, log)
|
2020-09-07 10:35:27 +02:00
|
|
|
} else {
|
2020-09-07 17:04:35 +02:00
|
|
|
err = f.Tmpl.Execute(f.dest, log)
|
2020-09-07 10:35:27 +02:00
|
|
|
}
|
|
|
|
f.mu.Unlock()
|
|
|
|
|
2020-09-09 13:43:26 +02:00
|
|
|
return true, err
|
2020-09-07 05:12:45 +02:00
|
|
|
}
|