Sync.swift 29 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712
  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 Realm
  19. import Realm.Private
  20. #if !(os(iOS) && (arch(i386) || arch(arm)))
  21. import Combine
  22. #endif
  23. /**
  24. An object representing a MongoDB Realm user.
  25. - see: `RLMUser`
  26. */
  27. public typealias User = RLMUser
  28. public extension User {
  29. /// Links the currently authenticated user with a new identity, where the identity is defined by the credential
  30. /// specified as a parameter. This will only be successful if this `User` is the currently authenticated
  31. /// with the client from which it was created. On success a new user will be returned with the new linked credentials.
  32. /// @param credentials The `Credentials` used to link the user to a new identity.
  33. /// @completion A completion that eventually return `Result.success(User)` with user's data or `Result.failure(Error)`.
  34. func linkUser(credentials: Credentials, _ completion: @escaping (Result<User, Error>) -> Void) {
  35. self.__linkUser(with: ObjectiveCSupport.convert(object: credentials)) { user, error in
  36. if let user = user {
  37. completion(.success(user))
  38. } else {
  39. completion(.failure(error ?? Realm.Error.callFailed))
  40. }
  41. }
  42. }
  43. }
  44. /**
  45. A manager which configures and manages MongoDB Realm synchronization-related
  46. functionality.
  47. - see: `RLMSyncManager`
  48. */
  49. public typealias SyncManager = RLMSyncManager
  50. /**
  51. Options for configuring timeouts and intervals in the sync client.
  52. - see: `RLMSyncTimeoutOptions`
  53. */
  54. public typealias SyncTimeoutOptions = RLMSyncTimeoutOptions
  55. /**
  56. A session object which represents communication between the client and server for a specific
  57. Realm.
  58. - see: `RLMSyncSession`
  59. */
  60. public typealias SyncSession = RLMSyncSession
  61. /**
  62. A closure type for a closure which can be set on the `SyncManager` to allow errors to be reported
  63. to the application.
  64. - see: `RLMSyncErrorReportingBlock`
  65. */
  66. public typealias ErrorReportingBlock = RLMSyncErrorReportingBlock
  67. /**
  68. A closure type for a closure which is used by certain APIs to asynchronously return a `SyncUser`
  69. object to the application.
  70. - see: `RLMUserCompletionBlock`
  71. */
  72. public typealias UserCompletionBlock = RLMUserCompletionBlock
  73. /**
  74. An error associated with the SDK's synchronization functionality. All errors reported by
  75. an error handler registered on the `SyncManager` are of this type.
  76. - see: `RLMSyncError`
  77. */
  78. public typealias SyncError = RLMSyncError
  79. extension SyncError {
  80. /**
  81. An opaque token allowing the user to take action after certain types of
  82. errors have been reported.
  83. - see: `RLMSyncErrorActionToken`
  84. */
  85. public typealias ActionToken = RLMSyncErrorActionToken
  86. /**
  87. Given a client reset error, extract and return the recovery file path
  88. and the action token.
  89. The action token can be passed into `SyncSession.immediatelyHandleError(_:)`
  90. to immediately delete the local copy of the Realm which experienced the
  91. client reset error. The local copy of the Realm must be deleted before
  92. your application attempts to open the Realm again.
  93. The recovery file path is the path to which the current copy of the Realm
  94. on disk will be saved once the client reset occurs.
  95. - warning: Do not call `SyncSession.immediatelyHandleError(_:)` until you are
  96. sure that all references to the Realm and managed objects belonging
  97. to the Realm have been nil'ed out, and that all autorelease pools
  98. containing these references have been drained.
  99. - see: `SyncError.ActionToken`, `SyncSession.immediatelyHandleError(_:)`
  100. */
  101. public func clientResetInfo() -> (String, SyncError.ActionToken)? {
  102. if code == SyncError.clientResetError,
  103. let recoveryPath = userInfo[kRLMSyncPathOfRealmBackupCopyKey] as? String,
  104. let token = _nsError.__rlmSync_errorActionToken() {
  105. return (recoveryPath, token)
  106. }
  107. return nil
  108. }
  109. /**
  110. Given a permission denied error, extract and return the action token.
  111. This action token can be passed into `SyncSession.immediatelyHandleError(_:)`
  112. to immediately delete the local copy of the Realm which experienced the
  113. permission denied error. The local copy of the Realm must be deleted before
  114. your application attempts to open the Realm again.
  115. - warning: Do not call `SyncSession.immediatelyHandleError(_:)` until you are
  116. sure that all references to the Realm and managed objects belonging
  117. to the Realm have been nil'ed out, and that all autorelease pools
  118. containing these references have been drained.
  119. - see: `SyncError.ActionToken`, `SyncSession.immediatelyHandleError(_:)`
  120. */
  121. public func deleteRealmUserInfo() -> SyncError.ActionToken? {
  122. return _nsError.__rlmSync_errorActionToken()
  123. }
  124. }
  125. /**
  126. An error associated with network requests made to the authentication server. This type of error
  127. may be returned in the callback block to `SyncUser.logIn()` upon certain types of failed login
  128. attempts (for example, if the request is malformed or if the server is experiencing an issue).
  129. - see: `RLMSyncAuthError`
  130. */
  131. public typealias SyncAuthError = RLMSyncAuthError
  132. /**
  133. An enum which can be used to specify the level of logging.
  134. - see: `RLMSyncLogLevel`
  135. */
  136. public typealias SyncLogLevel = RLMSyncLogLevel
  137. /**
  138. A data type whose values represent different authentication providers that can be used with
  139. MongoDB Realm.
  140. - see: `RLMIdentityProvider`
  141. */
  142. public typealias Provider = RLMIdentityProvider
  143. /**
  144. * How the Realm client should validate the identity of the server for secure connections.
  145. *
  146. * By default, when connecting to MongoDB Realm over HTTPS, Realm will
  147. * validate the server's HTTPS certificate using the system trust store and root
  148. * certificates. For additional protection against man-in-the-middle (MITM)
  149. * attacks and similar vulnerabilities, you can pin a certificate or public key,
  150. * and reject all others, even if they are signed by a trusted CA.
  151. */
  152. @frozen public enum ServerValidationPolicy {
  153. /// Perform no validation and accept potentially invalid certificates.
  154. ///
  155. /// - warning: DO NOT USE THIS OPTION IN PRODUCTION.
  156. case none
  157. /// Use the default server trust evaluation based on the system-wide CA
  158. /// store. Any certificate signed by a trusted CA will be accepted.
  159. case system
  160. /// Use a specific pinned certificate to validate the server identify.
  161. ///
  162. /// This will only connect to a server if one of the server certificates
  163. /// matches the certificate stored at the given local path and that
  164. /// certificate has a valid trust chain.
  165. ///
  166. /// On macOS, the certificate files may be in any of the formats supported
  167. /// by SecItemImport(), including PEM and .cer (see SecExternalFormat for a
  168. /// complete list of possible formats). On iOS and other platforms, only
  169. /// DER .cer files are supported.
  170. case pinCertificate(path: URL)
  171. }
  172. /**
  173. A `SyncConfiguration` represents configuration parameters for Realms intended to sync with
  174. MongoDB Realm.
  175. */
  176. @frozen public struct SyncConfiguration {
  177. /// The `SyncUser` who owns the Realm that this configuration should open.
  178. public let user: User
  179. /**
  180. The value this Realm is partitioned on. The partition key is a property defined in
  181. MongoDB Realm. All classes with a property with this value will be synchronized to the
  182. Realm.
  183. */
  184. public let partitionValue: AnyBSON?
  185. /**
  186. A policy that determines what should happen when all references to Realms opened by this
  187. configuration go out of scope.
  188. */
  189. internal let stopPolicy: RLMSyncStopPolicy
  190. /**
  191. By default, Realm.asyncOpen() swallows non-fatal connection errors such as
  192. a connection attempt timing out and simply retries until it succeeds. If
  193. this is set to `true`, instead the error will be reported to the callback
  194. and the async open will be cancelled.
  195. */
  196. public let cancelAsyncOpenOnNonFatalErrors: Bool
  197. internal init(config: RLMSyncConfiguration) {
  198. self.user = config.user
  199. self.stopPolicy = config.stopPolicy
  200. self.partitionValue = ObjectiveCSupport.convert(object: config.partitionValue)
  201. self.cancelAsyncOpenOnNonFatalErrors = config.cancelAsyncOpenOnNonFatalErrors
  202. }
  203. func asConfig() -> RLMSyncConfiguration {
  204. let c = RLMSyncConfiguration(user: user,
  205. partitionValue: partitionValue.map(ObjectiveCSupport.convertBson),
  206. stopPolicy: stopPolicy)
  207. c.cancelAsyncOpenOnNonFatalErrors = cancelAsyncOpenOnNonFatalErrors
  208. return c
  209. }
  210. }
  211. /// Structure providing an interface to call a MongoDB Realm function with the provided name and arguments.
  212. ///
  213. /// user.functions.sum([1, 2, 3, 4, 5]) { sum, error in
  214. /// guard case let .int64(value) = sum else {
  215. /// print(error?.localizedDescription)
  216. /// }
  217. ///
  218. /// assert(value == 15)
  219. /// }
  220. ///
  221. /// The dynamic member name (`sum` in the above example) is directly associated with the function name.
  222. /// The first argument is the `BSONArray` of arguments to be provided to the function.
  223. /// The second and final argument is the completion handler to call when the function call is complete.
  224. /// This handler is executed on a non-main global `DispatchQueue`.
  225. @dynamicMemberLookup
  226. @frozen public struct Functions {
  227. private let user: User
  228. fileprivate init(user: User) {
  229. self.user = user
  230. }
  231. /// A closure type for receiving the completion of a remote function call.
  232. public typealias FunctionCompletionHandler = (AnyBSON?, Error?) -> Void
  233. /// A closure type for the dynamic remote function type.
  234. public typealias Function = ([AnyBSON], @escaping FunctionCompletionHandler) -> Void
  235. /// The implementation of @dynamicMemberLookup that allows for dynamic remote function calls.
  236. public subscript(dynamicMember string: String) -> Function {
  237. return { (arguments: [AnyBSON], completionHandler: @escaping FunctionCompletionHandler) in
  238. let objcArgs = arguments.map(ObjectiveCSupport.convertBson)
  239. self.user.__callFunctionNamed(string, arguments: objcArgs) { (bson: RLMBSON?, error: Error?) in
  240. completionHandler(bson.map(ObjectiveCSupport.convertBson) ?? .none, error)
  241. }
  242. }
  243. }
  244. /// A closure type for receiving the completion result of a remote function call.
  245. public typealias ResultFunctionCompletionHandler = (Result<AnyBSON, Error>) -> Void
  246. /// A closure type for the dynamic remote function type.
  247. public typealias ResultFunction = ([AnyBSON], @escaping ResultFunctionCompletionHandler) -> Void
  248. /// The implementation of @dynamicMemberLookup that allows for dynamic remote function calls.
  249. public subscript(dynamicMember string: String) -> ResultFunction {
  250. return { (arguments: [AnyBSON], completionHandler: @escaping ResultFunctionCompletionHandler) in
  251. let objcArgs = arguments.map(ObjectiveCSupport.convertBson)
  252. self.user.__callFunctionNamed(string, arguments: objcArgs) { (bson: RLMBSON?, error: Error?) in
  253. if let b = bson.map(ObjectiveCSupport.convertBson), let bson = b {
  254. completionHandler(.success(bson))
  255. } else {
  256. completionHandler(.failure(error ?? Realm.Error.callFailed))
  257. }
  258. }
  259. }
  260. }
  261. #if !(os(iOS) && (arch(i386) || arch(arm)))
  262. /// The implementation of @dynamicMemberLookup that allows for dynamic remote function calls.
  263. @available(OSX 10.15, watchOS 6.0, iOS 13.0, iOSApplicationExtension 13.0, OSXApplicationExtension 10.15, tvOS 13.0, macCatalyst 13.0, macCatalystApplicationExtension 13.0, *)
  264. public subscript(dynamicMember string: String) -> ([AnyBSON]) -> Future<AnyBSON, Error> {
  265. return { (arguments: [AnyBSON]) in
  266. return Future<AnyBSON, Error> { self[dynamicMember: string](arguments, $0) }
  267. }
  268. }
  269. #endif
  270. }
  271. public extension User {
  272. /**
  273. Create a sync configuration instance.
  274. Additional settings can be optionally specified. Descriptions of these
  275. settings follow.
  276. `enableSSLValidation` is true by default. It can be disabled for debugging
  277. purposes.
  278. - warning: NEVER disable SSL validation for a system running in production.
  279. */
  280. func configuration<T: BSON>(partitionValue: T) -> Realm.Configuration {
  281. let config = self.__configuration(withPartitionValue: ObjectiveCSupport.convert(object: AnyBSON(partitionValue)))
  282. return ObjectiveCSupport.convert(object: config)
  283. }
  284. /**
  285. Create a sync configuration instance.
  286. - parameter partitionValue: Takes `nil` as a partition value.
  287. - parameter cancelAsyncOpenOnNonFatalErrors: By default, Realm.asyncOpen()
  288. swallows non-fatal connection errors such as a connection attempt timing
  289. out and simply retries until it succeeds. If this is set to `true`, instead
  290. the error will be reported to the callback and the async open will be
  291. cancelled.
  292. - warning: NEVER disable SSL validation for a system running in production.
  293. */
  294. func configuration(partitionValue: AnyBSON,
  295. cancelAsyncOpenOnNonFatalErrors: Bool = false) -> Realm.Configuration {
  296. let config = self.__configuration(withPartitionValue: ObjectiveCSupport.convert(object: partitionValue))
  297. let syncConfig = config.syncConfiguration!
  298. syncConfig.cancelAsyncOpenOnNonFatalErrors = cancelAsyncOpenOnNonFatalErrors
  299. config.syncConfiguration = syncConfig
  300. return ObjectiveCSupport.convert(object: config)
  301. }
  302. /**
  303. Create a sync configuration instance.
  304. - parameter partitionValue: The `BSON` value the Realm is partitioned on.
  305. - parameter cancelAsyncOpenOnNonFatalErrors: By default, Realm.asyncOpen()
  306. swallows non-fatal connection errors such as a connection attempt timing
  307. out and simply retries until it succeeds. If this is set to `true`, instead
  308. the error will be reported to the callback and the async open will be
  309. cancelled.
  310. - warning: NEVER disable SSL validation for a system running in production.
  311. */
  312. func configuration<T: BSON>(partitionValue: T,
  313. cancelAsyncOpenOnNonFatalErrors: Bool = false) -> Realm.Configuration {
  314. let config = self.__configuration(withPartitionValue: ObjectiveCSupport.convert(object: AnyBSON(partitionValue)))
  315. let syncConfig = config.syncConfiguration!
  316. syncConfig.cancelAsyncOpenOnNonFatalErrors = cancelAsyncOpenOnNonFatalErrors
  317. config.syncConfiguration = syncConfig
  318. return ObjectiveCSupport.convert(object: config)
  319. }
  320. /**
  321. The custom data of the user.
  322. This is configured in your MongoDB Realm App.
  323. */
  324. var customData: Document {
  325. guard let rlmCustomData = self.__customData as RLMBSON?,
  326. let anyBSON = ObjectiveCSupport.convert(object: rlmCustomData),
  327. case let .document(customData) = anyBSON else {
  328. return [:]
  329. }
  330. return customData
  331. }
  332. /// A client for interacting with a remote MongoDB instance
  333. /// - Parameter serviceName: The name of the MongoDB service
  334. /// - Returns: A `MongoClient` which is used for interacting with a remote MongoDB service
  335. func mongoClient(_ serviceName: String) -> MongoClient {
  336. return self.__mongoClient(withServiceName: serviceName)
  337. }
  338. /// Call a MongoDB Realm function with the provided name and arguments.
  339. ///
  340. /// user.functions.sum([1, 2, 3, 4, 5]) { sum, error in
  341. /// guard case let .int64(value) = sum else {
  342. /// print(error?.localizedDescription)
  343. /// }
  344. ///
  345. /// assert(value == 15)
  346. /// }
  347. ///
  348. /// The dynamic member name (`sum` in the above example) is directly associated with the function name.
  349. /// The first argument is the `BSONArray` of arguments to be provided to the function.
  350. /// The second and final argument is the completion handler to call when the function call is complete.
  351. /// This handler is executed on a non-main global `DispatchQueue`.
  352. var functions: Functions {
  353. return Functions(user: self)
  354. }
  355. }
  356. public extension SyncSession {
  357. /**
  358. The current state of the session represented by a session object.
  359. - see: `RLMSyncSessionState`
  360. */
  361. typealias State = RLMSyncSessionState
  362. /**
  363. The current state of a sync session's connection.
  364. - see: `RLMSyncConnectionState`
  365. */
  366. typealias ConnectionState = RLMSyncConnectionState
  367. /**
  368. The transfer direction (upload or download) tracked by a given progress notification block.
  369. Progress notification blocks can be registered on sessions if your app wishes to be informed
  370. how many bytes have been uploaded or downloaded, for example to show progress indicator UIs.
  371. */
  372. enum ProgressDirection {
  373. /// For monitoring upload progress.
  374. case upload
  375. /// For monitoring download progress.
  376. case download
  377. }
  378. /**
  379. The desired behavior of a progress notification block.
  380. Progress notification blocks can be registered on sessions if your app wishes to be informed
  381. how many bytes have been uploaded or downloaded, for example to show progress indicator UIs.
  382. */
  383. enum ProgressMode {
  384. /**
  385. The block will be called forever, or until it is unregistered by calling
  386. `ProgressNotificationToken.invalidate()`.
  387. Notifications will always report the latest number of transferred bytes, and the
  388. most up-to-date number of total transferrable bytes.
  389. */
  390. case reportIndefinitely
  391. /**
  392. The block will, upon registration, store the total number of bytes
  393. to be transferred. When invoked, it will always report the most up-to-date number
  394. of transferrable bytes out of that original number of transferrable bytes.
  395. When the number of transferred bytes reaches or exceeds the
  396. number of transferrable bytes, the block will be unregistered.
  397. */
  398. case forCurrentlyOutstandingWork
  399. }
  400. /**
  401. A token corresponding to a progress notification block.
  402. Call `invalidate()` on the token to stop notifications. If the notification block has already
  403. been automatically stopped, calling `invalidate()` does nothing. `invalidate()` should be called
  404. before the token is destroyed.
  405. */
  406. typealias ProgressNotificationToken = RLMProgressNotificationToken
  407. /**
  408. A struct encapsulating progress information, as well as useful helper methods.
  409. */
  410. struct Progress {
  411. /// The number of bytes that have been transferred.
  412. public let transferredBytes: Int
  413. /**
  414. The total number of transferrable bytes (bytes that have been transferred,
  415. plus bytes pending transfer).
  416. If the notification block is tracking downloads, this number represents the size of the
  417. changesets generated by all other clients using the Realm.
  418. If the notification block is tracking uploads, this number represents the size of the
  419. changesets representing the local changes on this client.
  420. */
  421. public let transferrableBytes: Int
  422. /// The fraction of bytes transferred out of all transferrable bytes. If this value is 1,
  423. /// no bytes are waiting to be transferred (either all bytes have already been transferred,
  424. /// or there are no bytes to be transferred in the first place).
  425. public var fractionTransferred: Double {
  426. if transferrableBytes == 0 {
  427. return 1
  428. }
  429. let percentage = Double(transferredBytes) / Double(transferrableBytes)
  430. return percentage > 1 ? 1 : percentage
  431. }
  432. /// Whether all pending bytes have already been transferred.
  433. public var isTransferComplete: Bool {
  434. return transferredBytes >= transferrableBytes
  435. }
  436. internal init(transferred: UInt, transferrable: UInt) {
  437. transferredBytes = Int(transferred)
  438. transferrableBytes = Int(transferrable)
  439. }
  440. }
  441. /**
  442. Register a progress notification block.
  443. If the session has already received progress information from the
  444. synchronization subsystem, the block will be called immediately. Otherwise, it
  445. will be called as soon as progress information becomes available.
  446. Multiple blocks can be registered with the same session at once. Each block
  447. will be invoked on a side queue devoted to progress notifications.
  448. The token returned by this method must be retained as long as progress
  449. notifications are desired, and the `invalidate()` method should be called on it
  450. when notifications are no longer needed and before the token is destroyed.
  451. If no token is returned, the notification block will never be called again.
  452. There are a number of reasons this might be true. If the session has previously
  453. experienced a fatal error it will not accept progress notification blocks. If
  454. the block was configured in the `forCurrentlyOutstandingWork` mode but there
  455. is no additional progress to report (for example, the number of transferrable bytes
  456. and transferred bytes are equal), the block will not be called again.
  457. - parameter direction: The transfer direction (upload or download) to track in this progress notification block.
  458. - parameter mode: The desired behavior of this progress notification block.
  459. - parameter block: The block to invoke when notifications are available.
  460. - returns: A token which must be held for as long as you want notifications to be delivered.
  461. - see: `ProgressDirection`, `Progress`, `ProgressNotificationToken`
  462. */
  463. func addProgressNotification(for direction: ProgressDirection,
  464. mode: ProgressMode,
  465. block: @escaping (Progress) -> Void) -> ProgressNotificationToken? {
  466. return __addProgressNotification(for: (direction == .upload ? .upload : .download),
  467. mode: (mode == .reportIndefinitely
  468. ? .reportIndefinitely
  469. : .forCurrentlyOutstandingWork)) { transferred, transferrable in
  470. block(Progress(transferred: transferred, transferrable: transferrable))
  471. }
  472. }
  473. }
  474. extension Realm {
  475. /// :nodoc:
  476. @available(*, unavailable, message: "Use Results.subscribe()")
  477. public func subscribe<T: Object>(to objects: T.Type, where: String,
  478. completion: @escaping (Results<T>?, Swift.Error?) -> Void) {
  479. fatalError()
  480. }
  481. /**
  482. Get the SyncSession used by this Realm. Will be nil if this is not a
  483. synchronized Realm.
  484. */
  485. public var syncSession: SyncSession? {
  486. return SyncSession(for: rlmRealm)
  487. }
  488. }
  489. #if !(os(iOS) && (arch(i386) || arch(arm)))
  490. @available(OSX 10.15, watchOS 6.0, iOS 13.0, iOSApplicationExtension 13.0, OSXApplicationExtension 10.15, tvOS 13.0, macCatalyst 13.0, macCatalystApplicationExtension 13.0, *)
  491. public extension User {
  492. /// Refresh a user's custom data. This will, in effect, refresh the user's auth session.
  493. /// @returns A publisher that eventually return `Dictionary` with user's data or `Error`.
  494. func refreshCustomData() -> Future<[AnyHashable: Any], Error> {
  495. return Future { self.refreshCustomData($0) }
  496. }
  497. /// Links the currently authenticated user with a new identity, where the identity is defined by the credential
  498. /// specified as a parameter. This will only be successful if this `User` is the currently authenticated
  499. /// with the client from which it was created. On success a new user will be returned with the new linked credentials.
  500. /// @param credentials The `Credentials` used to link the user to a new identity.
  501. /// @returns A publisher that eventually return `Result.success` or `Error`.
  502. func linkUser(credentials: Credentials) -> Future<User, Error> {
  503. return Future { self.linkUser(credentials: credentials, $0) }
  504. }
  505. /// Removes the user
  506. /// This logs out and destroys the session related to this user. The completion block will return an error
  507. /// if the user is not found or is already removed.
  508. /// @returns A publisher that eventually return `Result.success` or `Error`.
  509. func remove() -> Future<Void, Error> {
  510. return Future<Void, Error> { promise in
  511. self.remove { error in
  512. if let error = error {
  513. promise(.failure(error))
  514. } else {
  515. promise(.success(()))
  516. }
  517. }
  518. }
  519. }
  520. /// Logs out the current user
  521. /// The users state will be set to `Removed` is they are an anonymous user or `LoggedOut` if they are authenticated by a username / password or third party auth clients
  522. //// If the logout request fails, this method will still clear local authentication state.
  523. /// @returns A publisher that eventually return `Result.success` or `Error`.
  524. func logOut() -> Future<Void, Error> {
  525. return Future<Void, Error> { promise in
  526. self.logOut { error in
  527. if let error = error {
  528. promise(.failure(error))
  529. } else {
  530. promise(.success(()))
  531. }
  532. }
  533. }
  534. }
  535. }
  536. /// :nodoc:
  537. @available(OSX 10.15, watchOS 6.0, iOS 13.0, iOSApplicationExtension 13.0, OSXApplicationExtension 10.15, tvOS 13.0, macCatalyst 13.0, macCatalystApplicationExtension 13.0, *)
  538. @frozen public struct UserSubscription: Subscription {
  539. private let user: User
  540. private let token: RLMUserSubscriptionToken
  541. internal init(user: User, token: RLMUserSubscriptionToken) {
  542. self.user = user
  543. self.token = token
  544. }
  545. /// A unique identifier for identifying publisher streams.
  546. public var combineIdentifier: CombineIdentifier {
  547. return CombineIdentifier(NSNumber(value: token.value))
  548. }
  549. /// This function is not implemented.
  550. ///
  551. /// Realm publishers do not support backpressure and so this function does nothing.
  552. public func request(_ demand: Subscribers.Demand) {
  553. }
  554. /// Stop emitting values on this subscription.
  555. public func cancel() {
  556. user.unsubscribe(token)
  557. }
  558. }
  559. /// :nodoc:
  560. @available(OSX 10.15, watchOS 6.0, iOS 13.0, iOSApplicationExtension 13.0, OSXApplicationExtension 10.15, tvOS 13.0, macCatalyst 13.0, macCatalystApplicationExtension 13.0, *)
  561. public class UserPublisher: Publisher {
  562. /// This publisher cannot fail.
  563. public typealias Failure = Never
  564. /// This publisher emits User.
  565. public typealias Output = User
  566. private let user: User
  567. internal init(_ user: User) {
  568. self.user = user
  569. }
  570. /// :nodoc:
  571. public func receive<S>(subscriber: S) where S: Subscriber, S.Failure == Never, Output == S.Input {
  572. let token = user.subscribe { _ in
  573. _ = subscriber.receive(self.user)
  574. }
  575. subscriber.receive(subscription: UserSubscription(user: user, token: token))
  576. }
  577. }
  578. @available(OSX 10.15, watchOS 6.0, iOS 13.0, iOSApplicationExtension 13.0, OSXApplicationExtension 10.15, tvOS 13.0, macCatalyst 13.0, macCatalystApplicationExtension 13.0, *)
  579. extension User: ObservableObject {
  580. /// A publisher that emits Void each time the user changes.
  581. ///
  582. /// Despite the name, this actually emits *after* the user has changed.
  583. public var objectWillChange: UserPublisher {
  584. return UserPublisher(self)
  585. }
  586. }
  587. #endif
  588. public extension User {
  589. /// Refresh a user's custom data. This will, in effect, refresh the user's auth session.
  590. /// @completion A completion that eventually return `Result.success(Dictionary)` with user's data or `Result.failure(Error)`.
  591. func refreshCustomData(_ completion: @escaping (Result<[AnyHashable: Any], Error>) -> Void) {
  592. self.refreshCustomData { customData, error in
  593. if let customData = customData {
  594. completion(.success(customData))
  595. } else {
  596. completion(.failure(error ?? Realm.Error.callFailed))
  597. }
  598. }
  599. }
  600. }