2018-06-10 22:50:03 +08:00
|
|
|
package observable
|
|
|
|
|
|
|
|
import (
|
|
|
|
"runtime"
|
|
|
|
"sync"
|
|
|
|
"testing"
|
|
|
|
"time"
|
|
|
|
)
|
|
|
|
|
|
|
|
func iterator(item []interface{}) chan interface{} {
|
|
|
|
ch := make(chan interface{})
|
|
|
|
go func() {
|
|
|
|
time.Sleep(100 * time.Millisecond)
|
|
|
|
for _, elm := range item {
|
|
|
|
ch <- elm
|
|
|
|
}
|
|
|
|
close(ch)
|
|
|
|
}()
|
|
|
|
return ch
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestObservable(t *testing.T) {
|
|
|
|
iter := iterator([]interface{}{1, 2, 3, 4, 5})
|
|
|
|
src := NewObservable(iter)
|
|
|
|
data, err := src.Subscribe()
|
|
|
|
if err != nil {
|
|
|
|
t.Error(err)
|
|
|
|
}
|
|
|
|
count := 0
|
2018-07-12 18:03:02 +08:00
|
|
|
for range data {
|
2018-06-10 22:50:03 +08:00
|
|
|
count = count + 1
|
|
|
|
}
|
|
|
|
if count != 5 {
|
|
|
|
t.Error("Revc number error")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestObservable_MutilSubscribe(t *testing.T) {
|
|
|
|
iter := iterator([]interface{}{1, 2, 3, 4, 5})
|
|
|
|
src := NewObservable(iter)
|
|
|
|
ch1, _ := src.Subscribe()
|
|
|
|
ch2, _ := src.Subscribe()
|
|
|
|
count := 0
|
|
|
|
|
|
|
|
var wg sync.WaitGroup
|
|
|
|
wg.Add(2)
|
|
|
|
waitCh := func(ch <-chan interface{}) {
|
2018-07-12 18:03:02 +08:00
|
|
|
for range ch {
|
2018-06-10 22:50:03 +08:00
|
|
|
count = count + 1
|
|
|
|
}
|
|
|
|
wg.Done()
|
|
|
|
}
|
|
|
|
go waitCh(ch1)
|
|
|
|
go waitCh(ch2)
|
|
|
|
wg.Wait()
|
|
|
|
if count != 10 {
|
|
|
|
t.Error("Revc number error")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestObservable_UnSubscribe(t *testing.T) {
|
|
|
|
iter := iterator([]interface{}{1, 2, 3, 4, 5})
|
|
|
|
src := NewObservable(iter)
|
|
|
|
data, err := src.Subscribe()
|
|
|
|
if err != nil {
|
|
|
|
t.Error(err)
|
|
|
|
}
|
|
|
|
src.UnSubscribe(data)
|
|
|
|
_, open := <-data
|
|
|
|
if open {
|
|
|
|
t.Error("Revc number error")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-12 18:03:02 +08:00
|
|
|
func TestObservable_SubscribeClosedSource(t *testing.T) {
|
|
|
|
iter := iterator([]interface{}{1})
|
|
|
|
src := NewObservable(iter)
|
|
|
|
data, _ := src.Subscribe()
|
|
|
|
<-data
|
|
|
|
|
|
|
|
_, closed := src.Subscribe()
|
|
|
|
if closed == nil {
|
|
|
|
t.Error("Observable should be closed")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestObservable_UnSubscribeWithNotExistSubscription(t *testing.T) {
|
|
|
|
sub := Subscription(make(chan interface{}))
|
|
|
|
iter := iterator([]interface{}{1})
|
|
|
|
src := NewObservable(iter)
|
|
|
|
src.UnSubscribe(sub)
|
|
|
|
}
|
|
|
|
|
2018-06-10 22:50:03 +08:00
|
|
|
func TestObservable_SubscribeGoroutineLeak(t *testing.T) {
|
|
|
|
// waiting for other goroutine recycle
|
|
|
|
time.Sleep(120 * time.Millisecond)
|
|
|
|
init := runtime.NumGoroutine()
|
|
|
|
iter := iterator([]interface{}{1, 2, 3, 4, 5})
|
|
|
|
src := NewObservable(iter)
|
|
|
|
max := 100
|
|
|
|
|
|
|
|
var list []Subscription
|
|
|
|
for i := 0; i < max; i++ {
|
|
|
|
ch, _ := src.Subscribe()
|
|
|
|
list = append(list, ch)
|
|
|
|
}
|
|
|
|
|
|
|
|
var wg sync.WaitGroup
|
|
|
|
wg.Add(max)
|
|
|
|
waitCh := func(ch <-chan interface{}) {
|
2018-07-12 18:03:02 +08:00
|
|
|
for range ch {
|
2018-06-10 22:50:03 +08:00
|
|
|
}
|
|
|
|
wg.Done()
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, ch := range list {
|
|
|
|
go waitCh(ch)
|
|
|
|
}
|
|
|
|
wg.Wait()
|
|
|
|
now := runtime.NumGoroutine()
|
|
|
|
if init != now {
|
|
|
|
t.Errorf("Goroutine Leak: init %d now %d", init, now)
|
|
|
|
}
|
|
|
|
}
|