| 
									
										
										
										
											2021-04-19 03:41:13 +08:00
										 |  |  | // Copyright (c) 2015-2021 MinIO, Inc.
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // This file is part of MinIO Object Storage stack
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // This program is free software: you can redistribute it and/or modify
 | 
					
						
							|  |  |  | // it under the terms of the GNU Affero General Public License as published by
 | 
					
						
							|  |  |  | // the Free Software Foundation, either version 3 of the License, or
 | 
					
						
							|  |  |  | // (at your option) any later version.
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // This program is distributed in the hope that it will be useful
 | 
					
						
							|  |  |  | // but WITHOUT ANY WARRANTY; without even the implied warranty of
 | 
					
						
							|  |  |  | // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | 
					
						
							|  |  |  | // GNU Affero General Public License for more details.
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // You should have received a copy of the GNU Affero General Public License
 | 
					
						
							|  |  |  | // along with this program.  If not, see <http://www.gnu.org/licenses/>.
 | 
					
						
							| 
									
										
										
										
											2018-05-10 06:59:45 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | package logger | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"context" | 
					
						
							| 
									
										
										
										
											2022-07-28 00:44:59 +08:00
										 |  |  | 	"errors" | 
					
						
							| 
									
										
										
										
											2018-05-10 06:59:45 +08:00
										 |  |  | 	"sync" | 
					
						
							|  |  |  | 	"time" | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-28 00:44:59 +08:00
										 |  |  | // LogOnce provides the function type for logger.LogOnceIf() function
 | 
					
						
							|  |  |  | type LogOnce func(ctx context.Context, err error, id string, errKind ...interface{}) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-10 06:59:45 +08:00
										 |  |  | // Holds a map of recently logged errors.
 | 
					
						
							|  |  |  | type logOnceType struct { | 
					
						
							| 
									
										
										
										
											2022-07-28 00:44:59 +08:00
										 |  |  | 	IDMap map[string]error | 
					
						
							| 
									
										
										
										
											2018-05-10 06:59:45 +08:00
										 |  |  | 	sync.Mutex | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-28 00:44:59 +08:00
										 |  |  | func (l *logOnceType) logOnceConsoleIf(ctx context.Context, err error, id string, errKind ...interface{}) { | 
					
						
							| 
									
										
										
										
											2022-05-12 22:20:58 +08:00
										 |  |  | 	if err == nil { | 
					
						
							|  |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2022-07-28 00:44:59 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	nerr := unwrapErrs(err) | 
					
						
							| 
									
										
										
										
											2022-05-12 22:20:58 +08:00
										 |  |  | 	l.Lock() | 
					
						
							| 
									
										
										
										
											2022-07-28 00:44:59 +08:00
										 |  |  | 	shouldLog := true | 
					
						
							|  |  |  | 	prevErr, ok := l.IDMap[id] | 
					
						
							|  |  |  | 	if !ok { | 
					
						
							|  |  |  | 		l.IDMap[id] = nerr | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		// if errors are equal do not log.
 | 
					
						
							|  |  |  | 		shouldLog = prevErr.Error() != nerr.Error() | 
					
						
							| 
									
										
										
										
											2022-05-12 22:20:58 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	l.Unlock() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if shouldLog { | 
					
						
							|  |  |  | 		consoleLogIf(ctx, err, errKind...) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-28 00:44:59 +08:00
										 |  |  | const unwrapErrsDepth = 3 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // unwrapErrs upto the point where errors.Unwrap(err) returns nil
 | 
					
						
							|  |  |  | func unwrapErrs(err error) (leafErr error) { | 
					
						
							|  |  |  | 	uerr := errors.Unwrap(err) | 
					
						
							|  |  |  | 	depth := 1 | 
					
						
							|  |  |  | 	for uerr != nil { | 
					
						
							|  |  |  | 		// Save the current `uerr`
 | 
					
						
							|  |  |  | 		leafErr = uerr | 
					
						
							|  |  |  | 		// continue to look for leaf errors underneath
 | 
					
						
							|  |  |  | 		uerr = errors.Unwrap(leafErr) | 
					
						
							|  |  |  | 		depth++ | 
					
						
							|  |  |  | 		if depth == unwrapErrsDepth { | 
					
						
							|  |  |  | 			// If we have reached enough depth we
 | 
					
						
							|  |  |  | 			// do not further recurse down, this
 | 
					
						
							|  |  |  | 			// is done to avoid any unnecessary
 | 
					
						
							|  |  |  | 			// latencies this might bring.
 | 
					
						
							|  |  |  | 			break | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2022-08-03 06:10:11 +08:00
										 |  |  | 	if uerr == nil { | 
					
						
							|  |  |  | 		leafErr = err | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2022-07-28 00:44:59 +08:00
										 |  |  | 	return leafErr | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-10 06:59:45 +08:00
										 |  |  | // One log message per error.
 | 
					
						
							| 
									
										
										
										
											2022-07-28 00:44:59 +08:00
										 |  |  | func (l *logOnceType) logOnceIf(ctx context.Context, err error, id string, errKind ...interface{}) { | 
					
						
							| 
									
										
										
										
											2018-05-10 06:59:45 +08:00
										 |  |  | 	if err == nil { | 
					
						
							|  |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2022-07-28 00:44:59 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	nerr := unwrapErrs(err) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-10 06:59:45 +08:00
										 |  |  | 	l.Lock() | 
					
						
							| 
									
										
										
										
											2022-07-28 00:44:59 +08:00
										 |  |  | 	shouldLog := true | 
					
						
							|  |  |  | 	prevErr, ok := l.IDMap[id] | 
					
						
							|  |  |  | 	if !ok { | 
					
						
							|  |  |  | 		l.IDMap[id] = nerr | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		// if errors are equal do not log.
 | 
					
						
							|  |  |  | 		shouldLog = prevErr.Error() != nerr.Error() | 
					
						
							| 
									
										
										
										
											2018-05-10 06:59:45 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	l.Unlock() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if shouldLog { | 
					
						
							| 
									
										
										
										
											2019-10-12 09:50:54 +08:00
										 |  |  | 		LogIf(ctx, err, errKind...) | 
					
						
							| 
									
										
										
										
											2018-05-10 06:59:45 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Cleanup the map every 30 minutes so that the log message is printed again for the user to notice.
 | 
					
						
							|  |  |  | func (l *logOnceType) cleanupRoutine() { | 
					
						
							|  |  |  | 	for { | 
					
						
							| 
									
										
										
										
											2019-02-13 20:59:36 +08:00
										 |  |  | 		l.Lock() | 
					
						
							| 
									
										
										
										
											2022-07-28 00:44:59 +08:00
										 |  |  | 		l.IDMap = make(map[string]error) | 
					
						
							| 
									
										
										
										
											2019-02-13 20:59:36 +08:00
										 |  |  | 		l.Unlock() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		time.Sleep(30 * time.Minute) | 
					
						
							| 
									
										
										
										
											2018-05-10 06:59:45 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Returns logOnceType
 | 
					
						
							|  |  |  | func newLogOnceType() *logOnceType { | 
					
						
							| 
									
										
										
										
											2022-07-28 00:44:59 +08:00
										 |  |  | 	l := &logOnceType{IDMap: make(map[string]error)} | 
					
						
							| 
									
										
										
										
											2018-05-10 06:59:45 +08:00
										 |  |  | 	go l.cleanupRoutine() | 
					
						
							|  |  |  | 	return l | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | var logOnce = newLogOnceType() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // LogOnceIf - Logs notification errors - once per error.
 | 
					
						
							|  |  |  | // id is a unique identifier for related log messages, refer to cmd/notification.go
 | 
					
						
							|  |  |  | // on how it is used.
 | 
					
						
							| 
									
										
										
										
											2022-07-28 00:44:59 +08:00
										 |  |  | func LogOnceIf(ctx context.Context, err error, id string, errKind ...interface{}) { | 
					
						
							| 
									
										
										
										
											2022-07-26 08:53:03 +08:00
										 |  |  | 	if logIgnoreError(err) { | 
					
						
							| 
									
										
										
										
											2020-10-31 05:55:50 +08:00
										 |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-10-12 09:50:54 +08:00
										 |  |  | 	logOnce.logOnceIf(ctx, err, id, errKind...) | 
					
						
							| 
									
										
										
										
											2018-05-10 06:59:45 +08:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2022-05-12 22:20:58 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | // LogOnceConsoleIf - similar to LogOnceIf but exclusively only logs to console target.
 | 
					
						
							| 
									
										
										
										
											2022-07-28 00:44:59 +08:00
										 |  |  | func LogOnceConsoleIf(ctx context.Context, err error, id string, errKind ...interface{}) { | 
					
						
							| 
									
										
										
										
											2022-07-26 08:53:03 +08:00
										 |  |  | 	if logIgnoreError(err) { | 
					
						
							| 
									
										
										
										
											2022-05-12 22:20:58 +08:00
										 |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	logOnce.logOnceConsoleIf(ctx, err, id, errKind...) | 
					
						
							|  |  |  | } |