mirror of
				https://github.com/caddyserver/caddy.git
				synced 2025-10-25 15:52:45 -04:00 
			
		
		
		
	
		
			
				
	
	
		
			825 lines
		
	
	
		
			25 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			825 lines
		
	
	
		
			25 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // 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 caddy
 | |
| 
 | |
| import (
 | |
| 	"encoding/json"
 | |
| 	"fmt"
 | |
| 	"io"
 | |
| 	"log"
 | |
| 	"os"
 | |
| 	"slices"
 | |
| 	"strings"
 | |
| 	"sync"
 | |
| 	"time"
 | |
| 
 | |
| 	"go.uber.org/zap"
 | |
| 	"go.uber.org/zap/zapcore"
 | |
| 	"golang.org/x/term"
 | |
| 
 | |
| 	"github.com/caddyserver/caddy/v2/internal"
 | |
| )
 | |
| 
 | |
| func init() {
 | |
| 	RegisterModule(StdoutWriter{})
 | |
| 	RegisterModule(StderrWriter{})
 | |
| 	RegisterModule(DiscardWriter{})
 | |
| }
 | |
| 
 | |
| // Logging facilitates logging within Caddy. The default log is
 | |
| // called "default" and you can customize it. You can also define
 | |
| // additional logs.
 | |
| //
 | |
| // By default, all logs at INFO level and higher are written to
 | |
| // standard error ("stderr" writer) in a human-readable format
 | |
| // ("console" encoder if stdout is an interactive terminal, "json"
 | |
| // encoder otherwise).
 | |
| //
 | |
| // All defined logs accept all log entries by default, but you
 | |
| // can filter by level and module/logger names. A logger's name
 | |
| // is the same as the module's name, but a module may append to
 | |
| // logger names for more specificity. For example, you can
 | |
| // filter logs emitted only by HTTP handlers using the name
 | |
| // "http.handlers", because all HTTP handler module names have
 | |
| // that prefix.
 | |
| //
 | |
| // Caddy logs (except the sink) are zero-allocation, so they are
 | |
| // very high-performing in terms of memory and CPU time. Enabling
 | |
| // sampling can further increase throughput on extremely high-load
 | |
| // servers.
 | |
| type Logging struct {
 | |
| 	// Sink is the destination for all unstructured logs emitted
 | |
| 	// from Go's standard library logger. These logs are common
 | |
| 	// in dependencies that are not designed specifically for use
 | |
| 	// in Caddy. Because it is global and unstructured, the sink
 | |
| 	// lacks most advanced features and customizations.
 | |
| 	Sink *SinkLog `json:"sink,omitempty"`
 | |
| 
 | |
| 	// Logs are your logs, keyed by an arbitrary name of your
 | |
| 	// choosing. The default log can be customized by defining
 | |
| 	// a log called "default". You can further define other logs
 | |
| 	// and filter what kinds of entries they accept.
 | |
| 	Logs map[string]*CustomLog `json:"logs,omitempty"`
 | |
| 
 | |
| 	// a list of all keys for open writers; all writers
 | |
| 	// that are opened to provision this logging config
 | |
| 	// must have their keys added to this list so they
 | |
| 	// can be closed when cleaning up
 | |
| 	writerKeys []string
 | |
| }
 | |
| 
 | |
| // openLogs sets up the config and opens all the configured writers.
 | |
| // It closes its logs when ctx is canceled, so it should clean up
 | |
| // after itself.
 | |
