|
|
@@ -16,6 +16,12 @@ import (
|
|
16
|
16
|
"github.com/stretchr/testify/suite"
|
|
17
|
17
|
)
|
|
18
|
18
|
|
|
|
19
|
+type netTimeoutError struct{}
|
|
|
20
|
+
|
|
|
21
|
+func (e netTimeoutError) Error() string { return "i/o timeout" }
|
|
|
22
|
+func (e netTimeoutError) Timeout() bool { return true }
|
|
|
23
|
+func (e netTimeoutError) Temporary() bool { return true }
|
|
|
24
|
+
|
|
19
|
25
|
type ConnRewindBaseConn struct {
|
|
20
|
26
|
testlib.EssentialsConnMock
|
|
21
|
27
|
|
|
|
@@ -291,6 +297,141 @@ func (suite *ConnProxyProtocolTestSuite) TearDownTest() {
|
|
291
|
297
|
suite.targetConnMock.AssertExpectations(suite.T())
|
|
292
|
298
|
}
|
|
293
|
299
|
|
|
|
300
|
+type IdleTrackerTestSuite struct {
|
|
|
301
|
+ suite.Suite
|
|
|
302
|
+}
|
|
|
303
|
+
|
|
|
304
|
+func (suite *IdleTrackerTestSuite) TestNewNotIdle() {
|
|
|
305
|
+ tracker := newIdleTracker(time.Second)
|
|
|
306
|
+ suite.False(tracker.isIdle())
|
|
|
307
|
+}
|
|
|
308
|
+
|
|
|
309
|
+func (suite *IdleTrackerTestSuite) TestIdleAfterTimeout() {
|
|
|
310
|
+ tracker := newIdleTracker(10 * time.Millisecond)
|
|
|
311
|
+ time.Sleep(20 * time.Millisecond)
|
|
|
312
|
+
|
|
|
313
|
+ suite.True(tracker.isIdle())
|
|
|
314
|
+}
|
|
|
315
|
+
|
|
|
316
|
+func (suite *IdleTrackerTestSuite) TestTouchResetsIdle() {
|
|
|
317
|
+ tracker := newIdleTracker(50 * time.Millisecond)
|
|
|
318
|
+ time.Sleep(30 * time.Millisecond)
|
|
|
319
|
+
|
|
|
320
|
+ tracker.touch()
|
|
|
321
|
+
|
|
|
322
|
+ suite.False(tracker.isIdle())
|
|
|
323
|
+}
|
|
|
324
|
+
|
|
|
325
|
+type ConnIdleTimeoutTestSuite struct {
|
|
|
326
|
+ suite.Suite
|
|
|
327
|
+
|
|
|
328
|
+ connMock *testlib.EssentialsConnMock
|
|
|
329
|
+ tracker *idleTracker
|
|
|
330
|
+ conn connIdleTimeout
|
|
|
331
|
+}
|
|
|
332
|
+
|
|
|
333
|
+func (suite *ConnIdleTimeoutTestSuite) SetupTest() {
|
|
|
334
|
+ suite.connMock = &testlib.EssentialsConnMock{}
|
|
|
335
|
+ suite.tracker = newIdleTracker(time.Second)
|
|
|
336
|
+ suite.conn = connIdleTimeout{
|
|
|
337
|
+ Conn: suite.connMock,
|
|
|
338
|
+ tracker: suite.tracker,
|
|
|
339
|
+ }
|
|
|
340
|
+}
|
|
|
341
|
+
|
|
|
342
|
+func (suite *ConnIdleTimeoutTestSuite) TearDownTest() {
|
|
|
343
|
+ suite.connMock.AssertExpectations(suite.T())
|
|
|
344
|
+}
|
|
|
345
|
+
|
|
|
346
|
+func (suite *ConnIdleTimeoutTestSuite) TestReadOk() {
|
|
|
347
|
+ suite.connMock.On("SetReadDeadline", mock.Anything).Return(nil)
|
|
|
348
|
+ suite.connMock.On("Read", mock.Anything).Once().Return(5, nil)
|
|
|
349
|
+
|
|
|
350
|
+ n, err := suite.conn.Read(make([]byte, 10))
|
|
|
351
|
+ suite.NoError(err)
|
|
|
352
|
+ suite.Equal(5, n)
|
|
|
353
|
+}
|
|
|
354
|
+
|
|
|
355
|
+func (suite *ConnIdleTimeoutTestSuite) TestReadNonTimeoutErr() {
|
|
|
356
|
+ suite.connMock.On("SetReadDeadline", mock.Anything).Return(nil)
|
|
|
357
|
+ suite.connMock.On("Read", mock.Anything).Once().Return(0, io.EOF)
|
|
|
358
|
+
|
|
|
359
|
+ n, err := suite.conn.Read(make([]byte, 10))
|
|
|
360
|
+ suite.True(errors.Is(err, io.EOF))
|
|
|
361
|
+ suite.Equal(0, n)
|
|
|
362
|
+}
|
|
|
363
|
+
|
|
|
364
|
+func (suite *ConnIdleTimeoutTestSuite) TestReadTimeoutRetriesWhenNotIdle() {
|
|
|
365
|
+ suite.connMock.On("SetReadDeadline", mock.Anything).Return(nil)
|
|
|
366
|
+ suite.connMock.On("Read", mock.Anything).Once().Return(0, netTimeoutError{})
|
|
|
367
|
+ suite.connMock.On("Read", mock.Anything).Once().Return(5, nil)
|
|
|
368
|
+
|
|
|
369
|
+ n, err := suite.conn.Read(make([]byte, 10))
|
|
|
370
|
+ suite.NoError(err)
|
|
|
371
|
+ suite.Equal(5, n)
|
|
|
372
|
+}
|
|
|
373
|
+
|
|
|
374
|
+func (suite *ConnIdleTimeoutTestSuite) TestReadTimeoutClosesWhenIdle() {
|
|
|
375
|
+ suite.tracker = newIdleTracker(time.Millisecond)
|
|
|
376
|
+ suite.conn = connIdleTimeout{
|
|
|
377
|
+ Conn: suite.connMock,
|
|
|
378
|
+ tracker: suite.tracker,
|
|
|
379
|
+ }
|
|
|
380
|
+
|
|
|
381
|
+ time.Sleep(5 * time.Millisecond)
|
|
|
382
|
+
|
|
|
383
|
+ suite.connMock.On("SetReadDeadline", mock.Anything).Return(nil)
|
|
|
384
|
+ suite.connMock.On("Read", mock.Anything).Once().Return(0, netTimeoutError{})
|
|
|
385
|
+
|
|
|
386
|
+ n, err := suite.conn.Read(make([]byte, 10))
|
|
|
387
|
+ suite.Equal(0, n)
|
|
|
388
|
+
|
|
|
389
|
+ netErr, ok := err.(net.Error) //nolint: errorlint
|
|
|
390
|
+ suite.True(ok)
|
|
|
391
|
+ suite.True(netErr.Timeout())
|
|
|
392
|
+}
|
|
|
393
|
+
|
|
|
394
|
+func (suite *ConnIdleTimeoutTestSuite) TestSharedTrackerPreventsFalseTimeout() {
|
|
|
395
|
+ connMock2 := &testlib.EssentialsConnMock{}
|
|
|
396
|
+ conn2 := connIdleTimeout{
|
|
|
397
|
+ Conn: connMock2,
|
|
|
398
|
+ tracker: suite.tracker,
|
|
|
399
|
+ }
|
|
|
400
|
+
|
|
|
401
|
+ connMock2.On("SetWriteDeadline", mock.Anything).Return(nil)
|
|
|
402
|
+ connMock2.On("Write", mock.Anything).Once().Return(5, nil)
|
|
|
403
|
+
|
|
|
404
|
+ _, _ = conn2.Write(make([]byte, 5))
|
|
|
405
|
+
|
|
|
406
|
+ suite.connMock.On("SetReadDeadline", mock.Anything).Return(nil)
|
|
|
407
|
+ suite.connMock.On("Read", mock.Anything).Once().Return(0, netTimeoutError{})
|
|
|
408
|
+ suite.connMock.On("Read", mock.Anything).Once().Return(3, nil)
|
|
|
409
|
+
|
|
|
410
|
+ n, err := suite.conn.Read(make([]byte, 10))
|
|
|
411
|
+ suite.NoError(err)
|
|
|
412
|
+ suite.Equal(3, n)
|
|
|
413
|
+
|
|
|
414
|
+ connMock2.AssertExpectations(suite.T())
|
|
|
415
|
+}
|
|
|
416
|
+
|
|
|
417
|
+func (suite *ConnIdleTimeoutTestSuite) TestWriteOk() {
|
|
|
418
|
+ suite.connMock.On("SetWriteDeadline", mock.Anything).Return(nil)
|
|
|
419
|
+ suite.connMock.On("Write", mock.Anything).Once().Return(5, nil)
|
|
|
420
|
+
|
|
|
421
|
+ n, err := suite.conn.Write(make([]byte, 5))
|
|
|
422
|
+ suite.NoError(err)
|
|
|
423
|
+ suite.Equal(5, n)
|
|
|
424
|
+}
|
|
|
425
|
+
|
|
|
426
|
+func (suite *ConnIdleTimeoutTestSuite) TestWriteErr() {
|
|
|
427
|
+ suite.connMock.On("SetWriteDeadline", mock.Anything).Return(nil)
|
|
|
428
|
+ suite.connMock.On("Write", mock.Anything).Once().Return(0, io.EOF)
|
|
|
429
|
+
|
|
|
430
|
+ n, err := suite.conn.Write(make([]byte, 5))
|
|
|
431
|
+ suite.True(errors.Is(err, io.EOF))
|
|
|
432
|
+ suite.Equal(0, n)
|
|
|
433
|
+}
|
|
|
434
|
+
|
|
294
|
435
|
func TestConnTraffic(t *testing.T) {
|
|
295
|
436
|
t.Parallel()
|
|
296
|
437
|
suite.Run(t, &ConnTrafficTestSuite{})
|
|
|
@@ -305,3 +446,13 @@ func TestConnProxyProtocol(t *testing.T) {
|
|
305
|
446
|
t.Parallel()
|
|
306
|
447
|
suite.Run(t, &ConnProxyProtocolTestSuite{})
|
|
307
|
448
|
}
|
|
|
449
|
+
|
|
|
450
|
+func TestIdleTracker(t *testing.T) {
|
|
|
451
|
+ t.Parallel()
|
|
|
452
|
+ suite.Run(t, &IdleTrackerTestSuite{})
|
|
|
453
|
+}
|
|
|
454
|
+
|
|
|
455
|
+func TestConnIdleTimeout(t *testing.T) {
|
|
|
456
|
+ t.Parallel()
|
|
|
457
|
+ suite.Run(t, &ConnIdleTimeoutTestSuite{})
|
|
|
458
|
+}
|