| 1 | #pragma once | 
|---|
| 2 |  | 
|---|
| 3 | #include <Access/IAccessStorage.h> | 
|---|
| 4 | #include <list> | 
|---|
| 5 | #include <memory> | 
|---|
| 6 | #include <mutex> | 
|---|
| 7 | #include <unordered_map> | 
|---|
| 8 |  | 
|---|
| 9 |  | 
|---|
| 10 | namespace DB | 
|---|
| 11 | { | 
|---|
| 12 | /// Implementation of IAccessStorage which keeps all data in memory. | 
|---|
| 13 | class MemoryAccessStorage : public IAccessStorage | 
|---|
| 14 | { | 
|---|
| 15 | public: | 
|---|
| 16 | MemoryAccessStorage(const String & storage_name_ = "memory"); | 
|---|
| 17 | ~MemoryAccessStorage() override; | 
|---|
| 18 |  | 
|---|
| 19 | /// Sets all entities at once. | 
|---|
| 20 | void setAll(const std::vector<AccessEntityPtr> & all_entities); | 
|---|
| 21 | void setAll(const std::vector<std::pair<UUID, AccessEntityPtr>> & all_entities); | 
|---|
| 22 |  | 
|---|
| 23 | private: | 
|---|
| 24 | std::optional<UUID> findImpl(std::type_index type, const String & name) const override; | 
|---|
| 25 | std::vector<UUID> findAllImpl(std::type_index type) const override; | 
|---|
| 26 | bool existsImpl(const UUID & id) const override; | 
|---|
| 27 | AccessEntityPtr readImpl(const UUID & id) const override; | 
|---|
| 28 | String readNameImpl(const UUID & id) const override; | 
|---|
| 29 | UUID insertImpl(const AccessEntityPtr & entity, bool replace_if_exists) override; | 
|---|
| 30 | void removeImpl(const UUID & id) override; | 
|---|
| 31 | void updateImpl(const UUID & id, const UpdateFunc & update_func) override; | 
|---|
| 32 | SubscriptionPtr subscribeForChangesImpl(const UUID & id, const OnChangedHandler & handler) const override; | 
|---|
| 33 | SubscriptionPtr subscribeForChangesImpl(std::type_index type, const OnChangedHandler & handler) const override; | 
|---|
| 34 | bool hasSubscriptionImpl(const UUID & id) const override; | 
|---|
| 35 | bool hasSubscriptionImpl(std::type_index type) const override; | 
|---|
| 36 |  | 
|---|
| 37 | struct Entry | 
|---|
| 38 | { | 
|---|
| 39 | UUID id; | 
|---|
| 40 | AccessEntityPtr entity; | 
|---|
| 41 | mutable std::list<OnChangedHandler> handlers_by_id; | 
|---|
| 42 | }; | 
|---|
| 43 |  | 
|---|
| 44 | void insertNoLock(const UUID & id, const AccessEntityPtr & entity, bool replace_if_exists, Notifications & notifications); | 
|---|
| 45 | void removeNoLock(const UUID & id, Notifications & notifications); | 
|---|
| 46 | void updateNoLock(const UUID & id, const UpdateFunc & update_func, Notifications & notifications); | 
|---|
| 47 | void setAllNoLock(const std::vector<std::pair<UUID, AccessEntityPtr>> & all_entities, Notifications & notifications); | 
|---|
| 48 | void prepareNotifications(const Entry & entry, bool remove, Notifications & notifications) const; | 
|---|
| 49 |  | 
|---|
| 50 | using NameTypePair = std::pair<String, std::type_index>; | 
|---|
| 51 | struct Hash | 
|---|
| 52 | { | 
|---|
| 53 | size_t operator()(const NameTypePair & key) const | 
|---|
| 54 | { | 
|---|
| 55 | return std::hash<String>{}(key.first) - std::hash<std::type_index>{}(key.second); | 
|---|
| 56 | } | 
|---|
| 57 | }; | 
|---|
| 58 |  | 
|---|
| 59 | mutable std::mutex mutex; | 
|---|
| 60 | std::unordered_map<UUID, Entry> entries;               /// We want to search entries both by ID and by the pair of name and type. | 
|---|
| 61 | std::unordered_map<NameTypePair, Entry *, Hash> names; /// and by the pair of name and type. | 
|---|
| 62 | mutable std::unordered_multimap<std::type_index, OnChangedHandler> handlers_by_type; | 
|---|
| 63 | std::shared_ptr<const MemoryAccessStorage *> shared_ptr_to_this; /// We need weak pointers to `this` to implement subscriptions. | 
|---|
| 64 | }; | 
|---|
| 65 | } | 
|---|
| 66 |  | 
|---|