mirror of
https://github.com/caddyserver/caddy.git
synced 2025-01-14 06:46:27 +03:00
logging: Implement log_append
handler (#6066)
* logging: Implement `extra_log` handler * Rename to `log_append` * Rename `skip_log` to `log_skip` --------- Co-authored-by: Matt Holt <mholt@users.noreply.github.com>
This commit is contained in:
parent
2a78c9c5e4
commit
0d44e3ecba
9 changed files with 239 additions and 12 deletions
|
@ -49,7 +49,8 @@ func init() {
|
||||||
RegisterDirective("handle_errors", parseHandleErrors)
|
RegisterDirective("handle_errors", parseHandleErrors)
|
||||||
RegisterHandlerDirective("invoke", parseInvoke)
|
RegisterHandlerDirective("invoke", parseInvoke)
|
||||||
RegisterDirective("log", parseLog)
|
RegisterDirective("log", parseLog)
|
||||||
RegisterHandlerDirective("skip_log", parseSkipLog)
|
RegisterHandlerDirective("skip_log", parseLogSkip)
|
||||||
|
RegisterHandlerDirective("log_skip", parseLogSkip)
|
||||||
}
|
}
|
||||||
|
|
||||||
// parseBind parses the bind directive. Syntax:
|
// parseBind parses the bind directive. Syntax:
|
||||||
|
@ -1038,13 +1039,19 @@ func parseLogHelper(h Helper, globalLogNames map[string]struct{}) ([]ConfigValue
|
||||||
return configValues, nil
|
return configValues, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// parseSkipLog parses the skip_log directive. Syntax:
|
// parseLogSkip parses the log_skip directive. Syntax:
|
||||||
//
|
//
|
||||||
// skip_log [<matcher>]
|
// log_skip [<matcher>]
|
||||||
func parseSkipLog(h Helper) (caddyhttp.MiddlewareHandler, error) {
|
func parseLogSkip(h Helper) (caddyhttp.MiddlewareHandler, error) {
|
||||||
h.Next() // consume directive name
|
h.Next() // consume directive name
|
||||||
|
|
||||||
|
// "skip_log" is deprecated, replaced by "log_skip"
|
||||||
|
if h.Val() == "skip_log" {
|
||||||
|
caddy.Log().Named("config.adapter.caddyfile").Warn("the 'skip_log' directive is deprecated, please use 'log_skip' instead!")
|
||||||
|
}
|
||||||
|
|
||||||
if h.NextArg() {
|
if h.NextArg() {
|
||||||
return nil, h.ArgErr()
|
return nil, h.ArgErr()
|
||||||
}
|
}
|
||||||
return caddyhttp.VarsMiddleware{"skip_log": true}, nil
|
return caddyhttp.VarsMiddleware{"log_skip": true}, nil
|
||||||
}
|
}
|
||||||
|
|
|
@ -43,7 +43,8 @@ var directiveOrder = []string{
|
||||||
"vars",
|
"vars",
|
||||||
"fs",
|
"fs",
|
||||||
"root",
|
"root",
|
||||||
"skip_log",
|
"log_append",
|
||||||
|
"log_skip",
|
||||||
|
|
||||||
"header",
|
"header",
|
||||||
"copy_response_headers", // only in reverse_proxy's handle_response
|
"copy_response_headers", // only in reverse_proxy's handle_response
|
||||||
|
|
71
caddytest/integration/caddyfile_adapt/log_add.caddyfiletest
Normal file
71
caddytest/integration/caddyfile_adapt/log_add.caddyfiletest
Normal file
|
@ -0,0 +1,71 @@
|
||||||
|
:80 {
|
||||||
|
log
|
||||||
|
|
||||||
|
vars foo foo
|
||||||
|
|
||||||
|
log_append const bar
|
||||||
|
log_append vars foo
|
||||||
|
log_append placeholder {path}
|
||||||
|
|
||||||
|
log_append /only-for-this-path secret value
|
||||||
|
}
|
||||||
|
----------
|
||||||
|
{
|
||||||
|
"apps": {
|
||||||
|
"http": {
|
||||||
|
"servers": {
|
||||||
|
"srv0": {
|
||||||
|
"listen": [
|
||||||
|
":80"
|
||||||
|
],
|
||||||
|
"routes": [
|
||||||
|
{
|
||||||
|
"handle": [
|
||||||
|
{
|
||||||
|
"foo": "foo",
|
||||||
|
"handler": "vars"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"match": [
|
||||||
|
{
|
||||||
|
"path": [
|
||||||
|
"/only-for-this-path"
|
||||||
|
]
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"handle": [
|
||||||
|
{
|
||||||
|
"handler": "log_append",
|
||||||
|
"key": "secret",
|
||||||
|
"value": "value"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"handle": [
|
||||||
|
{
|
||||||
|
"handler": "log_append",
|
||||||
|
"key": "const",
|
||||||
|
"value": "bar"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"handler": "log_append",
|
||||||
|
"key": "vars",
|
||||||
|
"value": "foo"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"handler": "log_append",
|
||||||
|
"key": "placeholder",
|
||||||
|
"value": "{http.request.uri.path}"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"logs": {}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,7 +1,7 @@
|
||||||
http://localhost:2020 {
|
http://localhost:2020 {
|
||||||
log
|
log
|
||||||
skip_log /first-hidden*
|
log_skip /first-hidden*
|
||||||
skip_log /second-hidden*
|
log_skip /second-hidden*
|
||||||
respond 200
|
respond 200
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -34,7 +34,7 @@ http://localhost:2020 {
|
||||||
"handle": [
|
"handle": [
|
||||||
{
|
{
|
||||||
"handler": "vars",
|
"handler": "vars",
|
||||||
"skip_log": true
|
"log_skip": true
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"match": [
|
"match": [
|
||||||
|
@ -49,7 +49,7 @@ http://localhost:2020 {
|
||||||
"handle": [
|
"handle": [
|
||||||
{
|
{
|
||||||
"handler": "vars",
|
"handler": "vars",
|
||||||
"skip_log": true
|
"log_skip": true
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"match": [
|
"match": [
|
||||||
|
|
|
@ -166,7 +166,7 @@ func (e *ExtraLogFields) Set(field zap.Field) {
|
||||||
const (
|
const (
|
||||||
// Variable name used to indicate that this request
|
// Variable name used to indicate that this request
|
||||||
// should be omitted from the access logs
|
// should be omitted from the access logs
|
||||||
SkipLogVar string = "skip_log"
|
LogSkipVar string = "log_skip"
|
||||||
|
|
||||||
// For adding additional fields to the access logs
|
// For adding additional fields to the access logs
|
||||||
ExtraLogFieldsCtxKey caddy.CtxKey = "extra_log_fields"
|
ExtraLogFieldsCtxKey caddy.CtxKey = "extra_log_fields"
|
||||||
|
|
53
modules/caddyhttp/logging/caddyfile.go
Normal file
53
modules/caddyhttp/logging/caddyfile.go
Normal file
|
@ -0,0 +1,53 @@
|
||||||
|
// Copyright 2015 Matthew Holt and The Caddy Authors
|
||||||
|
//
|
||||||
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
// you may not use this file except in compliance with the License.
|
||||||
|
// You may obtain a copy of the License at
|
||||||
|
//
|
||||||
|
// http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
//
|
||||||
|
// Unless required by applicable law or agreed to in writing, software
|
||||||
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
// See the License for the specific language governing permissions and
|
||||||
|
// limitations under the License.
|
||||||
|
|
||||||
|
package logging
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/caddyserver/caddy/v2/caddyconfig/caddyfile"
|
||||||
|
"github.com/caddyserver/caddy/v2/caddyconfig/httpcaddyfile"
|
||||||
|
"github.com/caddyserver/caddy/v2/modules/caddyhttp"
|
||||||
|
)
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
httpcaddyfile.RegisterHandlerDirective("log_append", parseCaddyfile)
|
||||||
|
}
|
||||||
|
|
||||||
|
// parseCaddyfile sets up the log_append handler from Caddyfile tokens. Syntax:
|
||||||
|
//
|
||||||
|
// log_append [<matcher>] <key> <value>
|
||||||
|
func parseCaddyfile(h httpcaddyfile.Helper) (caddyhttp.MiddlewareHandler, error) {
|
||||||
|
handler := new(LogAppend)
|
||||||
|
err := handler.UnmarshalCaddyfile(h.Dispenser)
|
||||||
|
return handler, err
|
||||||
|
}
|
||||||
|
|
||||||
|
// UnmarshalCaddyfile implements caddyfile.Unmarshaler.
|
||||||
|
func (h *LogAppend) UnmarshalCaddyfile(d *caddyfile.Dispenser) error {
|
||||||
|
d.Next() // consume directive name
|
||||||
|
if !d.NextArg() {
|
||||||
|
return d.ArgErr()
|
||||||
|
}
|
||||||
|
h.Key = d.Val()
|
||||||
|
if !d.NextArg() {
|
||||||
|
return d.ArgErr()
|
||||||
|
}
|
||||||
|
h.Value = d.Val()
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Interface guards
|
||||||
|
var (
|
||||||
|
_ caddyfile.Unmarshaler = (*LogAppend)(nil)
|
||||||
|
)
|
94
modules/caddyhttp/logging/logadd.go
Normal file
94
modules/caddyhttp/logging/logadd.go
Normal file
|
@ -0,0 +1,94 @@
|
||||||
|
// Copyright 2015 Matthew Holt and The Caddy Authors
|
||||||
|
//
|
||||||
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
// you may not use this file except in compliance with the License.
|
||||||
|
// You may obtain a copy of the License at
|
||||||
|
//
|
||||||
|
// http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
//
|
||||||
|
// Unless required by applicable law or agreed to in writing, software
|
||||||
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
// See the License for the specific language governing permissions and
|
||||||
|
// limitations under the License.
|
||||||
|
|
||||||
|
package logging
|
||||||
|
|
||||||
|
import (
|
||||||
|
"net/http"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"go.uber.org/zap"
|
||||||
|
|
||||||
|
"github.com/caddyserver/caddy/v2"
|
||||||
|
"github.com/caddyserver/caddy/v2/modules/caddyhttp"
|
||||||
|
)
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
caddy.RegisterModule(LogAppend{})
|
||||||
|
}
|
||||||
|
|
||||||
|
// LogAppend implements a middleware that takes a key and value, where
|
||||||
|
// the key is the name of a log field and the value is a placeholder,
|
||||||
|
// or variable key, or constant value to use for that field.
|
||||||
|
type LogAppend struct {
|
||||||
|
// Key is the name of the log field.
|
||||||
|
Key string `json:"key,omitempty"`
|
||||||
|
|
||||||
|
// Value is the value to use for the log field.
|
||||||
|
// If it is a placeholder (with surrounding `{}`),
|
||||||
|
// it will be evaluated when the log is written.
|
||||||
|
// If the value is a key that exists in the `vars`
|
||||||
|
// map, the value of that key will be used. Otherwise
|
||||||
|
// the value will be used as-is as a constant string.
|
||||||
|
Value string `json:"value,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
// CaddyModule returns the Caddy module information.
|
||||||
|
func (LogAppend) CaddyModule() caddy.ModuleInfo {
|
||||||
|
return caddy.ModuleInfo{
|
||||||
|
ID: "http.handlers.log_append",
|
||||||
|
New: func() caddy.Module { return new(LogAppend) },
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h LogAppend) ServeHTTP(w http.ResponseWriter, r *http.Request, next caddyhttp.Handler) error {
|
||||||
|
// Run the next handler in the chain first.
|
||||||
|
// If an error occurs, we still want to add
|
||||||
|
// any extra log fields that we can, so we
|
||||||
|
// hold onto the error and return it later.
|
||||||
|
handlerErr := next.ServeHTTP(w, r)
|
||||||
|
|
||||||
|
// On the way back up the chain, add the extra log field
|
||||||
|
ctx := r.Context()
|
||||||
|
|
||||||
|
vars := ctx.Value(caddyhttp.VarsCtxKey).(map[string]any)
|
||||||
|
repl := ctx.Value(caddy.ReplacerCtxKey).(*caddy.Replacer)
|
||||||
|
extra := ctx.Value(caddyhttp.ExtraLogFieldsCtxKey).(*caddyhttp.ExtraLogFields)
|
||||||
|
|
||||||
|
var varValue any
|
||||||
|
if strings.HasPrefix(h.Value, "{") &&
|
||||||
|
strings.HasSuffix(h.Value, "}") &&
|
||||||
|
strings.Count(h.Value, "{") == 1 {
|
||||||
|
// the value looks like a placeholder, so get its value
|
||||||
|
varValue, _ = repl.Get(strings.Trim(h.Value, "{}"))
|
||||||
|
} else if val, ok := vars[h.Value]; ok {
|
||||||
|
// the value is a key in the vars map
|
||||||
|
varValue = val
|
||||||
|
} else {
|
||||||
|
// the value is a constant string
|
||||||
|
varValue = h.Value
|
||||||
|
}
|
||||||
|
|
||||||
|
// Add the field to the extra log fields.
|
||||||
|
// We use zap.Any because it will reflect
|
||||||
|
// to the correct type for us.
|
||||||
|
extra.Add(zap.Any(h.Key, varValue))
|
||||||
|
|
||||||
|
return handlerErr
|
||||||
|
}
|
||||||
|
|
||||||
|
// Interface guards
|
||||||
|
var (
|
||||||
|
_ caddyhttp.MiddlewareHandler = (*LogAppend)(nil)
|
||||||
|
)
|
|
@ -715,7 +715,7 @@ func (s *Server) logRequest(
|
||||||
repl *caddy.Replacer, bodyReader *lengthReader, shouldLogCredentials bool,
|
repl *caddy.Replacer, bodyReader *lengthReader, shouldLogCredentials bool,
|
||||||
) {
|
) {
|
||||||
// this request may be flagged as omitted from the logs
|
// this request may be flagged as omitted from the logs
|
||||||
if skipLog, ok := GetVar(r.Context(), SkipLogVar).(bool); ok && skipLog {
|
if skip, ok := GetVar(r.Context(), LogSkipVar).(bool); ok && skip {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -10,6 +10,7 @@ import (
|
||||||
_ "github.com/caddyserver/caddy/v2/modules/caddyhttp/encode/zstd"
|
_ "github.com/caddyserver/caddy/v2/modules/caddyhttp/encode/zstd"
|
||||||
_ "github.com/caddyserver/caddy/v2/modules/caddyhttp/fileserver"
|
_ "github.com/caddyserver/caddy/v2/modules/caddyhttp/fileserver"
|
||||||
_ "github.com/caddyserver/caddy/v2/modules/caddyhttp/headers"
|
_ "github.com/caddyserver/caddy/v2/modules/caddyhttp/headers"
|
||||||
|
_ "github.com/caddyserver/caddy/v2/modules/caddyhttp/logging"
|
||||||
_ "github.com/caddyserver/caddy/v2/modules/caddyhttp/map"
|
_ "github.com/caddyserver/caddy/v2/modules/caddyhttp/map"
|
||||||
_ "github.com/caddyserver/caddy/v2/modules/caddyhttp/proxyprotocol"
|
_ "github.com/caddyserver/caddy/v2/modules/caddyhttp/proxyprotocol"
|
||||||
_ "github.com/caddyserver/caddy/v2/modules/caddyhttp/push"
|
_ "github.com/caddyserver/caddy/v2/modules/caddyhttp/push"
|
||||||
|
|
Loading…
Reference in a new issue