mixed.hpp 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656
  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_MIXED_HPP
  19. #define REALM_MIXED_HPP
  20. #include <cstdint> // int64_t - not part of C++03, not even required by C++11 (see C++11 section 18.4.1)
  21. #include <cstddef> // size_t
  22. #include <cstring>
  23. #include <realm/keys.hpp>
  24. #include <realm/binary_data.hpp>
  25. #include <realm/data_type.hpp>
  26. #include <realm/string_data.hpp>
  27. #include <realm/timestamp.hpp>
  28. #include <realm/decimal128.hpp>
  29. #include <realm/object_id.hpp>
  30. #include <realm/uuid.hpp>
  31. #include <realm/util/assert.hpp>
  32. #include <realm/utilities.hpp>
  33. namespace realm {
  34. /// This class represents a polymorphic Realm value.
  35. ///
  36. /// At any particular moment an instance of this class stores a
  37. /// definite value of a definite type. If, for instance, that is an
  38. /// integer value, you may call get<int64_t>() to extract that value. You
  39. /// may call get_type() to discover what type of value is currently
  40. /// stored. Calling get<int64_t>() on an instance that does not store an
  41. /// integer, has undefined behavior, and likewise for all the other
  42. /// types that can be stored.
  43. ///
  44. /// It is crucial to understand that the act of extracting a value of
  45. /// a particular type requires definite knowledge about the stored
  46. /// type. Calling a getter method for any particular type, that is not
  47. /// the same type as the stored value, has undefined behavior.
  48. ///
  49. /// While values of numeric types are contained directly in a Mixed
  50. /// instance, character and binary data are merely referenced. A Mixed
  51. /// instance never owns the referenced data, nor does it in any other
  52. /// way attempt to manage its lifetime.
  53. ///
  54. /// For compatibility with C style strings, when a string (character
  55. /// data) is stored in a Realm database, it is always followed by a
  56. /// terminating null character. This is also true when strings are
  57. /// stored in a mixed type column. This means that in the following
  58. /// code, if the 'mixed' value of the 8th row stores a string, then \c
  59. /// c_str will always point to a null-terminated string:
  60. ///
  61. /// \code{.cpp}
  62. ///
  63. /// const char* c_str = my_table[7].mixed.data(); // Always null-terminated
  64. ///
  65. /// \endcode
  66. ///
  67. /// Note that this assumption does not hold in general for strings in
  68. /// instances of Mixed. Indeed there is nothing stopping you from
  69. /// constructing a new Mixed instance that refers to a string without
  70. /// a terminating null character.
  71. ///
  72. /// At the present time no soultion has been found that would allow
  73. /// for a Mixed instance to directly store a reference to a table. The
  74. /// problem is roughly as follows: From most points of view, the
  75. /// desirable thing to do, would be to store the table reference in a
  76. /// Mixed instance as a plain pointer without any ownership
  77. /// semantics. This would have no negative impact on the performance
  78. /// of copying and destroying Mixed instances, and it would serve just
  79. /// fine for passing a table as argument when setting the value of an
  80. /// entry in a mixed column. In that case a copy of the referenced
  81. /// table would be inserted into the mixed column.
  82. ///
  83. /// On the other hand, when retrieving a table reference from a mixed
  84. /// column, storing it as a plain pointer in a Mixed instance is no
  85. /// longer an acceptable option. The complex rules for managing the
  86. /// lifetime of a Table instance, that represents a subtable,
  87. /// necessitates the use of a "smart pointer" such as
  88. /// TableRef. Enhancing the Mixed class to be able to act as a
  89. /// TableRef would be possible, but would also lead to several new
  90. /// problems. One problem is the risk of a Mixed instance outliving a
  91. /// stack allocated Table instance that it references. This would be a
  92. /// fatal error. Another problem is the impact that the nontrivial
  93. /// table reference has on the performance of copying and destroying
  94. /// Mixed instances.
  95. ///
  96. /// \sa StringData
  97. class Mixed {
  98. public:
  99. Mixed() noexcept
  100. : m_type(0)
  101. {
  102. }
  103. Mixed(util::None) noexcept
  104. : Mixed()
  105. {
  106. }
  107. Mixed(realm::null) noexcept
  108. : Mixed()
  109. {
  110. }
  111. Mixed(int i) noexcept
  112. : Mixed(int64_t(i))
  113. {
  114. }
  115. Mixed(int64_t) noexcept;
  116. Mixed(bool) noexcept;
  117. Mixed(float) noexcept;
  118. Mixed(double) noexcept;
  119. Mixed(util::Optional<int64_t>) noexcept;
  120. Mixed(util::Optional<bool>) noexcept;
  121. Mixed(util::Optional<float>) noexcept;
  122. Mixed(util::Optional<double>) noexcept;
  123. Mixed(StringData) noexcept;
  124. Mixed(BinaryData) noexcept;
  125. Mixed(Timestamp) noexcept;
  126. Mixed(Decimal128);
  127. Mixed(ObjectId) noexcept;
  128. Mixed(util::Optional<ObjectId>) noexcept;
  129. Mixed(ObjKey) noexcept;
  130. Mixed(ObjLink) noexcept;
  131. Mixed(UUID) noexcept;
  132. Mixed(util::Optional<UUID>) noexcept;
  133. Mixed(const Obj&) noexcept;
  134. // These are shortcuts for Mixed(StringData(c_str)), and are
  135. // needed to avoid unwanted implicit conversion of char* to bool.
  136. Mixed(char* c_str) noexcept
  137. : Mixed(StringData(c_str))
  138. {
  139. }
  140. Mixed(const char* c_str) noexcept
  141. : Mixed(StringData(c_str))
  142. {
  143. }
  144. Mixed(const std::string& s) noexcept
  145. : Mixed(StringData(s))
  146. {
  147. }
  148. ~Mixed() noexcept
  149. {
  150. }
  151. DataType get_type() const noexcept
  152. {
  153. REALM_ASSERT(m_type);
  154. return DataType(m_type - 1);
  155. }
  156. template <class... Tail>
  157. bool is_type(DataType head, Tail... tail) const noexcept
  158. {
  159. return _is_type(head, tail...);
  160. }
  161. static bool types_are_comparable(const Mixed& l, const Mixed& r);
  162. static bool data_types_are_comparable(DataType l_type, DataType r_type);
  163. template <class T>
  164. T get() const noexcept;
  165. template <class T>
  166. T export_to_type() const noexcept;
  167. // These functions are kept to be backwards compatible
  168. int64_t get_int() const;
  169. bool get_bool() const;
  170. float get_float() const;
  171. double get_double() const;
  172. StringData get_string() const;
  173. BinaryData get_binary() const;
  174. Timestamp get_timestamp() const;
  175. Decimal128 get_decimal() const;
  176. ObjectId get_object_id() const;
  177. UUID get_uuid() const;
  178. ObjLink get_link() const;
  179. bool is_null() const;
  180. bool accumulate_numeric_to(Decimal128& destination) const;
  181. bool is_unresolved_link() const;
  182. bool is_same_type(const Mixed& b) const;
  183. // Will use utf8_compare for strings
  184. int compare(const Mixed& b) const;
  185. // Will compare strings as arrays of signed chars
  186. int compare_signed(const Mixed& b) const;
  187. bool operator==(const Mixed& other) const
  188. {
  189. return compare(other) == 0;
  190. }
  191. bool operator!=(const Mixed& other) const
  192. {
  193. return compare(other) != 0;
  194. }
  195. bool operator<(const Mixed& other) const
  196. {
  197. return compare(other) < 0;
  198. }
  199. bool operator>(const Mixed& other) const
  200. {
  201. return compare(other) > 0;
  202. }
  203. bool operator<=(const Mixed& other) const
  204. {
  205. return compare(other) <= 0;
  206. }
  207. bool operator>=(const Mixed& other) const
  208. {
  209. return compare(other) >= 0;
  210. }
  211. size_t hash() const;
  212. StringData get_index_data(std::array<char, 16>&) const;
  213. void use_buffer(std::string& buf);
  214. protected:
  215. friend std::ostream& operator<<(std::ostream& out, const Mixed& m);
  216. uint32_t m_type;
  217. union {
  218. int64_t int_val;
  219. bool bool_val;
  220. float float_val;
  221. double double_val;
  222. StringData string_val;
  223. BinaryData binary_val;
  224. Timestamp date_val;
  225. ObjectId id_val;
  226. Decimal128 decimal_val;
  227. ObjLink link_val;
  228. UUID uuid_val;
  229. };
  230. private:
  231. static bool _is_type() noexcept
  232. {
  233. return false;
  234. }
  235. bool _is_type(DataType type) const noexcept
  236. {
  237. return m_type == unsigned(int(type) + 1);
  238. }
  239. template <class... Tail>
  240. bool _is_type(DataType head, Tail... tail) const noexcept
  241. {
  242. return _is_type(head) || _is_type(tail...);
  243. }
  244. };
  245. // Implementation:
  246. inline Mixed::Mixed(int64_t v) noexcept
  247. {
  248. m_type = int(type_Int) + 1;
  249. int_val = v;
  250. }
  251. inline Mixed::Mixed(bool v) noexcept
  252. {
  253. m_type = int(type_Bool) + 1;
  254. bool_val = v;
  255. }
  256. inline Mixed::Mixed(float v) noexcept
  257. {
  258. if (null::is_null_float(v)) {
  259. m_type = 0;
  260. }
  261. else {
  262. m_type = int(type_Float) + 1;
  263. float_val = v;
  264. }
  265. }
  266. inline Mixed::Mixed(double v) noexcept
  267. {
  268. if (null::is_null_float(v)) {
  269. m_type = 0;
  270. }
  271. else {
  272. m_type = int(type_Double) + 1;
  273. double_val = v;
  274. }
  275. }
  276. inline Mixed::Mixed(util::Optional<int64_t> v) noexcept
  277. {
  278. if (v) {
  279. m_type = int(type_Int) + 1;
  280. int_val = *v;
  281. }
  282. else {
  283. m_type = 0;
  284. }
  285. }
  286. inline Mixed::Mixed(util::Optional<bool> v) noexcept
  287. {
  288. if (v) {
  289. m_type = int(type_Bool) + 1;
  290. bool_val = *v;
  291. }
  292. else {
  293. m_type = 0;
  294. }
  295. }
  296. inline Mixed::Mixed(util::Optional<float> v) noexcept
  297. {
  298. if (v && !null::is_null_float(*v)) {
  299. m_type = int(type_Float) + 1;
  300. float_val = *v;
  301. }
  302. else {
  303. m_type = 0;
  304. }
  305. }
  306. inline Mixed::Mixed(util::Optional<double> v) noexcept
  307. {
  308. if (v && !null::is_null_float(*v)) {
  309. m_type = int(type_Double) + 1;
  310. double_val = *v;
  311. }
  312. else {
  313. m_type = 0;
  314. }
  315. }
  316. inline Mixed::Mixed(util::Optional<ObjectId> v) noexcept
  317. {
  318. if (v) {
  319. m_type = int(type_ObjectId) + 1;
  320. id_val = *v;
  321. }
  322. else {
  323. m_type = 0;
  324. }
  325. }
  326. inline Mixed::Mixed(util::Optional<UUID> v) noexcept
  327. {
  328. if (v) {
  329. m_type = int(type_UUID) + 1;
  330. uuid_val = *v;
  331. }
  332. else {
  333. m_type = 0;
  334. }
  335. }
  336. inline Mixed::Mixed(StringData v) noexcept
  337. {
  338. if (!v.is_null()) {
  339. m_type = int(type_String) + 1;
  340. string_val = v;
  341. }
  342. else {
  343. m_type = 0;
  344. }
  345. }
  346. inline Mixed::Mixed(BinaryData v) noexcept
  347. {
  348. if (!v.is_null()) {
  349. m_type = int(type_Binary) + 1;
  350. binary_val = v;
  351. }
  352. else {
  353. m_type = 0;
  354. }
  355. }
  356. inline Mixed::Mixed(Timestamp v) noexcept
  357. {
  358. if (!v.is_null()) {
  359. m_type = int(type_Timestamp) + 1;
  360. date_val = v;
  361. }
  362. else {
  363. m_type = 0;
  364. }
  365. }
  366. inline Mixed::Mixed(Decimal128 v)
  367. {
  368. if (!v.is_null()) {
  369. m_type = int(type_Decimal) + 1;
  370. decimal_val = v;
  371. }
  372. else {
  373. m_type = 0;
  374. }
  375. }
  376. inline Mixed::Mixed(ObjectId v) noexcept
  377. {
  378. m_type = int(type_ObjectId) + 1;
  379. id_val = v;
  380. }
  381. inline Mixed::Mixed(UUID v) noexcept
  382. {
  383. m_type = int(type_UUID) + 1;
  384. uuid_val = v;
  385. }
  386. inline Mixed::Mixed(ObjKey v) noexcept
  387. {
  388. if (v) {
  389. m_type = int(type_Link) + 1;
  390. int_val = v.value;
  391. }
  392. else {
  393. m_type = 0;
  394. }
  395. }
  396. inline Mixed::Mixed(ObjLink v) noexcept
  397. {
  398. if (v) {
  399. m_type = int(type_TypedLink) + 1;
  400. link_val = v;
  401. }
  402. else {
  403. m_type = 0;
  404. }
  405. }
  406. template <>
  407. inline null Mixed::get<null>() const noexcept
  408. {
  409. REALM_ASSERT(m_type == 0);
  410. return {};
  411. }
  412. template <>
  413. inline int64_t Mixed::get<int64_t>() const noexcept
  414. {
  415. REALM_ASSERT(get_type() == type_Int);
  416. return int_val;
  417. }
  418. template <>
  419. inline int Mixed::get<int>() const noexcept
  420. {
  421. REALM_ASSERT(get_type() == type_Int);
  422. return int(int_val);
  423. }
  424. inline int64_t Mixed::get_int() const
  425. {
  426. return get<int64_t>();
  427. }
  428. template <>
  429. inline bool Mixed::get<bool>() const noexcept
  430. {
  431. REALM_ASSERT(get_type() == type_Bool);
  432. return bool_val;
  433. }
  434. inline bool Mixed::get_bool() const
  435. {
  436. return get<bool>();
  437. }
  438. template <>
  439. inline float Mixed::get<float>() const noexcept
  440. {
  441. REALM_ASSERT(get_type() == type_Float);
  442. return float_val;
  443. }
  444. inline float Mixed::get_float() const
  445. {
  446. return get<float>();
  447. }
  448. template <>
  449. inline double Mixed::get<double>() const noexcept
  450. {
  451. REALM_ASSERT(get_type() == type_Double);
  452. return double_val;
  453. }
  454. inline double Mixed::get_double() const
  455. {
  456. return get<double>();
  457. }
  458. template <>
  459. inline StringData Mixed::get<StringData>() const noexcept
  460. {
  461. if (is_null())
  462. return StringData();
  463. REALM_ASSERT(get_type() == type_String);
  464. return string_val;
  465. }
  466. inline StringData Mixed::get_string() const
  467. {
  468. return get<StringData>();
  469. }
  470. template <>
  471. inline BinaryData Mixed::get<BinaryData>() const noexcept
  472. {
  473. if (is_null())
  474. return BinaryData();
  475. if (get_type() == type_Binary) {
  476. return binary_val;
  477. }
  478. REALM_ASSERT(get_type() == type_String);
  479. return BinaryData(string_val.data(), string_val.size());
  480. }
  481. inline BinaryData Mixed::get_binary() const
  482. {
  483. return get<BinaryData>();
  484. }
  485. template <>
  486. inline Timestamp Mixed::get<Timestamp>() const noexcept
  487. {
  488. REALM_ASSERT(get_type() == type_Timestamp);
  489. return date_val;
  490. }
  491. inline Timestamp Mixed::get_timestamp() const
  492. {
  493. return get<Timestamp>();
  494. }
  495. template <>
  496. inline Decimal128 Mixed::get<Decimal128>() const noexcept
  497. {
  498. REALM_ASSERT(get_type() == type_Decimal);
  499. return decimal_val;
  500. }
  501. inline Decimal128 Mixed::get_decimal() const
  502. {
  503. return get<Decimal128>();
  504. }
  505. template <>
  506. inline ObjectId Mixed::get<ObjectId>() const noexcept
  507. {
  508. REALM_ASSERT(get_type() == type_ObjectId);
  509. return id_val;
  510. }
  511. inline ObjectId Mixed::get_object_id() const
  512. {
  513. return get<ObjectId>();
  514. }
  515. template <>
  516. inline UUID Mixed::get<UUID>() const noexcept
  517. {
  518. REALM_ASSERT(get_type() == type_UUID);
  519. return uuid_val;
  520. }
  521. inline UUID Mixed::get_uuid() const
  522. {
  523. return get<UUID>();
  524. }
  525. template <>
  526. inline ObjKey Mixed::get<ObjKey>() const noexcept
  527. {
  528. if (get_type() == type_TypedLink)
  529. return link_val.get_obj_key();
  530. REALM_ASSERT(get_type() == type_Link);
  531. return ObjKey(int_val);
  532. }
  533. template <>
  534. inline ObjLink Mixed::get<ObjLink>() const noexcept
  535. {
  536. REALM_ASSERT(get_type() == type_TypedLink);
  537. return link_val;
  538. }
  539. template <>
  540. inline Mixed Mixed::get<Mixed>() const noexcept
  541. {
  542. return *this;
  543. }
  544. inline ObjLink Mixed::get_link() const
  545. {
  546. return get<ObjLink>();
  547. }
  548. inline bool Mixed::is_null() const
  549. {
  550. return (m_type == 0);
  551. }
  552. inline bool Mixed::is_same_type(const Mixed& b) const
  553. {
  554. return (m_type == b.m_type);
  555. }
  556. inline bool Mixed::is_unresolved_link() const
  557. {
  558. if (is_null()) {
  559. return false;
  560. }
  561. else if (get_type() == type_TypedLink) {
  562. return get<ObjLink>().is_unresolved();
  563. }
  564. else if (get_type() == type_Link) {
  565. return get<ObjKey>().is_unresolved();
  566. }
  567. return false;
  568. }
  569. std::ostream& operator<<(std::ostream& out, const Mixed& m);
  570. } // namespace realm
  571. namespace std {
  572. template <>
  573. struct hash<::realm::Mixed> {
  574. inline size_t operator()(const ::realm::Mixed& m) const noexcept
  575. {
  576. return m.hash();
  577. }
  578. };
  579. } // namespace std
  580. #endif // REALM_MIXED_HPP