PersistedProperty.swift 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432
  1. ////////////////////////////////////////////////////////////////////////////
  2. //
  3. // Copyright 2021 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 Realm
  19. import Realm.Private
  20. /// @Persisted is used to declare properties on Object subclasses which should be
  21. /// managed by Realm.
  22. ///
  23. /// Example of usage:
  24. /// ```
  25. /// class MyModel: Object {
  26. /// // A basic property declaration. A property with no
  27. /// // default value supplied will default to `nil` for
  28. /// // Optional types, zero for numeric types, false for Bool,
  29. /// // an empty string/data, and a new random value for UUID
  30. /// // and ObjectID.
  31. /// @Persisted var basicIntProperty: Int
  32. ///
  33. /// // Custom default values can be specified with the
  34. /// // standard Swift syntax
  35. /// @Persisted var intWithCustomDefault: Int = 5
  36. ///
  37. /// // Properties can be indexed by passing `indexed: true`
  38. /// // to the initializer.
  39. /// @Persisted(indexed: true) var indexedString: String
  40. ///
  41. /// // Properties can set as the class's primary key by
  42. /// // passing `primaryKey: true` to the initializer
  43. /// @Persisted(primaryKey: true) var _id: ObjectId
  44. ///
  45. /// // List and set properties should always be declared
  46. /// // with `: List` rather than `= List()`
  47. /// @Persisted var listProperty: List<Int>
  48. /// @Persisted var setProperty: MutableSet<MyObject>
  49. ///
  50. /// // LinkingObjects properties require setting the source
  51. /// // object link property name in the initializer
  52. /// @Persisted(originProperty: "outgoingLink")
  53. /// var incomingLinks: LinkingObjects<OtherModel>
  54. ///
  55. /// // Properties which are not marked with @Persisted will
  56. /// // be ignored entirely by Realm.
  57. /// var ignoredProperty = true
  58. /// }
  59. /// ```
  60. ///
  61. /// Int, Bool, String, ObjectId and Date properties can be indexed by passing
  62. /// `indexed: true` to the initializer. Indexing a property improves the
  63. /// performance of equality queries on that property, at the cost of slightly
  64. /// worse write performance. No other operations currently use the index.
  65. ///
  66. /// A property can be set as the class's primary key by passing `primaryKey: true`
  67. /// to the initializer. Compound primary keys are not supported, and setting
  68. /// more than one property as the primary key will throw an exception at
  69. /// runtime. Only Int, String, UUID and ObjectID properties can be made the
  70. /// primary key, and when using MongoDB Realm, the primary key must be named
  71. /// `_id`. The primary key property can only be mutated on unmanaged objects,
  72. /// and mutating it on an object which has been added to a Realm will throw an
  73. /// exception.
  74. ///
  75. /// Properties can optionally be given a default value using the standard Swift
  76. /// syntax. If no default value is given, a value will be generated on first
  77. /// access: `nil` for all Optional types, zero for numeric types, false for
  78. /// Bool, an empty string/data, and a new random value for UUID and ObjectID.
  79. /// List and MutableSet properties *should not* be defined by setting them to a
  80. /// default value of an empty List/MutableSet. Doing so will work, but will
  81. /// result in worse performance when accessing objects managed by a Realm.
  82. /// Similarly, ObjectID properties *should not* be initialized to
  83. /// `ObjectID.generate()`, as doing so will result in extra ObjectIDs being
  84. /// generated and then discarded when reading from a Realm.
  85. ///
  86. /// If a class has at least one @Persisted property, all other properties will be
  87. /// ignored by Realm. This means that they will not be persisted and will not
  88. /// be usable in queries and other operations such as sorting and aggregates
  89. /// which require a managed property.
  90. ///
  91. /// @Persisted cannot be used anywhere other than as a property on an Object or
  92. /// EmbeddedObject subclass and trying to use it in other places will result in
  93. /// runtime errors.
  94. @propertyWrapper
  95. public struct Persisted<Value: _Persistable> {
  96. private var storage: PropertyStorage<Value>
  97. /// :nodoc:
  98. @available(*, unavailable, message: "@Persisted can only be used as a property on a Realm object")
  99. public var wrappedValue: Value {
  100. // The static subscript below is called instead of this when the property
  101. // wrapper is used on an ObjectBase subclass, which is the only thing we support.
  102. get { fatalError("called wrappedValue getter") }
  103. // swiftlint:disable:next unused_setter_value
  104. set { fatalError("called wrappedValue setter") }
  105. }
  106. /// Declares a property which is lazily initialized to the type's default value.
  107. public init() {
  108. storage = .unmanagedNoDefault(indexed: false, primary: false)
  109. }
  110. /// Declares a property which defaults to the given value.
  111. public init(wrappedValue value: Value) {
  112. storage = .unmanaged(value: value, indexed: false, primary: false)
  113. }
  114. /// :nodoc:
  115. public static subscript<EnclosingSelf: ObjectBase>(
  116. _enclosingInstance observed: EnclosingSelf,
  117. wrapped wrappedKeyPath: ReferenceWritableKeyPath<EnclosingSelf, Value>,
  118. storage storageKeyPath: ReferenceWritableKeyPath<EnclosingSelf, Self>
  119. ) -> Value {
  120. get {
  121. return observed[keyPath: storageKeyPath].get(observed)
  122. }
  123. set {
  124. observed[keyPath: storageKeyPath].set(observed, value: newValue)
  125. }
  126. }
  127. // Called via RLMInitializeSwiftAccessor() to initialize the wrapper on a
  128. // newly created managed accessor object.
  129. internal mutating func initialize(_ object: ObjectBase, key: PropertyKey) {
  130. storage = .managed(key: key)
  131. }
  132. // Collection types use this instead of the above because when promoting a
  133. // unmanaged object to a managed object we want to reuse the existing collection
  134. // object if it exists. Currently it always will exist because we read the
  135. // value of the property first, but there's a potential optimization to
  136. // skip initializing it on that read.
  137. internal mutating func initializeCollection(_ object: ObjectBase, key: PropertyKey) -> Value? {
  138. if case let .unmanaged(value, _, _) = storage {
  139. storage = .managedCached(value: value, key: key)
  140. return value
  141. }
  142. if case let .unmanagedObserved(value, _) = storage {
  143. storage = .managedCached(value: value, key: key)
  144. return value
  145. }
  146. storage = .managed(key: key)
  147. return nil
  148. }
  149. internal mutating func get(_ object: ObjectBase) -> Value {
  150. switch storage {
  151. case let .unmanaged(value, _, _):
  152. return value
  153. case .unmanagedNoDefault:
  154. let value = Value._rlmDefaultValue(false)
  155. storage = .unmanaged(value: value)
  156. return value
  157. case let .unmanagedObserved(value, key):
  158. if let lastAccessedNames = object.lastAccessedNames {
  159. var name: String = ""
  160. if Value._rlmType == .linkingObjects {
  161. name = RLMObjectBaseObjectSchema(object)!.computedProperties[Int(key)].name
  162. } else {
  163. name = RLMObjectBaseObjectSchema(object)!.properties[Int(key)].name
  164. }
  165. lastAccessedNames.add(name)
  166. return Value._rlmKeyPathRecorder(with: lastAccessedNames)
  167. }
  168. return value
  169. case let .managed(key):
  170. let v = Value._rlmGetProperty(object, key)
  171. if Value._rlmRequiresCaching {
  172. // Collection types are initialized once and stored on the
  173. // object rather than on every access. Non-collection types
  174. // cannot be cached without some mechanism for knowing when to
  175. // reread them which we don't currently have.
  176. storage = .managedCached(value: v, key: key)
  177. }
  178. return v
  179. case let .managedCached(value, _):
  180. return value
  181. }
  182. }
  183. internal mutating func set(_ object: ObjectBase, value: Value) {
  184. if value is MutableRealmCollection {
  185. (get(object) as! MutableRealmCollection).assign(value)
  186. return
  187. }
  188. switch storage {
  189. case let .unmanagedObserved(_, key):
  190. let name = RLMObjectBaseObjectSchema(object)!.properties[Int(key)].name
  191. object.willChangeValue(forKey: name)
  192. storage = .unmanagedObserved(value: value, key: key)
  193. object.didChangeValue(forKey: name)
  194. case .managed(let key), .managedCached(_, let key):
  195. Value._rlmSetProperty(object, key, value)
  196. case .unmanaged, .unmanagedNoDefault:
  197. storage = .unmanaged(value: value, indexed: false, primary: false)
  198. }
  199. }
  200. // Initialize an unmanaged property for observation
  201. internal mutating func observe(_ object: ObjectBase, property: RLMProperty) {
  202. let value: Value
  203. switch storage {
  204. case let .unmanaged(v, _, _):
  205. value = v
  206. case .unmanagedNoDefault:
  207. value = Value._rlmDefaultValue(false)
  208. case .unmanagedObserved, .managed, .managedCached:
  209. return
  210. }
  211. // Mutating a collection triggers a KVO notification on the parent, so
  212. // we need to ensure that the collection has a pointer to its parent.
  213. if let value = value as? MutableRealmCollection {
  214. value.setParent(object, property)
  215. }
  216. storage = .unmanagedObserved(value: value, key: PropertyKey(property.index))
  217. }
  218. }
  219. extension Persisted: Decodable where Value: Decodable {
  220. public init(from decoder: Decoder) throws {
  221. storage = .unmanaged(value: try decoder.decodeOptional(Value.self), indexed: false, primary: false)
  222. }
  223. }
  224. extension Persisted: Encodable where Value: Encodable {
  225. public func encode(to encoder: Encoder) throws {
  226. switch storage {
  227. case .unmanaged(let value, _, _):
  228. try value.encode(to: encoder)
  229. case .unmanagedObserved(let value, _):
  230. try value.encode(to: encoder)
  231. case .unmanagedNoDefault:
  232. try Value._rlmDefaultValue(false).encode(to: encoder)
  233. default:
  234. // We need a reference to the parent object to be able to read from
  235. // a managed property. There's probably a way to do this with some
  236. // sort of custom adapter that keeps track of the current parent
  237. // at each level of recursion, but it's not trivial.
  238. throw EncodingError.invalidValue(self, .init(codingPath: encoder.codingPath, debugDescription: "Only unmanaged Realm objects can be encoded using automatic Codable synthesis. You must explicitly define encode(to:) on your model class to support managed Realm objects."))
  239. }
  240. }
  241. }
  242. /// :nodoc:
  243. /// Protocol for a PropertyWrapper to properly handle Coding when the wrappedValue is Optional
  244. public protocol OptionalCodingWrapper {
  245. associatedtype WrappedType: ExpressibleByNilLiteral
  246. init(wrappedValue: WrappedType)
  247. }
  248. /// :nodoc:
  249. extension KeyedDecodingContainer {
  250. // This is used to override the default decoding behaviour for OptionalCodingWrapper to allow a value to avoid a missing key Error
  251. public func decode<T>(_ type: T.Type, forKey key: KeyedDecodingContainer<K>.Key) throws -> T where T: Decodable, T: OptionalCodingWrapper {
  252. return try decodeIfPresent(T.self, forKey: key) ?? T(wrappedValue: nil)
  253. }
  254. }
  255. extension Persisted: OptionalCodingWrapper where Value: ExpressibleByNilLiteral {
  256. }
  257. /**
  258. An enum type which can be used with @Persisted.
  259. Persisting an enum in Realm requires that it have a raw value and that the raw value by a type which Realm can store.
  260. The enum also has to be explicitly marked as conforming to this protocol as Swift does not let us do so implicitly.
  261. ```
  262. enum IntEnum: Int, PersistableEnum {
  263. case first = 1
  264. case second = 2
  265. case third = 7
  266. }
  267. enum StringEnum: String, PersistableEnum {
  268. case first = "a"
  269. case second = "b"
  270. case third = "g"
  271. }
  272. ```
  273. If the Realm contains a value which is not a valid member of the enum (such as
  274. if it was written by a different sync client which disagrees on which values
  275. are valid), optional enum properties will return `nil`, and non-optional
  276. properties will abort the process.
  277. */
  278. public protocol PersistableEnum: _OptionalPersistable, RawRepresentable, CaseIterable, RealmEnum {
  279. }
  280. extension PersistableEnum {
  281. /// :nodoc:
  282. public init() { self = Self.allCases.first! }
  283. }
  284. /// A type which can be indexed.
  285. ///
  286. /// This protocol is merely a tag and declaring additional types as conforming
  287. /// to it will simply result in runtime errors rather than compile-time errors.
  288. public protocol _Indexable {}
  289. extension Persisted where Value._RealmValue: _Indexable {
  290. /// Declares an indexed property which is lazily initialized to the type's default value.
  291. public init(indexed: Bool) {
  292. storage = .unmanagedNoDefault(indexed: indexed)
  293. }
  294. /// Declares an indexed property which defaults to the given value.
  295. public init(wrappedValue value: Value, indexed: Bool) {
  296. storage = .unmanaged(value: value, indexed: indexed)
  297. }
  298. }
  299. /// A type which can be made the primary key of an object.
  300. ///
  301. /// This protocol is merely a tag and declaring additional types as conforming
  302. /// to it will simply result in runtime errors rather than compile-time errors.
  303. public protocol _PrimaryKey {}
  304. extension Persisted where Value._RealmValue: _PrimaryKey {
  305. /// Declares the primary key property which is lazily initialized to the type's default value.
  306. public init(primaryKey: Bool) {
  307. storage = .unmanagedNoDefault(primary: primaryKey)
  308. }
  309. /// Declares the primary key property which defaults to the given value.
  310. public init(wrappedValue value: Value, primaryKey: Bool) {
  311. storage = .unmanaged(value: value, primary: primaryKey)
  312. }
  313. }
  314. /// :nodoc:
  315. // Constraining the LinkingObjects initializer to only LinkingObjects require
  316. // doing so via a protocol which only that type conforms to.
  317. public protocol LinkingObjectsProtocol {
  318. init(fromType: Element.Type, property: String)
  319. associatedtype Element
  320. }
  321. extension Persisted where Value: LinkingObjectsProtocol {
  322. /// Declares a LinkingObjects property with the given origin property name.
  323. ///
  324. /// - param originProperty: The name of the property on the linking object type which links to this object.
  325. public init(originProperty: String) {
  326. self.init(wrappedValue: Value(fromType: Value.Element.self, property: originProperty))
  327. }
  328. }
  329. extension LinkingObjects: LinkingObjectsProtocol {}
  330. // MARK: - Implementation
  331. /// :nodoc:
  332. extension Persisted: DiscoverablePersistedProperty where Value: _Persistable {
  333. public static var _rlmType: PropertyType { Value._rlmType }
  334. public static var _rlmOptional: Bool { Value._rlmOptional }
  335. public static var _rlmRequireObjc: Bool { false }
  336. public static func _rlmPopulateProperty(_ prop: RLMProperty) {
  337. // The label reported by Mirror has an underscore prefix added to it
  338. // as it's the actual storage rather than the compiler-magic getter/setter
  339. prop.name = String(prop.name.dropFirst())
  340. Value._rlmPopulateProperty(prop)
  341. Value._rlmSetAccessor(prop)
  342. }
  343. public func _rlmPopulateProperty(_ prop: RLMProperty) {
  344. switch storage {
  345. case let .unmanaged(value, indexed, primary):
  346. value._rlmPopulateProperty(prop)
  347. prop.indexed = indexed || primary
  348. prop.isPrimary = primary
  349. case let .unmanagedNoDefault(indexed, primary):
  350. prop.indexed = indexed || primary
  351. prop.isPrimary = primary
  352. default:
  353. fatalError()
  354. }
  355. }
  356. }
  357. // The actual storage for modern properties on objects.
  358. //
  359. // A newly created @Persisted will be either .unmanaged or .unmanagedNoDefault
  360. // depending on whether the user supplied a default value with `= value` when
  361. // defining the property. .unmanagedNoDefault turns into .unmanaged the first
  362. // time the property is read from, using a default value generated for the type.
  363. // If an unmanaged object is observed, that specific property is switched to
  364. // .unmanagedObserved so that the property can look up its name in the setter.
  365. //
  366. // When a new managed accessor is created, all properties are set to .managed.
  367. // When an existing unmanaged object is added to a Realm, existing non-collection
  368. // properties are set to .unmanaged, and collections are set to .managedCached,
  369. // reusing the existing instance of the collection (which are themselves promoted
  370. // to managed).
  371. //
  372. // The indexed and primary members of the unmanaged cases are used only for
  373. // schema discovery and are not always preserved once the Persisted is actually
  374. // used for anything.
  375. private enum PropertyStorage<T> {
  376. // An unmanaged value. This is used as the initial state if the user did
  377. // supply a default value, or if an unmanaged property is read or written
  378. // (but not observed).
  379. case unmanaged(value: T, indexed: Bool = false, primary: Bool = false)
  380. // The property is unmanaged and does not yet have a value. This state is
  381. // used if the user does not supply a default value in their model definition
  382. // and will be converted to the zero/empty value for the type when this
  383. // property is first used.
  384. case unmanagedNoDefault(indexed: Bool = false, primary: Bool = false)
  385. // The property is unmanaged and the parent object has (or previously had)
  386. // KVO observers, so we performed the additional initialization to set the
  387. // property key on each property. We do not track indexed/primary in this
  388. // state because those are needed only for schema discovery. An unmanaged
  389. // property never transitions from this state back to .unmanaged.
  390. case unmanagedObserved(value: T, key: PropertyKey)
  391. // The property is managed and so only needs to store the key to get/set
  392. // the value on the parent object.
  393. case managed(key: PropertyKey)
  394. // The property is managed and is storing a value which will be returned each
  395. // time. This is used only for collection properties, which are themselves
  396. // live objects and so only need to be created once. Caching them is both a
  397. // performance optimization (creating them involves a few memory allocations)
  398. // and is required for KVO to work correctly.
  399. case managedCached(value: T, key: PropertyKey)
  400. }