| func (logging *Logging) openLogs(ctx Context) error {
 | |
| 	// make sure to deallocate resources when context is done
 | |
| 	ctx.OnCancel(func() {
 | |
| 		err := logging.closeLogs()
 | |
| 		if err != nil {
 | |
| 			Log().Error("closing logs", zap.Error(err))
 | |
| 		}
 | |
| 	})
 | |
| 
 | |
| 	// set up the "sink" log first (std lib's default global logger)
 | |
| 	if logging.Sink != nil {
 | |
| 		err := logging.Sink.provision(ctx, logging)
 | |
| 		if err != nil {
 | |
| 			return fmt.Errorf("setting up sink log: %v", err)
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	// as a special case, set up the default structured Caddy log next
 | |
| 	if err := logging.setupNewDefault(ctx); err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 
 | |
| 	// then set up any other custom logs
 | |
| 	for name, l := range logging.Logs {
 | |
| 		// the default log is already set up
 | |
| 		if name == DefaultLoggerName {
 | |
| 			continue
 | |
| 		}
 | |
| 
 | |
| 		err := l.provision(ctx, logging)
 | |
| 		if err != nil {
 | |
| 			return fmt.Errorf("setting up custom log '%s': %v", name, err)
 | |
| 		}
 | |
| 
 | |
| 		// Any other logs that use the discard writer can be deleted
 | |
| 		// entirely. This avoids encoding and processing of each
 | |
| 		// log entry that would just be thrown away anyway. Notably,
 | |
| 		// we do not reach this point for the default log, which MUST
 | |
| 		// exist, otherwise core log emissions would panic because
 | |
| 		// they use the Log() function directly which expects a non-nil
 | |
| 		// logger. Even if we keep logs with a discard writer, they
 | |
| 		// have a nop core, and keeping them at all seems unnecessary.
 | |
| 		if _, ok := l.writerOpener.(*DiscardWriter); ok {
 | |
| 			delete(logging.Logs, name)
 | |
| 			continue
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| func (logging *Logging) setupNewDefault(ctx Context) error {
 | |
| 	if logging.Logs == nil {
 | |
| 		logging.Logs = make(map[string]*CustomLog)
 | |
| 	}
 | |
| 
 | |
| 	// extract the user-defined default log, if any
 | |
| 	newDefault := new(defaultCustomLog)
 | |
| 	if userDefault, ok := logging.Logs[DefaultLoggerName]; ok {
 | |
| 		newDefault.CustomLog = userDefault
 | |
| 	} else {
 | |
| 		// if none, make one with our own default settings
 | |
| 		var err error
 | |
| 		newDefault, err = newDefaultProductionLog()
 | |
| 		if err != nil {
 | |
| 			return fmt.Errorf("setting up default Caddy log: %v", err)
 | |
| 		}
 | |
| 		logging.Logs[DefaultLoggerName] = newDefault.CustomLog
 | |
| 	}
 | |
| 
 | |
| 	// options for the default logger
 | |
| 	options, err := newDefault.CustomLog.buildOptions()
 | |
| 	if err != nil {
 | |
| 		return fmt.Errorf("setting up default log: %v", err)
 | |
| 	}
 | |
| 
 | |
| 	// set up this new log
 | |
| 	err = newDefault.CustomLog.provision(ctx, logging)
 | |
| 	if err != nil {
 | |
| 		return fmt.Errorf("setting up default log: %v", err)
 | |
| 	}
 | |
| 
 | |
| 	filteringCore := &filteringCore{newDefault.CustomLog.core, newDefault.CustomLog}
 | |
| 	newDefault.logger = zap.New(filteringCore, options...)
 | |
| 
 | |
| 	// redirect the default caddy logs
 | |
| 	defaultLoggerMu.Lock()
 | |
| 	oldDefault := defaultLogger
 | |
| 	defaultLogger = newDefault
 | |
| 	defaultLoggerMu.Unlock()
 | |
| 
 | |
| 	// if the new writer is different, indicate it in the logs for convenience
 | |
| 	var newDefaultLogWriterKey, currentDefaultLogWriterKey string
 | |
| 	var newDefaultLogWriterStr, currentDefaultLogWriterStr string
 | |
| 	if newDefault.writerOpener != nil {
 | |
| 		newDefaultLogWriterKey = newDefault.writerOpener.WriterKey()
 | |
| 		newDefaultLogWriterStr = newDefault.writerOpener.String()
 | |
| 	}
 | |
| 	if oldDefault.writerOpener != nil {
 | |
| 		currentDefaultLogWriterKey = oldDefault.writerOpener.WriterKey()
 | |
| 		currentDefaultLogWriterStr = oldDefault.writerOpener.String()
 | |
| 	}
 | |
| 	if newDefaultLogWriterKey != currentDefaultLogWriterKey {
 | |
| 		oldDefault.logger.Info("redirected default logger",
 | |
| 			zap.String("from", currentDefaultLogWriterStr),
 | |
| 			zap.String("to", newDefaultLogWriterStr),
 | |
| 		)
 | |
| 	}
 | |
| 
 | |
| 	// if we had a buffered core, flush its contents ASAP
 | |
| 	// before we try to log anything else, so the order of
 | |
| 	// logs is preserved
 | |
| 	if oldBufferCore, ok := oldDefault.logger.Core().(*internal.LogBufferCore); ok {
 | |
| 		oldBufferCore.FlushTo(newDefault.logger)
 | |
| 	}
 | |
| 
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| // closeLogs cleans up resources allocated during openLogs.
 | |
| // A successful call to openLogs calls this automatically
 | |
| // when the context is canceled.
 | |
| func (logging *Logging) closeLogs() error {
 | |
| 	for _, key := range logging.writerKeys {
 | |
| 		_, err := writers.Delete(key)
 | |
| 		if err != nil {
 | |
| 			log.Printf("[ERROR] Closing log writer %v: %v", key, err)
 | |
| 		}
 | |
| 	}
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| // Logger returns a logger that is ready for the module to use.
 | |
| func (logging *Logging) Logger(mod Module) *zap.Logger {
 | |
| 	modID := string(mod.CaddyModule().ID)
 | |
| 	var cores []zapcore.Core
 | |
| 	var options []zap.Option
 | |
| 
 | |
| 	if logging != nil {
 | |
| 		for _, l := range logging.Logs {
 | |
| 			if l.matchesModule(modID) {
 | |
| 				if len(l.Include) == 0 && len(l.Exclude) == 0 {
 | |
| 					cores = append(cores, l.core)
 | |
| 					continue
 | |
| 				}
 | |
| 				if len(options) == 0 {
 | |
| 					newOptions, err := l.buildOptions()
 | |
| 					if err != nil {
 | |
| 						Log().Error("building options for logger", zap.String("module", modID), zap.Error(err))
 | |
| 					}
 | |
| 					options = newOptions
 | |
| 				}
 | |
| 				cores = append(cores, &filteringCore{Core: l.core, cl: l})
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	multiCore := zapcore.NewTee(cores...)
 | |
| 
 | |
| 	return zap.New(multiCore, options...).Named(modID)
 | |
| }
 | |
| 
 | |
| // openWriter opens a writer using opener, and returns true if
 | |
| // the writer is new, or false if the writer already exists.
 | |
| func (logging *Logging) openWriter(opener WriterOpener) (io.WriteCloser, bool, error) {
 | |
| 	key := opener.WriterKey()
 | |
| 	writer, loaded, err := writers.LoadOrNew(key, func() (Destructor, error) {
 | |
| 		w, err := opener.OpenWriter()
 | |
| 		return writerDestructor{w}, err
 | |
| 	})
 | |
| 	if err != nil {
 | |
| 		return nil, false, err
 | |
| 	}
 | |
| 	logging.writerKeys = append(logging.writerKeys, key)
 | |
| 	return writer.(io.WriteCloser), !loaded, nil
 | |
| }
 | |
| 
 | |
| // WriterOpener is a module that can open a log writer.
 | |
| // It can return a human-readable string representation
 | |
| // of itself so that operators can understand where
 | |
| // the logs are going.
 | |
| type WriterOpener interface {
 | |
| 	fmt.Stringer
 | |
| 
 | |
| 	// WriterKey is a string that uniquely identifies this
 | |
| 	// writer configuration. It is not shown to humans.
 | |
| 	WriterKey() string
 | |
| 
 | |
| 	// OpenWriter opens a log for writing. The writer
 | |
| 	// should be safe for concurrent use but need not
 | |
| 	// be synchronous.
 | |
| 	OpenWriter() (io.WriteCloser, error)
 | |
| }
 | |
| 
 | |
| // IsWriterStandardStream returns true if the input is a
 | |
| // writer-opener to a standard stream (stdout, stderr).
 | |
| func IsWriterStandardStream(wo WriterOpener) bool {
 | |
| 	switch wo.(type) {
 | |
| 	case StdoutWriter, StderrWriter,
 | |
| 		*StdoutWriter, *StderrWriter:
 | |
| 		return true
 | |
| 	}
 | |
| 	return false
 | |
| }
 | |
| 
 | |
| type writerDestructor struct {
 | |
| 	io.WriteCloser
 | |
| }
 | |
| 
 | |
| func (wdest writerDestructor) Destruct() error {
 | |
| 	return wdest.Close()
 | |
| }
 | |
| 
 | |
| // BaseLog contains the common logging parameters for logging.
 | |
| type BaseLog struct {
 | |
| 	// The module that writes out log entries for the sink.
 | |
| 	WriterRaw json.RawMessage `json:"writer,omitempty" caddy:"namespace=caddy.logging.writers inline_key=output"`
 | |
| 
 | |
| 	// The encoder is how the log entries are formatted or encoded.
 | |
| 	EncoderRaw json.RawMessage `json:"encoder,omitempty" caddy:"namespace=caddy.logging.encoders inline_key=format"`
 | |
| 
 | |
| 	// Tees entries through a zap.Core module which can extract
 | |
| 	// log entry metadata and fields for further processing.
 | |
| 	CoreRaw json.RawMessage `json:"core,omitempty" caddy:"namespace=caddy.logging.cores inline_key=module"`
 | |
| 
 | |
| 	// Level is the minimum level to emit, and is inclusive.
 | |
| 	// Possible levels: DEBUG, INFO, WARN, ERROR, PANIC, and FATAL
 | |
| 	Level string `json:"level,omitempty"`
 | |
| 
 | |
| 	// Sampling configures log entry sampling. If enabled,
 | |
| 	// only some log entries will be emitted. This is useful
 | |
| 	// for improving performance on extremely high-pressure
 | |
| 	// servers.
 | |
| 	Sampling *LogSampling `json:"sampling,omitempty"`
 | |
| 
 | |
| 	// If true, the log entry will include the caller's
 | |
| 	// file name and line number. Default off.
 | |
| 	WithCaller bool `json:"with_caller,omitempty"`
 | |
| 
 | |
| 	// If non-zero, and `with_caller` is true, this many
 | |
| 	// stack frames will be skipped when determining the
 | |
| 	// caller. Default 0.
 | |
| 	WithCallerSkip int `json:"with_caller_skip,omitempty"`
 | |
| 
 | |
| 	// If not empty, the log entry will include a stack trace
 | |
| 	// for all logs at the given level or higher. See `level`
 | |
| 	// for possible values. Default off.
 | |
| 	WithStacktrace string `json:"with_stacktrace,omitempty"`
 | |
| 
 | |
| 	writerOpener WriterOpener
 | |
| 	writer       io.WriteCloser
 | |
| 	encoder      zapcore.Encoder
 | |
| 	levelEnabler zapcore.LevelEnabler
 | |
| 	core         zapcore.Core
 | |
| }
 | |
| 
 | |
| func (cl *BaseLog) provisionCommon(ctx Context, logging *Logging) error {
 | |
| 	if cl.WriterRaw != nil {
 | |
| 		mod, err := ctx.LoadModule(cl, "WriterRaw")
 | |
| 		if err != nil {
 | |
| 			return fmt.Errorf("loading log writer module: %v", err)
 | |
| 		}
 | |
| 		cl.writerOpener = mod.(WriterOpener)
 | |
| 	}
 | |
| 	if cl.writerOpener == nil {
 | |
| 		cl.writerOpener = StderrWriter{}
 | |
| 	}
 | |
| 	var err error
 | |
| 	cl.writer, _, err = logging.openWriter(cl.writerOpener)
 | |
| 	if err != nil {
 | |
| 		return fmt.Errorf("opening log writer using %#v: %v", cl.writerOpener, err)
 | |
| 	}
 | |
| 
 | |
| 	// set up the log level
 | |
| 	cl.levelEnabler, err = parseLevel(cl.Level)
 | |
| 	if err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 
 | |
| 	if cl.EncoderRaw != nil {
 | |
| 		mod, err := ctx.LoadModule(cl, "EncoderRaw")
 | |
| 		if err != nil {
 | |
| 			return fmt.Errorf("loading log encoder module: %v", err)
 | |
| 		}
 | |
| 		cl.encoder = mod.(zapcore.Encoder)
 | |
| 
 | |
| 		// if the encoder module needs the writer to determine
 | |
| 		// the correct default to use for a nested encoder, we
 | |
| 		// pass it down as a secondary provisioning step
 | |
| 		if cfd, ok := mod.(ConfiguresFormatterDefault); ok {
 | |
| 			if err := cfd.ConfigureDefaultFormat(cl.writerOpener); err != nil {
 | |
| 				return fmt.Errorf("configuring default format for encoder module: %v", err)
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 	if cl.encoder == nil {
 | |
| 		cl.encoder = newDefaultProductionLogEncoder(cl.writerOpener)
 | |
| 	}
 | |
| 	cl.buildCore()
 | |
| 	if cl.CoreRaw != nil {
 | |
| 		mod, err := ctx.LoadModule(cl, "CoreRaw")
 | |
| 		if err != nil {
 | |
| 			return fmt.Errorf("loading log core module: %v", err)
 | |
| 		}
 | |
| 		core := mod.(zapcore.Core)
 | |
| 		cl.core = zapcore.NewTee(cl.core, core)
 | |
| 	}
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| func (cl *BaseLog) buildCore() {
 | |
| 	// logs which only discard their output don't need
 | |
| 	// to perform encoding or any other processing steps
 | |
| 	// at all, so just shortcut to a nop core instead
 | |
| 	if _, ok := cl.writerOpener.(*DiscardWriter); ok {
 | |
| 		cl.core = zapcore.NewNopCore()
 | |
| 		return
 | |
| 	}
 | |
| 	c := zapcore.NewCore(
 | |
| 		cl.encoder,
 | |
| 		zapcore.AddSync(cl.writer),
 | |
| 		cl.levelEnabler,
 | |
| 	)
 | |
| 	if cl.Sampling != nil {
 | |
| 		if cl.Sampling.Interval == 0 {
 | |
| 			cl.Sampling.Interval = 1 * time.Second
 | |
| 		}
 | |
| 		if cl.Sampling.First == 0 {
 | |
| 			cl.Sampling.First = 100
 | |
| 		}
 | |
| 		if cl.Sampling.Thereafter == 0 {
 | |
| 			cl.Sampling.Thereafter = 100
 | |
| 		}
 | |
| 		c = zapcore.NewSamplerWithOptions(c, cl.Sampling.Interval,
 | |
| 			cl.Sampling.First, cl.Sampling.Thereafter)
 | |
| 	}
 | |
| 	cl.core = c
 | |
| }
 | |
| 
 | |
| func (cl *BaseLog) buildOptions() ([]zap.Option, error) {
 | |
| 	var options []zap.Option
 | |
| 	if cl.WithCaller {
 | |
| 		options = append(options, zap.AddCaller())
 | |
| 		if cl.WithCallerSkip != 0 {
 | |
| 			options = append(options, zap.AddCallerSkip(cl.WithCallerSkip))
 | |
| 		}
 | |
| 	}
 | |
| 	if cl.WithStacktrace != "" {
 | |
| 		levelEnabler, err := parseLevel(cl.WithStacktrace)
 | |
| 		if err != nil {
 | |
| 			return options, fmt.Errorf("setting up default Caddy log: %v", err)
 | |
| 		}
 | |
| 		options = append(options, zap.AddStacktrace(levelEnabler))
 | |
| 	}
 | |
| 	return options, nil
 | |
| }
 | |
| 
 | |
| // SinkLog configures the default Go standard library
 | |
| // global logger in the log package. This is necessary because
 | |
| // module dependencies which are not built specifically for
 | |
| // Caddy will use the standard logger. This is also known as
 | |
| // the "sink" logger.
 | |
| type SinkLog struct {
 | |
| 	BaseLog
 | |
| }
 | |
| 
 | |
| func (sll *SinkLog) provision(ctx Context, logging *Logging) error {
 | |
| 	if err := sll.provisionCommon(ctx, logging); err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 
 | |
| 	options, err := sll.buildOptions()
 | |
| 	if err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 
 | |
| 	logger := zap.New(sll.core, options...)
 | |
| 	ctx.cleanupFuncs = append(ctx.cleanupFuncs, zap.RedirectStdLog(logger))
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| // CustomLog represents a custom logger configuration.
 | |
| //
 | |
| // By default, a log will emit all log entries. Some entries
 | |
| // will be skipped if sampling is enabled. Further, the Include
 | |
| // and Exclude parameters define which loggers (by name) are
 | |
| // allowed or rejected from emitting in this log. If both Include
 | |
| // and Exclude are populated, their values must be mutually
 | |
| // exclusive, and longer namespaces have priority. If neither
 | |
| // are populated, all logs are emitted.
 | |
| type CustomLog struct {
 | |
| 	BaseLog
 | |
| 
 | |
| 	// Include defines the names of loggers to emit in this
 | |
| 	// log. For example, to include only logs emitted by the
 | |
| 	// admin API, you would include "admin.api".
 | |
| 	Include []string `json:"include,omitempty"`
 | |
| 
 | |
| 	// Exclude defines the names of loggers that should be
 | |
| 	// skipped by this log. For example, to exclude only
 | |
| 	// HTTP access logs, you would exclude "http.log.access".
 | |
| 	Exclude []string `json:"exclude,omitempty"`
 | |
| }
 | |
| 
 | |
| func (cl *CustomLog) provision(ctx Context, logging *Logging) error {
 | |
| 	if err := cl.provisionCommon(ctx, logging); err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 
 | |
| 	// If both Include and Exclude lists are populated, then each item must
 | |
| 	// be a superspace or subspace of an item in the other list, because
 | |
| 	// populating both lists means that any given item is either a rule
 | |
| 	// or an exception to another rule. But if the item is not a super-
 | |
| 	// or sub-space of any item in the other list, it is neither a rule
 | |
| 	// nor an exception, and is a contradiction. Ensure, too, that the
 | |
| 	// sets do not intersect, which is also a contradiction.
 | |
| 	if len(cl.Include) > 0 && len(cl.Exclude) > 0 {
 | |
| 		// prevent intersections
 | |
| 		for _, allow := range cl.Include {
 | |
| 			if slices.Contains(cl.Exclude, allow) {
 | |
| 				return fmt.Errorf("include and exclude must not intersect, but found %s in both lists", allow)
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		// ensure namespaces are nested
 | |
| 	outer:
 | |
| 		for _, allow := range cl.Include {
 | |
| 			for _, deny := range cl.Exclude {
 | |
| 				if strings.HasPrefix(allow+".", deny+".") ||
 | |
| 					strings.HasPrefix(deny+".", allow+".") {
 | |
| 					continue outer
 | |
| 				}
 | |
| 			}
 | |
| 			return fmt.Errorf("when both include and exclude are populated, each element must be a superspace or subspace of one in the other list; check '%s' in include", allow)
 | |
| 		}
 | |
| 	}
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| func (cl *CustomLog) matchesModule(moduleID string) bool {
 | |
| 	return cl.loggerAllowed(moduleID, true)
 | |
| }
 | |
| 
 | |
| // loggerAllowed returns true if name is allowed to emit
 | |
| // to cl. isModule should be true if name is the name of
 | |
| // a module and you want to see if ANY of that module's
 | |
| // logs would be permitted.
 | |
| func (cl *CustomLog) loggerAllowed(name string, isModule bool) bool {
 | |
| 	// accept all loggers by default
 | |
| 	if len(cl.Include) == 0 && len(cl.Exclude) == 0 {
 | |
| 		return true
 | |
| 	}
 | |
| 
 | |
| 	// append a dot so that partial names don't match
 | |
| 	// (i.e. we don't want "foo.b" to match "foo.bar"); we
 | |
| 	// will also have to append a dot when we do HasPrefix
 | |
| 	// below to compensate for when namespaces are equal
 | |
| 	if name != "" && name != "*" && name != "." {
 | |
| 		name += "."
 | |
| 	}
 | |
| 
 | |
| 	var longestAccept, longestReject int
 | |
| 
 | |
| 	if len(cl.Include) > 0 {
 | |
| 		for _, namespace := range cl.Include {
 | |
| 			var hasPrefix bool
 | |
| 			if isModule {
 | |
| 				hasPrefix = strings.HasPrefix(namespace+".", name)
 | |
| 			} else {
 | |
| 				hasPrefix = strings.HasPrefix(name, namespace+".")
 | |
| 			}
 | |
| 			if hasPrefix && len(namespace) > longestAccept {
 | |
| 				longestAccept = len(namespace)
 | |
| 			}
 | |
| 		}
 | |
| 		// the include list was populated, meaning that
 | |
| 		// a match in this list is absolutely required
 | |
| 		// if we are to accept the entry
 | |
| 		if longestAccept == 0 {
 | |
| 			return false
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if len(cl.Exclude) > 0 {
 | |
| 		for _, namespace := range cl.Exclude {
 | |
| 			// * == all logs emitted by modules
 | |
| 			// . == all logs emitted by core
 | |
| 			if (namespace == "*" && name != ".") ||
 | |
| 				(namespace == "." && name == ".") {
 | |
| 				return false
 | |
| 			}
 | |
| 			if strings.HasPrefix(name, namespace+".") &&
 | |
| 				len(namespace) > longestReject {
 | |
| 				longestReject = len(namespace)
 | |
| 			}
 | |
| 		}
 | |
| 		// the reject list is populated, so we have to
 | |
| 		// reject this entry if its match is better
 | |
| 		// than the best from the accept list
 | |
| 		if longestReject > longestAccept {
 | |
| 			return false
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return (longestAccept > longestReject) ||
 | |
| 		(len(cl.Include) == 0 && longestReject == 0)
 | |
| }
 | |
| 
 | |
| // filteringCore filters log entries based on logger name,
 | |
| // according to the rules of a CustomLog.
 | |
| type filteringCore struct {
 | |
| 	zapcore.Core
 | |
| 	cl *CustomLog
 | |
| }
 | |
| 
 | |
| // With properly wraps With.
 | |
| func (fc *filteringCore) With(fields []zapcore.Field) zapcore.Core {
 | |
| 	return &filteringCore{
 | |
| 		Core: fc.Core.With(fields),
 | |
| 		cl:   fc.cl,
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // Check only allows the log entry if its logger name
 | |
| // is allowed from the include/exclude rules of fc.cl.
 | |
| func (fc *filteringCore) Check(e zapcore.Entry, ce *zapcore.CheckedEntry) *zapcore.CheckedEntry {
 | |
| 	if fc.cl.loggerAllowed(e.LoggerName, false) {
 | |
| 		return fc.Core.Check(e, ce)
 | |
| 	}
 | |
| 	return ce
 | |
| }
 | |
| 
 | |
| // LogSampling configures log entry sampling.
 | |
| type LogSampling struct {
 | |
| 	// The window over which to conduct sampling.
 | |
| 	Interval time.Duration `json:"interval,omitempty"`
 | |
| 
 | |
| 	// Log this many entries within a given level and
 | |
| 	// message for each interval.
 | |
| 	First int `json:"first,omitempty"`
 | |
| 
 | |
| 	// If more entries with the same level and message
 | |
| 	// are seen during the same interval, keep one in
 | |
| 	// this many entries until the end of the interval.
 | |
| 	Thereafter int `json:"thereafter,omitempty"`
 | |
| }
 | |
| 
 | |
| type (
 | |
| 	// StdoutWriter writes logs to standard out.
 | |
| 	StdoutWriter struct{}
 | |
| 
 | |
| 	// StderrWriter writes logs to standard error.
 | |
| 	StderrWriter struct{}
 | |
| 
 | |
| 	// DiscardWriter discards all writes.
 | |
| 	DiscardWriter struct{}
 | |
| )
 | |
| 
 | |
| // CaddyModule returns the Caddy module information.
 | |
| func (StdoutWriter) CaddyModule() ModuleInfo {
 | |
| 	return ModuleInfo{
 | |
| 		ID:  "caddy.logging.writers.stdout",
 | |
| 		New: func() Module { return new(StdoutWriter) },
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // CaddyModule returns the Caddy module information.
 | |
| func (StderrWriter) CaddyModule() ModuleInfo {
 | |
| 	return ModuleInfo{
 | |
| 		ID:  "caddy.logging.writers.stderr",
 | |
| 		New: func() Module { return new(StderrWriter) },
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // CaddyModule returns the Caddy module information.
 | |
| func (DiscardWriter) CaddyModule() ModuleInfo {
 | |
| 	return ModuleInfo{
 | |
| 		ID:  "caddy.logging.writers.discard",
 | |
| 		New: func() Module { return new(DiscardWriter) },
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func (StdoutWriter) String() string  { return "stdout" }
 | |
| func (StderrWriter) String() string  { return "stderr" }
 | |
| func (DiscardWriter) String() string { return "discard" }
 | |
| 
 | |
| // WriterKey returns a unique key representing stdout.
 | |
| func (StdoutWriter) WriterKey() string { return "std:out" }
 | |
| 
 | |
| // WriterKey returns a unique key representing stderr.
 | |
| func (StderrWriter) WriterKey() string { return "std:err" }
 | |
| 
 | |
| // WriterKey returns a unique key representing discard.
 | |
| func (DiscardWriter) WriterKey() string { return "discard" }
 | |
| 
 | |
| // OpenWriter returns os.Stdout that can't be closed.
 | |
| func (StdoutWriter) OpenWriter() (io.WriteCloser, error) {
 | |
| 	return notClosable{os.Stdout}, nil
 | |
| }
 | |
| 
 | |
| // OpenWriter returns os.Stderr that can't be closed.
 | |
| func (StderrWriter) OpenWriter() (io.WriteCloser, error) {
 | |
| 	return notClosable{os.Stderr}, nil
 | |
| }
 | |
| 
 | |
| // OpenWriter returns io.Discard that can't be closed.
 | |
| func (DiscardWriter) OpenWriter() (io.WriteCloser, error) {
 | |
| 	return notClosable{io.Discard}, nil
 | |
| }
 | |
| 
 | |
| // notClosable is an io.WriteCloser that can't be closed.
 | |
| type notClosable struct{ io.Writer }
 | |
| 
 | |
| func (fc notClosable) Close() error { return nil }
 | |
| 
 | |
| type defaultCustomLog struct {
 | |
| 	*CustomLog
 | |
| 	logger *zap.Logger
 | |
| }
 | |
| 
 | |
| // newDefaultProductionLog configures a custom log that is
 | |
| // intended for use by default if no other log is specified
 | |
| // in a config. It writes to stderr, uses the console encoder,
 | |
| // and enables INFO-level logs and higher.
 | |
| func newDefaultProductionLog() (*defaultCustomLog, error) {
 | |
| 	cl := new(CustomLog)
 | |
| 	cl.writerOpener = StderrWriter{}
 | |
| 	var err error
 | |
| 	cl.writer, err = cl.writerOpener.OpenWriter()
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 	cl.encoder = newDefaultProductionLogEncoder(cl.writerOpener)
 | |
| 	cl.levelEnabler = zapcore.InfoLevel
 | |
| 
 | |
| 	cl.buildCore()
 | |
| 
 | |
| 	logger := zap.New(cl.core)
 | |
| 
 | |
| 	// capture logs from other libraries which
 | |
| 	// may not be using zap logging directly
 | |
| 	_ = zap.RedirectStdLog(logger)
 | |
| 
 | |
| 	return &defaultCustomLog{
 | |
| 		CustomLog: cl,
 | |
| 		logger:    logger,
 | |
| 	}, nil
 | |
| }
 | |
| 
 | |
| func newDefaultProductionLogEncoder(wo WriterOpener) zapcore.Encoder {
 | |
| 	encCfg := zap.NewProductionEncoderConfig()
 | |
| 	if IsWriterStandardStream(wo) && term.IsTerminal(int(os.Stderr.Fd())) {
 | |
| 		// if interactive terminal, make output more human-readable by default
 | |
| 		encCfg.EncodeTime = func(ts time.Time, encoder zapcore.PrimitiveArrayEncoder) {
 | |
| 			encoder.AppendString(ts.UTC().Format("2006/01/02 15:04:05.000"))
 | |
| 		}
 | |
| 		if coloringEnabled {
 | |
| 			encCfg.EncodeLevel = zapcore.CapitalColorLevelEncoder
 | |
| 		}
 | |
| 
 | |
| 		return zapcore.NewConsoleEncoder(encCfg)
 | |
| 	}
 | |
| 	return zapcore.NewJSONEncoder(encCfg)
 | |
| }
 | |
| 
 | |
| func parseLevel(levelInput string) (zapcore.LevelEnabler, error) {
 | |
| 	repl := NewReplacer()
 | |
| 	level, err := repl.ReplaceOrErr(levelInput, true, true)
 | |
| 	if err != nil {
 | |
| 		return nil, fmt.Errorf("invalid log level: %v", err)
 | |
| 	}
 | |
| 	level = strings.ToLower(level)
 | |
| 
 | |
| 	// set up the log level
 | |
| 	switch level {
 | |
| 	case "debug":
 | |
| 		return zapcore.DebugLevel, nil
 | |
| 	case "", "info":
 | |
| 		return zapcore.InfoLevel, nil
 | |
| 	case "warn":
 | |
| 		return zapcore.WarnLevel, nil
 | |
| 	case "error":
 | |
| 		return zapcore.ErrorLevel, nil
 | |
| 	case "panic":
 | |
| 		return zapcore.PanicLevel, nil
 | |
| 	case "fatal":
 | |
| 		return zapcore.FatalLevel, nil
 | |
| 	default:
 | |
| 		return nil, fmt.Errorf("unrecognized log level: %s", level)
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // Log returns the current default logger.
 | |
| func Log() *zap.Logger {
 | |
| 	defaultLoggerMu.RLock()
 | |
| 	defer defaultLoggerMu.RUnlock()
 | |
| 	return defaultLogger.logger
 | |
| }
 | |
| 
 | |
| // BufferedLog sets the default logger to one that buffers
 | |
| // logs before a config is loaded.
 | |
| // Returns the buffered logger, the original default logger
 | |
| // (for flushing on errors), and the buffer core so that the
 | |
| // caller can flush the logs after the config is loaded or
 | |
| // fails to load.
 | |
| func BufferedLog() (*zap.Logger, *zap.Logger, *internal.LogBufferCore) {
 | |
| 	defaultLoggerMu.Lock()
 | |
| 	defer defaultLoggerMu.Unlock()
 | |
| 	origLogger := defaultLogger.logger
 | |
| 	bufferCore := internal.NewLogBufferCore(zap.InfoLevel)
 | |
| 	defaultLogger.logger = zap.New(bufferCore)
 | |
| 	return defaultLogger.logger, origLogger, bufferCore
 | |
| }
 | |
| 
 | |
| var (
 | |
| 	coloringEnabled  = os.Getenv("NO_COLOR") == "" && os.Getenv("TERM") != "xterm-mono"
 | |
| 	defaultLogger, _ = newDefaultProductionLog()
 | |
| 	defaultLoggerMu  sync.RWMutex
 | |
| )
 | |
| 
 | |
| var writers = NewUsagePool()
 | |
| 
 | |
| // ConfiguresFormatterDefault is an optional interface that
 | |
| // encoder modules can implement to configure the default
 | |
| // format of their encoder. This is useful for encoders
 | |
| // which nest an encoder, that needs to know the writer
 | |
| // in order to determine the correct default.
 | |
| type ConfiguresFormatterDefault interface {
 | |
| 	ConfigureDefaultFormat(WriterOpener) error
 | |
| }
 | |
| 
 | |
| const DefaultLoggerName = "default"
 | |
| 
 | |
| // Interface guards
 | |
| var (
 | |
| 	_ io.WriteCloser = (*notClosable)(nil)
 | |
| 	_ WriterOpener   = (*StdoutWriter)(nil)
 | |
| 	_ WriterOpener   = (*StderrWriter)(nil)
 | |
| )
 |