377 lines
7.2 KiB
Go
377 lines
7.2 KiB
Go
/*
|
|
* Copyright 2020 The Dragonfly Authors
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*/
|
|
|
|
package set
|
|
|
|
import (
|
|
"math/rand"
|
|
"runtime"
|
|
"sync"
|
|
"testing"
|
|
|
|
"github.com/stretchr/testify/assert"
|
|
)
|
|
|
|
const N = 1000
|
|
|
|
func TestSafeSetAdd(t *testing.T) {
|
|
tests := []struct {
|
|
name string
|
|
value string
|
|
expect func(t *testing.T, ok bool, s SafeSet[string], value string)
|
|
}{
|
|
{
|
|
name: "add value",
|
|
value: "foo",
|
|
expect: func(t *testing.T, ok bool, s SafeSet[string], value string) {
|
|
assert := assert.New(t)
|
|
assert.Equal(ok, true)
|
|
assert.Equal(s.Values(), []string{value})
|
|
},
|
|
},
|
|
{
|
|
name: "add value failed",
|
|
value: "foo",
|
|
expect: func(t *testing.T, _ bool, s SafeSet[string], value string) {
|
|
assert := assert.New(t)
|
|
ok := s.Add("foo")
|
|
assert.Equal(ok, false)
|
|
assert.Equal(s.Values(), []string{value})
|
|
},
|
|
},
|
|
}
|
|
|
|
for _, tc := range tests {
|
|
t.Run(tc.name, func(t *testing.T) {
|
|
s := NewSafeSet[string]()
|
|
tc.expect(t, s.Add(tc.value), s, tc.value)
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestSafeSetAdd_Concurrent(t *testing.T) {
|
|
runtime.GOMAXPROCS(2)
|
|
|
|
s := NewSafeSet[int]()
|
|
nums := rand.Perm(N)
|
|
|
|
var wg sync.WaitGroup
|
|
wg.Add(len(nums))
|
|
for i := range len(nums) {
|
|
go func(i int) {
|
|
s.Add(i)
|
|
wg.Done()
|
|
}(i)
|
|
}
|
|
|
|
wg.Wait()
|
|
for _, n := range nums {
|
|
if !s.Contains(n) {
|
|
t.Errorf("Set is missing element: %v", n)
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestSafeSetDelete(t *testing.T) {
|
|
tests := []struct {
|
|
name string
|
|
value string
|
|
expect func(t *testing.T, s SafeSet[string], value string)
|
|
}{
|
|
{
|
|
name: "delete value",
|
|
value: "foo",
|
|
expect: func(t *testing.T, s SafeSet[string], value string) {
|
|
assert := assert.New(t)
|
|
s.Delete(value)
|
|
assert.Equal(s.Len(), uint(0))
|
|
},
|
|
},
|
|
{
|
|
name: "delete value does not exist",
|
|
value: "foo",
|
|
expect: func(t *testing.T, s SafeSet[string], _ string) {
|
|
assert := assert.New(t)
|
|
s.Delete("bar")
|
|
assert.Equal(s.Len(), uint(1))
|
|
},
|
|
},
|
|
}
|
|
|
|
for _, tc := range tests {
|
|
t.Run(tc.name, func(t *testing.T) {
|
|
s := NewSafeSet[string]()
|
|
s.Add(tc.value)
|
|
tc.expect(t, s, tc.value)
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestSafeSetDelete_Concurrent(t *testing.T) {
|
|
runtime.GOMAXPROCS(2)
|
|
|
|
s := NewSafeSet[int]()
|
|
nums := rand.Perm(N)
|
|
for _, v := range nums {
|
|
s.Add(v)
|
|
}
|
|
|
|
var wg sync.WaitGroup
|
|
wg.Add(len(nums))
|
|
for _, v := range nums {
|
|
go func(i int) {
|
|
s.Delete(i)
|
|
wg.Done()
|
|
}(v)
|
|
}
|
|
wg.Wait()
|
|
|
|
if s.Len() != 0 {
|
|
t.Errorf("Expected len 0; got %v", s.Len())
|
|
}
|
|
}
|
|
|
|
func TestSafeSetContains(t *testing.T) {
|
|
tests := []struct {
|
|
name string
|
|
value string
|
|
expect func(t *testing.T, s SafeSet[string], value string)
|
|
}{
|
|
{
|
|
name: "contains value",
|
|
value: "foo",
|
|
expect: func(t *testing.T, s SafeSet[string], value string) {
|
|
assert := assert.New(t)
|
|
assert.Equal(s.Contains(string(value)), true)
|
|
},
|
|
},
|
|
{
|
|
name: "contains value does not exist",
|
|
value: "foo",
|
|
expect: func(t *testing.T, s SafeSet[string], _ string) {
|
|
assert := assert.New(t)
|
|
assert.Equal(s.Contains("bar"), false)
|
|
},
|
|
},
|
|
}
|
|
|
|
for _, tc := range tests {
|
|
t.Run(tc.name, func(t *testing.T) {
|
|
s := NewSafeSet[string]()
|
|
s.Add(tc.value)
|
|
tc.expect(t, s, tc.value)
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestSafeSetContains_Concurrent(t *testing.T) {
|
|
runtime.GOMAXPROCS(2)
|
|
|
|
s := NewSafeSet[int]()
|
|
nums := rand.Perm(N)
|
|
interfaces := make([]int, 0)
|
|
for _, v := range nums {
|
|
s.Add(v)
|
|
interfaces = append(interfaces, v)
|
|
}
|
|
|
|
var wg sync.WaitGroup
|
|
for range nums {
|
|
wg.Add(1)
|
|
go func() {
|
|
s.Contains(interfaces...)
|
|
wg.Done()
|
|
}()
|
|
}
|
|
wg.Wait()
|
|
}
|
|
|
|
func TestSetSafeLen(t *testing.T) {
|
|
tests := []struct {
|
|
name string
|
|
expect func(t *testing.T, s SafeSet[string])
|
|
}{
|
|
{
|
|
name: "get length",
|
|
expect: func(t *testing.T, s SafeSet[string]) {
|
|
assert := assert.New(t)
|
|
s.Add("foo")
|
|
assert.Equal(s.Len(), uint(1))
|
|
},
|
|
},
|
|
{
|
|
name: "get empty set length",
|
|
expect: func(t *testing.T, s SafeSet[string]) {
|
|
assert := assert.New(t)
|
|
assert.Equal(s.Len(), uint(0))
|
|
},
|
|
},
|
|
}
|
|
|
|
for _, tc := range tests {
|
|
t.Run(tc.name, func(t *testing.T) {
|
|
s := NewSafeSet[string]()
|
|
tc.expect(t, s)
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestSafeSetLen_Concurrent(t *testing.T) {
|
|
runtime.GOMAXPROCS(2)
|
|
|
|
s := NewSafeSet[int]()
|
|
|
|
var wg sync.WaitGroup
|
|
wg.Add(1)
|
|
go func() {
|
|
elems := s.Len()
|
|
for range N {
|
|
newElems := s.Len()
|
|
if newElems < elems {
|
|
t.Errorf("Len shrunk from %v to %v", elems, newElems)
|
|
}
|
|
}
|
|
wg.Done()
|
|
}()
|
|
|
|
for range N {
|
|
s.Add(rand.Int())
|
|
}
|
|
wg.Wait()
|
|
}
|
|
|
|
func TestSafeSetValues(t *testing.T) {
|
|
tests := []struct {
|
|
name string
|
|
expect func(t *testing.T, s SafeSet[string])
|
|
}{
|
|
{
|
|
name: "get values",
|
|
expect: func(t *testing.T, s SafeSet[string]) {
|
|
assert := assert.New(t)
|
|
s.Add("foo")
|
|
assert.Equal(s.Values(), []string{"foo"})
|
|
},
|
|
},
|
|
{
|
|
name: "get empty values",
|
|
expect: func(t *testing.T, s SafeSet[string]) {
|
|
assert := assert.New(t)
|
|
assert.Equal(s.Values(), []string(nil))
|
|
},
|
|
},
|
|
{
|
|
name: "get multi values",
|
|
expect: func(t *testing.T, s SafeSet[string]) {
|
|
assert := assert.New(t)
|
|
s.Add("foo")
|
|
s.Add("bar")
|
|
assert.Contains(s.Values(), "bar")
|
|
assert.Contains(s.Values(), "foo")
|
|
},
|
|
},
|
|
}
|
|
|
|
for _, tc := range tests {
|
|
t.Run(tc.name, func(t *testing.T) {
|
|
s := NewSafeSet[string]()
|
|
tc.expect(t, s)
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestSafeSetValues_Concurrent(t *testing.T) {
|
|
runtime.GOMAXPROCS(2)
|
|
|
|
s := NewSafeSet[int]()
|
|
|
|
var wg sync.WaitGroup
|
|
wg.Add(1)
|
|
go func() {
|
|
elems := s.Values()
|
|
for range N {
|
|
newElems := s.Values()
|
|
if len(newElems) < len(elems) {
|
|
t.Errorf("Values shrunk from %v to %v", elems, newElems)
|
|
}
|
|
}
|
|
wg.Done()
|
|
}()
|
|
|
|
for i := range N {
|
|
s.Add(i)
|
|
}
|
|
wg.Wait()
|
|
}
|
|
|
|
func TestSafeSetClear(t *testing.T) {
|
|
tests := []struct {
|
|
name string
|
|
expect func(t *testing.T, s SafeSet[string])
|
|
}{
|
|
{
|
|
name: "clear empty set",
|
|
expect: func(t *testing.T, s SafeSet[string]) {
|
|
assert := assert.New(t)
|
|
s.Clear()
|
|
assert.Equal(s.Values(), []string(nil))
|
|
},
|
|
},
|
|
{
|
|
name: "clear set",
|
|
expect: func(t *testing.T, s SafeSet[string]) {
|
|
assert := assert.New(t)
|
|
assert.Equal(s.Add("foo"), true)
|
|
s.Clear()
|
|
assert.Equal(s.Values(), []string(nil))
|
|
assert.Equal(s.Add("foo"), true)
|
|
assert.Equal(s.Values(), []string{"foo"})
|
|
},
|
|
},
|
|
}
|
|
|
|
for _, tc := range tests {
|
|
t.Run(tc.name, func(t *testing.T) {
|
|
s := NewSafeSet[string]()
|
|
tc.expect(t, s)
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestSafeSetClear_Concurrent(t *testing.T) {
|
|
runtime.GOMAXPROCS(2)
|
|
|
|
s := NewSafeSet[int]()
|
|
nums := rand.Perm(N)
|
|
|
|
var wg sync.WaitGroup
|
|
wg.Add(len(nums))
|
|
for i := range len(nums) {
|
|
go func(i int) {
|
|
s.Add(i)
|
|
s.Clear()
|
|
wg.Done()
|
|
}(i)
|
|
}
|
|
|
|
wg.Wait()
|
|
for _, n := range nums {
|
|
if s.Contains(n) {
|
|
t.Errorf("SafeSet contains element: %v", n)
|
|
}
|
|
}
|
|
}
|