File: sortedset_test.go

package info (click to toggle)
golang-github-segmentio-asm 1.2.0%2Bgit20231107.1cfacc8-1
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid
  • size: 932 kB
  • sloc: asm: 6,093; makefile: 32
file content (115 lines) | stat: -rw-r--r-- 2,859 bytes parent folder | download
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
package sortedset

import (
	"bytes"
	"encoding/hex"
	"math/rand"
	"sort"
	"testing"
)

func assertArraysEqual(t *testing.T, expected, actual []byte, size int) {
	t.Helper()

	if !bytes.Equal(expected, actual) {
		t.Logf("\nexpected (%d):\n%s\nfound (%d):\n%s",
			len(expected), hex.Dump(expected),
			len(actual), hex.Dump(actual))
		t.Fatal("arrays are not equal")
	}
}

func randomSortedArray(prng *rand.Rand, size int, count int, repeatChance float64) (array []byte, uniques []byte) {
	if count == 0 {
		return nil, nil
	}

	// Generate `count` random chunks of `size` bytes and then sort them.
	pool := make([]byte, size*count)
	prng.Read(pool)
	sortArray(pool, size)

	// Sanity checks — the items must be unique and sorted.
	for i := size; i < len(pool); i += size {
		switch bytes.Compare(pool[i-size:i], pool[i:i+size]) {
		case 0:
			panic("duplicate item in pool")
		case 1:
			panic("not sorted correctly")
		}
	}

	array = make([]byte, 0, size*count)

	// Build an array from the pool of unique items, using the configurable
	// chance of repeat. A repeatChance of 0 will yield an array where every
	// item is unique, while a repeatChance of 1 will yield an array where
	// every item is a duplicate of the first item.
	uniq := size
	for i := 0; i < count; i++ {
		array = append(array, pool[uniq-size:uniq]...)
		if prng.Float64() >= repeatChance && i != count-1 {
			uniq += size
		}
	}

	// Return a second array with just the unique items.
	uniques = pool[:uniq]
	return
}

func randomSortedSet(prng *rand.Rand, size int, count int) []byte {
	_, set := randomSortedArray(prng, size, count, 0.0)
	return set
}

func randomSortedSetPair(prng *rand.Rand, size int, count int, overlapChance float64) ([]byte, []byte) {
	setA := randomSortedSet(prng, size, count)
	setB := randomSortedSet(prng, size, count)

	// Sanity check: there must be no duplicates.
	if len(combineArrays(setA, setB, size)) != count*size*2 {
		panic("sorted sets overlap")
	}

	// Build a new set by taking items from both setA and setB depending
	// on the value of overlapChance.
	split := int(float64(count)*overlapChance) * size
	overlap := combineArrays(setA[:split], setB[:len(setB)-split], size)

	return setA, overlap
}

func combineArrays(a, b []byte, size int) []byte {
	return sortArray(append(append([]byte{}, a...), b...), size)
}

func sortArray(b []byte, size int) []byte {
	sort.Sort(&chunks{b: b, size: size})
	return b
}

type chunks struct {
	b    []byte
	size int
	tmp  []byte
}

func (s *chunks) Len() int {
	return len(s.b) / s.size
}

func (s *chunks) Less(i, j int) bool {
	return bytes.Compare(s.slice(i), s.slice(j)) < 0
}

func (s *chunks) Swap(i, j int) {
	tmp := make([]byte, s.size)
	copy(tmp, s.slice(j))
	copy(s.slice(j), s.slice(i))
	copy(s.slice(i), tmp)
}

func (s *chunks) slice(i int) []byte {
	return s.b[i*s.size : (i+1)*s.size]
}