RLMClassInfo.hpp 5.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148
  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. #import <Foundation/Foundation.h>
  19. #import <realm/table_ref.hpp>
  20. #import <realm/util/optional.hpp>
  21. #import <unordered_map>
  22. #import <vector>
  23. namespace realm {
  24. class ObjectSchema;
  25. class Schema;
  26. struct Property;
  27. struct ColKey;
  28. struct TableKey;
  29. }
  30. class RLMObservationInfo;
  31. @class RLMRealm, RLMSchema, RLMObjectSchema, RLMProperty;
  32. NS_ASSUME_NONNULL_BEGIN
  33. namespace std {
  34. // Add specializations so that NSString can be used as the key for hash containers
  35. template<> struct hash<NSString *> {
  36. size_t operator()(__unsafe_unretained NSString *const str) const {
  37. return [str hash];
  38. }
  39. };
  40. template<> struct equal_to<NSString *> {
  41. bool operator()(__unsafe_unretained NSString * lhs, __unsafe_unretained NSString *rhs) const {
  42. return [lhs isEqualToString:rhs];
  43. }
  44. };
  45. }
  46. // The per-RLMRealm object schema information which stores the cached table
  47. // reference, handles table column lookups, and tracks observed objects
  48. class RLMClassInfo {
  49. public:
  50. RLMClassInfo(RLMRealm *, RLMObjectSchema *, const realm::ObjectSchema *);
  51. RLMClassInfo(RLMRealm *realm, RLMObjectSchema *rlmObjectSchema,
  52. std::unique_ptr<realm::ObjectSchema> objectSchema);
  53. __unsafe_unretained RLMRealm *const realm;
  54. __unsafe_unretained RLMObjectSchema *const rlmObjectSchema;
  55. const realm::ObjectSchema *const objectSchema;
  56. // Storage for the functionality in RLMObservation for handling indirect
  57. // changes to KVO-observed things
  58. std::vector<RLMObservationInfo *> observedObjects;
  59. // Get the table for this object type. Will return nullptr only if it's a
  60. // read-only Realm that is missing the table entirely.
  61. realm::TableRef table() const;
  62. // Get the RLMProperty for a given table column, or `nil` if it is a column
  63. // not used by the current schema
  64. RLMProperty *_Nullable propertyForTableColumn(realm::ColKey) const noexcept;
  65. // Get the RLMProperty that's used as the primary key, or `nil` if there is
  66. // no primary key for the current schema
  67. RLMProperty *_Nullable propertyForPrimaryKey() const noexcept;
  68. // Get the table column for the given property. The property must be a valid
  69. // persisted property.
  70. realm::ColKey tableColumn(NSString *propertyName) const;
  71. realm::ColKey tableColumn(RLMProperty *property) const;
  72. // Get the table column key for the given computed property. The property
  73. // must be a valid computed property.
  74. // Subscripting a `realm::ObjectSchema->computed_properties[property.index]`
  75. // does not return a valid colKey, unlike subscripting persisted_properties.
  76. // This method retrieves a valid column key for computed properties by
  77. // getting the opposite table column of the origin's "forward" link.
  78. realm::ColKey computedTableColumn(RLMProperty *property) const;
  79. // Get the info for the target of the link at the given property index.
  80. RLMClassInfo &linkTargetType(size_t propertyIndex);
  81. // Get the info for the target of the given property
  82. RLMClassInfo &linkTargetType(realm::Property const& property);
  83. // Get the corresponding ClassInfo for the given Realm
  84. RLMClassInfo &resolve(RLMRealm *);
  85. // Return true if the RLMObjectSchema is for a Swift class
  86. bool isSwiftClass() const noexcept;
  87. // Returns true if this was a dynamically added type
  88. bool isDynamic() const noexcept;
  89. private:
  90. // If the ObjectSchema is not owned by the realm instance
  91. // we need to manually manage the ownership of the object.
  92. std::unique_ptr<realm::ObjectSchema> dynamicObjectSchema;
  93. [[maybe_unused]] RLMObjectSchema *_Nullable dynamicRLMObjectSchema;
  94. };
  95. // A per-RLMRealm object schema map which stores RLMClassInfo keyed on the name
  96. class RLMSchemaInfo {
  97. using impl = std::unordered_map<NSString *, RLMClassInfo>;
  98. public:
  99. RLMSchemaInfo() = default;
  100. RLMSchemaInfo(RLMRealm *realm);
  101. RLMSchemaInfo clone(realm::Schema const& source_schema, RLMRealm *target_realm);
  102. // Look up by name, throwing if it's not present
  103. RLMClassInfo& operator[](NSString *name);
  104. // Look up by table key, return none if its not present.
  105. RLMClassInfo* operator[](realm::TableKey const& tableKey);
  106. // Emplaces a locally derived object schema into RLMSchemaInfo. This is used
  107. // when creating objects dynamically that are not registered in the Cocoa schema.
  108. // Note: `RLMClassInfo` assumes ownership of `schema`.
  109. void appendDynamicObjectSchema(std::unique_ptr<realm::ObjectSchema> schema,
  110. RLMObjectSchema *objectSchema,
  111. RLMRealm *const target_realm);
  112. impl::iterator begin() noexcept;
  113. impl::iterator end() noexcept;
  114. impl::const_iterator begin() const noexcept;
  115. impl::const_iterator end() const noexcept;
  116. private:
  117. std::unordered_map<NSString *, RLMClassInfo> m_objects;
  118. };
  119. NS_ASSUME_NONNULL_END