mirror of
https://gitee.com/milvus-io/milvus.git
synced 2024-12-04 12:59:23 +08:00
875ad88d84
relate: https://github.com/milvus-io/milvus/issues/28948 Signed-off-by: aoiasd <zhicheng.yue@zilliz.com>
205 lines
5.1 KiB
Go
205 lines
5.1 KiB
Go
// Licensed to the LF AI & Data foundation under one
|
|
// or more contributor license agreements. See the NOTICE file
|
|
// distributed with this work for additional information
|
|
// regarding copyright ownership. The ASF licenses this file
|
|
// to you 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 accesslog
|
|
|
|
import (
|
|
"io"
|
|
"strconv"
|
|
"sync"
|
|
"time"
|
|
|
|
"go.uber.org/atomic"
|
|
"go.uber.org/zap"
|
|
|
|
"github.com/milvus-io/milvus/internal/proxy/accesslog/info"
|
|
configEvent "github.com/milvus-io/milvus/pkg/config"
|
|
"github.com/milvus-io/milvus/pkg/log"
|
|
"github.com/milvus-io/milvus/pkg/util/paramtable"
|
|
)
|
|
|
|
var (
|
|
_globalL *AccessLogger
|
|
once sync.Once
|
|
)
|
|
|
|
type AccessLogger struct {
|
|
enable atomic.Bool
|
|
writer io.Writer
|
|
formatters *FormatterManger
|
|
mu sync.RWMutex
|
|
}
|
|
|
|
func NewAccessLogger() *AccessLogger {
|
|
return &AccessLogger{}
|
|
}
|
|
|
|
func (l *AccessLogger) init(params *paramtable.ComponentParam) error {
|
|
formatters, err := initFormatter(¶ms.ProxyCfg.AccessLog)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
l.formatters = formatters
|
|
|
|
writer, err := initWriter(¶ms.ProxyCfg.AccessLog, ¶ms.MinioCfg)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
l.writer = writer
|
|
return nil
|
|
}
|
|
|
|
func (l *AccessLogger) Init(params *paramtable.ComponentParam) error {
|
|
if params.ProxyCfg.AccessLog.Enable.GetAsBool() {
|
|
l.mu.Lock()
|
|
defer l.mu.Unlock()
|
|
|
|
err := l.init(params)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
l.enable.Store(true)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (l *AccessLogger) SetEnable(enable bool) error {
|
|
l.mu.Lock()
|
|
defer l.mu.Unlock()
|
|
|
|
if l.enable.Load() == enable {
|
|
return nil
|
|
}
|
|
|
|
if enable {
|
|
log.Info("start enable access log")
|
|
params := paramtable.Get()
|
|
err := l.init(params)
|
|
if err != nil {
|
|
log.Warn("enable access log failed", zap.Error(err))
|
|
return err
|
|
}
|
|
} else {
|
|
log.Info("start close access log")
|
|
if write, ok := l.writer.(*RotateWriter); ok {
|
|
write.Close()
|
|
}
|
|
}
|
|
|
|
l.enable.Store(enable)
|
|
return nil
|
|
}
|
|
|
|
func (l *AccessLogger) Write(info info.AccessInfo) bool {
|
|
if !l.enable.Load() {
|
|
return false
|
|
}
|
|
|
|
l.mu.RLock()
|
|
defer l.mu.RUnlock()
|
|
|
|
method := info.MethodName()
|
|
formatter, ok := l.formatters.GetByMethod(method)
|
|
if !ok {
|
|
return false
|
|
}
|
|
_, err := l.writer.Write([]byte(formatter.Format(info)))
|
|
if err != nil {
|
|
log.Warn("write access log failed", zap.Error(err))
|
|
return false
|
|
}
|
|
return true
|
|
}
|
|
|
|
func InitAccessLogger(params *paramtable.ComponentParam) {
|
|
once.Do(func() {
|
|
logger := NewAccessLogger()
|
|
// support dynamic param
|
|
params.Watch(params.ProxyCfg.AccessLog.Enable.Key, configEvent.NewHandler("enable accesslog", func(event *configEvent.Event) {
|
|
value, err := strconv.ParseBool(event.Value)
|
|
if err != nil {
|
|
log.Warn("Failed to parse bool value", zap.String("v", event.Value), zap.Error(err))
|
|
return
|
|
}
|
|
logger.SetEnable(value)
|
|
}))
|
|
|
|
err := logger.Init(params)
|
|
if err != nil {
|
|
log.Warn("Init access logger failed", zap.Error(err))
|
|
}
|
|
_globalL = logger
|
|
info.ClusterPrefix.Store(params.CommonCfg.ClusterPrefix.GetValue())
|
|
log.Info("Init access logger success")
|
|
})
|
|
}
|
|
|
|
func initFormatter(logCfg *paramtable.AccessLogConfig) (*FormatterManger, error) {
|
|
formatterManger := NewFormatterManger()
|
|
formatMap := make(map[string]string) // fommatter name -> formatter format
|
|
methodMap := make(map[string][]string) // fommatter name -> formatter owner method
|
|
for key, value := range logCfg.Formatter.GetValue() {
|
|
formatterName, option, err := parseConfigKey(key)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if option == fomaterkey {
|
|
formatMap[formatterName] = value
|
|
} else if option == methodKey {
|
|
methodMap[formatterName] = paramtable.ParseAsStings(value)
|
|
}
|
|
}
|
|
|
|
for name, format := range formatMap {
|
|
formatterManger.Add(name, format)
|
|
if methods, ok := methodMap[name]; ok {
|
|
formatterManger.SetMethod(name, methods...)
|
|
}
|
|
}
|
|
|
|
return formatterManger, nil
|
|
}
|
|
|
|
// initAccessLogger initializes a zap access logger for proxy
|
|
func initWriter(logCfg *paramtable.AccessLogConfig, minioCfg *paramtable.MinioConfig) (io.Writer, error) {
|
|
if len(logCfg.Filename.GetValue()) > 0 {
|
|
lg, err := NewRotateWriter(logCfg, minioCfg)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if logCfg.CacheSize.GetAsInt() > 0 {
|
|
clg := NewCacheWriterWithCloser(lg, lg, logCfg.CacheSize.GetAsInt(), logCfg.CacheFlushInterval.GetAsDuration(time.Second))
|
|
return clg, nil
|
|
}
|
|
return lg, nil
|
|
}
|
|
|
|
// wirte to stdout when filename = ""
|
|
stdout, _, err := zap.Open([]string{"stdout"}...)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if logCfg.CacheSize.GetAsInt() > 0 {
|
|
lg := NewCacheWriter(stdout, logCfg.CacheSize.GetAsInt(), logCfg.CacheFlushInterval.GetAsDuration(time.Second))
|
|
return lg, nil
|
|
}
|
|
|
|
return stdout, nil
|
|
}
|