diff --git a/.gitignore b/.gitignore index 3b29812..f129058 100644 --- a/.gitignore +++ b/.gitignore @@ -1,9 +1,9 @@ .DS_Store /.build /Packages -/*.xcodeproj xcuserdata/ DerivedData/ -.swiftpm/config/registries.json +.swiftpm/configuration/registries.json .swiftpm/xcode/package.xcworkspace/contents.xcworkspacedata .netrc +Package.resolved \ No newline at end of file diff --git a/.swiftpm/xcode/xcshareddata/xcschemes/Reducer.xcscheme b/.swiftpm/xcode/xcshareddata/xcschemes/Reducer.xcscheme index 46117df..c81fbb9 100644 --- a/.swiftpm/xcode/xcshareddata/xcschemes/Reducer.xcscheme +++ b/.swiftpm/xcode/xcshareddata/xcschemes/Reducer.xcscheme @@ -77,6 +77,16 @@ ReferencedContainer = "container:"> + + + + State @@ -29,6 +30,8 @@ public extension Reduce { func start() async throws { } + func error(_ error: any Error, action: Action) async { } + func shouldCancel(_ current: Action, _ upcoming: Action) -> Bool { false } @@ -77,6 +80,7 @@ open class ProxyReduce: Reduce { public var initialState: State private let _start: ((@escaping (Mutation) -> Void) async throws -> Void)? + private let _error: ((@escaping (Mutation) -> Void, any Error, Action) async -> Void)? private let _mutate: ((State, Action, @escaping (Mutation) -> Void) async throws -> Void)? private let _reduce: ((State, Mutation) -> State)? private let _shouldCancel: ((Action, Action) -> Bool)? @@ -85,6 +89,7 @@ open class ProxyReduce: Reduce { public init( initialState: State, start: ((@escaping (Mutation) -> Void) async throws -> Void)? = nil, + error: ((@escaping (Mutation) -> Void, any Error, Action) async -> Void)? = nil, mutate: ((State, Action, @escaping (Mutation) -> Void) async throws -> Void)? = nil, reduce: ((State, Mutation) -> State)? = nil, shouldCancel: ((Action, Action) -> Bool)? = nil @@ -92,6 +97,7 @@ open class ProxyReduce: Reduce { self.initialState = initialState self._start = start + self._error = error self._mutate = mutate self._reduce = reduce self._shouldCancel = shouldCancel @@ -103,6 +109,9 @@ open class ProxyReduce: Reduce { start: { _ in try await reduce.start() }, + error: { _, error, action in + await reduce.error(error, action: action) + }, mutate: { _, action, _ in try await reduce.mutate(action: action) }, @@ -126,6 +135,10 @@ open class ProxyReduce: Reduce { try await _start?({ [weak self] mutation in self?.mutator?(mutation) }) } + public func error(_ error: any Error, action: Action) async { + await _error?({ [weak self] mutation in self?.mutator?(mutation) }, error, action) + } + open func mutate(action: Action) async throws { try await _mutate?(currentState, action, { [weak self] mutation in self?.mutator?(mutation) }) } diff --git a/Sources/Reducer/Reducer.swift b/Sources/Reducer/Reducer.swift index ed0c85a..268c769 100644 --- a/Sources/Reducer/Reducer.swift +++ b/Sources/Reducer/Reducer.swift @@ -115,7 +115,11 @@ open class Reducer: ObservableObject, Mutable { action, with: Task { @MainActor in // Mutate state from action. - try? await reduce.mutate(action: action) + do { + try await reduce.mutate(action: action) + } catch { + await reduce.error(error, action: action) + } } )) } diff --git a/Tests/ReducerTests/Model/ErrorReduce.swift b/Tests/ReducerTests/Model/ErrorReduce.swift new file mode 100644 index 0000000..119a962 --- /dev/null +++ b/Tests/ReducerTests/Model/ErrorReduce.swift @@ -0,0 +1,54 @@ +// +// ErrorReduce.swift +// Reducer +// +// Created by JSilver on 11/3/24. +// + +import Foundation +import Reducer + +@Reduce +@MainActor +class ErrorReduce { + enum Action { + case occurError(any Error) + } + + enum Mutation { + case setCount(Int) + } + + struct State { + var count: Int + } + + let initialState: State + + init() { + self.initialState = State( + count: 0 + ) + } + + func error(_ error: any Error, action: Action) async { + mutate(.setCount(currentState.count + 1)) + } + + func mutate(action: Action) async throws { + switch action { + case let .occurError(error): + throw error + } + } + + func reduce(state: State, mutation: Mutation) -> State { + var state = state + + switch mutation { + case let .setCount(count): + state.count = count + return state + } + } +} diff --git a/Tests/ReducerTests/Model/TestError.swift b/Tests/ReducerTests/Model/TestError.swift new file mode 100644 index 0000000..421c4c4 --- /dev/null +++ b/Tests/ReducerTests/Model/TestError.swift @@ -0,0 +1,16 @@ +// +// TestError.swift +// Reducer +// +// Created by JSilver on 11/3/24. +// + +import Foundation + +struct TestError: Error { + let description: String + + init(_ description: String) { + self.description = description + } +} diff --git a/Tests/ReducerTests/ReducerTests.swift b/Tests/ReducerTests/ReducerTests.swift index e1393ce..1bf1797 100644 --- a/Tests/ReducerTests/ReducerTests.swift +++ b/Tests/ReducerTests/ReducerTests.swift @@ -9,6 +9,7 @@ import XCTest @testable import Reducer import Combine +@MainActor final class ReducerTests: XCTestCase { // MARK: - Property @@ -22,7 +23,6 @@ final class ReducerTests: XCTestCase { } // MARK: - Test - @MainActor func test_that_count_increases_when_receiving_increase_action() async throws { // Given let reducer = Reducer( @@ -46,7 +46,6 @@ final class ReducerTests: XCTestCase { XCTAssertEqual(result, [0, 1, 2]) } - @MainActor func test_that_count_does_not_mutate_when_receiving_same_action_twice() async throws { // Given let reducer = Reducer( @@ -70,7 +69,6 @@ final class ReducerTests: XCTestCase { XCTAssertEqual(result, [0, 1]) } - @MainActor func test_that_all_action_cancelled_when_reducer_deinit() async throws { // Given var reducer: Reducer? = Reducer( @@ -96,7 +94,6 @@ final class ReducerTests: XCTestCase { XCTAssertEqual(result, [0]) } - @MainActor func test_that_count_increases_when_mutate_in_start() async throws { // Given let reducer = Reducer(TimerReduce( @@ -113,7 +110,6 @@ final class ReducerTests: XCTestCase { XCTAssertGreaterThan(result.last ?? 0, 0) } - @MainActor func test_that_count_increases_when_await_mutate_in_start() async throws { // Given let reducer = Reducer(AwaitStartReduce( @@ -130,7 +126,6 @@ final class ReducerTests: XCTestCase { XCTAssertEqual(result, [0, 1]) } - @MainActor func test_that_reducer_should_be_able_to_assign_proxy_reduce() async throws { // Given var reducer = Reducer(CountIncreaseReduce( @@ -170,7 +165,6 @@ final class ReducerTests: XCTestCase { XCTAssertEqual(result, [0, 10]) } - @MainActor func test_that_initial_count_is_100_when_proxy_set_initial_count() async throws { // Given let reducer = Reducer(proxy: .init( @@ -189,7 +183,6 @@ final class ReducerTests: XCTestCase { XCTAssertEqual(result, [100]) } - @MainActor func test_that_count_increases_when_proxy_receiving_increase_action() async throws { // Given let reducer = Reducer(proxy: .init( @@ -226,7 +219,6 @@ final class ReducerTests: XCTestCase { XCTAssertEqual(result, [0, 1]) } - @MainActor func test_that_count_increases_10_when_proxy_receiving_increase_action() async throws { // Given let reducer = Reducer(proxy: .init( @@ -264,7 +256,6 @@ final class ReducerTests: XCTestCase { XCTAssertEqual(result, [0, 10, 20]) } - @MainActor func test_that_count_does_not_mutate_when_proxy_same_action_twice() async throws { // Given let reducer = Reducer(proxy: .init( @@ -304,7 +295,6 @@ final class ReducerTests: XCTestCase { XCTAssertEqual(result, [0, 1]) } - @MainActor func test_that_count_increases_when_proxy_mutate_in_start() async throws { // Given var cancellable: AnyCancellable? = nil @@ -336,7 +326,6 @@ final class ReducerTests: XCTestCase { XCTAssertGreaterThan(result.last ?? 0, 0) } - @MainActor func test_that_reducer_can_assign_proxy_inherited_reduce() async throws { // Given let reducer = Reducer( @@ -357,4 +346,22 @@ final class ReducerTests: XCTestCase { // Then XCTAssertEqual(result, [0, 10]) } + + func test_that_count_increases_when_error_is_thrown_in_mutation() async throws { + // Given + let sut = Reducer(ErrorReduce()) + + // When + Task { + sut.action(.occurError(TestError("increase count"))) + } + + let result = try await wait( + sut.$state.map(\.count), + timeout: 1 + ) + + // Then + XCTAssertEqual(result, [0, 1]) + } }