spec.hpp 7.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276
  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_SPEC_HPP
  19. #define REALM_SPEC_HPP
  20. #include <realm/util/features.h>
  21. #include <realm/array.hpp>
  22. #include <realm/array_string_short.hpp>
  23. #include <realm/data_type.hpp>
  24. #include <realm/column_type.hpp>
  25. #include <realm/keys.hpp>
  26. namespace realm {
  27. class Table;
  28. class Group;
  29. class Spec {
  30. public:
  31. ~Spec() noexcept;
  32. Allocator& get_alloc() const noexcept;
  33. // insert column at index
  34. void insert_column(size_t column_ndx, ColKey column_key, ColumnType type, StringData name,
  35. int attr = col_attr_None);
  36. ColKey get_key(size_t column_ndx) const;
  37. void rename_column(size_t column_ndx, StringData new_name);
  38. /// Erase the column at the specified index.
  39. ///
  40. /// This function is guaranteed to *never* throw if the spec is
  41. /// used in a non-transactional context, or if the spec has
  42. /// already been successfully modified within the current write
  43. /// transaction.
  44. void erase_column(size_t column_ndx);
  45. // Column info
  46. size_t get_column_count() const noexcept;
  47. size_t get_public_column_count() const noexcept;
  48. ColumnType get_column_type(size_t column_ndx) const noexcept;
  49. StringData get_column_name(size_t column_ndx) const noexcept;
  50. /// Returns size_t(-1) if the specified column is not found.
  51. size_t get_column_index(StringData name) const noexcept;
  52. // Column Attributes
  53. ColumnAttrMask get_column_attr(size_t column_ndx) const noexcept;
  54. void set_dictionary_key_type(size_t column_ndx, DataType key_type);
  55. DataType get_dictionary_key_type(size_t column_ndx) const;
  56. // Auto Enumerated string columns
  57. void upgrade_string_to_enum(size_t column_ndx, ref_type keys_ref);
  58. size_t _get_enumkeys_ndx(size_t column_ndx) const noexcept;
  59. bool is_string_enum_type(size_t column_ndx) const noexcept;
  60. ref_type get_enumkeys_ref(size_t column_ndx, ArrayParent*& keys_parent) noexcept;
  61. //@{
  62. /// Compare two table specs for equality.
  63. bool operator==(const Spec&) const noexcept;
  64. bool operator!=(const Spec&) const noexcept;
  65. //@}
  66. void detach() noexcept;
  67. void destroy() noexcept;
  68. size_t get_ndx_in_parent() const noexcept;
  69. void set_ndx_in_parent(size_t) noexcept;
  70. void verify() const;
  71. private:
  72. // Underlying array structure.
  73. //
  74. Array m_top;
  75. Array m_types; // 1st slot in m_top
  76. ArrayStringShort m_names; // 2nd slot in m_top
  77. Array m_attr; // 3rd slot in m_top
  78. // 4th slot in m_top is vacant
  79. Array m_enumkeys; // 5th slot in m_top
  80. Array m_keys; // 6th slot in m_top
  81. size_t m_num_public_columns;
  82. Spec(Allocator&) noexcept; // Unattached
  83. bool init(ref_type) noexcept;
  84. void init(MemRef) noexcept;
  85. void update_internals() noexcept;
  86. // Returns true in case the ref has changed.
  87. bool init_from_parent() noexcept;
  88. ref_type get_ref() const noexcept;
  89. /// Called in the context of Group::commit() to ensure that
  90. /// attached table accessors stay valid across a commit. Please
  91. /// note that this works only for non-transactional commits. Table
  92. /// accessors obtained during a transaction are always detached
  93. /// when the transaction ends.
  94. void update_from_parent() noexcept;
  95. void set_parent(ArrayParent*, size_t ndx_in_parent) noexcept;
  96. void set_column_attr(size_t column_ndx, ColumnAttrMask attr);
  97. // Migration
  98. bool convert_column_attributes();
  99. bool convert_column_keys(TableKey table_key);
  100. void fix_column_keys(TableKey table_key);
  101. bool has_subspec()
  102. {
  103. return m_top.get(3) != 0;
  104. }
  105. void destroy_subspec()
  106. {
  107. Node::destroy(m_top.get_as_ref(3), m_top.get_alloc());
  108. m_top.set(3, 0);
  109. }
  110. TableKey get_opposite_link_table_key(size_t column_ndx) const noexcept;
  111. size_t get_origin_column_ndx(size_t backlink_col_ndx) const noexcept;
  112. ColKey find_backlink_column(TableKey origin_table_key, size_t spec_ndx) const noexcept;
  113. // Generate a column key only from state in the spec.
  114. ColKey update_colkey(ColKey existing_key, size_t spec_ndx, TableKey table_key);
  115. /// Construct an empty spec and return just the reference to the
  116. /// underlying memory.
  117. static MemRef create_empty_spec(Allocator&);
  118. size_t get_subspec_ndx(size_t column_ndx) const noexcept;
  119. friend class Group;
  120. friend class Table;
  121. };
  122. // Implementation:
  123. inline Allocator& Spec::get_alloc() const noexcept
  124. {
  125. return m_top.get_alloc();
  126. }
  127. // Uninitialized Spec (call init() to init)
  128. inline Spec::Spec(Allocator& alloc) noexcept
  129. : m_top(alloc)
  130. , m_types(alloc)
  131. , m_names(alloc)
  132. , m_attr(alloc)
  133. , m_enumkeys(alloc)
  134. , m_keys(alloc)
  135. {
  136. m_types.set_parent(&m_top, 0);
  137. m_names.set_parent(&m_top, 1);
  138. m_attr.set_parent(&m_top, 2);
  139. m_enumkeys.set_parent(&m_top, 4);
  140. m_keys.set_parent(&m_top, 5);
  141. }
  142. inline bool Spec::init_from_parent() noexcept
  143. {
  144. ref_type ref = m_top.get_ref_from_parent();
  145. return init(ref);
  146. }
  147. inline void Spec::destroy() noexcept
  148. {
  149. m_top.destroy_deep();
  150. }
  151. inline size_t Spec::get_ndx_in_parent() const noexcept
  152. {
  153. return m_top.get_ndx_in_parent();
  154. }
  155. inline void Spec::set_ndx_in_parent(size_t ndx) noexcept
  156. {
  157. m_top.set_ndx_in_parent(ndx);
  158. }
  159. inline ref_type Spec::get_ref() const noexcept
  160. {
  161. return m_top.get_ref();
  162. }
  163. inline void Spec::set_parent(ArrayParent* parent, size_t ndx_in_parent) noexcept
  164. {
  165. m_top.set_parent(parent, ndx_in_parent);
  166. }
  167. inline void Spec::rename_column(size_t column_ndx, StringData new_name)
  168. {
  169. REALM_ASSERT(column_ndx < m_types.size());
  170. m_names.set(column_ndx, new_name);
  171. }
  172. inline size_t Spec::get_column_count() const noexcept
  173. {
  174. // This is the total count of columns, including backlinks (not public)
  175. return m_types.size();
  176. }
  177. inline size_t Spec::get_public_column_count() const noexcept
  178. {
  179. return m_num_public_columns;
  180. }
  181. inline ColumnType Spec::get_column_type(size_t ndx) const noexcept
  182. {
  183. REALM_ASSERT(ndx < get_column_count());
  184. return ColumnType(int(m_types.get(ndx) & 0xFFFF));
  185. }
  186. inline ColumnAttrMask Spec::get_column_attr(size_t ndx) const noexcept
  187. {
  188. REALM_ASSERT(ndx < get_column_count());
  189. return ColumnAttrMask(m_attr.get(ndx));
  190. }
  191. inline void Spec::set_dictionary_key_type(size_t ndx, DataType key_type)
  192. {
  193. auto type = m_types.get(ndx);
  194. m_types.set(ndx, (type & 0xFFFF) + (int64_t(key_type) << 16));
  195. }
  196. inline DataType Spec::get_dictionary_key_type(size_t ndx) const
  197. {
  198. REALM_ASSERT(ndx < get_column_count());
  199. return DataType(int(m_types.get(ndx) >> 16));
  200. }
  201. inline void Spec::set_column_attr(size_t column_ndx, ColumnAttrMask attr)
  202. {
  203. REALM_ASSERT(column_ndx < get_column_count());
  204. // At this point we only allow one attr at a time
  205. // so setting it will overwrite existing. In the future
  206. // we will allow combinations.
  207. m_attr.set(column_ndx, attr.m_value);
  208. update_internals();
  209. }
  210. inline StringData Spec::get_column_name(size_t ndx) const noexcept
  211. {
  212. return m_names.get(ndx);
  213. }
  214. inline size_t Spec::get_column_index(StringData name) const noexcept
  215. {
  216. return m_names.find_first(name);
  217. }
  218. inline bool Spec::operator!=(const Spec& s) const noexcept
  219. {
  220. return !(*this == s);
  221. }
  222. } // namespace realm
  223. #endif // REALM_SPEC_HPP