12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229 |
- /*************************************************************************
- *
- * 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
- #include <set>
- 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;
- };
- template <class T>
- class Set final : public CollectionBaseImpl<SetBase> {
- public:
- using Base = CollectionBaseImpl<SetBase>;
- 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();
- if (ndx >= current_size) {
- throw std::out_of_range("Index out of range");
- }
- 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;
- }
- private:
- friend class LnkSet;
- mutable std::unique_ptr<BPlusTree<T>> m_tree;
- using Base::m_col_key;
- using Base::m_obj;
- using Base::m_valid;
- void create()
- {
- m_tree->create();
- m_valid = true;
- }
- REALM_NOINLINE bool init_from_parent() const final
- {
- m_valid = m_tree->init_from_parent();
- update_content_version();
- return m_valid;
- }
- REALM_NOINLINE void ensure_created()
- {
- if (!m_valid && m_obj.is_valid()) {
- create();
- }
- }
- 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);
- };
- 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)
- {
- update_unresolved();
- }
- 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:
- bool is_obj_valid(size_t) const noexcept final;
- 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;
- bool do_update_if_needed() const final
- {
- return m_set.update_if_needed();
- }
- bool do_init_from_parent() const final
- {
- return m_set.init_from_parent();
- }
- 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();
- /// 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_utf8 function implements these rules. If that
- // function is changed we should either implement the rules here or
- // upgrade all Set<Mixed> columns.
- 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
- return a.compare(b) == 0;
- }
- };
- template <class T>
- inline Set<T>::Set(const Obj& obj, ColKey col_key)
- : Base(obj, col_key)
- , m_tree(new BPlusTree<value_type>(obj.get_alloc()))
- {
- if (!col_key.is_set()) {
- throw LogicError(LogicError::collection_type_mismatch);
- }
- check_column_type<value_type>(m_col_key);
- m_tree->set_parent(this, 0); // ndx not used, implicit in m_owner
- if (m_obj) {
- // Fine because init_from_parent() is final.
- this->init_from_parent();
- }
- }
- template <class T>
- inline Set<T>::Set(const Set& other)
- : Base(static_cast<const Base&>(other))
- {
- // FIXME: If the other side needed an update, we could be using a stale ref
- // below.
- REALM_ASSERT(!other.update_if_needed());
- if (other.m_tree) {
- Allocator& alloc = other.m_tree->get_alloc();
- m_tree = std::make_unique<BPlusTree<T>>(alloc);
- m_tree->set_parent(this, 0);
- if (m_valid)
- m_tree->init_from_ref(other.m_tree->get_ref());
- }
- }
- 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) {
- m_tree.reset();
- if (other.m_tree) {
- Allocator& alloc = other.m_tree->get_alloc();
- m_tree = std::make_unique<BPlusTree<T>>(alloc);
- m_tree->set_parent(this, 0);
- if (m_valid) {
- m_tree->init_from_ref(other.m_tree->get_ref());
- }
- }
- }
- 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);
- }
- }
- 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 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();
- 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 LogicError(LogicError::column_not_nullable);
- ensure_created();
- this->ensure_writeable();
- 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)
- {
- update_if_needed();
- this->ensure_writeable();
- auto it = find_impl(value);
- 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
- {
- if (!is_attached())
- return 0;
- update_if_needed();
- if (!m_valid) {
- return 0;
- }
- return m_tree->size();
- }
- 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()
- {
- ensure_created();
- update_if_needed();
- this->ensure_writeable();
- 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
- {
- update_if_needed();
- return MinHelper<T>::eval(*m_tree, return_ndx);
- }
- template <class T>
- inline util::Optional<Mixed> Set<T>::max(size_t* return_ndx) const
- {
- update_if_needed();
- return MaxHelper<T>::eval(*m_tree, return_ndx);
- }
- template <class T>
- inline util::Optional<Mixed> Set<T>::sum(size_t* return_cnt) const
- {
- update_if_needed();
- return SumHelper<T>::eval(*m_tree, return_cnt);
- }
- template <class T>
- inline util::Optional<Mixed> Set<T>::avg(size_t* return_cnt) const
- {
- update_if_needed();
- return AverageHelper<T>::eval(*m_tree, 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 <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();
- }
- namespace {
- template <class T>
- auto convert_to_set(const CollectionBase& rhs, bool nullable)
- {
- std::set<T, SetElementLessThan<T>> ret;
- for (size_t i = 0; i < rhs.size(); i++) {
- auto val = rhs.get_any(i);
- if constexpr (std::is_same_v<T, Mixed>) {
- ret.emplace(val);
- }
- else {
- if (val.is_type(ColumnTypeTraits<T>::id)) {
- ret.emplace(val.get<T>());
- }
- else if (val.is_null() && nullable) {
- ret.emplace(BPlusTree<T>::default_value(true));
- }
- }
- }
- return ret;
- }
- } // namespace
- 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<T>(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<T>(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<T>(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<T>(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<T>(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<T>(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<T>(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<T>(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<T>(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<T>(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
- {
- update_if_needed();
- return m_set.get(virtual2real(ndx));
- }
- inline size_t LnkSet::find(ObjKey value) const
- {
- update_if_needed();
- if (value.is_unresolved()) {
- return not_found;
- }
- 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);
- 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) {
- 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();
- return m_set.get_any(virtual2real(ndx));
- }
- inline std::pair<size_t, bool> LnkSet::insert_null()
- {
- update_if_needed();
- auto [ndx, inserted] = m_set.insert_null();
- 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) {
- 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);
- 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) {
- ndx = real2virtual(ndx);
- }
- return {ndx, erased};
- }
- inline void LnkSet::clear()
- {
- m_set.clear();
- clear_unresolved();
- }
- inline util::Optional<Mixed> LnkSet::min(size_t* return_ndx) const
- {
- 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
- {
- 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;
- if (value.get_type() != type_Link)
- return not_found;
- size_t found = find(value.get<ObjKey>());
- if (found != not_found) {
- found = real2virtual(found);
- }
- return found;
- }
- inline bool LnkSet::is_obj_valid(size_t) const noexcept
- {
- // LnkSet cannot contain NULL links.
- return true;
- }
- 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();
- }
- inline void LnkSet::assign_intersection(const CollectionBase& rhs)
- {
- m_set.assign_intersection(rhs);
- update_unresolved();
- }
- inline void LnkSet::assign_difference(const CollectionBase& rhs)
- {
- m_set.assign_difference(rhs);
- update_unresolved();
- }
- inline void LnkSet::assign_symmetric_difference(const CollectionBase& rhs)
- {
- m_set.assign_symmetric_difference(rhs);
- update_unresolved();
- }
- } // namespace realm
- #endif // REALM_SET_HPP
|