mirror of
https://gitlab.com/pulsechaincom/erigon-pulse.git
synced 2025-01-20 09:21:11 +00:00
231e468e19
git-subtree-dir: erigon-lib git-subtree-mainline: 3c8cbda8098cc073a668b9e9b0aafe6c361f17da git-subtree-split: 93d9c9d9fe4bd8a49f7a98a6bce0f0da7094c7d3
768 lines
18 KiB
Go
768 lines
18 KiB
Go
/*
|
|
Copyright 2021 Erigon contributors
|
|
|
|
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 compress
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"fmt"
|
|
"math/rand"
|
|
"os"
|
|
"path/filepath"
|
|
"strings"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/ledgerwatch/log/v3"
|
|
"github.com/stretchr/testify/require"
|
|
)
|
|
|
|
func prepareLoremDict(t *testing.T) *Decompressor {
|
|
t.Helper()
|
|
logger := log.New()
|
|
tmpDir := t.TempDir()
|
|
file := filepath.Join(tmpDir, "compressed")
|
|
t.Name()
|
|
c, err := NewCompressor(context.Background(), t.Name(), file, tmpDir, 1, 2, log.LvlDebug, logger)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer c.Close()
|
|
for k, w := range loremStrings {
|
|
if err = c.AddWord([]byte(fmt.Sprintf("%s %d", w, k))); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
}
|
|
if err = c.Compress(); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
var d *Decompressor
|
|
if d, err = NewDecompressor(file); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
return d
|
|
}
|
|
|
|
func TestDecompressSkip(t *testing.T) {
|
|
d := prepareLoremDict(t)
|
|
defer d.Close()
|
|
g := d.MakeGetter()
|
|
i := 0
|
|
for g.HasNext() {
|
|
w := loremStrings[i]
|
|
if i%2 == 0 {
|
|
g.Skip()
|
|
} else {
|
|
word, _ := g.Next(nil)
|
|
expected := fmt.Sprintf("%s %d", w, i)
|
|
if string(word) != expected {
|
|
t.Errorf("expected %s, got (hex) %s", expected, word)
|
|
}
|
|
}
|
|
i++
|
|
}
|
|
}
|
|
|
|
func TestDecompressMatchOK(t *testing.T) {
|
|
d := prepareLoremDict(t)
|
|
defer d.Close()
|
|
g := d.MakeGetter()
|
|
i := 0
|
|
for g.HasNext() {
|
|
w := loremStrings[i]
|
|
if i%2 != 0 {
|
|
expected := fmt.Sprintf("%s %d", w, i)
|
|
ok, _ := g.Match([]byte(expected))
|
|
if !ok {
|
|
t.Errorf("expexted match with %s", expected)
|
|
}
|
|
} else {
|
|
word, _ := g.Next(nil)
|
|
expected := fmt.Sprintf("%s %d", w, i)
|
|
if string(word) != expected {
|
|
t.Errorf("expected %s, got (hex) %s", expected, word)
|
|
}
|
|
}
|
|
i++
|
|
}
|
|
}
|
|
|
|
func TestDecompressMatchCmpOK(t *testing.T) {
|
|
d := prepareLoremDict(t)
|
|
defer d.Close()
|
|
g := d.MakeGetter()
|
|
i := 0
|
|
for g.HasNext() {
|
|
w := loremStrings[i]
|
|
if i%2 != 0 {
|
|
expected := fmt.Sprintf("%s %d", w, i)
|
|
result := g.MatchCmp([]byte(expected))
|
|
if result != 0 {
|
|
t.Errorf("expexted match with %s", expected)
|
|
}
|
|
} else {
|
|
word, _ := g.Next(nil)
|
|
expected := fmt.Sprintf("%s %d", w, i)
|
|
if string(word) != expected {
|
|
t.Errorf("expected %s, got (hex) %s", expected, word)
|
|
}
|
|
}
|
|
i++
|
|
}
|
|
}
|
|
|
|
func prepareStupidDict(t *testing.T, size int) *Decompressor {
|
|
t.Helper()
|
|
logger := log.New()
|
|
tmpDir := t.TempDir()
|
|
file := filepath.Join(tmpDir, "compressed2")
|
|
t.Name()
|
|
c, err := NewCompressor(context.Background(), t.Name(), file, tmpDir, 1, 2, log.LvlDebug, logger)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer c.Close()
|
|
for i := 0; i < size; i++ {
|
|
if err = c.AddWord([]byte(fmt.Sprintf("word-%d", i))); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
}
|
|
if err = c.Compress(); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
var d *Decompressor
|
|
if d, err = NewDecompressor(file); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
return d
|
|
}
|
|
|
|
func TestDecompressMatchOKCondensed(t *testing.T) {
|
|
condensePatternTableBitThreshold = 4
|
|
d := prepareStupidDict(t, 10000)
|
|
defer func() { condensePatternTableBitThreshold = 9 }()
|
|
defer d.Close()
|
|
|
|
g := d.MakeGetter()
|
|
i := 0
|
|
for g.HasNext() {
|
|
if i%2 != 0 {
|
|
expected := fmt.Sprintf("word-%d", i)
|
|
ok, _ := g.Match([]byte(expected))
|
|
if !ok {
|
|
t.Errorf("expexted match with %s", expected)
|
|
}
|
|
} else {
|
|
word, _ := g.Next(nil)
|
|
expected := fmt.Sprintf("word-%d", i)
|
|
if string(word) != expected {
|
|
t.Errorf("expected %s, got (hex) %s", expected, word)
|
|
}
|
|
}
|
|
i++
|
|
}
|
|
}
|
|
|
|
func TestDecompressMatchNotOK(t *testing.T) {
|
|
d := prepareLoremDict(t)
|
|
defer d.Close()
|
|
g := d.MakeGetter()
|
|
i := 0
|
|
skipCount := 0
|
|
for g.HasNext() {
|
|
w := loremStrings[i]
|
|
expected := fmt.Sprintf("%s %d", w, i+1)
|
|
ok, _ := g.Match([]byte(expected))
|
|
if ok {
|
|
t.Errorf("not expexted match with %s", expected)
|
|
} else {
|
|
g.Skip()
|
|
skipCount++
|
|
}
|
|
i++
|
|
}
|
|
if skipCount != i {
|
|
t.Errorf("something wrong with match logic")
|
|
}
|
|
}
|
|
|
|
func TestDecompressMatchPrefix(t *testing.T) {
|
|
d := prepareLoremDict(t)
|
|
defer d.Close()
|
|
g := d.MakeGetter()
|
|
i := 0
|
|
skipCount := 0
|
|
for g.HasNext() {
|
|
w := loremStrings[i]
|
|
expected := []byte(fmt.Sprintf("%s %d", w, i+1))
|
|
expected = expected[:len(expected)/2]
|
|
if !g.MatchPrefix(expected) {
|
|
t.Errorf("expexted match with %s", expected)
|
|
}
|
|
g.Skip()
|
|
skipCount++
|
|
i++
|
|
}
|
|
if skipCount != i {
|
|
t.Errorf("something wrong with match logic")
|
|
}
|
|
g.Reset(0)
|
|
skipCount = 0
|
|
i = 0
|
|
for g.HasNext() {
|
|
w := loremStrings[i]
|
|
expected := []byte(fmt.Sprintf("%s %d", w, i+1))
|
|
expected = expected[:len(expected)/2]
|
|
if len(expected) > 0 {
|
|
expected[len(expected)-1]++
|
|
if g.MatchPrefix(expected) {
|
|
t.Errorf("not expexted match with %s", expected)
|
|
}
|
|
}
|
|
g.Skip()
|
|
skipCount++
|
|
i++
|
|
}
|
|
}
|
|
|
|
func TestDecompressMatchPrefixCmp(t *testing.T) {
|
|
d := prepareLoremDict(t)
|
|
defer d.Close()
|
|
g := d.MakeGetter()
|
|
i := 0
|
|
skipCount := 0
|
|
for g.HasNext() {
|
|
w := loremStrings[i]
|
|
expected := []byte(fmt.Sprintf("%s %d", w, i+1))
|
|
expected = expected[:len(expected)/2]
|
|
cmp := g.MatchPrefixCmp(expected)
|
|
if cmp != 0 {
|
|
t.Errorf("expexted match with %s", expected)
|
|
}
|
|
g.Skip()
|
|
skipCount++
|
|
i++
|
|
}
|
|
if skipCount != i {
|
|
t.Errorf("something wrong with match logic")
|
|
}
|
|
g.Reset(0)
|
|
skipCount = 0
|
|
i = 0
|
|
for g.HasNext() {
|
|
w := loremStrings[i]
|
|
expected := []byte(fmt.Sprintf("%s %d", w, i+1))
|
|
expected = expected[:len(expected)/2]
|
|
if len(expected) > 0 {
|
|
expected[len(expected)-1]++
|
|
cmp := g.MatchPrefixCmp(expected)
|
|
if cmp == 0 {
|
|
t.Errorf("not expexted match with %s", expected)
|
|
}
|
|
}
|
|
g.Skip()
|
|
skipCount++
|
|
i++
|
|
}
|
|
}
|
|
|
|
func prepareLoremDictUncompressed(t *testing.T) *Decompressor {
|
|
t.Helper()
|
|
logger := log.New()
|
|
tmpDir := t.TempDir()
|
|
file := filepath.Join(tmpDir, "compressed")
|
|
t.Name()
|
|
c, err := NewCompressor(context.Background(), t.Name(), file, tmpDir, 1, 2, log.LvlDebug, logger)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer c.Close()
|
|
for k, w := range loremStrings {
|
|
if err = c.AddUncompressedWord([]byte(fmt.Sprintf("%s %d", w, k))); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
}
|
|
if err = c.Compress(); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
var d *Decompressor
|
|
if d, err = NewDecompressor(file); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
return d
|
|
}
|
|
|
|
func TestUncompressed(t *testing.T) {
|
|
d := prepareLoremDictUncompressed(t)
|
|
defer d.Close()
|
|
g := d.MakeGetter()
|
|
i := 0
|
|
for g.HasNext() {
|
|
w := loremStrings[i]
|
|
expected := []byte(fmt.Sprintf("%s %d", w, i+1))
|
|
expected = expected[:len(expected)/2]
|
|
actual, _ := g.NextUncompressed()
|
|
if bytes.Equal(expected, actual) {
|
|
t.Errorf("expected %s, actual %s", expected, actual)
|
|
}
|
|
i++
|
|
}
|
|
}
|
|
|
|
const lorem = `Lorem ipsum dolor sit amet consectetur adipiscing elit sed do eiusmod tempor incididunt ut labore et
|
|
dolore magna aliqua Ut enim ad minim veniam quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo
|
|
consequat Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur
|
|
Excepteur sint occaecat cupidatat non proident sunt in culpa qui officia deserunt mollit anim id est laborum`
|
|
|
|
var loremStrings = strings.Split(lorem, " ")
|
|
|
|
func TestDecompressTorrent(t *testing.T) {
|
|
t.Skip()
|
|
|
|
fpath := "/mnt/data/chains/mainnet/snapshots/v1-014000-014500-transactions.seg"
|
|
st, err := os.Stat(fpath)
|
|
require.NoError(t, err)
|
|
fmt.Printf("file: %v, size: %d\n", st.Name(), st.Size())
|
|
|
|
condensePatternTableBitThreshold = 9
|
|
fmt.Printf("bit threshold: %d\n", condensePatternTableBitThreshold)
|
|
d, err := NewDecompressor(fpath)
|
|
|
|
require.NoError(t, err)
|
|
defer d.Close()
|
|
|
|
getter := d.MakeGetter()
|
|
_ = getter
|
|
|
|
for getter.HasNext() {
|
|
_, sz := getter.Next(nil)
|
|
// fmt.Printf("%x\n", buf)
|
|
require.NotZero(t, sz)
|
|
}
|
|
}
|
|
|
|
const N = 100
|
|
|
|
var WORDS = [N][]byte{}
|
|
var WORD_FLAGS = [N]bool{} // false - uncompressed word, true - compressed word
|
|
var INPUT_FLAGS = []int{} // []byte or nil input
|
|
|
|
func randWord() []byte {
|
|
size := rand.Intn(256) // size of the word
|
|
word := make([]byte, size)
|
|
for i := 0; i < size; i++ {
|
|
word[i] = byte(rand.Intn(256))
|
|
}
|
|
return word
|
|
}
|
|
|
|
func generateRandWords() {
|
|
for i := 0; i < N-2; i++ {
|
|
WORDS[i] = randWord()
|
|
}
|
|
// make sure we have at least 2 emtpy []byte
|
|
WORDS[N-2] = []byte{}
|
|
WORDS[N-1] = []byte{}
|
|
}
|
|
|
|
func randIntInRange(min, max int) int {
|
|
return (rand.Intn(max-min) + min)
|
|
}
|
|
|
|
func clearPrevDict() {
|
|
WORDS = [N][]byte{}
|
|
WORD_FLAGS = [N]bool{}
|
|
INPUT_FLAGS = []int{}
|
|
}
|
|
|
|
func prepareRandomDict(t *testing.T) *Decompressor {
|
|
t.Helper()
|
|
logger := log.New()
|
|
tmpDir := t.TempDir()
|
|
file := filepath.Join(tmpDir, "complex")
|
|
t.Name()
|
|
c, err := NewCompressor(context.Background(), t.Name(), file, tmpDir, 1, 2, log.LvlDebug, logger)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
// c.DisableFsync()
|
|
defer c.Close()
|
|
clearPrevDict()
|
|
rand.Seed(time.Now().UnixNano())
|
|
generateRandWords()
|
|
|
|
idx := 0
|
|
for idx < N {
|
|
n := rand.Intn(2)
|
|
switch n {
|
|
case 0: // input case
|
|
word := WORDS[idx]
|
|
m := rand.Intn(2)
|
|
if m == 1 {
|
|
if err = c.AddWord(word); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
WORD_FLAGS[idx] = true
|
|
} else {
|
|
if err = c.AddUncompressedWord(word); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
}
|
|
idx++
|
|
INPUT_FLAGS = append(INPUT_FLAGS, n)
|
|
case 1: // nil word
|
|
if err = c.AddWord(nil); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
INPUT_FLAGS = append(INPUT_FLAGS, n)
|
|
default:
|
|
t.Fatal(fmt.Errorf("case %d\n", n))
|
|
}
|
|
}
|
|
|
|
if err = c.Compress(); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
var d *Decompressor
|
|
if d, err = NewDecompressor(file); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
return d
|
|
}
|
|
|
|
func TestDecompressRandomMatchCmp(t *testing.T) {
|
|
d := prepareRandomDict(t)
|
|
defer d.Close()
|
|
|
|
if d.wordsCount != uint64(len(INPUT_FLAGS)) {
|
|
t.Fatalf("TestDecompressRandomDict: d.wordsCount != len(INPUT_FLAGS)")
|
|
}
|
|
|
|
g := d.MakeGetter()
|
|
|
|
word_idx := 0
|
|
input_idx := 0
|
|
total := 0
|
|
// check for existing and non existing keys
|
|
for g.HasNext() {
|
|
pos := g.dataP
|
|
if INPUT_FLAGS[input_idx] == 0 { // []byte input
|
|
notExpected := string(WORDS[word_idx]) + "z"
|
|
cmp := g.MatchCmp([]byte(notExpected))
|
|
if cmp == 0 {
|
|
t.Fatalf("not expected match: %v\n got: %v\n", []byte(notExpected), WORDS[word_idx])
|
|
}
|
|
|
|
expected := WORDS[word_idx]
|
|
cmp = g.MatchCmp(expected) // move offset to the next pos
|
|
if cmp != 0 {
|
|
savePos := g.dataP
|
|
g.Reset(pos)
|
|
word, nextPos := g.Next(nil)
|
|
if nextPos != savePos {
|
|
t.Fatalf("nextPos %d != savePos %d\n", nextPos, savePos)
|
|
}
|
|
if bytes.Compare(expected, word) != cmp {
|
|
fmt.Printf("1 expected: %v, acutal %v, cmp %d\n", expected, word, cmp)
|
|
}
|
|
t.Fatalf("expected match: %v\n got: %v\n", expected, word)
|
|
}
|
|
word_idx++
|
|
} else { // nil input
|
|
notExpected := []byte{0}
|
|
cmp := g.MatchCmp(notExpected)
|
|
if cmp == 0 {
|
|
t.Fatal("not expected match []byte{0} with nil\n")
|
|
}
|
|
|
|
expected := []byte{}
|
|
cmp = g.MatchCmp(nil)
|
|
if cmp != 0 {
|
|
savePos := g.dataP
|
|
g.Reset(pos)
|
|
word, nextPos := g.Next(nil)
|
|
if nextPos != savePos {
|
|
t.Fatalf("nextPos %d != savePos %d\n", nextPos, savePos)
|
|
}
|
|
if bytes.Compare(expected, word) != cmp {
|
|
fmt.Printf("2 expected: %v, acutal %v, cmp %d\n", expected, word, cmp)
|
|
}
|
|
t.Fatalf("expected match: %v\n got: %v\n", expected, word)
|
|
}
|
|
}
|
|
input_idx++
|
|
total++
|
|
}
|
|
if total != int(d.wordsCount) {
|
|
t.Fatalf("expected word count: %d, got %d\n", int(d.wordsCount), total)
|
|
}
|
|
}
|
|
|
|
func TestDecompressRandomMatchBool(t *testing.T) {
|
|
d := prepareRandomDict(t)
|
|
defer d.Close()
|
|
|
|
if d.wordsCount != uint64(len(INPUT_FLAGS)) {
|
|
t.Fatalf("TestDecompressRandomDict: d.wordsCount != len(INPUT_FLAGS)")
|
|
}
|
|
|
|
g := d.MakeGetter()
|
|
|
|
word_idx := 0
|
|
input_idx := 0
|
|
total := 0
|
|
// check for existing and non existing keys
|
|
for g.HasNext() {
|
|
pos := g.dataP
|
|
if INPUT_FLAGS[input_idx] == 0 { // []byte input
|
|
notExpected := string(WORDS[word_idx]) + "z"
|
|
ok, _ := g.Match([]byte(notExpected))
|
|
if ok {
|
|
t.Fatalf("not expected match: %v\n got: %v\n", []byte(notExpected), WORDS[word_idx])
|
|
}
|
|
|
|
expected := WORDS[word_idx]
|
|
ok, _ = g.Match(expected)
|
|
if !ok {
|
|
g.Reset(pos)
|
|
word, _ := g.Next(nil)
|
|
if bytes.Compare(expected, word) != 0 {
|
|
fmt.Printf("1 expected: %v, acutal %v, ok %v\n", expected, word, ok)
|
|
}
|
|
t.Fatalf("expected match: %v\n got: %v\n", expected, word)
|
|
}
|
|
word_idx++
|
|
} else { // nil input
|
|
notExpected := []byte{0}
|
|
ok, _ := g.Match(notExpected)
|
|
if ok {
|
|
t.Fatal("not expected match []byte{0} with nil\n")
|
|
}
|
|
|
|
expected := []byte{}
|
|
ok, _ = g.Match(nil)
|
|
if !ok {
|
|
g.Reset(pos)
|
|
word, _ := g.Next(nil)
|
|
if bytes.Compare(expected, word) != 0 {
|
|
fmt.Printf("2 expected: %v, acutal %v, ok %v\n", expected, word, ok)
|
|
}
|
|
t.Fatalf("expected match: %v\n got: %v\n", expected, word)
|
|
}
|
|
}
|
|
input_idx++
|
|
total++
|
|
}
|
|
if total != int(d.wordsCount) {
|
|
t.Fatalf("expected word count: %d, got %d\n", int(d.wordsCount), total)
|
|
}
|
|
}
|
|
|
|
func TestDecompressRandomFastNext(t *testing.T) {
|
|
d := prepareRandomDict(t)
|
|
defer d.Close()
|
|
|
|
if d.wordsCount != uint64(len(INPUT_FLAGS)) {
|
|
t.Fatalf("TestDecompressRandomDict: d.wordsCount != len(INPUT_FLAGS)")
|
|
}
|
|
|
|
g := d.MakeGetter()
|
|
|
|
word_idx := 0
|
|
input_idx := 0
|
|
total := 0
|
|
buf := make([]byte, (1 << 23))
|
|
// check for existing and non existing keys
|
|
for g.HasNext() {
|
|
if INPUT_FLAGS[input_idx] == 0 { // []byte input
|
|
expected := WORDS[word_idx]
|
|
word, _ := g.FastNext(buf)
|
|
if bytes.Compare(expected, word) != 0 {
|
|
t.Fatalf("1 expected: %v, got %v\n", expected, word)
|
|
}
|
|
word_idx++
|
|
} else { // nil input
|
|
expected := []byte{}
|
|
word, _ := g.FastNext(buf)
|
|
if bytes.Compare(expected, word) != 0 {
|
|
t.Fatalf("2 expected: %v, got %v\n", expected, word)
|
|
}
|
|
}
|
|
input_idx++
|
|
total++
|
|
}
|
|
if total != int(d.wordsCount) {
|
|
t.Fatalf("expected word count: %d, got %d\n", int(d.wordsCount), total)
|
|
}
|
|
}
|
|
|
|
// func TestDecompressRandomDict(t *testing.T) {
|
|
// d := prepareRandomDict(t)
|
|
// defer d.Close()
|
|
|
|
// if d.wordsCount != uint64(len(INPUT_FLAGS)) {
|
|
// t.Fatalf("TestDecompressRandomDict: d.wordsCount != len(INPUT_FLAGS)")
|
|
// }
|
|
|
|
// g := d.MakeGetter()
|
|
|
|
// word_idx := 0
|
|
// input_idx := 0
|
|
// total := 0
|
|
// // check for existing and non existing keys
|
|
// for g.HasNext() {
|
|
// pos := g.dataP
|
|
// if INPUT_FLAGS[input_idx] == 0 { // []byte input
|
|
// notExpected := string(WORDS[word_idx]) + "z"
|
|
// ok, _ := g.Match([]byte(notExpected))
|
|
// if ok {
|
|
// t.Fatalf("not expected match: %s\n got: %s\n", notExpected, WORDS[word_idx])
|
|
// }
|
|
|
|
// expected := WORDS[word_idx]
|
|
// ok, _ = g.Match(expected)
|
|
// if !ok {
|
|
// g.Reset(pos)
|
|
// word, _ := g.Next(nil)
|
|
// t.Fatalf("expected match: %s\n got: %s\n", expected, word)
|
|
// }
|
|
// word_idx++
|
|
// } else { // nil input
|
|
// notExpected := []byte{0}
|
|
// ok, _ := g.Match(notExpected)
|
|
// if ok {
|
|
// t.Fatal("not expected match []byte{0} with nil\n")
|
|
// }
|
|
|
|
// expected := []byte{}
|
|
// ok, _ = g.Match(nil)
|
|
// if !ok {
|
|
// g.Reset(pos)
|
|
// word, _ := g.Next(nil)
|
|
// t.Fatalf("expected match: %s\n got: %s\n", expected, word)
|
|
// }
|
|
// }
|
|
// input_idx++
|
|
// total++
|
|
// }
|
|
// if total != int(d.wordsCount) {
|
|
// t.Fatalf("expected word count: %d, got %d\n", int(d.wordsCount), total)
|
|
// }
|
|
|
|
// // TODO: check for non existing keys, suffixes, prefixes
|
|
// g.Reset(0)
|
|
|
|
// word_idx = 0
|
|
// input_idx = 0
|
|
// // check for existing and non existing prefixes
|
|
// var notExpected = []byte{2, 3, 4}
|
|
// for g.HasNext() {
|
|
|
|
// if INPUT_FLAGS[input_idx] == 0 { // []byte input
|
|
// expected := WORDS[word_idx]
|
|
// prefix_size := len(expected) / 2
|
|
// if len(expected)/2 > 3 {
|
|
// prefix_size = randIntInRange(3, len(expected)/2)
|
|
// }
|
|
// expected = expected[:prefix_size]
|
|
// if len(expected) > 0 {
|
|
// if !g.MatchPrefix(expected) {
|
|
// t.Errorf("expected match with %s", expected)
|
|
// }
|
|
// expected[len(expected)-1]++
|
|
// if g.MatchPrefix(expected) {
|
|
// t.Errorf("not expected match with %s", expected)
|
|
// }
|
|
// } else {
|
|
// if !g.MatchPrefix([]byte{}) {
|
|
// t.Error("expected match with empty []byte")
|
|
// }
|
|
// if g.MatchPrefix(notExpected) {
|
|
// t.Error("not expected empty []byte to match with []byte{2, 3, 4}")
|
|
// }
|
|
// }
|
|
// word_idx++
|
|
// } else { // nil input
|
|
// if !g.MatchPrefix(nil) {
|
|
// t.Error("expected match with nil")
|
|
// }
|
|
// if g.MatchPrefix(notExpected) {
|
|
// t.Error("not expected nil to match with []byte{2, 3, 4}")
|
|
// }
|
|
// }
|
|
|
|
// g.Skip()
|
|
// input_idx++
|
|
// }
|
|
|
|
// g.Reset(0)
|
|
|
|
// word_idx = 0
|
|
// input_idx = 0
|
|
// // check for existing and non existing suffixes
|
|
// notExpected = []byte{2, 3, 4}
|
|
// for g.HasNext() {
|
|
|
|
// if INPUT_FLAGS[input_idx] == 0 { // []byte input
|
|
// suffix := WORDS[word_idx]
|
|
// if len(suffix) > 1 {
|
|
// prefix := suffix[:len(suffix)/2]
|
|
// suffix = suffix[len(suffix)/2:]
|
|
// equal := reflect.DeepEqual(prefix, suffix)
|
|
// // check existing suffixes
|
|
// if g.MatchPrefix(suffix) { // suffix has to be equal to prefix
|
|
// if !equal {
|
|
// t.Fatalf("MatchPrefix(suffix) expected match: prefix is unequal to suffix %v != %v, full slice %v\n", prefix, suffix, WORDS[word_idx])
|
|
// }
|
|
// } else { // suffix has not to be the same as prefix
|
|
// if equal {
|
|
// t.Fatalf("MatchPrefix(suffix) expected unmatch: prefix is equal to suffix %v != %v, full slice %v\n", prefix, suffix, WORDS[word_idx])
|
|
// }
|
|
// }
|
|
|
|
// if len(suffix) > 0 {
|
|
// suffix[0]++
|
|
// if g.MatchPrefix(suffix) && reflect.DeepEqual(prefix, suffix) {
|
|
// t.Fatalf("MatchPrefix(suffix) not expected match: prefix is unequal to suffix %v != %v, full slice %v\n", prefix, suffix, WORDS[word_idx])
|
|
// }
|
|
// }
|
|
|
|
// g.Skip()
|
|
// } else {
|
|
// ok, _ := g.Match(suffix)
|
|
// if !ok {
|
|
// t.Fatal("Match(suffix): expected match suffix")
|
|
// }
|
|
// }
|
|
// word_idx++
|
|
// } else { // nil input
|
|
// if !g.MatchPrefix(nil) {
|
|
// t.Error("MatchPrefix(suffix): expected match with nil")
|
|
// }
|
|
// if g.MatchPrefix(notExpected) {
|
|
// t.Error("MatchPrefix(suffix): not expected nil to match with []byte{2, 3, 4}")
|
|
// }
|
|
// ok, _ := g.Match(nil)
|
|
// if !ok {
|
|
// t.Errorf("Match(suffix): expected to match with nil")
|
|
// }
|
|
// }
|
|
|
|
// input_idx++
|
|
// }
|
|
// }
|