query_conditions.hpp 28 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011
  1. /*************************************************************************
  2. *
  3. * Copyright 2016 Realm Inc.
  4. *
  5. * Licensed under the Apache License, Version 2.0 (the "License");
  6. * you may not use this file except in compliance with the License.
  7. * You may obtain a copy of the License at
  8. *
  9. * http://www.apache.org/licenses/LICENSE-2.0
  10. *
  11. * Unless required by applicable law or agreed to in writing, software
  12. * distributed under the License is distributed on an "AS IS" BASIS,
  13. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  14. * See the License for the specific language governing permissions and
  15. * limitations under the License.
  16. *
  17. **************************************************************************/
  18. #ifndef REALM_QUERY_CONDITIONS_HPP
  19. #define REALM_QUERY_CONDITIONS_HPP
  20. #include <cstdint>
  21. #include <string>
  22. #include <realm/query_state.hpp>
  23. #include <realm/unicode.hpp>
  24. #include <realm/binary_data.hpp>
  25. #include <realm/query_value.hpp>
  26. #include <realm/mixed.hpp>
  27. #include <realm/utilities.hpp>
  28. namespace realm {
  29. // Quick hack to make "Queries with Integer null columns" able to compile in Visual Studio 2015 which doesn't full
  30. // support sfinae
  31. // (real cause hasn't been investigated yet, cannot exclude that we don't obey c++11 standard)
  32. struct HackClass {
  33. template <class A, class B, class C>
  34. bool can_match(A, B, C)
  35. {
  36. REALM_ASSERT(false);
  37. return false;
  38. }
  39. template <class A, class B, class C>
  40. bool will_match(A, B, C)
  41. {
  42. REALM_ASSERT(false);
  43. return false;
  44. }
  45. };
  46. // Does v2 contain v1?
  47. struct Contains : public HackClass {
  48. bool operator()(StringData v1, const char*, const char*, StringData v2, bool = false, bool = false) const
  49. {
  50. return v2.contains(v1);
  51. }
  52. bool operator()(StringData v1, StringData v2, bool = false, bool = false) const
  53. {
  54. return v2.contains(v1);
  55. }
  56. bool operator()(BinaryData v1, BinaryData v2, bool = false, bool = false) const
  57. {
  58. return v2.contains(v1);
  59. }
  60. bool operator()(StringData v1, const std::array<uint8_t, 256> &charmap, StringData v2) const
  61. {
  62. return v2.contains(v1, charmap);
  63. }
  64. bool operator()(const QueryValue& m1, const QueryValue& m2) const
  65. {
  66. if (m1.is_null())
  67. return !m2.is_null();
  68. if (m1.is_type(type_String, type_Binary) && Mixed::types_are_comparable(m1, m2)) {
  69. BinaryData b1 = m1.get_binary();
  70. BinaryData b2 = m2.get_binary();
  71. return operator()(b1, b2, false, false);
  72. }
  73. return false;
  74. }
  75. template <class A, class B>
  76. bool operator()(A, B) const
  77. {
  78. REALM_ASSERT(false);
  79. return false;
  80. }
  81. template <class A, class B, class C, class D>
  82. bool operator()(A, B, C, D) const
  83. {
  84. REALM_ASSERT(false);
  85. return false;
  86. }
  87. bool operator()(int64_t, int64_t, bool, bool) const
  88. {
  89. REALM_ASSERT(false);
  90. return false;
  91. }
  92. static std::string description()
  93. {
  94. return "CONTAINS";
  95. }
  96. static const int condition = -1;
  97. };
  98. // Does v2 contain something like v1 (wildcard matching)?
  99. struct Like : public HackClass {
  100. bool operator()(StringData v1, const char*, const char*, StringData v2, bool = false, bool = false) const
  101. {
  102. return v2.like(v1);
  103. }
  104. bool operator()(BinaryData b1, const char*, const char*, BinaryData b2, bool = false, bool = false) const
  105. {
  106. StringData s1(b1.data(), b1.size());
  107. StringData s2(b2.data(), b2.size());
  108. return s2.like(s1);
  109. }
  110. bool operator()(StringData v1, StringData v2, bool = false, bool = false) const
  111. {
  112. return v2.like(v1);
  113. }
  114. bool operator()(BinaryData b1, BinaryData b2, bool = false, bool = false) const
  115. {
  116. StringData s1(b1.data(), b1.size());
  117. StringData s2(b2.data(), b2.size());
  118. return s2.like(s1);
  119. }
  120. bool operator()(const QueryValue& m1, const QueryValue& m2) const
  121. {
  122. if (m1.is_null() && m2.is_null())
  123. return true;
  124. if (m1.is_type(type_String, type_Binary) && Mixed::types_are_comparable(m1, m2)) {
  125. BinaryData b1 = m1.get_binary();
  126. BinaryData b2 = m2.get_binary();
  127. return operator()(b1, b2, false, false);
  128. }
  129. return false;
  130. }
  131. template <class A, class B>
  132. bool operator()(A, B) const
  133. {
  134. REALM_ASSERT(false);
  135. return false;
  136. }
  137. template <class A, class B, class C, class D>
  138. bool operator()(A, B, C, D) const
  139. {
  140. REALM_ASSERT(false);
  141. return false;
  142. }
  143. bool operator()(int64_t, int64_t, bool, bool) const
  144. {
  145. REALM_ASSERT(false);
  146. return false;
  147. }
  148. static std::string description()
  149. {
  150. return "LIKE";
  151. }
  152. static const int condition = -1;
  153. };
  154. // Does v2 begin with v1?
  155. struct BeginsWith : public HackClass {
  156. bool operator()(StringData v1, const char*, const char*, StringData v2, bool = false, bool = false) const
  157. {
  158. return v2.begins_with(v1);
  159. }
  160. bool operator()(StringData v1, StringData v2, bool = false, bool = false) const
  161. {
  162. return v2.begins_with(v1);
  163. }
  164. bool operator()(BinaryData v1, BinaryData v2, bool = false, bool = false) const
  165. {
  166. return v2.begins_with(v1);
  167. }
  168. bool operator()(const QueryValue& m1, const QueryValue& m2) const
  169. {
  170. if (m1.is_type(type_String, type_Binary) && Mixed::types_are_comparable(m1, m2)) {
  171. BinaryData b1 = m1.get_binary();
  172. BinaryData b2 = m2.get_binary();
  173. return b2.begins_with(b1);
  174. }
  175. return false;
  176. }
  177. template <class A, class B, class C, class D>
  178. bool operator()(A, B, C, D) const
  179. {
  180. REALM_ASSERT(false);
  181. return false;
  182. }
  183. template <class A, class B>
  184. bool operator()(A, B) const
  185. {
  186. REALM_ASSERT(false);
  187. return false;
  188. }
  189. static std::string description()
  190. {
  191. return "BEGINSWITH";
  192. }
  193. static const int condition = -1;
  194. };
  195. // Does v2 end with v1?
  196. struct EndsWith : public HackClass {
  197. bool operator()(StringData v1, const char*, const char*, StringData v2, bool = false, bool = false) const
  198. {
  199. return v2.ends_with(v1);
  200. }
  201. bool operator()(StringData v1, StringData v2, bool = false, bool = false) const
  202. {
  203. return v2.ends_with(v1);
  204. }
  205. bool operator()(BinaryData v1, BinaryData v2, bool = false, bool = false) const
  206. {
  207. return v2.ends_with(v1);
  208. }
  209. bool operator()(const QueryValue& m1, const QueryValue& m2) const
  210. {
  211. if (m1.is_type(type_String, type_Binary) && Mixed::types_are_comparable(m1, m2)) {
  212. BinaryData b1 = m1.get_binary();
  213. BinaryData b2 = m2.get_binary();
  214. return operator()(b1, b2, false, false);
  215. }
  216. return false;
  217. }
  218. template <class A, class B>
  219. bool operator()(A, B) const
  220. {
  221. REALM_ASSERT(false);
  222. return false;
  223. }
  224. template <class A, class B, class C, class D>
  225. bool operator()(A, B, C, D) const
  226. {
  227. REALM_ASSERT(false);
  228. return false;
  229. }
  230. static std::string description()
  231. {
  232. return "ENDSWITH";
  233. }
  234. static const int condition = -1;
  235. };
  236. struct Equal {
  237. static const int avx = 0x00; // _CMP_EQ_OQ
  238. // bool operator()(const bool v1, const bool v2, bool v1null = false, bool v2null = false) const { return v1 ==
  239. // v2; }
  240. bool operator()(StringData v1, const char*, const char*, StringData v2, bool = false, bool = false) const
  241. {
  242. return v1 == v2;
  243. }
  244. bool operator()(BinaryData v1, BinaryData v2, bool = false, bool = false) const
  245. {
  246. return v1 == v2;
  247. }
  248. bool operator()(const QueryValue& m1, const QueryValue& m2) const
  249. {
  250. return (m1.is_null() && m2.is_null()) || (Mixed::types_are_comparable(m1, m2) && (m1 == m2));
  251. }
  252. template <class T>
  253. bool operator()(const T& v1, const T& v2, bool v1null = false, bool v2null = false) const
  254. {
  255. return (v1null && v2null) || (!v1null && !v2null && v1 == v2);
  256. }
  257. static const int condition = cond_Equal;
  258. bool can_match(int64_t v, int64_t lbound, int64_t ubound)
  259. {
  260. return (v >= lbound && v <= ubound);
  261. }
  262. bool will_match(int64_t v, int64_t lbound, int64_t ubound)
  263. {
  264. return (v == 0 && ubound == 0 && lbound == 0);
  265. }
  266. static std::string description()
  267. {
  268. return "==";
  269. }
  270. };
  271. struct NotEqual {
  272. static const int avx = 0x0B; // _CMP_FALSE_OQ
  273. bool operator()(StringData v1, const char*, const char*, StringData v2, bool = false, bool = false) const
  274. {
  275. return v1 != v2;
  276. }
  277. // bool operator()(BinaryData v1, BinaryData v2, bool = false, bool = false) const { return v1 != v2; }
  278. template <class T>
  279. bool operator()(const T& v1, const T& v2, bool v1null = false, bool v2null = false) const
  280. {
  281. if (!v1null && !v2null)
  282. return v1 != v2;
  283. if (v1null && v2null)
  284. return false;
  285. return true;
  286. }
  287. bool operator()(const QueryValue& m1, const Mixed& m2) const
  288. {
  289. return !Equal()(m1, m2);
  290. }
  291. static const int condition = cond_NotEqual;
  292. bool can_match(int64_t v, int64_t lbound, int64_t ubound)
  293. {
  294. return !(v == 0 && ubound == 0 && lbound == 0);
  295. }
  296. bool will_match(int64_t v, int64_t lbound, int64_t ubound)
  297. {
  298. return (v > ubound || v < lbound);
  299. }
  300. template <class A, class B, class C, class D>
  301. bool operator()(A, B, C, D) const = delete;
  302. static std::string description()
  303. {
  304. return "!=";
  305. }
  306. };
  307. // Does v2 contain v1?
  308. struct ContainsIns : public HackClass {
  309. bool operator()(StringData v1, const char* v1_upper, const char* v1_lower, StringData v2, bool = false,
  310. bool = false) const
  311. {
  312. if (v2.is_null() && !v1.is_null())
  313. return false;
  314. if (v1.size() == 0 && !v2.is_null())
  315. return true;
  316. return search_case_fold(v2, v1_upper, v1_lower, v1.size()) != v2.size();
  317. }
  318. // Slow version, used if caller hasn't stored an upper and lower case version
  319. bool operator()(StringData v1, StringData v2, bool = false, bool = false) const
  320. {
  321. if (v2.is_null() && !v1.is_null())
  322. return false;
  323. if (v1.size() == 0 && !v2.is_null())
  324. return true;
  325. std::string v1_upper = case_map(v1, true, IgnoreErrors);
  326. std::string v1_lower = case_map(v1, false, IgnoreErrors);
  327. return search_case_fold(v2, v1_upper.c_str(), v1_lower.c_str(), v1.size()) != v2.size();
  328. }
  329. bool operator()(BinaryData b1, BinaryData b2, bool = false, bool = false) const
  330. {
  331. StringData s1(b1.data(), b1.size());
  332. StringData s2(b2.data(), b2.size());
  333. return this->operator()(s1, s2, false, false);
  334. }
  335. // Case insensitive Boyer-Moore version
  336. bool operator()(StringData v1, const char* v1_upper, const char* v1_lower, const std::array<uint8_t, 256> &charmap, StringData v2) const
  337. {
  338. if (v2.is_null() && !v1.is_null())
  339. return false;
  340. if (v1.size() == 0 && !v2.is_null())
  341. return true;
  342. return contains_ins(v2, v1_upper, v1_lower, v1.size(), charmap);
  343. }
  344. bool operator()(const QueryValue& m1, const QueryValue& m2) const
  345. {
  346. if (m1.is_null())
  347. return !m2.is_null();
  348. if (m1.is_type(type_String, type_Binary) && Mixed::types_are_comparable(m1, m2)) {
  349. BinaryData b1 = m1.get_binary();
  350. BinaryData b2 = m2.get_binary();
  351. return operator()(b1, b2, false, false);
  352. }
  353. return false;
  354. }
  355. template <class A, class B>
  356. bool operator()(A, B) const
  357. {
  358. REALM_ASSERT(false);
  359. return false;
  360. }
  361. template <class A, class B, class C, class D>
  362. bool operator()(A, B, C, D) const
  363. {
  364. REALM_ASSERT(false);
  365. return false;
  366. }
  367. bool operator()(int64_t, int64_t, bool, bool) const
  368. {
  369. REALM_ASSERT(false);
  370. return false;
  371. }
  372. static std::string description()
  373. {
  374. return "CONTAINS[c]";
  375. }
  376. static const int condition = -1;
  377. };
  378. // Does v2 contain something like v1 (wildcard matching)?
  379. struct LikeIns : public HackClass {
  380. bool operator()(StringData v1, const char* v1_upper, const char* v1_lower, StringData v2, bool = false,
  381. bool = false) const
  382. {
  383. if (v2.is_null() || v1.is_null()) {
  384. return (v2.is_null() && v1.is_null());
  385. }
  386. return string_like_ins(v2, v1_lower, v1_upper);
  387. }
  388. bool operator()(BinaryData b1, const char* b1_upper, const char* b1_lower, BinaryData b2, bool = false,
  389. bool = false) const
  390. {
  391. if (b2.is_null() || b1.is_null()) {
  392. return (b2.is_null() && b1.is_null());
  393. }
  394. StringData s2(b2.data(), b2.size());
  395. return string_like_ins(s2, b1_lower, b1_upper);
  396. }
  397. // Slow version, used if caller hasn't stored an upper and lower case version
  398. bool operator()(StringData v1, StringData v2, bool = false, bool = false) const
  399. {
  400. if (v2.is_null() || v1.is_null()) {
  401. return (v2.is_null() && v1.is_null());
  402. }
  403. std::string v1_upper = case_map(v1, true, IgnoreErrors);
  404. std::string v1_lower = case_map(v1, false, IgnoreErrors);
  405. return string_like_ins(v2, v1_lower, v1_upper);
  406. }
  407. bool operator()(BinaryData b1, BinaryData b2, bool = false, bool = false) const
  408. {
  409. if (b2.is_null() || b1.is_null()) {
  410. return (b2.is_null() && b1.is_null());
  411. }
  412. StringData s1(b1.data(), b1.size());
  413. StringData s2(b2.data(), b2.size());
  414. std::string s1_upper = case_map(s1, true, IgnoreErrors);
  415. std::string s1_lower = case_map(s1, false, IgnoreErrors);
  416. return string_like_ins(s2, s1_lower, s1_upper);
  417. }
  418. bool operator()(const QueryValue& m1, const QueryValue& m2) const
  419. {
  420. if (m1.is_null() && m2.is_null())
  421. return true;
  422. if (m1.is_type(type_String, type_Binary) && Mixed::types_are_comparable(m1, m2)) {
  423. BinaryData b1 = m1.get_binary();
  424. BinaryData b2 = m2.get_binary();
  425. return operator()(b1, b2, false, false);
  426. }
  427. return false;
  428. }
  429. template <class A, class B>
  430. bool operator()(A, B) const
  431. {
  432. REALM_ASSERT(false);
  433. return false;
  434. }
  435. template <class A, class B, class C, class D>
  436. bool operator()(A, B, C, D) const
  437. {
  438. REALM_ASSERT(false);
  439. return false;
  440. }
  441. bool operator()(int64_t, int64_t, bool, bool) const
  442. {
  443. REALM_ASSERT(false);
  444. return false;
  445. }
  446. static std::string description()
  447. {
  448. return "LIKE[c]";
  449. }
  450. static const int condition = -1;
  451. };
  452. // Does v2 begin with v1?
  453. struct BeginsWithIns : public HackClass {
  454. bool operator()(StringData v1, const char* v1_upper, const char* v1_lower, StringData v2, bool = false,
  455. bool = false) const
  456. {
  457. if (v2.is_null() && !v1.is_null())
  458. return false;
  459. return v1.size() <= v2.size() && equal_case_fold(v2.prefix(v1.size()), v1_upper, v1_lower);
  460. }
  461. // Slow version, used if caller hasn't stored an upper and lower case version
  462. bool operator()(StringData v1, StringData v2, bool = false, bool = false) const
  463. {
  464. if (v2.is_null() && !v1.is_null())
  465. return false;
  466. if (v1.size() > v2.size())
  467. return false;
  468. std::string v1_upper = case_map(v1, true, IgnoreErrors);
  469. std::string v1_lower = case_map(v1, false, IgnoreErrors);
  470. return equal_case_fold(v2.prefix(v1.size()), v1_upper.c_str(), v1_lower.c_str());
  471. }
  472. bool operator()(BinaryData b1, BinaryData b2, bool = false, bool = false) const
  473. {
  474. StringData s1(b1.data(), b1.size());
  475. StringData s2(b2.data(), b2.size());
  476. return this->operator()(s1, s2, false, false);
  477. }
  478. bool operator()(const QueryValue& m1, const QueryValue& m2) const
  479. {
  480. if (m1.is_type(type_String, type_Binary) && Mixed::types_are_comparable(m1, m2)) {
  481. BinaryData b1 = m1.get_binary();
  482. BinaryData b2 = m2.get_binary();
  483. return operator()(b1, b2, false, false);
  484. }
  485. return false;
  486. }
  487. template <class A, class B>
  488. bool operator()(A, B) const
  489. {
  490. REALM_ASSERT(false);
  491. return false;
  492. }
  493. template <class A, class B, class C, class D>
  494. bool operator()(A, B, C, D) const
  495. {
  496. REALM_ASSERT(false);
  497. return false;
  498. }
  499. bool operator()(int64_t, int64_t, bool, bool) const
  500. {
  501. REALM_ASSERT(false);
  502. return false;
  503. }
  504. static std::string description()
  505. {
  506. return "BEGINSWITH[c]";
  507. }
  508. static const int condition = -1;
  509. };
  510. // Does v2 end with v1?
  511. struct EndsWithIns : public HackClass {
  512. bool operator()(StringData v1, const char* v1_upper, const char* v1_lower, StringData v2, bool = false,
  513. bool = false) const
  514. {
  515. if (v2.is_null() && !v1.is_null())
  516. return false;
  517. return v1.size() <= v2.size() && equal_case_fold(v2.suffix(v1.size()), v1_upper, v1_lower);
  518. }
  519. // Slow version, used if caller hasn't stored an upper and lower case version
  520. bool operator()(StringData v1, StringData v2, bool = false, bool = false) const
  521. {
  522. if (v2.is_null() && !v1.is_null())
  523. return false;
  524. if (v1.size() > v2.size())
  525. return false;
  526. std::string v1_upper = case_map(v1, true, IgnoreErrors);
  527. std::string v1_lower = case_map(v1, false, IgnoreErrors);
  528. return equal_case_fold(v2.suffix(v1.size()), v1_upper.c_str(), v1_lower.c_str());
  529. }
  530. bool operator()(BinaryData b1, BinaryData b2, bool = false, bool = false) const
  531. {
  532. StringData s1(b1.data(), b1.size());
  533. StringData s2(b2.data(), b2.size());
  534. return this->operator()(s1, s2, false, false);
  535. }
  536. bool operator()(const QueryValue& m1, const QueryValue& m2) const
  537. {
  538. if (m1.is_type(type_String, type_Binary) && Mixed::types_are_comparable(m1, m2)) {
  539. BinaryData b1 = m1.get_binary();
  540. BinaryData b2 = m2.get_binary();
  541. return operator()(b1, b2, false, false);
  542. }
  543. return false;
  544. }
  545. template <class A, class B>
  546. bool operator()(A, B) const
  547. {
  548. REALM_ASSERT(false);
  549. return false;
  550. }
  551. template <class A, class B, class C, class D>
  552. bool operator()(A, B, C, D) const
  553. {
  554. REALM_ASSERT(false);
  555. return false;
  556. }
  557. bool operator()(int64_t, int64_t, bool, bool) const
  558. {
  559. REALM_ASSERT(false);
  560. return false;
  561. }
  562. static std::string description()
  563. {
  564. return "ENDSWITH[c]";
  565. }
  566. static const int condition = -1;
  567. };
  568. struct EqualIns : public HackClass {
  569. bool operator()(StringData v1, const char* v1_upper, const char* v1_lower, StringData v2, bool = false,
  570. bool = false) const
  571. {
  572. if (v1.is_null() != v2.is_null())
  573. return false;
  574. return v1.size() == v2.size() && equal_case_fold(v2, v1_upper, v1_lower);
  575. }
  576. // Slow version, used if caller hasn't stored an upper and lower case version
  577. bool operator()(StringData v1, StringData v2, bool = false, bool = false) const
  578. {
  579. if (v1.is_null() != v2.is_null())
  580. return false;
  581. if (v1.size() != v2.size())
  582. return false;
  583. std::string v1_upper = case_map(v1, true, IgnoreErrors);
  584. std::string v1_lower = case_map(v1, false, IgnoreErrors);
  585. return equal_case_fold(v2, v1_upper.c_str(), v1_lower.c_str());
  586. }
  587. bool operator()(BinaryData b1, BinaryData b2, bool = false, bool = false) const
  588. {
  589. StringData s1(b1.data(), b1.size());
  590. StringData s2(b2.data(), b2.size());
  591. return this->operator()(s1, s2, false, false);
  592. }
  593. bool operator()(const QueryValue& m1, const QueryValue& m2) const
  594. {
  595. if (m1.is_null() && m2.is_null()) {
  596. return true;
  597. }
  598. else if (Mixed::types_are_comparable(m1, m2)) {
  599. if (m1.is_type(type_String, type_Binary)) {
  600. return operator()(m1.get_binary(), m2.get_binary(), false, false);
  601. }
  602. else {
  603. return m1 == m2;
  604. }
  605. }
  606. return false;
  607. }
  608. template <class A, class B>
  609. bool operator()(A, B) const
  610. {
  611. REALM_ASSERT(false);
  612. return false;
  613. }
  614. template <class A, class B, class C, class D>
  615. bool operator()(A, B, C, D) const
  616. {
  617. REALM_ASSERT(false);
  618. return false;
  619. }
  620. bool operator()(int64_t, int64_t, bool, bool) const
  621. {
  622. REALM_ASSERT(false);
  623. return false;
  624. }
  625. static std::string description()
  626. {
  627. return "==[c]";
  628. }
  629. static const int condition = -1;
  630. };
  631. struct NotEqualIns : public HackClass {
  632. bool operator()(StringData v1, const char* v1_upper, const char* v1_lower, StringData v2, bool = false,
  633. bool = false) const
  634. {
  635. if (v1.is_null() != v2.is_null())
  636. return true;
  637. return v1.size() != v2.size() || !equal_case_fold(v2, v1_upper, v1_lower);
  638. }
  639. // Slow version, used if caller hasn't stored an upper and lower case version
  640. bool operator()(StringData v1, StringData v2, bool = false, bool = false) const
  641. {
  642. if (v1.is_null() != v2.is_null())
  643. return true;
  644. if (v1.size() != v2.size())
  645. return true;
  646. std::string v1_upper = case_map(v1, true, IgnoreErrors);
  647. std::string v1_lower = case_map(v1, false, IgnoreErrors);
  648. return !equal_case_fold(v2, v1_upper.c_str(), v1_lower.c_str());
  649. }
  650. bool operator()(BinaryData b1, BinaryData b2, bool = false, bool = false) const
  651. {
  652. StringData s1(b1.data(), b1.size());
  653. StringData s2(b2.data(), b2.size());
  654. return this->operator()(s1, s2, false, false);
  655. }
  656. bool operator()(const QueryValue& m1, const QueryValue& m2) const
  657. {
  658. return !EqualIns()(m1, m2);
  659. }
  660. template <class A, class B>
  661. bool operator()(A, B) const
  662. {
  663. REALM_ASSERT(false);
  664. return false;
  665. }
  666. template <class A, class B, class C, class D>
  667. bool operator()(A, B, C, D) const
  668. {
  669. REALM_ASSERT(false);
  670. return false;
  671. }
  672. static std::string description()
  673. {
  674. return "!=[c]";
  675. }
  676. static const int condition = -1;
  677. };
  678. struct Greater {
  679. static const int avx = 0x1E; // _CMP_GT_OQ
  680. template <class T>
  681. bool operator()(const T& v1, const T& v2, bool v1null = false, bool v2null = false) const
  682. {
  683. if (v1null || v2null)
  684. return false;
  685. return v1 > v2;
  686. }
  687. bool operator()(const QueryValue& m1, const QueryValue& m2) const
  688. {
  689. return Mixed::types_are_comparable(m1, m2) && (m1 > m2);
  690. }
  691. static const int condition = cond_Greater;
  692. template <class A, class B, class C, class D>
  693. bool operator()(A, B, C, D) const
  694. {
  695. REALM_ASSERT(false);
  696. return false;
  697. }
  698. bool can_match(int64_t v, int64_t lbound, int64_t ubound)
  699. {
  700. static_cast<void>(lbound);
  701. return ubound > v;
  702. }
  703. bool will_match(int64_t v, int64_t lbound, int64_t ubound)
  704. {
  705. static_cast<void>(ubound);
  706. return lbound > v;
  707. }
  708. static std::string description()
  709. {
  710. return ">";
  711. }
  712. };
  713. struct None {
  714. template <class T>
  715. bool operator()(const T&, const T&, bool = false, bool = false) const
  716. {
  717. return true;
  718. }
  719. static const int condition = cond_None;
  720. template <class A, class B, class C, class D>
  721. bool operator()(A, B, C, D) const
  722. {
  723. REALM_ASSERT(false);
  724. return false;
  725. }
  726. bool can_match(int64_t v, int64_t lbound, int64_t ubound)
  727. {
  728. static_cast<void>(lbound);
  729. static_cast<void>(ubound);
  730. static_cast<void>(v);
  731. return true;
  732. }
  733. bool will_match(int64_t v, int64_t lbound, int64_t ubound)
  734. {
  735. static_cast<void>(lbound);
  736. static_cast<void>(ubound);
  737. static_cast<void>(v);
  738. return true;
  739. }
  740. static std::string description()
  741. {
  742. return "none";
  743. }
  744. };
  745. struct NotNull {
  746. template <class T>
  747. bool operator()(const T&, const T&, bool v = false, bool = false) const
  748. {
  749. return !v;
  750. }
  751. static const int condition = cond_LeftNotNull;
  752. template <class A, class B, class C, class D>
  753. bool operator()(A, B, C, D) const
  754. {
  755. REALM_ASSERT(false);
  756. return false;
  757. }
  758. bool can_match(int64_t v, int64_t lbound, int64_t ubound)
  759. {
  760. static_cast<void>(lbound);
  761. static_cast<void>(ubound);
  762. static_cast<void>(v);
  763. return true;
  764. }
  765. bool will_match(int64_t v, int64_t lbound, int64_t ubound)
  766. {
  767. static_cast<void>(lbound);
  768. static_cast<void>(ubound);
  769. static_cast<void>(v);
  770. return true;
  771. }
  772. static std::string description()
  773. {
  774. return "!= NULL";
  775. }
  776. };
  777. struct Less {
  778. static const int avx = 0x11; // _CMP_LT_OQ
  779. template <class T>
  780. bool operator()(const T& v1, const T& v2, bool v1null = false, bool v2null = false) const
  781. {
  782. if (v1null || v2null)
  783. return false;
  784. return v1 < v2;
  785. }
  786. bool operator()(const QueryValue& m1, const QueryValue& m2) const
  787. {
  788. return Mixed::types_are_comparable(m1, m2) && (m1 < m2);
  789. }
  790. template <class A, class B, class C, class D>
  791. bool operator()(A, B, C, D) const
  792. {
  793. REALM_ASSERT(false);
  794. return false;
  795. }
  796. static const int condition = cond_Less;
  797. bool can_match(int64_t v, int64_t lbound, int64_t ubound)
  798. {
  799. static_cast<void>(ubound);
  800. return lbound < v;
  801. }
  802. bool will_match(int64_t v, int64_t lbound, int64_t ubound)
  803. {
  804. static_cast<void>(lbound);
  805. return ubound < v;
  806. }
  807. static std::string description()
  808. {
  809. return "<";
  810. }
  811. };
  812. struct LessEqual : public HackClass {
  813. static const int avx = 0x12; // _CMP_LE_OQ
  814. template <class T>
  815. bool operator()(const T& v1, const T& v2, bool v1null = false, bool v2null = false) const
  816. {
  817. if (v1null && v2null)
  818. return true;
  819. return (!v1null && !v2null && v1 <= v2);
  820. }
  821. bool operator()(const util::Optional<bool>& v1, const util::Optional<bool>& v2, bool v1null, bool v2null) const
  822. {
  823. if (v1null && v2null)
  824. return false;
  825. return (!v1null && !v2null && *v1 <= *v2);
  826. }
  827. bool operator()(const QueryValue& m1, const QueryValue& m2) const
  828. {
  829. return (m1.is_null() && m2.is_null()) || (Mixed::types_are_comparable(m1, m2) && (m1 <= m2));
  830. }
  831. template <class A, class B, class C, class D>
  832. bool operator()(A, B, C, D) const
  833. {
  834. REALM_ASSERT(false);
  835. return false;
  836. }
  837. static std::string description()
  838. {
  839. return "<=";
  840. }
  841. static const int condition = -1;
  842. };
  843. struct GreaterEqual : public HackClass {
  844. static const int avx = 0x1D; // _CMP_GE_OQ
  845. template <class T>
  846. bool operator()(const T& v1, const T& v2, bool v1null = false, bool v2null = false) const
  847. {
  848. if (v1null && v2null)
  849. return true;
  850. return (!v1null && !v2null && v1 >= v2);
  851. }
  852. bool operator()(const util::Optional<bool>& v1, const util::Optional<bool>& v2, bool v1null, bool v2null) const
  853. {
  854. if (v1null && v2null)
  855. return false;
  856. return (!v1null && !v2null && *v1 >= *v2);
  857. }
  858. bool operator()(const QueryValue& m1, const QueryValue& m2) const
  859. {
  860. return (m1.is_null() && m2.is_null()) || (Mixed::types_are_comparable(m1, m2) && (m1 >= m2));
  861. }
  862. template <class A, class B, class C, class D>
  863. bool operator()(A, B, C, D) const
  864. {
  865. REALM_ASSERT(false);
  866. return false;
  867. }
  868. static std::string description()
  869. {
  870. return ">=";
  871. }
  872. static const int condition = -1;
  873. };
  874. // CompareLess is a temporary hack to have a generalized way to compare any realm types. Todo, enable correct <
  875. // operator of StringData (currently gives circular header dependency with utf8.hpp)
  876. template <class T>
  877. struct CompareLess {
  878. static bool compare(T v1, T v2, bool = false, bool = false)
  879. {
  880. return v1 < v2;
  881. }
  882. };
  883. template <>
  884. struct CompareLess<StringData> {
  885. static bool compare(StringData v1, StringData v2, bool = false, bool = false)
  886. {
  887. bool ret = utf8_compare(v1.data(), v2.data());
  888. return ret;
  889. }
  890. };
  891. } // namespace realm
  892. #endif // REALM_QUERY_CONDITIONS_HPP