Add DB_BACKEND_INMEMORY; deprecate InMemory::open()
[xapian.git] / xapian-core / tests / queryparsertest.cc
blobf2e26fb001ab59a0a21469d8258dff6790a06f61
1 /* queryparsertest.cc: Tests of Xapian::QueryParser
3 * Copyright (C) 2002,2003,2004,2005,2006,2007,2008,2009,2010,2011,2012,2013,2015,2016 Olly Betts
4 * Copyright (C) 2007,2009 Lemur Consulting Ltd
6 * This program is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU General Public License as
8 * published by the Free Software Foundation; either version 2 of the
9 * License, or (at your option) any later version.
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
16 * You should have received a copy of the GNU General Public License
17 * along with this program; if not, write to the Free Software
18 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301
19 * USA
22 #include <config.h>
24 #include <xapian.h>
26 #include "cputimer.h"
27 #include "str.h"
28 #include "stringutils.h"
30 #include <cmath>
31 #include <iostream>
32 #include <string>
33 #include <vector>
34 #include "safesysstat.h" // For mkdir().
36 using namespace std;
38 #include "testsuite.h"
39 #include "testutils.h"
41 struct test {
42 const char *query;
43 const char *expect;
46 static const test test_or_queries[] = {
47 { "simple-example", "(simple@1 PHRASE 2 example@2)" },
48 { "time_t", "Ztime_t@1" },
49 { "stock -cooking", "(Zstock@1 AND_NOT Zcook@2)" },
50 { "foo -baz bar", "((Zfoo@1 OR Zbar@3) AND_NOT Zbaz@2)" },
51 { "d- school report", "(Zd@1 OR (Zschool@2 OR Zreport@3))" },
52 { "gtk+ -gnome", "(Zgtk+@1 AND_NOT Zgnome@2)" },
53 { "c++ -d--", "(Zc++@1 AND_NOT Zd@2)" },
54 { "Mg2+ Cl-", "(mg2+@1 OR cl@2)" },
55 { "\"c++ library\"", "(c++@1 PHRASE 2 library@2)" },
56 { "A&L A&RMCO AD&D", "(a&l@1 OR a&rmco@2 OR ad&d@3)" },
57 { "C# vs C++", "(c#@1 OR Zvs@2 OR c++@3)" },
58 { "j##", "Zj##@1" },
59 { "a#b", "(Za@1 OR Zb@2)" },
60 { "O.K. U.N.C.L.E XY.Z.", "((ok@1 OR uncle@2) OR (xy@3 PHRASE 2 z@4))" },
61 { "author:orwell animal farm", "(ZAorwel@1 OR Zanim@2 OR Zfarm@3)" },
62 { "author:Orwell Animal Farm", "(Aorwell@1 OR animal@2 OR farm@3)" },
63 // Regression test for bug reported in 0.9.6.
64 { "author:\"orwell\" title:\"animal\"", "(Aorwell@1 OR XTanimal@2)" },
65 // Regression test for bug related to one reported in 0.9.6.
66 { "author:(orwell) title:(animal)", "(ZAorwel@1 OR ZXTanim@2)" },
67 // Regression test for bug caused by fix for previous bug.
68 { "author:\"milne, a.a.\"", "(Amilne@1 PHRASE 3 Aa@2 PHRASE 3 Aa@3)" },
69 { "author:\"milne a.a.\"", "(Amilne@1 PHRASE 3 Aa@2 PHRASE 3 Aa@3)" },
70 // Regression test for bug reported in 0.9.7.
71 { "site:/path/name", "0 * H/path/name" },
72 // Regression test for bug introduced (and fixed) in SVN prior to 1.0.0.
73 { "author:/path/name", "(Apath@1 PHRASE 2 Aname@2)" },
74 // Feature tests for change to allow phrase generators after prefix in 1.2.4.
75 { "author:/path", "ZApath@1" },
76 { "author:-Foo", "Afoo@1" },
77 { "author:/", "Zauthor@1" },
78 { "author::", "Zauthor@1" },
79 { "author:/ foo", "(Zauthor@1 OR Zfoo@2)" },
80 { "author:: foo", "(Zauthor@1 OR Zfoo@2)" },
81 { "author::foo", "(author@1 PHRASE 2 foo@2)" },
82 { "author:/ AND foo", "(Zauthor@1 AND Zfoo@2)" },
83 { "author:: AND foo", "(Zauthor@1 AND Zfoo@2)" },
84 { "foo AND author:/", "(Zfoo@1 AND Zauthor@2)" },
85 { "foo AND author::", "(Zfoo@1 AND Zauthor@2)" },
86 // Regression test for bug introduced into (and fixed) in SVN prior to 1.0.0.
87 { "author:(title::case)", "(Atitle@1 PHRASE 2 Acase@2)" },
88 // Regression test for bug fixed in 1.0.4 - the '+' would be ignored there
89 // because the whitespace after the '"' wasn't noticed.
90 { "\"hello world\" +python", "(Zpython@3 AND_MAYBE (hello@1 PHRASE 2 world@2))" },
91 // In 1.1.0, NON_SPACING_MARK was added as a word character.
92 { "\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" },
93 // In 1.1.4, ENCLOSING_MARK and COMBINING_SPACING_MARK were added, and
94 // code to ignore several zero-width space characters was added.
95 { "\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" },
96 { "unmatched\"", "unmatched@1" },
97 { "unmatched \" \" ", "Zunmatch@1" },
98 { "hyphen-ated\" ", "(hyphen@1 PHRASE 2 ated@2)" },
99 { "hyphen-ated\" \"", "(hyphen@1 PHRASE 2 ated@2)" },
100 { "\"1.4\"", "1.4@1" },
101 { "\"1.\"", "1@1" },
102 { "\"A#.B.\"", "(a#@1 PHRASE 2 b@2)" },
103 { "\" Xapian QueryParser\" parses queries", "((xapian@1 PHRASE 2 queryparser@2) OR (Zpars@3 OR Zqueri@4))" },
104 { "\" xapian queryParser\" parses queries", "((xapian@1 PHRASE 2 queryparser@2) OR (Zpars@3 OR Zqueri@4))" },
105 { "h\xc3\xb6hle", "Zh\xc3\xb6hle@1" },
106 { "one +two three", "(Ztwo@2 AND_MAYBE (Zone@1 OR Zthree@3))" },
107 { "subject:test other", "(ZXTtest@1 OR Zother@2)" },
108 { "subject:\"space flight\"", "(XTspace@1 PHRASE 2 XTflight@2)" },
109 { "author:(twain OR poe) OR flight", "((ZAtwain@1 OR ZApoe@2) OR Zflight@3)" },
110 { "author:(twain OR title:pit OR poe)", "((ZAtwain@1 OR ZXTpit@2) OR ZApoe@3)" },
111 { "title:2001 title:space", "(XT2001@1 OR ZXTspace@2)" },
112 { "(title:help)", "ZXThelp@1" },
113 { "beer NOT \"orange juice\"", "(Zbeer@1 AND_NOT (orange@2 PHRASE 2 juice@3))" },
114 { "beer AND NOT lager", "(Zbeer@1 AND_NOT Zlager@2)" },
115 { "beer AND -lager", "(Zbeer@1 AND_NOT Zlager@2)" },
116 { "beer AND +lager", "(Zbeer@1 AND Zlager@2)" },
117 { "A OR B NOT C", "(a@1 OR (b@2 AND_NOT c@3))" },
118 { "A OR B AND NOT C", "(a@1 OR (b@2 AND_NOT c@3))" },
119 { "A OR B AND -C", "(a@1 OR (b@2 AND_NOT c@3))" },
120 { "A OR B AND +C", "(a@1 OR (b@2 AND c@3))" },
121 { "A OR B XOR C", "(a@1 OR (b@2 XOR c@3))" },
122 { "A XOR B NOT C", "(a@1 XOR (b@2 AND_NOT c@3))" },
123 { "one AND two", "(Zone@1 AND Ztwo@2)" },
124 { "one A.N.D. two", "(Zone@1 OR and@2 OR Ztwo@3)" },
125 { "one \xc3\x81ND two", "(Zone@1 OR \xc3\xa1nd@2 OR Ztwo@3)" },
126 { "one author:AND two", "(Zone@1 OR Aand@2 OR Ztwo@3)" },
127 { "author:hyphen-ated", "(Ahyphen@1 PHRASE 2 Aated@2)" },
128 { "cvs site:xapian.org", "(Zcvs@1 FILTER Hxapian.org)" },
129 { "cvs -site:xapian.org", "(Zcvs@1 AND_NOT Hxapian.org)" },
130 { "foo -site:xapian.org bar", "((Zfoo@1 OR Zbar@2) AND_NOT Hxapian.org)" },
131 { "site:xapian.org mail", "(Zmail@1 FILTER Hxapian.org)" },
132 { "-site:xapian.org mail", "(Zmail@1 AND_NOT Hxapian.org)" },
133 { "mail AND -site:xapian.org", "(Zmail@1 AND_NOT 0 * Hxapian.org)" },
134 { "-Wredundant-decls", "(wredundant@1 PHRASE 2 decls@2)" },
135 { "site:xapian.org", "0 * Hxapian.org" },
136 { "mug +site:xapian.org -site:cvs.xapian.org", "((Zmug@1 FILTER Hxapian.org) AND_NOT Hcvs.xapian.org)" },
137 { "mug -site:cvs.xapian.org +site:xapian.org", "((Zmug@1 FILTER Hxapian.org) AND_NOT Hcvs.xapian.org)" },
138 { "mug +site:xapian.org AND -site:cvs.xapian.org", "((Zmug@1 FILTER Hxapian.org) AND_NOT 0 * Hcvs.xapian.org)" },
139 { "mug site:xapian.org AND -site:cvs.xapian.org", "((Zmug@1 FILTER Hxapian.org) AND_NOT 0 * Hcvs.xapian.org)" },
140 { "mug site:xapian.org AND +site:cvs.xapian.org", "((Zmug@1 FILTER Hxapian.org) AND 0 * Hcvs.xapian.org)" },
141 { "NOT windows", "Syntax: <expression> NOT <expression>" },
142 { "a AND (NOT b)", "Syntax: <expression> NOT <expression>" },
143 { "AND NOT windows", "Syntax: <expression> AND NOT <expression>" },
144 { "AND -windows", "Syntax: <expression> AND <expression>" },
145 { "gordian NOT", "Syntax: <expression> NOT <expression>" },
146 { "gordian AND NOT", "Syntax: <expression> AND NOT <expression>" },
147 { "gordian AND -", "Syntax: <expression> AND <expression>" },
148 { "foo OR (something AND)", "Syntax: <expression> AND <expression>" },
149 { "OR foo", "Syntax: <expression> OR <expression>" },
150 { "XOR", "Syntax: <expression> XOR <expression>" },
151 { "hard\xa0space", "(Zhard@1 OR Zspace@2)" },
152 { " white\r\nspace\ttest ", "(Zwhite@1 OR Zspace@2 OR Ztest@3)" },
153 { "one AND two three", "(Zone@1 AND (Ztwo@2 OR Zthree@3))" },
154 { "one two AND three", "((Zone@1 OR Ztwo@2) AND Zthree@3)" },
155 { "one AND two/three", "(Zone@1 AND (two@2 PHRASE 2 three@3))" },
156 { "one AND /two/three", "(Zone@1 AND (two@2 PHRASE 2 three@3))" },
157 { "one AND/two/three", "(Zone@1 AND (two@2 PHRASE 2 three@3))" },
158 { "one +/two/three", "((two@2 PHRASE 2 three@3) AND_MAYBE Zone@1)" },
159 { "one//two", "(one@1 PHRASE 2 two@2)" },
160 { "\"missing quote", "(missing@1 PHRASE 2 quote@2)" },
161 { "DVD+RW", "(dvd@1 OR rw@2)" }, // Would a phrase be better?
162 { "+\"must have\" optional", "((must@1 PHRASE 2 have@2) AND_MAYBE Zoption@3)" },
163 { "one NEAR two NEAR three", "(one@1 NEAR 12 two@2 NEAR 12 three@3)" },
164 { "something NEAR/3 else", "(something@1 NEAR 4 else@2)" },
165 { "a NEAR/6 b NEAR c", "(a@1 NEAR 8 b@2 NEAR 8 c@3)" },
166 { "something ADJ else", "(something@1 PHRASE 11 else@2)" },
167 { "something ADJ/3 else", "(something@1 PHRASE 4 else@2)" },
168 { "a ADJ/6 b ADJ c", "(a@1 PHRASE 8 b@2 PHRASE 8 c@3)" },
169 // Regression test - Unicode character values were truncated to 8 bits
170 // before testing C_isdigit(), so this rather artificial example parsed
171 // to: (a@1 NEAR 262 b@2)
172 { "a NEAR/\xc4\xb5 b", "((Za@1 OR (near@2 PHRASE 2 \xc4\xb5@3)) OR Zb@4)" },
173 { "a ADJ/\xc4\xb5 b", "((Za@1 OR (adj@2 PHRASE 2 \xc4\xb5@3)) OR Zb@4)" },
174 // Regression test - the first two cases were parsed as if the '/' were a
175 // space, which was inconsistent with the second two. Fixed in 1.2.5.
176 { "a NEAR/b", "(Za@1 OR (near@2 PHRASE 2 b@3))" },
177 { "a ADJ/b", "(Za@1 OR (adj@2 PHRASE 2 b@3))" },
178 { "a NEAR/b c", "((Za@1 OR (near@2 PHRASE 2 b@3)) OR Zc@4)" },
179 { "a ADJ/b c", "((Za@1 OR (adj@2 PHRASE 2 b@3)) OR Zc@4)" },
180 // Regression tests - + and - didn't work on bracketed subexpressions prior
181 // to 1.0.2.
182 { "+(one two) three", "((Zone@1 OR Ztwo@2) AND_MAYBE Zthree@3)" },
183 { "zero -(one two)", "(Zzero@1 AND_NOT (Zone@2 OR Ztwo@3))" },
184 // Feature tests that ':' is inserted between prefix and term correctly:
185 { "category:Foo", "0 * XCAT:Foo" },
186 { "category:foo", "0 * XCATfoo" },
187 { "category:\xc3\x96oo", "0 * XCAT\xc3\x96oo" },
188 // Feature tests for quoted boolean terms:
189 { "category:\"Hello world\"", "0 * XCAT:Hello world" },
190 { "category:\"literal \"\"\"", "0 * XCATliteral \"" },
191 { "category:\" \"", "0 * XCAT " },
192 { "category:\"\"", "0 * XCAT" },
193 { "category:\"(unterminated)", "0 * XCAT(unterminated)" },
194 // Feature tests for curly double quotes:
195 { "“curly quotes”", "(curly@1 PHRASE 2 quotes@2)" },
196 // Feature tests for implicitly closing brackets:
197 { "(foo", "Zfoo@1" },
198 { "(foo XOR bar", "(Zfoo@1 XOR Zbar@2)" },
199 { "(foo XOR (bar AND baz)", "(Zfoo@1 XOR (Zbar@2 AND Zbaz@3))" },
200 { "(foo XOR (bar AND baz", "(Zfoo@1 XOR (Zbar@2 AND Zbaz@3))" },
201 // Slightly arbitrarily we accept mismatched quotes.
202 { "\"curly quotes”", "(curly@1 PHRASE 2 quotes@2)" },
203 { "“curly quotes\"", "(curly@1 PHRASE 2 quotes@2)" },
204 { "“curly quotes“", "(curly@1 PHRASE 2 quotes@2)" },
205 { "”curly quotes”", "(curly@1 PHRASE 2 quotes@2)" },
206 { "author:“orwell” title:“animal\"", "(Aorwell@1 OR XTanimal@2)" },
207 { "author:\"orwell” title:“animal”", "(Aorwell@1 OR XTanimal@2)" },
208 { "author:“milne, a.a.”", "(Amilne@1 PHRASE 3 Aa@2 PHRASE 3 Aa@3)" },
209 { "author:“milne, a.a.\"", "(Amilne@1 PHRASE 3 Aa@2 PHRASE 3 Aa@3)" },
210 { "author:\"milne a.a.”", "(Amilne@1 PHRASE 3 Aa@2 PHRASE 3 Aa@3)" },
211 { "“hello world” +python", "(Zpython@3 AND_MAYBE (hello@1 PHRASE 2 world@2))" },
212 { "unmatched“", "Zunmatch@1" },
213 { "unmatched”", "Zunmatch@1" },
214 { "unmatched “ ” ", "Zunmatch@1" },
215 { "unmatched \" ” ", "Zunmatch@1" },
216 { "unmatched “ \" ", "Zunmatch@1" },
217 { "hyphen-ated“ ", "(hyphen@1 PHRASE 2 ated@2)" },
218 { "hyphen-ated” ", "(hyphen@1 PHRASE 2 ated@2)" },
219 { "hyphen-ated“ ”", "(hyphen@1 PHRASE 2 ated@2)" },
220 { "hyphen-ated“ \"", "(hyphen@1 PHRASE 2 ated@2)" },
221 { "hyphen-ated\" ”", "(hyphen@1 PHRASE 2 ated@2)" },
222 { "“1.4”", "1.4@1" },
223 { "“1.\"", "1@1" },
224 { "\"A#.B.”", "(a#@1 PHRASE 2 b@2)" },
225 { "“ Xapian QueryParser” parses queries", "((xapian@1 PHRASE 2 queryparser@2) OR (Zpars@3 OR Zqueri@4))" },
226 { "“ xapian queryParser\" parses queries", "((xapian@1 PHRASE 2 queryparser@2) OR (Zpars@3 OR Zqueri@4))" },
227 { "beer NOT “orange juice”", "(Zbeer@1 AND_NOT (orange@2 PHRASE 2 juice@3))" },
228 { "“missing quote", "(missing@1 PHRASE 2 quote@2)" },
229 { "+“must have” optional", "((must@1 PHRASE 2 have@2) AND_MAYBE Zoption@3)" },
230 { "category:“Hello world”", "0 * XCAT:Hello world" },
231 { "category:“literal \"\"”", "0 * XCATliteral \"" },
232 { "category:“ ”", "0 * XCAT " },
233 { "category:\" \"", "0 * XCAT ”" },
234 { "category:\" ”", "0 * XCAT ”" },
235 { "category:“ \"", "0 * XCAT " },
236 { "category:“”", "0 * XCAT" },
237 { "category:\"\"", "0 * XCAT”" },
238 { "category:\"”", "0 * XCAT”" },
239 { "category:“\"", "0 * XCAT" },
240 { "category:“(unterminated)", "0 * XCAT(unterminated)" },
241 // Real world examples from tweakers.net:
242 { "Call to undefined function: imagecreate()", "(call@1 OR Zto@2 OR Zundefin@3 OR Zfunction@4 OR imagecreate@5)" },
243 { "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))" },
244 { "php date() nedelands", "((Zphp@1 OR date@2) OR Znedeland@3)" },
245 { "wget domein --http-user", "((Zwget@1 OR Zdomein@2) OR (http@3 PHRASE 2 user@4))" },
246 { "@home problemen", "(Zhome@1 OR Zproblemen@2)" },
247 { "'ipacsum'", "Zipacsum@1" },
248 { "canal + ", "Zcanal@1" },
249 { "/var/run/mysqld/mysqld.sock", "(var@1 PHRASE 5 run@2 PHRASE 5 mysqld@3 PHRASE 5 mysqld@4 PHRASE 5 sock@5)" },
250 { "\"QSI-161 drivers\"", "(qsi@1 PHRASE 3 161@2 PHRASE 3 drivers@3)" },
251 { "\"e-cube\" barebone", "((e@1 PHRASE 2 cube@2) OR Zbarebon@3)" },
252 { "\"./httpd: symbol not found: dlopen\"", "(httpd@1 PHRASE 5 symbol@2 PHRASE 5 not@3 PHRASE 5 found@4 PHRASE 5 dlopen@5)" },
253 { "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)" },
254 { "location.href = \"\"", "(location@1 PHRASE 2 href@2)" },
255 { "method=\"post\" action=\"\">", "((method@1 OR post@2) OR action@3)" },
256 { "behuizing 19\" inch", "((Zbehuiz@1 OR 19@2) OR inch@3)" },
257 { "19\" rack", "(19@1 OR rack@2)" },
258 { "3,5\" mainboard", "(3,5@1 OR mainboard@2)" },
259 { "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)" },
260 { "data error (clic redundancy check)", "((Zdata@1 OR Zerror@2) OR (Zclic@3 OR Zredund@4 OR Zcheck@5))" },
261 { "? mediaplayer 9\"", "(Zmediaplay@1 OR 9@2)" },
262 { "date(\"w\")", "(date@1 OR w@2)" },
263 { "Syntaxisfout (operator ontbreekt ASP", "(syntaxisfout@1 OR (Zoper@2 OR Zontbreekt@3 OR asp@4))" },
264 { "Request.ServerVariables(\"logon_user\")", "((request@1 PHRASE 2 servervariables@2) OR logon_user@3)" },
265 { "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))" },
266 { "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))" },
267 { "ip_masq_new(proto=TCP)", "((ip_masq_new@1 OR proto@2) OR tcp@3)" },
268 { "\"document.write(\"", "(document@1 PHRASE 2 write@2)" },
269 { "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))" },
270 { "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))" },
271 { "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))" },
272 { "\"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)" },
273 { "\") 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)" },
274 { "\"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)" },
275 { "$structure = imap_header($mbox, $tt);", "(((Zstructur@1 OR imap_header@2) OR Zmbox@3) OR Ztt@4)" },
276 { "\"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)" },
277 { "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))" },
278 { "ereg_replace(\"\\\\\",\"\\/\"", "ereg_replace@1" },
279 { "\\\\\"divx+geen+geluid\\\\\"", "(divx@1 PHRASE 3 geen@2 PHRASE 3 geluid@3)" },
280 { "lcase(\"string\")", "(lcase@1 OR string@2)" },
281 { "isEmpty( ) functie in visual basic", "(isempty@1 OR (Zfuncti@2 OR Zin@3 OR Zvisual@4 OR Zbasic@5))" },
282 { "*** stop: 0x0000001E (0xC0000005,0x00000000,0x00000000,0x00000000)", "((Zstop@1 OR 0x0000001e@2) OR 0xc0000005,0x00000000,0x00000000,0x00000000@3)" },
283 { "\"ctrl+v+c+a fout\"", "(ctrl@1 PHRASE 5 v@2 PHRASE 5 c@3 PHRASE 5 a@4 PHRASE 5 fout@5)" },
284 { "Server.CreateObject(\"ADODB.connection\")", "((server@1 PHRASE 2 createobject@2) OR (adodb@3 PHRASE 2 connection@4))" },
285 { "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))" },
286 { "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))" },
287 { "delphi CreateOleObject(\"MSXML2.DomDocument\")", "((Zdelphi@1 OR createoleobject@2) OR (msxml2@3 PHRASE 2 domdocument@4))" },
288 { "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))" },
289 { "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))" },
290 { "asp ' en \"", "(Zasp@1 OR Zen@2)" },
291 { "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)))" },
292 { "session_set_cookie_params(echo \"hoi\")", "((session_set_cookie_params@1 OR Zecho@2) OR hoi@3)" },
293 { "windows update werkt niet (windows se", "((Zwindow@1 OR Zupdat@2 OR Zwerkt@3 OR Zniet@4) OR (Zwindow@5 OR Zse@6))" },
294 { "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))" },
295 { "sony +(u20 u-20)", "((Zu20@2 OR (u@3 PHRASE 2 20@4)) AND_MAYBE Zsoni@1)" },
296 { "[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)))" },
297 { "directories lokaal php (uitlezen OR inladen)", "((Zdirectori@1 OR Zlokaal@2 OR Zphp@3) OR (Zuitlezen@4 OR Zinladen@5))" },
298 { "(multi pc modem)+ (line sync)", "((Zmulti@1 OR Zpc@2 OR Zmodem@3) OR (Zline@4 OR Zsync@5))" },
299 { "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))" },
300 { "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))" },
301 { "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))" },
302 { "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))" },
303 { "\"arp -s\" ip veranderen", "((arp@1 PHRASE 2 s@2) OR (Zip@3 OR Zveranderen@4))" },
304 { "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))" },
305 { "$datum = date(\"d-m-Y\");", "((Zdatum@1 OR date@2) OR (d@3 PHRASE 3 m@4 PHRASE 3 y@5))" },
306 { "\"'\" +asp", "Zasp@1" },
307 { "+session +[", "Zsession@1" },
308 { "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))" },
309 { "\"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)" },
310 { "\"+irq +veranderen +xp\"", "(irq@1 PHRASE 3 veranderen@2 PHRASE 3 xp@3)" },
311 { "\"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)" },
312 { "mkdir() failed (File exists) php", "(((mkdir@1 OR Zfail@2) OR (file@3 OR Zexist@4)) OR Zphp@5)" },
313 { "laatsteIndex(int n)", "(laatsteindex@1 OR (Zint@2 OR Zn@3))" },
314 { "\"line+in\" OR \"c8783\"", "((line@1 PHRASE 2 in@2) OR c8783@3)" },
315 { "if ($_POST['Submit'])", "(Zif@1 OR (_post@2 OR submit@3))" },
316 { "NEC DVD+-RW ND-1300A", "((nec@1 OR (dvd+@2 PHRASE 2 rw@3)) OR (nd@4 PHRASE 2 1300a@5))" },
317 { "*String not found* (*String not found*.)", "((string@1 OR Znot@2 OR found@3) OR (string@4 OR Znot@5 OR found@6))" },
318 { "MSI G4Ti4200-TD 128MB (GeForce4 Ti4200)", "(((msi@1 OR (g4ti4200@2 PHRASE 2 td@3)) OR 128mb@4) OR (geforce4@5 OR ti4200@6))" },
319 { "href=\"#\"", "href@1" },
320 { "Request.ServerVariables(\"REMOTE_USER\") javascript", "(((request@1 PHRASE 2 servervariables@2) OR remote_user@3) OR Zjavascript@4)" },
321 { "XF86Config(-4) waar", "((xf86config@1 OR 4@2) OR Zwaar@3)" },
322 { "Unknown (tag 2000)", "(unknown@1 OR (Ztag@2 OR 2000@3))" },
323 { "KT4V(MS-6712)", "(kt4v@1 OR (ms@2 PHRASE 2 6712@3))" },
324 { "scheduled+AND+nieuwsgroepen+AND+updaten", "((Zschedul@1 AND Znieuwsgroepen@2) AND Zupdaten@3)" },
325 { "137(netbios-ns)", "(137@1 OR (netbios@2 PHRASE 2 ns@3))" },
326 { "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))" },
327 { "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)" },
328 { "wat is code van \" teken", "((Zwat@1 OR Zis@2 OR Zcode@3 OR Zvan@4) OR teken@5)" },
329 { "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))" },
330 { "Permission denied (publickey,password,keyboard-interactive).", "((permission@1 OR Zdeni@2) OR ((Zpublickey@3 OR Zpassword@4) OR (keyboard@5 PHRASE 2 interactive@6)))" },
331 { "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))" },
332 { "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)" },
333 { "\"2020 NEAR zoom\"", "(2020@1 PHRASE 3 near@2 PHRASE 3 zoom@3)" },
334 { "setcookie(\"naam\",\"$user\");", "((setcookie@1 OR naam@2) OR user@3)" },
335 { "MSI 645 Ultra (MS-6547) Ver1", "(((msi@1 OR 645@2 OR ultra@3) OR (ms@4 PHRASE 2 6547@5)) OR ver1@6)" },
336 { "if ($HTTP", "(Zif@1 OR http@2)" },
337 { "data error(cyclic redundancy check)", "((Zdata@1 OR error@2) OR (Zcyclic@3 OR Zredund@4 OR Zcheck@5))" },
338 { "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)" },
339 { "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))" },
340 { "Call Shell(\"notepad.exe\",", "((call@1 OR shell@2) OR (notepad@3 PHRASE 2 exe@4))" },
341 { "2.5\" harddisk converter", "(2.5@1 OR (harddisk@2 PHRASE 2 converter@3))" },
342 { "creative labs \"dvd+rw\"", "((Zcreativ@1 OR Zlab@2) OR (dvd@3 PHRASE 2 rw@4))" },
343 { "\"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)" },
344 { "ati radeon \"driver cleaner", "((Zati@1 OR Zradeon@2) OR (driver@3 PHRASE 2 cleaner@4))" },
345 { "\"../\" path", "Zpath@1" },
346 { "(novell client) workstation only", "((Znovel@1 OR Zclient@2) OR (Zworkstat@3 OR Zonli@4))" },
347 { "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)" },
348 { "\"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)" },
349 { "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)" },
350 { "Forwarden van domeinnaam (naar HTTP adres)", "((forwarden@1 OR Zvan@2 OR Zdomeinnaam@3) OR (Znaar@4 OR http@5 OR Zadr@6))" },
351 { "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))" },
352 { "httpd (no pid file) not running", "((Zhttpd@1 OR (Zno@2 OR Zpid@3 OR Zfile@4)) OR (Znot@5 OR Zrun@6))" },
353 { "apache httpd (pid file) not running", "(((Zapach@1 OR Zhttpd@2) OR (Zpid@3 OR Zfile@4)) OR (Znot@5 OR Zrun@6))" },
354 { "Klasse is niet geregistreerd (Fout=80040154).", "((klasse@1 OR Zis@2 OR Zniet@3 OR Zgeregistreerd@4) OR (fout@5 OR 80040154@6))" },
355 { "\"dvd+r\" \"dvd-r\"", "((dvd@1 PHRASE 2 r@2) OR (dvd@3 PHRASE 2 r@4))" },
356 { "\"=\" tekens uit csvfile", "(Zteken@1 OR Zuit@2 OR Zcsvfile@3)" },
357 { "libc.so.6(GLIBC_2.3)", "((libc@1 PHRASE 3 so@2 PHRASE 3 6@3) OR glibc_2.3@4)" },
358 { "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))" },
359 { "(t-mobile) bereik", "((t@1 PHRASE 2 mobile@2) OR Zbereik@3)" },
360 { "error LNK2001: unresolved external symbol \"public", "((Zerror@1 OR lnk2001@2 OR Zunresolv@3 OR Zextern@4 OR Zsymbol@5) OR public@6)" },
361 { "patch linux exploit -p)", "((Zpatch@1 OR Zlinux@2 OR Zexploit@3) OR Zp@4)" },
362 { "MYD not found (Errcode: 2)", "((myd@1 OR Znot@2 OR Zfound@3) OR (errcode@4 OR 2@5))" },
363 { "ob_start(\"ob_gzhandler\"); file download", "((ob_start@1 OR ob_gzhandler@2) OR (Zfile@3 OR Zdownload@4))" },
364 { "ECS Elitegroup K7VZA (VIA VT8363/VT8363A)", "((ecs@1 OR elitegroup@2 OR k7vza@3) OR (via@4 OR (vt8363@5 PHRASE 2 vt8363a@6)))" },
365 { "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))" },
366 { "Javascript:history.go(-1)", "((javascript@1 PHRASE 3 history@2 PHRASE 3 go@3) OR 1@4)" },
367 { "java :) als icon", "(Zjava@1 OR (Zal@2 OR Zicon@3))" },
368 { "onmouseover=setPointer(this", "((onmouseover@1 OR setpointer@2) OR Zthis@3)" },
369 { "\" in vbscript", "(in@1 PHRASE 2 vbscript@2)" },
370 { "IRC (FAQ OR (hulp NEAR bij))", "(irc@1 OR (faq@2 OR (hulp@3 NEAR 11 bij@4)))" },
371 { "setProperty(\"McSquare\"+i, _xscale, _xscale++);", "((((setproperty@1 OR mcsquare@2) OR Zi@3) OR _xscale@4) OR _xscale++@5)" },
372 { "[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)" },
373 { "(php.ini) (memory_limit)", "((php@1 PHRASE 2 ini@2) OR Zmemory_limit@3)" },
374 { "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)" },
375 { "VXD NAVEX()@)", "(vxd@1 OR navex@2)" },
376 { "\"Iiyama AS4314UT 17\" \"", "(iiyama@1 PHRASE 3 as4314ut@2 PHRASE 3 17@3)" },
377 { "include (\"$id.html\");", "(Zinclud@1 OR (id@2 PHRASE 2 html@3))" },
378 { "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)" },
379 { "(program files\\common) opstarten", "((Zprogram@1 OR (files@2 PHRASE 2 common@3)) OR Zopstarten@4)" },
380 { "java \" string", "(Zjava@1 OR string@2)" },
381 { "+=", "" },
382 { "php +=", "Zphp@1" },
383 { "[php] ereg_replace(\".\"", "(Zphp@1 OR ereg_replace@2)" },
384 { "\"echo -e\" kleur", "((echo@1 PHRASE 2 e@2) OR Zkleur@3)" },
385 { "adobe premiere \"-1\"", "((Zadob@1 OR Zpremier@2) OR 1@3)" },
386 { "DVD brander \"+\" en \"-\"", "((dvd@1 OR Zbrander@2) OR Zen@3)" },
387 { "inspirion \"dvd+R\"", "(Zinspirion@1 OR (dvd@2 PHRASE 2 r@3))" },
388 { "asp 0x80040E14)", "(Zasp@1 OR 0x80040e14@2)" },
389 { "\"e-tech motorola router", "(e@1 PHRASE 4 tech@2 PHRASE 4 motorola@3 PHRASE 4 router@4)" },
390 { "bluetooth '1.3.2.19\"", "(Zbluetooth@1 OR 1.3.2.19@2)" },
391 { "ms +-connect", "(Zms@1 OR Zconnect@2)" },
392 { "php+print+\"", "(Zphp@1 OR print+@2)" },
393 { "athlon 1400 :welke videokaart\"", "((Zathlon@1 OR 1400@2) OR (Zwelk@3 OR videokaart@4))" },
394 { "+-dvd", "Zdvd@1" },
395 { "glftpd \"-new-\"", "(Zglftpd@1 OR new@2)" },
396 { "\"scandisk + dos5.0", "(scandisk@1 PHRASE 2 dos5.0@2)" },
397 { "socket\\(\\)", "socket@1" },
398 { "msn (e-tech) router", "((Zmsn@1 OR (e@2 PHRASE 2 tech@3)) OR Zrouter@4)" },
399 { "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)" },
400 { "\"CF+bluetooth\"", "(cf@1 PHRASE 2 bluetooth@2)" },
401 { "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))" },
402 { "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)" },
403 { "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))" },
404 { "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)))" },
405 { "\"muziek 2x zo snel\"\"", "(muziek@1 PHRASE 4 2x@2 PHRASE 4 zo@3 PHRASE 4 snel@4)" },
406 { "execCommand('inserthorizontalrule'", "(execcommand@1 OR Zinserthorizontalrul@2)" },
407 { "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))" },
408 { "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)" },
409 { "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))" },
410 { "'\"><br>bla</br>", "(br@1 PHRASE 3 bla@2 PHRASE 3 br@3)" },
411 { "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))" },
412 { "\"(P5A-b)\"", "(p5a@1 PHRASE 2 b@2)" },
413 { "(13,5 > 13) == no-go!", "((13,5@1 OR 13@2) OR (no@3 PHRASE 2 go@4))" },
414 { "eth not found \"ifconfig -a\"", "((Zeth@1 OR Znot@2 OR Zfound@3) OR (ifconfig@4 PHRASE 2 a@5))" },
415 { "<META NAME=\"ROBOTS", "((meta@1 OR name@2) OR robots@3)" },
416 { "lp0: using parport0 (interrupt-driven)", "((Zlp0@1 OR (Zuse@2 OR Zparport0@3)) OR (interrupt@4 PHRASE 2 driven@5))" },
417 { "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)" },
418 { "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))" },
419 { "header(\"Content Type: text/html\");", "((header@1 OR (content@2 OR type@3)) OR (text@4 PHRASE 2 html@5))" },
420 { "\"-RW\" \"+RW\"", "(rw@1 OR rw@2)" },
421 { "\"cresta digital answering machine", "(cresta@1 PHRASE 4 digital@2 PHRASE 4 answering@3 PHRASE 4 machine@4)" },
422 { "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)))" },
423 { "c++ fopen \"r+t\"", "((Zc++@1 OR Zfopen@2) OR (r@3 PHRASE 2 t@4))" },
424 { "c++ fopen (r+t)", "((Zc++@1 OR Zfopen@2) OR (Zr@3 OR Zt@4))" },
425 { "\"DVD+R\"", "(dvd@1 PHRASE 2 r@2)" },
426 { "Class.forName(\"jdbc.odbc.JdbcOdbcDriver\");", "((class@1 PHRASE 2 forname@2) OR (jdbc@3 PHRASE 3 odbc@4 PHRASE 3 jdbcodbcdriver@5))" },
427 { "perl(find.pl)", "(perl@1 OR (find@2 PHRASE 2 pl@3))" },
428 { "\"-5v\" voeding", "(5v@1 OR Zvoed@2)" },
429 { "\"-5v\" power supply", "(5v@1 OR (Zpower@2 OR Zsuppli@3))" },
430 { "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))" },
431 { "(error $2108) Borland", "((Zerror@1 OR 2108@2) OR borland@3)" },
432 { "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))" },
433 { "Elektrotechniek + \"hoe bevalt het?\"\"", "(elektrotechniek@1 OR (hoe@2 PHRASE 3 bevalt@3 PHRASE 3 het@4))" },
434 { "Shortcuts in menu (java", "((shortcuts@1 OR Zin@2 OR Zmenu@3) OR Zjava@4)" },
435 { "detonator+settings\"", "(Zdeton@1 OR settings@2)" },
436 { "(ez-bios) convert", "((ez@1 PHRASE 2 bios@2) OR Zconvert@3)" },
437 { "Sparkle 7100M4 64MB (GeForce4 MX440)", "((sparkle@1 OR 7100m4@2 OR 64mb@3) OR (geforce4@4 OR mx440@5))" },
438 { "freebsd \"boek OR newbie\"", "(Zfreebsd@1 OR (boek@2 PHRASE 3 or@3 PHRASE 3 newbie@4))" },
439 { "for (;;) c++", "(Zfor@1 OR Zc++@2)" },
440 { "1700+-2100+", "(1700+@1 PHRASE 2 2100+@2)" },
441 { "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))" },
442 { "NEC DV-5800B (Bul", "((nec@1 OR (dv@2 PHRASE 2 5800b@3)) OR bul@4)" },
443 { "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))" },
444 { "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))" },
445 { "'q ben\"", "(Zq@1 OR ben@2)" },
446 { "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))" },
447 { "\xc3\xb6ude onderdelen\"", "(Z\xc3\xb6ude@1 OR onderdelen@2)" },
448 { "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))" },
449 { "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))" },
450 { "cybercom \"dvd+r\"", "(Zcybercom@1 OR (dvd@2 PHRASE 2 r@3))" },
451 { "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))" },
452 { "relais +/-", "Zrelai@1" },
453 { "formules (slepen OR doortrekken) excel", "((Zformul@1 OR (Zslepen@2 OR Zdoortrekken@3)) OR Zexcel@4)" },
454 { "\"%English", "english@1" },
455 { "select max( mysql", "((Zselect@1 OR max@2) OR Zmysql@3)" },
456 { "leejow(saait", "(leejow@1 OR Zsaait@2)" },
457 { "'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))" },
458 { "K7T Turbo 2 (MS-6330)", "((k7t@1 OR turbo@2 OR 2@3) OR (ms@4 PHRASE 2 6330@5))" },
459 { "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))" },
460 { "\"cannot find -lz\"", "(cannot@1 PHRASE 3 find@2 PHRASE 3 lz@3)" },
461 { "undefined reference to `mysql_drop_db'\"", "((Zundefin@1 OR Zrefer@2 OR Zto@3) OR Zmysql_drop_db@4)" },
462 { "search form asp \"%'", "(Zsearch@1 OR Zform@2 OR Zasp@3)" },
463 { "(dvd+r) kwaliteit", "((Zdvd@1 OR Zr@2) OR Zkwaliteit@3)" },
464 { "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))" },
465 { "geluid (schokt OR hapert)", "(Zgeluid@1 OR (Zschokt@2 OR Zhapert@3))" },
466 { "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)" },
467 { "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))" },
468 { "(library qt-mt) not found", "((Zlibrari@1 OR (qt@2 PHRASE 2 mt@3)) OR (Znot@4 OR Zfound@5))" },
469 { "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))" },
470 { "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))" },
471 { "Titan TTC-D5TB(4/CU35)", "((titan@1 OR (ttc@2 PHRASE 2 d5tb@3)) OR (4@4 PHRASE 2 cu35@5))" },
472 { "[php] date( min", "((Zphp@1 OR date@2) OR Zmin@3)" },
473 { "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)" },
474 { "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))" },
475 { "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))" },
476 { "ati linux drivers (4.3.0)", "((Zati@1 OR Zlinux@2 OR Zdriver@3) OR 4.3.0@4)" },
477 { "ENCAPSED_AND_WHITESPACE", "encapsed_and_whitespace@1" },
478 { "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))" },
479 { "welke dvd \"+r\" media", "(((Zwelk@1 OR Zdvd@2) OR r@3) OR Zmedia@4)" },
480 { "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))" },
481 { "dvd +/-", "Zdvd@1" },
482 { "7vaxp +voltage mod\"", "(Zvoltag@2 AND_MAYBE (7vaxp@1 OR mod@3))" },
483 { "lpt port (SPP/EPP) is enabled", "(((Zlpt@1 OR Zport@2) OR (spp@3 PHRASE 2 epp@4)) OR (Zis@5 OR Zenabl@6))" },
484 { "getenv(\"HTTP_REFERER\")", "(getenv@1 OR http_referer@2)" },
485 { "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)" },
486 { "Exception number: c0000005 (access violation)", "((exception@1 OR Znumber@2 OR Zc0000005@3) OR (Zaccess@4 OR Zviolat@5))" },
487 { "header(\"Content-type:application/octetstream\");", "(header@1 OR (content@2 PHRASE 4 type@3 PHRASE 4 application@4 PHRASE 4 octetstream@5))" },
488 { "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)))" },
489 { "(001.part.met", "(001@1 PHRASE 3 part@2 PHRASE 3 met@3)" },
490 { "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))" },
491 { "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)" },
492 { "dvd \"+\" \"-\"", "Zdvd@1" },
493 { "bericht ( %)", "Zbericht@1" },
494 { "2500+ of 2600+ (niett OC)", "((2500+@1 OR Zof@2 OR 2600+@3) OR (Zniett@4 OR oc@5))" },
495 { "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))" },
496 { "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))" },
497 { "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))" },
498 { "GA-8IHXP(3.0)", "((ga@1 PHRASE 2 8ihxp@2) OR 3.0@3)" },
499 { "8IHXP(3.0)", "(8ihxp@1 OR 3.0@2)" },
500 { "na\xc2\xb7si (de ~ (m.))", "(Zna\xc2\xb7si@1 OR (Zde@2 OR Zm@3))" },
501 { "header(\"Content-Disposition: attachment;", "(header@1 OR (content@2 PHRASE 3 disposition@3 PHRASE 3 attachment@4))" },
502 { "\"header(\"Content-Disposition: attachment;\"", "((header@1 OR (content@2 PHRASE 2 disposition@3)) OR Zattach@4)" },
503 { "\"Beep -f\"", "(beep@1 PHRASE 2 f@2)" },
504 { "kraan NEAR (Elektrisch OR Electrisch)", "((Zkraan@1 OR near@2) OR (elektrisch@3 OR or@4 OR electrisch@5))" },
505 { "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))" },
506 { "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))" },
507 { "ac3 (0x2000) \"Dolby Laboratories,", "((Zac3@1 OR 0x2000@2) OR (dolby@3 PHRASE 2 laboratories@4))" },
508 { "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))" },
509 { "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))" },
510 { "Motion JPEG (MJPEG codec)", "((motion@1 OR jpeg@2) OR (mjpeg@3 OR Zcodec@4))" },
511 { ": zoomtext\"", "zoomtext@1" },
512 { "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))" },
513 { "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))" },
514 { "\"\"wa is fase\"", "(Zwa@1 OR Zis@2 OR fase@3)" },
515 { "<v:imagedata src=\"", "((v@1 PHRASE 2 imagedata@2) OR src@3)" },
516 { "system(play ringin.wav); ?>", "((system@1 OR Zplay@2) OR (ringin@3 PHRASE 2 wav@4))" },
517 { "\"perfect NEAR systems\"", "(perfect@1 PHRASE 3 near@2 PHRASE 3 systems@3)" },
518 { "LoadLibrary(\"mainta/gamex86.dll\") failed", "((loadlibrary@1 OR (mainta@2 PHRASE 3 gamex86@3 PHRASE 3 dll@4)) OR Zfail@5)" },
519 { "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)" },
520 { "secundaire IDE-controller (dubbele fifo)", "((Zsecundair@1 OR (ide@2 PHRASE 2 controller@3)) OR (Zdubbel@4 OR Zfifo@5))" },
521 { "\"Postal2+Explorer.exe\"", "(postal2@1 PHRASE 3 explorer@2 PHRASE 3 exe@3)" },
522 { "COUNT(*)", "count@1" },
523 { "Nuttige Windows progs (1/11)", "((nuttige@1 OR windows@2 OR Zprog@3) OR (1@4 PHRASE 2 11@5))" },
524 { "if(usercode==passcode==)", "((if@1 OR usercode@2) OR passcode@3)" },
525 { "lg 8160b (dvd+r)", "((Zlg@1 OR 8160b@2) OR (Zdvd@3 OR Zr@4))" },
526 { "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)))" },
527 { "'ipod pakt tags niet\"", "(Zipod@1 OR Zpakt@2 OR Ztag@3 OR niet@4)" },
528 { "\"DVD+/-R\"", "(dvd+@1 PHRASE 2 r@2)" },
529 { "\"DVD+R DVD-R\"", "(dvd@1 PHRASE 4 r@2 PHRASE 4 dvd@3 PHRASE 4 r@4)" },
530 { "php ;) in een array zetten", "(Zphp@1 OR (Zin@2 OR Zeen@3 OR Zarray@4 OR Zzetten@5))" },
531 { "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)" },
532 { "creative (soundblaster OR sb) 128", "((Zcreativ@1 OR (Zsoundblast@2 OR Zsb@3)) OR 128@4)" },
533 { "Can't open file: (errno: 145)", "((can't@1 OR Zopen@2 OR Zfile@3) OR (Zerrno@4 OR 145@5))" },
534 { "Formateren lukt niet(98,XP)", "(((formateren@1 OR Zlukt@2 OR niet@3) OR 98@4) OR xp@5)" },
535 { "access denied (java.io.", "((Zaccess@1 OR Zdeni@2) OR (java@3 PHRASE 2 io@4))" },
536 { "(access denied (java.io.)", "((Zaccess@1 OR Zdeni@2) OR (java@3 PHRASE 2 io@4))" },
537 { "wil niet installeren ( crc fouten)", "((Zwil@1 OR Zniet@2 OR Zinstalleren@3) OR (Zcrc@4 OR Zfouten@5))" },
538 { "(DVD+RW) brandsoftware meerdere", "((dvd@1 OR rw@2) OR (Zbrandsoftwar@3 OR Zmeerder@4))" },
539 { "(database OF databases) EN geheugen", "((Zdatabas@1 OR of@2 OR Zdatabas@3) OR (en@4 OR Zgeheugen@5))" },
540 { "(server 2003) winroute", "((Zserver@1 OR 2003@2) OR Zwinrout@3)" },
541 { "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))" },
542 { "(draadloos OR wireless) netwerk", "((Zdraadloo@1 OR Zwireless@2) OR Znetwerk@3)" },
543 { "localtime(time(NULL));", "((localtime@1 OR time@2) OR null@3)" },
544 { "ob_start(\"ob_gzhandler\");", "(ob_start@1 OR ob_gzhandler@2)" },
545 { "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))" },
546 { "COM+-gebeurtenissysteem", "(com+@1 PHRASE 2 gebeurtenissysteem@2)" },
547 { "rcpthosts (#5.7.1)", "(Zrcpthost@1 OR 5.7.1@2)" },
548 { "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))" },
549 { "window.open( scrollbar", "((window@1 PHRASE 2 open@2) OR Zscrollbar@3)" },
550 { "T68i truc ->", "(t68i@1 OR Ztruc@2)" },
551 { "T68i ->", "t68i@1" },
552 { "\"de lijn is bezet\"\"", "(de@1 PHRASE 4 lijn@2 PHRASE 4 is@3 PHRASE 4 bezet@4)" },
553 { "if (eregi(\"", "(Zif@1 OR eregi@2)" },
554 { "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))" },
555 { "execCommand(\"Paste\");", "(execcommand@1 OR paste@2)" },
556 { "\"-1 unread\"", "(1@1 PHRASE 2 unread@2)" },
557 { "\"www.historical-fire-engines", "(www@1 PHRASE 4 historical@2 PHRASE 4 fire@3 PHRASE 4 engines@4)" },
558 { "\"DVD+RW\" erase", "((dvd@1 PHRASE 2 rw@2) OR Zeras@3)" },
559 { "[showjekamer)", "Zshowjekam@1" },
560 { "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)" },
561 { "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))" },
562 { "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))" },
563 { "vervangen # \"/", "Zvervangen@1" },
564 { "vervangen # /\"", "Zvervangen@1" },
565 { "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)" },
566 { "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))" },
567 { "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))" },
568 { "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))" },
569 { "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))" },
570 { "mag mijn waarschuwing nu weg ? ;)", "(Zmag@1 OR Zmijn@2 OR Zwaarschuw@3 OR Znu@4 OR Zweg@5)" },
571 { "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))" },
572 { "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))" },
573 { "(browser 19) citrix", "((Zbrowser@1 OR 19@2) OR Zcitrix@3)" },
574 { "preg_replace (.*?)", "Zpreg_replac@1" },
575 { "formule excel #naam\"?\"", "((Zformul@1 OR Zexcel@2) OR naam@3)" },
576 { "->", "" },
577 { "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))" },
578 { "<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))" },
579 { "\"rpm -e httpd\"", "(rpm@1 PHRASE 3 e@2 PHRASE 3 httpd@3)" },
580 { "automatisch op All Flis (*.*)", "(Zautomatisch@1 OR Zop@2 OR all@3 OR flis@4)" },
581 { "(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))" },
582 { "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))" },
583 { "\"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)" },
584 { "(./) chmod.sh", "(chmod@1 PHRASE 2 sh@2)" },
585 { "document.write(ssg(\" html", "(((document@1 PHRASE 2 write@2) OR ssg@3) OR html@4)" },
586 { "superstack \"mac+adressen\"", "(Zsuperstack@1 OR (mac@2 PHRASE 2 adressen@3))" },
587 { "IIS getenv(REMOTE_HOST)_", "(((iis@1 OR getenv@2) OR remote_host@3) OR _@4)" },
588 { "IIS en getenv(REMOTE_HOST)", "((iis@1 OR Zen@2 OR getenv@3) OR remote_host@4)" },
589 { "php getenv(\"HTTP_REFERER\")", "((Zphp@1 OR getenv@2) OR http_referer@3)" },
590 { "nec+-1300", "(nec+@1 PHRASE 2 1300@2)" },
591 { "smbpasswd script \"-s\"", "((Zsmbpasswd@1 OR Zscript@2) OR s@3)" },
592 { "leestekens \" \xc3\xb6 \xc3\xab", "(Zleesteken@1 OR (\xc3\xb6@2 PHRASE 2 \xc3\xab@3))" },
593 { "freesco and (all seeing eye)", "((Zfreesco@1 OR Zand@2) OR (Zall@3 OR Zsee@4 OR Zeye@5))" },
594 { "('all seeing eye') and freesco", "((Zall@1 OR Zsee@2 OR Zeye@3) OR (Zand@4 OR Zfreesco@5))" },
595 { "\"[......\"", "" },
596 { "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)))" },
597 { "gegevensfout (cyclishe redundantiecontrole)", "(Zgegevensfout@1 OR (Zcyclish@2 OR Zredundantiecontrol@3))" },
598 { "firmware versie waar NEC\"", "(Zfirmwar@1 OR Zversi@2 OR Zwaar@3 OR nec@4)" },
599 { "nu.nl \"-1\"", "((nu@1 PHRASE 2 nl@2) OR 1@3)" },
600 { "provider+-webspace", "(provider+@1 PHRASE 2 webspace@2)" },
601 { "verschil \"dvd+rw\" \"dvd-rw\"", "((Zverschil@1 OR (dvd@2 PHRASE 2 rw@3)) OR (dvd@4 PHRASE 2 rw@5))" },
602 { "(dhcp client) + hangt", "((Zdhcp@1 OR Zclient@2) OR Zhangt@3)" },
603 { "MSI 875P Neo-FIS2R (Intel 875P)", "(((msi@1 OR 875p@2) OR (neo@3 PHRASE 2 fis2r@4)) OR (intel@5 OR 875p@6))" },
604 { "voeding passief gekoeld\"", "(Zvoed@1 OR Zpassief@2 OR gekoeld@3)" },
605 { "if (mysql_num_rows($resultaat)==1)", "(((Zif@1 OR mysql_num_rows@2) OR Zresultaat@3) OR 1@4)" },
606 { "Server.CreateObject(\"Persits.Upload.1\")", "((server@1 PHRASE 2 createobject@2) OR (persits@3 PHRASE 3 upload@4 PHRASE 3 1@5))" },
607 { "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))" },
608 { "if (cod>9999999", "(Zif@1 OR (cod@2 OR 9999999@3))" },
609 { "\"rm -rf /bin/laden\"", "(rm@1 PHRASE 4 rf@2 PHRASE 4 bin@3 PHRASE 4 laden@4)" },
610 { "\">>> 0) & 0xFF\"", "(0@1 PHRASE 2 0xff@2)" },
611 { "<!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))" },
612 { "<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)" },
613 { "linux humor :)", "(Zlinux@1 OR Zhumor@2)" },
614 { "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))" },
615 { "remote_smtp defer (-44)", "((Zremote_smtp@1 OR Zdefer@2) OR 44@3)" },
616 { "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)" },
617 { "Koper + amoniak (NH2", "((koper@1 OR Zamoniak@2) OR nh2@3)" },
618 { "nec dvd -/+r", "((Znec@1 OR Zdvd@2) AND_NOT Zr@3)" },
619 { "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)" },
620 { "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))" },
621 { "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))" },
622 { "\"~\" + \"c:\\\"", "Zc@1" },
623 { "mysql count(*)", "(Zmysql@1 OR count@2)" },
624 { "for %f in (*.*) do", "((Zfor@1 OR (Zf@2 OR Zin@3)) OR Zdo@4)" },
625 { "raar \"~\" bestand", "(Zraar@1 OR Zbestand@2)" },
626 { "NEC DVD +-R/RW 1300", "(((nec@1 OR dvd@2) OR (r@3 PHRASE 2 rw@4)) OR 1300@5)" },
627 { "approved (ref: 38446-263)", "(Zapprov@1 OR (Zref@2 OR (38446@3 PHRASE 2 263@4)))" },
628 { "GA-7VRXP(2.0)", "((ga@1 PHRASE 2 7vrxp@2) OR 2.0@3)" },
629 { "~ Could not retrieve directory listing for \"/\"", "(could@1 OR Znot@2 OR Zretriev@3 OR Zdirectori@4 OR Zlist@5 OR Zfor@6)" },
630 { "asp CreateObject(\"Word.Document\")", "((Zasp@1 OR createobject@2) OR (word@3 PHRASE 2 document@4))" },
631 { "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))" },
632 { "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)))" },
633 { "parent.document.getElementById(\\\"leftmenu\\\").cols", "(((parent@1 PHRASE 3 document@2 PHRASE 3 getelementbyid@3) OR leftmenu@4) OR Zcol@5)" },
634 { "<% if not isEmpty(Request.QueryString) then", "(((Zif@1 OR Znot@2 OR isempty@3) OR (request@4 PHRASE 2 querystring@5)) OR Zthen@6)" },
635 { "Active Desktop (Hier issie)", "((active@1 OR desktop@2) OR (hier@3 OR Zissi@4))" },
636 { "Asus A7V8X (LAN + Sound)", "((asus@1 OR a7v8x@2) OR (lan@3 OR sound@4))" },
637 { "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)" },
638 { "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)" },
639 { "session_register(\"login\");", "(session_register@1 OR login@2)" },
640 { "\"kylix+ndmb\"", "(kylix@1 PHRASE 2 ndmb@2)" },
641 { "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))" },
642 { "If ($_SESSION[\"Login\"] == 1)", "(if@1 OR ((_session@2 OR login@3) OR 1@4))" },
643 { "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))" },
644 { "ASRock K7VT2 (incl. LAN)", "((asrock@1 OR k7vt2@2) OR (Zincl@3 OR lan@4))" },
645 { "+windows98 +(geen communicatie) +ie5", "((Zwindows98@1 AND (Zgeen@2 OR Zcommunicati@3)) AND Zie5@4)" },
646 { "\"xterm -fn\"", "(xterm@1 PHRASE 2 fn@2)" },
647 { "IRQL_NOT_LESS_OR_EQUAL", "irql_not_less_or_equal@1" },
648 { "access query \"NOT IN\"", "((Zaccess@1 OR Zqueri@2) OR (not@3 PHRASE 2 in@4))" },
649 { "\"phrase one \"phrase two\"", "((phrase@1 PHRASE 2 one@2) OR (Zphrase@3 OR two@4))" },
650 { "NEAR 207 46 249 27", "(near@1 OR 207@2 OR 46@3 OR 249@4 OR 27@5)" },
651 { "- NEAR 12V voeding", "(near@1 OR 12v@2 OR Zvoed@3)" },
652 { "waarom \"~\" in directorynaam", "(Zwaarom@1 OR (Zin@2 OR Zdirectorynaam@3))" },
653 { "cd'r NEAR toebehoren", "(cd'r@1 NEAR 11 toebehoren@2)" },
654 { "site:1 site:2", "0 * (H1 OR H2)" },
655 { "site:1 site2:2", "0 * (H1 AND J2)" },
656 { "site:1 site:2 site2:2", "0 * ((H1 OR H2) AND J2)" },
657 { "site:1 OR site:2", "(0 * H1 OR 0 * H2)" },
658 { "site:1 AND site:2", "(0 * H1 AND 0 * H2)" },
659 { "foo AND site:2", "(Zfoo@1 AND 0 * H2)" },
660 // Non-exclusive boolean prefixes feature tests (ticket#402):
661 { "category:1 category:2", "0 * (XCAT1 AND XCAT2)" },
662 { "category:1 site2:2", "0 * (J2 AND XCAT1)" },
663 { "category:1 category:2 site2:2", "0 * (J2 AND (XCAT1 AND XCAT2))" },
664 { "category:1 OR category:2", "(0 * XCAT1 OR 0 * XCAT2)" },
665 { "category:1 AND category:2", "(0 * XCAT1 AND 0 * XCAT2)" },
666 { "foo AND category:2", "(Zfoo@1 AND 0 * XCAT2)" },
667 // Regression test for combining multiple non-exclusive prefixes, fixed in
668 // 1.2.22 and 1.3.4.
669 { "category:1 dogegory:2", "0 * (XCAT1 AND XDOG2)" },
670 { "A site:1 site:2", "(a@1 FILTER (H1 OR H2))" },
671 #if 0
672 { "A (site:1 OR site:2)", "(a@1 FILTER (H1 OR H2))" },
673 #endif
674 { "A site:1 site2:2", "(a@1 FILTER (H1 AND J2))" },
675 { "A site:1 site:2 site2:2", "(a@1 FILTER ((H1 OR H2) AND J2))" },
676 #if 0
677 { "A site:1 OR site:2", "(a@1 FILTER (H1 OR H2))" },
678 #endif
679 { "A site:1 AND site:2", "((a@1 FILTER H1) AND 0 * H2)" },
680 { "site:xapian.org OR site:www.xapian.org", "(0 * Hxapian.org OR 0 * Hwww.xapian.org)" },
681 { "site:xapian.org site:www.xapian.org", "0 * (Hxapian.org OR Hwww.xapian.org)" },
682 { "site:xapian.org AND site:www.xapian.org", "(0 * Hxapian.org AND 0 * Hwww.xapian.org)" },
683 { "Xapian site:xapian.org site:www.xapian.org", "(xapian@1 FILTER (Hxapian.org OR Hwww.xapian.org))" },
684 { "author:richard author:olly writer:charlie", "(ZArichard@1 OR ZAolli@2 OR ZAcharli@3)"},
685 { "author:richard NEAR title:book", "(Arichard@1 NEAR 11 XTbook@2)"},
686 { "authortitle:richard NEAR title:book", "((Arichard@1 OR XTrichard@1) NEAR 11 XTbook@2)" },
687 { "multisite:xapian.org", "0 * (Hxapian.org OR Jxapian.org)"},
688 { "authortitle:richard", "(ZArichard@1 OR ZXTrichard@1)"},
689 { "multisite:xapian.org site:www.xapian.org author:richard authortitle:richard", "((ZArichard@1 OR (ZArichard@2 OR ZXTrichard@2)) FILTER (Hwww.xapian.org AND (Hxapian.org OR Jxapian.org)))" },
690 { "authortitle:richard-boulton", "((Arichard@1 PHRASE 2 Aboulton@2) OR (XTrichard@1 PHRASE 2 XTboulton@2))"},
691 { "authortitle:\"richard boulton\"", "((Arichard@1 PHRASE 2 Aboulton@2) OR (XTrichard@1 PHRASE 2 XTboulton@2))"},
692 // Test FLAG_CJK_NGRAM isn't on by default:
693 { "久有归天愿", "Z久有归天愿@1" },
694 { NULL, "CJK" }, // Enable FLAG_CJK_NGRAM
695 // Test non-CJK queries still parse the same:
696 { "gtk+ -gnome", "(Zgtk+@1 AND_NOT Zgnome@2)" },
697 { "“curly quotes”", "(curly@1 PHRASE 2 quotes@2)" },
698 // Test n-gram generation:
699 { "久有归天愿", "(久@1 AND 久有@1 AND 有@1 AND 有归@1 AND 归@1 AND 归天@1 AND 天@1 AND 天愿@1 AND 愿@1)" },
700 { "久有 归天愿", "((久@1 AND 久有@1 AND 有@1) OR (归@2 AND 归天@2 AND 天@2 AND 天愿@2 AND 愿@2))" },
701 { "久有!归天愿", "((久@1 AND 久有@1 AND 有@1) OR (归@2 AND 归天@2 AND 天@2 AND 天愿@2 AND 愿@2))" },
702 { "title:久有 归 天愿", "(((XT久@1 AND XT久有@1 AND XT有@1) OR 归@2) OR (天@3 AND 天愿@3 AND 愿@3))" },
703 { "h众ello万众", "(((Zh@1 OR 众@2) OR Zello@3) OR (万@4 AND 万众@4 AND 众@4))" },
704 { "世(の中)TEST_tm", "((世@1 OR (の@2 AND の中@2 AND 中@2)) OR test_tm@3)" },
705 { "다녀 AND 와야", "((다@1 AND 다녀@1 AND 녀@1) AND (와@2 AND 와야@2 AND 야@2))" },
706 { "authortitle:학술 OR 연구를", "((A학@1 AND XT학@1 AND A학술@1 AND XT학술@1 AND A술@1 AND XT술@1) OR (연@2 AND 연구@2 AND 구@2 AND 구를@2 AND 를@2))" },
707 // FIXME: These should really filter by bigrams to accelerate:
708 { "\"久有归\"", "(久@1 PHRASE 3 有@1 PHRASE 3 归@1)" },
709 { "\"久有test归\"", "(久@1 PHRASE 4 有@1 PHRASE 4 test@2 PHRASE 4 归@3)" },
710 // FIXME: this should work: { "久 NEAR 有", "(久@1 NEAR 11 有@2)" },
711 { NULL, NULL }
714 static bool test_queryparser1()
716 Xapian::QueryParser queryparser;
717 queryparser.set_stemmer(Xapian::Stem("english"));
718 queryparser.set_stemming_strategy(Xapian::QueryParser::STEM_SOME);
719 queryparser.add_prefix("author", "A");
720 queryparser.add_prefix("writer", "A");
721 queryparser.add_prefix("title", "XT");
722 queryparser.add_prefix("subject", "XT");
723 queryparser.add_prefix("authortitle", "A");
724 queryparser.add_prefix("authortitle", "XT");
725 queryparser.add_boolean_prefix("site", "H");
726 queryparser.add_boolean_prefix("site2", "J");
727 queryparser.add_boolean_prefix("multisite", "H");
728 queryparser.add_boolean_prefix("multisite", "J");
729 queryparser.add_boolean_prefix("category", "XCAT", false);
730 queryparser.add_boolean_prefix("dogegory", "XDOG", false);
731 TEST_EXCEPTION(Xapian::InvalidOperationError,
732 queryparser.add_boolean_prefix("authortitle", "B");
734 TEST_EXCEPTION(Xapian::InvalidOperationError,
735 queryparser.add_prefix("multisite", "B");
737 unsigned flags = queryparser.FLAG_DEFAULT;
738 for (const test *p = test_or_queries; ; ++p) {
739 if (!p->query) {
740 if (!p->expect) break;
741 if (strcmp(p->expect, "CJK") == 0) {
742 flags = queryparser.FLAG_DEFAULT|queryparser.FLAG_CJK_NGRAM;
743 continue;
745 FAIL_TEST(string("Unknown flag code: ") + p->expect);
747 string expect, parsed;
748 if (p->expect)
749 expect = p->expect;
750 else
751 expect = "parse error";
752 try {
753 Xapian::Query qobj = queryparser.parse_query(p->query, flags);
754 parsed = qobj.get_description();
755 expect = string("Query(") + expect + ')';
756 } catch (const Xapian::QueryParserError &e) {
757 parsed = e.get_msg();
758 } catch (const Xapian::Error &e) {
759 parsed = e.get_description();
760 } catch (...) {
761 parsed = "Unknown exception!";
763 tout << "Query: " << p->query << '\n';
764 TEST_STRINGS_EQUAL(parsed, expect);
766 return true;
769 static const test test_and_queries[] = {
770 { "internet explorer title:(http www)", "((Zinternet@1 AND Zexplor@2) AND (ZXThttp@3 AND ZXTwww@4))" },
771 // Regression test for bug in 0.9.2 and earlier - this would give
772 // (two@2 AND_MAYBE (one@1 AND three@3))
773 { "one +two three", "((Zone@1 AND Ztwo@2) AND Zthree@3)" },
774 { "hello -title:\"hello world\"", "(Zhello@1 AND_NOT (XThello@2 PHRASE 2 XTworld@3))" },
775 // Regression test for bug fixed in 1.0.4 - the '-' would be ignored there
776 // because the whitespace after the '"' wasn't noticed.
777 { "\"hello world\" -C++", "((hello@1 PHRASE 2 world@2) AND_NOT c++@3)" },
778 // Regression tests for bug fixed in 1.0.4 - queries with only boolean
779 // filter and HATE terms weren't accepted.
780 { "-cup site:world", "(0 * Hworld AND_NOT Zcup@1)" },
781 { "site:world -cup", "(0 * Hworld AND_NOT Zcup@1)" },
782 // Regression test for bug fixed in 1.0.4 - the KET token for ')' was lost.
783 { "(site:world) -cup", "(0 * Hworld AND_NOT Zcup@1)" },
784 // Regression test for bug fixed in 1.0.4 - a boolean filter term between
785 // probabilistic terms caused a parse error (probably broken during the
786 // addition of synonym support in 1.0.2).
787 { "foo site:xapian.org bar", "((Zfoo@1 AND Zbar@2) FILTER Hxapian.org)" },
788 // Add coverage for other cases similar to the above.
789 { "a b site:xapian.org", "((Za@1 AND Zb@2) FILTER Hxapian.org)" },
790 { "site:xapian.org a b", "((Za@1 AND Zb@2) FILTER Hxapian.org)" },
791 { NULL, "CJK" }, // Enable FLAG_CJK_NGRAM
792 // Test n-gram generation:
793 { "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)))" },
794 { "洛伊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)" },
795 { NULL, NULL }
798 // With default_op = OP_AND.
799 static bool test_qp_default_op1()
801 Xapian::QueryParser queryparser;
802 queryparser.set_stemmer(Xapian::Stem("english"));
803 queryparser.set_stemming_strategy(Xapian::QueryParser::STEM_SOME);
804 queryparser.add_prefix("author", "A");
805 queryparser.add_prefix("title", "XT");
806 queryparser.add_prefix("subject", "XT");
807 queryparser.add_boolean_prefix("site", "H");
808 queryparser.set_default_op(Xapian::Query::OP_AND);
809 unsigned flags = queryparser.FLAG_DEFAULT;
810 for (const test *p = test_and_queries; ; ++p) {
811 if (!p->query) {
812 if (!p->expect) break;
813 if (strcmp(p->expect, "CJK") == 0) {
814 flags = queryparser.FLAG_DEFAULT|queryparser.FLAG_CJK_NGRAM;
815 continue;
817 FAIL_TEST(string("Unknown flag code: ") + p->expect);
819 string expect, parsed;
820 if (p->expect)
821 expect = p->expect;
822 else
823 expect = "parse error";
824 try {
825 Xapian::Query qobj = queryparser.parse_query(p->query, flags);
826 parsed = qobj.get_description();
827 expect = string("Query(") + expect + ')';
828 } catch (const Xapian::QueryParserError &e) {
829 parsed = e.get_msg();
830 } catch (const Xapian::Error &e) {
831 parsed = e.get_description();
832 } catch (...) {
833 parsed = "Unknown exception!";
835 tout << "Query: " << p->query << '\n';
836 TEST_STRINGS_EQUAL(parsed, expect);
838 return true;
841 // Feature test for specify the default prefix (new in Xapian 1.0.0).
842 static bool test_qp_default_prefix1()
844 Xapian::QueryParser qp;
845 qp.set_stemmer(Xapian::Stem("english"));
846 qp.set_stemming_strategy(Xapian::QueryParser::STEM_SOME);
847 qp.add_prefix("title", "XT");
849 Xapian::Query qobj;
850 qobj = qp.parse_query("hello world", 0, "A");
851 TEST_STRINGS_EQUAL(qobj.get_description(), "Query((ZAhello@1 OR ZAworld@2))");
852 qobj = qp.parse_query("me title:stuff", 0, "A");
853 TEST_STRINGS_EQUAL(qobj.get_description(), "Query((ZAme@1 OR ZXTstuff@2))");
854 qobj = qp.parse_query("title:(stuff) me", Xapian::QueryParser::FLAG_BOOLEAN, "A");
855 TEST_STRINGS_EQUAL(qobj.get_description(), "Query((ZXTstuff@1 OR ZAme@2))");
856 qobj = qp.parse_query("英国 title:文森hello", qp.FLAG_CJK_NGRAM, "A");
857 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))");
858 return true;
861 // Feature test for setting the default prefix with add_prefix()
862 // (new in Xapian 1.0.3).
863 static bool test_qp_default_prefix2()
865 Xapian::QueryParser qp;
866 qp.set_stemmer(Xapian::Stem("english"));
867 qp.set_stemming_strategy(Xapian::QueryParser::STEM_SOME);
869 // test that default prefixes can only be set with add_prefix().
870 TEST_EXCEPTION(Xapian::UnimplementedError,
871 qp.add_boolean_prefix("", "B");
874 qp.add_prefix("title", "XT");
875 qp.add_prefix("", "A");
877 Xapian::Query qobj;
878 qobj = qp.parse_query("hello world", 0);
879 TEST_STRINGS_EQUAL(qobj.get_description(), "Query((ZAhello@1 OR ZAworld@2))");
880 qobj = qp.parse_query("me title:stuff", 0);
881 TEST_STRINGS_EQUAL(qobj.get_description(), "Query((ZAme@1 OR ZXTstuff@2))");
882 qobj = qp.parse_query("title:(stuff) me", Xapian::QueryParser::FLAG_BOOLEAN);
883 TEST_STRINGS_EQUAL(qobj.get_description(), "Query((ZXTstuff@1 OR ZAme@2))");
885 qobj = qp.parse_query("hello world", 0, "B");
886 TEST_STRINGS_EQUAL(qobj.get_description(), "Query((ZBhello@1 OR ZBworld@2))");
887 qobj = qp.parse_query("me title:stuff", 0, "B");
888 TEST_STRINGS_EQUAL(qobj.get_description(), "Query((ZBme@1 OR ZXTstuff@2))");
889 qobj = qp.parse_query("title:(stuff) me", Xapian::QueryParser::FLAG_BOOLEAN, "B");
890 TEST_STRINGS_EQUAL(qobj.get_description(), "Query((ZXTstuff@1 OR ZBme@2))");
892 qp.add_prefix("", "B");
893 qobj = qp.parse_query("me-us title:(stuff) me", Xapian::QueryParser::FLAG_BOOLEAN);
894 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)))");
895 qobj = qp.parse_query("me-us title:(stuff) me", Xapian::QueryParser::FLAG_BOOLEAN, "C");
896 TEST_STRINGS_EQUAL(qobj.get_description(), "Query((((Cme@1 PHRASE 2 Cus@2) OR ZXTstuff@3) OR ZCme@4))");
898 qobj = qp.parse_query("me-us title:\"not-me\"", Xapian::QueryParser::FLAG_PHRASE);
899 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)))");
900 return true;
903 // Test query with odd characters in.
904 static bool test_qp_odd_chars1()
906 Xapian::QueryParser qp;
907 string query("\x01weird\x00stuff\x7f", 13);
908 Xapian::Query qobj = qp.parse_query(query);
909 tout << "Query: " << query << '\n';
910 TEST_STRINGS_EQUAL(qobj.get_description(), "Query((weird@1 OR stuff@2))"); // FIXME: should these be stemmed?
911 return true;
914 // Test right truncation.
915 static bool test_qp_flag_wildcard1()
917 #ifndef XAPIAN_HAS_INMEMORY_BACKEND
918 SKIP_TEST("Testcase requires the InMemory backend which is disabled");
919 #else
920 Xapian::WritableDatabase db(string(), Xapian::DB_BACKEND_INMEMORY);
921 Xapian::Document doc;
922 doc.add_term("abc");
923 doc.add_term("main");
924 doc.add_term("muscat");
925 doc.add_term("muscle");
926 doc.add_term("musclebound");
927 doc.add_term("muscular");
928 doc.add_term("mutton");
929 db.add_document(doc);
930 Xapian::QueryParser qp;
931 qp.set_database(db);
932 Xapian::Query qobj = qp.parse_query("ab*", Xapian::QueryParser::FLAG_WILDCARD);
933 TEST_STRINGS_EQUAL(qobj.get_description(), "Query((SYNONYM WILDCARD OR ab))");
934 qobj = qp.parse_query("muscle*", Xapian::QueryParser::FLAG_WILDCARD);
935 TEST_STRINGS_EQUAL(qobj.get_description(), "Query((SYNONYM WILDCARD OR muscle))");
936 qobj = qp.parse_query("meat*", Xapian::QueryParser::FLAG_WILDCARD);
937 TEST_STRINGS_EQUAL(qobj.get_description(), "Query((SYNONYM WILDCARD OR meat))");
938 qobj = qp.parse_query("musc*", Xapian::QueryParser::FLAG_WILDCARD);
939 TEST_STRINGS_EQUAL(qobj.get_description(), "Query((SYNONYM WILDCARD OR musc))");
940 qobj = qp.parse_query("mutt*", Xapian::QueryParser::FLAG_WILDCARD);
941 TEST_STRINGS_EQUAL(qobj.get_description(), "Query((SYNONYM WILDCARD OR mutt))");
942 // Regression test (we weren't lowercasing terms before checking if they
943 // were in the database or not):
944 qobj = qp.parse_query("mUTTON++");
945 TEST_STRINGS_EQUAL(qobj.get_description(), "Query(mutton@1)");
946 // Regression test: check that wildcards work with +terms.
947 unsigned flags = Xapian::QueryParser::FLAG_WILDCARD |
948 Xapian::QueryParser::FLAG_LOVEHATE;
949 qobj = qp.parse_query("+mai* main", flags);
950 TEST_STRINGS_EQUAL(qobj.get_description(), "Query(((SYNONYM WILDCARD OR mai) AND_MAYBE main@2))");
951 // Regression test (if we had a +term which was a wildcard and wasn't
952 // present, the query could still match documents).
953 qobj = qp.parse_query("foo* main", flags);
954 TEST_STRINGS_EQUAL(qobj.get_description(), "Query(((SYNONYM WILDCARD OR foo) OR main@2))");
955 qobj = qp.parse_query("main foo*", flags);
956 TEST_STRINGS_EQUAL(qobj.get_description(), "Query((main@1 OR (SYNONYM WILDCARD OR foo)))");
957 qobj = qp.parse_query("+foo* main", flags);
958 TEST_STRINGS_EQUAL(qobj.get_description(), "Query(((SYNONYM WILDCARD OR foo) AND_MAYBE main@2))");
959 qobj = qp.parse_query("main +foo*", flags);
960 TEST_STRINGS_EQUAL(qobj.get_description(), "Query(((SYNONYM WILDCARD OR foo) AND_MAYBE main@1))");
961 qobj = qp.parse_query("foo* +main", flags);
962 TEST_STRINGS_EQUAL(qobj.get_description(), "Query((main@2 AND_MAYBE (SYNONYM WILDCARD OR foo)))");
963 qobj = qp.parse_query("+main foo*", flags);
964 TEST_STRINGS_EQUAL(qobj.get_description(), "Query((main@1 AND_MAYBE (SYNONYM WILDCARD OR foo)))");
965 qobj = qp.parse_query("+foo* +main", flags);
966 TEST_STRINGS_EQUAL(qobj.get_description(), "Query(((SYNONYM WILDCARD OR foo) AND main@2))");
967 qobj = qp.parse_query("+main +foo*", flags);
968 TEST_STRINGS_EQUAL(qobj.get_description(), "Query((main@1 AND (SYNONYM WILDCARD OR foo)))");
969 qobj = qp.parse_query("foo* mai", flags);
970 TEST_STRINGS_EQUAL(qobj.get_description(), "Query(((SYNONYM WILDCARD OR foo) OR mai@2))");
971 qobj = qp.parse_query("mai foo*", flags);
972 TEST_STRINGS_EQUAL(qobj.get_description(), "Query((mai@1 OR (SYNONYM WILDCARD OR foo)))");
973 qobj = qp.parse_query("+foo* mai", flags);
974 TEST_STRINGS_EQUAL(qobj.get_description(), "Query(((SYNONYM WILDCARD OR foo) AND_MAYBE mai@2))");
975 qobj = qp.parse_query("mai +foo*", flags);
976 TEST_STRINGS_EQUAL(qobj.get_description(), "Query(((SYNONYM WILDCARD OR foo) AND_MAYBE mai@1))");
977 qobj = qp.parse_query("foo* +mai", flags);
978 TEST_STRINGS_EQUAL(qobj.get_description(), "Query((mai@2 AND_MAYBE (SYNONYM WILDCARD OR foo)))");
979 qobj = qp.parse_query("+mai foo*", flags);
980 TEST_STRINGS_EQUAL(qobj.get_description(), "Query((mai@1 AND_MAYBE (SYNONYM WILDCARD OR foo)))");
981 qobj = qp.parse_query("+foo* +mai", flags);
982 TEST_STRINGS_EQUAL(qobj.get_description(), "Query(((SYNONYM WILDCARD OR foo) AND mai@2))");
983 qobj = qp.parse_query("+mai +foo*", flags);
984 TEST_STRINGS_EQUAL(qobj.get_description(), "Query((mai@1 AND (SYNONYM WILDCARD OR foo)))");
985 qobj = qp.parse_query("-foo* main", flags);
986 TEST_STRINGS_EQUAL(qobj.get_description(), "Query((main@2 AND_NOT (SYNONYM WILDCARD OR foo)))");
987 qobj = qp.parse_query("main -foo*", flags);
988 TEST_STRINGS_EQUAL(qobj.get_description(), "Query((main@1 AND_NOT (SYNONYM WILDCARD OR foo)))");
989 qobj = qp.parse_query("main -foo* -bar", flags);
990 TEST_STRINGS_EQUAL(qobj.get_description(), "Query((main@1 AND_NOT ((SYNONYM WILDCARD OR foo) OR bar@3)))");
991 qobj = qp.parse_query("main -bar -foo*", flags);
992 TEST_STRINGS_EQUAL(qobj.get_description(), "Query((main@1 AND_NOT (bar@2 OR (SYNONYM WILDCARD OR foo))))");
993 // Check with OP_AND too.
994 qp.set_default_op(Xapian::Query::OP_AND);
995 qobj = qp.parse_query("foo* main", flags);
996 TEST_STRINGS_EQUAL(qobj.get_description(), "Query(((SYNONYM WILDCARD OR foo) AND main@2))");
997 qobj = qp.parse_query("main foo*", flags);
998 TEST_STRINGS_EQUAL(qobj.get_description(), "Query((main@1 AND (SYNONYM WILDCARD OR foo)))");
999 qp.set_default_op(Xapian::Query::OP_AND);
1000 qobj = qp.parse_query("+foo* main", flags);
1001 TEST_STRINGS_EQUAL(qobj.get_description(), "Query(((SYNONYM WILDCARD OR foo) AND main@2))");
1002 qobj = qp.parse_query("main +foo*", flags);
1003 TEST_STRINGS_EQUAL(qobj.get_description(), "Query((main@1 AND (SYNONYM WILDCARD OR foo)))");
1004 qobj = qp.parse_query("-foo* main", flags);
1005 TEST_STRINGS_EQUAL(qobj.get_description(), "Query((main@2 AND_NOT (SYNONYM WILDCARD OR foo)))");
1006 qobj = qp.parse_query("main -foo*", flags);
1007 TEST_STRINGS_EQUAL(qobj.get_description(), "Query((main@1 AND_NOT (SYNONYM WILDCARD OR foo)))");
1008 // Check empty wildcard followed by negation.
1009 qobj = qp.parse_query("foo* -main", Xapian::QueryParser::FLAG_LOVEHATE|Xapian::QueryParser::FLAG_WILDCARD);
1010 TEST_STRINGS_EQUAL(qobj.get_description(), "Query(((SYNONYM WILDCARD OR foo) AND_NOT main@2))");
1011 // Regression test for bug#484 fixed in 1.2.1 and 1.0.21.
1012 qobj = qp.parse_query("abc muscl* main", flags);
1013 TEST_STRINGS_EQUAL(qobj.get_description(), "Query(((abc@1 AND (SYNONYM WILDCARD OR muscl)) AND main@3))");
1014 return true;
1015 #endif
1018 // Test right truncation with prefixes.
1019 static bool test_qp_flag_wildcard2()
1021 #ifndef XAPIAN_HAS_INMEMORY_BACKEND
1022 SKIP_TEST("Testcase requires the InMemory backend which is disabled");
1023 #else
1024 Xapian::WritableDatabase db(string(), Xapian::DB_BACKEND_INMEMORY);
1025 Xapian::Document doc;
1026 doc.add_term("Aheinlein");
1027 doc.add_term("Ahuxley");
1028 doc.add_term("hello");
1029 db.add_document(doc);
1030 Xapian::QueryParser qp;
1031 qp.set_database(db);
1032 qp.add_prefix("author", "A");
1033 Xapian::Query qobj;
1034 qobj = qp.parse_query("author:h*", Xapian::QueryParser::FLAG_WILDCARD);
1035 TEST_STRINGS_EQUAL(qobj.get_description(), "Query((SYNONYM WILDCARD OR Ah))");
1036 qobj = qp.parse_query("author:h* test", Xapian::QueryParser::FLAG_WILDCARD);
1037 TEST_STRINGS_EQUAL(qobj.get_description(), "Query(((SYNONYM WILDCARD OR Ah) OR test@2))");
1038 return true;
1039 #endif
1042 #ifdef XAPIAN_HAS_INMEMORY_BACKEND
1043 static void
1044 test_qp_flag_wildcard1_helper(const Xapian::Database &db,
1045 Xapian::termcount max_expansion,
1046 const string & query_string)
1048 Xapian::QueryParser qp;
1049 qp.set_database(db);
1050 qp.set_max_expansion(max_expansion);
1051 Xapian::Enquire e(db);
1052 e.set_query(qp.parse_query(query_string, Xapian::QueryParser::FLAG_WILDCARD));
1053 // The exception for expanding too much may happen at parse time or later
1054 // so we need to calculate the MSet too.
1055 e.get_mset(0, 10);
1057 #endif
1059 // Test right truncation with a limit on expansion.
1060 static bool test_qp_flag_wildcard3()
1062 #ifndef XAPIAN_HAS_INMEMORY_BACKEND
1063 SKIP_TEST("Testcase requires the InMemory backend which is disabled");
1064 #else
1065 Xapian::WritableDatabase db(string(), Xapian::DB_BACKEND_INMEMORY);
1066 Xapian::Document doc;
1067 doc.add_term("abc");
1068 doc.add_term("main");
1069 doc.add_term("muscat");
1070 doc.add_term("muscle");
1071 doc.add_term("musclebound");
1072 doc.add_term("muscular");
1073 doc.add_term("mutton");
1074 db.add_document(doc);
1076 // Test that a max of 0 doesn't set a limit.
1077 test_qp_flag_wildcard1_helper(db, 0, "z*");
1078 test_qp_flag_wildcard1_helper(db, 0, "m*");
1080 // These cases should expand to the limit given.
1081 test_qp_flag_wildcard1_helper(db, 1, "z*");
1082 test_qp_flag_wildcard1_helper(db, 1, "ab*");
1083 test_qp_flag_wildcard1_helper(db, 2, "muscle*");
1084 test_qp_flag_wildcard1_helper(db, 4, "musc*");
1085 test_qp_flag_wildcard1_helper(db, 4, "mus*");
1086 test_qp_flag_wildcard1_helper(db, 5, "mu*");
1087 test_qp_flag_wildcard1_helper(db, 6, "m*");
1089 // These cases should expand to one more than the limit.
1090 TEST_EXCEPTION(Xapian::WildcardError,
1091 test_qp_flag_wildcard1_helper(db, 1, "muscle*"));
1092 TEST_EXCEPTION(Xapian::WildcardError,
1093 test_qp_flag_wildcard1_helper(db, 3, "musc*"));
1094 TEST_EXCEPTION(Xapian::WildcardError,
1095 test_qp_flag_wildcard1_helper(db, 3, "mus*"));
1096 TEST_EXCEPTION(Xapian::WildcardError,
1097 test_qp_flag_wildcard1_helper(db, 4, "mu*"));
1098 TEST_EXCEPTION(Xapian::WildcardError,
1099 test_qp_flag_wildcard1_helper(db, 5, "m*"));
1101 return true;
1102 #endif
1105 // Test partial queries.
1106 static bool test_qp_flag_partial1()
1108 #ifndef XAPIAN_HAS_INMEMORY_BACKEND
1109 SKIP_TEST("Testcase requires the InMemory backend which is disabled");
1110 #else
1111 Xapian::WritableDatabase db(string(), Xapian::DB_BACKEND_INMEMORY);
1112 Xapian::Document doc;
1113 Xapian::Stem stemmer("english");
1114 doc.add_term("abc");
1115 doc.add_term("main");
1116 doc.add_term("muscat");
1117 doc.add_term("muscle");
1118 doc.add_term("musclebound");
1119 doc.add_term("muscular");
1120 doc.add_term("mutton");
1121 doc.add_term("Z" + stemmer("outside"));
1122 doc.add_term("Z" + stemmer("out"));
1123 doc.add_term("outside");
1124 doc.add_term("out");
1125 doc.add_term("XTcove");
1126 doc.add_term("XTcows");
1127 doc.add_term("XTcowl");
1128 doc.add_term("XTcox");
1129 doc.add_term("ZXTcow");
1130 doc.add_term("XONEpartial");
1131 doc.add_term("XONEpartial2");
1132 doc.add_term("XTWOpartial3");
1133 doc.add_term("XTWOpartial4");
1134 db.add_document(doc);
1135 Xapian::QueryParser qp;
1136 qp.set_database(db);
1137 qp.set_stemmer(stemmer);
1138 qp.set_stemming_strategy(Xapian::QueryParser::STEM_SOME);
1139 qp.add_prefix("title", "XT");
1140 qp.add_prefix("double", "XONE");
1141 qp.add_prefix("double", "XTWO");
1143 // Check behaviour with unstemmed terms
1144 Xapian::Query qobj = qp.parse_query("a", Xapian::QueryParser::FLAG_PARTIAL);
1145 TEST_STRINGS_EQUAL(qobj.get_description(), "Query(((SYNONYM WILDCARD OR a) OR Za@1))");
1146 qobj = qp.parse_query("ab", Xapian::QueryParser::FLAG_PARTIAL);
1147 TEST_STRINGS_EQUAL(qobj.get_description(), "Query(((SYNONYM WILDCARD OR ab) OR Zab@1))");
1148 qobj = qp.parse_query("muscle", Xapian::QueryParser::FLAG_PARTIAL);
1149 TEST_STRINGS_EQUAL(qobj.get_description(), "Query(((SYNONYM WILDCARD OR muscle) OR Zmuscl@1))");
1150 qobj = qp.parse_query("meat", Xapian::QueryParser::FLAG_PARTIAL);
1151 TEST_STRINGS_EQUAL(qobj.get_description(), "Query(((SYNONYM WILDCARD OR meat) OR Zmeat@1))");
1152 qobj = qp.parse_query("musc", Xapian::QueryParser::FLAG_PARTIAL);
1153 TEST_STRINGS_EQUAL(qobj.get_description(), "Query(((SYNONYM WILDCARD OR musc) OR Zmusc@1))");
1154 qobj = qp.parse_query("mutt", Xapian::QueryParser::FLAG_PARTIAL);
1155 TEST_STRINGS_EQUAL(qobj.get_description(), "Query(((SYNONYM WILDCARD OR mutt) OR Zmutt@1))");
1156 qobj = qp.parse_query("abc musc", Xapian::QueryParser::FLAG_PARTIAL);
1157 TEST_STRINGS_EQUAL(qobj.get_description(), "Query((Zabc@1 OR ((SYNONYM WILDCARD OR musc) OR Zmusc@2)))");
1158 qobj = qp.parse_query("a* mutt", Xapian::QueryParser::FLAG_PARTIAL | Xapian::QueryParser::FLAG_WILDCARD);
1159 TEST_STRINGS_EQUAL(qobj.get_description(), "Query(((SYNONYM WILDCARD OR a) OR ((SYNONYM WILDCARD OR mutt) OR Zmutt@2)))");
1161 // Check behaviour with stemmed terms, and stem strategy STEM_SOME.
1162 qobj = qp.parse_query("o", Xapian::QueryParser::FLAG_PARTIAL);
1163 TEST_STRINGS_EQUAL(qobj.get_description(), "Query(((SYNONYM WILDCARD OR o) OR Zo@1))");
1164 qobj = qp.parse_query("ou", Xapian::QueryParser::FLAG_PARTIAL);
1165 TEST_STRINGS_EQUAL(qobj.get_description(), "Query(((SYNONYM WILDCARD OR ou) OR Zou@1))");
1166 qobj = qp.parse_query("out", Xapian::QueryParser::FLAG_PARTIAL);
1167 TEST_STRINGS_EQUAL(qobj.get_description(), "Query(((SYNONYM WILDCARD OR out) OR Zout@1))");
1168 qobj = qp.parse_query("outs", Xapian::QueryParser::FLAG_PARTIAL);
1169 TEST_STRINGS_EQUAL(qobj.get_description(), "Query(((SYNONYM WILDCARD OR outs) OR Zout@1))");
1170 qobj = qp.parse_query("outsi", Xapian::QueryParser::FLAG_PARTIAL);
1171 TEST_STRINGS_EQUAL(qobj.get_description(), "Query(((SYNONYM WILDCARD OR outsi) OR Zoutsi@1))");
1172 qobj = qp.parse_query("outsid", Xapian::QueryParser::FLAG_PARTIAL);
1173 TEST_STRINGS_EQUAL(qobj.get_description(), "Query(((SYNONYM WILDCARD OR outsid) OR Zoutsid@1))");
1174 qobj = qp.parse_query("outside", Xapian::QueryParser::FLAG_PARTIAL);
1175 TEST_STRINGS_EQUAL(qobj.get_description(), "Query(((SYNONYM WILDCARD OR outside) OR Zoutsid@1))");
1177 // Check behaviour with capitalised terms, and stem strategy STEM_SOME.
1178 qobj = qp.parse_query("Out", Xapian::QueryParser::FLAG_PARTIAL);
1179 TEST_STRINGS_EQUAL(qobj.get_description(), "Query(((SYNONYM WILDCARD OR out) OR out@1))");
1180 qobj = qp.parse_query("Outs", Xapian::QueryParser::FLAG_PARTIAL);
1181 TEST_STRINGS_EQUAL(qobj.get_description(), "Query(((SYNONYM WILDCARD OR outs) OR outs@1))");
1182 qobj = qp.parse_query("Outside", Xapian::QueryParser::FLAG_PARTIAL);
1183 TEST_STRINGS_EQUAL(qobj.get_description(), "Query(((SYNONYM WILDCARD OR outside) OR outside@1))");
1184 // FIXME: Used to be this, but we aren't currently doing this change:
1185 // TEST_STRINGS_EQUAL(qobj.get_description(), "Query(outside@1#2)");
1187 // And now with stemming strategy STEM_ALL.
1188 qp.set_stemming_strategy(Xapian::QueryParser::STEM_ALL);
1189 qobj = qp.parse_query("Out", Xapian::QueryParser::FLAG_PARTIAL);
1190 TEST_STRINGS_EQUAL(qobj.get_description(), "Query(((SYNONYM WILDCARD OR out) OR out@1))");
1191 qobj = qp.parse_query("Outs", Xapian::QueryParser::FLAG_PARTIAL);
1192 TEST_STRINGS_EQUAL(qobj.get_description(), "Query(((SYNONYM WILDCARD OR outs) OR out@1))");
1193 qobj = qp.parse_query("Outside", Xapian::QueryParser::FLAG_PARTIAL);
1194 TEST_STRINGS_EQUAL(qobj.get_description(), "Query(((SYNONYM WILDCARD OR outside) OR outsid@1))");
1196 // And now with stemming strategy STEM_ALL_Z.
1197 qp.set_stemming_strategy(Xapian::QueryParser::STEM_ALL_Z);
1198 qobj = qp.parse_query("Out", Xapian::QueryParser::FLAG_PARTIAL);
1199 TEST_STRINGS_EQUAL(qobj.get_description(), "Query(((SYNONYM WILDCARD OR out) OR Zout@1))");
1200 qobj = qp.parse_query("Outs", Xapian::QueryParser::FLAG_PARTIAL);
1201 TEST_STRINGS_EQUAL(qobj.get_description(), "Query(((SYNONYM WILDCARD OR outs) OR Zout@1))");
1202 qobj = qp.parse_query("Outside", Xapian::QueryParser::FLAG_PARTIAL);
1203 TEST_STRINGS_EQUAL(qobj.get_description(), "Query(((SYNONYM WILDCARD OR outside) OR Zoutsid@1))");
1205 // Check handling of a case with a prefix.
1206 qp.set_stemming_strategy(Xapian::QueryParser::STEM_SOME);
1207 qobj = qp.parse_query("title:cow", Xapian::QueryParser::FLAG_PARTIAL);
1208 TEST_STRINGS_EQUAL(qobj.get_description(), "Query(((SYNONYM WILDCARD OR XTcow) OR ZXTcow@1))");
1209 qobj = qp.parse_query("title:cows", Xapian::QueryParser::FLAG_PARTIAL);
1210 TEST_STRINGS_EQUAL(qobj.get_description(), "Query(((SYNONYM WILDCARD OR XTcows) OR ZXTcow@1))");
1211 qobj = qp.parse_query("title:Cow", Xapian::QueryParser::FLAG_PARTIAL);
1212 TEST_STRINGS_EQUAL(qobj.get_description(), "Query(((SYNONYM WILDCARD OR XTcow) OR XTcow@1))");
1213 qobj = qp.parse_query("title:Cows", Xapian::QueryParser::FLAG_PARTIAL);
1214 TEST_STRINGS_EQUAL(qobj.get_description(), "Query(((SYNONYM WILDCARD OR XTcows) OR XTcows@1))");
1215 // FIXME: Used to be this, but we aren't currently doing this change:
1216 // TEST_STRINGS_EQUAL(qobj.get_description(), "Query(XTcows@1#2)");
1218 // Regression test - the initial version of the multi-prefix code would
1219 // inflate the wqf of the "parsed as normal" version of a partial term
1220 // by multiplying it by the number of prefixes mapped to.
1221 qobj = qp.parse_query("double:vision", Xapian::QueryParser::FLAG_PARTIAL);
1222 TEST_STRINGS_EQUAL(qobj.get_description(), "Query(((WILDCARD OR XONEvision SYNONYM WILDCARD OR XTWOvision) OR (ZXONEvision@1 SYNONYM ZXTWOvision@1)))");
1224 // Test handling of FLAG_PARTIAL when there's more than one prefix.
1225 qobj = qp.parse_query("double:part", Xapian::QueryParser::FLAG_PARTIAL);
1226 TEST_STRINGS_EQUAL(qobj.get_description(), "Query(((WILDCARD OR XONEpart SYNONYM WILDCARD OR XTWOpart) OR (ZXONEpart@1 SYNONYM ZXTWOpart@1)))");
1228 // Test handling of FLAG_PARTIAL when there's more than one prefix, without
1229 // stemming.
1230 qp.set_stemming_strategy(Xapian::QueryParser::STEM_NONE);
1231 qobj = qp.parse_query("double:part", Xapian::QueryParser::FLAG_PARTIAL);
1232 TEST_STRINGS_EQUAL(qobj.get_description(), "Query(((WILDCARD OR XONEpart SYNONYM WILDCARD OR XTWOpart) OR (XONEpart@1 SYNONYM XTWOpart@1)))");
1233 qobj = qp.parse_query("double:partial", Xapian::QueryParser::FLAG_PARTIAL);
1234 TEST_STRINGS_EQUAL(qobj.get_description(), "Query(((WILDCARD OR XONEpartial SYNONYM WILDCARD OR XTWOpartial) OR (XONEpartial@1 SYNONYM XTWOpartial@1)))");
1236 return true;
1237 #endif
1240 static bool test_qp_flag_bool_any_case1()
1242 using Xapian::QueryParser;
1243 Xapian::QueryParser qp;
1244 Xapian::Query qobj;
1245 qobj = qp.parse_query("to and fro", QueryParser::FLAG_BOOLEAN | QueryParser::FLAG_BOOLEAN_ANY_CASE);
1246 TEST_STRINGS_EQUAL(qobj.get_description(), "Query((to@1 AND fro@2))");
1247 qobj = qp.parse_query("to and fro", QueryParser::FLAG_BOOLEAN);
1248 TEST_STRINGS_EQUAL(qobj.get_description(), "Query((to@1 OR and@2 OR fro@3))");
1249 // Regression test for bug in 0.9.4 and earlier.
1250 qobj = qp.parse_query("to And fro", QueryParser::FLAG_BOOLEAN | QueryParser::FLAG_BOOLEAN_ANY_CASE);
1251 TEST_STRINGS_EQUAL(qobj.get_description(), "Query((to@1 AND fro@2))");
1252 qobj = qp.parse_query("to And fro", QueryParser::FLAG_BOOLEAN);
1253 TEST_STRINGS_EQUAL(qobj.get_description(), "Query((to@1 OR and@2 OR fro@3))");
1254 return true;
1257 static const test test_stop_queries[] = {
1258 { "test the queryparser", "(test@1 AND queryparser@3)" },
1259 // Regression test for bug in 0.9.6 and earlier. This would fail to
1260 // parse.
1261 { "test AND the AND queryparser", "((test@1 AND the@2) AND queryparser@3)" },
1262 // 0.9.6 and earlier ignored a stopword even if it was the only term.
1263 // More recent versions don't ever treat a single term as a stopword.
1264 { "the", "the@1" },
1265 // 1.2.2 and earlier ignored an all-stopword query with multiple terms,
1266 // which prevents 'to be or not to be' for being searchable unless the
1267 // user made it into a phrase query or prefixed all terms with '+'
1268 // (ticket#245).
1269 { "an the a", "(an@1 AND the@2 AND a@3)" },
1270 // Regression test for bug in initial version of the patch for the
1271 // "all-stopword" case.
1272 { "the AND a an", "(the@1 AND (a@2 AND an@3))" },
1273 { NULL, NULL }
1276 static bool test_qp_stopper1()
1278 Xapian::QueryParser qp;
1279 const char * stopwords[] = { "a", "an", "the" };
1280 Xapian::SimpleStopper stop(stopwords, stopwords + 3);
1281 qp.set_stopper(&stop);
1282 qp.set_default_op(Xapian::Query::OP_AND);
1283 for (const test *p = test_stop_queries; p->query; ++p) {
1284 string expect, parsed;
1285 if (p->expect)
1286 expect = p->expect;
1287 else
1288 expect = "parse error";
1289 try {
1290 Xapian::Query qobj = qp.parse_query(p->query);
1291 parsed = qobj.get_description();
1292 expect = string("Query(") + expect + ')';
1293 } catch (const Xapian::QueryParserError &e) {
1294 parsed = e.get_msg();
1295 } catch (const Xapian::Error &e) {
1296 parsed = e.get_description();
1297 } catch (...) {
1298 parsed = "Unknown exception!";
1300 tout << "Query: " << p->query << '\n';
1301 TEST_STRINGS_EQUAL(parsed, expect);
1303 return true;
1306 static const test test_pure_not_queries[] = {
1307 { "NOT windows", "(<alldocuments> AND_NOT Zwindow@1)" },
1308 { "a AND (NOT b)", "(Za@1 AND (<alldocuments> AND_NOT Zb@2))" },
1309 { "AND NOT windows", "Syntax: <expression> AND NOT <expression>" },
1310 { "gordian NOT", "Syntax: <expression> NOT <expression>" },
1311 { "gordian AND NOT", "Syntax: <expression> AND NOT <expression>" },
1312 { NULL, NULL }
1315 static bool test_qp_flag_pure_not1()
1317 using Xapian::QueryParser;
1318 Xapian::QueryParser qp;
1319 qp.set_stemmer(Xapian::Stem("english"));
1320 qp.set_stemming_strategy(QueryParser::STEM_SOME);
1321 for (const test *p = test_pure_not_queries; p->query; ++p) {
1322 string expect, parsed;
1323 if (p->expect)
1324 expect = p->expect;
1325 else
1326 expect = "parse error";
1327 try {
1328 Xapian::Query qobj = qp.parse_query(p->query,
1329 QueryParser::FLAG_BOOLEAN |
1330 QueryParser::FLAG_PURE_NOT);
1331 parsed = qobj.get_description();
1332 expect = string("Query(") + expect + ')';
1333 } catch (const Xapian::QueryParserError &e) {
1334 parsed = e.get_msg();
1335 } catch (const Xapian::Error &e) {
1336 parsed = e.get_description();
1337 } catch (...) {
1338 parsed = "Unknown exception!";
1340 tout << "Query: " << p->query << '\n';
1341 TEST_STRINGS_EQUAL(parsed, expect);
1343 return true;
1346 // Debatable if this is a regression test or a feature test, as it's not
1347 // obvious is this was a bug fix or a new feature. Either way, it first
1348 // appeared in Xapian 1.0.0.
1349 static bool test_qp_unstem_boolean_prefix()
1351 Xapian::QueryParser qp;
1352 qp.add_boolean_prefix("test", "XTEST");
1353 Xapian::Query q = qp.parse_query("hello test:foo");
1354 TEST_STRINGS_EQUAL(q.get_description(), "Query((hello@1 FILTER XTESTfoo))");
1355 Xapian::TermIterator u = qp.unstem_begin("XTESTfoo");
1356 TEST(u != qp.unstem_end("XTESTfoo"));
1357 TEST_EQUAL(*u, "test:foo");
1358 ++u;
1359 TEST(u == qp.unstem_end("XTESTfoo"));
1360 return true;
1363 static const test test_value_range1_queries[] = {
1364 { "a..b", "0 * VALUE_RANGE 1 a b" },
1365 { "$50..100", "0 * VALUE_RANGE 1 $50 100" },
1366 { "$50..$99", "0 * VALUE_RANGE 1 $50 $99" },
1367 { "$50..$100", "" }, // start > range
1368 { "02/03/1979..10/12/1980", "0 * VALUE_RANGE 1 02/03/1979 10/12/1980" },
1369 { "a..b hello", "(hello@1 FILTER VALUE_RANGE 1 a b)" },
1370 { "hello a..b", "(hello@1 FILTER VALUE_RANGE 1 a b)" },
1371 { "hello a..b world", "((hello@1 OR world@2) FILTER VALUE_RANGE 1 a b)" },
1372 { "hello a..b test:foo", "(hello@1 FILTER (VALUE_RANGE 1 a b AND XTESTfoo))" },
1373 { "hello a..b test:foo test:bar", "(hello@1 FILTER (VALUE_RANGE 1 a b AND (XTESTfoo OR XTESTbar)))" },
1374 { "hello a..b c..d test:foo", "(hello@1 FILTER ((VALUE_RANGE 1 a b OR VALUE_RANGE 1 c d) AND XTESTfoo))" },
1375 { "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)))" },
1376 { "-5..7", "0 * VALUE_RANGE 1 -5 7" },
1377 { "hello -5..7", "(hello@1 FILTER VALUE_RANGE 1 -5 7)" },
1378 { "-5..7 hello", "(hello@1 FILTER VALUE_RANGE 1 -5 7)" },
1379 { "\"time flies\" 09:00..12:30", "((time@1 PHRASE 2 flies@2) FILTER VALUE_RANGE 1 09:00 12:30)" },
1380 // Feature test for single-ended ranges (ticket#480):
1381 { "..b", "0 * VALUE_LE 1 b" },
1382 { "a..", "0 * VALUE_GE 1 a" },
1383 // Test for expanded set of characters allowed in range start:
1384 { "10:30+1300..11:00+1300", "0 * VALUE_RANGE 1 10:30+1300 11:00+1300" },
1385 { NULL, NULL }
1388 // Simple test of ValueRangeProcessor class.
1389 static bool test_qp_value_range1()
1391 Xapian::QueryParser qp;
1392 qp.add_boolean_prefix("test", "XTEST");
1393 Xapian::StringValueRangeProcessor vrp(1);
1394 qp.add_valuerangeprocessor(&vrp);
1395 for (const test *p = test_value_range1_queries; p->query; ++p) {
1396 string expect, parsed;
1397 if (p->expect)
1398 expect = p->expect;
1399 else
1400 expect = "parse error";
1401 try {
1402 Xapian::Query qobj = qp.parse_query(p->query);
1403 parsed = qobj.get_description();
1404 expect = string("Query(") + expect + ')';
1405 } catch (const Xapian::QueryParserError &e) {
1406 parsed = e.get_msg();
1407 } catch (const Xapian::Error &e) {
1408 parsed = e.get_description();
1409 } catch (...) {
1410 parsed = "Unknown exception!";
1412 tout << "Query: " << p->query << '\n';
1413 TEST_STRINGS_EQUAL(parsed, expect);
1415 return true;
1418 static const test test_value_range2_queries[] = {
1419 { "a..b", "0 * VALUE_RANGE 3 a b" },
1420 { "1..12", "0 * VALUE_RANGE 2 \\xa0 \\xae" },
1421 { "20070201..20070228", "0 * VALUE_RANGE 1 20070201 20070228" },
1422 { "$10..20", "0 * VALUE_RANGE 4 \\xad \\xb1" },
1423 { "$10..$20", "0 * VALUE_RANGE 4 \\xad \\xb1" },
1424 // Feature test for single-ended ranges (ticket#480):
1425 { "$..20", "0 * VALUE_LE 4 \\xb1" },
1426 { "..$20", "0 * VALUE_LE 3 $20" }, // FIXME: probably should parse as $..20
1427 { "$10..", "0 * VALUE_GE 4 \\xad" },
1428 { "12..42kg", "0 * VALUE_RANGE 5 \\xae \\xb5@" },
1429 { "12kg..42kg", "0 * VALUE_RANGE 5 \\xae \\xb5@" },
1430 { "12kg..42", "0 * VALUE_RANGE 3 12kg 42" },
1431 { "10..$20", "" }, // start > end
1432 { "1999-03-12..2020-12-30", "0 * VALUE_RANGE 1 19990312 20201230" },
1433 { "1999/03/12..2020/12/30", "0 * VALUE_RANGE 1 19990312 20201230" },
1434 { "1999.03.12..2020.12.30", "0 * VALUE_RANGE 1 19990312 20201230" },
1435 // Feature test for single-ended ranges (ticket#480):
1436 { "..2020.12.30", "0 * VALUE_LE 1 20201230" },
1437 { "1999.03.12..", "0 * VALUE_GE 1 19990312" },
1438 { "12/03/99..12/04/01", "0 * VALUE_RANGE 1 19990312 20010412" },
1439 { "03-12-99..04-14-01", "0 * VALUE_RANGE 1 19990312 20010414" },
1440 { "(test:a..test:b hello)", "(hello@1 FILTER VALUE_RANGE 3 test:a test:b)" },
1441 { "12..42kg 5..6kg 1..12", "0 * (VALUE_RANGE 2 \\xa0 \\xae AND (VALUE_RANGE 5 \\xae \\xb5@ OR VALUE_RANGE 5 \\xa9 \\xaa))" },
1442 // Check that a VRP which fails to match doesn't remove a prefix or suffix.
1443 // 1.0.13/1.1.1 and earlier got this wrong in some cases.
1444 { "$12a..13", "0 * VALUE_RANGE 3 $12a 13" },
1445 { "$12..13b", "0 * VALUE_RANGE 3 $12 13b" },
1446 { "$12..12kg", "0 * VALUE_RANGE 3 $12 12kg" },
1447 { "12..b12kg", "0 * VALUE_RANGE 3 12 b12kg" },
1448 { NULL, NULL }
1451 // Test chaining of ValueRangeProcessor classes.
1452 static bool test_qp_value_range2()
1454 Xapian::QueryParser qp;
1455 qp.add_boolean_prefix("test", "XTEST");
1456 Xapian::DateValueRangeProcessor vrp_date(1);
1457 Xapian::NumberValueRangeProcessor vrp_num(2);
1458 Xapian::StringValueRangeProcessor vrp_str(3);
1459 Xapian::NumberValueRangeProcessor vrp_cash(4, "$");
1460 Xapian::NumberValueRangeProcessor vrp_weight(5, "kg", false);
1461 qp.add_valuerangeprocessor(&vrp_date);
1462 qp.add_valuerangeprocessor(&vrp_num);
1463 qp.add_valuerangeprocessor(&vrp_cash);
1464 qp.add_valuerangeprocessor(&vrp_weight);
1465 qp.add_valuerangeprocessor(&vrp_str);
1466 for (const test *p = test_value_range2_queries; p->query; ++p) {
1467 string expect, parsed;
1468 if (p->expect)
1469 expect = p->expect;
1470 else
1471 expect = "parse error";
1472 try {
1473 Xapian::Query qobj = qp.parse_query(p->query);
1474 parsed = qobj.get_description();
1475 expect = string("Query(") + expect + ')';
1476 } catch (const Xapian::QueryParserError &e) {
1477 parsed = e.get_msg();
1478 } catch (const Xapian::Error &e) {
1479 parsed = e.get_description();
1480 } catch (...) {
1481 parsed = "Unknown exception!";
1483 tout << "Query: " << p->query << '\n';
1484 TEST_STRINGS_EQUAL(parsed, expect);
1486 return true;
1489 // Test NumberValueRangeProcessors with actual data.
1490 static bool test_qp_value_range3()
1492 #ifndef XAPIAN_HAS_INMEMORY_BACKEND
1493 SKIP_TEST("Testcase requires the InMemory backend which is disabled");
1494 #else
1495 Xapian::WritableDatabase db(string(), Xapian::DB_BACKEND_INMEMORY);
1496 double low = -10;
1497 int steps = 60;
1498 double step = 0.5;
1500 for (int i = 0; i <= steps; ++i) {
1501 double v = low + i * step;
1502 Xapian::Document doc;
1503 doc.add_value(1, Xapian::sortable_serialise(v));
1504 db.add_document(doc);
1507 Xapian::NumberValueRangeProcessor vrp_num(1);
1508 Xapian::QueryParser qp;
1509 qp.add_valuerangeprocessor(&vrp_num);
1511 for (int j = 0; j <= steps; ++j) {
1512 double start = low + j * step;
1513 for (int k = 0; k <= steps; ++k) {
1514 double end = low + k * step;
1515 string query = str(start) + ".." + str(end);
1516 tout << "Query: " << query << '\n';
1517 Xapian::Query qobj = qp.parse_query(query);
1518 Xapian::Enquire enq(db);
1519 enq.set_query(qobj);
1520 Xapian::MSet mset = enq.get_mset(0, steps + 1);
1521 if (end < start) {
1522 TEST_EQUAL(mset.size(), 0);
1523 } else {
1524 TEST_EQUAL(mset.size(), 1u + (k - j));
1525 for (unsigned int m = 0; m != mset.size(); ++m) {
1526 double v = start + m * step;
1527 TEST_EQUAL(mset[m].get_document().get_value(1),
1528 Xapian::sortable_serialise(v));
1533 return true;
1534 #endif
1537 static const test test_value_range4_queries[] = {
1538 { "id:19254@foo..example.com", "0 * Q19254@foo..example.com" },
1539 { "hello:world", "0 * XHELLOworld" },
1540 { "hello:mum..world", "0 * VALUE_RANGE 1 mum world" },
1541 { NULL, NULL }
1544 /** Test a boolean filter which happens to contain "..".
1546 * Regression test for bug fixed in 1.2.3.
1548 * Also test that the same prefix can be set for a valuerange and filter.
1550 static bool test_qp_value_range4()
1552 Xapian::QueryParser qp;
1553 qp.add_boolean_prefix("id", "Q");
1554 qp.add_boolean_prefix("hello", "XHELLO");
1555 Xapian::StringValueRangeProcessor vrp_str(1, "hello:");
1556 qp.add_valuerangeprocessor(&vrp_str);
1557 for (const test *p = test_value_range4_queries; p->query; ++p) {
1558 string expect, parsed;
1559 if (p->expect)
1560 expect = p->expect;
1561 else
1562 expect = "parse error";
1563 try {
1564 Xapian::Query qobj = qp.parse_query(p->query);
1565 parsed = qobj.get_description();
1566 expect = string("Query(") + expect + ')';
1567 } catch (const Xapian::QueryParserError &e) {
1568 parsed = e.get_msg();
1569 } catch (const Xapian::Error &e) {
1570 parsed = e.get_description();
1571 } catch (...) {
1572 parsed = "Unknown exception!";
1574 tout << "Query: " << p->query << '\n';
1575 TEST_STRINGS_EQUAL(parsed, expect);
1577 return true;
1581 static const test test_value_daterange1_queries[] = {
1582 { "12/03/99..12/04/01", "0 * VALUE_RANGE 1 19991203 20011204" },
1583 { "03-12-99..04-14-01", "0 * VALUE_RANGE 1 19990312 20010414" },
1584 { "01/30/60..02/02/59", "0 * VALUE_RANGE 1 19600130 20590202" },
1585 { "1999-03-12..2001-04-14", "0 * VALUE_RANGE 1 19990312 20010414" },
1586 { "12/03/99..02", "Unknown range operation" },
1587 { "1999-03-12..2001", "Unknown range operation" },
1588 { NULL, NULL }
1591 // Test DateValueRangeProcessor
1592 static bool test_qp_value_daterange1()
1594 Xapian::QueryParser qp;
1595 Xapian::DateValueRangeProcessor vrp_date(1, true, 1960);
1596 qp.add_valuerangeprocessor(&vrp_date);
1597 for (const test *p = test_value_daterange1_queries; p->query; ++p) {
1598 string expect, parsed;
1599 if (p->expect)
1600 expect = p->expect;
1601 else
1602 expect = "parse error";
1603 try {
1604 Xapian::Query qobj = qp.parse_query(p->query);
1605 parsed = qobj.get_description();
1606 expect = string("Query(") + expect + ')';
1607 } catch (const Xapian::QueryParserError &e) {
1608 parsed = e.get_msg();
1609 } catch (const Xapian::Error &e) {
1610 parsed = e.get_description();
1611 } catch (...) {
1612 parsed = "Unknown exception!";
1614 tout << "Query: " << p->query << '\n';
1615 TEST_STRINGS_EQUAL(parsed, expect);
1617 return true;
1620 static const test test_value_daterange2_queries[] = {
1621 { "created:12/03/99..12/04/01", "0 * VALUE_RANGE 1 19991203 20011204" },
1622 { "modified:03-12-99..04-14-01", "0 * VALUE_RANGE 2 19990312 20010414" },
1623 { "accessed:01/30/70..02/02/69", "0 * VALUE_RANGE 3 19700130 20690202" },
1624 // In <=1.2.12, and in 1.3.0, this gave "Unknown range operation":
1625 { "deleted:12/03/99..12/04/01", "0 * VALUE_RANGE 4 19990312 20010412" },
1626 { "1999-03-12..2001-04-14", "Unknown range operation" },
1627 { "12/03/99..created:12/04/01", "Unknown range operation" },
1628 { "12/03/99created:..12/04/01", "Unknown range operation" },
1629 { "12/03/99..12/04/01created:", "Unknown range operation" },
1630 { "12/03/99..02", "Unknown range operation" },
1631 { "1999-03-12..2001", "Unknown range operation" },
1632 { NULL, NULL }
1635 // Feature test DateValueRangeProcessor with prefixes (added in 1.1.2).
1636 static bool test_qp_value_daterange2()
1638 Xapian::QueryParser qp;
1639 Xapian::DateValueRangeProcessor vrp_cdate(1, "created:", true, true, 1970);
1640 Xapian::DateValueRangeProcessor vrp_mdate(2, "modified:", true, true, 1970);
1641 Xapian::DateValueRangeProcessor vrp_adate(3, "accessed:", true, true, 1970);
1642 // Regression test - here a const char * was taken as a bool rather than a
1643 // std::string when resolving the overloaded forms. Fixed in 1.2.13 and
1644 // 1.3.1.
1645 Xapian::DateValueRangeProcessor vrp_ddate(4, "deleted:");
1646 qp.add_valuerangeprocessor(&vrp_cdate);
1647 qp.add_valuerangeprocessor(&vrp_mdate);
1648 qp.add_valuerangeprocessor(&vrp_adate);
1649 qp.add_valuerangeprocessor(&vrp_ddate);
1650 for (const test *p = test_value_daterange2_queries; p->query; ++p) {
1651 string expect, parsed;
1652 if (p->expect)
1653 expect = p->expect;
1654 else
1655 expect = "parse error";
1656 try {
1657 Xapian::Query qobj = qp.parse_query(p->query);
1658 parsed = qobj.get_description();
1659 expect = string("Query(") + expect + ')';
1660 } catch (const Xapian::QueryParserError &e) {
1661 parsed = e.get_msg();
1662 } catch (const Xapian::Error &e) {
1663 parsed = e.get_description();
1664 } catch (...) {
1665 parsed = "Unknown exception!";
1667 tout << "Query: " << p->query << '\n';
1668 TEST_STRINGS_EQUAL(parsed, expect);
1670 return true;
1673 static const test test_value_stringrange1_queries[] = {
1674 { "tag:bar..foo", "0 * VALUE_RANGE 1 bar foo" },
1675 { "bar..foo", "0 * VALUE_RANGE 0 bar foo" },
1676 { NULL, NULL }
1679 // Feature test StringValueRangeProcessor with prefixes (added in 1.1.2).
1680 static bool test_qp_value_stringrange1()
1682 Xapian::QueryParser qp;
1683 Xapian::StringValueRangeProcessor vrp_default(0);
1684 Xapian::StringValueRangeProcessor vrp_tag(1, "tag:", true);
1685 qp.add_valuerangeprocessor(&vrp_tag);
1686 qp.add_valuerangeprocessor(&vrp_default);
1687 for (const test *p = test_value_stringrange1_queries; p->query; ++p) {
1688 string expect, parsed;
1689 if (p->expect)
1690 expect = p->expect;
1691 else
1692 expect = "parse error";
1693 try {
1694 Xapian::Query qobj = qp.parse_query(p->query);
1695 parsed = qobj.get_description();
1696 expect = string("Query(") + expect + ')';
1697 } catch (const Xapian::QueryParserError &e) {
1698 parsed = e.get_msg();
1699 } catch (const Xapian::Error &e) {
1700 parsed = e.get_description();
1701 } catch (...) {
1702 parsed = "Unknown exception!";
1704 tout << "Query: " << p->query << '\n';
1705 TEST_STRINGS_EQUAL(parsed, expect);
1707 return true;
1710 struct AuthorValueRangeProcessor : public Xapian::ValueRangeProcessor {
1711 AuthorValueRangeProcessor() {}
1713 Xapian::valueno operator()(std::string &begin, std::string &end) {
1714 if (!startswith(begin, "author:"))
1715 return Xapian::BAD_VALUENO;
1716 begin.erase(0, 7);
1717 begin = Xapian::Unicode::tolower(begin);
1718 end = Xapian::Unicode::tolower(end);
1719 return 4;
1723 static const test test_value_customrange1_queries[] = {
1724 { "mars author:Asimov..Bradbury", "(mars@1 FILTER VALUE_RANGE 4 asimov bradbury)" },
1725 { NULL, NULL }
1728 // Test custom ValueRangeProcessor subclass.
1729 static bool test_qp_value_customrange1()
1731 Xapian::QueryParser qp;
1732 AuthorValueRangeProcessor vrp_author;
1733 qp.add_valuerangeprocessor(&vrp_author);
1734 for (const test *p = test_value_customrange1_queries; p->query; ++p) {
1735 string expect, parsed;
1736 if (p->expect)
1737 expect = p->expect;
1738 else
1739 expect = "parse error";
1740 try {
1741 Xapian::Query qobj = qp.parse_query(p->query);
1742 parsed = qobj.get_description();
1743 expect = string("Query(") + expect + ')';
1744 } catch (const Xapian::QueryParserError &e) {
1745 parsed = e.get_msg();
1746 } catch (const Xapian::Error &e) {
1747 parsed = e.get_description();
1748 } catch (...) {
1749 parsed = "Unknown exception!";
1751 tout << "Query: " << p->query << '\n';
1752 TEST_STRINGS_EQUAL(parsed, expect);
1754 return true;
1757 class TitleFieldProcessor : public Xapian::FieldProcessor {
1758 Xapian::Query operator()(const std::string & str) {
1759 if (str == "all")
1760 return Xapian::Query::MatchAll;
1761 return Xapian::Query("S" + str);
1765 class HostFieldProcessor : public Xapian::FieldProcessor {
1766 Xapian::Query operator()(const std::string & str) {
1767 if (str == "*")
1768 return Xapian::Query::MatchAll;
1769 string res = "H";
1770 for (string::const_iterator i = str.begin(); i != str.end(); ++i)
1771 res += C_tolower(*i);
1772 return Xapian::Query(res);
1776 static const test test_fieldproc1_queries[] = {
1777 { "title:test", "Stest" },
1778 { "title:all", "<alldocuments>" },
1779 { "host:Xapian.org", "0 * Hxapian.org" },
1780 { "host:*", "0 * <alldocuments>" },
1781 { "host:\"Space Station.Example.Org\"", "0 * Hspace station.example.org" },
1782 { NULL, NULL }
1785 // FieldProcessor test.
1786 static bool test_qp_fieldproc1()
1788 Xapian::QueryParser qp;
1789 TitleFieldProcessor title_fproc;
1790 HostFieldProcessor host_fproc;
1791 qp.add_prefix("title", &title_fproc);
1792 qp.add_boolean_prefix("host", &host_fproc);
1793 for (const test *p = test_fieldproc1_queries; p->query; ++p) {
1794 string expect, parsed;
1795 if (p->expect)
1796 expect = p->expect;
1797 else
1798 expect = "parse error";
1799 try {
1800 Xapian::Query qobj = qp.parse_query(p->query);
1801 parsed = qobj.get_description();
1802 expect = string("Query(") + expect + ')';
1803 } catch (const Xapian::QueryParserError &e) {
1804 parsed = e.get_msg();
1805 } catch (const Xapian::Error &e) {
1806 parsed = e.get_description();
1807 } catch (...) {
1808 parsed = "Unknown exception!";
1810 tout << "Query: " << p->query << '\n';
1811 TEST_STRINGS_EQUAL(parsed, expect);
1813 return true;
1816 class DateRangeFieldProcessor : public Xapian::FieldProcessor {
1817 Xapian::Query operator()(const std::string & str) {
1818 // In reality, these would be built from the current date, but for
1819 // testing it is much simpler to fix the date.
1820 if (str == "today")
1821 return Xapian::Query(Xapian::Query::OP_VALUE_GE, 1, "20120725");
1822 if (str == "this week")
1823 return Xapian::Query(Xapian::Query::OP_VALUE_GE, 1, "20120723");
1824 if (str == "this month")
1825 return Xapian::Query(Xapian::Query::OP_VALUE_GE, 1, "20120701");
1826 if (str == "this year")
1827 return Xapian::Query(Xapian::Query::OP_VALUE_GE, 1, "20120101");
1828 if (str == "this decade")
1829 return Xapian::Query(Xapian::Query::OP_VALUE_GE, 1, "20100101");
1830 if (str == "this century")
1831 return Xapian::Query(Xapian::Query::OP_VALUE_GE, 1, "20000101");
1832 throw Xapian::QueryParserError("Didn't understand date specification '" + str + "'");
1836 static const test test_fieldproc2_queries[] = {
1837 { "date:\"this week\"", "0 * VALUE_GE 1 20120723" },
1838 { "date:23/7/2012..25/7/2012", "0 * VALUE_RANGE 1 20120723 20120725" },
1839 { NULL, NULL }
1842 // Test using FieldProcessor and ValueRangeProcessor together.
1843 static bool test_qp_fieldproc2()
1845 Xapian::QueryParser qp;
1846 DateRangeFieldProcessor date_fproc;
1847 qp.add_boolean_prefix("date", &date_fproc);
1848 Xapian::DateValueRangeProcessor vrp_date(1, "date:");
1849 qp.add_valuerangeprocessor(&vrp_date);
1850 for (const test *p = test_fieldproc2_queries; p->query; ++p) {
1851 string expect, parsed;
1852 if (p->expect)
1853 expect = p->expect;
1854 else
1855 expect = "parse error";
1856 try {
1857 Xapian::Query qobj = qp.parse_query(p->query);
1858 parsed = qobj.get_description();
1859 expect = string("Query(") + expect + ')';
1860 } catch (const Xapian::QueryParserError &e) {
1861 parsed = e.get_msg();
1862 } catch (const Xapian::Error &e) {
1863 parsed = e.get_description();
1864 } catch (...) {
1865 parsed = "Unknown exception!";
1867 tout << "Query: " << p->query << '\n';
1868 TEST_STRINGS_EQUAL(parsed, expect);
1870 return true;
1873 static bool test_qp_stoplist1()
1875 Xapian::QueryParser qp;
1876 const char * stopwords[] = { "a", "an", "the" };
1877 Xapian::SimpleStopper stop(stopwords, stopwords + 3);
1878 qp.set_stopper(&stop);
1880 Xapian::TermIterator i;
1882 Xapian::Query query1 = qp.parse_query("some mice");
1883 i = qp.stoplist_begin();
1884 TEST(i == qp.stoplist_end());
1886 Xapian::Query query2 = qp.parse_query("the cat");
1887 i = qp.stoplist_begin();
1888 TEST(i != qp.stoplist_end());
1889 TEST_EQUAL(*i, "the");
1890 ++i;
1891 TEST(i == qp.stoplist_end());
1893 // Regression test - prior to Xapian 1.0.0 the stoplist wasn't being cleared
1894 // when a new query was parsed.
1895 Xapian::Query query3 = qp.parse_query("an aardvark");
1896 i = qp.stoplist_begin();
1897 TEST(i != qp.stoplist_end());
1898 TEST_EQUAL(*i, "an");
1899 ++i;
1900 TEST(i == qp.stoplist_end());
1902 return true;
1905 static const test test_mispelled_queries[] = {
1906 { "doucment search", "document search" },
1907 { "doucment seeacrh", "document search" },
1908 { "docment seeacrh test", "document search test" },
1909 { "\"paragahp pineapple\"", "\"paragraph pineapple\"" },
1910 { "\"paragahp pineapple\"", "\"paragraph pineapple\"" },
1911 { "test S.E.A.R.C.", "" },
1912 { "this AND that", "" },
1913 { "documento", "document" },
1914 { "documento-documento", "document-document" },
1915 { "documento-searcho", "document-search" },
1916 { "test saerch", "test search" },
1917 { "paragraf search", "paragraph search" },
1918 { NULL, NULL }
1921 // Test spelling correction in the QueryParser.
1922 static bool test_qp_spell1()
1924 mkdir(".chert", 0755);
1925 string dbdir = ".chert/qp_spell1";
1926 Xapian::WritableDatabase db(dbdir, Xapian::DB_CREATE_OR_OVERWRITE);
1928 Xapian::Document doc;
1929 doc.add_term("document", 6);
1930 doc.add_term("search", 7);
1931 doc.add_term("saerch", 1);
1932 doc.add_term("paragraph", 8);
1933 doc.add_term("paragraf", 2);
1934 db.add_document(doc);
1936 db.add_spelling("document");
1937 db.add_spelling("search");
1938 db.add_spelling("paragraph");
1939 db.add_spelling("band");
1941 Xapian::QueryParser qp;
1942 qp.set_stemming_strategy(Xapian::QueryParser::STEM_SOME);
1943 qp.set_database(db);
1945 for (const test *p = test_mispelled_queries; p->query; ++p) {
1946 Xapian::Query q;
1947 q = qp.parse_query(p->query,
1948 Xapian::QueryParser::FLAG_SPELLING_CORRECTION |
1949 Xapian::QueryParser::FLAG_BOOLEAN );
1950 tout << "Query: " << p->query << endl;
1951 TEST_STRINGS_EQUAL(qp.get_corrected_query_string(), p->expect);
1954 return true;
1957 // Test spelling correction in the QueryParser with multiple databases.
1958 static bool test_qp_spell2()
1960 mkdir(".chert", 0755);
1961 string dbdir = ".chert/qp_spell2";
1962 Xapian::WritableDatabase db1(dbdir, Xapian::DB_CREATE_OR_OVERWRITE);
1964 db1.add_spelling("document");
1965 db1.add_spelling("search");
1966 db1.commit();
1968 dbdir = ".chert/qp_spell2b";
1969 Xapian::WritableDatabase db2(dbdir, Xapian::DB_CREATE_OR_OVERWRITE);
1971 db2.add_spelling("document");
1972 db2.add_spelling("paragraph");
1973 db2.add_spelling("band");
1975 Xapian::Database db;
1976 db.add_database(db1);
1977 db.add_database(db2);
1979 Xapian::QueryParser qp;
1980 qp.set_stemming_strategy(Xapian::QueryParser::STEM_SOME);
1981 qp.set_database(db);
1983 for (const test *p = test_mispelled_queries; p->query; ++p) {
1984 Xapian::Query q;
1985 q = qp.parse_query(p->query,
1986 Xapian::QueryParser::FLAG_SPELLING_CORRECTION |
1987 Xapian::QueryParser::FLAG_BOOLEAN );
1988 tout << "Query: " << p->query << endl;
1989 TEST_STRINGS_EQUAL(qp.get_corrected_query_string(), p->expect);
1992 return true;
1995 static const test test_mispelled_wildcard_queries[] = {
1996 { "doucment", "document" },
1997 { "doucment*", "" },
1998 { "doucment* seearch", "doucment* search" },
1999 { "doucment* search", "" },
2000 { NULL, NULL }
2003 // Test spelling correction in the QueryParser with wildcards.
2004 // Regression test for bug fixed in 1.1.3 and 1.0.17.
2005 static bool test_qp_spellwild1()
2007 mkdir(".chert", 0755);
2008 string dbdir = ".chert/qp_spellwild1";
2009 Xapian::WritableDatabase db(dbdir, Xapian::DB_CREATE_OR_OVERWRITE);
2011 db.add_spelling("document");
2012 db.add_spelling("search");
2013 db.add_spelling("paragraph");
2014 db.add_spelling("band");
2016 Xapian::QueryParser qp;
2017 qp.set_database(db);
2019 const test *p;
2020 for (p = test_mispelled_queries; p->query; ++p) {
2021 Xapian::Query q;
2022 q = qp.parse_query(p->query,
2023 Xapian::QueryParser::FLAG_SPELLING_CORRECTION |
2024 Xapian::QueryParser::FLAG_BOOLEAN |
2025 Xapian::QueryParser::FLAG_WILDCARD);
2026 tout << "Query: " << p->query << endl;
2027 TEST_STRINGS_EQUAL(qp.get_corrected_query_string(), p->expect);
2029 for (p = test_mispelled_wildcard_queries; p->query; ++p) {
2030 Xapian::Query q;
2031 q = qp.parse_query(p->query,
2032 Xapian::QueryParser::FLAG_SPELLING_CORRECTION |
2033 Xapian::QueryParser::FLAG_BOOLEAN |
2034 Xapian::QueryParser::FLAG_WILDCARD);
2035 tout << "Query: " << p->query << endl;
2036 TEST_STRINGS_EQUAL(qp.get_corrected_query_string(), p->expect);
2039 return true;
2042 static const test test_mispelled_partial_queries[] = {
2043 { "doucment", "" },
2044 { "doucment ", "document " },
2045 { "documen", "" },
2046 { "documen ", "document " },
2047 { "seearch documen", "search documen" },
2048 { "search documen", "" },
2049 { NULL, NULL }
2052 // Test spelling correction in the QueryParser with FLAG_PARTIAL.
2053 // Regression test for bug fixed in 1.1.3 and 1.0.17.
2054 static bool test_qp_spellpartial1()
2056 mkdir(".chert", 0755);
2057 string dbdir = ".chert/qp_spellpartial1";
2058 Xapian::WritableDatabase db(dbdir, Xapian::DB_CREATE_OR_OVERWRITE);
2060 db.add_spelling("document");
2061 db.add_spelling("search");
2062 db.add_spelling("paragraph");
2063 db.add_spelling("band");
2065 Xapian::QueryParser qp;
2066 qp.set_database(db);
2068 for (const test *p = test_mispelled_partial_queries; p->query; ++p) {
2069 Xapian::Query q;
2070 q = qp.parse_query(p->query,
2071 Xapian::QueryParser::FLAG_SPELLING_CORRECTION |
2072 Xapian::QueryParser::FLAG_PARTIAL);
2073 tout << "Query: " << p->query << endl;
2074 TEST_STRINGS_EQUAL(qp.get_corrected_query_string(), p->expect);
2077 return true;
2080 static const test test_synonym_queries[] = {
2081 { "searching", "(Zsearch@1 SYNONYM Zfind@1 SYNONYM Zlocate@1)" },
2082 { "search", "(Zsearch@1 SYNONYM find@1)" },
2083 { "Search", "(search@1 SYNONYM find@1)" },
2084 { "Searching", "searching@1" },
2085 { "searching OR terms", "((Zsearch@1 SYNONYM Zfind@1 SYNONYM Zlocate@1) OR Zterm@2)" },
2086 { "search OR terms", "((Zsearch@1 SYNONYM find@1) OR Zterm@2)" },
2087 { "search +terms", "(Zterm@2 AND_MAYBE (Zsearch@1 SYNONYM find@1))" },
2088 { "search -terms", "((Zsearch@1 SYNONYM find@1) AND_NOT Zterm@2)" },
2089 { "+search terms", "((Zsearch@1 SYNONYM find@1) AND_MAYBE Zterm@2)" },
2090 { "-search terms", "(Zterm@2 AND_NOT (Zsearch@1 SYNONYM find@1))" },
2091 { "search terms", "((Zsearch@1 SYNONYM find@1) OR Zterm@2)" },
2092 // Shouldn't trigger synonyms:
2093 { "\"search terms\"", "(search@1 PHRASE 2 terms@2)" },
2094 // Check that setting FLAG_AUTO_SYNONYMS doesn't enable multi-word
2095 // synonyms. Regression test for bug fixed in 1.3.0 and 1.2.9.
2096 { "regression test", "(Zregress@1 OR Ztest@2)" },
2097 { NULL, NULL }
2100 // Test single term synonyms in the QueryParser.
2101 static bool test_qp_synonym1()
2103 mkdir(".chert", 0755);
2104 string dbdir = ".chert/qp_synonym1";
2105 Xapian::WritableDatabase db(dbdir, Xapian::DB_CREATE_OR_OVERWRITE);
2107 db.add_synonym("Zsearch", "Zfind");
2108 db.add_synonym("Zsearch", "Zlocate");
2109 db.add_synonym("search", "find");
2110 db.add_synonym("Zseek", "Zsearch");
2111 db.add_synonym("regression test", "magic");
2113 db.commit();
2115 Xapian::QueryParser qp;
2116 qp.set_stemmer(Xapian::Stem("english"));
2117 qp.set_stemming_strategy(Xapian::QueryParser::STEM_SOME);
2118 qp.set_database(db);
2120 for (const test *p = test_synonym_queries; p->query; ++p) {
2121 string expect = "Query(";
2122 expect += p->expect;
2123 expect += ')';
2124 Xapian::Query q;
2125 q = qp.parse_query(p->query, qp.FLAG_AUTO_SYNONYMS|qp.FLAG_DEFAULT);
2126 tout << "Query: " << p->query << endl;
2127 TEST_STRINGS_EQUAL(q.get_description(), expect);
2130 return true;
2133 static const test test_multi_synonym_queries[] = {
2134 { "sun OR tan OR cream", "((Zsun@1 OR Ztan@2) OR Zcream@3)" },
2135 { "sun tan", "((Zsun@1 OR Ztan@2) SYNONYM bathe@1)" },
2136 { "sun tan cream", "((Zsun@1 OR Ztan@2 OR Zcream@3) SYNONYM lotion@1)" },
2137 { "beach sun tan holiday", "(Zbeach@1 OR ((Zsun@2 OR Ztan@3) SYNONYM bathe@2) OR Zholiday@4)" },
2138 { "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))" },
2139 { "single", "(Zsingl@1 SYNONYM record@1)" },
2140 { NULL, NULL }
2143 // Test multi term synonyms in the QueryParser.
2144 static bool test_qp_synonym2()
2146 mkdir(".chert", 0755);
2147 string dbdir = ".chert/qp_synonym2";
2148 Xapian::WritableDatabase db(dbdir, Xapian::DB_CREATE_OR_OVERWRITE);
2150 db.add_synonym("sun tan cream", "lotion");
2151 db.add_synonym("sun tan", "bathe");
2152 db.add_synonym("single", "record");
2154 db.commit();
2156 Xapian::QueryParser qp;
2157 qp.set_stemmer(Xapian::Stem("english"));
2158 qp.set_stemming_strategy(Xapian::QueryParser::STEM_SOME);
2159 qp.set_database(db);
2161 for (const test *p = test_multi_synonym_queries; p->query; ++p) {
2162 string expect = "Query(";
2163 expect += p->expect;
2164 expect += ')';
2165 Xapian::Query q;
2166 q = qp.parse_query(p->query,
2167 Xapian::QueryParser::FLAG_AUTO_MULTIWORD_SYNONYMS |
2168 Xapian::QueryParser::FLAG_DEFAULT);
2169 tout << "Query: " << p->query << endl;
2170 TEST_STRINGS_EQUAL(q.get_description(), expect);
2173 return true;
2176 static const test test_synonym_op_queries[] = {
2177 { "searching", "Zsearch@1" },
2178 { "~searching", "(Zsearch@1 SYNONYM Zfind@1 SYNONYM Zlocate@1)" },
2179 { "~search", "(Zsearch@1 SYNONYM find@1)" },
2180 { "~Search", "(search@1 SYNONYM find@1)" },
2181 { "~Searching", "searching@1" },
2182 { "~searching OR terms", "((Zsearch@1 SYNONYM Zfind@1 SYNONYM Zlocate@1) OR Zterm@2)" },
2183 { "~search OR terms", "((Zsearch@1 SYNONYM find@1) OR Zterm@2)" },
2184 { "~search +terms", "(Zterm@2 AND_MAYBE (Zsearch@1 SYNONYM find@1))" },
2185 { "~search -terms", "((Zsearch@1 SYNONYM find@1) AND_NOT Zterm@2)" },
2186 { "+~search terms", "((Zsearch@1 SYNONYM find@1) AND_MAYBE Zterm@2)" },
2187 { "-~search terms", "(Zterm@2 AND_NOT (Zsearch@1 SYNONYM find@1))" },
2188 { "~search terms", "((Zsearch@1 SYNONYM find@1) OR Zterm@2)" },
2189 { "~foo:search", "(ZXFOOsearch@1 SYNONYM prefixated@1)" },
2190 // FIXME: should look for multi-term synonym...
2191 { "~\"search terms\"", "(search@1 PHRASE 2 terms@2)" },
2192 { NULL, NULL }
2195 // Test the synonym operator in the QueryParser.
2196 static bool test_qp_synonym3()
2198 mkdir(".chert", 0755);
2199 string dbdir = ".chert/qp_synonym3";
2200 Xapian::WritableDatabase db(dbdir, Xapian::DB_CREATE_OR_OVERWRITE);
2202 db.add_synonym("Zsearch", "Zfind");
2203 db.add_synonym("Zsearch", "Zlocate");
2204 db.add_synonym("search", "find");
2205 db.add_synonym("Zseek", "Zsearch");
2206 db.add_synonym("ZXFOOsearch", "prefixated");
2208 db.commit();
2210 Xapian::QueryParser qp;
2211 qp.set_stemmer(Xapian::Stem("english"));
2212 qp.set_stemming_strategy(Xapian::QueryParser::STEM_SOME);
2213 qp.set_database(db);
2214 qp.add_prefix("foo", "XFOO");
2216 for (const test *p = test_synonym_op_queries; p->query; ++p) {
2217 string expect = "Query(";
2218 expect += p->expect;
2219 expect += ')';
2220 Xapian::Query q;
2221 q = qp.parse_query(p->query,
2222 Xapian::QueryParser::FLAG_SYNONYM |
2223 Xapian::QueryParser::FLAG_BOOLEAN |
2224 Xapian::QueryParser::FLAG_LOVEHATE |
2225 Xapian::QueryParser::FLAG_PHRASE );
2226 tout << "Query: " << p->query << endl;
2227 TEST_STRINGS_EQUAL(q.get_description(), expect);
2230 return true;
2233 static const test test_stem_all_queries[] = {
2234 { "\"chemical engineers\"", "(chemic@1 PHRASE 2 engin@2)" },
2235 { "chemical NEAR engineers", "(chemic@1 NEAR 11 engin@2)" },
2236 { "chemical engineers", "(chemic@1 OR engin@2)" },
2237 { "title:(chemical engineers)", "(XTchemic@1 OR XTengin@2)" },
2238 { NULL, NULL }
2241 static bool test_qp_stem_all1()
2243 Xapian::QueryParser qp;
2244 qp.set_stemmer(Xapian::Stem("english"));
2245 qp.set_stemming_strategy(qp.STEM_ALL);
2246 qp.add_prefix("title", "XT");
2247 for (const test *p = test_stem_all_queries; p->query; ++p) {
2248 string expect, parsed;
2249 if (p->expect)
2250 expect = p->expect;
2251 else
2252 expect = "parse error";
2253 try {
2254 Xapian::Query qobj = qp.parse_query(p->query);
2255 parsed = qobj.get_description();
2256 expect = string("Query(") + expect + ')';
2257 } catch (const Xapian::QueryParserError &e) {
2258 parsed = e.get_msg();
2259 } catch (const Xapian::Error &e) {
2260 parsed = e.get_description();
2261 } catch (...) {
2262 parsed = "Unknown exception!";
2264 tout << "Query: " << p->query << '\n';
2265 TEST_STRINGS_EQUAL(parsed, expect);
2267 return true;
2270 static const test test_stem_all_z_queries[] = {
2271 { "\"chemical engineers\"", "(Zchemic@1 PHRASE 2 Zengin@2)" },
2272 { "chemical NEAR engineers", "(Zchemic@1 NEAR 11 Zengin@2)" },
2273 { "chemical engineers", "(Zchemic@1 OR Zengin@2)" },
2274 { "title:(chemical engineers)", "(ZXTchemic@1 OR ZXTengin@2)" },
2275 { NULL, NULL }
2278 static bool test_qp_stem_all_z1()
2280 Xapian::QueryParser qp;
2281 qp.set_stemmer(Xapian::Stem("english"));
2282 qp.set_stemming_strategy(qp.STEM_ALL_Z);
2283 qp.add_prefix("title", "XT");
2284 for (const test *p = test_stem_all_z_queries; p->query; ++p) {
2285 string expect, parsed;
2286 if (p->expect)
2287 expect = p->expect;
2288 else
2289 expect = "parse error";
2290 try {
2291 Xapian::Query qobj = qp.parse_query(p->query);
2292 parsed = qobj.get_description();
2293 expect = string("Query(") + expect + ')';
2294 } catch (const Xapian::QueryParserError &e) {
2295 parsed = e.get_msg();
2296 } catch (const Xapian::Error &e) {
2297 parsed = e.get_description();
2298 } catch (...) {
2299 parsed = "Unknown exception!";
2301 tout << "Query: " << p->query << '\n';
2302 TEST_STRINGS_EQUAL(parsed, expect);
2304 return true;
2307 static double
2308 time_query_parse(const Xapian::Database & db, const string & q,
2309 int repetitions, unsigned flags)
2311 Xapian::QueryParser qp;
2312 qp.set_database(db);
2313 CPUTimer timer;
2314 std::vector<Xapian::Query> qs;
2315 qs.reserve(repetitions);
2316 for (int i = 0; i != repetitions; ++i) {
2317 qs.push_back(qp.parse_query(q, flags));
2319 if (repetitions > 1) {
2320 Xapian::Query qc(Xapian::Query::OP_OR, qs.begin(), qs.end());
2322 return timer.get_time();
2325 static void
2326 qp_scale1_helper(const Xapian::Database &db, const string & q, unsigned n,
2327 unsigned flags)
2329 double time1;
2330 while (true) {
2331 time1 = time_query_parse(db, q, n, flags);
2332 if (time1 != 0.0) break;
2334 // The first test completed before the timer ticked at all, so increase
2335 // the number of repetitions and retry.
2336 unsigned n_new = n * 10;
2337 if (n_new < n)
2338 SKIP_TEST("Can't count enough repetitions to be able to time test");
2339 n = n_new;
2342 n /= 5;
2344 string q_n;
2345 q_n.reserve(q.size() * n);
2346 for (unsigned i = n; i != 0; --i) {
2347 q_n += q;
2350 // Time 5 repetitions so we average random variations a bit.
2351 double time2 = time_query_parse(db, q_n, 5, flags);
2352 tout << "small=" << time1 << "s, large=" << time2 << "s\n";
2354 // Allow a factor of 2.15 difference, to cover random variation and a
2355 // native time interval which isn't an exact multiple of 1/CLK_TCK.
2356 TEST_REL(time2,<,time1 * 2.15);
2359 // Regression test: check that query parser doesn't scale very badly with the
2360 // size of the query.
2361 static bool test_qp_scale1()
2363 mkdir(".chert", 0755);
2364 string dbdir = ".chert/qp_scale1";
2365 Xapian::WritableDatabase db(dbdir, Xapian::DB_CREATE_OR_OVERWRITE);
2367 db.add_synonym("foo", "bar");
2368 db.commit();
2370 string q1("foo ");
2371 string q1b("baz ");
2372 const unsigned repetitions = 5000;
2374 // A long multiword synonym.
2375 string syn;
2376 for (int j = 60; j != 0; --j) {
2377 syn += q1;
2379 syn.resize(syn.size() - 1);
2381 unsigned synflags = Xapian::QueryParser::FLAG_DEFAULT |
2382 Xapian::QueryParser::FLAG_SYNONYM |
2383 Xapian::QueryParser::FLAG_AUTO_MULTIWORD_SYNONYMS;
2385 // First, we test a simple query.
2386 qp_scale1_helper(db, q1, repetitions, Xapian::QueryParser::FLAG_DEFAULT);
2388 // If synonyms are enabled, a different code-path is followed.
2389 // Test a query which has no synonyms.
2390 qp_scale1_helper(db, q1b, repetitions, synflags);
2392 // Test a query which has short synonyms.
2393 qp_scale1_helper(db, q1, repetitions, synflags);
2395 // Add a synonym for the whole query, to test that code path.
2396 db.add_synonym(syn, "bar");
2397 db.commit();
2399 qp_scale1_helper(db, q1, repetitions, synflags);
2401 return true;
2404 static const test test_near_queries[] = {
2405 { "simple-example", "(simple@1 PHRASE 2 example@2)" },
2406 { "stock -cooking", "(Zstock@1 AND_NOT Zcook@2)" },
2407 // FIXME: these give NEAR 2
2408 // { "foo -baz bar", "((foo@1 NEAR 11 bar@3) AND_NOT Zbaz@2)" },
2409 // { "one +two three", "(Ztwo@2 AND_MAYBE (one@1 NEAR 11 three@3))" },
2410 { "foo bar", "(foo@1 NEAR 11 bar@2)" },
2411 { "foo bar baz", "(foo@1 NEAR 12 bar@2 NEAR 12 baz@3)" },
2412 { "gtk+ -gnome", "(Zgtk+@1 AND_NOT Zgnome@2)" },
2413 { "c++ -d--", "(Zc++@1 AND_NOT Zd@2)" },
2414 { "\"c++ library\"", "(c++@1 PHRASE 2 library@2)" },
2415 { "author:orwell animal farm", "(Aorwell@1 NEAR 12 animal@2 NEAR 12 farm@3)" },
2416 { "author:Orwell Animal Farm", "(Aorwell@1 NEAR 12 animal@2 NEAR 12 farm@3)" },
2417 { "beer NOT \"orange juice\"", "(Zbeer@1 AND_NOT (orange@2 PHRASE 2 juice@3))" },
2418 { "beer AND NOT lager", "(Zbeer@1 AND_NOT Zlager@2)" },
2419 { "A OR B NOT C", "(a@1 OR (b@2 AND_NOT c@3))" },
2420 { "A OR B AND NOT C", "(a@1 OR (b@2 AND_NOT c@3))" },
2421 { "A OR B XOR C", "(a@1 OR (b@2 XOR c@3))" },
2422 { "A XOR B NOT C", "(a@1 XOR (b@2 AND_NOT c@3))" },
2423 { "one AND two", "(Zone@1 AND Ztwo@2)" },
2424 { "NOT windows", "Syntax: <expression> NOT <expression>" },
2425 { "a AND (NOT b)", "Syntax: <expression> NOT <expression>" },
2426 { "AND NOT windows", "Syntax: <expression> AND NOT <expression>" },
2427 { "gordian NOT", "Syntax: <expression> NOT <expression>" },
2428 { "gordian AND NOT", "Syntax: <expression> AND NOT <expression>" },
2429 { "foo OR (something AND)", "Syntax: <expression> AND <expression>" },
2430 { "OR foo", "Syntax: <expression> OR <expression>" },
2431 { "XOR", "Syntax: <expression> XOR <expression>" },
2432 { "hard\xa0space", "(hard@1 NEAR 11 space@2)" },
2433 { NULL, NULL }
2436 static bool test_qp_near1()
2438 Xapian::QueryParser queryparser;
2439 queryparser.set_stemmer(Xapian::Stem("english"));
2440 queryparser.set_stemming_strategy(Xapian::QueryParser::STEM_SOME);
2441 queryparser.add_prefix("author", "A");
2442 queryparser.add_prefix("writer", "A");
2443 queryparser.add_prefix("title", "XT");
2444 queryparser.add_prefix("subject", "XT");
2445 queryparser.add_prefix("authortitle", "A");
2446 queryparser.add_prefix("authortitle", "XT");
2447 queryparser.add_boolean_prefix("site", "H");
2448 queryparser.add_boolean_prefix("site2", "J");
2449 queryparser.add_boolean_prefix("multisite", "H");
2450 queryparser.add_boolean_prefix("multisite", "J");
2451 queryparser.add_boolean_prefix("category", "XCAT", false);
2452 queryparser.add_boolean_prefix("dogegory", "XDOG", false);
2453 queryparser.set_default_op(Xapian::Query::OP_NEAR);
2454 for (const test *p = test_near_queries; p->query; ++p) {
2455 string expect, parsed;
2456 if (p->expect)
2457 expect = p->expect;
2458 else
2459 expect = "parse error";
2460 try {
2461 Xapian::Query qobj = queryparser.parse_query(p->query);
2462 parsed = qobj.get_description();
2463 expect = string("Query(") + expect + ')';
2464 } catch (const Xapian::QueryParserError &e) {
2465 parsed = e.get_msg();
2466 } catch (const Xapian::Error &e) {
2467 parsed = e.get_description();
2468 } catch (...) {
2469 parsed = "Unknown exception!";
2471 tout << "Query: " << p->query << '\n';
2472 TEST_STRINGS_EQUAL(parsed, expect);
2474 return true;
2477 static const test test_phrase_queries[] = {
2478 { "simple-example", "(simple@1 PHRASE 2 example@2)" },
2479 { "stock -cooking", "(Zstock@1 AND_NOT Zcook@2)" },
2480 // FIXME: these give PHRASE 2
2481 // { "foo -baz bar", "((foo@1 PHRASE 11 bar@3) AND_NOT Zbaz@2)" },
2482 // { "one +two three", "(Ztwo@2 AND_MAYBE (one@1 PHRASE 11 three@3))" },
2483 { "foo bar", "(foo@1 PHRASE 11 bar@2)" },
2484 { "foo bar baz", "(foo@1 PHRASE 12 bar@2 PHRASE 12 baz@3)" },
2485 { "gtk+ -gnome", "(Zgtk+@1 AND_NOT Zgnome@2)" },
2486 { "c++ -d--", "(Zc++@1 AND_NOT Zd@2)" },
2487 { "\"c++ library\"", "(c++@1 PHRASE 2 library@2)" },
2488 { "author:orwell animal farm", "(Aorwell@1 PHRASE 12 animal@2 PHRASE 12 farm@3)" },
2489 { "author:Orwell Animal Farm", "(Aorwell@1 PHRASE 12 animal@2 PHRASE 12 farm@3)" },
2490 { "beer NOT \"orange juice\"", "(Zbeer@1 AND_NOT (orange@2 PHRASE 2 juice@3))" },
2491 { "beer AND NOT lager", "(Zbeer@1 AND_NOT Zlager@2)" },
2492 { "A OR B NOT C", "(a@1 OR (b@2 AND_NOT c@3))" },
2493 { "A OR B AND NOT C", "(a@1 OR (b@2 AND_NOT c@3))" },
2494 { "A OR B XOR C", "(a@1 OR (b@2 XOR c@3))" },
2495 { "A XOR B NOT C", "(a@1 XOR (b@2 AND_NOT c@3))" },
2496 { "one AND two", "(Zone@1 AND Ztwo@2)" },
2497 { "NOT windows", "Syntax: <expression> NOT <expression>" },
2498 { "a AND (NOT b)", "Syntax: <expression> NOT <expression>" },
2499 { "AND NOT windows", "Syntax: <expression> AND NOT <expression>" },
2500 { "gordian NOT", "Syntax: <expression> NOT <expression>" },
2501 { "gordian AND NOT", "Syntax: <expression> AND NOT <expression>" },
2502 { "foo OR (something AND)", "Syntax: <expression> AND <expression>" },
2503 { "OR foo", "Syntax: <expression> OR <expression>" },
2504 { "XOR", "Syntax: <expression> XOR <expression>" },
2505 { "hard\xa0space", "(hard@1 PHRASE 11 space@2)" },
2506 // FIXME: this isn't what we want, but fixing phrase to work with
2507 // subqueries first might be the best approach.
2508 // FIXME: this isn't currently reimplemented:
2509 // { "(one AND two) three", "((Zone@1 PHRASE 11 Zthree@3) AND (Ztwo@2 PHRASE 11 Zthree@3))" },
2510 { NULL, NULL }
2513 static bool test_qp_phrase1()
2515 Xapian::QueryParser queryparser;
2516 queryparser.set_stemmer(Xapian::Stem("english"));
2517 queryparser.set_stemming_strategy(Xapian::QueryParser::STEM_SOME);
2518 queryparser.add_prefix("author", "A");
2519 queryparser.add_prefix("writer", "A");
2520 queryparser.add_prefix("title", "XT");
2521 queryparser.add_prefix("subject", "XT");
2522 queryparser.add_prefix("authortitle", "A");
2523 queryparser.add_prefix("authortitle", "XT");
2524 queryparser.add_boolean_prefix("site", "H");
2525 queryparser.add_boolean_prefix("site2", "J");
2526 queryparser.add_boolean_prefix("multisite", "H");
2527 queryparser.add_boolean_prefix("multisite", "J");
2528 queryparser.add_boolean_prefix("category", "XCAT", false);
2529 queryparser.add_boolean_prefix("dogegory", "XDOG", false);
2530 queryparser.set_default_op(Xapian::Query::OP_PHRASE);
2531 for (const test *p = test_phrase_queries; p->query; ++p) {
2532 string expect, parsed;
2533 if (p->expect)
2534 expect = p->expect;
2535 else
2536 expect = "parse error";
2537 try {
2538 Xapian::Query qobj = queryparser.parse_query(p->query);
2539 parsed = qobj.get_description();
2540 expect = string("Query(") + expect + ')';
2541 } catch (const Xapian::QueryParserError &e) {
2542 parsed = e.get_msg();
2543 } catch (const Xapian::Error &e) {
2544 parsed = e.get_description();
2545 } catch (...) {
2546 parsed = "Unknown exception!";
2548 tout << "Query: " << p->query << '\n';
2549 TEST_STRINGS_EQUAL(parsed, expect);
2551 return true;
2554 static const test test_stopword_group_or_queries[] = {
2555 { "this is a test", "test@4" },
2556 { "test*", "(SYNONYM WILDCARD OR test)" },
2557 { "a test*", "(SYNONYM WILDCARD OR test)" },
2558 { "is a test*", "(SYNONYM WILDCARD OR test)" },
2559 { "this is a test*", "(SYNONYM WILDCARD OR test)" },
2560 { "this is a us* test*", "((SYNONYM WILDCARD OR us) OR (SYNONYM WILDCARD OR test))" },
2561 { "this is a user test*", "(user@4 OR (SYNONYM WILDCARD OR test))" },
2562 { NULL, NULL }
2565 static const test test_stopword_group_and_queries[] = {
2566 { "this is a test", "test@4" },
2567 { "test*", "(SYNONYM WILDCARD OR test)" },
2568 { "a test*", "(SYNONYM WILDCARD OR test)" },
2569 // Two stopwords + one wildcard failed in 1.0.16
2570 { "is a test*", "(SYNONYM WILDCARD OR test)" },
2571 // Three stopwords + one wildcard failed in 1.0.16
2572 { "this is a test*", "(SYNONYM WILDCARD OR test)" },
2573 // Three stopwords + two wildcards failed in 1.0.16
2574 { "this is a us* test*", "((SYNONYM WILDCARD OR us) AND (SYNONYM WILDCARD OR test))" },
2575 { "this is a user test*", "(user@4 AND (SYNONYM WILDCARD OR test))" },
2576 { NULL, NULL }
2579 // Regression test for bug fixed in 1.0.17 and 1.1.3.
2580 static bool test_qp_stopword_group1()
2582 #ifndef XAPIAN_HAS_INMEMORY_BACKEND
2583 SKIP_TEST("Testcase requires the InMemory backend which is disabled");
2584 #else
2585 Xapian::WritableDatabase db(string(), Xapian::DB_BACKEND_INMEMORY);
2586 Xapian::Document doc;
2587 doc.add_term("test");
2588 doc.add_term("tester");
2589 doc.add_term("testable");
2590 doc.add_term("user");
2591 db.add_document(doc);
2593 Xapian::SimpleStopper stopper;
2594 stopper.add("this");
2595 stopper.add("is");
2596 stopper.add("a");
2598 Xapian::QueryParser qp;
2599 qp.set_stopper(&stopper);
2600 qp.set_database(db);
2602 // Process test cases with OP_OR first, then with OP_AND.
2603 qp.set_default_op(Xapian::Query::OP_OR);
2604 const test *p = test_stopword_group_or_queries;
2605 for (int i = 1; i <= 2; ++i) {
2606 for ( ; p->query; ++p) {
2607 string expect, parsed;
2608 if (p->expect)
2609 expect = p->expect;
2610 else
2611 expect = "parse error";
2612 try {
2613 Xapian::Query qobj = qp.parse_query(p->query, qp.FLAG_WILDCARD);
2614 parsed = qobj.get_description();
2615 expect = string("Query(") + expect + ')';
2616 } catch (const Xapian::QueryParserError &e) {
2617 parsed = e.get_msg();
2618 } catch (const Xapian::Error &e) {
2619 parsed = e.get_description();
2620 } catch (...) {
2621 parsed = "Unknown exception!";
2623 tout << "Query: " << p->query << '\n';
2624 TEST_STRINGS_EQUAL(parsed, expect);
2627 qp.set_default_op(Xapian::Query::OP_AND);
2628 p = test_stopword_group_and_queries;
2631 return true;
2632 #endif
2635 /// Check that QueryParser::set_default_op() rejects inappropriate ops.
2636 static bool test_qp_default_op2()
2638 Xapian::QueryParser qp;
2639 static const Xapian::Query::op ops[] = {
2640 Xapian::Query::OP_AND_NOT,
2641 Xapian::Query::OP_XOR,
2642 Xapian::Query::OP_AND_MAYBE,
2643 Xapian::Query::OP_FILTER,
2644 Xapian::Query::OP_VALUE_RANGE,
2645 Xapian::Query::OP_SCALE_WEIGHT,
2646 Xapian::Query::OP_VALUE_GE,
2647 Xapian::Query::OP_VALUE_LE
2649 const Xapian::Query::op * p;
2650 for (p = ops; p - ops != sizeof(ops) / sizeof(*ops); ++p) {
2651 tout << *p << endl;
2652 TEST_EXCEPTION(Xapian::InvalidArgumentError,
2653 qp.set_default_op(*p));
2654 TEST_EQUAL(qp.get_default_op(), Xapian::Query::OP_OR);
2656 return true;
2659 struct qp_default_op3_test {
2660 Xapian::Query::op op;
2661 const char *expect;
2664 /// Check that QueryParser::set_default_op() accepts appropriate ops.
2665 static bool test_qp_default_op3()
2667 Xapian::QueryParser qp;
2668 static const qp_default_op3_test tests[] = {
2669 { Xapian::Query::OP_AND,
2670 "Query((a@1 AND b@2 AND c@3))" },
2671 { Xapian::Query::OP_OR,
2672 "Query((a@1 OR b@2 OR c@3))" },
2673 { Xapian::Query::OP_PHRASE,
2674 "Query((a@1 PHRASE 12 b@2 PHRASE 12 c@3))" },
2675 { Xapian::Query::OP_NEAR,
2676 "Query((a@1 NEAR 12 b@2 NEAR 12 c@3))" },
2677 { Xapian::Query::OP_ELITE_SET,
2678 "Query((a@1 ELITE_SET 10 b@2 ELITE_SET 10 c@3))" },
2679 { Xapian::Query::OP_SYNONYM,
2680 "Query((a@1 SYNONYM b@2 SYNONYM c@3))" },
2682 const qp_default_op3_test * p;
2683 for (p = tests; p - tests != sizeof(tests) / sizeof(*tests); ++p) {
2684 tout << p ->op<< endl;
2685 qp.set_default_op(p->op);
2686 // Check that get_default_op() returns what we just set.
2687 TEST_EQUAL(qp.get_default_op(), p->op);
2688 TEST_EQUAL(qp.parse_query("A B C").get_description(), p->expect);
2690 return true;
2693 /// Test that the default strategy is now STEM_SOME (as of 1.3.1).
2694 static bool test_qp_defaultstrategysome1()
2696 Xapian::QueryParser qp;
2697 qp.set_stemmer(Xapian::Stem("en"));
2698 TEST_EQUAL(qp.parse_query("testing").get_description(), "Query(Ztest@1)");
2699 return true;
2702 /// Test cases for the QueryParser.
2703 static const test_desc tests[] = {
2704 TESTCASE(queryparser1),
2705 TESTCASE(qp_default_op1),
2706 TESTCASE(qp_odd_chars1),
2707 TESTCASE(qp_flag_wildcard1),
2708 TESTCASE(qp_flag_wildcard2),
2709 TESTCASE(qp_flag_wildcard3),
2710 TESTCASE(qp_flag_partial1),
2711 TESTCASE(qp_flag_bool_any_case1),
2712 TESTCASE(qp_stopper1),
2713 TESTCASE(qp_flag_pure_not1),
2714 TESTCASE(qp_unstem_boolean_prefix),
2715 TESTCASE(qp_default_prefix1),
2716 TESTCASE(qp_default_prefix2),
2717 TESTCASE(qp_value_range1),
2718 TESTCASE(qp_value_range2),
2719 TESTCASE(qp_value_range3),
2720 TESTCASE(qp_value_range4),
2721 TESTCASE(qp_value_daterange1),
2722 TESTCASE(qp_value_daterange2),
2723 TESTCASE(qp_value_stringrange1),
2724 TESTCASE(qp_value_customrange1),
2725 TESTCASE(qp_fieldproc1),
2726 TESTCASE(qp_fieldproc2),
2727 TESTCASE(qp_stoplist1),
2728 TESTCASE(qp_spell1),
2729 TESTCASE(qp_spell2),
2730 TESTCASE(qp_spellwild1),
2731 TESTCASE(qp_spellpartial1),
2732 TESTCASE(qp_synonym1),
2733 TESTCASE(qp_synonym2),
2734 TESTCASE(qp_synonym3),
2735 TESTCASE(qp_stem_all1),
2736 TESTCASE(qp_stem_all_z1),
2737 TESTCASE(qp_scale1),
2738 TESTCASE(qp_near1),
2739 TESTCASE(qp_phrase1),
2740 TESTCASE(qp_stopword_group1),
2741 TESTCASE(qp_default_op2),
2742 TESTCASE(qp_default_op3),
2743 TESTCASE(qp_defaultstrategysome1),
2744 END_OF_TESTCASES
2747 int main(int argc, char **argv)
2748 try {
2749 test_driver::parse_command_line(argc, argv);
2750 return test_driver::run(tests);
2751 } catch (const char * e) {
2752 cout << e << endl;
2753 return 1;