123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317 |
- /*************************************************************************
- *
- * Copyright 2020 Realm Inc.
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- *
- **************************************************************************/
- #ifndef REALM_SET_HPP
- #define REALM_SET_HPP
- #include <realm/collection.hpp>
- #include <realm/bplustree.hpp>
- #include <realm/array_key.hpp>
- #include <numeric> // std::iota
- namespace realm {
- class SetBase : public CollectionBase {
- public:
- using CollectionBase::CollectionBase;
- virtual ~SetBase() {}
- virtual SetBasePtr clone() const = 0;
- virtual std::pair<size_t, bool> insert_null() = 0;
- virtual std::pair<size_t, bool> erase_null() = 0;
- virtual std::pair<size_t, bool> insert_any(Mixed value) = 0;
- virtual std::pair<size_t, bool> erase_any(Mixed value) = 0;
- protected:
- void insert_repl(Replication* repl, size_t index, Mixed value) const;
- void erase_repl(Replication* repl, size_t index, Mixed value) const;
- void clear_repl(Replication* repl) const;
- static std::vector<Mixed> convert_to_mixed_set(const CollectionBase& rhs);
- };
- template <class T>
- class Set final : public CollectionBaseImpl<SetBase, Set<T>> {
- public:
- using Base = CollectionBaseImpl<SetBase, Set>;
- using value_type = T;
- using iterator = CollectionIterator<Set<T>>;
- Set() = default;
- Set(const Obj& owner, ColKey col_key);
- Set(const Set& other);
- Set(Set&& other) noexcept;
- Set& operator=(const Set& other);
- Set& operator=(Set&& other) noexcept;
- using Base::operator==;
- using Base::operator!=;
- SetBasePtr clone() const final
- {
- return std::make_unique<Set<T>>(*this);
- }
- T get(size_t ndx) const
- {
- const auto current_size = size();
- CollectionBase::validate_index("get()", ndx, current_size);
- return m_tree->get(ndx);
- }
- iterator begin() const noexcept
- {
- return iterator{this, 0};
- }
- iterator end() const noexcept
- {
- return iterator{this, size()};
- }
- size_t find_first(const T& value) const
- {
- return find(value);
- }
- template <class Func>
- void find_all(T value, Func&& func) const
- {
- size_t found = find(value);
- if (found != not_found) {
- func(found);
- }
- }
- bool is_subset_of(const CollectionBase&) const;
- bool is_strict_subset_of(const CollectionBase& rhs) const;
- bool is_superset_of(const CollectionBase& rhs) const;
- bool is_strict_superset_of(const CollectionBase& rhs) const;
- bool intersects(const CollectionBase& rhs) const;
- bool set_equals(const CollectionBase& rhs) const;
- void assign_union(const CollectionBase&);
- void assign_intersection(const CollectionBase&);
- void assign_difference(const CollectionBase&);
- void assign_symmetric_difference(const CollectionBase&);
- /// Insert a value into the set if it does not already exist, returning the index of the inserted value,
- /// or the index of the already-existing value.
- std::pair<size_t, bool> insert(T value);
- /// Find the index of a value in the set, or `size_t(-1)` if it is not in the set.
- size_t find(T value) const;
- /// Erase an element from the set, returning true if the set contained the element.
- std::pair<size_t, bool> erase(T value);
- // Overriding members of CollectionBase:
- size_t size() const final;
- bool is_null(size_t ndx) const final;
- Mixed get_any(size_t ndx) const final
- {
- return get(ndx);
- }
- void clear() final;
- util::Optional<Mixed> min(size_t* return_ndx = nullptr) const final;
- util::Optional<Mixed> max(size_t* return_ndx = nullptr) const final;
- util::Optional<Mixed> sum(size_t* return_cnt = nullptr) const final;
- util::Optional<Mixed> avg(size_t* return_cnt = nullptr) const final;
- std::unique_ptr<CollectionBase> clone_collection() const final
- {
- return std::make_unique<Set<T>>(*this);
- }
- void sort(std::vector<size_t>& indices, bool ascending = true) const final;
- void distinct(std::vector<size_t>& indices, util::Optional<bool> sort_order = util::none) const final;
- // Overriding members of SetBase:
- size_t find_any(Mixed) const final;
- std::pair<size_t, bool> insert_null() final;
- std::pair<size_t, bool> erase_null() final;
- std::pair<size_t, bool> insert_any(Mixed value) final;
- std::pair<size_t, bool> erase_any(Mixed value) final;
- const BPlusTree<T>& get_tree() const
- {
- return *m_tree;
- }
- UpdateStatus update_if_needed() const final
- {
- auto status = Base::update_if_needed();
- switch (status) {
- case UpdateStatus::Detached: {
- m_tree.reset();
- return UpdateStatus::Detached;
- }
- case UpdateStatus::NoChange:
- if (m_tree && m_tree->is_attached()) {
- return UpdateStatus::NoChange;
- }
- // The tree has not been initialized yet for this accessor, so
- // perform lazy initialization by treating it as an update.
- [[fallthrough]];
- case UpdateStatus::Updated: {
- bool attached = init_from_parent(false);
- return attached ? UpdateStatus::Updated : UpdateStatus::Detached;
- }
- }
- REALM_UNREACHABLE();
- }
- UpdateStatus ensure_created() final
- {
- auto status = Base::ensure_created();
- switch (status) {
- case UpdateStatus::Detached:
- break; // Not possible (would have thrown earlier).
- case UpdateStatus::NoChange: {
- if (m_tree && m_tree->is_attached()) {
- return UpdateStatus::NoChange;
- }
- // The tree has not been initialized yet for this accessor, so
- // perform lazy initialization by treating it as an update.
- [[fallthrough]];
- }
- case UpdateStatus::Updated: {
- bool attached = init_from_parent(true);
- REALM_ASSERT(attached);
- return attached ? UpdateStatus::Updated : UpdateStatus::Detached;
- }
- }
- REALM_UNREACHABLE();
- }
- void migrate();
- private:
- // Friend because it needs access to `m_tree` in the implementation of
- // `ObjCollectionBase::get_mutable_tree()`.
- friend class LnkSet;
- // BPlusTree must be wrapped in an `std::unique_ptr` because it is not
- // default-constructible, due to its `Allocator&` member.
- mutable std::unique_ptr<BPlusTree<T>> m_tree;
- using Base::bump_content_version;
- using Base::m_col_key;
- using Base::m_nullable;
- using Base::m_obj;
- bool init_from_parent(bool allow_create) const
- {
- if (!m_tree) {
- m_tree.reset(new BPlusTree<T>(m_obj.get_alloc()));
- const ArrayParent* parent = this;
- m_tree->set_parent(const_cast<ArrayParent*>(parent), 0);
- }
- if (m_tree->init_from_parent()) {
- // All is well
- return true;
- }
- if (!allow_create) {
- return false;
- }
- // The ref in the column was NULL, create the tree in place.
- m_tree->create();
- REALM_ASSERT(m_tree->is_attached());
- return true;
- }
- /// Update the accessor and return true if it is attached after the update.
- inline bool update() const
- {
- return update_if_needed() != UpdateStatus::Detached;
- }
- // `do_` methods here perform the action after preconditions have been
- // checked (bounds check, writability, etc.).
- void do_insert(size_t ndx, T value);
- void do_erase(size_t ndx);
- void do_clear();
- iterator find_impl(const T& value) const;
- template <class It1, class It2>
- bool is_subset_of(It1, It2) const;
- template <class It1, class It2>
- bool is_superset_of(It1, It2) const;
- template <class It1, class It2>
- bool intersects(It1, It2) const;
- template <class It1, class It2>
- void assign_union(It1, It2);
- template <class It1, class It2>
- void assign_intersection(It1, It2);
- template <class It1, class It2>
- void assign_difference(It1, It2);
- template <class It1, class It2>
- void assign_symmetric_difference(It1, It2);
- static std::vector<T> convert_to_set(const CollectionBase& rhs, bool nullable);
- };
- class LnkSet final : public ObjCollectionBase<SetBase> {
- public:
- using Base = ObjCollectionBase<SetBase>;
- using value_type = ObjKey;
- using iterator = CollectionIterator<LnkSet>;
- LnkSet() = default;
- LnkSet(const Obj& owner, ColKey col_key)
- : m_set(owner, col_key)
- {
- }
- LnkSet(const LnkSet&) = default;
- LnkSet(LnkSet&&) = default;
- LnkSet& operator=(const LnkSet&) = default;
- LnkSet& operator=(LnkSet&&) = default;
- bool operator==(const LnkSet& other) const;
- bool operator!=(const LnkSet& other) const;
- ObjKey get(size_t ndx) const;
- size_t find(ObjKey) const;
- size_t find_first(ObjKey) const;
- std::pair<size_t, bool> insert(ObjKey);
- std::pair<size_t, bool> erase(ObjKey);
- bool is_subset_of(const CollectionBase&) const;
- bool is_strict_subset_of(const CollectionBase& rhs) const;
- bool is_superset_of(const CollectionBase& rhs) const;
- bool is_strict_superset_of(const CollectionBase& rhs) const;
- bool intersects(const CollectionBase& rhs) const;
- bool set_equals(const CollectionBase& rhs) const;
- void assign_union(const CollectionBase&);
- void assign_intersection(const CollectionBase&);
- void assign_difference(const CollectionBase&);
- void assign_symmetric_difference(const CollectionBase&);
- // Overriding members of CollectionBase:
- using CollectionBase::get_owner_key;
- CollectionBasePtr clone_collection() const
- {
- return clone_linkset();
- }
- size_t size() const final;
- bool is_null(size_t ndx) const final;
- Mixed get_any(size_t ndx) const final;
- void clear() final;
- util::Optional<Mixed> min(size_t* return_ndx = nullptr) const final;
- util::Optional<Mixed> max(size_t* return_ndx = nullptr) const final;
- util::Optional<Mixed> sum(size_t* return_cnt = nullptr) const final;
- util::Optional<Mixed> avg(size_t* return_cnt = nullptr) const final;
- void sort(std::vector<size_t>& indices, bool ascending = true) const final;
- void distinct(std::vector<size_t>& indices, util::Optional<bool> sort_order = util::none) const final;
- const Obj& get_obj() const noexcept final;
- bool is_attached() const final;
- bool has_changed() const final;
- ColKey get_col_key() const noexcept final;
- // Overriding members of SetBase:
- SetBasePtr clone() const
- {
- return clone_linkset();
- }
- // Overriding members of ObjList:
- LinkCollectionPtr clone_obj_list() const final
- {
- return clone_linkset();
- }
- size_t find_any(Mixed) const final;
- std::pair<size_t, bool> insert_null() final;
- std::pair<size_t, bool> erase_null() final;
- std::pair<size_t, bool> insert_any(Mixed value) final;
- std::pair<size_t, bool> erase_any(Mixed value) final;
- // Overriding members of ObjList:
- Obj get_object(size_t ndx) const final;
- ObjKey get_key(size_t ndx) const final;
- // LnkSet interface:
- std::unique_ptr<LnkSet> clone_linkset() const
- {
- // FIXME: The copy constructor requires this.
- update_if_needed();
- return std::make_unique<LnkSet>(*this);
- }
- template <class Func>
- void find_all(ObjKey value, Func&& func) const
- {
- if (value.is_unresolved()) {
- return;
- }
- m_set.find_all(value, [&](size_t ndx) {
- func(real2virtual(ndx));
- });
- }
- TableView get_sorted_view(SortDescriptor order) const;
- TableView get_sorted_view(ColKey column_key, bool ascending = true);
- void remove_target_row(size_t link_ndx);
- void remove_all_target_rows();
- iterator begin() const noexcept
- {
- return iterator{this, 0};
- }
- iterator end() const noexcept
- {
- return iterator{this, size()};
- }
- private:
- Set<ObjKey> m_set;
- // Overriding members of ObjCollectionBase:
- UpdateStatus do_update_if_needed() const final
- {
- return m_set.update_if_needed();
- }
- BPlusTree<ObjKey>* get_mutable_tree() const final
- {
- return m_set.m_tree.get();
- }
- };
- template <>
- void Set<ObjKey>::do_insert(size_t, ObjKey);
- template <>
- void Set<ObjKey>::do_erase(size_t);
- template <>
- void Set<ObjKey>::do_clear();
- template <>
- void Set<ObjLink>::do_insert(size_t, ObjLink);
- template <>
- void Set<ObjLink>::do_erase(size_t);
- template <>
- void Set<Mixed>::do_insert(size_t, Mixed);
- template <>
- void Set<Mixed>::do_erase(size_t);
- template <>
- void Set<Mixed>::do_clear();
- template <>
- void Set<Mixed>::migrate();
- /// Compare set elements.
- ///
- /// We cannot use `std::less<>` directly, because the ordering of set elements
- /// impacts the file format. For primitive types this is trivial (and can indeed
- /// be just `std::less<>`), but for example `Mixed` has specialized comparison
- /// that defines equality of numeric types.
- template <class T>
- struct SetElementLessThan {
- bool operator()(const T& a, const T& b) const noexcept
- {
- // CAUTION: This routine is technically part of the file format, because
- // it determines the storage order of Set elements.
- return a < b;
- }
- };
- template <class T>
- struct SetElementEquals {
- bool operator()(const T& a, const T& b) const noexcept
- {
- // CAUTION: This routine is technically part of the file format, because
- // it determines the storage order of Set elements.
- return a == b;
- }
- };
- template <>
- struct SetElementLessThan<Mixed> {
- bool operator()(const Mixed& a, const Mixed& b) const noexcept
- {
- // CAUTION: This routine is technically part of the file format, because
- // it determines the storage order of Set elements.
- // These are the rules for comparison of Mixed types in a Set<Mixed>:
- // - If both values are null they are equal
- // - If only one value is null, that value is lesser than the other
- // - All numeric types are compared as the corresponding real numbers
- // would compare. So integer 3 equals double 3.
- // - String and binary types are compared using lexicographical comparison.
- // - All other types are compared using the comparison operators defined
- // for the types.
- // - If two values have different types, the rank of the types are compared.
- // the rank is as follows:
- // boolean
- // numeric
- // string
- // binary
- // Timestamp
- // ObjectId
- // UUID
- // TypedLink
- // Link
- //
- // The current Mixed::compare function implements these rules except when comparing
- // string and binary. If that function is changed we should either implement the rules
- // here or upgrade all Set<Mixed> columns.
- if (a.is_type(type_String) && b.is_type(type_Binary)) {
- return true;
- }
- if (a.is_type(type_Binary) && b.is_type(type_String)) {
- return false;
- }
- return a.compare(b) < 0;
- }
- };
- template <>
- struct SetElementEquals<Mixed> {
- bool operator()(const Mixed& a, const Mixed& b) const noexcept
- {
- // CAUTION: This routine is technically part of the file format, because
- // it determines the storage order of Set elements.
- // See comments above
- if (a.is_type(type_String) && b.is_type(type_Binary)) {
- return false;
- }
- if (a.is_type(type_Binary) && b.is_type(type_String)) {
- return false;
- }
- return a.compare(b) == 0;
- }
- };
- template <class T>
- inline Set<T>::Set(const Obj& obj, ColKey col_key)
- : Base(obj, col_key)
- {
- if (!col_key.is_set()) {
- throw InvalidArgument(ErrorCodes::TypeMismatch, "Property not a set");
- }
- check_column_type<value_type>(m_col_key);
- }
- template <class T>
- inline Set<T>::Set(const Set& other)
- : Base(static_cast<const Base&>(other))
- {
- // Reset the content version so we can rely on init_from_parent() being
- // called lazily when the accessor is used.
- Base::reset_content_version();
- }
- template <class T>
- inline Set<T>::Set(Set&& other) noexcept
- : Base(static_cast<Base&&>(other))
- , m_tree(std::exchange(other.m_tree, nullptr))
- {
- if (m_tree) {
- m_tree->set_parent(this, 0);
- }
- }
- template <class T>
- inline Set<T>& Set<T>::operator=(const Set& other)
- {
- Base::operator=(static_cast<const Base&>(other));
- if (this != &other) {
- // Just reset the pointer and rely on init_from_parent() being called
- // when the accessor is actually used.
- m_tree.reset();
- Base::reset_content_version();
- }
- return *this;
- }
- template <class T>
- inline Set<T>& Set<T>::operator=(Set&& other) noexcept
- {
- Base::operator=(static_cast<Base&&>(other));
- if (this != &other) {
- m_tree = std::exchange(other.m_tree, nullptr);
- if (m_tree) {
- m_tree->set_parent(this, 0);
- // Note: We do not need to call reset_content_version(), because we
- // took both `m_tree` and `m_content_version` from `other`.
- }
- }
- return *this;
- }
- template <typename U>
- Set<U> Obj::get_set(ColKey col_key) const
- {
- return Set<U>(*this, col_key);
- }
- template <typename U>
- inline SetPtr<U> Obj::get_set_ptr(ColKey col_key) const
- {
- return std::make_unique<Set<U>>(*this, col_key);
- }
- inline LnkSet Obj::get_linkset(ColKey col_key) const
- {
- return LnkSet{*this, col_key};
- }
- inline LnkSet Obj::get_linkset(StringData col_name) const
- {
- return get_linkset(get_column_key(col_name));
- }
- inline LnkSetPtr Obj::get_linkset_ptr(ColKey col_key) const
- {
- return std::make_unique<LnkSet>(*this, col_key);
- }
- template <class T>
- size_t Set<T>::find(T value) const
- {
- auto it = find_impl(value);
- if (it != end() && SetElementEquals<T>{}(*it, value)) {
- return it.index();
- }
- return npos;
- }
- template <class T>
- size_t Set<T>::find_any(Mixed value) const
- {
- if constexpr (std::is_same_v<T, Mixed>) {
- return find(value);
- }
- else {
- if (value.is_null()) {
- if (!m_nullable) {
- return not_found;
- }
- return find(BPlusTree<T>::default_value(true));
- }
- else {
- return find(value.get<typename util::RemoveOptional<T>::type>());
- }
- }
- }
- template <class T>
- REALM_NOINLINE auto Set<T>::find_impl(const T& value) const -> iterator
- {
- auto b = this->begin();
- auto e = this->end(); // Note: This ends up calling `update_if_needed()`.
- return std::lower_bound(b, e, value, SetElementLessThan<T>{});
- }
- template <class T>
- std::pair<size_t, bool> Set<T>::insert(T value)
- {
- update_if_needed();
- if (value_is_null(value) && !m_nullable)
- throw InvalidArgument(ErrorCodes::PropertyNotNullable,
- util::format("Set: %1", CollectionBase::get_property_name()));
- ensure_created();
- auto it = find_impl(value);
- if (it != this->end() && SetElementEquals<T>{}(*it, value)) {
- return {it.index(), false};
- }
- if (Replication* repl = m_obj.get_replication()) {
- // FIXME: We should emit an instruction regardless of element presence for the purposes of conflict
- // resolution in synchronized databases. The reason is that the new insertion may come at a later time
- // than an interleaving erase instruction, so emitting the instruction ensures that last "write" wins.
- this->insert_repl(repl, it.index(), value);
- }
- do_insert(it.index(), value);
- bump_content_version();
- return {it.index(), true};
- }
- template <class T>
- std::pair<size_t, bool> Set<T>::insert_any(Mixed value)
- {
- if constexpr (std::is_same_v<T, Mixed>) {
- return insert(value);
- }
- else {
- if (value.is_null()) {
- return insert_null();
- }
- else {
- return insert(value.get<typename util::RemoveOptional<T>::type>());
- }
- }
- }
- template <class T>
- std::pair<size_t, bool> Set<T>::erase(T value)
- {
- auto it = find_impl(value); // Note: This ends up calling `update_if_needed()`.
- if (it == end() || !SetElementEquals<T>{}(*it, value)) {
- return {npos, false};
- }
- if (Replication* repl = m_obj.get_replication()) {
- this->erase_repl(repl, it.index(), value);
- }
- do_erase(it.index());
- bump_content_version();
- return {it.index(), true};
- }
- template <class T>
- std::pair<size_t, bool> Set<T>::erase_any(Mixed value)
- {
- if constexpr (std::is_same_v<T, Mixed>) {
- return erase(value);
- }
- else {
- if (value.is_null()) {
- return erase_null();
- }
- else {
- return erase(value.get<typename util::RemoveOptional<T>::type>());
- }
- }
- }
- template <class T>
- inline std::pair<size_t, bool> Set<T>::insert_null()
- {
- return insert(BPlusTree<T>::default_value(this->m_nullable));
- }
- template <class T>
- std::pair<size_t, bool> Set<T>::erase_null()
- {
- return erase(BPlusTree<T>::default_value(this->m_nullable));
- }
- template <class T>
- REALM_NOINLINE size_t Set<T>::size() const
- {
- return update() ? m_tree->size() : 0;
- }
- template <class T>
- inline bool Set<T>::is_null(size_t ndx) const
- {
- return m_nullable && value_is_null(get(ndx));
- }
- template <class T>
- inline void Set<T>::clear()
- {
- if (size() > 0) {
- if (Replication* repl = this->m_obj.get_replication()) {
- this->clear_repl(repl);
- }
- do_clear();
- bump_content_version();
- }
- }
- template <class T>
- inline util::Optional<Mixed> Set<T>::min(size_t* return_ndx) const
- {
- if (update()) {
- return MinHelper<T>::eval(*m_tree, return_ndx);
- }
- return MinHelper<T>::not_found(return_ndx);
- }
- template <class T>
- inline util::Optional<Mixed> Set<T>::max(size_t* return_ndx) const
- {
- if (update()) {
- return MaxHelper<T>::eval(*m_tree, return_ndx);
- }
- return MaxHelper<T>::not_found(return_ndx);
- }
- template <class T>
- inline util::Optional<Mixed> Set<T>::sum(size_t* return_cnt) const
- {
- if (update()) {
- return SumHelper<T>::eval(*m_tree, return_cnt);
- }
- return SumHelper<T>::not_found(return_cnt);
- }
- template <class T>
- inline util::Optional<Mixed> Set<T>::avg(size_t* return_cnt) const
- {
- if (update()) {
- return AverageHelper<T>::eval(*m_tree, return_cnt);
- }
- return AverageHelper<T>::not_found(return_cnt);
- }
- void set_sorted_indices(size_t sz, std::vector<size_t>& indices, bool ascending);
- template <class T>
- inline void Set<T>::sort(std::vector<size_t>& indices, bool ascending) const
- {
- auto sz = size();
- set_sorted_indices(sz, indices, ascending);
- }
- template <>
- void Set<Mixed>::sort(std::vector<size_t>& indices, bool ascending) const;
- template <class T>
- inline void Set<T>::distinct(std::vector<size_t>& indices, util::Optional<bool> sort_order) const
- {
- auto ascending = !sort_order || *sort_order;
- sort(indices, ascending);
- }
- template <class T>
- inline void Set<T>::do_insert(size_t ndx, T value)
- {
- m_tree->insert(ndx, value);
- }
- template <class T>
- inline void Set<T>::do_erase(size_t ndx)
- {
- m_tree->erase(ndx);
- }
- template <class T>
- inline void Set<T>::do_clear()
- {
- m_tree->clear();
- }
- template <class T>
- std::vector<T> Set<T>::convert_to_set(const CollectionBase& rhs, bool nullable)
- {
- if constexpr (std::is_same_v<T, Mixed>) {
- return SetBase::convert_to_mixed_set(rhs);
- }
- std::vector<Mixed> mixed = SetBase::convert_to_mixed_set(rhs);
- std::vector<T> ret;
- ret.reserve(mixed.size());
- for (auto&& val : mixed) {
- if constexpr (std::is_same_v<T, ObjKey>) {
- static_cast<void>(nullable);
- if (val.is_type(type_Link, type_TypedLink)) {
- ret.push_back(val.get<ObjKey>());
- }
- }
- else {
- if (val.is_type(ColumnTypeTraits<T>::id)) {
- ret.push_back(val.get<T>());
- }
- else if (val.is_null() && nullable) {
- ret.push_back(BPlusTree<T>::default_value(true));
- }
- }
- }
- return ret;
- }
- template <class T>
- bool Set<T>::is_subset_of(const CollectionBase& rhs) const
- {
- if (auto other_set = dynamic_cast<const Set<T>*>(&rhs)) {
- return is_subset_of(other_set->begin(), other_set->end());
- }
- auto other_set = convert_to_set(rhs, m_nullable);
- return is_subset_of(other_set.begin(), other_set.end());
- }
- template <class T>
- template <class It1, class It2>
- bool Set<T>::is_subset_of(It1 first, It2 last) const
- {
- return std::includes(first, last, begin(), end(), SetElementLessThan<T>{});
- }
- template <class T>
- bool Set<T>::is_strict_subset_of(const CollectionBase& rhs) const
- {
- if (auto other_set = dynamic_cast<const Set<T>*>(&rhs)) {
- return size() != rhs.size() && is_subset_of(other_set->begin(), other_set->end());
- }
- auto other_set = convert_to_set(rhs, m_nullable);
- return size() != other_set.size() && is_subset_of(other_set.begin(), other_set.end());
- }
- template <class T>
- bool Set<T>::is_superset_of(const CollectionBase& rhs) const
- {
- if (auto other_set = dynamic_cast<const Set<T>*>(&rhs)) {
- return is_superset_of(other_set->begin(), other_set->end());
- }
- auto other_set = convert_to_set(rhs, m_nullable);
- return is_superset_of(other_set.begin(), other_set.end());
- }
- template <class T>
- template <class It1, class It2>
- bool Set<T>::is_superset_of(It1 first, It2 last) const
- {
- return std::includes(begin(), end(), first, last, SetElementLessThan<T>{});
- }
- template <class T>
- bool Set<T>::is_strict_superset_of(const CollectionBase& rhs) const
- {
- if (auto other_set = dynamic_cast<const Set<T>*>(&rhs)) {
- return size() != rhs.size() && is_superset_of(other_set->begin(), other_set->end());
- }
- auto other_set = convert_to_set(rhs, m_nullable);
- return size() != other_set.size() && is_superset_of(other_set.begin(), other_set.end());
- }
- template <class T>
- bool Set<T>::intersects(const CollectionBase& rhs) const
- {
- if (auto other_set = dynamic_cast<const Set<T>*>(&rhs)) {
- return intersects(other_set->begin(), other_set->end());
- }
- auto other_set = convert_to_set(rhs, m_nullable);
- return intersects(other_set.begin(), other_set.end());
- }
- template <class T>
- template <class It1, class It2>
- bool Set<T>::intersects(It1 first, It2 last) const
- {
- SetElementLessThan<T> less;
- auto it = begin();
- while (it != end() && first != last) {
- if (less(*it, *first)) {
- ++it;
- }
- else if (less(*first, *it)) {
- ++first;
- }
- else {
- return true;
- }
- }
- return false;
- }
- template <class T>
- bool Set<T>::set_equals(const CollectionBase& rhs) const
- {
- if (auto other_set = dynamic_cast<const Set<T>*>(&rhs)) {
- return size() == rhs.size() && is_subset_of(other_set->begin(), other_set->end());
- }
- auto other_set = convert_to_set(rhs, m_nullable);
- return size() == other_set.size() && is_subset_of(other_set.begin(), other_set.end());
- }
- template <class T>
- inline void Set<T>::assign_union(const CollectionBase& rhs)
- {
- if (auto other_set = dynamic_cast<const Set<T>*>(&rhs)) {
- return assign_union(other_set->begin(), other_set->end());
- }
- auto other_set = convert_to_set(rhs, m_nullable);
- return assign_union(other_set.begin(), other_set.end());
- }
- template <class T>
- template <class It1, class It2>
- void Set<T>::assign_union(It1 first, It2 last)
- {
- std::vector<T> the_diff;
- std::set_difference(first, last, begin(), end(), std::back_inserter(the_diff), SetElementLessThan<T>{});
- // 'the_diff' now contains all the elements that are in foreign set, but not in 'this'
- // Now insert those elements
- for (auto&& value : the_diff) {
- insert(value);
- }
- }
- template <class T>
- inline void Set<T>::assign_intersection(const CollectionBase& rhs)
- {
- if (auto other_set = dynamic_cast<const Set<T>*>(&rhs)) {
- return assign_intersection(other_set->begin(), other_set->end());
- }
- auto other_set = convert_to_set(rhs, m_nullable);
- return assign_intersection(other_set.begin(), other_set.end());
- }
- template <class T>
- template <class It1, class It2>
- void Set<T>::assign_intersection(It1 first, It2 last)
- {
- std::vector<T> intersection;
- std::set_intersection(first, last, begin(), end(), std::back_inserter(intersection), SetElementLessThan<T>{});
- clear();
- // Elements in intersection comes from foreign set, so ok to use here
- for (auto&& value : intersection) {
- insert(value);
- }
- }
- template <class T>
- inline void Set<T>::assign_difference(const CollectionBase& rhs)
- {
- if (auto other_set = dynamic_cast<const Set<T>*>(&rhs)) {
- return assign_difference(other_set->begin(), other_set->end());
- }
- auto other_set = convert_to_set(rhs, m_nullable);
- return assign_difference(other_set.begin(), other_set.end());
- }
- template <class T>
- template <class It1, class It2>
- void Set<T>::assign_difference(It1 first, It2 last)
- {
- std::vector<T> intersection;
- std::set_intersection(first, last, begin(), end(), std::back_inserter(intersection), SetElementLessThan<T>{});
- // 'intersection' now contains all the elements that are in both foreign set and 'this'.
- // Remove those elements. The elements comes from the foreign set, so ok to refer to.
- for (auto&& value : intersection) {
- erase(value);
- }
- }
- template <class T>
- inline void Set<T>::assign_symmetric_difference(const CollectionBase& rhs)
- {
- if (auto other_set = dynamic_cast<const Set<T>*>(&rhs)) {
- return assign_symmetric_difference(other_set->begin(), other_set->end());
- }
- auto other_set = convert_to_set(rhs, m_nullable);
- return assign_symmetric_difference(other_set.begin(), other_set.end());
- }
- template <class T>
- template <class It1, class It2>
- void Set<T>::assign_symmetric_difference(It1 first, It2 last)
- {
- std::vector<T> difference;
- std::set_difference(first, last, begin(), end(), std::back_inserter(difference), SetElementLessThan<T>{});
- std::vector<T> intersection;
- std::set_intersection(first, last, begin(), end(), std::back_inserter(intersection), SetElementLessThan<T>{});
- // Now remove the common elements and add the differences
- for (auto&& value : intersection) {
- erase(value);
- }
- for (auto&& value : difference) {
- insert(value);
- }
- }
- inline bool LnkSet::operator==(const LnkSet& other) const
- {
- return m_set == other.m_set;
- }
- inline bool LnkSet::operator!=(const LnkSet& other) const
- {
- return m_set != other.m_set;
- }
- inline ObjKey LnkSet::get(size_t ndx) const
- {
- const auto current_size = size();
- if (ndx >= current_size) {
- throw OutOfBounds(util::format("Invalid index into set: %1", CollectionBase::get_property_name()), ndx,
- current_size);
- }
- return m_set.m_tree->get(virtual2real(ndx));
- }
- inline size_t LnkSet::find(ObjKey value) const
- {
- if (value.is_unresolved()) {
- return not_found;
- }
- update_if_needed();
- size_t ndx = m_set.find(value);
- if (ndx == not_found) {
- return not_found;
- }
- return real2virtual(ndx);
- }
- inline size_t LnkSet::find_first(ObjKey value) const
- {
- return find(value);
- }
- inline size_t LnkSet::size() const
- {
- update_if_needed();
- return m_set.size() - num_unresolved();
- }
- inline std::pair<size_t, bool> LnkSet::insert(ObjKey value)
- {
- REALM_ASSERT(!value.is_unresolved());
- update_if_needed();
- auto [ndx, inserted] = m_set.insert(value);
- if (inserted) {
- update_unresolved(UpdateStatus::Updated);
- }
- return {real2virtual(ndx), inserted};
- }
- inline std::pair<size_t, bool> LnkSet::erase(ObjKey value)
- {
- REALM_ASSERT(!value.is_unresolved());
- update_if_needed();
- auto [ndx, removed] = m_set.erase(value);
- if (removed) {
- update_unresolved(UpdateStatus::Updated);
- ndx = real2virtual(ndx);
- }
- return {ndx, removed};
- }
- inline bool LnkSet::is_null(size_t ndx) const
- {
- update_if_needed();
- return m_set.is_null(virtual2real(ndx));
- }
- inline Mixed LnkSet::get_any(size_t ndx) const
- {
- update_if_needed();
- auto obj_key = m_set.get(virtual2real(ndx));
- return ObjLink{get_target_table()->get_key(), obj_key};
- }
- inline std::pair<size_t, bool> LnkSet::insert_null()
- {
- update_if_needed();
- auto [ndx, inserted] = m_set.insert_null();
- if (inserted) {
- update_unresolved(UpdateStatus::Updated);
- }
- return {real2virtual(ndx), inserted};
- }
- inline std::pair<size_t, bool> LnkSet::erase_null()
- {
- update_if_needed();
- auto [ndx, erased] = m_set.erase_null();
- if (erased) {
- update_unresolved(UpdateStatus::Updated);
- ndx = real2virtual(ndx);
- }
- return {ndx, erased};
- }
- inline std::pair<size_t, bool> LnkSet::insert_any(Mixed value)
- {
- update_if_needed();
- auto [ndx, inserted] = m_set.insert_any(value);
- if (inserted) {
- update_unresolved(UpdateStatus::Updated);
- }
- return {real2virtual(ndx), inserted};
- }
- inline std::pair<size_t, bool> LnkSet::erase_any(Mixed value)
- {
- auto [ndx, erased] = m_set.erase_any(value);
- if (erased) {
- update_unresolved(UpdateStatus::Updated);
- ndx = real2virtual(ndx);
- }
- return {ndx, erased};
- }
- inline void LnkSet::clear()
- {
- // Note: Explicit call to `ensure_writable()` not needed, because we
- // explicitly call `clear_unresolved()`.
- m_set.clear();
- clear_unresolved();
- }
- inline util::Optional<Mixed> LnkSet::min(size_t* return_ndx) const
- {
- update_if_needed();
- size_t found = not_found;
- auto value = m_set.min(&found);
- if (found != not_found && return_ndx) {
- *return_ndx = real2virtual(found);
- }
- return value;
- }
- inline util::Optional<Mixed> LnkSet::max(size_t* return_ndx) const
- {
- update_if_needed();
- size_t found = not_found;
- auto value = m_set.max(&found);
- if (found != not_found && return_ndx) {
- *return_ndx = real2virtual(found);
- }
- return value;
- }
- inline util::Optional<Mixed> LnkSet::sum(size_t* return_cnt) const
- {
- static_cast<void>(return_cnt);
- REALM_TERMINATE("Not implemented");
- }
- inline util::Optional<Mixed> LnkSet::avg(size_t* return_cnt) const
- {
- static_cast<void>(return_cnt);
- REALM_TERMINATE("Not implemented");
- }
- inline void LnkSet::sort(std::vector<size_t>& indices, bool ascending) const
- {
- update_if_needed();
- // Map the input indices to real indices.
- std::transform(indices.begin(), indices.end(), indices.begin(), [this](size_t ndx) {
- return virtual2real(ndx);
- });
- m_set.sort(indices, ascending);
- // Map the output indices to virtual indices.
- std::transform(indices.begin(), indices.end(), indices.begin(), [this](size_t ndx) {
- return real2virtual(ndx);
- });
- }
- inline void LnkSet::distinct(std::vector<size_t>& indices, util::Optional<bool> sort_order) const
- {
- update_if_needed();
- // Map the input indices to real indices.
- std::transform(indices.begin(), indices.end(), indices.begin(), [this](size_t ndx) {
- return virtual2real(ndx);
- });
- m_set.distinct(indices, sort_order);
- // Map the output indices to virtual indices.
- std::transform(indices.begin(), indices.end(), indices.begin(), [this](size_t ndx) {
- return real2virtual(ndx);
- });
- }
- inline const Obj& LnkSet::get_obj() const noexcept
- {
- return m_set.get_obj();
- }
- inline bool LnkSet::is_attached() const
- {
- return m_set.is_attached();
- }
- inline bool LnkSet::has_changed() const
- {
- return m_set.has_changed();
- }
- inline ColKey LnkSet::get_col_key() const noexcept
- {
- return m_set.get_col_key();
- }
- inline size_t LnkSet::find_any(Mixed value) const
- {
- if (value.is_null())
- return not_found;
- const auto type = value.get_type();
- if (type == type_Link) {
- return find(value.get<ObjKey>());
- }
- if (type == type_TypedLink) {
- auto link = value.get_link();
- if (link.get_table_key() == get_target_table()->get_key()) {
- return find(link.get_obj_key());
- }
- }
- return not_found;
- }
- inline Obj LnkSet::get_object(size_t ndx) const
- {
- ObjKey key = get(ndx);
- return get_target_table()->get_object(key);
- }
- inline ObjKey LnkSet::get_key(size_t ndx) const
- {
- return get(ndx);
- }
- inline void LnkSet::assign_union(const CollectionBase& rhs)
- {
- m_set.assign_union(rhs);
- update_unresolved(UpdateStatus::Updated);
- }
- inline void LnkSet::assign_intersection(const CollectionBase& rhs)
- {
- m_set.assign_intersection(rhs);
- update_unresolved(UpdateStatus::Updated);
- }
- inline void LnkSet::assign_difference(const CollectionBase& rhs)
- {
- m_set.assign_difference(rhs);
- update_unresolved(UpdateStatus::Updated);
- }
- inline void LnkSet::assign_symmetric_difference(const CollectionBase& rhs)
- {
- m_set.assign_symmetric_difference(rhs);
- update_unresolved(UpdateStatus::Updated);
- }
- } // namespace realm
- #endif // REALM_SET_HPP
|