Files
swift-composable-architectu…/Sources/ComposableArchitecture/Effect.swift
Stephen Celis dc9dd315ee Cleanup (#1107)
2022-05-24 15:15:57 -04:00

409 lines
15 KiB
Swift

import Combine
import Foundation
/// The ``Effect`` type encapsulates a unit of work that can be run in the outside world, and can
/// feed data back to the ``Store``. It is the perfect place to do side effects, such as network
/// requests, saving/loading from disk, creating timers, interacting with dependencies, and more.
///
/// Effects are returned from reducers so that the ``Store`` can perform the effects after the
/// reducer is done running. It is important to note that ``Store`` is not thread safe, and so all
/// effects must receive values on the same thread, **and** if the store is being used to drive UI
/// then it must receive values on the main thread.
///
/// An effect simply wraps a `Publisher` value and provides some convenience initializers for
/// constructing some common types of effects.
public struct Effect<Output, Failure: Error>: Publisher {
public let upstream: AnyPublisher<Output, Failure>
/// Initializes an effect that wraps a publisher. Each emission of the wrapped publisher will be
/// emitted by the effect.
///
/// This initializer is useful for turning any publisher into an effect. For example:
///
/// ```swift
/// Effect(
/// NotificationCenter.default
/// .publisher(for: UIApplication.userDidTakeScreenshotNotification)
/// )
/// ```
///
/// Alternatively, you can use the `.eraseToEffect()` method that is defined on the `Publisher`
/// protocol:
///
/// ```swift
/// NotificationCenter.default
/// .publisher(for: UIApplication.userDidTakeScreenshotNotification)
/// .eraseToEffect()
/// ```
///
/// - Parameter publisher: A publisher.
public init<P: Publisher>(_ publisher: P) where P.Output == Output, P.Failure == Failure {
self.upstream = publisher.eraseToAnyPublisher()
}
public func receive<S>(
subscriber: S
) where S: Combine.Subscriber, Failure == S.Failure, Output == S.Input {
self.upstream.subscribe(subscriber)
}
/// Initializes an effect that immediately emits the value passed in.
///
/// - Parameter value: The value that is immediately emitted by the effect.
public init(value: Output) {
self.init(Just(value).setFailureType(to: Failure.self))
}
/// Initializes an effect that immediately fails with the error passed in.
///
/// - Parameter error: The error that is immediately emitted by the effect.
public init(error: Failure) {
// NB: Ideally we'd return a `Fail` publisher here, but due to a bug in iOS 13 that publisher
// can crash when used with certain combinations of operators such as `.retry.catch`. The
// bug was fixed in iOS 14, but to remain compatible with iOS 13 and higher we need to do
// a little trickery to fail in a slightly different way.
self.init(
Deferred {
Future { $0(.failure(error)) }
}
)
}
/// An effect that does nothing and completes immediately. Useful for situations where you must
/// return an effect, but you don't need to do anything.
public static var none: Effect {
Empty(completeImmediately: true).eraseToEffect()
}
/// Creates an effect that can supply a single value asynchronously in the future.
///
/// This can be helpful for converting APIs that are callback-based into ones that deal with
/// ``Effect``s.
///
/// For example, to create an effect that delivers an integer after waiting a second:
///
/// ```swift
/// Effect<Int, Never>.future { callback in
/// DispatchQueue.main.asyncAfter(deadline: .now() + 1) {
/// callback(.success(42))
/// }
/// }
/// ```
///
/// Note that you can only deliver a single value to the `callback`. If you send more they will be
/// discarded:
///
/// ```swift
/// Effect<Int, Never>.future { callback in
/// DispatchQueue.main.asyncAfter(deadline: .now() + 1) {
/// callback(.success(42))
/// callback(.success(1729)) // Will not be emitted by the effect
/// }
/// }
/// ```
///
/// If you need to deliver more than one value to the effect, you should use the ``Effect``
/// initializer that accepts a ``Subscriber`` value.
///
/// - Parameter attemptToFulfill: A closure that takes a `callback` as an argument which can be
/// used to feed it `Result<Output, Failure>` values.
public static func future(
_ attemptToFulfill: @escaping (@escaping (Result<Output, Failure>) -> Void) -> Void
) -> Effect {
Deferred { Future(attemptToFulfill) }.eraseToEffect()
}
/// Initializes an effect that lazily executes some work in the real world and synchronously sends
/// that data back into the store.
///
/// For example, to load a user from some JSON on the disk, one can wrap that work in an effect:
///
/// ```swift
/// Effect<User, Error>.result {
/// let fileUrl = URL(
/// fileURLWithPath: NSSearchPathForDirectoriesInDomains(
/// .documentDirectory, .userDomainMask, true
/// )[0]
/// )
/// .appendingPathComponent("user.json")
///
/// let result = Result<User, Error> {
/// let data = try Data(contentsOf: fileUrl)
/// return try JSONDecoder().decode(User.self, from: $0)
/// }
///
/// return result
/// }
/// ```
///
/// - Parameter attemptToFulfill: A closure encapsulating some work to execute in the real world.
/// - Returns: An effect.
public static func result(_ attemptToFulfill: @escaping () -> Result<Output, Failure>) -> Self {
Deferred { Future { $0(attemptToFulfill()) } }.eraseToEffect()
}
/// Initializes an effect from a callback that can send as many values as it wants, and can send
/// a completion.
///
/// This initializer is useful for bridging callback APIs, delegate APIs, and manager APIs to the
/// ``Effect`` type. One can wrap those APIs in an Effect so that its events are sent through the
/// effect, which allows the reducer to handle them.
///
/// For example, one can create an effect to ask for access to `MPMediaLibrary`. It can start by
/// sending the current status immediately, and then if the current status is `notDetermined` it
/// can request authorization, and once a status is received it can send that back to the effect:
///
/// ```swift
/// Effect.run { subscriber in
/// subscriber.send(MPMediaLibrary.authorizationStatus())
///
/// guard MPMediaLibrary.authorizationStatus() == .notDetermined else {
/// subscriber.send(completion: .finished)
/// return AnyCancellable {}
/// }
///
/// MPMediaLibrary.requestAuthorization { status in
/// subscriber.send(status)
/// subscriber.send(completion: .finished)
/// }
/// return AnyCancellable {
/// // Typically clean up resources that were created here, but this effect doesn't
/// // have any.
/// }
/// }
/// ```
///
/// - Parameter work: A closure that accepts a ``Subscriber`` value and returns a cancellable.
/// When the ``Effect`` is completed, the cancellable will be used to clean up any resources
/// created when the effect was started.
public static func run(
_ work: @escaping (Effect.Subscriber) -> Cancellable
) -> Self {
AnyPublisher.create(work).eraseToEffect()
}
/// Concatenates a variadic list of effects together into a single effect, which runs the effects
/// one after the other.
///
/// - Warning: Combine's `Publishers.Concatenate` operator, which this function uses, can leak
/// when its suffix is a `Publishers.MergeMany` operator, which is used throughout the
/// Composable Architecture in functions like ``Reducer/combine(_:)-1ern2``.
///
/// Feedback filed: <https://gist.github.com/mbrandonw/611c8352e1bd1c22461bd505e320ab58>
///
/// - Parameter effects: A variadic list of effects.
/// - Returns: A new effect
public static func concatenate(_ effects: Effect...) -> Effect {
.concatenate(effects)
}
/// Concatenates a collection of effects together into a single effect, which runs the effects one
/// after the other.
///
/// - Warning: Combine's `Publishers.Concatenate` operator, which this function uses, can leak
/// when its suffix is a `Publishers.MergeMany` operator, which is used throughout the
/// Composable Architecture in functions like ``Reducer/combine(_:)-1ern2``.
///
/// Feedback filed: <https://gist.github.com/mbrandonw/611c8352e1bd1c22461bd505e320ab58>
///
/// - Parameter effects: A collection of effects.
/// - Returns: A new effect
public static func concatenate<C: Collection>(
_ effects: C
) -> Effect where C.Element == Effect {
guard let first = effects.first else { return .none }
return
effects
.dropFirst()
.reduce(into: first) { effects, effect in
effects = effects.append(effect).eraseToEffect()
}
}
/// Merges a variadic list of effects together into a single effect, which runs the effects at the
/// same time.
///
/// - Parameter effects: A list of effects.
/// - Returns: A new effect
public static func merge(
_ effects: Effect...
) -> Effect {
.merge(effects)
}
/// Merges a sequence of effects together into a single effect, which runs the effects at the same
/// time.
///
/// - Parameter effects: A sequence of effects.
/// - Returns: A new effect
public static func merge<S: Sequence>(_ effects: S) -> Effect where S.Element == Effect {
Publishers.MergeMany(effects).eraseToEffect()
}
/// Creates an effect that executes some work in the real world that doesn't need to feed data
/// back into the store. If an error is thrown, the effect will complete and the error will be ignored.
///
/// - Parameter work: A closure encapsulating some work to execute in the real world.
/// - Returns: An effect.
public static func fireAndForget(_ work: @escaping () throws -> Void) -> Effect {
// NB: Ideally we'd return a `Deferred` wrapping an `Empty(completeImmediately: true)`, but
// due to a bug in iOS 13.2 that publisher will never complete. The bug was fixed in
// iOS 13.3, but to remain compatible with iOS 13.2 and higher we need to do a little
// trickery to make sure the deferred publisher completes.
Deferred { () -> Publishers.CompactMap<Result<Output?, Failure>.Publisher, Output> in
try? work()
return Just<Output?>(nil)
.setFailureType(to: Failure.self)
.compactMap { $0 }
}
.eraseToEffect()
}
/// Transforms all elements from the upstream effect with a provided closure.
///
/// - Parameter transform: A closure that transforms the upstream effect's output to a new output.
/// - Returns: A publisher that uses the provided closure to map elements from the upstream effect
/// to new elements that it then publishes.
public func map<T>(_ transform: @escaping (Output) -> T) -> Effect<T, Failure> {
.init(self.map(transform) as Publishers.Map<Self, T>)
}
}
extension Effect where Failure == Swift.Error {
/// Initializes an effect that lazily executes some work in the real world and synchronously sends
/// that data back into the store.
///
/// For example, to load a user from some JSON on the disk, one can wrap that work in an effect:
///
/// ```swift
/// Effect<User, Error>.catching {
/// let fileUrl = URL(
/// fileURLWithPath: NSSearchPathForDirectoriesInDomains(
/// .documentDirectory, .userDomainMask, true
/// )[0]
/// )
/// .appendingPathComponent("user.json")
///
/// let data = try Data(contentsOf: fileUrl)
/// return try JSONDecoder().decode(User.self, from: $0)
/// }
/// ```
///
/// - Parameter work: A closure encapsulating some work to execute in the real world.
/// - Returns: An effect.
public static func catching(_ work: @escaping () throws -> Output) -> Self {
.future { $0(Result { try work() }) }
}
}
extension Publisher {
/// Turns any publisher into an ``Effect``.
///
/// This can be useful for when you perform a chain of publisher transformations in a reducer, and
/// you need to convert that publisher to an effect so that you can return it from the reducer:
///
/// ```swift
/// case .buttonTapped:
/// return fetchUser(id: 1)
/// .filter(\.isAdmin)
/// .eraseToEffect()
/// ```
///
/// - Returns: An effect that wraps `self`.
public func eraseToEffect() -> Effect<Output, Failure> {
Effect(self)
}
/// Turns any publisher into an ``Effect``.
///
/// This is a convenience operator for writing ``Effect/eraseToEffect()`` followed by `map`.
///
/// ```swift
/// case .buttonTapped:
/// return fetchUser(id: 1)
/// .filter(\.isAdmin)
/// .eraseToEffect(ProfileAction.adminUserFetched)
/// ```
///
/// - Parameters:
/// - transform: A mapping function that converts `Output` to another type.
/// - Returns: An effect that wraps `self` after mapping `Output` values.
public func eraseToEffect<T>(
_ transform: @escaping (Output) -> T
) -> Effect<T, Failure> {
self.map(transform)
.eraseToEffect()
}
/// Turns any publisher into an ``Effect`` that cannot fail by wrapping its output and failure in
/// a result.
///
/// This can be useful when you are working with a failing API but want to deliver its data to an
/// action that handles both success and failure.
///
/// ```swift
/// case .buttonTapped:
/// return environment.fetchUser(id: 1)
/// .catchToEffect()
/// .map(ProfileAction.userResponse)
/// ```
///
/// - Returns: An effect that wraps `self`.
public func catchToEffect() -> Effect<Result<Output, Failure>, Never> {
self.map(Result.success)
.catch { Just(.failure($0)) }
.eraseToEffect()
}
/// Turns any publisher into an ``Effect`` that cannot fail by wrapping its output and failure
/// into a result and then applying passed in function to it.
///
/// This is a convenience operator for writing ``Effect/catchToEffect()`` followed by `map`.
///
/// ```swift
/// case .buttonTapped:
/// return environment.fetchUser(id: 1)
/// .catchToEffect(ProfileAction.userResponse)
/// ```
///
/// - Parameters:
/// - transform: A mapping function that converts `Result<Output,Failure>` to another type.
/// - Returns: An effect that wraps `self`.
public func catchToEffect<T>(
_ transform: @escaping (Result<Output, Failure>) -> T
) -> Effect<T, Never> {
self
.map { transform(.success($0)) }
.catch { Just(transform(.failure($0))) }
.eraseToEffect()
}
/// Turns any publisher into an ``Effect`` for any output and failure type by ignoring all output
/// and any failure.
///
/// This is useful for times you want to fire off an effect but don't want to feed any data back
/// into the system. It can automatically promote an effect to your reducer's domain.
///
/// ```swift
/// case .buttonTapped:
/// return analyticsClient.track("Button Tapped")
/// .fireAndForget()
/// ```
///
/// - Parameters:
/// - outputType: An output type.
/// - failureType: A failure type.
/// - Returns: An effect that never produces output or errors.
public func fireAndForget<NewOutput, NewFailure>(
outputType: NewOutput.Type = NewOutput.self,
failureType: NewFailure.Type = NewFailure.self
) -> Effect<NewOutput, NewFailure> {
return
self
.flatMap { _ in Empty<NewOutput, Failure>() }
.catch { _ in Empty() }
.eraseToEffect()
}
}