-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathfuture_test.go
146 lines (118 loc) · 2.73 KB
/
future_test.go
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
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
package future_test
import (
"context"
"fmt"
"testing"
"time"
future "github.com/uudashr/go-future"
)
func TestFuture_Get_immediate(t *testing.T) {
fut, setResult := future.New[string]()
want := "Hello"
setResult(want, nil)
ctx, cancel := context.WithTimeout(context.Background(), 500*time.Millisecond)
defer cancel()
got, err := fut.Get(ctx)
if err != nil {
t.Fatal(err)
}
if got != want {
t.Fatal("got:", got, "want:", want)
}
}
func TestFuture_Listen_immediate(t *testing.T) {
fut, setResult := future.New[string]()
want := "Hello"
setResult(want, nil)
done := make(chan struct{})
var got string
var gotErr error
fut.Listen(func(val string, err error) {
got, gotErr = val, err
close(done)
})
<-done
if gotErr != nil {
t.Fatal(gotErr)
}
if got != want {
t.Fatal("got:", got, "want:", want)
}
}
func TestFuture_Get_async(t *testing.T) {
fut, setResult := future.New[string]()
want := "Hello"
time.AfterFunc(10*time.Millisecond, func() {
setResult(want, nil)
})
ctx, cancel := context.WithTimeout(context.Background(), 500*time.Millisecond)
defer cancel()
got, err := fut.Get(ctx)
if err != nil {
t.Fatal(err)
}
if got != want {
t.Fatal("got:", got, "want:", want)
}
}
func TestFuture_Get_timeout(t *testing.T) {
fut, _ := future.New[string]()
// no result
ctx, cancel := context.WithTimeout(context.Background(), 500*time.Millisecond)
defer cancel()
_, err := fut.Get(ctx)
if got, want := err, context.DeadlineExceeded; got != want {
t.Fatal("got:", got, "want:", want)
}
}
// Basic usage of the future API.
func ExampleFuture() {
// Future function
doThings := func() *future.Future[string] {
fut, setResult := future.New[string]()
time.AfterFunc(10*time.Millisecond, func() {
setResult("OK", nil)
})
return fut
}
// Usage
res := doThings()
val, _ := res.Get(context.Background())
fmt.Println(val)
// Output: OK
}
// Use Listen to get callback style. This can remove the need of extra gouroutine.
func ExampleFuture_callback() {
// Future function
doThings := func() *future.Future[string] {
fut, setResult := future.New[string]()
time.AfterFunc(10*time.Millisecond, func() {
setResult("OK", nil)
})
return fut
}
// Usage
done := make(chan struct{})
res := doThings()
res.Listen(func(val string, err error) {
fmt.Println(val)
close(done)
})
<-done
// Output: OK
}
// Use existing sync function call into async using future.
func ExampleCall() {
// Sync function
greet := func() (string, error) {
time.Sleep(10 * time.Millisecond)
return "Hello World!", nil
}
// Use sync function as async using future
fut := future.Call(func() (string, error) {
return greet()
})
v, _ := fut.Get(context.Background())
fmt.Println(v)
// Output: Hello World!
}