package rand
|
|
|
|
import (
|
|
"bytes"
|
|
"encoding/json"
|
|
"fmt"
|
|
mrand "math/rand"
|
|
"sync"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/stretchr/testify/assert"
|
|
)
|
|
|
|
func TestRandStr(t *testing.T) {
|
|
l := 243
|
|
s := RandStr(l)
|
|
assert.Equal(t, l, len(s))
|
|
}
|
|
|
|
func TestRandBytes(t *testing.T) {
|
|
l := 243
|
|
b := RandBytes(l)
|
|
assert.Equal(t, l, len(b))
|
|
}
|
|
|
|
func TestRandIntn(t *testing.T) {
|
|
n := 243
|
|
for i := 0; i < 100; i++ {
|
|
x := RandIntn(n)
|
|
assert.True(t, x < n)
|
|
}
|
|
}
|
|
|
|
// Test to make sure that we never call math.rand().
|
|
// We do this by ensuring that outputs are deterministic.
|
|
func TestDeterminism(t *testing.T) {
|
|
var firstOutput string
|
|
|
|
// Set math/rand's seed for the sake of debugging this test.
|
|
// (It isn't strictly necessary).
|
|
mrand.Seed(1)
|
|
|
|
for i := 0; i < 100; i++ {
|
|
output := testThemAll()
|
|
if i == 0 {
|
|
firstOutput = output
|
|
} else if firstOutput != output {
|
|
t.Errorf("run #%d's output was different from first run.\nfirst: %v\nlast: %v",
|
|
i, firstOutput, output)
|
|
}
|
|
}
|
|
}
|
|
|
|
func testThemAll() string {
|
|
|
|
// Such determinism.
|
|
grand.reset(1)
|
|
|
|
// Use it.
|
|
out := new(bytes.Buffer)
|
|
perm := RandPerm(10)
|
|
blob, _ := json.Marshal(perm)
|
|
fmt.Fprintf(out, "perm: %s\n", blob)
|
|
fmt.Fprintf(out, "randInt: %d\n", RandInt())
|
|
fmt.Fprintf(out, "randUint: %d\n", RandUint())
|
|
fmt.Fprintf(out, "randIntn: %d\n", RandIntn(97))
|
|
fmt.Fprintf(out, "randInt31: %d\n", RandInt31())
|
|
fmt.Fprintf(out, "randInt32: %d\n", RandInt32())
|
|
fmt.Fprintf(out, "randInt63: %d\n", RandInt63())
|
|
fmt.Fprintf(out, "randInt64: %d\n", RandInt64())
|
|
fmt.Fprintf(out, "randUint32: %d\n", RandUint32())
|
|
fmt.Fprintf(out, "randUint64: %d\n", RandUint64())
|
|
return out.String()
|
|
}
|
|
|
|
func TestRngConcurrencySafety(t *testing.T) {
|
|
var wg sync.WaitGroup
|
|
for i := 0; i < 100; i++ {
|
|
wg.Add(1)
|
|
go func() {
|
|
defer wg.Done()
|
|
|
|
_ = RandUint64()
|
|
<-time.After(time.Millisecond * time.Duration(RandIntn(100)))
|
|
_ = RandPerm(3)
|
|
}()
|
|
}
|
|
wg.Wait()
|
|
}
|
|
|
|
func BenchmarkRandBytes10B(b *testing.B) {
|
|
benchmarkRandBytes(b, 10)
|
|
}
|
|
func BenchmarkRandBytes100B(b *testing.B) {
|
|
benchmarkRandBytes(b, 100)
|
|
}
|
|
func BenchmarkRandBytes1KiB(b *testing.B) {
|
|
benchmarkRandBytes(b, 1024)
|
|
}
|
|
func BenchmarkRandBytes10KiB(b *testing.B) {
|
|
benchmarkRandBytes(b, 10*1024)
|
|
}
|
|
func BenchmarkRandBytes100KiB(b *testing.B) {
|
|
benchmarkRandBytes(b, 100*1024)
|
|
}
|
|
func BenchmarkRandBytes1MiB(b *testing.B) {
|
|
benchmarkRandBytes(b, 1024*1024)
|
|
}
|
|
|
|
func benchmarkRandBytes(b *testing.B, n int) {
|
|
for i := 0; i < b.N; i++ {
|
|
_ = RandBytes(n)
|
|
}
|
|
b.ReportAllocs()
|
|
}
|