github.com/livekit/protocol@v1.16.1-0.20240517185851-47e4c6bba773/rpc/logging.go (about) 1 // Copyright 2023 LiveKit, Inc. 2 // 3 // Licensed under the Apache License, Version 2.0 (the "License"); 4 // you may not use this file except in compliance with the License. 5 // You may obtain a copy of the License at 6 // 7 // http://www.apache.org/licenses/LICENSE-2.0 8 // 9 // Unless required by applicable law or agreed to in writing, software 10 // distributed under the License is distributed on an "AS IS" BASIS, 11 // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 12 // See the License for the specific language governing permissions and 13 // limitations under the License. 14 15 package rpc 16 17 import ( 18 "context" 19 "time" 20 21 "github.com/puzpuzpuz/xsync/v3" 22 "google.golang.org/protobuf/proto" 23 24 "github.com/livekit/protocol/logger" 25 "github.com/livekit/psrpc" 26 ) 27 28 type loggerCache struct { 29 m *xsync.MapOf[string, logger.Logger] 30 } 31 32 func newLoggerCache() loggerCache { 33 return loggerCache{m: xsync.NewMapOf[string, logger.Logger]()} 34 } 35 36 func (c loggerCache) Get(info psrpc.RPCInfo, l logger.Logger) logger.Logger { 37 wl, _ := c.m.LoadOrCompute(info.Method, func() logger.Logger { 38 return l.WithComponent("psrpc").WithComponent(info.Service).WithComponent(info.Method) 39 }) 40 return wl 41 } 42 43 func WithClientLogger(logger logger.Logger) psrpc.ClientOption { 44 return psrpc.WithClientOptions( 45 psrpc.WithClientRPCInterceptors(newClientRPCLoggerInterceptor(logger)), 46 psrpc.WithClientMultiRPCInterceptors(newMultiRPCLoggerInterceptor(logger)), 47 psrpc.WithClientStreamInterceptors(newStreamLoggerInterceptor(logger)), 48 ) 49 } 50 51 func WithServerLogger(logger logger.Logger) psrpc.ServerOption { 52 return psrpc.WithServerOptions( 53 psrpc.WithServerRPCInterceptors(newServerRPCLoggerInterceptor(logger)), 54 psrpc.WithServerStreamInterceptors(newStreamLoggerInterceptor(logger)), 55 ) 56 } 57 58 func newClientRPCLoggerInterceptor(l logger.Logger) psrpc.ClientRPCInterceptor { 59 loggers := newLoggerCache() 60 return func(rpcInfo psrpc.RPCInfo, next psrpc.ClientRPCHandler) psrpc.ClientRPCHandler { 61 l := loggers.Get(rpcInfo, l) 62 return func(ctx context.Context, req proto.Message, opts ...psrpc.RequestOption) (res proto.Message, err error) { 63 start := time.Now() 64 defer func() { 65 if err != nil { 66 l.Warnw("client error", err, "topic", rpcInfo.Topic, "request", logger.Proto(req), "response", logger.Proto(res), "duration", time.Since(start)) 67 } else { 68 l.Debugw("client response", "topic", rpcInfo.Topic, "request", logger.Proto(req), "response", logger.Proto(res), "duration", time.Since(start)) 69 } 70 }() 71 return next(ctx, req, opts...) 72 } 73 } 74 } 75 76 func newServerRPCLoggerInterceptor(l logger.Logger) psrpc.ServerRPCInterceptor { 77 loggers := newLoggerCache() 78 return func(ctx context.Context, req proto.Message, rpcInfo psrpc.RPCInfo, handler psrpc.ServerRPCHandler) (res proto.Message, err error) { 79 l := loggers.Get(rpcInfo, l) 80 start := time.Now() 81 defer func() { 82 if err != nil { 83 l.Warnw("server error", err, "topic", rpcInfo.Topic, "request", logger.Proto(req), "response", logger.Proto(res), "duration", time.Since(start)) 84 } else { 85 l.Debugw("server response", "topic", rpcInfo.Topic, "request", logger.Proto(req), "response", logger.Proto(res), "duration", time.Since(start)) 86 } 87 }() 88 return handler(ctx, req) 89 } 90 } 91 92 func newStreamLoggerInterceptor(l logger.Logger) psrpc.StreamInterceptor { 93 loggers := newLoggerCache() 94 return func(rpcInfo psrpc.RPCInfo, next psrpc.StreamHandler) psrpc.StreamHandler { 95 l := loggers.Get(rpcInfo, l).WithValues("topic", rpcInfo.Topic) 96 l.Debugw("stream opened") 97 return &streamLoggerInterceptor{ 98 StreamHandler: next, 99 logger: l, 100 } 101 } 102 } 103 104 type streamLoggerInterceptor struct { 105 psrpc.StreamHandler 106 logger logger.Logger 107 } 108 109 func (s *streamLoggerInterceptor) Recv(msg proto.Message) (err error) { 110 s.logger.Debugw("received message", "message", logger.Proto(msg)) 111 return s.StreamHandler.Recv(msg) 112 } 113 114 func (s *streamLoggerInterceptor) Send(msg proto.Message, opts ...psrpc.StreamOption) (err error) { 115 start := time.Now() 116 defer func() { 117 if err != nil { 118 s.logger.Warnw("failed to send message", err, "message", logger.Proto(msg), "duration", time.Since(start)) 119 } else { 120 s.logger.Debugw("sent message", "message", logger.Proto(msg), "duration", time.Since(start)) 121 } 122 }() 123 return s.StreamHandler.Send(msg, opts...) 124 } 125 126 func (s *streamLoggerInterceptor) Close(cause error) error { 127 s.logger.Debugw("stream closed") 128 return s.StreamHandler.Close(cause) 129 } 130 131 func newMultiRPCLoggerInterceptor(l logger.Logger) psrpc.ClientMultiRPCInterceptor { 132 loggers := newLoggerCache() 133 return func(rpcInfo psrpc.RPCInfo, next psrpc.ClientMultiRPCHandler) psrpc.ClientMultiRPCHandler { 134 return &multiRPCLoggerInterceptor{ 135 ClientMultiRPCHandler: next, 136 logger: loggers.Get(rpcInfo, l).WithValues("topic", rpcInfo.Topic), 137 start: time.Now(), 138 } 139 } 140 } 141 142 type multiRPCLoggerInterceptor struct { 143 psrpc.ClientMultiRPCHandler 144 logger logger.Logger 145 start time.Time 146 responseCount int 147 errorCount int 148 } 149 150 func (r *multiRPCLoggerInterceptor) Send(ctx context.Context, req proto.Message, opts ...psrpc.RequestOption) error { 151 r.start = time.Now() 152 r.logger.Debugw("multirpc opened", "request", logger.Proto(req)) 153 return r.ClientMultiRPCHandler.Send(ctx, req, opts...) 154 } 155 156 func (r *multiRPCLoggerInterceptor) Recv(msg proto.Message, err error) { 157 if err != nil { 158 r.logger.Warnw("received error", err) 159 r.errorCount++ 160 } else { 161 r.logger.Debugw("received response", "response", logger.Proto(msg)) 162 r.responseCount++ 163 } 164 r.ClientMultiRPCHandler.Recv(msg, err) 165 } 166 167 func (r *multiRPCLoggerInterceptor) Close() { 168 r.logger.Debugw("multirpc closed", "responseCount", r.responseCount, "errorCount", r.errorCount) 169 r.ClientMultiRPCHandler.Close() 170 }