add non-block write log in asynchronous mode (#5150)
* add non-block write log in asynchronous mode --------- Co-authored-by: chenhaokun <chenhaokun@itiger.com>
This commit is contained in:
		
							parent
							
								
									fb76377a3e
								
							
						
					
					
						commit
						f1dea5b811
					
				| @ -1,4 +1,5 @@ | ||||
| # developing | ||||
| - [add non-block write log in asynchronous mode](https://github.com/beego/beego/pull/5150) | ||||
| - [Fix 5126: support bloom filter cache](https://github.com/beego/beego/pull/5126) | ||||
| - [Fix 5117: support write though cache](https://github.com/beego/beego/pull/5117) | ||||
| - [add read through for cache module](https://github.com/beego/beego/pull/5116) | ||||
|  | ||||
| @ -29,7 +29,6 @@ | ||||
| //	log.Warn("warning") | ||||
| //	log.Debug("debug") | ||||
| //	log.Critical("critical") | ||||
| // | ||||
| package logs | ||||
| 
 | ||||
| import ( | ||||
| @ -115,6 +114,9 @@ type BeeLogger struct { | ||||
| 	enableFuncCallDepth bool | ||||
| 	enableFullFilePath  bool | ||||
| 	asynchronous        bool | ||||
| 	// Whether to discard logs when buffer is full and asynchronous is true | ||||
| 	// No discard by default | ||||
| 	logWithNonBlocking  bool | ||||
| 	wg                  sync.WaitGroup | ||||
| 	level               int | ||||
| 	loggerFuncCallDepth int | ||||
| @ -175,6 +177,16 @@ func (bl *BeeLogger) Async(msgLen ...int64) *BeeLogger { | ||||
| 	return bl | ||||
| } | ||||
| 
 | ||||
| // AsyncNonBlockWrite Non-blocking write in asynchronous mode | ||||
| // Only works if asynchronous write logging is set | ||||
| func (bl *BeeLogger) AsyncNonBlockWrite() *BeeLogger { | ||||
| 	if !bl.asynchronous { | ||||
| 		return bl | ||||
| 	} | ||||
| 	bl.logWithNonBlocking = true | ||||
| 	return bl | ||||
| } | ||||
| 
 | ||||
| // SetLogger provides a given logger adapter into BeeLogger with config string. | ||||
| // config must in in JSON format like {"interval":360}} | ||||
| func (bl *BeeLogger) setLogger(adapterName string, configs ...string) error { | ||||
| @ -312,8 +324,17 @@ func (bl *BeeLogger) writeMsg(lm *LogMsg) error { | ||||
| 		logM.FilePath = lm.FilePath | ||||
| 		logM.LineNumber = lm.LineNumber | ||||
| 		logM.Prefix = lm.Prefix | ||||
| 
 | ||||
| 		if bl.outputs != nil { | ||||
| 			bl.msgChan <- lm | ||||
| 			if bl.logWithNonBlocking { | ||||
| 				select { | ||||
| 				case bl.msgChan <- lm: | ||||
| 				// discard log when channel is full | ||||
| 				default: | ||||
| 				} | ||||
| 			} else { | ||||
| 				bl.msgChan <- lm | ||||
| 			} | ||||
| 		} else { | ||||
| 			logMsgPool.Put(lm) | ||||
| 		} | ||||
|  | ||||
| @ -15,7 +15,11 @@ | ||||
| package logs | ||||
| 
 | ||||
| import ( | ||||
| 	"encoding/json" | ||||
| 	"fmt" | ||||
| 	"io" | ||||
| 	"testing" | ||||
| 	"time" | ||||
| 
 | ||||
| 	"github.com/stretchr/testify/assert" | ||||
| ) | ||||
| @ -30,3 +34,95 @@ func TestBeeLoggerDelLogger(t *testing.T) { | ||||
| 	SetPrefix("aaa") | ||||
| 	Info("hello") | ||||
| } | ||||
| 
 | ||||
| type mockLogger struct { | ||||
| 	*logWriter | ||||
| 	WriteCost time.Duration `json:"write_cost"` // Simulated log writing time consuming | ||||
| 	writeCnt  int           // Count add 1 when writing log success, just for test result | ||||
| } | ||||
| 
 | ||||
| func NewMockLogger() Logger { | ||||
| 	return &mockLogger{ | ||||
| 		logWriter: &logWriter{writer: io.Discard}, | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| func (m *mockLogger) Init(config string) error { | ||||
| 	return json.Unmarshal([]byte(config), m) | ||||
| } | ||||
| 
 | ||||
| func (m *mockLogger) WriteMsg(lm *LogMsg) error { | ||||
| 	m.Lock() | ||||
| 	msg := lm.Msg | ||||
| 	msg += "\n" | ||||
| 
 | ||||
| 	time.Sleep(m.WriteCost) | ||||
| 	if _, err := m.writer.Write([]byte(msg)); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 
 | ||||
| 	m.writeCnt++ | ||||
| 	m.Unlock() | ||||
| 	return nil | ||||
| } | ||||
| 
 | ||||
| func (m *mockLogger) GetCnt() int { | ||||
| 	return m.writeCnt | ||||
| } | ||||
| 
 | ||||
| func (*mockLogger) Destroy()                    {} | ||||
| func (*mockLogger) Flush()                      {} | ||||
| func (*mockLogger) SetFormatter(_ LogFormatter) {} | ||||
| 
 | ||||
| func TestBeeLogger_AsyncNonBlockWrite(t *testing.T) { | ||||
| 	testCases := []struct { | ||||
| 		name         string | ||||
| 		before       func() | ||||
| 		after        func() | ||||
| 		msgLen       int64 | ||||
| 		writeCost    time.Duration | ||||
| 		sendInterval time.Duration | ||||
| 		writeCnt     int | ||||
| 	}{ | ||||
| 		{ | ||||
| 			// Write log time is less than send log time, no blocking | ||||
| 			name: "mock1", | ||||
| 			after: func() { | ||||
| 				_ = beeLogger.DelLogger("mock1") | ||||
| 			}, | ||||
| 			msgLen:       5, | ||||
| 			writeCnt:     10, | ||||
| 			writeCost:    200 * time.Millisecond, | ||||
| 			sendInterval: 300 * time.Millisecond, | ||||
| 		}, | ||||
| 		{ | ||||
| 			// Write log time is less than send log time, discarded when blocking | ||||
| 			name: "mock2", | ||||
| 			after: func() { | ||||
| 				_ = beeLogger.DelLogger("mock2") | ||||
| 			}, | ||||
| 			writeCnt:     5, | ||||
| 			msgLen:       5, | ||||
| 			writeCost:    200 * time.Millisecond, | ||||
| 			sendInterval: 10 * time.Millisecond, | ||||
| 		}, | ||||
| 	} | ||||
| 	for _, tc := range testCases { | ||||
| 		t.Run(tc.name, func(t *testing.T) { | ||||
| 			Register(tc.name, NewMockLogger) | ||||
| 			err := beeLogger.SetLogger(tc.name, fmt.Sprintf(`{"write_cost": %d}`, tc.writeCost)) | ||||
| 			assert.Nil(t, err) | ||||
| 
 | ||||
| 			l := beeLogger.Async(tc.msgLen) | ||||
| 			l.AsyncNonBlockWrite() | ||||
| 
 | ||||
| 			for i := 0; i < 10; i++ { | ||||
| 				time.Sleep(tc.sendInterval) | ||||
| 				l.Info(fmt.Sprintf("----%d----", i)) | ||||
| 			} | ||||
| 			time.Sleep(1 * time.Second) | ||||
| 			assert.Equal(t, tc.writeCnt, l.outputs[0].Logger.(*mockLogger).writeCnt) | ||||
| 			tc.after() | ||||
| 		}) | ||||
| 	} | ||||
| } | ||||
|  | ||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user