forked from simongog/sdsl-lite
-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathsearch_bidirectional_test.cpp
154 lines (133 loc) · 5.61 KB
/
search_bidirectional_test.cpp
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
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
#include <sdsl/suffix_arrays.hpp>
#include <sdsl/construct.hpp>
#include "gtest/gtest.h"
#include <iostream>
#include <vector>
#include <string>
namespace
{
using namespace sdsl;
using namespace std;
typedef sdsl::int_vector<>::size_type size_type;
string test_file;
string test_file_rev;
template<class T>
class search_bidirectional_test : public ::testing::Test { };
using testing::Types;
typedef Types<
csa_wt<wt_blcd<>, 32, 32, sa_order_sa_sampling<>, isa_sampling<>, byte_alphabet>,
csa_wt<wt_blcd<>, 32, 32, sa_order_sa_sampling<>, isa_sampling<>, succinct_byte_alphabet<> >,
csa_wt<wt_hutu<>, 32, 32, sa_order_sa_sampling<>, isa_sampling<>, byte_alphabet>,
csa_wt<wt_hutu<>, 32, 32, sa_order_sa_sampling<>, isa_sampling<>, succinct_byte_alphabet<> >,
csa_wt<wt_hutu<bit_vector_il<> >, 32, 32, sa_order_sa_sampling<>, isa_sampling<>, byte_alphabet>
> Implementations;
TYPED_TEST_CASE(search_bidirectional_test, Implementations);
//! Compare bidirectional search and backward search
TYPED_TEST(search_bidirectional_test, bidirectional_search)
{
bool debug = false;
TypeParam csa1;
TypeParam csa1_rev;
construct(csa1, test_file, 1);
construct(csa1_rev, test_file_rev, 1);
std::mt19937_64 rng(13);
std::uniform_int_distribution<uint64_t> distribution(0, csa1.size()-1);
for (size_type h = 0; h<1000; ++h) {
//search for an existing pattern forward and backward using bidirectional_search:
size_type x = 4; // number of characters that are added to the pattern in each step
size_type steps = 10; // maximal number of patternparts that are searched for
size_type start = distribution(rng); //inclusive
size_type end = start; //exclusive
bool forward = false;
size_type l_rev = 0;
size_type r_rev = csa1_rev.size()-1;
size_type l = 0;
size_type r = csa1.size()-1;
size_type occ = csa1.size(); // size of the interval equals the number of occurrences of the pattern in the text (init empty pattern)
size_type i,pos;
// alternating forward and backward search using bidirectional_search: alternating every x characters
for (size_type rep = 0; rep<steps and start>=x and end<=csa1.size()-x ; ++rep) {
string newpat = "";
if (forward) {
//forward
i = 0;
pos = end;
for (size_type j=0; j<x; ++j) {
newpat.push_back(csa1.text[pos+j]);
}
occ = bidirectional_search_forward(csa1, csa1_rev, l, r, l_rev, r_rev, newpat.begin(), newpat.end(), l, r, l_rev, r_rev);
i = newpat.size();
end += i;
} else {
//backward
i = 0;
pos = start-1;
for (size_type j=0; j<x; ++j) {
newpat.push_back(csa1.text[pos-x+1+j]);
}
occ = bidirectional_search_backward(csa1, csa1_rev, l, r, l_rev, r_rev, newpat.begin(), newpat.end(), l, r, l_rev, r_rev);
i = newpat.size();
start -= i;
}
//output
if (debug) {
cout << "pattern (at text[" << start << ".." << end-1 << "]):" << endl;
for (size_type j=start; j<end; ++j) {
cout << csa1.text[j];
}
cout << endl;
if (occ) {
cout << "interval of pattern in csa1 is [" << l << ".." << r << "]" << endl;
cout << "interval of reverse pattern in csa1_rev is [" << l_rev << ".." << r_rev << "]" << endl;
}
cout << endl;
}
ASSERT_LT((size_type)0, occ) << "Pattern not found in input."; // make sure pattern was found in input (it has to be because we took part of the input as pattern)
{
//check using backward_search
string pat = "";
for (size_type j=0; j<end-start; ++j) {
pat.push_back(csa1.text[start+j]);
}
size_type b_l,b_r;
size_type b_occ = backward_search(csa1, 0, csa1.size()-1, pat.begin(), pat.end(), b_l, b_r);
ASSERT_EQ(b_occ, occ) << "Bidirectional_search and backward_search found different number of occurrences of the pattern.";
ASSERT_EQ(b_l, l) << "Bidirectional_search and backward_search found different left border";
ASSERT_EQ(b_r, r) << "Bidirectional_search and backward_search found different right border";
}
//change direction
forward = !forward;
}
}
}
} // namespace
int main(int argc, char** argv)
{
::testing::InitGoogleTest(&argc, argv);
if (argc < 2) {
cout << "Usage: " << argv[0] << " test_file" << endl;
cout << " (1) Reverses test_file; stores it in test_file_rev." << endl;
cout << " (2) Performs tests." << endl;
cout << " (3) Deletes test_file_reverse." << endl;
return 1;
}
test_file = argv[1];
test_file_rev = test_file + "_rev";
{
//reverse input
int_vector<8> text;
load_vector_from_file(text, test_file, 1);
size_type n = text.size();
int_vector<8> text_rev(n);
for (size_type i=0; i<n; i++) {
text_rev[n-1-i] = text[i];
}
char* text2 = (char*)text_rev.data();
ofstream of(test_file_rev, ofstream::binary);
of.write(text2, n);
of.close();
}
int result = RUN_ALL_TESTS();
sdsl::remove(test_file_rev);
return result;
}