@@ -17,38 +17,39 @@ limitations under the License.
17
17
package watch
18
18
19
19
import (
20
+ "context"
20
21
"errors"
21
22
"time"
22
23
24
+ "github.com/golang/glog"
23
25
"k8s.io/apimachinery/pkg/util/wait"
26
+ "k8s.io/apimachinery/pkg/watch"
24
27
)
25
28
26
29
// ConditionFunc returns true if the condition has been reached, false if it has not been reached yet,
27
30
// or an error if the condition cannot be checked and should terminate. In general, it is better to define
28
31
// level driven conditions over edge driven conditions (pod has ready=true, vs pod modified and ready changed
29
32
// from false to true).
30
- type ConditionFunc func (event Event ) (bool , error )
33
+ type ConditionFunc func (event watch. Event ) (bool , error )
31
34
32
- // ErrWatchClosed is returned when the watch channel is closed before timeout in Until .
33
- var ErrWatchClosed = errors .New ("watch closed before Until timeout" )
35
+ // ErrWatchClosed is returned when the watch channel is closed before timeout in UntilWithoutRetry .
36
+ var ErrWatchClosed = errors .New ("watch closed before UntilWithoutRetry timeout" )
34
37
35
- // Until reads items from the watch until each provided condition succeeds, and then returns the last watch
38
+ // UntilWithoutRetry reads items from the watch until each provided condition succeeds, and then returns the last watch
36
39
// encountered. The first condition that returns an error terminates the watch (and the event is also returned).
37
40
// If no event has been received, the returned event will be nil.
38
41
// Conditions are satisfied sequentially so as to provide a useful primitive for higher level composition.
39
- // A zero timeout means to wait forever.
40
- func Until (timeout time.Duration , watcher Interface , conditions ... ConditionFunc ) (* Event , error ) {
42
+ // Waits until context deadline or until context is canceled.
43
+ //
44
+ // Warning: Unless you have a very specific use case (probably a special Watcher) don't use this function!!!
45
+ // Warning: This will fail e.g. on API timeouts and/or 'too old resource version' error.
46
+ // Warning: You are most probably looking for a function *Until* or *UntilWithSync* below,
47
+ // Warning: solving such issues.
48
+ // TODO: Consider making this function private to prevent misuse when the other occurrences in our codebase are gone.
49
+ func UntilWithoutRetry (ctx context.Context , watcher watch.Interface , conditions ... ConditionFunc ) (* watch.Event , error ) {
41
50
ch := watcher .ResultChan ()
42
51
defer watcher .Stop ()
43
- var after <- chan time.Time
44
- if timeout > 0 {
45
- after = time .After (timeout )
46
- } else {
47
- ch := make (chan time.Time )
48
- defer close (ch )
49
- after = ch
50
- }
51
- var lastEvent * Event
52
+ var lastEvent * watch.Event
52
53
for _ , condition := range conditions {
53
54
// check the next condition against the previous event and short circuit waiting for the next watch
54
55
if lastEvent != nil {
@@ -69,7 +70,6 @@ func Until(timeout time.Duration, watcher Interface, conditions ...ConditionFunc
69
70
}
70
71
lastEvent = & event
71
72
72
- // TODO: check for watch expired error and retry watch from latest point?
73
73
done , err := condition (event )
74
74
if err != nil {
75
75
return lastEvent , err
@@ -78,10 +78,25 @@ func Until(timeout time.Duration, watcher Interface, conditions ...ConditionFunc
78
78
break ConditionSucceeded
79
79
}
80
80
81
- case <- after :
81
+ case <- ctx . Done () :
82
82
return lastEvent , wait .ErrWaitTimeout
83
83
}
84
84
}
85
85
}
86
86
return lastEvent , nil
87
87
}
88
+
89
+ // ContextWithOptionalTimeout wraps context.WithTimeout and handles infinite timeouts expressed as 0 duration.
90
+ func ContextWithOptionalTimeout (parent context.Context , timeout time.Duration ) (context.Context , context.CancelFunc ) {
91
+ if timeout < 0 {
92
+ // This should be handled in validation
93
+ glog .Errorf ("Timeout for context shall not be negative!" )
94
+ timeout = 0
95
+ }
96
+
97
+ if timeout == 0 {
98
+ return context .WithCancel (parent )
99
+ }
100
+
101
+ return context .WithTimeout (parent , timeout )
102
+ }
0 commit comments