xapian-core  1.4.22
api_queryparser.cc
Go to the documentation of this file.
1 
4 /* Copyright (C) 2002,2003,2004,2005,2006,2007,2008,2009,2010,2011,2012,2013,2015,2016,2019 Olly Betts
5  * Copyright (C) 2006,2007,2009 Lemur Consulting Ltd
6  *
7  * This program is free software; you can redistribute it and/or
8  * modify it under the terms of the GNU General Public License as
9  * published by the Free Software Foundation; either version 2 of the
10  * License, or (at your option) any later version.
11  *
12  * This program is distributed in the hope that it will be useful,
13  * but WITHOUT ANY WARRANTY; without even the implied warranty of
14  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15  * GNU General Public License for more details.
16  *
17  * You should have received a copy of the GNU General Public License
18  * along with this program; if not, write to the Free Software
19  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301
20  * USA
21  */
22 
23 #include <config.h>
24 
25 #include "api_queryparser.h"
26 
27 #define XAPIAN_DEPRECATED(D) D
28 #include <xapian.h>
29 
30 #include "apitest.h"
31 #include "cputimer.h"
32 #include "str.h"
33 #include "stringutils.h"
34 
35 #include <string>
36 #include <vector>
37 
38 using namespace std;
39 
40 #include "testsuite.h"
41 #include "testutils.h"
42 
43 struct test {
44  const char *query;
45  const char *expect;
46 };
47 
48 static const test test_or_queries[] = {
49  { "simple-example", "(simple@1 PHRASE 2 example@2)" },
50  { "time_t", "Ztime_t@1" },
51  { "stock -cooking", "(Zstock@1 AND_NOT Zcook@2)" },
52  { "foo -baz bar", "((Zfoo@1 OR Zbar@3) AND_NOT Zbaz@2)" },
53  { "d- school report", "(Zd@1 OR (Zschool@2 OR Zreport@3))" },
54  { "gtk+ -gnome", "(Zgtk+@1 AND_NOT Zgnome@2)" },
55  { "c++ -d--", "(Zc++@1 AND_NOT Zd@2)" },
56  { "Mg2+ Cl-", "(mg2+@1 OR cl@2)" },
57  { "\"c++ library\"", "(c++@1 PHRASE 2 library@2)" },
58  { "A&L A&RMCO AD&D", "(a&l@1 OR a&rmco@2 OR ad&d@3)" },
59  { "C# vs C++", "(c#@1 OR Zvs@2 OR c++@3)" },
60  { "j##", "Zj##@1" },
61  { "a#b", "(Za@1 OR Zb@2)" },
62  { "O.K. U.N.C.L.E XY.Z.", "(ok@1 OR uncle@2 OR (xy@3 PHRASE 2 z@4))" },
63  { "author:orwell animal farm", "(ZAorwel@1 OR Zanim@2 OR Zfarm@3)" },
64  { "author:Orwell Animal Farm", "(Aorwell@1 OR animal@2 OR farm@3)" },
65  // Regression test for bug reported in 0.9.6.
66  { "author:\"orwell\" title:\"animal\"", "(Aorwell@1 OR XTanimal@2)" },
67  // Regression test for bug related to one reported in 0.9.6.
68  { "author:(orwell) title:(animal)", "(ZAorwel@1 OR ZXTanim@2)" },
69  // Regression test for bug caused by fix for previous bug.
70  { "author:\"milne, a.a.\"", "(Amilne@1 PHRASE 3 Aa@2 PHRASE 3 Aa@3)" },
71  { "author:\"milne a.a.\"", "(Amilne@1 PHRASE 3 Aa@2 PHRASE 3 Aa@3)" },
72  // Regression test for bug reported in 0.9.7.
73  { "site:/path/name", "0 * H/path/name" },
74  // Regression test for bug introduced (and fixed) in SVN prior to 1.0.0.
75  { "author:/path/name", "(Apath@1 PHRASE 2 Aname@2)" },
76  // Feature tests for change to allow phrase generators after prefix in 1.2.4.
77  { "author:/path", "ZApath@1" },
78  { "author:-Foo", "Afoo@1" },
79  { "author:/", "Zauthor@1" },
80  { "author::", "Zauthor@1" },
81  { "author:/ foo", "(Zauthor@1 OR Zfoo@2)" },
82  { "author:: foo", "(Zauthor@1 OR Zfoo@2)" },
83  { "author::foo", "(author@1 PHRASE 2 foo@2)" },
84  { "author:/ AND foo", "(Zauthor@1 AND Zfoo@2)" },
85  { "author:: AND foo", "(Zauthor@1 AND Zfoo@2)" },
86  { "foo AND author:/", "(Zfoo@1 AND Zauthor@2)" },
87  { "foo AND author::", "(Zfoo@1 AND Zauthor@2)" },
88  // Regression test for bug introduced into (and fixed) in SVN prior to 1.0.0.
89  { "author:(title::case)", "(Atitle@1 PHRASE 2 Acase@2)" },
90  // Regression test for bug fixed in 1.0.4 - the '+' would be ignored there
91  // because the whitespace after the '"' wasn't noticed.
92  { "\"hello world\" +python", "(Zpython@3 AND_MAYBE (hello@1 PHRASE 2 world@2))" },
93  // In 1.1.0, NON_SPACING_MARK was added as a word character.
94  { "\xd8\xa7\xd9\x84\xd8\xb1\xd9\x91\xd8\xad\xd9\x85\xd9\x86", "Z\xd8\xa7\xd9\x84\xd8\xb1\xd9\x91\xd8\xad\xd9\x85\xd9\x86@1" },
95  // In 1.1.4, ENCLOSING_MARK and COMBINING_SPACING_MARK were added, and
96  // code to ignore several zero-width space characters was added.
97  { "\xe1\x80\x9d\xe1\x80\xae\xe2\x80\x8b\xe1\x80\x80\xe1\x80\xae\xe2\x80\x8b\xe1\x80\x95\xe1\x80\xad\xe2\x80\x8b\xe1\x80\x9e\xe1\x80\xaf\xe1\x80\xb6\xe1\x80\xb8\xe2\x80\x8b\xe1\x80\x85\xe1\x80\xbd\xe1\x80\xb2\xe2\x80\x8b\xe1\x80\x9e\xe1\x80\xb0\xe2\x80\x8b\xe1\x80\x99\xe1\x80\xbb\xe1\x80\xac\xe1\x80\xb8\xe1\x80\x80", "Z\xe1\x80\x9d\xe1\x80\xae\xe1\x80\x80\xe1\x80\xae\xe1\x80\x95\xe1\x80\xad\xe1\x80\x9e\xe1\x80\xaf\xe1\x80\xb6\xe1\x80\xb8\xe1\x80\x85\xe1\x80\xbd\xe1\x80\xb2\xe1\x80\x9e\xe1\x80\xb0\xe1\x80\x99\xe1\x80\xbb\xe1\x80\xac\xe1\x80\xb8\xe1\x80\x80@1" },
98  { "unmatched\"", "unmatched@1" },
99  { "unmatched \" \" ", "Zunmatch@1" },
100  { "hyphen-ated\" ", "(hyphen@1 PHRASE 2 ated@2)" },
101  { "hyphen-ated\" \"", "(hyphen@1 PHRASE 2 ated@2)" },
102  { "\"1.4\"", "1.4@1" },
103  { "\"1.\"", "1@1" },
104  { "\"A#.B.\"", "(a#@1 PHRASE 2 b@2)" },
105  { "\" Xapian QueryParser\" parses queries", "((xapian@1 PHRASE 2 queryparser@2) OR (Zpars@3 OR Zqueri@4))" },
106  { "\" xapian queryParser\" parses queries", "((xapian@1 PHRASE 2 queryparser@2) OR (Zpars@3 OR Zqueri@4))" },
107  { "h\xc3\xb6hle", "Zh\xc3\xb6hle@1" },
108  { "one +two three", "(Ztwo@2 AND_MAYBE (Zone@1 OR Zthree@3))" },
109  { "subject:test other", "(ZXTtest@1 OR Zother@2)" },
110  { "subject:\"space flight\"", "(XTspace@1 PHRASE 2 XTflight@2)" },
111  { "author:(twain OR poe) OR flight", "(ZAtwain@1 OR ZApoe@2 OR Zflight@3)" },
112  { "author:(twain OR title:pit OR poe)", "(ZAtwain@1 OR ZXTpit@2 OR ZApoe@3)" },
113  { "title:2001 title:space", "(XT2001@1 OR ZXTspace@2)" },
114  { "(title:help)", "ZXThelp@1" },
115  { "beer NOT \"orange juice\"", "(Zbeer@1 AND_NOT (orange@2 PHRASE 2 juice@3))" },
116  { "beer AND NOT lager", "(Zbeer@1 AND_NOT Zlager@2)" },
117  { "beer AND -lager", "(Zbeer@1 AND_NOT Zlager@2)" },
118  { "beer AND +lager", "(Zbeer@1 AND Zlager@2)" },
119  { "A OR B NOT C", "(a@1 OR (b@2 AND_NOT c@3))" },
120  { "A OR B AND NOT C", "(a@1 OR (b@2 AND_NOT c@3))" },
121  { "A OR B AND -C", "(a@1 OR (b@2 AND_NOT c@3))" },
122  { "A OR B AND +C", "(a@1 OR (b@2 AND c@3))" },
123  { "A OR B XOR C", "(a@1 OR (b@2 XOR c@3))" },
124  { "A XOR B NOT C", "(a@1 XOR (b@2 AND_NOT c@3))" },
125  { "one AND two", "(Zone@1 AND Ztwo@2)" },
126  { "one A.N.D. two", "(Zone@1 OR and@2 OR Ztwo@3)" },
127  { "one \xc3\x81ND two", "(Zone@1 OR \xc3\xa1nd@2 OR Ztwo@3)" },
128  { "one author:AND two", "(Zone@1 OR Aand@2 OR Ztwo@3)" },
129  { "author:hyphen-ated", "(Ahyphen@1 PHRASE 2 Aated@2)" },
130  { "cvs site:xapian.org", "(Zcvs@1 FILTER Hxapian.org)" },
131  { "cvs -site:xapian.org", "(Zcvs@1 AND_NOT Hxapian.org)" },
132  { "foo -site:xapian.org bar", "((Zfoo@1 OR Zbar@2) AND_NOT Hxapian.org)" },
133  { "site:xapian.org mail", "(Zmail@1 FILTER Hxapian.org)" },
134  { "-site:xapian.org mail", "(Zmail@1 AND_NOT Hxapian.org)" },
135  { "mail AND -site:xapian.org", "(Zmail@1 AND_NOT Hxapian.org)" },
136  { "-Wredundant-decls", "(wredundant@1 PHRASE 2 decls@2)" },
137  { "site:xapian.org", "0 * Hxapian.org" },
138  { "mug +site:xapian.org -site:cvs.xapian.org", "((Zmug@1 FILTER Hxapian.org) AND_NOT Hcvs.xapian.org)" },
139  { "mug -site:cvs.xapian.org +site:xapian.org", "((Zmug@1 FILTER Hxapian.org) AND_NOT Hcvs.xapian.org)" },
140  { "mug +site:xapian.org AND -site:cvs.xapian.org", "((Zmug@1 FILTER Hxapian.org) AND_NOT Hcvs.xapian.org)" },
141  { "mug site:xapian.org AND -site:cvs.xapian.org", "((Zmug@1 FILTER Hxapian.org) AND_NOT Hcvs.xapian.org)" },
142  { "mug site:xapian.org AND +site:cvs.xapian.org", "((Zmug@1 FILTER Hxapian.org) AND 0 * Hcvs.xapian.org)" },
143  { "NOT windows", "Syntax: <expression> NOT <expression>" },
144  { "a AND (NOT b)", "Syntax: <expression> NOT <expression>" },
145  { "AND NOT windows", "Syntax: <expression> AND NOT <expression>" },
146  { "AND -windows", "Syntax: <expression> AND <expression>" },
147  { "gordian NOT", "Syntax: <expression> NOT <expression>" },
148  { "gordian AND NOT", "Syntax: <expression> AND NOT <expression>" },
149  { "gordian AND -", "Syntax: <expression> AND <expression>" },
150  { "foo OR (something AND)", "Syntax: <expression> AND <expression>" },
151  { "OR foo", "Syntax: <expression> OR <expression>" },
152  { "XOR", "Syntax: <expression> XOR <expression>" },
153  // Regression test for bug fix in 1.4.13.
154  { "a OR -b", "Syntax: <expression> OR <expression>" },
155  { "hard\xa0space", "(Zhard@1 OR Zspace@2)" },
156  { " white\r\nspace\ttest ", "(Zwhite@1 OR Zspace@2 OR Ztest@3)" },
157  { "one AND two three", "(Zone@1 AND (Ztwo@2 OR Zthree@3))" },
158  { "one two AND three", "((Zone@1 OR Ztwo@2) AND Zthree@3)" },
159  { "one AND two/three", "(Zone@1 AND (two@2 PHRASE 2 three@3))" },
160  { "one AND /two/three", "(Zone@1 AND (two@2 PHRASE 2 three@3))" },
161  { "one AND/two/three", "(Zone@1 AND (two@2 PHRASE 2 three@3))" },
162  { "one +/two/three", "((two@2 PHRASE 2 three@3) AND_MAYBE Zone@1)" },
163  { "one//two", "(one@1 PHRASE 2 two@2)" },
164  { "\"missing quote", "(missing@1 PHRASE 2 quote@2)" },
165  { "DVD+RW", "(dvd@1 OR rw@2)" }, // Would a phrase be better?
166  { "+\"must have\" optional", "((must@1 PHRASE 2 have@2) AND_MAYBE Zoption@3)" },
167  { "one NEAR two NEAR three", "(one@1 NEAR 12 two@2 NEAR 12 three@3)" },
168  { "something NEAR/3 else", "(something@1 NEAR 4 else@2)" },
169  { "a NEAR/6 b NEAR c", "(a@1 NEAR 8 b@2 NEAR 8 c@3)" },
170  { "something ADJ else", "(something@1 PHRASE 11 else@2)" },
171  { "something ADJ/3 else", "(something@1 PHRASE 4 else@2)" },
172  { "a ADJ/6 b ADJ c", "(a@1 PHRASE 8 b@2 PHRASE 8 c@3)" },
173  // Regression test - Unicode character values were truncated to 8 bits
174  // before testing C_isdigit(), so this rather artificial example parsed
175  // to: (a@1 NEAR 262 b@2)
176  { "a NEAR/\xc4\xb5 b", "(Za@1 OR (near@2 PHRASE 2 \xc4\xb5@3) OR Zb@4)" },
177  { "a ADJ/\xc4\xb5 b", "(Za@1 OR (adj@2 PHRASE 2 \xc4\xb5@3) OR Zb@4)" },
178  // Regression test - the first two cases were parsed as if the '/' were a
179  // space, which was inconsistent with the second two. Fixed in 1.2.5.
180  { "a NEAR/b", "(Za@1 OR (near@2 PHRASE 2 b@3))" },
181  { "a ADJ/b", "(Za@1 OR (adj@2 PHRASE 2 b@3))" },
182  { "a NEAR/b c", "(Za@1 OR (near@2 PHRASE 2 b@3) OR Zc@4)" },
183  { "a ADJ/b c", "(Za@1 OR (adj@2 PHRASE 2 b@3) OR Zc@4)" },
184  // Regression tests - + and - didn't work on bracketed subexpressions prior
185  // to 1.0.2.
186  { "+(one two) three", "((Zone@1 OR Ztwo@2) AND_MAYBE Zthree@3)" },
187  { "zero -(one two)", "(Zzero@1 AND_NOT (Zone@2 OR Ztwo@3))" },
188  // Feature tests that ':' is inserted between prefix and term correctly:
189  { "category:Foo", "0 * XCAT:Foo" },
190  { "category:foo", "0 * XCATfoo" },
191  { "category:\xc3\x96oo", "0 * XCAT\xc3\x96oo" },
192  { "category::colon", "0 * XCAT::colon" },
193  // Feature tests for quoted boolean terms:
194  { "category:\"Hello world\"", "0 * XCAT:Hello world" },
195  { "category:\"literal \"\"\"", "0 * XCATliteral \"" },
196  { "category:\" \"", "0 * XCAT " },
197  { "category:\"\"", "0 * XCAT" },
198  { "category:\"(unterminated)", "0 * XCAT(unterminated)" },
199  // Feature tests for curly double quotes:
200  { "“curly quotes”", "(curly@1 PHRASE 2 quotes@2)" },
201  // Test "" inside quoted phrase doesn't end the phrase (for consistency
202  // with "" being an escape " in a quoted boolean term.
203  { "subject:\"foo\"\"bar\"", "(XTfoo@1 PHRASE 2 XTbar@2)" },
204  // Feature tests for implicitly closing brackets:
205  { "(foo", "Zfoo@1" },
206  { "(foo XOR bar", "(Zfoo@1 XOR Zbar@2)" },
207  { "(foo XOR (bar AND baz)", "(Zfoo@1 XOR (Zbar@2 AND Zbaz@3))" },
208  { "(foo XOR (bar AND baz", "(Zfoo@1 XOR (Zbar@2 AND Zbaz@3))" },
209  // Slightly arbitrarily we accept mismatched quotes.
210  { "\"curly quotes”", "(curly@1 PHRASE 2 quotes@2)" },
211  { "“curly quotes\"", "(curly@1 PHRASE 2 quotes@2)" },
212  { "“curly quotes“", "(curly@1 PHRASE 2 quotes@2)" },
213  { "”curly quotes”", "(curly@1 PHRASE 2 quotes@2)" },
214  { "author:“orwell” title:“animal\"", "(Aorwell@1 OR XTanimal@2)" },
215  { "author:\"orwell” title:“animal”", "(Aorwell@1 OR XTanimal@2)" },
216  { "author:“milne, a.a.”", "(Amilne@1 PHRASE 3 Aa@2 PHRASE 3 Aa@3)" },
217  { "author:“milne, a.a.\"", "(Amilne@1 PHRASE 3 Aa@2 PHRASE 3 Aa@3)" },
218  { "author:\"milne a.a.”", "(Amilne@1 PHRASE 3 Aa@2 PHRASE 3 Aa@3)" },
219  { "“hello world” +python", "(Zpython@3 AND_MAYBE (hello@1 PHRASE 2 world@2))" },
220  { "unmatched“", "Zunmatch@1" },
221  { "unmatched”", "Zunmatch@1" },
222  { "unmatched “ ” ", "Zunmatch@1" },
223  { "unmatched \" ” ", "Zunmatch@1" },
224  { "unmatched “ \" ", "Zunmatch@1" },
225  { "hyphen-ated“ ", "(hyphen@1 PHRASE 2 ated@2)" },
226  { "hyphen-ated” ", "(hyphen@1 PHRASE 2 ated@2)" },
227  { "hyphen-ated“ ”", "(hyphen@1 PHRASE 2 ated@2)" },
228  { "hyphen-ated“ \"", "(hyphen@1 PHRASE 2 ated@2)" },
229  { "hyphen-ated\" ”", "(hyphen@1 PHRASE 2 ated@2)" },
230  { "“1.4”", "1.4@1" },
231  { "“1.\"", "1@1" },
232  { "\"A#.B.”", "(a#@1 PHRASE 2 b@2)" },
233  { "“ Xapian QueryParser” parses queries", "((xapian@1 PHRASE 2 queryparser@2) OR (Zpars@3 OR Zqueri@4))" },
234  { "“ xapian queryParser\" parses queries", "((xapian@1 PHRASE 2 queryparser@2) OR (Zpars@3 OR Zqueri@4))" },
235  { "beer NOT “orange juice”", "(Zbeer@1 AND_NOT (orange@2 PHRASE 2 juice@3))" },
236  { "“missing quote", "(missing@1 PHRASE 2 quote@2)" },
237  { "+“must have” optional", "((must@1 PHRASE 2 have@2) AND_MAYBE Zoption@3)" },
238  { "category:“Hello world”", "0 * XCAT:Hello world" },
239  { "category:“literal \"\"”", "0 * XCATliteral \"" },
240  { "category:“ ”", "0 * XCAT " },
241  { "category:\" ”\"", "0 * XCAT ”" },
242  { "category:\" ”", "0 * XCAT ”" },
243  { "category:“ \"", "0 * XCAT " },
244  { "category:“”", "0 * XCAT" },
245  { "category:\"”\"", "0 * XCAT”" },
246  { "category:\"”", "0 * XCAT”" },
247  { "category:“\"", "0 * XCAT" },
248  { "category:“(unterminated)", "0 * XCAT(unterminated)" },
249  // Real world examples from tweakers.net:
250  { "Call to undefined function: imagecreate()", "(call@1 OR Zto@2 OR Zundefin@3 OR Zfunction@4 OR imagecreate@5)" },
251  { "mysql_fetch_row(): supplied argument is not a valid MySQL result resource", "(mysql_fetch_row@1 OR (Zsuppli@2 OR Zargument@3 OR Zis@4 OR Znot@5 OR Za@6 OR Zvalid@7 OR mysql@8 OR Zresult@9 OR Zresourc@10))" },
252  { "php date() nedelands", "(Zphp@1 OR date@2 OR Znedeland@3)" },
253  { "wget domein --http-user", "(Zwget@1 OR Zdomein@2 OR (http@3 PHRASE 2 user@4))" },
254  { "@home problemen", "(Zhome@1 OR Zproblemen@2)" },
255  { "'ipacsum'", "Zipacsum@1" },
256  { "canal + ", "Zcanal@1" },
257  { "/var/run/mysqld/mysqld.sock", "(var@1 PHRASE 5 run@2 PHRASE 5 mysqld@3 PHRASE 5 mysqld@4 PHRASE 5 sock@5)" },
258  { "\"QSI-161 drivers\"", "(qsi@1 PHRASE 3 161@2 PHRASE 3 drivers@3)" },
259  { "\"e-cube\" barebone", "((e@1 PHRASE 2 cube@2) OR Zbarebon@3)" },
260  { "\"./httpd: symbol not found: dlopen\"", "(httpd@1 PHRASE 5 symbol@2 PHRASE 5 not@3 PHRASE 5 found@4 PHRASE 5 dlopen@5)" },
261  { "ERROR 2003: Can't connect to MySQL server on 'localhost' (10061)", "(error@1 OR 2003@2 OR can't@3 OR Zconnect@4 OR Zto@5 OR mysql@6 OR Zserver@7 OR Zon@8 OR Zlocalhost@9 OR 10061@10)" },
262  { "location.href = \"\"", "(location@1 PHRASE 2 href@2)" },
263  { "method=\"post\" action=\"\">", "(method@1 OR post@2 OR action@3)" },
264  { "behuizing 19\" inch", "(Zbehuiz@1 OR 19@2 OR inch@3)" },
265  { "19\" rack", "(19@1 OR rack@2)" },
266  { "3,5\" mainboard", "(3,5@1 OR mainboard@2)" },
267  { "553 sorry, that domain isn't in my list of allowed rcpthosts (#5.7.1)", "(553@1 OR Zsorri@2 OR (Zthat@3 OR Zdomain@4 OR Zisn't@5 OR Zin@6 OR Zmy@7 OR Zlist@8 OR Zof@9 OR Zallow@10 OR Zrcpthost@11) OR 5.7.1@12)" },
268  { "data error (clic redundancy check)", "(Zdata@1 OR Zerror@2 OR (Zclic@3 OR Zredund@4 OR Zcheck@5))" },
269  { "? mediaplayer 9\"", "(Zmediaplay@1 OR 9@2)" },
270  { "date(\"w\")", "(date@1 OR w@2)" },
271  { "Syntaxisfout (operator ontbreekt ASP", "(syntaxisfout@1 OR (Zoper@2 OR Zontbreekt@3 OR asp@4))" },
272  { "Request.ServerVariables(\"logon_user\")", "((request@1 PHRASE 2 servervariables@2) OR logon_user@3)" },
273  { "ASP \"request.form\" van \\\"enctype=\"MULTIPART/FORM-DATA\"\\\"", "(asp@1 OR (request@2 PHRASE 2 form@3) OR Zvan@4 OR enctype@5 OR (multipart@6 PHRASE 3 form@7 PHRASE 3 data@8))" },
274  { "USER ftp (Login failed): Invalid shell: /sbin/nologin", "(user@1 OR Zftp@2 OR (login@3 OR Zfail@4) OR (invalid@5 OR Zshell@6) OR (sbin@7 PHRASE 2 nologin@8))" },
275  { "ip_masq_new(proto=TCP)", "(ip_masq_new@1 OR proto@2 OR tcp@3)" },
276  { "\"document.write(\"", "(document@1 PHRASE 2 write@2)" },
277  { "ERROR 1045: Access denied for user: 'root@localhost' (Using password: NO)", "(error@1 OR 1045@2 OR access@3 OR Zdeni@4 OR Zfor@5 OR Zuser@6 OR (root@7 PHRASE 2 localhost@8) OR (using@9 OR Zpassword@10 OR no@11))" },
278  { "TIP !! subtitles op TV-out (via DVD max g400)", "(tip@1 OR (Zsubtitl@2 OR Zop@3) OR (tv@4 PHRASE 2 out@5) OR (Zvia@6 OR dvd@7 OR Zmax@8 OR Zg400@9))" },
279  { "Gigabyte 8PE667 (de Ultra versie) of Asus A7N8X Deluxe", "(gigabyte@1 OR 8pe667@2 OR (Zde@3 OR ultra@4 OR Zversi@5) OR (Zof@6 OR asus@7 OR a7n8x@8 OR deluxe@9))" },
280  { "\"1) Ze testen 8x AF op de GFFX tegen \"", "(1@1 PHRASE 9 ze@2 PHRASE 9 testen@3 PHRASE 9 8x@4 PHRASE 9 af@5 PHRASE 9 op@6 PHRASE 9 de@7 PHRASE 9 gffx@8 PHRASE 9 tegen@9)" },
281  { "\") Ze houden geen rekening met de kwaliteit van AF. Als ze dat gedaan hadden dan waren ze tot de conclusie gekomen dat Performance AF (dus Bilinear AF) op de 9700Pro goed te vergelijken is met Balanced AF op de GFFX. En dan hadden ze ook gezien dat de GFFX niet kan tippen aan de Quality AF van de 9700Pro.\"", "(ze@1 PHRASE 59 houden@2 PHRASE 59 geen@3 PHRASE 59 rekening@4 PHRASE 59 met@5 PHRASE 59 de@6 PHRASE 59 kwaliteit@7 PHRASE 59 van@8 PHRASE 59 af@9 PHRASE 59 als@10 PHRASE 59 ze@11 PHRASE 59 dat@12 PHRASE 59 gedaan@13 PHRASE 59 hadden@14 PHRASE 59 dan@15 PHRASE 59 waren@16 PHRASE 59 ze@17 PHRASE 59 tot@18 PHRASE 59 de@19 PHRASE 59 conclusie@20 PHRASE 59 gekomen@21 PHRASE 59 dat@22 PHRASE 59 performance@23 PHRASE 59 af@24 PHRASE 59 dus@25 PHRASE 59 bilinear@26 PHRASE 59 af@27 PHRASE 59 op@28 PHRASE 59 de@29 PHRASE 59 9700pro@30 PHRASE 59 goed@31 PHRASE 59 te@32 PHRASE 59 vergelijken@33 PHRASE 59 is@34 PHRASE 59 met@35 PHRASE 59 balanced@36 PHRASE 59 af@37 PHRASE 59 op@38 PHRASE 59 de@39 PHRASE 59 gffx@40 PHRASE 59 en@41 PHRASE 59 dan@42 PHRASE 59 hadden@43 PHRASE 59 ze@44 PHRASE 59 ook@45 PHRASE 59 gezien@46 PHRASE 59 dat@47 PHRASE 59 de@48 PHRASE 59 gffx@49 PHRASE 59 niet@50 PHRASE 59 kan@51 PHRASE 59 tippen@52 PHRASE 59 aan@53 PHRASE 59 de@54 PHRASE 59 quality@55 PHRASE 59 af@56 PHRASE 59 van@57 PHRASE 59 de@58 PHRASE 59 9700pro@59)" },
282  { "\"Ze houden geen rekening met de kwaliteit van AF. Als ze dat gedaan hadden dan waren ze tot de conclusie gekomen dat Performance AF (dus Bilinear AF) op de 9700Pro goed te vergelijken is met Balanced AF op de GFFX. En dan hadden ze ook gezien dat de GFFX niet kan tippen aan de Quality AF van de 9700Pro.\"", "(ze@1 PHRASE 59 houden@2 PHRASE 59 geen@3 PHRASE 59 rekening@4 PHRASE 59 met@5 PHRASE 59 de@6 PHRASE 59 kwaliteit@7 PHRASE 59 van@8 PHRASE 59 af@9 PHRASE 59 als@10 PHRASE 59 ze@11 PHRASE 59 dat@12 PHRASE 59 gedaan@13 PHRASE 59 hadden@14 PHRASE 59 dan@15 PHRASE 59 waren@16 PHRASE 59 ze@17 PHRASE 59 tot@18 PHRASE 59 de@19 PHRASE 59 conclusie@20 PHRASE 59 gekomen@21 PHRASE 59 dat@22 PHRASE 59 performance@23 PHRASE 59 af@24 PHRASE 59 dus@25 PHRASE 59 bilinear@26 PHRASE 59 af@27 PHRASE 59 op@28 PHRASE 59 de@29 PHRASE 59 9700pro@30 PHRASE 59 goed@31 PHRASE 59 te@32 PHRASE 59 vergelijken@33 PHRASE 59 is@34 PHRASE 59 met@35 PHRASE 59 balanced@36 PHRASE 59 af@37 PHRASE 59 op@38 PHRASE 59 de@39 PHRASE 59 gffx@40 PHRASE 59 en@41 PHRASE 59 dan@42 PHRASE 59 hadden@43 PHRASE 59 ze@44 PHRASE 59 ook@45 PHRASE 59 gezien@46 PHRASE 59 dat@47 PHRASE 59 de@48 PHRASE 59 gffx@49 PHRASE 59 niet@50 PHRASE 59 kan@51 PHRASE 59 tippen@52 PHRASE 59 aan@53 PHRASE 59 de@54 PHRASE 59 quality@55 PHRASE 59 af@56 PHRASE 59 van@57 PHRASE 59 de@58 PHRASE 59 9700pro@59)" },
283  { "$structure = imap_header($mbox, $tt);", "(Zstructur@1 OR imap_header@2 OR Zmbox@3 OR Ztt@4)" },
284  { "\"ifup: Could not get a valid interface name: -> skipped\"", "(ifup@1 PHRASE 9 could@2 PHRASE 9 not@3 PHRASE 9 get@4 PHRASE 9 a@5 PHRASE 9 valid@6 PHRASE 9 interface@7 PHRASE 9 name@8 PHRASE 9 skipped@9)" },
285  { "Er kan geen combinatie van filters worden gevonden om de gegevensstroom te genereren. (Error=80040218)", "(er@1 OR Zkan@2 OR Zgeen@3 OR Zcombinati@4 OR Zvan@5 OR Zfilter@6 OR Zworden@7 OR Zgevonden@8 OR Zom@9 OR Zde@10 OR Zgegevensstroom@11 OR Zte@12 OR Zgenereren@13 OR (error@14 OR 80040218@15))" },
286  { "ereg_replace(\"\\\\\",\"\\/\"", "ereg_replace@1" },
287  { "\\\\\"divx+geen+geluid\\\\\"", "(divx@1 PHRASE 3 geen@2 PHRASE 3 geluid@3)" },
288  { "lcase(\"string\")", "(lcase@1 OR string@2)" },
289  { "isEmpty( ) functie in visual basic", "(isempty@1 OR (Zfuncti@2 OR Zin@3 OR Zvisual@4 OR Zbasic@5))" },
290  { "*** stop: 0x0000001E (0xC0000005,0x00000000,0x00000000,0x00000000)", "(Zstop@1 OR 0x0000001e@2 OR 0xc0000005,0x00000000,0x00000000,0x00000000@3)" },
291  { "\"ctrl+v+c+a fout\"", "(ctrl@1 PHRASE 5 v@2 PHRASE 5 c@3 PHRASE 5 a@4 PHRASE 5 fout@5)" },
292  { "Server.CreateObject(\"ADODB.connection\")", "((server@1 PHRASE 2 createobject@2) OR (adodb@3 PHRASE 2 connection@4))" },
293  { "Presario 6277EA-XP model P4/28 GHz-120GB-DVD-CDRW (512MBWXP) (470048-012)", "(presario@1 OR (6277ea@2 PHRASE 2 xp@3) OR Zmodel@4 OR (p4@5 PHRASE 2 28@6) OR (ghz@7 PHRASE 4 120gb@8 PHRASE 4 dvd@9 PHRASE 4 cdrw@10) OR 512mbwxp@11 OR (470048@12 PHRASE 2 012@13))" },
294  { "Failed to connect agent. (AGENT=dbaxchg2, EC=UserId =NUll)", "(failed@1 OR Zto@2 OR Zconnect@3 OR Zagent@4 OR (agent@5 OR Zdbaxchg2@6 OR ec@7 OR userid@8 OR null@9))" },
295  { "delphi CreateOleObject(\"MSXML2.DomDocument\")", "(Zdelphi@1 OR createoleobject@2 OR (msxml2@3 PHRASE 2 domdocument@4))" },
296  { "Unhandled exeption in IEXPLORE.EXE (FTAPP.DLL)", "(unhandled@1 OR Zexept@2 OR Zin@3 OR (iexplore@4 PHRASE 2 exe@5) OR (ftapp@6 PHRASE 2 dll@7))" },
297  { "IBM High Rate Wireless LAN PCI Adapter (Low Profile Enabled)", "(ibm@1 OR high@2 OR rate@3 OR wireless@4 OR lan@5 OR pci@6 OR adapter@7 OR (low@8 OR profile@9 OR enabled@10))" },
298  { "asp ' en \"", "(Zasp@1 OR Zen@2)" },
299  { "Hercules 3D Prophet 8500 LE 64MB (OEM, Radeon 8500 LE)", "(hercules@1 OR 3d@2 OR prophet@3 OR 8500@4 OR le@5 OR 64mb@6 OR (oem@7 OR (radeon@8 OR 8500@9 OR le@10)))" },
300  { "session_set_cookie_params(echo \"hoi\")", "(session_set_cookie_params@1 OR Zecho@2 OR hoi@3)" },
301  { "windows update werkt niet (windows se", "(Zwindow@1 OR Zupdat@2 OR Zwerkt@3 OR Zniet@4 OR (Zwindow@5 OR Zse@6))" },
302  { "De statuscode van de fout is ( 0 x 4 , 0 , 0 , 0 )", "(de@1 OR Zstatuscod@2 OR Zvan@3 OR Zde@4 OR Zfout@5 OR Zis@6 OR (0@7 OR Zx@8 OR 4@9 OR 0@10 OR 0@11 OR 0@12))" },
303  { "sony +(u20 u-20)", "((Zu20@2 OR (u@3 PHRASE 2 20@4)) AND_MAYBE Zsoni@1)" },
304  { "[crit] (17)File exists: unable to create scoreboard (name-based shared memory failure)", "(Zcrit@1 OR 17@2 OR (file@3 OR Zexist@4 OR Zunabl@5 OR Zto@6 OR Zcreat@7 OR Zscoreboard@8) OR ((name@9 PHRASE 2 based@10) OR (Zshare@11 OR Zmemori@12 OR Zfailur@13)))" },
305  { "directories lokaal php (uitlezen OR inladen)", "(Zdirectori@1 OR Zlokaal@2 OR Zphp@3 OR (Zuitlezen@4 OR Zinladen@5))" },
306  { "(multi pc modem)+ (line sync)", "(Zmulti@1 OR Zpc@2 OR Zmodem@3 OR (Zline@4 OR Zsync@5))" },
307  { "xp 5.1.2600.0 (xpclient.010817-1148)", "(Zxp@1 OR 5.1.2600.0@2 OR (xpclient@3 PHRASE 3 010817@4 PHRASE 3 1148@5))" },
308  { "DirectDraw test results: Failure at step 5 (User verification of rectangles): HRESULT = 0x00000000 (error code) Direct3D 7 test results: Failure at step 32 (User verification of Direct3D rendering): HRESULT = 0x00000000 (error code) Direct3D 8 test results: Failure at step 32 (User verification of Direct3D rendering): HRESULT = 0x00000000 (error code) Direct3D 9 test results: Failure at step 32 (User verification of Direct3D rendering): HRESULT = 0x00000000 (error code)", "(directdraw@1 OR Ztest@2 OR Zresult@3 OR failure@4 OR Zat@5 OR Zstep@6 OR 5@7 OR (user@8 OR Zverif@9 OR Zof@10 OR Zrectangl@11) OR hresult@12 OR 0x00000000@13 OR (Zerror@14 OR Zcode@15) OR (direct3d@16 OR 7@17 OR Ztest@18 OR Zresult@19 OR failure@20 OR Zat@21 OR Zstep@22 OR 32@23) OR (user@24 OR Zverif@25 OR Zof@26 OR direct3d@27 OR Zrender@28) OR hresult@29 OR 0x00000000@30 OR (Zerror@31 OR Zcode@32) OR (direct3d@33 OR 8@34 OR Ztest@35 OR Zresult@36 OR failure@37 OR Zat@38 OR Zstep@39 OR 32@40) OR (user@41 OR Zverif@42 OR Zof@43 OR direct3d@44 OR Zrender@45) OR hresult@46 OR 0x00000000@47 OR (Zerror@48 OR Zcode@49) OR (direct3d@50 OR 9@51 OR Ztest@52 OR Zresult@53 OR failure@54 OR Zat@55 OR Zstep@56 OR 32@57) OR (user@58 OR Zverif@59 OR Zof@60 OR direct3d@61 OR Zrender@62) OR hresult@63 OR 0x00000000@64 OR (Zerror@65 OR Zcode@66))" },
309  { "Thermaltake Aquarius II waterkoeling (kompleet voor P4 en XP)", "(thermaltake@1 OR aquarius@2 OR ii@3 OR Zwaterkoel@4 OR (Zkompleet@5 OR Zvoor@6 OR p4@7 OR Zen@8 OR xp@9))" },
310  { "E3501 unable to add job to database (EC=-2005)", "(e3501@1 OR Zunabl@2 OR Zto@3 OR Zadd@4 OR Zjob@5 OR Zto@6 OR Zdatabas@7 OR (ec@8 OR 2005@9))" },
311  { "\"arp -s\" ip veranderen", "((arp@1 PHRASE 2 s@2) OR (Zip@3 OR Zveranderen@4))" },
312  { "header(\"content-type: application/octet-stream\");", "(header@1 OR (content@2 PHRASE 2 type@3) OR (application@4 PHRASE 3 octet@5 PHRASE 3 stream@6))" },
313  { "$datum = date(\"d-m-Y\");", "(Zdatum@1 OR date@2 OR (d@3 PHRASE 3 m@4 PHRASE 3 y@5))" },
314  { "\"'\" +asp", "Zasp@1" },
315  { "+session +[", "Zsession@1" },
316  { "Dit apparaat kan niet starten. (Code 10)", "(dit@1 OR Zapparaat@2 OR Zkan@3 OR Zniet@4 OR Zstarten@5 OR (code@6 OR 10@7))" },
317  { "\"You cannot use the Administration program while the Domino Server is running. Either shut down the Domino Server (but keep the file server running) or choose the ican labeled 'Lotus Notes' instead.\"", "(you@1 PHRASE 32 cannot@2 PHRASE 32 use@3 PHRASE 32 the@4 PHRASE 32 administration@5 PHRASE 32 program@6 PHRASE 32 while@7 PHRASE 32 the@8 PHRASE 32 domino@9 PHRASE 32 server@10 PHRASE 32 is@11 PHRASE 32 running@12 PHRASE 32 either@13 PHRASE 32 shut@14 PHRASE 32 down@15 PHRASE 32 the@16 PHRASE 32 domino@17 PHRASE 32 server@18 PHRASE 32 but@19 PHRASE 32 keep@20 PHRASE 32 the@21 PHRASE 32 file@22 PHRASE 32 server@23 PHRASE 32 running@24 PHRASE 32 or@25 PHRASE 32 choose@26 PHRASE 32 the@27 PHRASE 32 ican@28 PHRASE 32 labeled@29 PHRASE 32 lotus@30 PHRASE 32 notes@31 PHRASE 32 instead@32)" },
318  { "\"+irq +veranderen +xp\"", "(irq@1 PHRASE 3 veranderen@2 PHRASE 3 xp@3)" },
319  { "\"is not a member of 'operator``global namespace''' + c++", "(is@1 PHRASE 9 not@2 PHRASE 9 a@3 PHRASE 9 member@4 PHRASE 9 of@5 PHRASE 9 operator@6 PHRASE 9 global@7 PHRASE 9 namespace@8 PHRASE 9 c++@9)" },
320  { "mkdir() failed (File exists) php", "(mkdir@1 OR Zfail@2 OR (file@3 OR Zexist@4) OR Zphp@5)" },
321  { "laatsteIndex(int n)", "(laatsteindex@1 OR (Zint@2 OR Zn@3))" },
322  { "\"line+in\" OR \"c8783\"", "((line@1 PHRASE 2 in@2) OR c8783@3)" },
323  { "if ($_POST['Submit'])", "(Zif@1 OR (_post@2 OR submit@3))" },
324  { "NEC DVD+-RW ND-1300A", "(nec@1 OR (dvd+@2 PHRASE 2 rw@3) OR (nd@4 PHRASE 2 1300a@5))" },
325  { "*String not found* (*String not found*.)", "(string@1 OR Znot@2 OR found@3 OR (string@4 OR Znot@5 OR found@6))" },
326  { "MSI G4Ti4200-TD 128MB (GeForce4 Ti4200)", "(msi@1 OR (g4ti4200@2 PHRASE 2 td@3) OR 128mb@4 OR (geforce4@5 OR ti4200@6))" },
327  { "href=\"#\"", "href@1" },
328  { "Request.ServerVariables(\"REMOTE_USER\") javascript", "((request@1 PHRASE 2 servervariables@2) OR remote_user@3 OR Zjavascript@4)" },
329  { "XF86Config(-4) waar", "(xf86config@1 OR 4@2 OR Zwaar@3)" },
330  { "Unknown (tag 2000)", "(unknown@1 OR (Ztag@2 OR 2000@3))" },
331  { "KT4V(MS-6712)", "(kt4v@1 OR (ms@2 PHRASE 2 6712@3))" },
332  { "scheduled+AND+nieuwsgroepen+AND+updaten", "(Zschedul@1 AND Znieuwsgroepen@2 AND Zupdaten@3)" },
333  { "137(netbios-ns)", "(137@1 OR (netbios@2 PHRASE 2 ns@3))" },
334  { "HARWARE ERROR, TRACKING SERVO (4:0X09:0X01)", "(harware@1 OR error@2 OR (tracking@3 OR servo@4) OR (4@5 PHRASE 3 0x09@6 PHRASE 3 0x01@7))" },
335  { "Chr(10) wat is code van \" teken", "(chr@1 OR 10@2 OR (Zwat@3 OR Zis@4 OR Zcode@5 OR Zvan@6) OR Zteken@7)" },
336  { "wat is code van \" teken", "(Zwat@1 OR Zis@2 OR Zcode@3 OR Zvan@4 OR teken@5)" },
337  { "The Jet VBA file (VBAJET.dll for 16-bit version, VBAJET32.dll version", "(the@1 OR jet@2 OR vba@3 OR Zfile@4 OR ((vbajet@5 PHRASE 2 dll@6) OR Zfor@7 OR (16@8 PHRASE 2 bit@9) OR Zversion@10 OR (vbajet32@11 PHRASE 2 dll@12) OR Zversion@13))" },
338  { "Permission denied (publickey,password,keyboard-interactive).", "(permission@1 OR Zdeni@2 OR (Zpublickey@3 OR Zpassword@4 OR (keyboard@5 PHRASE 2 interactive@6)))" },
339  { "De lees- of schrijfbewerking (\"written\") op het geheugen is mislukt", "(de@1 OR Zlee@2 OR Zof@3 OR Zschrijfbewerk@4 OR written@5 OR (Zop@6 OR Zhet@7 OR Zgeheugen@8 OR Zis@9 OR Zmislukt@10))" },
340  { "Primary IDE channel no 80 conductor cable installed\"", "(primary@1 OR ide@2 OR Zchannel@3 OR Zno@4 OR 80@5 OR Zconductor@6 OR Zcabl@7 OR installed@8)" },
341  { "\"2020 NEAR zoom\"", "(2020@1 PHRASE 3 near@2 PHRASE 3 zoom@3)" },
342  { "setcookie(\"naam\",\"$user\");", "(setcookie@1 OR naam@2 OR user@3)" },
343  { "MSI 645 Ultra (MS-6547) Ver1", "(msi@1 OR 645@2 OR ultra@3 OR (ms@4 PHRASE 2 6547@5) OR ver1@6)" },
344  { "if ($HTTP", "(Zif@1 OR http@2)" },
345  { "data error(cyclic redundancy check)", "(Zdata@1 OR error@2 OR (Zcyclic@3 OR Zredund@4 OR Zcheck@5))" },
346  { "UObject::StaticAllocateObject <- (NULL None) <- UObject::StaticConstructObject <- InitEngine", "((uobject@1 PHRASE 2 staticallocateobject@2) OR (null@3 OR none@4) OR (uobject@5 PHRASE 2 staticconstructobject@6) OR initengine@7)" },
347  { "Failure at step 8 (Creating 3D Device)", "(failure@1 OR Zat@2 OR Zstep@3 OR 8@4 OR (creating@5 OR 3d@6 OR device@7))" },
348  { "Call Shell(\"notepad.exe\",", "(call@1 OR shell@2 OR (notepad@3 PHRASE 2 exe@4))" },
349  { "2.5\" harddisk converter", "(2.5@1 OR (harddisk@2 PHRASE 2 converter@3))" },
350  { "creative labs \"dvd+rw\"", "(Zcreativ@1 OR Zlab@2 OR (dvd@3 PHRASE 2 rw@4))" },
351  { "\"het beleid van deze computer staat u niet toe interactief", "(het@1 PHRASE 10 beleid@2 PHRASE 10 van@3 PHRASE 10 deze@4 PHRASE 10 computer@5 PHRASE 10 staat@6 PHRASE 10 u@7 PHRASE 10 niet@8 PHRASE 10 toe@9 PHRASE 10 interactief@10)" },
352  { "ati radeon \"driver cleaner", "(Zati@1 OR Zradeon@2 OR (driver@3 PHRASE 2 cleaner@4))" },
353  { "\"../\" path", "Zpath@1" },
354  { "(novell client) workstation only", "(Znovel@1 OR Zclient@2 OR (Zworkstat@3 OR Zonli@4))" },
355  { "Unable to find libgd.(a|so) anywhere", "(unable@1 OR Zto@2 OR Zfind@3 OR Zlibgd@4 OR Za@5 OR Zso@6 OR Zanywher@7)" },
356  { "\"libstdc++-libc6.1-1.so.2\"", "(libstdc++@1 PHRASE 5 libc6.1@2 PHRASE 5 1@3 PHRASE 5 so@4 PHRASE 5 2@5)" },
357  { "ipsec_setup (/etc/ipsec.conf, line 1) cannot open configuration file \"/etc/ipsec.conf\" -- `' aborted", "(Zipsec_setup@1 OR ((etc@2 PHRASE 3 ipsec@3 PHRASE 3 conf@4) OR (Zline@5 OR 1@6)) OR (Zcannot@7 OR Zopen@8 OR Zconfigur@9 OR Zfile@10) OR (etc@11 PHRASE 3 ipsec@12 PHRASE 3 conf@13) OR Zabort@14)" },
358  { "Forwarden van domeinnaam (naar HTTP adres)", "(forwarden@1 OR Zvan@2 OR Zdomeinnaam@3 OR (Znaar@4 OR http@5 OR Zadr@6))" },
359  { "Compaq HP, 146.8 GB (MPN-286716-B22) Hard Drives", "(compaq@1 OR hp@2 OR (146.8@3 OR gb@4) OR (mpn@5 PHRASE 3 286716@6 PHRASE 3 b22@7) OR (hard@8 OR drives@9))" },
360  { "httpd (no pid file) not running", "(Zhttpd@1 OR (Zno@2 OR Zpid@3 OR Zfile@4) OR (Znot@5 OR Zrun@6))" },
361  { "apache httpd (pid file) not running", "(Zapach@1 OR Zhttpd@2 OR (Zpid@3 OR Zfile@4) OR (Znot@5 OR Zrun@6))" },
362  { "Klasse is niet geregistreerd (Fout=80040154).", "(klasse@1 OR Zis@2 OR Zniet@3 OR Zgeregistreerd@4 OR (fout@5 OR 80040154@6))" },
363  { "\"dvd+r\" \"dvd-r\"", "((dvd@1 PHRASE 2 r@2) OR (dvd@3 PHRASE 2 r@4))" },
364  { "\"=\" tekens uit csvfile", "(Zteken@1 OR Zuit@2 OR Zcsvfile@3)" },
365  { "libc.so.6(GLIBC_2.3)", "((libc@1 PHRASE 3 so@2 PHRASE 3 6@3) OR glibc_2.3@4)" },
366  { "Sitecom Broadband xDSL / Cable Router 4S (DC-202)", "(sitecom@1 OR broadband@2 OR Zxdsl@3 OR (cable@4 OR router@5 OR 4s@6) OR (dc@7 PHRASE 2 202@8))" },
367  { "(t-mobile) bereik", "((t@1 PHRASE 2 mobile@2) OR Zbereik@3)" },
368  { "error LNK2001: unresolved external symbol \"public", "(Zerror@1 OR lnk2001@2 OR Zunresolv@3 OR Zextern@4 OR Zsymbol@5 OR public@6)" },
369  { "patch linux exploit -p)", "(Zpatch@1 OR Zlinux@2 OR Zexploit@3 OR Zp@4)" },
370  { "MYD not found (Errcode: 2)", "(myd@1 OR Znot@2 OR Zfound@3 OR (errcode@4 OR 2@5))" },
371  { "ob_start(\"ob_gzhandler\"); file download", "(ob_start@1 OR ob_gzhandler@2 OR (Zfile@3 OR Zdownload@4))" },
372  { "ECS Elitegroup K7VZA (VIA VT8363/VT8363A)", "(ecs@1 OR elitegroup@2 OR k7vza@3 OR (via@4 OR (vt8363@5 PHRASE 2 vt8363a@6)))" },
373  { "ASUS A7V8X (LAN + Serial-ATA + Firewire + Raid + Audio)", "(asus@1 OR a7v8x@2 OR (lan@3 OR (serial@4 PHRASE 2 ata@5) OR firewire@6 OR raid@7 OR audio@8))" },
374  { "Javascript:history.go(-1)", "((javascript@1 PHRASE 3 history@2 PHRASE 3 go@3) OR 1@4)" },
375  { "java :) als icon", "(Zjava@1 OR (Zal@2 OR Zicon@3))" },
376  { "onmouseover=setPointer(this", "(onmouseover@1 OR setpointer@2 OR Zthis@3)" },
377  { "\" in vbscript", "(in@1 PHRASE 2 vbscript@2)" },
378  { "IRC (FAQ OR (hulp NEAR bij))", "(irc@1 OR (faq@2 OR (hulp@3 NEAR 11 bij@4)))" },
379  { "setProperty(\"McSquare\"+i, _xscale, _xscale++);", "(setproperty@1 OR mcsquare@2 OR Zi@3 OR _xscale@4 OR _xscale++@5)" },
380  { "[warn] Apache does not support line-end comments. Consider using quotes around argument: \"#-1\"", "(Zwarn@1 OR (apache@2 OR Zdoe@3 OR Znot@4 OR Zsupport@5) OR (line@6 PHRASE 2 end@7) OR Zcomment@8 OR (consider@9 OR Zuse@10 OR Zquot@11 OR Zaround@12 OR Zargument@13) OR 1@14)" },
381  { "(php.ini) (memory_limit)", "((php@1 PHRASE 2 ini@2) OR Zmemory_limit@3)" },
382  { "line 8: syntax error near unexpected token `kernel_thread(f'", "(Zline@1 OR 8@2 OR Zsyntax@3 OR Zerror@4 OR Znear@5 OR Zunexpect@6 OR Ztoken@7 OR kernel_thread@8 OR Zf@9)" },
383  { "VXD NAVEX()@)", "(vxd@1 OR navex@2)" },
384  { "\"Iiyama AS4314UT 17\" \"", "(iiyama@1 PHRASE 3 as4314ut@2 PHRASE 3 17@3)" },
385  { "include (\"$id.html\");", "(Zinclud@1 OR (id@2 PHRASE 2 html@3))" },
386  { "include id.Today's date is: <? print (date (\"M d, Y\")); ?>hp", "(Zinclud@1 OR (id@2 PHRASE 2 today's@3) OR (Zdate@4 OR Zis@5) OR Zprint@6 OR (Zdate@7 OR (m@8 PHRASE 3 d@9 PHRASE 3 y@10)) OR Zhp@11)" },
387  { "(program files\\common) opstarten", "(Zprogram@1 OR (files@2 PHRASE 2 common@3) OR Zopstarten@4)" },
388  { "java \" string", "(Zjava@1 OR string@2)" },
389  { "+=", "" },
390  { "php +=", "Zphp@1" },
391  { "[php] ereg_replace(\".\"", "(Zphp@1 OR ereg_replace@2)" },
392  { "\"echo -e\" kleur", "((echo@1 PHRASE 2 e@2) OR Zkleur@3)" },
393  { "adobe premiere \"-1\"", "(Zadob@1 OR Zpremier@2 OR 1@3)" },
394  { "DVD brander \"+\" en \"-\"", "(dvd@1 OR Zbrander@2 OR Zen@3)" },
395  { "inspirion \"dvd+R\"", "(Zinspirion@1 OR (dvd@2 PHRASE 2 r@3))" },
396  { "asp 0x80040E14)", "(Zasp@1 OR 0x80040e14@2)" },
397  { "\"e-tech motorola router", "(e@1 PHRASE 4 tech@2 PHRASE 4 motorola@3 PHRASE 4 router@4)" },
398  { "bluetooth '1.3.2.19\"", "(Zbluetooth@1 OR 1.3.2.19@2)" },
399  { "ms +-connect", "(Zms@1 OR Zconnect@2)" },
400  { "php+print+\"", "(Zphp@1 OR print+@2)" },
401  { "athlon 1400 :welke videokaart\"", "(Zathlon@1 OR 1400@2 OR (Zwelk@3 OR videokaart@4))" },
402  { "+-dvd", "Zdvd@1" },
403  { "glftpd \"-new-\"", "(Zglftpd@1 OR new@2)" },
404  { "\"scandisk + dos5.0", "(scandisk@1 PHRASE 2 dos5.0@2)" },
405  { "socket\\(\\)", "socket@1" },
406  { "msn (e-tech) router", "(Zmsn@1 OR (e@2 PHRASE 2 tech@3) OR Zrouter@4)" },
407  { "Het grote Epox 8k3a+ ervaring/prob topic\"", "(het@1 OR Zgrote@2 OR epox@3 OR 8k3a+@4 OR (ervaring@5 PHRASE 2 prob@6) OR topic@7)" },
408  { "\"CF+bluetooth\"", "(cf@1 PHRASE 2 bluetooth@2)" },
409  { "kwaliteit (s-video) composite verschil tv out", "(Zkwaliteit@1 OR (s@2 PHRASE 2 video@3) OR (Zcomposit@4 OR Zverschil@5 OR Ztv@6 OR Zout@7))" },
410  { "Wie kan deze oude hardware nog gebruiken\" Deel", "(wie@1 OR Zkan@2 OR Zdeze@3 OR Zoud@4 OR Zhardwar@5 OR Znog@6 OR gebruiken@7 OR deel@8)" },
411  { "Public Declare Sub Sleep Lib \"kernel32\" (ByVal dwMilliseconds As Long)", "(public@1 OR declare@2 OR sub@3 OR sleep@4 OR lib@5 OR kernel32@6 OR (byval@7 OR Zdwmillisecond@8 OR as@9 OR long@10))" },
412  { "for inclusion (include_path='.:/usr/share/php')", "(Zfor@1 OR Zinclus@2 OR (include_path@3 OR (usr@4 PHRASE 3 share@5 PHRASE 3 php@6)))" },
413  { "\"muziek 2x zo snel\"\"", "(muziek@1 PHRASE 4 2x@2 PHRASE 4 zo@3 PHRASE 4 snel@4)" },
414  { "execCommand('inserthorizontalrule'", "(execcommand@1 OR Zinserthorizontalrul@2)" },
415  { "specs: IBM PS/2, Intel 8086 @ 25 mhz!!, 2 mb intern, 50 mb hd, 5.5\" floppy drive, toetsenbord en geen muis", "(Zspec@1 OR ibm@2 OR (ps@3 PHRASE 2 2@4) OR (intel@5 OR 8086@6) OR (25@7 OR Zmhz@8) OR (2@9 OR Zmb@10 OR Zintern@11) OR (50@12 OR Zmb@13 OR Zhd@14) OR 5.5@15 OR (floppy@16 PHRASE 6 drive@17 PHRASE 6 toetsenbord@18 PHRASE 6 en@19 PHRASE 6 geen@20 PHRASE 6 muis@21))" },
416  { "History: GetEventTool <- GetMusicManager <- GetMusicScript <- DMCallRoutine <- AMusicScriptEvent::execCallRoutine <- UObject::execClassContext <- (U2GameInfo M08A1.U2GameInfo0 @ Function U2.U2GameInfo.NotifyLevelChangeEnd : 0075 line 744) <- UObject::ProcessEvent <- (U2GameInfo M08A1.U2GameInfo0, Function U2.U2GameInfo.NotifyLevelChangeEnd) <- UGameEngine::LoadMap <- LocalMapURL <- UGameEngine::Browse <- ServerTravel <- UGameEngine::Tick <- UpdateWorld <- MainLoop", "(history@1 OR geteventtool@2 OR getmusicmanager@3 OR getmusicscript@4 OR dmcallroutine@5 OR (amusicscriptevent@6 PHRASE 2 execcallroutine@7) OR (uobject@8 PHRASE 2 execclasscontext@9) OR (u2gameinfo@10 OR (m08a1@11 PHRASE 2 u2gameinfo0@12) OR function@13 OR (u2@14 PHRASE 3 u2gameinfo@15 PHRASE 3 notifylevelchangeend@16) OR (0075@17 OR Zline@18 OR 744@19)) OR (uobject@20 PHRASE 2 processevent@21) OR (u2gameinfo@22 OR (m08a1@23 PHRASE 2 u2gameinfo0@24) OR function@25 OR (u2@26 PHRASE 3 u2gameinfo@27 PHRASE 3 notifylevelchangeend@28)) OR (ugameengine@29 PHRASE 2 loadmap@30) OR localmapurl@31 OR (ugameengine@32 PHRASE 2 browse@33) OR servertravel@34 OR (ugameengine@35 PHRASE 2 tick@36) OR updateworld@37 OR mainloop@38)" },
417  { "Support AMD XP 2400+ & 2600+ (K7T Turbo2 only)", "(support@1 OR amd@2 OR xp@3 OR 2400+@4 OR 2600+@5 OR (k7t@6 OR turbo2@7 OR Zonli@8))" },
418  { "'\"><br>bla</br>", "(br@1 PHRASE 3 bla@2 PHRASE 3 br@3)" },
419  { "The instruction at \"0x30053409\" referenced memory at \"0x06460504\". The memory could not be \"read'. Click OK to terminate the application.", "(the@1 OR Zinstruct@2 OR Zat@3 OR 0x30053409@4 OR (Zreferenc@5 OR Zmemori@6 OR Zat@7) OR 0x06460504@8 OR (the@9 OR Zmemori@10 OR Zcould@11 OR Znot@12 OR Zbe@13) OR (read@14 PHRASE 7 click@15 PHRASE 7 ok@16 PHRASE 7 to@17 PHRASE 7 terminate@18 PHRASE 7 the@19 PHRASE 7 application@20))" },
420  { "\"(P5A-b)\"", "(p5a@1 PHRASE 2 b@2)" },
421  { "(13,5 > 13) == no-go!", "(13,5@1 OR 13@2 OR (no@3 PHRASE 2 go@4))" },
422  { "eth not found \"ifconfig -a\"", "(Zeth@1 OR Znot@2 OR Zfound@3 OR (ifconfig@4 PHRASE 2 a@5))" },
423  { "<META NAME=\"ROBOTS", "(meta@1 OR name@2 OR robots@3)" },
424  { "lp0: using parport0 (interrupt-driven)", "(Zlp0@1 OR (Zuse@2 OR Zparport0@3) OR (interrupt@4 PHRASE 2 driven@5))" },
425  { "ULTRA PC-TUNING, COOLING & MODDING (4,6)", "(ultra@1 OR (pc@2 PHRASE 2 tuning@3) OR cooling@4 OR modding@5 OR 4,6@6)" },
426  { "512MB PC2700 DDR SDRAM Rood (Dane-Elec)", "(512mb@1 OR pc2700@2 OR ddr@3 OR sdram@4 OR rood@5 OR (dane@6 PHRASE 2 elec@7))" },
427  { "header(\"Content Type: text/html\");", "(header@1 OR (content@2 OR type@3) OR (text@4 PHRASE 2 html@5))" },
428  { "\"-RW\" \"+RW\"", "(rw@1 OR rw@2)" },
429  { "\"cresta digital answering machine", "(cresta@1 PHRASE 4 digital@2 PHRASE 4 answering@3 PHRASE 4 machine@4)" },
430  { "Arctic Super Silent PRO TC (Athlon/P3 - 2,3 GHz)", "(arctic@1 OR super@2 OR silent@3 OR pro@4 OR tc@5 OR ((athlon@6 PHRASE 2 p3@7) OR (2,3@8 OR ghz@9)))" },
431  { "c++ fopen \"r+t\"", "(Zc++@1 OR Zfopen@2 OR (r@3 PHRASE 2 t@4))" },
432  { "c++ fopen (r+t)", "(Zc++@1 OR Zfopen@2 OR (Zr@3 OR Zt@4))" },
433  { "\"DVD+R\"", "(dvd@1 PHRASE 2 r@2)" },
434  { "Class.forName(\"jdbc.odbc.JdbcOdbcDriver\");", "((class@1 PHRASE 2 forname@2) OR (jdbc@3 PHRASE 3 odbc@4 PHRASE 3 jdbcodbcdriver@5))" },
435  { "perl(find.pl)", "(perl@1 OR (find@2 PHRASE 2 pl@3))" },
436  { "\"-5v\" voeding", "(5v@1 OR Zvoed@2)" },
437  { "\"-5v\" power supply", "(5v@1 OR (Zpower@2 OR Zsuppli@3))" },
438  { "An Error occurred whie attempting to initialize the Borland Database Engine (error $2108)", "(an@1 OR error@2 OR Zoccur@3 OR Zwhie@4 OR Zattempt@5 OR Zto@6 OR Ziniti@7 OR Zthe@8 OR borland@9 OR database@10 OR engine@11 OR (Zerror@12 OR 2108@13))" },
439  { "(error $2108) Borland", "(Zerror@1 OR 2108@2 OR borland@3)" },
440  { "On Friday 04 April 2003 09:32, Edwin van Eersel wrote: > ik voel me eigenlijk wel behoorlijk kut :)", "(on@1 OR friday@2 OR 04@3 OR april@4 OR 2003@5 OR (09@6 PHRASE 2 32@7) OR (edwin@8 OR Zvan@9 OR eersel@10 OR Zwrote@11) OR (Zik@12 OR Zvoel@13 OR Zme@14 OR Zeigenlijk@15 OR Zwel@16 OR Zbehoorlijk@17 OR Zkut@18))" },
441  { "Elektrotechniek + \"hoe bevalt het?\"\"", "(elektrotechniek@1 OR (hoe@2 PHRASE 3 bevalt@3 PHRASE 3 het@4))" },
442  { "Shortcuts in menu (java", "(shortcuts@1 OR Zin@2 OR Zmenu@3 OR Zjava@4)" },
443  { "detonator+settings\"", "(Zdeton@1 OR settings@2)" },
444  { "(ez-bios) convert", "((ez@1 PHRASE 2 bios@2) OR Zconvert@3)" },
445  { "Sparkle 7100M4 64MB (GeForce4 MX440)", "(sparkle@1 OR 7100m4@2 OR 64mb@3 OR (geforce4@4 OR mx440@5))" },
446  { "freebsd \"boek OR newbie\"", "(Zfreebsd@1 OR (boek@2 PHRASE 3 or@3 PHRASE 3 newbie@4))" },
447  { "for (;;) c++", "(Zfor@1 OR Zc++@2)" },
448  { "1700+-2100+", "(1700+@1 PHRASE 2 2100+@2)" },
449  { "PHP Warning: Invalid library (maybe not a PHP library) 'libmysqlclient.so'", "(php@1 OR warning@2 OR invalid@3 OR Zlibrari@4 OR (Zmayb@5 OR Znot@6 OR Za@7 OR php@8 OR Zlibrari@9) OR (libmysqlclient@10 PHRASE 2 so@11))" },
450  { "NEC DV-5800B (Bul", "(nec@1 OR (dv@2 PHRASE 2 5800b@3) OR bul@4)" },
451  { "org.jdom.input.SAXBuilder.<init>(SAXBuilder.java)", "((org@1 PHRASE 4 jdom@2 PHRASE 4 input@3 PHRASE 4 saxbuilder@4) OR init@5 OR (saxbuilder@6 PHRASE 2 java@7))" },
452  { "AMD Athlon XP 2500+ (1,83GHz, 512KB)", "(amd@1 OR athlon@2 OR xp@3 OR 2500+@4 OR (1,83ghz@5 OR 512kb@6))" },
453  { "'q ben\"", "(Zq@1 OR ben@2)" },
454  { "getsmbfilepwent: malformed password entry (uid not number)", "(Zgetsmbfilepw@1 OR (Zmalform@2 OR Zpassword@3 OR Zentri@4) OR (Zuid@5 OR Znot@6 OR Znumber@7))" },
455  { "\xc3\xb6ude onderdelen\"", "(Z\xc3\xb6ude@1 OR onderdelen@2)" },
456  { "Heeft iemand enig idee waarom de pioneer (zelf met originele firmware van pioneer) bij mij niet wil flashen ??", "(heeft@1 OR Ziemand@2 OR Zenig@3 OR Zide@4 OR Zwaarom@5 OR Zde@6 OR Zpioneer@7 OR (Zzelf@8 OR Zmet@9 OR Zoriginel@10 OR Zfirmwar@11 OR Zvan@12 OR Zpioneer@13) OR (Zbij@14 OR Zmij@15 OR Zniet@16 OR Zwil@17 OR Zflashen@18))" },
457  { "asus a7v266 bios nieuw -(a7v266-e)", "((Zasus@1 OR Za7v266@2 OR Zbio@3 OR Znieuw@4) AND_NOT (a7v266@5 PHRASE 2 e@6))" },
458  { "cybercom \"dvd+r\"", "(Zcybercom@1 OR (dvd@2 PHRASE 2 r@3))" },
459  { "AMD PCNET Family Ethernet Adapter (PCI-ISA)", "(amd@1 OR pcnet@2 OR family@3 OR ethernet@4 OR adapter@5 OR (pci@6 PHRASE 2 isa@7))" },
460  { "relais +/-", "Zrelai@1" },
461  { "formules (slepen OR doortrekken) excel", "(Zformul@1 OR (Zslepen@2 OR Zdoortrekken@3) OR Zexcel@4)" },
462  { "\"%English", "english@1" },
463  { "select max( mysql", "(Zselect@1 OR max@2 OR Zmysql@3)" },
464  { "leejow(saait", "(leejow@1 OR Zsaait@2)" },
465  { "'Windows 2000 Advanced Server\" netwerkverbinding valt steeds weg", "(windows@1 OR 2000@2 OR advanced@3 OR server@4 OR (netwerkverbinding@5 PHRASE 4 valt@6 PHRASE 4 steeds@7 PHRASE 4 weg@8))" },
466  { "K7T Turbo 2 (MS-6330)", "(k7t@1 OR turbo@2 OR 2@3 OR (ms@4 PHRASE 2 6330@5))" },
467  { "failed to receive data from the client agent. (ec=1)", "(Zfail@1 OR Zto@2 OR Zreceiv@3 OR Zdata@4 OR Zfrom@5 OR Zthe@6 OR Zclient@7 OR Zagent@8 OR (ec@9 OR 1@10))" },
468  { "\"cannot find -lz\"", "(cannot@1 PHRASE 3 find@2 PHRASE 3 lz@3)" },
469  { "undefined reference to `mysql_drop_db'\"", "(Zundefin@1 OR Zrefer@2 OR Zto@3 OR Zmysql_drop_db@4)" },
470  { "search form asp \"%'", "(Zsearch@1 OR Zform@2 OR Zasp@3)" },
471  { "(dvd+r) kwaliteit", "(Zdvd@1 OR Zr@2 OR Zkwaliteit@3)" },
472  { "Fatal error: Allowed memory size of 8388608 bytes exhausted (tried to allocate 35 bytes)", "(fatal@1 OR Zerror@2 OR allowed@3 OR Zmemori@4 OR Zsize@5 OR Zof@6 OR 8388608@7 OR Zbyte@8 OR Zexhaust@9 OR (Ztri@10 OR Zto@11 OR Zalloc@12 OR 35@13 OR Zbyte@14))" },
473  { "geluid (schokt OR hapert)", "(Zgeluid@1 OR (Zschokt@2 OR Zhapert@3))" },
474  { "Het wordt pas echt leuk als het hard staat!! >:)", "(het@1 OR Zwordt@2 OR Zpas@3 OR Zecht@4 OR Zleuk@5 OR Zal@6 OR Zhet@7 OR Zhard@8 OR Zstaat@9)" },
475  { "Uw configuratie bestand bevat instellingen (root zonder wachtwoord) die betrekking hebben tot de standaard MySQL account. Uw MySQL server draait met deze standaard waardes, en is open voor ongewilde toegang, het wordt dus aangeraden dit op te lossen", "(uw@1 OR Zconfigurati@2 OR Zbestand@3 OR Zbevat@4 OR Zinstellingen@5 OR (Zroot@6 OR Zzonder@7 OR Zwachtwoord@8) OR (Zdie@9 OR Zbetrekk@10 OR Zhebben@11 OR Ztot@12 OR Zde@13 OR Zstandaard@14 OR mysql@15 OR Zaccount@16 OR uw@17 OR mysql@18 OR Zserver@19 OR Zdraait@20 OR Zmet@21 OR Zdeze@22 OR Zstandaard@23 OR Zwaard@24) OR (Zen@25 OR Zis@26 OR Zopen@27 OR Zvoor@28 OR Zongewild@29 OR Ztoegang@30) OR (Zhet@31 OR Zwordt@32 OR Zdus@33 OR Zaangeraden@34 OR Zdit@35 OR Zop@36 OR Zte@37 OR Zlossen@38))" },
476  { "(library qt-mt) not found", "(Zlibrari@1 OR (qt@2 PHRASE 2 mt@3) OR (Znot@4 OR Zfound@5))" },
477  { "Qt (>= Qt 3.0.3) (library qt-mt) not found", "(qt@1 OR (qt@2 OR 3.0.3@3) OR (Zlibrari@4 OR (qt@5 PHRASE 2 mt@6)) OR (Znot@7 OR Zfound@8))" },
478  { "setup was unable to find (or could not read) the language specific setup resource dll, unable to continue. Please reboot and try again.", "(Zsetup@1 OR Zwas@2 OR Zunabl@3 OR Zto@4 OR Zfind@5 OR (Zor@6 OR Zcould@7 OR Znot@8 OR Zread@9) OR (Zthe@10 OR Zlanguag@11 OR Zspecif@12 OR Zsetup@13 OR Zresourc@14 OR Zdll@15) OR (Zunabl@16 OR Zto@17 OR Zcontinu@18 OR please@19 OR Zreboot@20 OR Zand@21 OR Ztri@22 OR Zagain@23))" },
479  { "Titan TTC-D5TB(4/CU35)", "(titan@1 OR (ttc@2 PHRASE 2 d5tb@3) OR (4@4 PHRASE 2 cu35@5))" },
480  { "[php] date( min", "(Zphp@1 OR date@2 OR Zmin@3)" },
481  { "EPOX EP-8RDA+ (nForce2 SPP+MCP-T) Rev. 1.1", "(epox@1 OR (ep@2 PHRASE 2 8rda+@3) OR (Znforce2@4 OR spp@5 OR (mcp@6 PHRASE 2 t@7)) OR rev@8 OR 1.1@9)" },
482  { "554 5.4.6 Too many hops 53 (25 max)", "(554@1 OR 5.4.6@2 OR too@3 OR Zmani@4 OR Zhop@5 OR 53@6 OR (25@7 OR Zmax@8))" },
483  { "ik had toch nog een vraagje: er zijn nu eigenlijk alleen maar schijfjes van 4.7GB alleen straks zullen er vast schijfjes van meer dan 4.7GB komen. Zal deze brander dit wel kunnen schijven?""?(na bijvoorbeeld een firmware update?) ben erg benieuwd", "(Zik@1 OR Zhad@2 OR Ztoch@3 OR Znog@4 OR Zeen@5 OR Zvraagj@6 OR Zer@7 OR Zzijn@8 OR Znu@9 OR Zeigenlijk@10 OR Zalleen@11 OR Zmaar@12 OR Zschijfj@13 OR Zvan@14 OR 4.7gb@15 OR Zalleen@16 OR Zstrak@17 OR Zzullen@18 OR Zer@19 OR Zvast@20 OR Zschijfj@21 OR Zvan@22 OR Zmeer@23 OR Zdan@24 OR 4.7gb@25 OR Zkomen@26 OR zal@27 OR Zdeze@28 OR Zbrander@29 OR Zdit@30 OR Zwel@31 OR Zkunnen@32 OR Zschijven@33 OR (Zna@34 OR Zbijvoorbeeld@35 OR Zeen@36 OR Zfirmwar@37 OR Zupdat@38) OR (Zben@39 OR Zerg@40 OR Zbenieuwd@41))" },
484  { "ati linux drivers (4.3.0)", "(Zati@1 OR Zlinux@2 OR Zdriver@3 OR 4.3.0@4)" },
485  { "ENCAPSED_AND_WHITESPACE", "encapsed_and_whitespace@1" },
486  { "lpadmin: add-printer (set device) failed: client-error-not-possible", "(Zlpadmin@1 OR (add@2 PHRASE 2 printer@3) OR (Zset@4 OR Zdevic@5) OR Zfail@6 OR (client@7 PHRASE 4 error@8 PHRASE 4 not@9 PHRASE 4 possible@10))" },
487  { "welke dvd \"+r\" media", "(Zwelk@1 OR Zdvd@2 OR r@3 OR Zmedia@4)" },
488  { "Warning: stat failed for fotos(errno=2 - No such file or directory)", "(warning@1 OR (Zstat@2 OR Zfail@3 OR Zfor@4 OR fotos@5) OR errno@6 OR 2@7 OR (no@8 OR Zsuch@9 OR Zfile@10 OR Zor@11 OR Zdirectori@12))" },
489  { "dvd +/-", "Zdvd@1" },
490  { "7vaxp +voltage mod\"", "(Zvoltag@2 AND_MAYBE (7vaxp@1 OR mod@3))" },
491  { "lpt port (SPP/EPP) is enabled", "(Zlpt@1 OR Zport@2 OR (spp@3 PHRASE 2 epp@4) OR (Zis@5 OR Zenabl@6))" },
492  { "getenv(\"HTTP_REFERER\")", "(getenv@1 OR http_referer@2)" },
493  { "Error setting display mode: CreateDevice failed (D3DERR_DRIVERINTERNALERROR)", "(error@1 OR Zset@2 OR Zdisplay@3 OR Zmode@4 OR createdevice@5 OR Zfail@6 OR d3derr_driverinternalerror@7)" },
494  { "Exception number: c0000005 (access violation)", "(exception@1 OR Znumber@2 OR Zc0000005@3 OR (Zaccess@4 OR Zviolat@5))" },
495  { "header(\"Content-type:application/octetstream\");", "(header@1 OR (content@2 PHRASE 4 type@3 PHRASE 4 application@4 PHRASE 4 octetstream@5))" },
496  { "java.security.AccessControlException: access denied (java.lang.RuntimePermission accessClassInPackage.sun.jdbc.odbc)", "((java@1 PHRASE 3 security@2 PHRASE 3 accesscontrolexception@3) OR (Zaccess@4 OR Zdeni@5) OR ((java@6 PHRASE 3 lang@7 PHRASE 3 runtimepermission@8) OR (accessclassinpackage@9 PHRASE 4 sun@10 PHRASE 4 jdbc@11 PHRASE 4 odbc@12)))" },
497  { "(001.part.met", "(001@1 PHRASE 3 part@2 PHRASE 3 met@3)" },
498  { "Warning: mail(): Use the -f option (5th param) to include valid reply-to address ! in /usr/home/vdb/www/mail.php on line 79", "(warning@1 OR mail@2 OR (use@3 OR Zthe@4) OR (Zf@5 OR Zoption@6) OR (5th@7 OR Zparam@8) OR (Zto@9 OR Zinclud@10 OR Zvalid@11) OR (reply@12 PHRASE 2 to@13) OR Zaddress@14 OR Zin@15 OR (usr@16 PHRASE 6 home@17 PHRASE 6 vdb@18 PHRASE 6 www@19 PHRASE 6 mail@20 PHRASE 6 php@21) OR (Zon@22 OR Zline@23 OR 79@24))" },
499  { "PHP Use the -f option (5th param)", "((php@1 OR use@2 OR Zthe@3 OR Zoption@5 OR (5th@6 OR Zparam@7)) AND_NOT Zf@4)" },
500  { "dvd \"+\" \"-\"", "Zdvd@1" },
501  { "bericht ( %)", "Zbericht@1" },
502  { "2500+ of 2600+ (niett OC)", "(2500+@1 OR Zof@2 OR 2600+@3 OR (Zniett@4 OR oc@5))" },
503  { "maxtor windows xp werkt The drivers for this device are not installed. (Code 28)", "(Zmaxtor@1 OR Zwindow@2 OR Zxp@3 OR Zwerkt@4 OR the@5 OR Zdriver@6 OR Zfor@7 OR Zthis@8 OR Zdevic@9 OR Zare@10 OR Znot@11 OR Zinstal@12 OR (code@13 OR 28@14))" },
504  { "Warning: stat failed for /mnt/web/react/got/react/board/non-www/headlines/tnet-headlines.txt (errno=2 - No such file or directory) in /mnt/web/react/got/react/global/non-www/templates/got/functions.inc.php on line 303", "(warning@1 OR (Zstat@2 OR Zfail@3 OR Zfor@4) OR (mnt@5 PHRASE 12 web@6 PHRASE 12 react@7 PHRASE 12 got@8 PHRASE 12 react@9 PHRASE 12 board@10 PHRASE 12 non@11 PHRASE 12 www@12 PHRASE 12 headlines@13 PHRASE 12 tnet@14 PHRASE 12 headlines@15 PHRASE 12 txt@16) OR (errno@17 OR 2@18 OR (no@19 OR Zsuch@20 OR Zfile@21 OR Zor@22 OR Zdirectori@23)) OR Zin@24 OR (mnt@25 PHRASE 13 web@26 PHRASE 13 react@27 PHRASE 13 got@28 PHRASE 13 react@29 PHRASE 13 global@30 PHRASE 13 non@31 PHRASE 13 www@32 PHRASE 13 templates@33 PHRASE 13 got@34 PHRASE 13 functions@35 PHRASE 13 inc@36 PHRASE 13 php@37) OR (Zon@38 OR Zline@39 OR 303@40))" },
505  { "apm: BIOS version 1.2 Flags 0x03 (Driver version 1.16)", "(Zapm@1 OR (bios@2 OR Zversion@3 OR 1.2@4 OR flags@5 OR 0x03@6) OR (driver@7 OR Zversion@8 OR 1.16@9))" },
506  { "GA-8IHXP(3.0)", "((ga@1 PHRASE 2 8ihxp@2) OR 3.0@3)" },
507  { "8IHXP(3.0)", "(8ihxp@1 OR 3.0@2)" },
508  { "na\xc2\xb7si (de ~ (m.))", "(Zna\xc2\xb7si@1 OR (Zde@2 OR Zm@3))" },
509  { "header(\"Content-Disposition: attachment;", "(header@1 OR (content@2 PHRASE 3 disposition@3 PHRASE 3 attachment@4))" },
510  { "\"header(\"Content-Disposition: attachment;\"", "(header@1 OR (content@2 PHRASE 2 disposition@3) OR Zattach@4)" },
511  { "\"Beep -f\"", "(beep@1 PHRASE 2 f@2)" },
512  { "kraan NEAR (Elektrisch OR Electrisch)", "(Zkraan@1 OR near@2 OR (elektrisch@3 OR or@4 OR electrisch@5))" },
513  { "checking for Qt... configure: error: Qt (>= Qt 3.0.2) (headers and libraries) not found. Please check your installation!", "(Zcheck@1 OR Zfor@2 OR qt@3 OR Zconfigur@4 OR Zerror@5 OR qt@6 OR (qt@7 OR 3.0.2@8) OR (Zheader@9 OR Zand@10 OR Zlibrari@11) OR (Znot@12 OR Zfound@13 OR please@14 OR Zcheck@15 OR Zyour@16 OR Zinstal@17))" },
514  { "parse error, unexpected '\\\"', expecting T_STRING or T_VARIABLE or T_NUM_STRING", "(Zpars@1 OR Zerror@2 OR Zunexpect@3 OR (expecting@4 PHRASE 6 t_string@5 PHRASE 6 or@6 PHRASE 6 t_variable@7 PHRASE 6 or@8 PHRASE 6 t_num_string@9))" },
515  { "ac3 (0x2000) \"Dolby Laboratories,", "(Zac3@1 OR 0x2000@2 OR (dolby@3 PHRASE 2 laboratories@4))" },
516  { "Movie.FileName=(\"../../../~animations/\"+lesson1.recordset.fields('column3')+\"Intro.avi\")", "((movie@1 PHRASE 2 filename@2) OR animations@3 OR (lesson1@4 PHRASE 3 recordset@5 PHRASE 3 fields@6) OR Zcolumn3@7 OR (intro@8 PHRASE 2 avi@9))" },
517  { "502 Permission Denied - Permission Denied - news.chello.nl -- http://www.chello.nl/ (Typhoon v1.2.3)", "(502@1 OR permission@2 OR denied@3 OR (permission@4 OR denied@5) OR (news@6 PHRASE 3 chello@7 PHRASE 3 nl@8) OR (http@9 PHRASE 4 www@10 PHRASE 4 chello@11 PHRASE 4 nl@12) OR (typhoon@13 OR Zv1.2.3@14))" },
518  { "Motion JPEG (MJPEG codec)", "(motion@1 OR jpeg@2 OR (mjpeg@3 OR Zcodec@4))" },
519  { ": zoomtext\"", "zoomtext@1" },
520  { "Your SORT command does not seem to support the \"-r -n -k 7\"", "(your@1 OR sort@2 OR Zcommand@3 OR Zdoe@4 OR Znot@5 OR Zseem@6 OR Zto@7 OR Zsupport@8 OR Zthe@9 OR (r@10 PHRASE 4 n@11 PHRASE 4 k@12 PHRASE 4 7@13))" },
521  { "Geef de naam van de MSDOS prompt op C:\\\\WINDOWS.COM\\\"", "(geef@1 OR Zde@2 OR Znaam@3 OR Zvan@4 OR Zde@5 OR msdos@6 OR Zprompt@7 OR Zop@8 OR (c@9 PHRASE 3 windows@10 PHRASE 3 com@11))" },
522  { "\"\"wa is fase\"", "(Zwa@1 OR Zis@2 OR fase@3)" },
523  { "<v:imagedata src=\"", "((v@1 PHRASE 2 imagedata@2) OR src@3)" },
524  { "system(play ringin.wav); ?>", "(system@1 OR Zplay@2 OR (ringin@3 PHRASE 2 wav@4))" },
525  { "\"perfect NEAR systems\"", "(perfect@1 PHRASE 3 near@2 PHRASE 3 systems@3)" },
526  { "LoadLibrary(\"mainta/gamex86.dll\") failed", "(loadlibrary@1 OR (mainta@2 PHRASE 3 gamex86@3 PHRASE 3 dll@4) OR Zfail@5)" },
527  { "DATE_FORMAT('1997-10-04 22:23:00', '%W %M %Y');", "(date_format@1 OR (1997@2 PHRASE 3 10@3 PHRASE 3 04@4) OR (22@5 PHRASE 3 23@6 PHRASE 3 00@7) OR w@8 OR m@9 OR y@10)" },
528  { "secundaire IDE-controller (dubbele fifo)", "(Zsecundair@1 OR (ide@2 PHRASE 2 controller@3) OR (Zdubbel@4 OR Zfifo@5))" },
529  { "\"Postal2+Explorer.exe\"", "(postal2@1 PHRASE 3 explorer@2 PHRASE 3 exe@3)" },
530  { "COUNT(*)", "count@1" },
531  { "Nuttige Windows progs (1/11)", "(nuttige@1 OR windows@2 OR Zprog@3 OR (1@4 PHRASE 2 11@5))" },
532  { "if(usercode==passcode==)", "(if@1 OR usercode@2 OR passcode@3)" },
533  { "lg 8160b (dvd+r)", "(Zlg@1 OR 8160b@2 OR (Zdvd@3 OR Zr@4))" },
534  { "iPAQ Pocket PC 2002 End User Update (EUU - Service Pack)", "(Zipaq@1 OR pocket@2 OR pc@3 OR 2002@4 OR end@5 OR user@6 OR update@7 OR (euu@8 OR (service@9 OR pack@10)))" },
535  { "'ipod pakt tags niet\"", "(Zipod@1 OR Zpakt@2 OR Ztag@3 OR niet@4)" },
536  { "\"DVD+/-R\"", "(dvd+@1 PHRASE 2 r@2)" },
537  { "\"DVD+R DVD-R\"", "(dvd@1 PHRASE 4 r@2 PHRASE 4 dvd@3 PHRASE 4 r@4)" },
538  { "php ;) in een array zetten", "(Zphp@1 OR (Zin@2 OR Zeen@3 OR Zarray@4 OR Zzetten@5))" },
539  { "De inhoud van uw advertentie is niet geschikt voor plaatsing op marktplaats! (001", "(de@1 OR Zinhoud@2 OR Zvan@3 OR Zuw@4 OR Zadvertenti@5 OR Zis@6 OR Zniet@7 OR Zgeschikt@8 OR Zvoor@9 OR Zplaats@10 OR Zop@11 OR Zmarktplaat@12 OR 001@13)" },
540  { "creative (soundblaster OR sb) 128", "(Zcreativ@1 OR (Zsoundblast@2 OR Zsb@3) OR 128@4)" },
541  { "Can't open file: (errno: 145)", "(can't@1 OR Zopen@2 OR Zfile@3 OR (Zerrno@4 OR 145@5))" },
542  { "Formateren lukt niet(98,XP)", "(formateren@1 OR Zlukt@2 OR niet@3 OR 98@4 OR xp@5)" },
543  { "access denied (java.io.", "(Zaccess@1 OR Zdeni@2 OR (java@3 PHRASE 2 io@4))" },
544  { "(access denied (java.io.)", "(Zaccess@1 OR Zdeni@2 OR (java@3 PHRASE 2 io@4))" },
545  { "wil niet installeren ( crc fouten)", "(Zwil@1 OR Zniet@2 OR Zinstalleren@3 OR (Zcrc@4 OR Zfouten@5))" },
546  { "(DVD+RW) brandsoftware meerdere", "(dvd@1 OR rw@2 OR (Zbrandsoftwar@3 OR Zmeerder@4))" },
547  { "(database OF databases) EN geheugen", "(Zdatabas@1 OR of@2 OR Zdatabas@3 OR (en@4 OR Zgeheugen@5))" },
548  { "(server 2003) winroute", "(Zserver@1 OR 2003@2 OR Zwinrout@3)" },
549  { "54MHz (kanaal 2 VHF) tot tenminste 806 MHz (kanaal 69 UHF)", "(54mhz@1 OR (Zkanaal@2 OR 2@3 OR vhf@4) OR (Ztot@5 OR Ztenminst@6 OR 806@7 OR mhz@8) OR (Zkanaal@9 OR 69@10 OR uhf@11))" },
550  { "(draadloos OR wireless) netwerk", "(Zdraadloo@1 OR Zwireless@2 OR Znetwerk@3)" },
551  { "localtime(time(NULL));", "(localtime@1 OR time@2 OR null@3)" },
552  { "ob_start(\"ob_gzhandler\");", "(ob_start@1 OR ob_gzhandler@2)" },
553  { "PPP Closed : LCP Time-out (VPN-0)", "(ppp@1 OR closed@2 OR lcp@3 OR (time@4 PHRASE 2 out@5) OR (vpn@6 PHRASE 2 0@7))" },
554  { "COM+-gebeurtenissysteem", "(com+@1 PHRASE 2 gebeurtenissysteem@2)" },
555  { "rcpthosts (#5.7.1)", "(Zrcpthost@1 OR 5.7.1@2)" },
556  { "Dit apparaat werkt niet goed omdat Windows de voor dit apparaat vereiste stuurprogramma's niet kan laden. (Code 31)", "(dit@1 OR Zapparaat@2 OR Zwerkt@3 OR Zniet@4 OR Zgo@5 OR Zomdat@6 OR windows@7 OR Zde@8 OR Zvoor@9 OR Zdit@10 OR Zapparaat@11 OR Zvereist@12 OR Zstuurprogramma@13 OR Zniet@14 OR Zkan@15 OR Zladen@16 OR (code@17 OR 31@18))" },
557  { "window.open( scrollbar", "((window@1 PHRASE 2 open@2) OR Zscrollbar@3)" },
558  { "T68i truc ->", "(t68i@1 OR Ztruc@2)" },
559  { "T68i ->", "t68i@1" },
560  { "\"de lijn is bezet\"\"", "(de@1 PHRASE 4 lijn@2 PHRASE 4 is@3 PHRASE 4 bezet@4)" },
561  { "if (eregi(\"", "(Zif@1 OR eregi@2)" },
562  { "This device is not working properly because Windows cannot load the drivers required for this device. (Code 31)", "(this@1 OR Zdevic@2 OR Zis@3 OR Znot@4 OR Zwork@5 OR Zproper@6 OR Zbecaus@7 OR windows@8 OR Zcannot@9 OR Zload@10 OR Zthe@11 OR Zdriver@12 OR Zrequir@13 OR Zfor@14 OR Zthis@15 OR Zdevic@16 OR (code@17 OR 31@18))" },
563  { "execCommand(\"Paste\");", "(execcommand@1 OR paste@2)" },
564  { "\"-1 unread\"", "(1@1 PHRASE 2 unread@2)" },
565  { "\"www.historical-fire-engines", "(www@1 PHRASE 4 historical@2 PHRASE 4 fire@3 PHRASE 4 engines@4)" },
566  { "\"DVD+RW\" erase", "((dvd@1 PHRASE 2 rw@2) OR Zeras@3)" },
567  { "[showjekamer)", "Zshowjekam@1" },
568  { "The description for Event ID 1 in Source True Vector Engine ) cannot be found. The local computer may not have the necessary registry information or message DLL files to display messages from a remote computer. You may be able to use the /AUXSOURC", "(the@1 OR Zdescript@2 OR Zfor@3 OR event@4 OR id@5 OR 1@6 OR Zin@7 OR source@8 OR true@9 OR vector@10 OR engine@11 OR (Zcannot@12 OR Zbe@13 OR Zfound@14 OR the@15 OR Zlocal@16 OR Zcomput@17 OR Zmay@18 OR Znot@19 OR Zhave@20 OR Zthe@21 OR Znecessari@22 OR Zregistri@23 OR Zinform@24 OR Zor@25 OR Zmessag@26 OR dll@27 OR Zfile@28 OR Zto@29 OR Zdisplay@30 OR Zmessag@31 OR Zfrom@32 OR Za@33 OR Zremot@34 OR Zcomput@35 OR you@36 OR Zmay@37 OR Zbe@38 OR Zabl@39 OR Zto@40 OR Zuse@41 OR Zthe@42) OR auxsourc@43)" },
569  { "org.apache.jasper.JasperException: This absolute uri (http://java.sun.com/jstl/core) cannot be resolved in either web.xml or the jar files deployed with this application", "((org@1 PHRASE 4 apache@2 PHRASE 4 jasper@3 PHRASE 4 jasperexception@4) OR (this@5 OR Zabsolut@6 OR Zuri@7) OR (http@8 PHRASE 6 java@9 PHRASE 6 sun@10 PHRASE 6 com@11 PHRASE 6 jstl@12 PHRASE 6 core@13) OR (Zcannot@14 OR Zbe@15 OR Zresolv@16 OR Zin@17 OR Zeither@18) OR (web@19 PHRASE 2 xml@20) OR (Zor@21 OR Zthe@22 OR Zjar@23 OR Zfile@24 OR Zdeploy@25 OR Zwith@26 OR Zthis@27 OR Zapplic@28))" },
570  { "This absolute uri (http://java.sun.com/jstl/core) cannot be resolved in either web.xml or the jar files deployed with this application", "(this@1 OR Zabsolut@2 OR Zuri@3 OR (http@4 PHRASE 6 java@5 PHRASE 6 sun@6 PHRASE 6 com@7 PHRASE 6 jstl@8 PHRASE 6 core@9) OR (Zcannot@10 OR Zbe@11 OR Zresolv@12 OR Zin@13 OR Zeither@14) OR (web@15 PHRASE 2 xml@16) OR (Zor@17 OR Zthe@18 OR Zjar@19 OR Zfile@20 OR Zdeploy@21 OR Zwith@22 OR Zthis@23 OR Zapplic@24))" },
571  { "vervangen # \"/", "Zvervangen@1" },
572  { "vervangen # /\"", "Zvervangen@1" },
573  { "while(list($key, $val) = each($HTTP_POST_VARS))", "(while@1 OR list@2 OR Zkey@3 OR Zval@4 OR each@5 OR http_post_vars@6)" },
574  { "PowerDVD does not support the current display mode. (DDraw Overlay mode is recommended)", "(powerdvd@1 OR Zdoe@2 OR Znot@3 OR Zsupport@4 OR Zthe@5 OR Zcurrent@6 OR Zdisplay@7 OR Zmode@8 OR (ddraw@9 OR overlay@10 OR Zmode@11 OR Zis@12 OR Zrecommend@13))" },
575  { "Warning: Unexpected character in input: '' (ASCII=92) state=1 highlight", "(warning@1 OR (unexpected@2 OR Zcharact@3 OR Zin@4 OR Zinput@5) OR (ascii@6 OR 92@7) OR state@8 OR (1@9 OR Zhighlight@10))" },
576  { "error: Qt-1.4 (headers and libraries) not found. Please check your installation!", "(Zerror@1 OR (qt@2 PHRASE 2 1.4@3) OR (Zheader@4 OR Zand@5 OR Zlibrari@6) OR (Znot@7 OR Zfound@8 OR please@9 OR Zcheck@10 OR Zyour@11 OR Zinstal@12))" },
577  { "Error while initializing the sound driver: device /dev/dsp can't be opened (No such device) The sound server will continue, using the null output device.", "(error@1 OR Zwhile@2 OR Ziniti@3 OR Zthe@4 OR Zsound@5 OR Zdriver@6 OR Zdevic@7 OR (dev@8 PHRASE 2 dsp@9) OR (Zcan't@10 OR Zbe@11 OR Zopen@12) OR (no@13 OR Zsuch@14 OR Zdevic@15) OR (the@16 OR Zsound@17 OR Zserver@18 OR Zwill@19 OR Zcontinu@20) OR (Zuse@21 OR Zthe@22 OR Znull@23 OR Zoutput@24 OR Zdevic@25))" },
578  { "mag mijn waarschuwing nu weg ? ;)", "(Zmag@1 OR Zmijn@2 OR Zwaarschuw@3 OR Znu@4 OR Zweg@5)" },
579  { "Abit NF7-S (nForce 2 Chipset) Rev 2.0", "(abit@1 OR (nf7@2 PHRASE 2 s@3) OR (Znforc@4 OR 2@5 OR chipset@6) OR (rev@7 OR 2.0@8))" },
580  { "Setup Could Not Verify the Integrity of the File\" Error Message Occurs When You Try to Install Windows XP Service Pack 1", "(setup@1 OR could@2 OR not@3 OR verify@4 OR Zthe@5 OR integrity@6 OR Zof@7 OR Zthe@8 OR file@9 OR (error@10 PHRASE 13 message@11 PHRASE 13 occurs@12 PHRASE 13 when@13 PHRASE 13 you@14 PHRASE 13 try@15 PHRASE 13 to@16 PHRASE 13 install@17 PHRASE 13 windows@18 PHRASE 13 xp@19 PHRASE 13 service@20 PHRASE 13 pack@21 PHRASE 13 1@22))" },
581  { "(browser 19) citrix", "(Zbrowser@1 OR 19@2 OR Zcitrix@3)" },
582  { "preg_replace (.*?)", "Zpreg_replac@1" },
583  { "formule excel #naam\"?\"", "(Zformul@1 OR Zexcel@2 OR naam@3)" },
584  { "->", "" },
585  { "De instructie op 0x77f436f7 verwijst naar geheugen op 0x007f4778. De lees-of schrijfbewerking (\"written\") op het geheugen is mislukt", "(de@1 OR Zinstructi@2 OR Zop@3 OR 0x77f436f7@4 OR Zverwijst@5 OR Znaar@6 OR Zgeheugen@7 OR Zop@8 OR 0x007f4778@9 OR de@10 OR (lees@11 PHRASE 2 of@12) OR Zschrijfbewerk@13 OR written@14 OR (Zop@15 OR Zhet@16 OR Zgeheugen@17 OR Zis@18 OR Zmislukt@19))" },
586  { "<iframe src=\"www.tweakers.net></iframe>", "(Zifram@1 OR src@2 OR (www@3 PHRASE 4 tweakers@4 PHRASE 4 net@5 PHRASE 4 iframe@6))" },
587  { "\"rpm -e httpd\"", "(rpm@1 PHRASE 3 e@2 PHRASE 3 httpd@3)" },
588  { "automatisch op All Flis (*.*)", "(Zautomatisch@1 OR Zop@2 OR all@3 OR flis@4)" },
589  { "(Windows; U; Windows NT 5.1; en-US; rv:1.3b) Gecko/20030210", "(windows@1 OR u@2 OR (windows@3 OR nt@4 OR 5.1@5) OR (en@6 PHRASE 2 us@7) OR (rv@8 PHRASE 2 1.3b@9) OR (gecko@10 PHRASE 2 20030210@11))" },
590  { "en-US; rv:1.3b) Gecko/20030210", "((en@1 PHRASE 2 us@2) OR (rv@3 PHRASE 2 1.3b@4) OR (gecko@5 PHRASE 2 20030210@6))" },
591  { "\"en-US; rv:1.3b) Gecko/20030210\"", "(en@1 PHRASE 6 us@2 PHRASE 6 rv@3 PHRASE 6 1.3b@4 PHRASE 6 gecko@5 PHRASE 6 20030210@6)" },
592  { "(./) chmod.sh", "(chmod@1 PHRASE 2 sh@2)" },
593  { "document.write(ssg(\" html", "((document@1 PHRASE 2 write@2) OR ssg@3 OR html@4)" },
594  { "superstack \"mac+adressen\"", "(Zsuperstack@1 OR (mac@2 PHRASE 2 adressen@3))" },
595  { "IIS getenv(REMOTE_HOST)_", "(iis@1 OR getenv@2 OR remote_host@3 OR _@4)" },
596  { "IIS en getenv(REMOTE_HOST)", "(iis@1 OR Zen@2 OR getenv@3 OR remote_host@4)" },
597  { "php getenv(\"HTTP_REFERER\")", "(Zphp@1 OR getenv@2 OR http_referer@3)" },
598  { "nec+-1300", "(nec+@1 PHRASE 2 1300@2)" },
599  { "smbpasswd script \"-s\"", "(Zsmbpasswd@1 OR Zscript@2 OR s@3)" },
600  { "leestekens \" \xc3\xb6 \xc3\xab", "(Zleesteken@1 OR (\xc3\xb6@2 PHRASE 2 \xc3\xab@3))" },
601  { "freesco and (all seeing eye)", "(Zfreesco@1 OR Zand@2 OR (Zall@3 OR Zsee@4 OR Zeye@5))" },
602  { "('all seeing eye') and freesco", "(Zall@1 OR Zsee@2 OR Zeye@3 OR (Zand@4 OR Zfreesco@5))" },
603  { "\"[......\"", "" },
604  { "Error = 11004 (500 No Data (Winsock error #11004))", "(error@1 OR 11004@2 OR (500@3 OR no@4 OR data@5 OR (winsock@6 OR Zerror@7 OR 11004@8)))" },
605  { "gegevensfout (cyclishe redundantiecontrole)", "(Zgegevensfout@1 OR (Zcyclish@2 OR Zredundantiecontrol@3))" },
606  { "firmware versie waar NEC\"", "(Zfirmwar@1 OR Zversi@2 OR Zwaar@3 OR nec@4)" },
607  { "nu.nl \"-1\"", "((nu@1 PHRASE 2 nl@2) OR 1@3)" },
608  { "provider+-webspace", "(provider+@1 PHRASE 2 webspace@2)" },
609  { "verschil \"dvd+rw\" \"dvd-rw\"", "(Zverschil@1 OR (dvd@2 PHRASE 2 rw@3) OR (dvd@4 PHRASE 2 rw@5))" },
610  { "(dhcp client) + hangt", "(Zdhcp@1 OR Zclient@2 OR Zhangt@3)" },
611  { "MSI 875P Neo-FIS2R (Intel 875P)", "(msi@1 OR 875p@2 OR (neo@3 PHRASE 2 fis2r@4) OR (intel@5 OR 875p@6))" },
612  { "voeding passief gekoeld\"", "(Zvoed@1 OR Zpassief@2 OR gekoeld@3)" },
613  { "if (mysql_num_rows($resultaat)==1)", "(Zif@1 OR mysql_num_rows@2 OR Zresultaat@3 OR 1@4)" },
614  { "Server.CreateObject(\"Persits.Upload.1\")", "((server@1 PHRASE 2 createobject@2) OR (persits@3 PHRASE 3 upload@4 PHRASE 3 1@5))" },
615  { "if(cod>9999999)cod=parseInt(cod/64)", "(if@1 OR cod@2 OR 9999999@3 OR cod@4 OR parseint@5 OR (cod@6 PHRASE 2 64@7))" },
616  { "if (cod>9999999", "(Zif@1 OR (cod@2 OR 9999999@3))" },
617  { "\"rm -rf /bin/laden\"", "(rm@1 PHRASE 4 rf@2 PHRASE 4 bin@3 PHRASE 4 laden@4)" },
618  { "\">>> 0) & 0xFF\"", "(0@1 PHRASE 2 0xff@2)" },
619  { "<!DOCTYPE HTML PUBLIC \"-//W3C//DTD HTML 4.01//EN\"> document.body.scrollHeight", "(doctype@1 OR html@2 OR public@3 OR (w3c@4 PHRASE 5 dtd@5 PHRASE 5 html@6 PHRASE 5 4.01@7 PHRASE 5 en@8) OR (document@9 PHRASE 3 body@10 PHRASE 3 scrollheight@11))" },
620  { "<BR>window.resizeBy(offsetX,offsetY)<P>kweet", "(br@1 OR (window@2 PHRASE 2 resizeby@3) OR Zoffsetx@4 OR Zoffseti@5 OR p@6 OR Zkweet@7)" },
621  { "linux humor :)", "(Zlinux@1 OR Zhumor@2)" },
622  { "ClassFactory kan aangevraagde klasse niet leveren (Fout=80040111)", "(classfactory@1 OR Zkan@2 OR Zaangevraagd@3 OR Zklass@4 OR Zniet@5 OR Zleveren@6 OR (fout@7 OR 80040111@8))" },
623  { "remote_smtp defer (-44)", "(Zremote_smtp@1 OR Zdefer@2 OR 44@3)" },
624  { "txtlogin.getText().trim().toUpperCase().intern() == inuser[2 * (i - 1) + 2].trim().toUpperCase().intern() && txtpass.getText().trim().toUpperCase().intern() == inuser[2 * (i - 1) + 3].trim().toUpperCase().intern())", "((txtlogin@1 PHRASE 2 gettext@2) OR trim@3 OR touppercase@4 OR intern@5 OR inuser@6 OR 2@7 OR Zi@8 OR 1@9 OR 2@10 OR trim@11 OR touppercase@12 OR intern@13 OR (txtpass@14 PHRASE 2 gettext@15) OR trim@16 OR touppercase@17 OR intern@18 OR inuser@19 OR 2@20 OR Zi@21 OR 1@22 OR 3@23 OR trim@24 OR touppercase@25 OR intern@26)" },
625  { "Koper + amoniak (NH2", "(koper@1 OR Zamoniak@2 OR nh2@3)" },
626  { "nec dvd -/+r", "((Znec@1 OR Zdvd@2) AND_NOT Zr@3)" },
627  { "er is een gereserveerde fout (-1104) opgetreden", "(Zer@1 OR Zis@2 OR Zeen@3 OR Zgereserveerd@4 OR Zfout@5 OR 1104@6 OR Zopgetreden@7)" },
628  { "Cor \\(CCN\\)'\" <cor.kloet@ccn.controlec.nl>", "(cor@1 OR ccn@2 OR (cor@3 PHRASE 5 kloet@4 PHRASE 5 ccn@5 PHRASE 5 controlec@6 PHRASE 5 nl@7))" },
629  { "Warning: Failed opening for inclusion (include_path='') in Unknown on line 0", "(warning@1 OR (failed@2 OR Zopen@3 OR Zfor@4 OR Zinclus@5) OR include_path@6 OR (Zin@7 OR unknown@8 OR Zon@9 OR Zline@10 OR 0@11))" },
630  { "\"~\" + \"c:\\\"", "Zc@1" },
631  { "mysql count(*)", "(Zmysql@1 OR count@2)" },
632  { "for %f in (*.*) do", "(Zfor@1 OR (Zf@2 OR Zin@3) OR Zdo@4)" },
633  { "raar \"~\" bestand", "(Zraar@1 OR Zbestand@2)" },
634  { "NEC DVD +-R/RW 1300", "(nec@1 OR dvd@2 OR (r@3 PHRASE 2 rw@4) OR 1300@5)" },
635  { "approved (ref: 38446-263)", "(Zapprov@1 OR (Zref@2 OR (38446@3 PHRASE 2 263@4)))" },
636  { "GA-7VRXP(2.0)", "((ga@1 PHRASE 2 7vrxp@2) OR 2.0@3)" },
637  { "~ Could not retrieve directory listing for \"/\"", "(could@1 OR Znot@2 OR Zretriev@3 OR Zdirectori@4 OR Zlist@5 OR Zfor@6)" },
638  { "asp CreateObject(\"Word.Document\")", "(Zasp@1 OR createobject@2 OR (word@3 PHRASE 2 document@4))" },
639  { "De lees- of schrijfbewerking (\"written\") op het geheugen is mislukt.", "(de@1 OR Zlee@2 OR Zof@3 OR Zschrijfbewerk@4 OR written@5 OR (Zop@6 OR Zhet@7 OR Zgeheugen@8 OR Zis@9 OR Zmislukt@10))" },
640  { "putStr (map (\\x -> chr (round (21/2 * x^3 - 92 * x^2 + 503/2 * x - 105))) [1..4])", "(Zputstr@1 OR (Zmap@2 OR ((Zx@3 OR (Zround@5 OR ((21@6 PHRASE 2 2@7) OR Zx@8 OR 3@9 OR 92@10 OR Zx@11 OR 2@12 OR (503@13 PHRASE 2 2@14) OR Zx@15 OR 105@16))) AND_NOT Zchr@4) OR (1@17 PHRASE 2 4@18)))" },
641  { "parent.document.getElementById(\\\"leftmenu\\\").cols", "((parent@1 PHRASE 3 document@2 PHRASE 3 getelementbyid@3) OR leftmenu@4 OR Zcol@5)" },
642  { "<% if not isEmpty(Request.QueryString) then", "(Zif@1 OR Znot@2 OR isempty@3 OR (request@4 PHRASE 2 querystring@5) OR Zthen@6)" },
643  { "Active Desktop (Hier issie)", "(active@1 OR desktop@2 OR (hier@3 OR Zissi@4))" },
644  { "Asus A7V8X (LAN + Sound)", "(asus@1 OR a7v8x@2 OR (lan@3 OR sound@4))" },
645  { "Novell This pentium class machine (or greater) lacks some required CPU feature(s", "(novell@1 OR this@2 OR Zpentium@3 OR Zclass@4 OR Zmachin@5 OR (Zor@6 OR Zgreater@7) OR (Zlack@8 OR Zsome@9 OR Zrequir@10 OR cpu@11 OR feature@12) OR Zs@13)" },
646  { "sql server install fails error code (-1)", "(Zsql@1 OR Zserver@2 OR Zinstal@3 OR Zfail@4 OR Zerror@5 OR Zcode@6 OR 1@7)" },
647  { "session_register(\"login\");", "(session_register@1 OR login@2)" },
648  { "\"kylix+ndmb\"", "(kylix@1 PHRASE 2 ndmb@2)" },
649  { "Cannot find imap library (libc-client.a).", "(cannot@1 OR Zfind@2 OR Zimap@3 OR Zlibrari@4 OR (libc@5 PHRASE 3 client@6 PHRASE 3 a@7))" },
650  { "If ($_SESSION[\"Login\"] == 1)", "(if@1 OR (_session@2 OR login@3 OR 1@4))" },
651  { "You have an error in your SQL syntax near '1')' at line 1", "(you@1 OR Zhave@2 OR Zan@3 OR Zerror@4 OR Zin@5 OR Zyour@6 OR sql@7 OR Zsyntax@8 OR Znear@9 OR 1@10 OR (Zat@11 OR Zline@12 OR 1@13))" },
652  { "ASRock K7VT2 (incl. LAN)", "(asrock@1 OR k7vt2@2 OR (Zincl@3 OR lan@4))" },
653  { "+windows98 +(geen communicatie) +ie5", "(Zwindows98@1 AND (Zgeen@2 OR Zcommunicati@3) AND Zie5@4)" },
654  { "\"xterm -fn\"", "(xterm@1 PHRASE 2 fn@2)" },
655  { "IRQL_NOT_LESS_OR_EQUAL", "irql_not_less_or_equal@1" },
656  { "access query \"NOT IN\"", "(Zaccess@1 OR Zqueri@2 OR (not@3 PHRASE 2 in@4))" },
657  { "\"phrase one \"phrase two\"", "((phrase@1 PHRASE 2 one@2) OR (Zphrase@3 OR two@4))" },
658  { "NEAR 207 46 249 27", "(near@1 OR 207@2 OR 46@3 OR 249@4 OR 27@5)" },
659  { "- NEAR 12V voeding", "(near@1 OR 12v@2 OR Zvoed@3)" },
660  { "waarom \"~\" in directorynaam", "(Zwaarom@1 OR (Zin@2 OR Zdirectorynaam@3))" },
661  { "cd'r NEAR toebehoren", "(cd'r@1 NEAR 11 toebehoren@2)" },
662  { "site:1 site:2", "0 * (H1 OR H2)" },
663  { "site:1 site2:2", "0 * (H1 AND J2)" },
664  { "site:1 site:2 site2:2", "0 * ((H1 OR H2) AND J2)" },
665  { "site:1 OR site:2", "(0 * H1 OR 0 * H2)" },
666  { "site:1 AND site:2", "(0 * H1 AND 0 * H2)" },
667  { "foo AND site:2", "(Zfoo@1 AND 0 * H2)" },
668  // Non-exclusive boolean prefixes feature tests (ticket#402):
669  { "category:1 category:2", "0 * (XCAT1 AND XCAT2)" },
670  { "category:1 site2:2", "0 * (XCAT1 AND J2)" },
671  { "category:1 category:2 site2:2", "0 * ((XCAT1 AND XCAT2) AND J2)" },
672  { "category:1 OR category:2", "(0 * XCAT1 OR 0 * XCAT2)" },
673  { "category:1 AND category:2", "(0 * XCAT1 AND 0 * XCAT2)" },
674  { "foo AND category:2", "(Zfoo@1 AND 0 * XCAT2)" },
675  // Regression test for combining multiple non-exclusive prefixes, fixed in
676  // 1.2.22 and 1.3.4.
677  { "category:1 dogegory:2", "0 * (XCAT1 AND XDOG2)" },
678  { "A site:1 site:2", "(a@1 FILTER (H1 OR H2))" },
679 #if 0
680  { "A (site:1 OR site:2)", "(a@1 FILTER (H1 OR H2))" },
681 #endif
682  { "A site:1 site2:2", "(a@1 FILTER (H1 AND J2))" },
683  { "A site:1 site:2 site2:2", "(a@1 FILTER ((H1 OR H2) AND J2))" },
684 #if 0
685  { "A site:1 OR site:2", "(a@1 FILTER (H1 OR H2))" },
686 #endif
687  { "A site:1 AND site:2", "((a@1 FILTER H1) AND 0 * H2)" },
688  { "site:xapian.org OR site:www.xapian.org", "(0 * Hxapian.org OR 0 * Hwww.xapian.org)" },
689  { "site:xapian.org site:www.xapian.org", "0 * (Hxapian.org OR Hwww.xapian.org)" },
690  { "site:xapian.org AND site:www.xapian.org", "(0 * Hxapian.org AND 0 * Hwww.xapian.org)" },
691  { "Xapian site:xapian.org site:www.xapian.org", "(xapian@1 FILTER (Hxapian.org OR Hwww.xapian.org))" },
692  { "author:richard author:olly writer:charlie", "(ZArichard@1 OR ZAolli@2 OR ZAcharli@3)"},
693  { "author:richard NEAR title:book", "(Arichard@1 NEAR 11 XTbook@2)"},
694  { "authortitle:richard NEAR title:book", "((Arichard@1 OR XTrichard@1) NEAR 11 XTbook@2)" },
695  { "multisite:xapian.org", "0 * (Hxapian.org OR Jxapian.org)"},
696  { "authortitle:richard", "(ZArichard@1 OR ZXTrichard@1)"},
697  { "multisite:xapian.org site:www.xapian.org author:richard authortitle:richard", "((ZArichard@1 OR (ZArichard@2 OR ZXTrichard@2)) FILTER ((Hxapian.org OR Jxapian.org) AND Hwww.xapian.org))" },
698  { "authortitle:richard-boulton", "((Arichard@1 PHRASE 2 Aboulton@2) OR (XTrichard@1 PHRASE 2 XTboulton@2))"},
699  { "authortitle:\"richard boulton\"", "((Arichard@1 PHRASE 2 Aboulton@2) OR (XTrichard@1 PHRASE 2 XTboulton@2))"},
700  // Test FLAG_CJK_NGRAM isn't on by default:
701  { "久有归天愿", "Z久有归天愿@1" },
702  { NULL, "CJK" }, // Enable FLAG_CJK_NGRAM
703  // Test non-CJK queries still parse the same:
704  { "gtk+ -gnome", "(Zgtk+@1 AND_NOT Zgnome@2)" },
705  { "“curly quotes”", "(curly@1 PHRASE 2 quotes@2)" },
706  // Test n-gram generation:
707  { "久有归天愿", "(久@1 AND 久有@1 AND 有@1 AND 有归@1 AND 归@1 AND 归天@1 AND 天@1 AND 天愿@1 AND 愿@1)" },
708  { "久有 归天愿", "((久@1 AND 久有@1 AND 有@1) OR (归@2 AND 归天@2 AND 天@2 AND 天愿@2 AND 愿@2))" },
709  { "久有!归天愿", "((久@1 AND 久有@1 AND 有@1) OR (归@2 AND 归天@2 AND 天@2 AND 天愿@2 AND 愿@2))" },
710  { "title:久有 归 天愿", "((XT久@1 AND XT久有@1 AND XT有@1) OR 归@2 OR (天@3 AND 天愿@3 AND 愿@3))" },
711  { "h众ello万众", "(Zh@1 OR 众@2 OR Zello@3 OR (万@4 AND 万众@4 AND 众@4))" },
712  { "世(の中)TEST_tm", "(世@1 OR (の@2 AND の中@2 AND 中@2) OR test_tm@3)" },
713  { "다녀 AND 와야", "(다@1 AND 다녀@1 AND 녀@1 AND (와@2 AND 와야@2 AND 야@2))" },
714  { "authortitle:학술 OR 연구를", "((A학@1 AND A학술@1 AND A술@1) OR (XT학@1 AND XT학술@1 AND XT술@1) OR (연@2 AND 연구@2 AND 구@2 AND 구를@2 AND 를@2))" },
715  // FIXME: These should really filter by bigrams to accelerate:
716  { "\"久有归\"", "(久@1 PHRASE 3 有@1 PHRASE 3 归@1)" },
717  { "\"久有test归\"", "(久@1 PHRASE 4 有@1 PHRASE 4 test@2 PHRASE 4 归@3)" },
718  // FIXME: this should work: { "久 NEAR 有", "(久@1 NEAR 11 有@2)" },
719  { NULL, NULL }
720 };
721 
722 DEFINE_TESTCASE(queryparser1, !backend) {
723  Xapian::QueryParser queryparser;
724  queryparser.set_stemmer(Xapian::Stem("english"));
726  queryparser.add_prefix("author", "A");
727  queryparser.add_prefix("writer", "A");
728  queryparser.add_prefix("title", "XT");
729  queryparser.add_prefix("subject", "XT");
730  queryparser.add_prefix("authortitle", "A");
731  queryparser.add_prefix("authortitle", "XT");
732  queryparser.add_boolean_prefix("site", "H");
733  queryparser.add_boolean_prefix("site2", "J");
734  queryparser.add_boolean_prefix("multisite", "H");
735  queryparser.add_boolean_prefix("multisite", "J");
736  queryparser.add_boolean_prefix("category", "XCAT", false);
737  queryparser.add_boolean_prefix("dogegory", "XDOG", false);
739  queryparser.add_boolean_prefix("authortitle", "B");
740  );
742  queryparser.add_prefix("multisite", "B");
743  );
744  unsigned flags = queryparser.FLAG_DEFAULT;
745  for (const test *p = test_or_queries; ; ++p) {
746  if (!p->query) {
747  if (!p->expect) break;
748  if (strcmp(p->expect, "CJK") == 0) {
749  flags = queryparser.FLAG_DEFAULT|queryparser.FLAG_CJK_NGRAM;
750  continue;
751  }
752  FAIL_TEST("Unknown flag code: " << p->expect);
753  }
754  string expect, parsed;
755  if (p->expect)
756  expect = p->expect;
757  else
758  expect = "parse error";
759  try {
760  Xapian::Query qobj = queryparser.parse_query(p->query, flags);
761  parsed = qobj.get_description();
762  expect = string("Query(") + expect + ')';
763  } catch (const Xapian::QueryParserError &e) {
764  parsed = e.get_msg();
765  } catch (const Xapian::Error &e) {
766  parsed = e.get_description();
767  } catch (...) {
768  parsed = "Unknown exception!";
769  }
770  tout << "Query: " << p->query << '\n';
771  TEST_STRINGS_EQUAL(parsed, expect);
772  }
773 }
774 
775 static const test test_and_queries[] = {
776  { "internet explorer title:(http www)", "(Zinternet@1 AND Zexplor@2 AND (ZXThttp@3 AND ZXTwww@4))" },
777  // Regression test for bug in 0.9.2 and earlier - this would give
778  // (two@2 AND_MAYBE (one@1 AND three@3))
779  { "one +two three", "(Zone@1 AND Ztwo@2 AND Zthree@3)" },
780  { "hello -title:\"hello world\"", "(Zhello@1 AND_NOT (XThello@2 PHRASE 2 XTworld@3))" },
781  // Regression test for bug fixed in 1.0.4 - the '-' would be ignored there
782  // because the whitespace after the '"' wasn't noticed.
783  { "\"hello world\" -C++", "((hello@1 PHRASE 2 world@2) AND_NOT c++@3)" },
784  // Regression tests for bug fixed in 1.0.4 - queries with only boolean
785  // filter and HATE terms weren't accepted.
786  { "-cup site:world", "(0 * Hworld AND_NOT Zcup@1)" },
787  { "site:world -cup", "(0 * Hworld AND_NOT Zcup@1)" },
788  // Regression test for bug fixed in 1.0.4 - the KET token for ')' was lost.
789  { "(site:world) -cup", "(0 * Hworld AND_NOT Zcup@1)" },
790  // Regression test for bug fixed in 1.0.4 - a boolean filter term between
791  // probabilistic terms caused a parse error (probably broken during the
792  // addition of synonym support in 1.0.2).
793  { "foo site:xapian.org bar", "((Zfoo@1 AND Zbar@2) FILTER Hxapian.org)" },
794  // Add coverage for other cases similar to the above.
795  { "a b site:xapian.org", "((Za@1 AND Zb@2) FILTER Hxapian.org)" },
796  { "site:xapian.org a b", "((Za@1 AND Zb@2) FILTER Hxapian.org)" },
797  { NULL, "CJK" }, // Enable FLAG_CJK_NGRAM
798  // Test n-gram generation:
799  { "author:험가 OR subject:万众 hello world!", "((A험@1 AND A험가@1 AND A가@1) OR (XT万@2 AND XT万众@2 AND XT众@2 AND (Zhello@3 AND Zworld@4)))" },
800  { "洛伊one儿差点two脸three", "(洛@1 AND 洛伊@1 AND 伊@1 AND Zone@2 AND (儿@3 AND 儿差@3 AND 差@3 AND 差点@3 AND 点@3) AND Ztwo@4 AND 脸@5 AND Zthree@6)" },
801  { NULL, NULL }
802 };
803 
804 // With default_op = OP_AND.
805 DEFINE_TESTCASE(qp_default_op1, !backend) {
806  Xapian::QueryParser queryparser;
807  queryparser.set_stemmer(Xapian::Stem("english"));
809  queryparser.add_prefix("author", "A");
810  queryparser.add_prefix("title", "XT");
811  queryparser.add_prefix("subject", "XT");
812  queryparser.add_boolean_prefix("site", "H");
814  unsigned flags = queryparser.FLAG_DEFAULT;
815  for (const test *p = test_and_queries; ; ++p) {
816  if (!p->query) {
817  if (!p->expect) break;
818  if (strcmp(p->expect, "CJK") == 0) {
819  flags = queryparser.FLAG_DEFAULT|queryparser.FLAG_CJK_NGRAM;
820  continue;
821  }
822  FAIL_TEST("Unknown flag code: " << p->expect);
823  }
824  string expect, parsed;
825  if (p->expect)
826  expect = p->expect;
827  else
828  expect = "parse error";
829  try {
830  Xapian::Query qobj = queryparser.parse_query(p->query, flags);
831  parsed = qobj.get_description();
832  expect = string("Query(") + expect + ')';
833  } catch (const Xapian::QueryParserError &e) {
834  parsed = e.get_msg();
835  } catch (const Xapian::Error &e) {
836  parsed = e.get_description();
837  } catch (...) {
838  parsed = "Unknown exception!";
839  }
840  tout << "Query: " << p->query << '\n';
841  TEST_STRINGS_EQUAL(parsed, expect);
842  }
843 }
844 
845 // Feature test for specify the default prefix (new in Xapian 1.0.0).
846 DEFINE_TESTCASE(qp_default_prefix1, !backend) {
848  qp.set_stemmer(Xapian::Stem("english"));
850  qp.add_prefix("title", "XT");
851 
852  Xapian::Query qobj;
853  qobj = qp.parse_query("hello world", 0, "A");
854  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((ZAhello@1 OR ZAworld@2))");
855  qobj = qp.parse_query("me title:stuff", 0, "A");
856  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((ZAme@1 OR ZXTstuff@2))");
857  qobj = qp.parse_query("title:(stuff) me", Xapian::QueryParser::FLAG_BOOLEAN, "A");
858  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((ZXTstuff@1 OR ZAme@2))");
859  qobj = qp.parse_query("英国 title:文森hello", qp.FLAG_CJK_NGRAM, "A");
860  TEST_STRINGS_EQUAL(qobj.get_description(), "Query(((A英@1 AND A英国@1 AND A国@1) OR (XT文@2 AND XT文森@2 AND XT森@2) OR ZAhello@3))");
861 }
862 
863 // Feature test for setting the default prefix with add_prefix()
864 // (new in Xapian 1.0.3).
865 DEFINE_TESTCASE(qp_default_prefix2, !backend) {
867  qp.set_stemmer(Xapian::Stem("english"));
869 
870  // test that default prefixes can only be set with add_prefix().
872  qp.add_boolean_prefix("", "B");
873  );
874 
875  qp.add_prefix("title", "XT");
876  qp.add_prefix("", "A");
877 
878  Xapian::Query qobj;
879  qobj = qp.parse_query("hello world", 0);
880  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((ZAhello@1 OR ZAworld@2))");
881  qobj = qp.parse_query("me title:stuff", 0);
882  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((ZAme@1 OR ZXTstuff@2))");
883  qobj = qp.parse_query("title:(stuff) me", Xapian::QueryParser::FLAG_BOOLEAN);
884  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((ZXTstuff@1 OR ZAme@2))");
885 
886  qobj = qp.parse_query("hello world", 0, "B");
887  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((ZBhello@1 OR ZBworld@2))");
888  qobj = qp.parse_query("me title:stuff", 0, "B");
889  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((ZBme@1 OR ZXTstuff@2))");
890  qobj = qp.parse_query("title:(stuff) me", Xapian::QueryParser::FLAG_BOOLEAN, "B");
891  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((ZXTstuff@1 OR ZBme@2))");
892 
893  qp.add_prefix("", "B");
894  qobj = qp.parse_query("me-us title:(stuff) me", Xapian::QueryParser::FLAG_BOOLEAN);
895  TEST_STRINGS_EQUAL(qobj.get_description(), "Query(((Ame@1 PHRASE 2 Aus@2) OR (Bme@1 PHRASE 2 Bus@2) OR ZXTstuff@3 OR (ZAme@4 OR ZBme@4)))");
896  qobj = qp.parse_query("me-us title:(stuff) me", Xapian::QueryParser::FLAG_BOOLEAN, "C");
897  TEST_STRINGS_EQUAL(qobj.get_description(), "Query(((Cme@1 PHRASE 2 Cus@2) OR ZXTstuff@3 OR ZCme@4))");
898 
899  qobj = qp.parse_query("me-us title:\"not-me\"", Xapian::QueryParser::FLAG_PHRASE);
900  TEST_STRINGS_EQUAL(qobj.get_description(), "Query(((Ame@1 PHRASE 2 Aus@2) OR (Bme@1 PHRASE 2 Bus@2) OR (XTnot@3 PHRASE 2 XTme@4)))");
901 }
902 
903 // Test query with odd characters in.
904 DEFINE_TESTCASE(qp_odd_chars1, !backend) {
906  string query("\x01weird\x00stuff\x7f", 13);
907  Xapian::Query qobj = qp.parse_query(query);
908  tout << "Query: " << query << '\n';
909  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((weird@1 OR stuff@2))"); // FIXME: should these be stemmed?
910 }
911 
912 // Test right truncation.
913 DEFINE_TESTCASE(qp_flag_wildcard1, writable) {
915  Xapian::Document doc;
916  doc.add_term("abc");
917  doc.add_term("main");
918  doc.add_term("muscat");
919  doc.add_term("muscle");
920  doc.add_term("musclebound");
921  doc.add_term("muscular");
922  doc.add_term("mutton");
923  db.add_document(doc);
925  qp.set_database(db);
927  TEST_STRINGS_EQUAL(qobj.get_description(), "Query(WILDCARD SYNONYM ab)");
928  qobj = qp.parse_query("muscle*", Xapian::QueryParser::FLAG_WILDCARD);
929  TEST_STRINGS_EQUAL(qobj.get_description(), "Query(WILDCARD SYNONYM muscle)");
931  TEST_STRINGS_EQUAL(qobj.get_description(), "Query(WILDCARD SYNONYM meat)");
933  TEST_STRINGS_EQUAL(qobj.get_description(), "Query(WILDCARD SYNONYM musc)");
935  TEST_STRINGS_EQUAL(qobj.get_description(), "Query(WILDCARD SYNONYM mutt)");
936  // Regression test (we weren't lowercasing terms before checking if they
937  // were in the database or not):
938  qobj = qp.parse_query("mUTTON++");
939  TEST_STRINGS_EQUAL(qobj.get_description(), "Query(mutton@1)");
940  // Regression test: check that wildcards work with +terms.
941  unsigned flags = Xapian::QueryParser::FLAG_WILDCARD |
943  qobj = qp.parse_query("+mai* main", flags);
944  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((WILDCARD SYNONYM mai AND_MAYBE main@2))");
945  // Regression test (if we had a +term which was a wildcard and wasn't
946  // present, the query could still match documents).
947  qobj = qp.parse_query("foo* main", flags);
948  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((WILDCARD SYNONYM foo OR main@2))");
949  qobj = qp.parse_query("main foo*", flags);
950  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((main@1 OR WILDCARD SYNONYM foo))");
951  qobj = qp.parse_query("+foo* main", flags);
952  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((WILDCARD SYNONYM foo AND_MAYBE main@2))");
953  qobj = qp.parse_query("main +foo*", flags);
954  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((WILDCARD SYNONYM foo AND_MAYBE main@1))");
955  qobj = qp.parse_query("foo* +main", flags);
956  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((main@2 AND_MAYBE WILDCARD SYNONYM foo))");
957  qobj = qp.parse_query("+main foo*", flags);
958  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((main@1 AND_MAYBE WILDCARD SYNONYM foo))");
959  qobj = qp.parse_query("+foo* +main", flags);
960  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((WILDCARD SYNONYM foo AND main@2))");
961  qobj = qp.parse_query("+main +foo*", flags);
962  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((main@1 AND WILDCARD SYNONYM foo))");
963  qobj = qp.parse_query("foo* mai", flags);
964  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((WILDCARD SYNONYM foo OR mai@2))");
965  qobj = qp.parse_query("mai foo*", flags);
966  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((mai@1 OR WILDCARD SYNONYM foo))");
967  qobj = qp.parse_query("+foo* mai", flags);
968  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((WILDCARD SYNONYM foo AND_MAYBE mai@2))");
969  qobj = qp.parse_query("mai +foo*", flags);
970  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((WILDCARD SYNONYM foo AND_MAYBE mai@1))");
971  qobj = qp.parse_query("foo* +mai", flags);
972  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((mai@2 AND_MAYBE WILDCARD SYNONYM foo))");
973  qobj = qp.parse_query("+mai foo*", flags);
974  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((mai@1 AND_MAYBE WILDCARD SYNONYM foo))");
975  qobj = qp.parse_query("+foo* +mai", flags);
976  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((WILDCARD SYNONYM foo AND mai@2))");
977  qobj = qp.parse_query("+mai +foo*", flags);
978  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((mai@1 AND WILDCARD SYNONYM foo))");
979  qobj = qp.parse_query("-foo* main", flags);
980  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((main@2 AND_NOT WILDCARD SYNONYM foo))");
981  qobj = qp.parse_query("main -foo*", flags);
982  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((main@1 AND_NOT WILDCARD SYNONYM foo))");
983  qobj = qp.parse_query("main -foo* -bar", flags);
984  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((main@1 AND_NOT (WILDCARD SYNONYM foo OR bar@3)))");
985  qobj = qp.parse_query("main -bar -foo*", flags);
986  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((main@1 AND_NOT (bar@2 OR WILDCARD SYNONYM foo)))");
987  // Check with OP_AND too.
989  qobj = qp.parse_query("foo* main", flags);
990  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((WILDCARD SYNONYM foo AND main@2))");
991  qobj = qp.parse_query("main foo*", flags);
992  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((main@1 AND WILDCARD SYNONYM foo))");
994  qobj = qp.parse_query("+foo* main", flags);
995  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((WILDCARD SYNONYM foo AND main@2))");
996  qobj = qp.parse_query("main +foo*", flags);
997  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((main@1 AND WILDCARD SYNONYM foo))");
998  qobj = qp.parse_query("-foo* main", flags);
999  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((main@2 AND_NOT WILDCARD SYNONYM foo))");
1000  qobj = qp.parse_query("main -foo*", flags);
1001  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((main@1 AND_NOT WILDCARD SYNONYM foo))");
1002  // Check empty wildcard followed by negation.
1004  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((WILDCARD SYNONYM foo AND_NOT main@2))");
1005  // Regression test for bug#484 fixed in 1.2.1 and 1.0.21.
1006  qobj = qp.parse_query("abc muscl* main", flags);
1007  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((abc@1 AND WILDCARD SYNONYM muscl AND main@3))");
1008 }
1009 
1010 // Test right truncation with prefixes.
1011 DEFINE_TESTCASE(qp_flag_wildcard2, writable) {
1013  Xapian::Document doc;
1014  doc.add_term("Aheinlein");
1015  doc.add_term("Ahuxley");
1016  doc.add_term("hello");
1017  db.add_document(doc);
1019  qp.set_database(db);
1020  qp.add_prefix("author", "A");
1021  Xapian::Query qobj;
1022  qobj = qp.parse_query("author:h*", Xapian::QueryParser::FLAG_WILDCARD);
1023  TEST_STRINGS_EQUAL(qobj.get_description(), "Query(WILDCARD SYNONYM Ah)");
1024  qobj = qp.parse_query("author:h* test", Xapian::QueryParser::FLAG_WILDCARD);
1025  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((WILDCARD SYNONYM Ah OR test@2))");
1026 }
1027 
1028 static void
1030  Xapian::termcount max_expansion,
1031  const string & query_string)
1032 {
1034  qp.set_database(db);
1035  qp.set_max_expansion(max_expansion);
1036  Xapian::Enquire e(db);
1038  // The exception for expanding too much may happen at parse time or later
1039  // so we need to calculate the MSet too.
1040  e.get_mset(0, 10);
1041 }
1042 
1043 // Test right truncation with a limit on expansion.
1044 DEFINE_TESTCASE(qp_flag_wildcard3, writable) {
1046  Xapian::Document doc;
1047  doc.add_term("abc");
1048  doc.add_term("main");
1049  doc.add_term("muscat");
1050  doc.add_term("muscle");
1051  doc.add_term("musclebound");
1052  doc.add_term("muscular");
1053  doc.add_term("mutton");
1054  db.add_document(doc);
1055 
1056  // Test that a max of 0 doesn't set a limit.
1057  test_qp_flag_wildcard3_helper(db, 0, "z*");
1058  test_qp_flag_wildcard3_helper(db, 0, "m*");
1059 
1060  // These cases should expand to the limit given.
1061  test_qp_flag_wildcard3_helper(db, 1, "z*");
1062  test_qp_flag_wildcard3_helper(db, 1, "ab*");
1063  test_qp_flag_wildcard3_helper(db, 2, "muscle*");
1064  test_qp_flag_wildcard3_helper(db, 4, "musc*");
1065  test_qp_flag_wildcard3_helper(db, 4, "mus*");
1066  test_qp_flag_wildcard3_helper(db, 5, "mu*");
1067  test_qp_flag_wildcard3_helper(db, 6, "m*");
1068 
1069  // These cases should expand to one more than the limit.
1071  test_qp_flag_wildcard3_helper(db, 1, "muscle*"));
1073  test_qp_flag_wildcard3_helper(db, 3, "musc*"));
1075  test_qp_flag_wildcard3_helper(db, 3, "mus*"));
1077  test_qp_flag_wildcard3_helper(db, 4, "mu*"));
1079  test_qp_flag_wildcard3_helper(db, 5, "m*"));
1080 }
1081 
1082 // Test partial queries.
1083 DEFINE_TESTCASE(qp_flag_partial1, writable) {
1085  Xapian::Document doc;
1086  Xapian::Stem stemmer("english");
1087  doc.add_term("abc");
1088  doc.add_term("main");
1089  doc.add_term("muscat");
1090  doc.add_term("muscle");
1091  doc.add_term("musclebound");
1092  doc.add_term("muscular");
1093  doc.add_term("mutton");
1094  doc.add_term("Z" + stemmer("outside"));
1095  doc.add_term("Z" + stemmer("out"));
1096  doc.add_term("outside");
1097  doc.add_term("out");
1098  doc.add_term("XTcove");
1099  doc.add_term("XTcows");
1100  doc.add_term("XTcowl");
1101  doc.add_term("XTcox");
1102  doc.add_term("ZXTcow");
1103  doc.add_term("XONEpartial");
1104  doc.add_term("XONEpartial2");
1105  doc.add_term("XTWOpartial3");
1106  doc.add_term("XTWOpartial4");
1107  db.add_document(doc);
1109  qp.set_database(db);
1110  qp.set_stemmer(stemmer);
1112  qp.add_prefix("title", "XT");
1113  qp.add_prefix("double", "XONE");
1114  qp.add_prefix("double", "XTWO");
1115 
1116  // Check behaviour with unstemmed terms
1118  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((WILDCARD SYNONYM a OR Za@1))");
1120  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((WILDCARD SYNONYM ab OR Zab@1))");
1121  qobj = qp.parse_query("muscle", Xapian::QueryParser::FLAG_PARTIAL);
1122  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((WILDCARD SYNONYM muscle OR Zmuscl@1))");
1124  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((WILDCARD SYNONYM meat OR Zmeat@1))");
1126  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((WILDCARD SYNONYM musc OR Zmusc@1))");
1128  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((WILDCARD SYNONYM mutt OR Zmutt@1))");
1129  qobj = qp.parse_query("abc musc", Xapian::QueryParser::FLAG_PARTIAL);
1130  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((Zabc@1 OR (WILDCARD SYNONYM musc OR Zmusc@2)))");
1132  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((WILDCARD SYNONYM a OR (WILDCARD SYNONYM mutt OR Zmutt@2)))");
1133 
1134  // Check behaviour with stemmed terms, and stem strategy STEM_SOME.
1136  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((WILDCARD SYNONYM o OR Zo@1))");
1138  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((WILDCARD SYNONYM ou OR Zou@1))");
1140  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((WILDCARD SYNONYM out OR Zout@1))");
1142  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((WILDCARD SYNONYM outs OR Zout@1))");
1143  qobj = qp.parse_query("outsi", Xapian::QueryParser::FLAG_PARTIAL);
1144  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((WILDCARD SYNONYM outsi OR Zoutsi@1))");
1145  qobj = qp.parse_query("outsid", Xapian::QueryParser::FLAG_PARTIAL);
1146  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((WILDCARD SYNONYM outsid OR Zoutsid@1))");
1147  qobj = qp.parse_query("outside", Xapian::QueryParser::FLAG_PARTIAL);
1148  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((WILDCARD SYNONYM outside OR Zoutsid@1))");
1149 
1150  // Check behaviour with capitalised terms, and stem strategy STEM_SOME.
1152  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((WILDCARD SYNONYM out OR out@1))");
1154  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((WILDCARD SYNONYM outs OR outs@1))");
1155  qobj = qp.parse_query("Outside", Xapian::QueryParser::FLAG_PARTIAL);
1156  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((WILDCARD SYNONYM outside OR outside@1))");
1157  // FIXME: Used to be this, but we aren't currently doing this change:
1158  // TEST_STRINGS_EQUAL(qobj.get_description(), "Query(outside@1#2)");
1159 
1160  // And now with stemming strategy STEM_SOME_FULL_POS.
1163  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((WILDCARD SYNONYM out OR out@1))");
1165  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((WILDCARD SYNONYM outs OR outs@1))");
1166  qobj = qp.parse_query("Outside", Xapian::QueryParser::FLAG_PARTIAL);
1167  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((WILDCARD SYNONYM outside OR outside@1))");
1168 
1169  // And now with stemming strategy STEM_ALL.
1172  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((WILDCARD SYNONYM out OR out@1))");
1174  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((WILDCARD SYNONYM outs OR out@1))");
1175  qobj = qp.parse_query("Outside", Xapian::QueryParser::FLAG_PARTIAL);
1176  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((WILDCARD SYNONYM outside OR outsid@1))");
1177 
1178  // And now with stemming strategy STEM_ALL_Z.
1181  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((WILDCARD SYNONYM out OR Zout@1))");
1183  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((WILDCARD SYNONYM outs OR Zout@1))");
1184  qobj = qp.parse_query("Outside", Xapian::QueryParser::FLAG_PARTIAL);
1185  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((WILDCARD SYNONYM outside OR Zoutsid@1))");
1186 
1187  // Check handling of a case with a prefix.
1189  qobj = qp.parse_query("title:cow", Xapian::QueryParser::FLAG_PARTIAL);
1190  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((WILDCARD SYNONYM XTcow OR ZXTcow@1))");
1191  qobj = qp.parse_query("title:cows", Xapian::QueryParser::FLAG_PARTIAL);
1192  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((WILDCARD SYNONYM XTcows OR ZXTcow@1))");
1193  qobj = qp.parse_query("title:Cow", Xapian::QueryParser::FLAG_PARTIAL);
1194  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((WILDCARD SYNONYM XTcow OR XTcow@1))");
1195  qobj = qp.parse_query("title:Cows", Xapian::QueryParser::FLAG_PARTIAL);
1196  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((WILDCARD SYNONYM XTcows OR XTcows@1))");
1197  // FIXME: Used to be this, but we aren't currently doing this change:
1198  // TEST_STRINGS_EQUAL(qobj.get_description(), "Query(XTcows@1#2)");
1199 
1200  // Regression test - the initial version of the multi-prefix code would
1201  // inflate the wqf of the "parsed as normal" version of a partial term
1202  // by multiplying it by the number of prefixes mapped to.
1203  qobj = qp.parse_query("double:vision", Xapian::QueryParser::FLAG_PARTIAL);
1204  TEST_STRINGS_EQUAL(qobj.get_description(), "Query(((WILDCARD OR XONEvision SYNONYM WILDCARD OR XTWOvision) OR (ZXONEvision@1 SYNONYM ZXTWOvision@1)))");
1205 
1206  // Test handling of FLAG_PARTIAL when there's more than one prefix.
1207  qobj = qp.parse_query("double:part", Xapian::QueryParser::FLAG_PARTIAL);
1208  TEST_STRINGS_EQUAL(qobj.get_description(), "Query(((WILDCARD OR XONEpart SYNONYM WILDCARD OR XTWOpart) OR (ZXONEpart@1 SYNONYM ZXTWOpart@1)))");
1209 
1210  // Test handling of FLAG_PARTIAL when there's more than one prefix, without
1211  // stemming.
1213  qobj = qp.parse_query("double:part", Xapian::QueryParser::FLAG_PARTIAL);
1214  TEST_STRINGS_EQUAL(qobj.get_description(), "Query(((WILDCARD OR XONEpart SYNONYM WILDCARD OR XTWOpart) OR (XONEpart@1 SYNONYM XTWOpart@1)))");
1215  qobj = qp.parse_query("double:partial", Xapian::QueryParser::FLAG_PARTIAL);
1216  TEST_STRINGS_EQUAL(qobj.get_description(), "Query(((WILDCARD OR XONEpartial SYNONYM WILDCARD OR XTWOpartial) OR (XONEpartial@1 SYNONYM XTWOpartial@1)))");
1217 }
1218 
1219 // Tests for document counts for wildcard queries.
1220 // Regression test for bug fixed in 1.0.0.
1221 DEFINE_TESTCASE(wildquery1, backend) {
1222  Xapian::QueryParser queryparser;
1223  unsigned flags = Xapian::QueryParser::FLAG_WILDCARD |
1225  queryparser.set_stemmer(Xapian::Stem("english"));
1227  Xapian::Database db = get_database("apitest_simpledata");
1228  queryparser.set_database(db);
1229  Xapian::Enquire enquire(db);
1230 
1231  Xapian::Query qobj = queryparser.parse_query("th*", flags);
1232  tout << qobj.get_description() << endl;
1233  enquire.set_query(qobj);
1234  Xapian::MSet mymset = enquire.get_mset(0, 10);
1235  // Check that 6 documents were returned.
1236  TEST_MSET_SIZE(mymset, 6);
1237 
1238  qobj = queryparser.parse_query("notindb* \"this\"", flags);
1239  tout << qobj.get_description() << endl;
1240  enquire.set_query(qobj);
1241  mymset = enquire.get_mset(0, 10);
1242  // Check that 6 documents were returned.
1243  TEST_MSET_SIZE(mymset, 6);
1244 
1245  qobj = queryparser.parse_query("+notindb* \"this\"", flags);
1246  tout << qobj.get_description() << endl;
1247  enquire.set_query(qobj);
1248  mymset = enquire.get_mset(0, 10);
1249  // Check that 0 documents were returned.
1250  TEST_MSET_SIZE(mymset, 0);
1251 }
1252 
1253 DEFINE_TESTCASE(qp_flag_bool_any_case1, !backend) {
1254  using Xapian::QueryParser;
1256  Xapian::Query qobj;
1257  qobj = qp.parse_query("to and fro", QueryParser::FLAG_BOOLEAN | QueryParser::FLAG_BOOLEAN_ANY_CASE);
1258  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((to@1 AND fro@2))");
1259  qobj = qp.parse_query("to and fro", QueryParser::FLAG_BOOLEAN);
1260  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((to@1 OR and@2 OR fro@3))");
1261  // Regression test for bug in 0.9.4 and earlier.
1262  qobj = qp.parse_query("to And fro", QueryParser::FLAG_BOOLEAN | QueryParser::FLAG_BOOLEAN_ANY_CASE);
1263  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((to@1 AND fro@2))");
1264  qobj = qp.parse_query("to And fro", QueryParser::FLAG_BOOLEAN);
1265  TEST_STRINGS_EQUAL(qobj.get_description(), "Query((to@1 OR and@2 OR fro@3))");
1266 }
1267 
1268 static const test test_stop_queries[] = {
1269  { "test the queryparser", "(test@1 AND queryparser@3)" },
1270  // Regression test for bug in 0.9.6 and earlier. This would fail to
1271  // parse.
1272  { "test AND the AND queryparser", "(test@1 AND the@2 AND queryparser@3)" },
1273  // 0.9.6 and earlier ignored a stopword even if it was the only term.
1274  // More recent versions don't ever treat a single term as a stopword.
1275  { "the", "the@1" },
1276  // 1.2.2 and earlier ignored an all-stopword query with multiple terms,
1277  // which prevents 'to be or not to be' for being searchable unless the
1278  // user made it into a phrase query or prefixed all terms with '+'
1279  // (ticket#245).
1280  { "an the a", "(an@1 AND the@2 AND a@3)" },
1281  // Regression test for bug in initial version of the patch for the
1282  // "all-stopword" case.
1283  { "the AND a an", "(the@1 AND (a@2 AND an@3))" },
1284  { NULL, NULL }
1285 };
1286 
1287 DEFINE_TESTCASE(qp_stopper1, !backend) {
1289  static const char * const stopwords[] = { "a", "an", "the" };
1290  Xapian::SimpleStopper stop(stopwords, stopwords + 3);
1291  qp.set_stopper(&stop);
1293  for (const test *p = test_stop_queries; p->query; ++p) {
1294  string expect, parsed;
1295  if (p->expect)
1296  expect = p->expect;
1297  else
1298  expect = "parse error";
1299  try {
1300  Xapian::Query qobj = qp.parse_query(p->query);
1301  parsed = qobj.get_description();
1302  expect = string("Query(") + expect + ')';
1303  } catch (const Xapian::QueryParserError &e) {
1304  parsed = e.get_msg();
1305  } catch (const Xapian::Error &e) {
1306  parsed = e.get_description();
1307  } catch (...) {
1308  parsed = "Unknown exception!";
1309  }
1310  tout << "Query: " << p->query << '\n';
1311  TEST_STRINGS_EQUAL(parsed, expect);
1312  }
1313 }
1314 
1315 static const test test_pure_not_queries[] = {
1316  { "NOT windows", "(<alldocuments> AND_NOT Zwindow@1)" },
1317  { "a AND (NOT b)", "(Za@1 AND (<alldocuments> AND_NOT Zb@2))" },
1318  { "AND NOT windows", "Syntax: <expression> AND NOT <expression>" },
1319  { "gordian NOT", "Syntax: <expression> NOT <expression>" },
1320  { "gordian AND NOT", "Syntax: <expression> AND NOT <expression>" },
1321  { NULL, NULL }
1322 };
1323 
1324 DEFINE_TESTCASE(qp_flag_pure_not1, !backend) {
1325  using Xapian::QueryParser;
1327  qp.set_stemmer(Xapian::Stem("english"));
1328  qp.set_stemming_strategy(QueryParser::STEM_SOME);
1329  for (const test *p = test_pure_not_queries; p->query; ++p) {
1330  string expect, parsed;
1331  if (p->expect)
1332  expect = p->expect;
1333  else
1334  expect = "parse error";
1335  try {
1336  Xapian::Query qobj = qp.parse_query(p->query,
1337  QueryParser::FLAG_BOOLEAN |
1338  QueryParser::FLAG_PURE_NOT);
1339  parsed = qobj.get_description();
1340  expect = string("Query(") + expect + ')';
1341  } catch (const Xapian::QueryParserError &e) {
1342  parsed = e.get_msg();
1343  } catch (const Xapian::Error &e) {
1344  parsed = e.get_description();
1345  } catch (...) {
1346  parsed = "Unknown exception!";
1347  }
1348  tout << "Query: " << p->query << '\n';
1349  TEST_STRINGS_EQUAL(parsed, expect);
1350  }
1351 }
1352 
1353 // Debatable if this is a regression test or a feature test, as it's not
1354 // obvious is this was a bug fix or a new feature. Either way, it first
1355 // appeared in Xapian 1.0.0.
1356 DEFINE_TESTCASE(qp_unstem_boolean_prefix, !backend) {
1358  qp.add_boolean_prefix("test", "XTEST");
1359  Xapian::Query q = qp.parse_query("hello test:foo");
1360  TEST_STRINGS_EQUAL(q.get_description(), "Query((hello@1 FILTER XTESTfoo))");
1361  Xapian::TermIterator u = qp.unstem_begin("XTESTfoo");
1362  TEST(u != qp.unstem_end("XTESTfoo"));
1363  TEST_EQUAL(*u, "test:foo");
1364  ++u;
1365  TEST(u == qp.unstem_end("XTESTfoo"));
1366 }
1367 
1368 // Feature test for FLAG_ACCUMULATE.
1369 DEFINE_TESTCASE(qp_accumulate, !backend) {
1371  Xapian::SimpleStopper stopper;
1372  stopper.add("a");
1373  stopper.add("the");
1374  qp.set_stopper(&stopper);
1375  qp.set_stemmer(Xapian::Stem("en"));
1376  qp.add_boolean_prefix("test", "XTEST");
1377  qp.add_prefix("foo", "XFOO");
1378  Xapian::Query q = qp.parse_query("a plains test:bools foo:fielded");
1379  tout << q.get_description() << '\n';
1380  {
1381  Xapian::TermIterator t = qp.unstem_begin("Zplain");
1382  TEST(t != qp.unstem_end("Zplain"));
1383  TEST_EQUAL(*t, "plains");
1384  ++t;
1385  TEST(t == qp.unstem_end("Zplain"));
1386  }
1387  {
1388  Xapian::TermIterator t = qp.unstem_begin("XTESTbools");
1389  TEST(t != qp.unstem_end("XTESTbools"));
1390  TEST_EQUAL(*t, "test:bools");
1391  ++t;
1392  TEST(t == qp.unstem_end("XTESTbools"));
1393  }
1394  {
1395  Xapian::TermIterator t = qp.unstem_begin("ZXFOOfield");
1396  TEST(t != qp.unstem_end("ZXFOOfield"));
1397  TEST_EQUAL(*t, "fielded");
1398  ++t;
1399  TEST(t == qp.unstem_end("ZXFOOfield"));
1400  }
1401  {
1403  TEST(t != qp.stoplist_end());
1404  TEST_EQUAL(*t, "a");
1405  ++t;
1406  TEST(t == qp.stoplist_end());
1407  }
1408  q = qp.parse_query("the plain foo:fields",
1409  qp.FLAG_DEFAULT | qp.FLAG_ACCUMULATE);
1410  tout << q.get_description() << '\n';
1411  {
1412  Xapian::TermIterator t = qp.unstem_begin("Zplain");
1413  TEST(t != qp.unstem_end("Zplain"));
1414  TEST_EQUAL(*t, "plains");
1415  ++t;
1416  TEST(t != qp.unstem_end("Zplain"));
1417  TEST_EQUAL(*t, "plain");
1418  ++t;
1419  TEST(t == qp.unstem_end("Zplain"));
1420  }
1421  {
1422  Xapian::TermIterator t = qp.unstem_begin("XTESTbools");
1423  TEST(t != qp.unstem_end("XTESTbools"));
1424  TEST_EQUAL(*t, "test:bools");
1425  ++t;
1426  TEST(t == qp.unstem_end("XTESTbools"));
1427  }
1428  {
1429  Xapian::TermIterator t = qp.unstem_begin("ZXFOOfield");
1430  TEST(t != qp.unstem_end("ZXFOOfield"));
1431  TEST_EQUAL(*t, "fielded");
1432  ++t;
1433  TEST(t != qp.unstem_end("ZXFOOfield"));
1434  TEST_EQUAL(*t, "fields");
1435  ++t;
1436  TEST(t == qp.unstem_end("ZXFOOfield"));
1437  }
1438  {
1440  TEST(t != qp.stoplist_end());
1441  TEST_EQUAL(*t, "a");
1442  ++t;
1443  TEST(t != qp.stoplist_end());
1444  TEST_EQUAL(*t, "the");
1445  ++t;
1446  TEST(t == qp.stoplist_end());
1447  }
1448  // Check things are reset without FLAG_ACCUMULATE.
1449  q = qp.parse_query("plains");
1450  tout << q.get_description() << '\n';
1451  {
1452  Xapian::TermIterator t = qp.unstem_begin("Zplain");
1453  TEST(t != qp.unstem_end("Zplain"));
1454  TEST_EQUAL(*t, "plains");
1455  ++t;
1456  TEST(t == qp.unstem_end("Zplain"));
1457  }
1458  {
1459  Xapian::TermIterator t = qp.unstem_begin("XTESTboolean");
1460  TEST(t == qp.unstem_end("XTESTboolean"));
1461  }
1462  {
1463  Xapian::TermIterator t = qp.unstem_begin("ZXFOOfield");
1464  TEST(t == qp.unstem_end("ZXFOOfield"));
1465  }
1466  {
1468  TEST(t == qp.stoplist_end());
1469  }
1470 }
1471 
1472 static const test test_value_range1_queries[] = {
1473  { "a..b", "VALUE_RANGE 1 a b" },
1474  { "$50..100", "VALUE_RANGE 1 $50 100" },
1475  { "$50..$99", "VALUE_RANGE 1 $50 $99" },
1476  { "$50..$100", "" }, // start > range
1477  { "02/03/1979..10/12/1980", "VALUE_RANGE 1 02/03/1979 10/12/1980" },
1478  { "a..b hello", "(hello@1 FILTER VALUE_RANGE 1 a b)" },
1479  { "hello a..b", "(hello@1 FILTER VALUE_RANGE 1 a b)" },
1480  { "hello a..b world", "((hello@1 OR world@2) FILTER VALUE_RANGE 1 a b)" },
1481  { "hello a..b test:foo", "(hello@1 FILTER (VALUE_RANGE 1 a b AND XTESTfoo))" },
1482  { "hello a..b test:foo test:bar", "(hello@1 FILTER (VALUE_RANGE 1 a b AND (XTESTfoo OR XTESTbar)))" },
1483  { "hello a..b c..d test:foo", "(hello@1 FILTER ((VALUE_RANGE 1 a b OR VALUE_RANGE 1 c d) AND XTESTfoo))" },
1484  { "hello a..b c..d test:foo test:bar", "(hello@1 FILTER ((VALUE_RANGE 1 a b OR VALUE_RANGE 1 c d) AND (XTESTfoo OR XTESTbar)))" },
1485  { "-5..7", "VALUE_RANGE 1 -5 7" },
1486  { "hello -5..7", "(hello@1 FILTER VALUE_RANGE 1 -5 7)" },
1487  { "-5..7 hello", "(hello@1 FILTER VALUE_RANGE 1 -5 7)" },
1488  { "\"time flies\" 09:00..12:30", "((time@1 PHRASE 2 flies@2) FILTER VALUE_RANGE 1 09:00 12:30)" },
1489  // Feature test for single-ended ranges (ticket#480):
1490  { "..b", "VALUE_LE 1 b" },
1491  { "a..", "VALUE_GE 1 a" },
1492  // Test for expanded set of characters allowed in range start:
1493  { "10:30+1300..11:00+1300", "VALUE_RANGE 1 10:30+1300 11:00+1300" },
1494  { NULL, NULL }
1495 };
1496 
1497 // Simple test of ValueRangeProcessor class.
1498 DEFINE_TESTCASE(qp_value_range1, !backend) {
1500  qp.add_boolean_prefix("test", "XTEST");
1502  qp.add_valuerangeprocessor(&vrp);
1503  for (const test *p = test_value_range1_queries; p->query; ++p) {
1504  string expect, parsed;
1505  if (p->expect)
1506  expect = p->expect;
1507  else
1508  expect = "parse error";
1509  try {
1510  Xapian::Query qobj = qp.parse_query(p->query);
1511  parsed = qobj.get_description();
1512  expect = string("Query(") + expect + ')';
1513  } catch (const Xapian::QueryParserError &e) {
1514  parsed = e.get_msg();
1515  } catch (const Xapian::Error &e) {
1516  parsed = e.get_description();
1517  } catch (...) {
1518  parsed = "Unknown exception!";
1519  }
1520  tout << "Query: " << p->query << '\n';
1521  TEST_STRINGS_EQUAL(parsed, expect);
1522  }
1523 }
1524 
1525 // Simple test of RangeProcessor class.
1526 DEFINE_TESTCASE(qp_range1, !backend) {
1528  qp.add_boolean_prefix("test", "XTEST");
1529  Xapian::RangeProcessor rp(1);
1530  qp.add_rangeprocessor(&rp);
1531  for (const test *p = test_value_range1_queries; p->query; ++p) {
1532  string expect, parsed;
1533  if (p->expect)
1534  expect = p->expect;
1535  else
1536  expect = "parse error";
1537  try {
1538  Xapian::Query qobj = qp.parse_query(p->query);
1539  parsed = qobj.get_description();
1540  expect = string("Query(") + expect + ')';
1541  } catch (const Xapian::QueryParserError &e) {
1542  parsed = e.get_msg();
1543  } catch (const Xapian::Error &e) {
1544  parsed = e.get_description();
1545  } catch (...) {
1546  parsed = "Unknown exception!";
1547  }
1548  tout << "Query: " << p->query << '\n';
1549  TEST_STRINGS_EQUAL(parsed, expect);
1550  }
1551 }
1552 
1553 static const test test_value_range2_queries[] = {
1554  { "a..b", "VALUE_RANGE 3 a b" },
1555  { "1..12", "VALUE_RANGE 2 \\xa0 \\xae" },
1556  { "20070201..20070228", "VALUE_RANGE 1 20070201 20070228" },
1557  { "$10..20", "VALUE_RANGE 4 \\xad \\xb1" },
1558  { "$10..$20", "VALUE_RANGE 4 \\xad \\xb1" },
1559  // Feature test for single-ended ranges (ticket#480):
1560  { "$..20", "VALUE_LE 4 \\xb1" },
1561  { "..$20", "VALUE_LE 3 $20" }, // FIXME: probably should parse as $..20
1562  { "$10..", "VALUE_GE 4 \\xad" },
1563  { "12..42kg", "VALUE_RANGE 5 \\xae \\xb5@" },
1564  { "12kg..42kg", "VALUE_RANGE 5 \\xae \\xb5@" },
1565  { "12kg..42", "VALUE_RANGE 3 12kg 42" },
1566  { "10..$20", "" }, // start > end
1567  { "1999-03-12..2020-12-30", "VALUE_RANGE 1 19990312 20201230" },
1568  { "1999/03/12..2020/12/30", "VALUE_RANGE 1 19990312 20201230" },
1569  { "1999.03.12..2020.12.30", "VALUE_RANGE 1 19990312 20201230" },
1570  // Feature test for single-ended ranges (ticket#480):
1571  { "..2020.12.30", "VALUE_LE 1 20201230" },
1572  { "1999.03.12..", "VALUE_GE 1 19990312" },
1573  { "12/03/99..12/04/01", "VALUE_RANGE 1 19990312 20010412" },
1574  { "03-12-99..04-14-01", "VALUE_RANGE 1 19990312 20010414" },
1575  { "1/2/3..2/3/4", "VALUE_RANGE 1 20030201 20040302" },
1576  { "(test:a..test:b hello)", "(hello@1 FILTER VALUE_RANGE 3 test:a test:b)" },
1577  { "12..42kg 5..6kg 1..12", "0 * (VALUE_RANGE 2 \\xa0 \\xae AND (VALUE_RANGE 5 \\xae \\xb5@ OR VALUE_RANGE 5 \\xa9 \\xaa))" },
1578  // Check that a VRP which fails to match doesn't remove a prefix or suffix.
1579  // 1.0.13/1.1.1 and earlier got this wrong in some cases.
1580  { "$12a..13", "VALUE_RANGE 3 $12a 13" },
1581  { "$12..13b", "VALUE_RANGE 3 $12 13b" },
1582  { "$12..12kg", "VALUE_RANGE 3 $12 12kg" },
1583  { "12..b12kg", "VALUE_RANGE 3 12 b12kg" },
1584  { NULL, NULL }
1585 };
1586 
1587 // Test chaining of ValueRangeProcessor classes.
1588 DEFINE_TESTCASE(qp_value_range2, !backend) {
1590  qp.add_boolean_prefix("test", "XTEST");
1591  Xapian::DateValueRangeProcessor vrp_date(1);
1594  Xapian::NumberValueRangeProcessor vrp_cash(4, "$");
1595  Xapian::NumberValueRangeProcessor vrp_weight(5, "kg", false);
1596  qp.add_valuerangeprocessor(&vrp_date);
1597  qp.add_valuerangeprocessor(&vrp_num);
1598  qp.add_valuerangeprocessor(&vrp_cash);
1599  qp.add_valuerangeprocessor(&vrp_weight);
1600  qp.add_valuerangeprocessor(&vrp_str);
1601  for (const test *p = test_value_range2_queries; p->query; ++p) {
1602  string expect, parsed;
1603  if (p->expect)
1604  expect = p->expect;
1605  else
1606  expect = "parse error";
1607  try {
1608  Xapian::Query qobj = qp.parse_query(p->query);
1609  parsed = qobj.get_description();
1610  expect = string("Query(") + expect + ')';
1611  } catch (const Xapian::QueryParserError &e) {
1612  parsed = e.get_msg();
1613  } catch (const Xapian::Error &e) {
1614  parsed = e.get_description();
1615  } catch (...) {
1616  parsed = "Unknown exception!";
1617  }
1618  tout << "Query: " << p->query << '\n';
1619  TEST_STRINGS_EQUAL(parsed, expect);
1620  }
1621 }
1622 
1623 // Test chaining of RangeProcessor classes.
1624 DEFINE_TESTCASE(qp_range2, !backend) {
1625  using Xapian::RP_REPEATED;
1626  using Xapian::RP_SUFFIX;
1628  qp.add_boolean_prefix("test", "XTEST");
1629  Xapian::DateRangeProcessor rp_date(1);
1630  Xapian::NumberRangeProcessor rp_num(2);
1631  Xapian::RangeProcessor rp_str(3);
1632  Xapian::NumberRangeProcessor rp_cash(4, "$", RP_REPEATED);
1634  qp.add_rangeprocessor(&rp_date);
1635  qp.add_rangeprocessor(&rp_num);
1636  qp.add_rangeprocessor(&rp_cash);
1637  qp.add_rangeprocessor(&rp_weight);
1638  qp.add_rangeprocessor(&rp_str);
1639  for (const test *p = test_value_range2_queries; p->query; ++p) {
1640  string expect, parsed;
1641  if (p->expect)
1642  expect = p->expect;
1643  else
1644  expect = "parse error";
1645  try {
1646  Xapian::Query qobj = qp.parse_query(p->query);
1647  parsed = qobj.get_description();
1648  expect = string("Query(") + expect + ')';
1649  } catch (const Xapian::QueryParserError &e) {
1650  parsed = e.get_msg();
1651  } catch (const Xapian::Error &e) {
1652  parsed = e.get_description();
1653  } catch (...) {
1654  parsed = "Unknown exception!";
1655  }
1656  tout << "Query: " << p->query << '\n';
1657  TEST_STRINGS_EQUAL(parsed, expect);
1658  }
1659 }
1660 
1661 // Test NumberValueRangeProcessors with actual data.
1662 DEFINE_TESTCASE(qp_value_range3, writable) {
1664  double low = -10;
1665  int steps = 60;
1666  double step = 0.5;
1667 
1668  for (int i = 0; i <= steps; ++i) {
1669  double v = low + i * step;
1670  Xapian::Document doc;
1672  db.add_document(doc);
1673  }
1674 
1677  qp.add_valuerangeprocessor(&vrp_num);
1678 
1679  for (int j = 0; j <= steps; ++j) {
1680  double start = low + j * step;
1681  for (int k = 0; k <= steps; ++k) {
1682  double end = low + k * step;
1683  string query = str(start) + ".." + str(end);
1684  tout << "Query: " << query << '\n';
1685  Xapian::Query qobj = qp.parse_query(query);
1686  Xapian::Enquire enq(db);
1687  enq.set_query(qobj);
1688  Xapian::MSet mset = enq.get_mset(0, steps + 1);
1689  if (end < start) {
1690  TEST_EQUAL(mset.size(), 0);
1691  } else {
1692  TEST_EQUAL(mset.size(), 1u + (k - j));
1693  for (unsigned int m = 0; m != mset.size(); ++m) {
1694  double v = start + m * step;
1695  TEST_EQUAL(mset[m].get_document().get_value(1),
1697  }
1698  }
1699  }
1700  }
1701 }
1702 
1703 // Test NumberRangeProcessors with actual data.
1704 DEFINE_TESTCASE(qp_range3, writable) {
1706  double low = -10;
1707  int steps = 60;
1708  double step = 0.5;
1709 
1710  for (int i = 0; i <= steps; ++i) {
1711  double v = low + i * step;
1712  Xapian::Document doc;
1714  db.add_document(doc);
1715  }
1716 
1717  Xapian::NumberRangeProcessor rp_num(1);
1719  qp.add_rangeprocessor(&rp_num);
1720 
1721  for (int j = 0; j <= steps; ++j) {
1722  double start = low + j * step;
1723  for (int k = 0; k <= steps; ++k) {
1724  double end = low + k * step;
1725  string query = str(start) + ".." + str(end);
1726  tout << "Query: " << query << '\n';
1727  Xapian::Query qobj = qp.parse_query(query);
1728  Xapian::Enquire enq(db);
1729  enq.set_query(qobj);
1730  Xapian::MSet mset = enq.get_mset(0, steps + 1);
1731  if (end < start) {
1732  TEST_EQUAL(mset.size(), 0);
1733  } else {
1734  TEST_EQUAL(mset.size(), 1u + (k - j));
1735  for (unsigned int m = 0; m != mset.size(); ++m) {
1736  double v = start + m * step;
1737  TEST_EQUAL(mset[m].get_document().get_value(1),
1739  }
1740  }
1741  }
1742  }
1743 }
1744 
1745 static const test test_value_range4_queries[] = {
1746  { "id:19254@foo..example.com", "0 * Q19254@foo..example.com" },
1747  { "hello:world", "0 * XHELLOworld" },
1748  { "hello:mum..world", "VALUE_RANGE 1 mum world" },
1749  { NULL, NULL }
1750 };
1751 
1758 DEFINE_TESTCASE(qp_value_range4, !backend) {
1760  qp.add_boolean_prefix("id", "Q");
1761  qp.add_boolean_prefix("hello", "XHELLO");
1762  Xapian::StringValueRangeProcessor vrp_str(1, "hello:");
1763  qp.add_valuerangeprocessor(&vrp_str);
1764  for (const test *p = test_value_range4_queries; p->query; ++p) {
1765  string expect, parsed;
1766  if (p->expect)
1767  expect = p->expect;
1768  else
1769  expect = "parse error";
1770  try {
1771  Xapian::Query qobj = qp.parse_query(p->query);
1772  parsed = qobj.get_description();
1773  expect = string("Query(") + expect + ')';
1774  } catch (const Xapian::QueryParserError &e) {
1775  parsed = e.get_msg();
1776  } catch (const Xapian::Error &e) {
1777  parsed = e.get_description();
1778  } catch (...) {
1779  parsed = "Unknown exception!";
1780  }
1781  tout << "Query: " << p->query << '\n';
1782  TEST_STRINGS_EQUAL(parsed, expect);
1783  }
1784 }
1785 
1792 DEFINE_TESTCASE(qp_range4, !backend) {
1794  qp.add_boolean_prefix("id", "Q");
1795  qp.add_boolean_prefix("hello", "XHELLO");
1796  Xapian::RangeProcessor rp_str(1, "hello:");
1797  qp.add_rangeprocessor(&rp_str);
1798  for (const test *p = test_value_range4_queries; p->query; ++p) {
1799  string expect, parsed;
1800  if (p->expect)
1801  expect = p->expect;
1802  else
1803  expect = "parse error";
1804  try {
1805  Xapian::Query qobj = qp.parse_query(p->query);
1806  parsed = qobj.get_description();
1807  expect = string("Query(") + expect + ')';
1808  } catch (const Xapian::QueryParserError &e) {
1809  parsed = e.get_msg();
1810  } catch (const Xapian::Error &e) {
1811  parsed = e.get_description();
1812  } catch (...) {
1813  parsed = "Unknown exception!";
1814  }
1815  tout << "Query: " << p->query << '\n';
1816  TEST_STRINGS_EQUAL(parsed, expect);
1817  }
1818 }
1819 
1820 static const test test_value_daterange1_queries[] = {
1821  { "12/03/99..12/04/01", "VALUE_RANGE 1 19991203 20011204" },
1822  { "03-12-99..04-14-01", "VALUE_RANGE 1 19990312 20010414" },
1823  { "01/30/60..02/02/59", "VALUE_RANGE 1 19600130 20590202" },
1824  { "1999-03-12..2001-04-14", "VALUE_RANGE 1 19990312 20010414" },
1825  { "12/03/99..02", "Unknown range operation" },
1826  { "1999-03-12..2001", "Unknown range operation" },
1827  { NULL, NULL }
1828 };
1829 
1830 // Test DateValueRangeProcessor
1831 DEFINE_TESTCASE(qp_value_daterange1, !backend) {
1833  Xapian::DateValueRangeProcessor vrp_date(1, true, 1960);
1834  qp.add_valuerangeprocessor(&vrp_date);
1835  for (const test *p = test_value_daterange1_queries; p->query; ++p) {
1836  string expect, parsed;
1837  if (p->expect)
1838  expect = p->expect;
1839  else
1840  expect = "parse error";
1841  try {
1842  Xapian::Query qobj = qp.parse_query(p->query);
1843  parsed = qobj.get_description();
1844  expect = string("Query(") + expect + ')';
1845  } catch (const Xapian::QueryParserError &e) {
1846  parsed = e.get_msg();
1847  } catch (const Xapian::Error &e) {
1848  parsed = e.get_description();
1849  } catch (...) {
1850  parsed = "Unknown exception!";
1851  }
1852  tout << "Query: " << p->query << '\n';
1853  TEST_STRINGS_EQUAL(parsed, expect);
1854  }
1855 }
1856 
1857 // Test DateRangeProcessor
1858 DEFINE_TESTCASE(qp_daterange1, !backend) {
1861  qp.add_rangeprocessor(&rp_date);
1862  for (const test *p = test_value_daterange1_queries; p->query; ++p) {
1863  string expect, parsed;
1864  if (p->expect)
1865  expect = p->expect;
1866  else
1867  expect = "parse error";
1868  try {
1869  Xapian::Query qobj = qp.parse_query(p->query);
1870  parsed = qobj.get_description();
1871  expect = string("Query(") + expect + ')';
1872  } catch (const Xapian::QueryParserError &e) {
1873  parsed = e.get_msg();
1874  } catch (const Xapian::Error &e) {
1875  parsed = e.get_description();
1876  } catch (...) {
1877  parsed = "Unknown exception!";
1878  }
1879  tout << "Query: " << p->query << '\n';
1880  TEST_STRINGS_EQUAL(parsed, expect);
1881  }
1882 }
1883 
1884 static const test test_value_daterange2_queries[] = {
1885  { "created:12/03/99..12/04/01", "VALUE_RANGE 1 19991203 20011204" },
1886  { "modified:03-12-99..04-14-01", "VALUE_RANGE 2 19990312 20010414" },
1887  { "accessed:01/30/70..02/02/69", "VALUE_RANGE 3 19700130 20690202" },
1888  // In <=1.2.12, and in 1.3.0, this gave "Unknown range operation":
1889  { "deleted:12/03/99..12/04/01", "VALUE_RANGE 4 19990312 20010412" },
1890  { "1999-03-12..2001-04-14", "Unknown range operation" },
1891  { "12/03/99..created:12/04/01", "Unknown range operation" },
1892  { "12/03/99created:..12/04/01", "Unknown range operation" },
1893  { "12/03/99..12/04/01created:", "Unknown range operation" },
1894  { "12/03/99..02", "Unknown range operation" },
1895  { "1999-03-12..2001", "Unknown range operation" },
1896  { NULL, NULL }
1897 };
1898 
1899 // Feature test DateValueRangeProcessor with prefixes (added in 1.1.2).
1900 DEFINE_TESTCASE(qp_value_daterange2, !backend) {
1902  Xapian::DateValueRangeProcessor vrp_cdate(1, "created:", true, true, 1970);
1903  Xapian::DateValueRangeProcessor vrp_mdate(2, "modified:", true, true, 1970);
1904  Xapian::DateValueRangeProcessor vrp_adate(3, "accessed:", true, true, 1970);
1905  // Regression test - here a const char * was taken as a bool rather than a
1906  // std::string when resolving the overloaded forms. Fixed in 1.2.13 and
1907  // 1.3.1.
1908  Xapian::DateValueRangeProcessor vrp_ddate(4, "deleted:");
1909  qp.add_valuerangeprocessor(&vrp_cdate);
1910  qp.add_valuerangeprocessor(&vrp_mdate);
1911  qp.add_valuerangeprocessor(&vrp_adate);
1912  qp.add_valuerangeprocessor(&vrp_ddate);
1913  for (const test *p = test_value_daterange2_queries; p->query; ++p) {
1914  string expect, parsed;
1915  if (p->expect)
1916  expect = p->expect;
1917  else
1918  expect = "parse error";
1919  try {
1920  Xapian::Query qobj = qp.parse_query(p->query);
1921  parsed = qobj.get_description();
1922  expect = string("Query(") + expect + ')';
1923  } catch (const Xapian::QueryParserError &e) {
1924  parsed = e.get_msg();
1925  } catch (const Xapian::Error &e) {
1926  parsed = e.get_description();
1927  } catch (...) {
1928  parsed = "Unknown exception!";
1929  }
1930  tout << "Query: " << p->query << '\n';
1931  TEST_STRINGS_EQUAL(parsed, expect);
1932  }
1933 }
1934 
1935 // Feature test DateRangeProcessor with prefixes (added in 1.1.2).
1936 DEFINE_TESTCASE(qp_daterange2, !backend) {
1939  Xapian::DateRangeProcessor rp_cdate(1, "created:", RP_DATE_PREFER_MDY, 1970);
1940  Xapian::DateRangeProcessor rp_mdate(2, "modified:", RP_DATE_PREFER_MDY, 1970);
1941  Xapian::DateRangeProcessor rp_adate(3, "accessed:", RP_DATE_PREFER_MDY, 1970);
1942  // Regression test - here a const char * was taken as a bool rather than a
1943  // std::string when resolving the overloaded forms. Fixed in 1.2.13 and
1944  // 1.3.1.
1945  Xapian::DateRangeProcessor rp_ddate(4, "deleted:");
1946  qp.add_rangeprocessor(&rp_cdate);
1947  qp.add_rangeprocessor(&rp_mdate);
1948  qp.add_rangeprocessor(&rp_adate);
1949  qp.add_rangeprocessor(&rp_ddate);
1950  for (const test *p = test_value_daterange2_queries; p->query; ++p) {
1951  string expect, parsed;
1952  if (p->expect)
1953  expect = p->expect;
1954  else
1955  expect = "parse error";
1956  try {
1957  Xapian::Query qobj = qp.parse_query(p->query);
1958  parsed = qobj.get_description();
1959  expect = string("Query(") + expect + ')';
1960  } catch (const Xapian::QueryParserError &e) {
1961  parsed = e.get_msg();
1962  } catch (const Xapian::Error &e) {
1963  parsed = e.get_description();
1964  } catch (...) {
1965  parsed = "Unknown exception!";
1966  }
1967  tout << "Query: " << p->query << '\n';
1968  TEST_STRINGS_EQUAL(parsed, expect);
1969  }
1970 }
1971 
1972 static const test test_value_stringrange1_queries[] = {
1973  { "tag:bar..foo", "VALUE_RANGE 1 bar foo" },
1974  { "bar..foo", "VALUE_RANGE 0 bar foo" },
1975  { NULL, NULL }
1976 };
1977 
1978 // Feature test StringValueRangeProcessor with prefixes (added in 1.1.2).
1979 DEFINE_TESTCASE(qp_value_stringrange1, !backend) {
1981  Xapian::StringValueRangeProcessor vrp_default(0);
1982  Xapian::StringValueRangeProcessor vrp_tag(1, "tag:", true);
1983  qp.add_valuerangeprocessor(&vrp_tag);
1984  qp.add_valuerangeprocessor(&vrp_default);
1985  for (const test *p = test_value_stringrange1_queries; p->query; ++p) {
1986  string expect, parsed;
1987  if (p->expect)
1988  expect = p->expect;
1989  else
1990  expect = "parse error";
1991  try {
1992  Xapian::Query qobj = qp.parse_query(p->query);
1993  parsed = qobj.get_description();
1994  expect = string("Query(") + expect + ')';
1995  } catch (const Xapian::QueryParserError &e) {
1996  parsed = e.get_msg();
1997  } catch (const Xapian::Error &e) {
1998  parsed = e.get_description();
1999  } catch (...) {
2000  parsed = "Unknown exception!";
2001  }
2002  tout << "Query: " << p->query << '\n';
2003  TEST_STRINGS_EQUAL(parsed, expect);
2004  }
2005 }
2006 
2007 // Feature test RangeProcessor with prefixes.
2008 DEFINE_TESTCASE(qp_stringrange1, !backend) {
2010  Xapian::RangeProcessor rp_default(0);
2011  Xapian::RangeProcessor rp_tag(1, "tag:");
2012  qp.add_rangeprocessor(&rp_tag);
2013  qp.add_rangeprocessor(&rp_default);
2014  for (const test *p = test_value_stringrange1_queries; p->query; ++p) {
2015  string expect, parsed;
2016  if (p->expect)
2017  expect = p->expect;
2018  else
2019  expect = "parse error";
2020  try {
2021  Xapian::Query qobj = qp.parse_query(p->query);
2022  parsed = qobj.get_description();
2023  expect = string("Query(") + expect + ')';
2024  } catch (const Xapian::QueryParserError &e) {
2025  parsed = e.get_msg();
2026  } catch (const Xapian::Error &e) {
2027  parsed = e.get_description();
2028  } catch (...) {
2029  parsed = "Unknown exception!";
2030  }
2031  tout << "Query: " << p->query << '\n';
2032  TEST_STRINGS_EQUAL(parsed, expect);
2033  }
2034 }
2035 
2036 static const test test_value_customrange1_queries[] = {
2037  { "mars author:Asimov..Bradbury", "(mars@1 FILTER VALUE_RANGE 4 asimov bradbury)" },
2038  { NULL, NULL }
2039 };
2040 
2043 
2044  Xapian::valueno operator()(std::string &begin, std::string &end) {
2045  if (!startswith(begin, "author:"))
2046  return Xapian::BAD_VALUENO;
2047  begin.erase(0, 7);
2048  begin = Xapian::Unicode::tolower(begin);
2049  end = Xapian::Unicode::tolower(end);
2050  return 4;
2051  }
2052 };
2053 
2054 // Test custom ValueRangeProcessor subclass.
2055 DEFINE_TESTCASE(qp_value_customrange1, !backend) {
2057  AuthorValueRangeProcessor vrp_author;
2058  qp.add_valuerangeprocessor(&vrp_author);
2059  for (const test *p = test_value_customrange1_queries; p->query; ++p) {
2060  string expect, parsed;
2061  if (p->expect)
2062  expect = p->expect;
2063  else
2064  expect = "parse error";
2065  try {
2066  Xapian::Query qobj = qp.parse_query(p->query);
2067  parsed = qobj.get_description();
2068  expect = string("Query(") + expect + ')';
2069  } catch (const Xapian::QueryParserError &e) {
2070  parsed = e.get_msg();
2071  } catch (const Xapian::Error &e) {
2072  parsed = e.get_description();
2073  } catch (...) {
2074  parsed = "Unknown exception!";
2075  }
2076  tout << "Query: " << p->query << '\n';
2077  TEST_STRINGS_EQUAL(parsed, expect);
2078  }
2079 }
2080 
2082  AuthorRangeProcessor() : Xapian::RangeProcessor(4, "author:") { }
2083 
2084  Xapian::Query operator()(const std::string& b, const std::string& e)
2085  {
2086  string begin = Xapian::Unicode::tolower(b);
2087  string end = Xapian::Unicode::tolower(e);
2088  return Xapian::RangeProcessor::operator()(begin, end);
2089  }
2090 };
2091 
2092 // Test custom RangeProcessor subclass.
2093 DEFINE_TESTCASE(qp_customrange1, !backend) {
2095  AuthorRangeProcessor rp_author;
2096  qp.add_rangeprocessor(&rp_author);
2097  for (const test *p = test_value_customrange1_queries; p->query; ++p) {
2098  string expect, parsed;
2099  if (p->expect)
2100  expect = p->expect;
2101  else
2102  expect = "parse error";
2103  try {
2104  Xapian::Query qobj = qp.parse_query(p->query);
2105  parsed = qobj.get_description();
2106  expect = string("Query(") + expect + ')';
2107  } catch (const Xapian::QueryParserError &e) {
2108  parsed = e.get_msg();
2109  } catch (const Xapian::Error &e) {
2110  parsed = e.get_description();
2111  } catch (...) {
2112  parsed = "Unknown exception!";
2113  }
2114  tout << "Query: " << p->query << '\n';
2115  TEST_STRINGS_EQUAL(parsed, expect);
2116  }
2117 }
2118 
2120  Xapian::Query operator()(const std::string & str) {
2121  if (str == "all")
2122  return Xapian::Query::MatchAll;
2123  return Xapian::Query("S" + str);
2124  }
2125 };
2126 
2128  Xapian::Query operator()(const std::string & str) {
2129  if (str == "*")
2130  return Xapian::Query::MatchAll;
2131  string res = "H";
2132  for (string::const_iterator i = str.begin(); i != str.end(); ++i)
2133  res += C_tolower(*i);
2134  return Xapian::Query(res);
2135  }
2136 };
2137 
2138 static const test test_fieldproc1_queries[] = {
2139  { "title:test", "Stest" },
2140  { "title:all", "<alldocuments>" },
2141  { "host:Xapian.org", "0 * Hxapian.org" },
2142  { "host:*", "0 * <alldocuments>" },
2143  { "host:\"Space Station.Example.Org\"", "0 * Hspace station.example.org" },
2144  { NULL, NULL }
2145 };
2146 
2147 // FieldProcessor test.
2148 DEFINE_TESTCASE(qp_fieldproc1, !backend) {
2150  TitleFieldProcessor title_fproc;
2151  HostFieldProcessor host_fproc;
2152  qp.add_prefix("title", &title_fproc);
2153  qp.add_boolean_prefix("host", &host_fproc);
2154  for (const test *p = test_fieldproc1_queries; p->query; ++p) {
2155  string expect, parsed;
2156  if (p->expect)
2157  expect = p->expect;
2158  else
2159  expect = "parse error";
2160  try {
2161  Xapian::Query qobj = qp.parse_query(p->query);
2162  parsed = qobj.get_description();
2163  expect = string("Query(") + expect + ')';
2164  } catch (const Xapian::QueryParserError &e) {
2165  parsed = e.get_msg();
2166  } catch (const Xapian::Error &e) {
2167  parsed = e.get_description();
2168  } catch (...) {
2169  parsed = "Unknown exception!";
2170  }
2171  tout << "Query: " << p->query << '\n';
2172  TEST_STRINGS_EQUAL(parsed, expect);
2173  }
2174 }
2175 
2177  Xapian::Query operator()(const std::string & str) {
2178  // In reality, these would be built from the current date, but for
2179  // testing it is much simpler to fix the date.
2180  if (str == "today")
2181  return Xapian::Query(Xapian::Query::OP_VALUE_GE, 1, "20120725");
2182  if (str == "this week")
2183  return Xapian::Query(Xapian::Query::OP_VALUE_GE, 1, "20120723");
2184  if (str == "this month")
2185  return Xapian::Query(Xapian::Query::OP_VALUE_GE, 1, "20120701");
2186  if (str == "this year")
2187  return Xapian::Query(Xapian::Query::OP_VALUE_GE, 1, "20120101");
2188  if (str == "this decade")
2189  return Xapian::Query(Xapian::Query::OP_VALUE_GE, 1, "20100101");
2190  if (str == "this century")
2191  return Xapian::Query(Xapian::Query::OP_VALUE_GE, 1, "20000101");
2192  throw Xapian::QueryParserError("Didn't understand date specification '" + str + "'");
2193  }
2194 };
2195 
2196 static const test test_fieldproc2_queries[] = {
2197  { "date:\"this week\"", "VALUE_GE 1 20120723" },
2198  { "date:23/7/2012..25/7/2012", "VALUE_RANGE 1 20120723 20120725" },
2199  { NULL, NULL }
2200 };
2201 
2202 // Test using FieldProcessor and ValueRangeProcessor together.
2203 DEFINE_TESTCASE(qp_fieldproc2, !backend) {
2205  DateRangeFieldProcessor date_fproc;
2206  qp.add_boolean_prefix("date", &date_fproc);
2207  Xapian::DateValueRangeProcessor vrp_date(1, "date:");
2208  qp.add_valuerangeprocessor(&vrp_date);
2209  for (const test *p = test_fieldproc2_queries; p->query; ++p) {
2210  string expect, parsed;
2211  if (p->expect)
2212  expect = p->expect;
2213  else
2214  expect = "parse error";
2215  try {
2216  Xapian::Query qobj = qp.parse_query(p->query);
2217  parsed = qobj.get_description();
2218  expect = string("Query(") + expect + ')';
2219  } catch (const Xapian::QueryParserError &e) {
2220  parsed = e.get_msg();
2221  } catch (const Xapian::Error &e) {
2222  parsed = e.get_description();
2223  } catch (...) {
2224  parsed = "Unknown exception!";
2225  }
2226  tout << "Query: " << p->query << '\n';
2227  TEST_STRINGS_EQUAL(parsed, expect);
2228  }
2229 }
2230 
2231 // Test using FieldProcessor and RangeProcessor together.
2232 DEFINE_TESTCASE(qp_fieldproc3, !backend) {
2234  DateRangeFieldProcessor date_fproc;
2235  qp.add_boolean_prefix("date", &date_fproc);
2236  Xapian::DateRangeProcessor rp_date(1, "date:");
2237  qp.add_rangeprocessor(&rp_date);
2238  for (const test *p = test_fieldproc2_queries; p->query; ++p) {
2239  string expect, parsed;
2240  if (p->expect)
2241  expect = p->expect;
2242  else
2243  expect = "parse error";
2244  try {
2245  Xapian::Query qobj = qp.parse_query(p->query);
2246  parsed = qobj.get_description();
2247  expect = string("Query(") + expect + ')';
2248  } catch (const Xapian::QueryParserError &e) {
2249  parsed = e.get_msg();
2250  } catch (const Xapian::Error &e) {
2251  parsed = e.get_description();
2252  } catch (...) {
2253  parsed = "Unknown exception!";
2254  }
2255  tout << "Query: " << p->query << '\n';
2256  TEST_STRINGS_EQUAL(parsed, expect);
2257  }
2258 }
2259 
2260 DEFINE_TESTCASE(qp_stoplist1, !backend) {
2262  static const char * const stopwords[] = { "a", "an", "the" };
2263  Xapian::SimpleStopper stop(stopwords, stopwords + 3);
2264  qp.set_stopper(&stop);
2265 
2267 
2268  Xapian::Query query1 = qp.parse_query("some mice");
2269  i = qp.stoplist_begin();
2270  TEST(i == qp.stoplist_end());
2271 
2272  Xapian::Query query2 = qp.parse_query("the cat");
2273  i = qp.stoplist_begin();
2274  TEST(i != qp.stoplist_end());
2275  TEST_EQUAL(*i, "the");
2276  ++i;
2277  TEST(i == qp.stoplist_end());
2278 
2279  // Regression test - prior to Xapian 1.0.0 the stoplist wasn't being cleared
2280  // when a new query was parsed.
2281  Xapian::Query query3 = qp.parse_query("an aardvark");
2282  i = qp.stoplist_begin();
2283  TEST(i != qp.stoplist_end());
2284  TEST_EQUAL(*i, "an");
2285  ++i;
2286  TEST(i == qp.stoplist_end());
2287 }
2288 
2289 static const test test_mispelled_queries[] = {
2290  { "doucment search", "document search" },
2291  { "doucment seeacrh", "document search" },
2292  { "docment seeacrh test", "document search test" },
2293  { "\"paragahp pineapple\"", "\"paragraph pineapple\"" },
2294  { "\"paragahp pineapple\"", "\"paragraph pineapple\"" },
2295  { "test S.E.A.R.C.", "" },
2296  { "this AND that", "" },
2297  { "documento", "document" },
2298  { "documento-documento", "document-document" },
2299  { "documento-searcho", "document-search" },
2300  { "test saerch", "test search" },
2301  { "paragraf search", "paragraph search" },
2302  { NULL, NULL }
2303 };
2304 
2305 // Test spelling correction in the QueryParser.
2306 DEFINE_TESTCASE(qp_spell1, spelling) {
2308 
2309  Xapian::Document doc;
2310  doc.add_term("document", 6);
2311  doc.add_term("search", 7);
2312  doc.add_term("saerch", 1);
2313  doc.add_term("paragraph", 8);
2314  doc.add_term("paragraf", 2);
2315  db.add_document(doc);
2316 
2317  db.add_spelling("document");
2318  db.add_spelling("search");
2319  db.add_spelling("paragraph");
2320  db.add_spelling("band");
2321 
2324  qp.set_database(db);
2325 
2326  for (const test *p = test_mispelled_queries; p->query; ++p) {
2327  Xapian::Query q;
2328  q = qp.parse_query(p->query,
2331  tout << "Query: " << p->query << endl;
2333  }
2334 }
2335 
2336 // Test spelling correction in the QueryParser with multiple databases.
2337 DEFINE_TESTCASE(qp_spell2, spelling)
2338 {
2340 
2341  db1.add_spelling("document");
2342  db1.add_spelling("search");
2343  db1.commit();
2344 
2346 
2347  db2.add_spelling("document");
2348  db2.add_spelling("paragraph");
2349  db2.add_spelling("band");
2350 
2351  Xapian::Database db;
2352  db.add_database(db1);
2353  db.add_database(db2);
2354 
2357  qp.set_database(db);
2358 
2359  for (const test *p = test_mispelled_queries; p->query; ++p) {
2360  Xapian::Query q;
2361  q = qp.parse_query(p->query,
2364  tout << "Query: " << p->query << endl;
2366  }
2367 }
2368 
2369 static const test test_mispelled_wildcard_queries[] = {
2370  { "doucment", "document" },
2371  { "doucment*", "" },
2372  { "doucment* seearch", "doucment* search" },
2373  { "doucment* search", "" },
2374  { NULL, NULL }
2375 };
2376 
2377 // Test spelling correction in the QueryParser with wildcards.
2378 // Regression test for bug fixed in 1.1.3 and 1.0.17.
2379 DEFINE_TESTCASE(qp_spellwild1, spelling) {
2381 
2382  db.add_spelling("document");
2383  db.add_spelling("search");
2384  db.add_spelling("paragraph");
2385  db.add_spelling("band");
2386 
2388  qp.set_database(db);
2389 
2390  const test *p;
2391  for (p = test_mispelled_queries; p->query; ++p) {
2392  Xapian::Query q;
2393  q = qp.parse_query(p->query,
2397  tout << "Query: " << p->query << endl;
2399  }
2400  for (p = test_mispelled_wildcard_queries; p->query; ++p) {
2401  Xapian::Query q;
2402  q = qp.parse_query(p->query,
2406  tout << "Query: " << p->query << endl;
2408  }
2409 }
2410 
2411 static const test test_mispelled_partial_queries[] = {
2412  { "doucment", "" },
2413  { "doucment ", "document " },
2414  { "documen", "" },
2415  { "documen ", "document " },
2416  { "seearch documen", "search documen" },
2417  { "search documen", "" },
2418  { NULL, NULL }
2419 };
2420 
2421 // Test spelling correction in the QueryParser with FLAG_PARTIAL.
2422 // Regression test for bug fixed in 1.1.3 and 1.0.17.
2423 DEFINE_TESTCASE(qp_spellpartial1, spelling) {
2425 
2426  db.add_spelling("document");
2427  db.add_spelling("search");
2428  db.add_spelling("paragraph");
2429  db.add_spelling("band");
2430 
2432  qp.set_database(db);
2433 
2434  for (const test *p = test_mispelled_partial_queries; p->query; ++p) {
2435  Xapian::Query q;
2436  q = qp.parse_query(p->query,
2439  tout << "Query: " << p->query << endl;
2441  }
2442 }
2443 
2444 static const test test_synonym_queries[] = {
2445  { "searching", "(Zsearch@1 SYNONYM Zfind@1 SYNONYM Zlocate@1)" },
2446  { "search", "(Zsearch@1 SYNONYM find@1)" },
2447  { "Search", "(search@1 SYNONYM find@1)" },
2448  { "Searching", "searching@1" },
2449  { "searching OR terms", "((Zsearch@1 SYNONYM Zfind@1 SYNONYM Zlocate@1) OR Zterm@2)" },
2450  { "search OR terms", "((Zsearch@1 SYNONYM find@1) OR Zterm@2)" },
2451  { "search +terms", "(Zterm@2 AND_MAYBE (Zsearch@1 SYNONYM find@1))" },
2452  { "search -terms", "((Zsearch@1 SYNONYM find@1) AND_NOT Zterm@2)" },
2453  { "+search terms", "((Zsearch@1 SYNONYM find@1) AND_MAYBE Zterm@2)" },
2454  { "-search terms", "(Zterm@2 AND_NOT (Zsearch@1 SYNONYM find@1))" },
2455  { "search terms", "((Zsearch@1 SYNONYM find@1) OR Zterm@2)" },
2456  // Shouldn't trigger synonyms:
2457  { "\"search terms\"", "(search@1 PHRASE 2 terms@2)" },
2458  // Check that setting FLAG_AUTO_SYNONYMS doesn't enable multi-word
2459  // synonyms. Regression test for bug fixed in 1.3.0 and 1.2.9.
2460  { "regression test", "(Zregress@1 OR Ztest@2)" },
2461  { NULL, NULL }
2462 };
2463 
2464 // Test single term synonyms in the QueryParser.
2465 DEFINE_TESTCASE(qp_synonym1, spelling) {
2467 
2468  db.add_synonym("Zsearch", "Zfind");
2469  db.add_synonym("Zsearch", "Zlocate");
2470  db.add_synonym("search", "find");
2471  db.add_synonym("Zseek", "Zsearch");
2472  db.add_synonym("regression test", "magic");
2473 
2474  db.commit();
2475 
2477  qp.set_stemmer(Xapian::Stem("english"));
2479  qp.set_database(db);
2480 
2481  for (const test *p = test_synonym_queries; p->query; ++p) {
2482  string expect = "Query(";
2483  expect += p->expect;
2484  expect += ')';
2485  Xapian::Query q;
2486  q = qp.parse_query(p->query, qp.FLAG_AUTO_SYNONYMS|qp.FLAG_DEFAULT);
2487  tout << "Query: " << p->query << endl;
2488  TEST_STRINGS_EQUAL(q.get_description(), expect);
2489  }
2490 }
2491 
2492 static const test test_multi_synonym_queries[] = {
2493  { "sun OR tan OR cream", "(Zsun@1 OR Ztan@2 OR Zcream@3)" },
2494  { "sun tan", "((Zsun@1 OR Ztan@2) SYNONYM bathe@1)" },
2495  { "sun tan cream", "((Zsun@1 OR Ztan@2 OR Zcream@3) SYNONYM lotion@1)" },
2496  { "beach sun tan holiday", "(Zbeach@1 OR ((Zsun@2 OR Ztan@3) SYNONYM bathe@2) OR Zholiday@4)" },
2497  { "sun tan sun tan cream", "(((Zsun@1 OR Ztan@2) SYNONYM bathe@1) OR ((Zsun@3 OR Ztan@4 OR Zcream@5) SYNONYM lotion@3))" },
2498  { "single", "(Zsingl@1 SYNONYM record@1)" },
2499  { NULL, NULL }
2500 };
2501 
2502 // Test multi term synonyms in the QueryParser.
2503 DEFINE_TESTCASE(qp_synonym2, synonyms) {
2505 
2506  db.add_synonym("sun tan cream", "lotion");
2507  db.add_synonym("sun tan", "bathe");
2508  db.add_synonym("single", "record");
2509 
2510  db.commit();
2511 
2513  qp.set_stemmer(Xapian::Stem("english"));
2515  qp.set_database(db);
2516 
2517  for (const test *p = test_multi_synonym_queries; p->query; ++p) {
2518  string expect = "Query(";
2519  expect += p->expect;
2520  expect += ')';
2521  Xapian::Query q;
2522  q = qp.parse_query(p->query,
2525  tout << "Query: " << p->query << endl;
2526  TEST_STRINGS_EQUAL(q.get_description(), expect);
2527  }
2528 }
2529 
2530 static const test test_synonym_op_queries[] = {
2531  { "searching", "Zsearch@1" },
2532  { "~searching", "(Zsearch@1 SYNONYM Zfind@1 SYNONYM Zlocate@1)" },
2533  { "~search", "(Zsearch@1 SYNONYM find@1)" },
2534  { "~Search", "(search@1 SYNONYM find@1)" },
2535  { "~Searching", "searching@1" },
2536  { "~searching OR terms", "((Zsearch@1 SYNONYM Zfind@1 SYNONYM Zlocate@1) OR Zterm@2)" },
2537  { "~search OR terms", "((Zsearch@1 SYNONYM find@1) OR Zterm@2)" },
2538  { "~search +terms", "(Zterm@2 AND_MAYBE (Zsearch@1 SYNONYM find@1))" },
2539  { "~search -terms", "((Zsearch@1 SYNONYM find@1) AND_NOT Zterm@2)" },
2540  { "+~search terms", "((Zsearch@1 SYNONYM find@1) AND_MAYBE Zterm@2)" },
2541  { "-~search terms", "(Zterm@2 AND_NOT (Zsearch@1 SYNONYM find@1))" },
2542  { "~search terms", "((Zsearch@1 SYNONYM find@1) OR Zterm@2)" },
2543  { "~foo:search", "(ZXFOOsearch@1 SYNONYM prefixated@1)" },
2544  // FIXME: should look for multi-term synonym...
2545  { "~\"search terms\"", "(search@1 PHRASE 2 terms@2)" },
2546  { NULL, NULL }
2547 };
2548 
2549 // Test the synonym operator in the QueryParser.
2550 DEFINE_TESTCASE(qp_synonym3, synonyms) {
2552 
2553  db.add_synonym("Zsearch", "Zfind");
2554  db.add_synonym("Zsearch", "Zlocate");
2555  db.add_synonym("search", "find");
2556  db.add_synonym("Zseek", "Zsearch");
2557  db.add_synonym("ZXFOOsearch", "prefixated");
2558 
2559  db.commit();
2560 
2562  qp.set_stemmer(Xapian::Stem("english"));
2564  qp.set_database(db);
2565  qp.add_prefix("foo", "XFOO");
2566 
2567  for (const test *p = test_synonym_op_queries; p->query; ++p) {
2568  string expect = "Query(";
2569  expect += p->expect;
2570  expect += ')';
2571  Xapian::Query q;
2572  q = qp.parse_query(p->query,
2577  tout << "Query: " << p->query << endl;
2578  TEST_STRINGS_EQUAL(q.get_description(), expect);
2579  }
2580 }
2581 
2582 static const test test_stem_all_queries[] = {
2583  { "\"chemical engineers\"", "(chemic@1 PHRASE 2 engin@2)" },
2584  { "chemical NEAR engineers", "(chemic@1 NEAR 11 engin@2)" },
2585  { "chemical engineers", "(chemic@1 OR engin@2)" },
2586  { "title:(chemical engineers)", "(XTchemic@1 OR XTengin@2)" },
2587  { NULL, NULL }
2588 };
2589 
2590 DEFINE_TESTCASE(qp_stem_all1, !backend) {
2592  qp.set_stemmer(Xapian::Stem("english"));
2594  qp.add_prefix("title", "XT");
2595  for (const test *p = test_stem_all_queries; p->query; ++p) {
2596  string expect, parsed;
2597  if (p->expect)
2598  expect = p->expect;
2599  else
2600  expect = "parse error";
2601  try {
2602  Xapian::Query qobj = qp.parse_query(p->query);
2603  parsed = qobj.get_description();
2604  expect = string("Query(") + expect + ')';
2605  } catch (const Xapian::QueryParserError &e) {
2606  parsed = e.get_msg();
2607  } catch (const Xapian::Error &e) {
2608  parsed = e.get_description();
2609  } catch (...) {
2610  parsed = "Unknown exception!";
2611  }
2612  tout << "Query: " << p->query << '\n';
2613  TEST_STRINGS_EQUAL(parsed, expect);
2614  }
2615 }
2616 
2617 static const test test_stem_all_z_queries[] = {
2618  { "\"chemical engineers\"", "(Zchemic@1 PHRASE 2 Zengin@2)" },
2619  { "chemical NEAR engineers", "(Zchemic@1 NEAR 11 Zengin@2)" },
2620  { "chemical engineers", "(Zchemic@1 OR Zengin@2)" },
2621  { "title:(chemical engineers)", "(ZXTchemic@1 OR ZXTengin@2)" },
2622  { NULL, NULL }
2623 };
2624 
2625 DEFINE_TESTCASE(qp_stem_all_z1, !backend) {
2627  qp.set_stemmer(Xapian::Stem("english"));
2629  qp.add_prefix("title", "XT");
2630  for (const test *p = test_stem_all_z_queries; p->query; ++p) {
2631  string expect, parsed;
2632  if (p->expect)
2633  expect = p->expect;
2634  else
2635  expect = "parse error";
2636  try {
2637  Xapian::Query qobj = qp.parse_query(p->query);
2638  parsed = qobj.get_description();
2639  expect = string("Query(") + expect + ')';
2640  } catch (const Xapian::QueryParserError &e) {
2641  parsed = e.get_msg();
2642  } catch (const Xapian::Error &e) {
2643  parsed = e.get_description();
2644  } catch (...) {
2645  parsed = "Unknown exception!";
2646  }
2647  tout << "Query: " << p->query << '\n';
2648  TEST_STRINGS_EQUAL(parsed, expect);
2649  }
2650 }
2651 
2652 static double
2653 time_query_parse(const Xapian::Database & db, const string & q,
2654  int repetitions, unsigned flags)
2655 {
2657  qp.set_database(db);
2658  CPUTimer timer;
2659  std::vector<Xapian::Query> qs;
2660  qs.reserve(repetitions);
2661  for (int i = 0; i != repetitions; ++i) {
2662  qs.push_back(qp.parse_query(q, flags));
2663  }
2664  if (repetitions > 1) {
2665  Xapian::Query qc(Xapian::Query::OP_OR, qs.begin(), qs.end());
2666  }
2667  return timer.get_time();
2668 }
2669 
2670 static void
2671 qp_scale1_helper(const Xapian::Database &db, const string & q, unsigned n,
2672  unsigned flags)
2673 {
2674  double time1;
2675  while (true) {
2676  time1 = time_query_parse(db, q, n, flags);
2677  if (time1 != 0.0) break;
2678 
2679  // The first test completed before the timer ticked at all, so increase
2680  // the number of repetitions and retry.
2681  unsigned n_new = n * 10;
2682  if (n_new < n)
2683  SKIP_TEST("Can't count enough repetitions to be able to time test");
2684  n = n_new;
2685  }
2686 
2687  n /= 5;
2688 
2689  string q_n;
2690  q_n.reserve(q.size() * n);
2691  for (unsigned i = n; i != 0; --i) {
2692  q_n += q;
2693  }
2694 
2695  // Time 5 repetitions so we average random variations a bit.
2696  double time2 = time_query_parse(db, q_n, 5, flags);
2697  tout << "small=" << time1 << "s, large=" << time2 << "s\n";
2698 
2699  // Allow a factor of 2.15 difference, to cover random variation and a
2700  // native time interval which isn't an exact multiple of 1/CLK_TCK.
2701  TEST_REL(time2,<,time1 * 2.15);
2702 }
2703 
2704 // Regression test: check that query parser doesn't scale very badly with the
2705 // size of the query.
2706 DEFINE_TESTCASE(qp_scale1, synonyms) {
2708 
2709  db.add_synonym("foo", "bar");
2710  db.commit();
2711 
2712  string q1("foo ");
2713  string q1b("baz ");
2714  const unsigned repetitions = 5000;
2715 
2716  // A long multiword synonym.
2717  string syn;
2718  for (int j = 50; j != 0; --j) {
2719  syn += q1;
2720  }
2721  syn.resize(syn.size() - 1);
2722 
2723  unsigned synflags = Xapian::QueryParser::FLAG_DEFAULT |
2726 
2727  // First, we test a simple query.
2729 
2730  // If synonyms are enabled, a different code-path is followed.
2731  // Test a query which has no synonyms.
2732  qp_scale1_helper(db, q1b, repetitions, synflags);
2733 
2734  // Test a query which has short synonyms.
2735  qp_scale1_helper(db, q1, repetitions, synflags);
2736 
2737  // Add a synonym for the whole query, to test that code path.
2738  db.add_synonym(syn, "bar");
2739  db.commit();
2740 
2741  qp_scale1_helper(db, q1, repetitions, synflags);
2742 }
2743 
2744 static const test test_near_queries[] = {
2745  { "simple-example", "(simple@1 PHRASE 2 example@2)" },
2746  { "stock -cooking", "(Zstock@1 AND_NOT Zcook@2)" },
2747 // FIXME: these give NEAR 2
2748 // { "foo -baz bar", "((foo@1 NEAR 11 bar@3) AND_NOT Zbaz@2)" },
2749 // { "one +two three", "(Ztwo@2 AND_MAYBE (one@1 NEAR 11 three@3))" },
2750  { "foo bar", "(foo@1 NEAR 11 bar@2)" },
2751  { "foo bar baz", "(foo@1 NEAR 12 bar@2 NEAR 12 baz@3)" },
2752  { "gtk+ -gnome", "(Zgtk+@1 AND_NOT Zgnome@2)" },
2753  { "c++ -d--", "(Zc++@1 AND_NOT Zd@2)" },
2754  { "\"c++ library\"", "(c++@1 PHRASE 2 library@2)" },
2755  { "author:orwell animal farm", "(Aorwell@1 NEAR 12 animal@2 NEAR 12 farm@3)" },
2756  { "author:Orwell Animal Farm", "(Aorwell@1 NEAR 12 animal@2 NEAR 12 farm@3)" },
2757  { "beer NOT \"orange juice\"", "(Zbeer@1 AND_NOT (orange@2 PHRASE 2 juice@3))" },
2758  { "beer AND NOT lager", "(Zbeer@1 AND_NOT Zlager@2)" },
2759  { "A OR B NOT C", "(a@1 OR (b@2 AND_NOT c@3))" },
2760  { "A OR B AND NOT C", "(a@1 OR (b@2 AND_NOT c@3))" },
2761  { "A OR B XOR C", "(a@1 OR (b@2 XOR c@3))" },
2762  { "A XOR B NOT C", "(a@1 XOR (b@2 AND_NOT c@3))" },
2763  { "one AND two", "(Zone@1 AND Ztwo@2)" },
2764  { "NOT windows", "Syntax: <expression> NOT <expression>" },
2765  { "a AND (NOT b)", "Syntax: <expression> NOT <expression>" },
2766  { "AND NOT windows", "Syntax: <expression> AND NOT <expression>" },
2767  { "gordian NOT", "Syntax: <expression> NOT <expression>" },
2768  { "gordian AND NOT", "Syntax: <expression> AND NOT <expression>" },
2769  { "foo OR (something AND)", "Syntax: <expression> AND <expression>" },
2770  { "OR foo", "Syntax: <expression> OR <expression>" },
2771  { "XOR", "Syntax: <expression> XOR <expression>" },
2772  { "hard\xa0space", "(hard@1 NEAR 11 space@2)" },
2773  { NULL, NULL }
2774 };
2775 
2776 DEFINE_TESTCASE(qp_near1, !backend) {
2777  Xapian::QueryParser queryparser;
2778  queryparser.set_stemmer(Xapian::Stem("english"));
2780  queryparser.add_prefix("author", "A");
2781  queryparser.add_prefix("writer", "A");
2782  queryparser.add_prefix("title", "XT");
2783  queryparser.add_prefix("subject", "XT");
2784  queryparser.add_prefix("authortitle", "A");
2785  queryparser.add_prefix("authortitle", "XT");
2786  queryparser.add_boolean_prefix("site", "H");
2787  queryparser.add_boolean_prefix("site2", "J");
2788  queryparser.add_boolean_prefix("multisite", "H");
2789  queryparser.add_boolean_prefix("multisite", "J");
2790  queryparser.add_boolean_prefix("category", "XCAT", false);
2791  queryparser.add_boolean_prefix("dogegory", "XDOG", false);
2793  for (const test *p = test_near_queries; p->query; ++p) {
2794  string expect, parsed;
2795  if (p->expect)
2796  expect = p->expect;
2797  else
2798  expect = "parse error";
2799  try {
2800  Xapian::Query qobj = queryparser.parse_query(p->query);
2801  parsed = qobj.get_description();
2802  expect = string("Query(") + expect + ')';
2803  } catch (const Xapian::QueryParserError &e) {
2804  parsed = e.get_msg();
2805  } catch (const Xapian::Error &e) {
2806  parsed = e.get_description();
2807  } catch (...) {
2808  parsed = "Unknown exception!";
2809  }
2810  tout << "Query: " << p->query << '\n';
2811  TEST_STRINGS_EQUAL(parsed, expect);
2812  }
2813 }
2814 
2815 static const test test_phrase_queries[] = {
2816  { "simple-example", "(simple@1 PHRASE 2 example@2)" },
2817  { "stock -cooking", "(Zstock@1 AND_NOT Zcook@2)" },
2818 // FIXME: these give PHRASE 2
2819 // { "foo -baz bar", "((foo@1 PHRASE 11 bar@3) AND_NOT Zbaz@2)" },
2820 // { "one +two three", "(Ztwo@2 AND_MAYBE (one@1 PHRASE 11 three@3))" },
2821  { "foo bar", "(foo@1 PHRASE 11 bar@2)" },
2822  { "foo bar baz", "(foo@1 PHRASE 12 bar@2 PHRASE 12 baz@3)" },
2823  { "gtk+ -gnome", "(Zgtk+@1 AND_NOT Zgnome@2)" },
2824  { "c++ -d--", "(Zc++@1 AND_NOT Zd@2)" },
2825  { "\"c++ library\"", "(c++@1 PHRASE 2 library@2)" },
2826  { "author:orwell animal farm", "(Aorwell@1 PHRASE 12 animal@2 PHRASE 12 farm@3)" },
2827  { "author:Orwell Animal Farm", "(Aorwell@1 PHRASE 12 animal@2 PHRASE 12 farm@3)" },
2828  { "beer NOT \"orange juice\"", "(Zbeer@1 AND_NOT (orange@2 PHRASE 2 juice@3))" },
2829  { "beer AND NOT lager", "(Zbeer@1 AND_NOT Zlager@2)" },
2830  { "A OR B NOT C", "(a@1 OR (b@2 AND_NOT c@3))" },
2831  { "A OR B AND NOT C", "(a@1 OR (b@2 AND_NOT c@3))" },
2832  { "A OR B XOR C", "(a@1 OR (b@2 XOR c@3))" },
2833  { "A XOR B NOT C", "(a@1 XOR (b@2 AND_NOT c@3))" },
2834  { "one AND two", "(Zone@1 AND Ztwo@2)" },
2835  { "NOT windows", "Syntax: <expression> NOT <expression>" },
2836  { "a AND (NOT b)", "Syntax: <expression> NOT <expression>" },
2837  { "AND NOT windows", "Syntax: <expression> AND NOT <expression>" },
2838  { "gordian NOT", "Syntax: <expression> NOT <expression>" },
2839  { "gordian AND NOT", "Syntax: <expression> AND NOT <expression>" },
2840  { "foo OR (something AND)", "Syntax: <expression> AND <expression>" },
2841  { "OR foo", "Syntax: <expression> OR <expression>" },
2842  { "XOR", "Syntax: <expression> XOR <expression>" },
2843  { "hard\xa0space", "(hard@1 PHRASE 11 space@2)" },
2844  // FIXME: this isn't what we want, but fixing phrase to work with
2845  // subqueries first might be the best approach.
2846  // FIXME: this isn't currently reimplemented:
2847  // { "(one AND two) three", "((Zone@1 PHRASE 11 Zthree@3) AND (Ztwo@2 PHRASE 11 Zthree@3))" },
2848  { NULL, NULL }
2849 };
2850 
2851 DEFINE_TESTCASE(qp_phrase1, !backend) {
2852  Xapian::QueryParser queryparser;
2853  queryparser.set_stemmer(Xapian::Stem("english"));
2855  queryparser.add_prefix("author", "A");
2856  queryparser.add_prefix("writer", "A");
2857  queryparser.add_prefix("title", "XT");
2858  queryparser.add_prefix("subject", "XT");
2859  queryparser.add_prefix("authortitle", "A");
2860  queryparser.add_prefix("authortitle", "XT");
2861  queryparser.add_boolean_prefix("site", "H");
2862  queryparser.add_boolean_prefix("site2", "J");
2863  queryparser.add_boolean_prefix("multisite", "H");
2864  queryparser.add_boolean_prefix("multisite", "J");
2865  queryparser.add_boolean_prefix("category", "XCAT", false);
2866  queryparser.add_boolean_prefix("dogegory", "XDOG", false);
2868  for (const test *p = test_phrase_queries; p->query; ++p) {
2869  string expect, parsed;
2870  if (p->expect)
2871  expect = p->expect;
2872  else
2873  expect = "parse error";
2874  try {
2875  Xapian::Query qobj = queryparser.parse_query(p->query);
2876  parsed = qobj.get_description();
2877  expect = string("Query(") + expect + ')';
2878  } catch (const Xapian::QueryParserError &e) {
2879  parsed = e.get_msg();
2880  } catch (const Xapian::Error &e) {
2881  parsed = e.get_description();
2882  } catch (...) {
2883  parsed = "Unknown exception!";
2884  }
2885  tout << "Query: " << p->query << '\n';
2886  TEST_STRINGS_EQUAL(parsed, expect);
2887  }
2888 }
2889 
2890 static const test test_stopword_group_or_queries[] = {
2891  { "this is a test", "test@4" },
2892  { "test*", "WILDCARD SYNONYM test" },
2893  { "a test*", "WILDCARD SYNONYM test" },
2894  { "is a test*", "WILDCARD SYNONYM test" },
2895  { "this is a test*", "WILDCARD SYNONYM test" },
2896  { "this is a us* test*", "(WILDCARD SYNONYM us OR WILDCARD SYNONYM test)" },
2897  { "this is a user test*", "(user@4 OR WILDCARD SYNONYM test)" },
2898  { NULL, NULL }
2899 };
2900 
2901 static const test test_stopword_group_and_queries[] = {
2902  { "this is a test", "test@4" },
2903  { "test*", "WILDCARD SYNONYM test" },
2904  { "a test*", "WILDCARD SYNONYM test" },
2905  // Two stopwords + one wildcard failed in 1.0.16
2906  { "is a test*", "WILDCARD SYNONYM test" },
2907  // Three stopwords + one wildcard failed in 1.0.16
2908  { "this is a test*", "WILDCARD SYNONYM test" },
2909  // Three stopwords + two wildcards failed in 1.0.16
2910  { "this is a us* test*", "(WILDCARD SYNONYM us AND WILDCARD SYNONYM test)" },
2911  { "this is a user test*", "(user@4 AND WILDCARD SYNONYM test)" },
2912  { NULL, NULL }
2913 };
2914 
2915 // Regression test for bug fixed in 1.0.17 and 1.1.3.
2916 DEFINE_TESTCASE(qp_stopword_group1, writable) {
2918  Xapian::Document doc;
2919  doc.add_term("test");
2920  doc.add_term("tester");
2921  doc.add_term("testable");
2922  doc.add_term("user");
2923  db.add_document(doc);
2924 
2925  Xapian::SimpleStopper stopper;
2926  stopper.add("this");
2927  stopper.add("is");
2928  stopper.add("a");
2929 
2931  qp.set_stopper(&stopper);
2932  qp.set_database(db);
2933 
2934  // Process test cases with OP_OR first, then with OP_AND.
2936  const test *p = test_stopword_group_or_queries;
2937  for (int i = 1; i <= 2; ++i) {
2938  for ( ; p->query; ++p) {
2939  string expect, parsed;
2940  if (p->expect)
2941  expect = p->expect;
2942  else
2943  expect = "parse error";
2944  try {
2945  Xapian::Query qobj = qp.parse_query(p->query, qp.FLAG_WILDCARD);
2946  parsed = qobj.get_description();
2947  expect = string("Query(") + expect + ')';
2948  } catch (const Xapian::QueryParserError &e) {
2949  parsed = e.get_msg();
2950  } catch (const Xapian::Error &e) {
2951  parsed = e.get_description();
2952  } catch (...) {
2953  parsed = "Unknown exception!";
2954  }
2955  tout << "Query: " << p->query << '\n';
2956  TEST_STRINGS_EQUAL(parsed, expect);
2957  }
2958 
2961  }
2962 }
2963 
2965 DEFINE_TESTCASE(qp_default_op2, !backend) {
2967  static const Xapian::Query::op ops[] = {
2976  };
2977  for (Xapian::Query::op op : ops) {
2978  tout << op << endl;
2980  qp.set_default_op(op));
2982  }
2983 }
2984 
2987  const char *expect;
2988 };
2989 
2991 DEFINE_TESTCASE(qp_default_op3, !backend) {
2993  static const qp_default_op3_test tests[] = {
2995  "Query((a@1 AND b@2 AND c@3))" },
2997  "Query((a@1 OR b@2 OR c@3))" },
2999  "Query((a@1 PHRASE 12 b@2 PHRASE 12 c@3))" },
3001  "Query((a@1 NEAR 12 b@2 NEAR 12 c@3))" },
3003  "Query((a@1 ELITE_SET 10 b@2 ELITE_SET 10 c@3))" },
3005  "Query((a@1 SYNONYM b@2 SYNONYM c@3))" },
3006  };
3007  const qp_default_op3_test * p;
3008  for (p = tests; p - tests != sizeof(tests) / sizeof(*tests); ++p) {
3009  tout << p->op << endl;
3010  qp.set_default_op(p->op);
3011  // Check that get_default_op() returns what we just set.
3012  TEST_EQUAL(qp.get_default_op(), p->op);
3013  TEST_EQUAL(qp.parse_query("A B C").get_description(), p->expect);
3014  }
3015 }
3016 
3018 DEFINE_TESTCASE(qp_defaultstrategysome1, !backend) {
3020  qp.set_stemmer(Xapian::Stem("en"));
3021  TEST_EQUAL(qp.parse_query("testing").get_description(), "Query(Ztest@1)");
3022 }
3023 
3025 DEFINE_TESTCASE(qp_stemsomefullpos, !backend) {
3027  qp.set_stemmer(Xapian::Stem("en"));
3029  TEST_EQUAL(qp.parse_query("terms NEAR testing").get_description(), "Query((Zterm@1 NEAR 11 Ztest@2))");
3030  TEST_EQUAL(qp.parse_query("terms ADJ testing").get_description(), "Query((Zterm@1 PHRASE 11 Ztest@2))");
3031 }
3032 
3033 DEFINE_TESTCASE(qp_nopos, !backend) {
3034  static const test tests[] = {
3035  { "no pos anyway", "(no@1 OR pos@2 OR anyway@3)" },
3036  { "w ADJ x", "(w@1 AND x@2)" },
3037  { "\"phrase q\" OR A NEAR/4 B", "((phrase@1 AND q@2) OR (a@3 AND b@4))" },
3038  // Check FLAG_NO_POSITIONS stays on if we reparse with fewer flags.
3039  { "a-b NEAR x", "((a@1 AND b@2) OR (near@3 OR x@4))" },
3040  };
3042  const auto flags = qp.FLAG_DEFAULT | qp.FLAG_NO_POSITIONS;
3043  for (const test& p : tests) {
3044  string expect, parsed;
3045  if (p.expect)
3046  expect = p.expect;
3047  else
3048  expect = "parse error";
3049  try {
3050  Xapian::Query q = qp.parse_query(p.query, flags);
3051  parsed = q.get_description();
3052  expect = string("Query(") + expect + ')';
3053  } catch (const Xapian::QueryParserError& e) {
3054  parsed = e.get_msg();
3055  } catch (const Xapian::Error& e) {
3056  parsed = e.get_description();
3057  } catch (...) {
3058  parsed = "Unknown exception!";
3059  }
3060  tout << "Query: " << p.query << '\n';
3061  TEST_STRINGS_EQUAL(parsed, expect);
3062  }
3063 }
#define TEST_MSET_SIZE(M, S)
Check MSet M has size S.
Definition: testutils.h:78
The Xapian namespace contains public interfaces for the Xapian library.
Definition: compactor.cc:80
Xapian::doccount size() const
Return number of items in this MSet object.
Definition: omenquire.cc:318
Xapian::docid add_document(const Xapian::Document &document)
Add a new document to the database.
Definition: omdatabase.cc:902
Xapian::valueno operator()(std::string &begin, std::string &end)
Check for a valid range of this type.
Handle a date range.
Definition: queryparser.h:520
void add_value(Xapian::valueno slot, const std::string &value)
Add a new value.
Definition: omdocument.cc:107
const char * query
static double time_query_parse(const Xapian::Database &db, const string &q, int repetitions, unsigned flags)
static const test test_phrase_queries[]
Simple implementation of Stopper class - this will suit most users.
Definition: queryparser.h:100
#define TEST(a)
Test a condition, without an additional explanation for failure.
Definition: testsuite.h:275
static const test test_near_queries[]
unsigned tolower(unsigned ch)
Convert a Unicode character to lowercase.
Definition: unicode.h:376
char C_tolower(char ch)
Definition: stringutils.h:178
void set_default_op(Query::op default_op)
Set the default operator.
Definition: queryparser.cc:102
static const test test_pure_not_queries[]
This class is used to access a database, or a group of databases.
Definition: database.h:68
static const test test_synonym_op_queries[]
Xapian::Query operator()(const std::string &b, const std::string &e)
Check for a valid range of this type.
static const Xapian::Query MatchAll
A query matching all documents.
Definition: query.h:75
Match documents which an odd number of subqueries match.
Definition: query.h:107
InvalidOperationError indicates the API was used in an invalid way.
Definition: error.h:283
Class representing a stemming algorithm.
Definition: stem.h:62
void add_valuerangeprocessor(Xapian::ValueRangeProcessor *vrproc)
Register a ValueRangeProcessor.
Definition: queryparser.h:1265
static const test test_value_range4_queries[]
void set_stopper(const Stopper *stop=NULL)
Set the stopper.
Definition: queryparser.cc:96
op
Query operators.
Definition: query.h:78
std::string get_corrected_query_string() const
Get the spelling-corrected query string.
Definition: queryparser.cc:242
Xapian::WritableDatabase get_writable_database(const string &dbname)
Definition: apitest.cc:87
TermIterator unstem_begin(const std::string &term) const
Begin iterator over unstemmed forms of the given stemmed query term.
Definition: queryparser.cc:221
Handle a number range.
Definition: queryparser.h:343
Xapian::Query operator()(const std::string &str)
Convert a field-prefixed string to a Query object.
const std::string & get_msg() const
Message giving details of the error, intended for human consumption.
Definition: error.h:122
Build a Xapian::Query object from a user query string.
Definition: queryparser.h:778
WildcardError indicates an error expanding a wildcarded query.
Definition: error.h:1013
a generic test suite engine
Class representing a list of search results.
Definition: mset.h:44
STL namespace.
virtual Xapian::Query operator()(const std::string &begin, const std::string &end)
Check for a valid range of this type.
MSet get_mset(Xapian::doccount first, Xapian::doccount maxitems, Xapian::doccount checkatleast=0, const RSet *omrset=0, const MatchDecider *mdecider=0) const
Get (a portion of) the match set for the current query.
Definition: omenquire.cc:932
Indicates a query string can&#39;t be parsed.
Definition: error.h:887
Convert types to std::string.
static void test_qp_flag_wildcard3_helper(const Xapian::Database &db, Xapian::termcount max_expansion, const string &query_string)
std::string sortable_serialise(double value)
Convert a floating point number to a string, preserving sort order.
Definition: queryparser.h:1347
void add(const std::string &word)
Add a single stop word.
Definition: queryparser.h:124
Produce a query which doesn&#39;t use positional information.
Definition: queryparser.h:912
static Xapian::Stem stemmer
Definition: stemtest.cc:41
Base class for value range processors.
Definition: queryparser.h:404
void set_max_expansion(Xapian::termcount max_expansion, int max_type=Xapian::Query::WILDCARD_LIMIT_ERROR, unsigned flags=FLAG_WILDCARD|FLAG_PARTIAL)
Specify the maximum expansion of a wildcard and/or partial term.
Definition: queryparser.cc:146
static const test test_or_queries[]
Enable automatic use of synonyms for single terms and groups of terms.
Definition: queryparser.h:871
static void qp_scale1_helper(const Xapian::Database &db, const string &q, unsigned n, unsigned flags)
void add_rangeprocessor(Xapian::RangeProcessor *range_proc, const std::string *grouping=NULL)
Register a RangeProcessor.
Definition: queryparser.cc:234
static const test test_stem_all_z_queries[]
Xapian::WritableDatabase get_named_writable_database(const std::string &name, const std::string &source)
Definition: apitest.cc:93
Enable partial matching.
Definition: queryparser.h:837
test functionality of the Xapian API
void set_stemmer(const Xapian::Stem &stemmer)
Set the stemmer.
Definition: queryparser.cc:84
static const test test_value_customrange1_queries[]
TermIterator unstem_end(const std::string &) const
End iterator over unstemmed forms of the given stemmed query term.
Definition: queryparser.h:1252
TermIterator stoplist_begin() const
Begin iterator over terms omitted from the query as stopwords.
Definition: queryparser.cc:214
Class for iterating over a list of terms.
Definition: termiterator.h:41
static const test test_fieldproc2_queries[]
unsigned XAPIAN_TERMCOUNT_BASE_TYPE termcount
A counts of terms.
Definition: types.h:72
#define TEST_REL(A, REL, B)
Test a relation holds,e.g. TEST_REL(a,>,b);.
Definition: testmacros.h:32
Xapian::Query operator()(const std::string &str)
Convert a field-prefixed string to a Query object.
InvalidArgumentError indicates an invalid parameter value was passed to the API.
Definition: error.h:241
const char * expect
Handle a date range.
Definition: queryparser.h:253
static const test test_fieldproc1_queries[]
Base class for field processors.
Definition: queryparser.h:729
Pick the best N subqueries and combine with OP_OR.
Definition: query.h:215
This class provides read/write access to a database.
Definition: database.h:785
static const test test_and_queries[]
void set_stemming_strategy(stem_strategy strategy)
Set the stemming strategy.
Definition: queryparser.cc:90
static const test test_stopword_group_or_queries[]
std::ostringstream tout
The debug printing stream.
Definition: testsuite.cc:103
Scale the weight contributed by a subquery.
Definition: query.h:166
Match only documents where all subqueries match near and in order.
Definition: query.h:152
Match the first subquery taking extra weight from other subqueries.
Definition: query.h:118
Public interfaces for the Xapian library.
Match like OP_AND but only taking weight from the first subquery.
Definition: query.h:128
Match only documents where a value slot is >= a given value.
Definition: query.h:223
static const test test_value_range1_queries[]
TermIterator stoplist_end() const
End iterator over terms omitted from the query as stopwords.
Definition: queryparser.h:1244
#define TEST_EXCEPTION(TYPE, CODE)
Check that CODE throws exactly Xapian exception TYPE.
Definition: testutils.h:109
Xapian::Query operator()(const std::string &str)
Convert a field-prefixed string to a Query object.
Match only documents where a value slot is within a given range.
Definition: query.h:158
void add_boolean_prefix(const std::string &field, const std::string &prefix, const std::string *grouping=NULL)
Add a boolean term prefix allowing the user to restrict a search with a boolean filter specified in t...
Definition: queryparser.cc:197
string str(int value)
Convert int to std::string.
Definition: str.cc:90
Match only documents where a value slot is <= a given value.
Definition: query.h:231
void commit()
Commit any pending modifications made to the database.
Definition: omdatabase.cc:857
Support quoted phrases.
Definition: queryparser.h:790
static const test test_stem_all_queries[]
bool startswith(const std::string &s, char pfx)
Definition: stringutils.h:46
static const test test_value_daterange2_queries[]
static const test_desc tests[]
The lists of tests to perform.
static const test test_stopword_group_and_queries[]
Query parse_query(const std::string &query_string, unsigned flags=FLAG_DEFAULT, const std::string &default_prefix=std::string())
Parse a query.
Definition: queryparser.cc:161
Handle a string range.
Definition: queryparser.h:466
void add_database(const Database &database)
Add an existing database (or group of databases) to those accessed by this object.
Definition: omdatabase.cc:148
void set_query(const Xapian::Query &query, Xapian::termcount qlen=0)
Set the query to run.
Definition: omenquire.cc:793
Base class for range processors.
Definition: queryparser.h:140
Match like OP_OR but weighting as if a single term.
Definition: query.h:239
static const test test_mispelled_queries[]
std::string get_description() const
Return a string describing this object.
Definition: error.cc:93
#define FAIL_TEST(MSG)
Fail the current testcase with message MSG.
Definition: testsuite.h:68
Match only documents which all subqueries match.
Definition: query.h:84
static Xapian::Query query(Xapian::Query::op op, const string &t1=string(), const string &t2=string(), const string &t3=string(), const string &t4=string(), const string &t5=string(), const string &t6=string(), const string &t7=string(), const string &t8=string(), const string &t9=string(), const string &t10=string())
Definition: api_anydb.cc:63
Xapian::Database get_database(const string &dbname)
Definition: apitest.cc:48
Enable automatic use of synonyms for single terms.
Definition: queryparser.h:864
static const test test_value_stringrange1_queries[]
double get_time() const
Return elapsed CPU time since object creation in seconds.
Definition: cputimer.h:34
DEFINE_TESTCASE(queryparser1, !backend)
Measure CPU time.
void set_database(const Database &db)
Specify the database being searched.
Definition: queryparser.cc:141
Handle a number range.
Definition: queryparser.h:662
Accumulate unstem and stoplist results.
Definition: queryparser.h:901
#define SKIP_TEST(MSG)
Skip the current testcase with message MSG.
Definition: testsuite.h:74
std::string get_description() const
Return a string describing this object.
Definition: query.cc:232
This class provides an interface to the information retrieval system for the purpose of searching...
Definition: enquire.h:152
Support AND, OR, etc and bracketed subexpressions.
Definition: queryparser.h:788
Match only documents where all subqueries match near each other.
Definition: query.h:140
Xapian::Query::op op
static const test test_synonym_queries[]
All exceptions thrown by Xapian are subclasses of Xapian::Error.
Definition: error.h:43
Match documents which the first subquery matches but no others do.
Definition: query.h:99
Match documents which at least one subquery matches.
Definition: query.h:92
unsigned valueno
The number for a value slot in a document.
Definition: types.h:108
Xapian-specific test helper functions and macros.
static const test test_multi_synonym_queries[]
Various handy helpers which std::string really should provide.
static const test test_mispelled_wildcard_queries[]
#define TEST_STRINGS_EQUAL(a, b)
Test for equality of two strings.
Definition: testsuite.h:287
Enable generation of n-grams from CJK text.
Definition: queryparser.h:886
static const test test_value_daterange1_queries[]
void add_synonym(const std::string &term, const std::string &synonym) const
Add a synonym for a term.
Definition: omdatabase.cc:1028
Class representing a query.
Definition: query.h:46
static const test test_value_range2_queries[]
void add_prefix(const std::string &field, const std::string &prefix)
Add a free-text field term prefix.
Definition: queryparser.cc:183
const valueno BAD_VALUENO
Reserved value to indicate "no valueno".
Definition: types.h:125
#define TEST_EQUAL(a, b)
Test for equality of two things.
Definition: testsuite.h:278
Query::op get_default_op() const
Get the current default operator.
Definition: queryparser.cc:135
Enable synonym operator &#39;~&#39;.
Definition: queryparser.h:858
static const test test_mispelled_partial_queries[]
A handle representing a document in a Xapian database.
Definition: document.h:61
UnimplementedError indicates an attempt to use an unimplemented feature.
Definition: error.h:325
void add_spelling(const std::string &word, Xapian::termcount freqinc=1) const
Add a word to the spelling dictionary.
Definition: omdatabase.cc:1004
static const test test_stop_queries[]
void add_term(const std::string &tname, Xapian::termcount wdfinc=1)
Add a term to the document, without positional information.
Definition: omdocument.cc:140