File: tickrunner_test.go

package info (click to toggle)
receptor 1.5.5-2
  • links: PTS, VCS
  • area: main
  • in suites: sid
  • size: 2,772 kB
  • sloc: python: 1,643; makefile: 305; sh: 174
file content (121 lines) | stat: -rw-r--r-- 3,908 bytes parent folder | download | duplicates (2)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
package tickrunner

import (
	"context"
	"sync"
	"testing"
	"time"
)

// SafeCounter is safe to use concurrently.
type SafeCounter struct {
	mu sync.Mutex
	v  int
}

// Inc increments the counter for the given key.
func (c *SafeCounter) Inc() {
	c.mu.Lock()
	// Lock so only one goroutine at a time can access the map c.v.
	c.v++
	c.mu.Unlock()
}

// Value returns the current value of the counter for the given key.
func (c *SafeCounter) Value() int {
	c.mu.Lock()
	// Lock so only one goroutine at a time can access the map c.v.
	defer c.mu.Unlock()

	return c.v
}

func TestRun(t *testing.T) {
	type testCase struct {
		name             string
		requestTime      int
		requestCount     int
		periodicInterval time.Duration
		defaultReqDelay  time.Duration
		expectedRunCount int
		/* This value must take into account the function's algorithm
		and the other two time.Duration values */
		waitForBeforeChecks time.Duration
	}

	tests := []testCase{
		{
			name:             "Run only with default periodicInterval once",
			periodicInterval: time.Duration(2) * time.Second,
			// Run can only execute one request with the set periodicInterval.
			expectedRunCount:    1,
			waitForBeforeChecks: time.Duration(3) * time.Second,
		},
		{
			name:             "Run only with default periodicInterval more than once",
			periodicInterval: time.Duration(2) * time.Second,
			// Run can only execute one request with the set periodicInterval.
			expectedRunCount:    3,
			waitForBeforeChecks: time.Duration(7) * time.Second,
		},
		{
			name:         "Run request inmediately",
			requestCount: 1,
			requestTime:  0,
			/* Setting this to a high value so it doesn't run at all
			with the default periodicInterval. We only want to test
			for the requests sent. */
			periodicInterval: time.Duration(300) * time.Second,
			defaultReqDelay:  time.Duration(1) * time.Second,
			// Run can only execute one request with the set periodicInterval.
			expectedRunCount:    1,
			waitForBeforeChecks: time.Duration(2) * time.Second,
		},
		{
			name:         "Run sending some requests overrides default periodicInterval",
			requestCount: 3,
			requestTime:  2,
			/* Setting this to a high value so it doesn't run at all
			with the default periodicInterval. We only want to test
			for the requests sent. */
			periodicInterval: time.Duration(300) * time.Second,
			defaultReqDelay:  time.Duration(1) * time.Second,
			/* Due to the design of the test itself, the requests get sent
			into the channel back to back with no time in between them
			This expectedRunCount is the correct value since only the
			first one will be run. */
			expectedRunCount:    1,
			waitForBeforeChecks: time.Duration(3) * time.Second,
		},
	}

	for _, tc := range tests {
		t.Run(tc.name, func(t *testing.T) {
			ctx, ctxCancel := context.WithCancel(context.Background())
			defer ctxCancel()

			/* Create a counter value and increment function to keep track of
			inner function calls count. */
			runCounter := SafeCounter{}
			runFunction := func() { runCounter.Inc() }
			runChan := Run(ctx, runFunction, tc.periodicInterval, tc.defaultReqDelay)

			/* Send the required requests through the returned channel
			These are inmediate back-to-back requests
			Following the function's logic, if there are too many requests coming in
			simultaneously, these should be batched since only the oldest one is taken
			into account before the set time passes. */
			for i := 0; i < tc.requestCount; i++ {
				runChan <- time.Duration(tc.requestTime) * time.Second
			}

			/* Since this function is time-based, lets wait for a calculated time
			before asserting any value so we avoid race conditions with
			non-blocking code. */
			time.Sleep(tc.waitForBeforeChecks)
			if runCounter.Value() != tc.expectedRunCount {
				t.Errorf("Run count: %d, Expected number of runs: %d", runCounter.Value(), tc.expectedRunCount)
			}
		})
	}
}