Files
swift-composable-architectu…/Tests/ComposableArchitectureTests/ViewStoreTests.swift
Stephen Celis cfcf8b43ad Rename ViewStore.suspend(while:) to yield(while:) (#1151)
The naming of `ViewStore.suspend(while:)` was inspired by
`Task.suspend()`, which was deprecated and renamed to `Task.yield()`
long ago. Let's do the same here and call it `ViewStore.yield(while:)`.
2022-06-20 09:36:47 -04:00

258 lines
7.0 KiB
Swift

import Combine
import ComposableArchitecture
import XCTest
final class ViewStoreTests: XCTestCase {
var cancellables: Set<AnyCancellable> = []
override func setUp() {
super.setUp()
equalityChecks = 0
subEqualityChecks = 0
}
func testPublisherFirehose() {
let store = Store(
initialState: 0,
reducer: Reducer<Int, Void, Void>.empty,
environment: ()
)
let viewStore = ViewStore(store)
var emissionCount = 0
viewStore.publisher
.sink { _ in emissionCount += 1 }
.store(in: &self.cancellables)
XCTAssertNoDifference(emissionCount, 1)
viewStore.send(())
XCTAssertNoDifference(emissionCount, 1)
viewStore.send(())
XCTAssertNoDifference(emissionCount, 1)
viewStore.send(())
XCTAssertNoDifference(emissionCount, 1)
}
func testEqualityChecks() {
let store = Store(
initialState: State(),
reducer: Reducer<State, Void, Void>.empty,
environment: ()
)
let store1 = store.scope(state: { $0 })
let store2 = store1.scope(state: { $0 })
let store3 = store2.scope(state: { $0 })
let store4 = store3.scope(state: { $0 })
let viewStore1 = ViewStore(store1)
let viewStore2 = ViewStore(store2)
let viewStore3 = ViewStore(store3)
let viewStore4 = ViewStore(store4)
viewStore1.publisher.sink { _ in }.store(in: &self.cancellables)
viewStore2.publisher.sink { _ in }.store(in: &self.cancellables)
viewStore3.publisher.sink { _ in }.store(in: &self.cancellables)
viewStore4.publisher.sink { _ in }.store(in: &self.cancellables)
viewStore1.publisher.substate.sink { _ in }.store(in: &self.cancellables)
viewStore2.publisher.substate.sink { _ in }.store(in: &self.cancellables)
viewStore3.publisher.substate.sink { _ in }.store(in: &self.cancellables)
viewStore4.publisher.substate.sink { _ in }.store(in: &self.cancellables)
XCTAssertNoDifference(0, equalityChecks)
XCTAssertNoDifference(0, subEqualityChecks)
viewStore4.send(())
XCTAssertNoDifference(4, equalityChecks)
XCTAssertNoDifference(4, subEqualityChecks)
viewStore4.send(())
XCTAssertNoDifference(8, equalityChecks)
XCTAssertNoDifference(8, subEqualityChecks)
viewStore4.send(())
XCTAssertNoDifference(12, equalityChecks)
XCTAssertNoDifference(12, subEqualityChecks)
viewStore4.send(())
XCTAssertNoDifference(16, equalityChecks)
XCTAssertNoDifference(16, subEqualityChecks)
}
func testAccessViewStoreStateInPublisherSink() {
let reducer = Reducer<Int, Void, Void> { count, _, _ in
count += 1
return .none
}
let store = Store(initialState: 0, reducer: reducer, environment: ())
let viewStore = ViewStore(store)
var results: [Int] = []
viewStore.publisher
.sink { _ in results.append(viewStore.state) }
.store(in: &self.cancellables)
viewStore.send(())
viewStore.send(())
viewStore.send(())
XCTAssertNoDifference([0, 1, 2, 3], results)
}
func testWillSet() {
let reducer = Reducer<Int, Void, Void> { count, _, _ in
count += 1
return .none
}
let store = Store(initialState: 0, reducer: reducer, environment: ())
let viewStore = ViewStore(store)
var results: [Int] = []
viewStore.objectWillChange
.sink { _ in results.append(viewStore.state) }
.store(in: &self.cancellables)
viewStore.send(())
viewStore.send(())
viewStore.send(())
XCTAssertNoDifference([0, 1, 2], results)
}
func testPublisherOwnsViewStore() {
let reducer = Reducer<Int, Void, Void> { count, _, _ in
count += 1
return .none
}
let store = Store(initialState: 0, reducer: reducer, environment: ())
var results: [Int] = []
ViewStore(store)
.publisher
.sink { results.append($0) }
.store(in: &self.cancellables)
ViewStore(store).send(())
XCTAssertNoDifference(results, [0, 1])
}
func testStorePublisherSubscriptionOrder() {
let reducer = Reducer<Int, Void, Void> { count, _, _ in
count += 1
return .none
}
let store = Store(initialState: 0, reducer: reducer, environment: ())
let viewStore = ViewStore(store)
var results: [Int] = []
viewStore.publisher
.sink { _ in results.append(0) }
.store(in: &self.cancellables)
viewStore.publisher
.sink { _ in results.append(1) }
.store(in: &self.cancellables)
viewStore.publisher
.sink { _ in results.append(2) }
.store(in: &self.cancellables)
XCTAssertNoDifference(results, [0, 1, 2])
for _ in 0..<9 {
viewStore.send(())
}
XCTAssertNoDifference(results, Array(repeating: [0, 1, 2], count: 10).flatMap { $0 })
}
#if canImport(_Concurrency) && compiler(>=5.5.2)
func testSendWhile() {
let expectation = self.expectation(description: "await")
Task { @MainActor in
enum Action {
case response
case tapped
}
let reducer = Reducer<Bool, Action, Void> { state, action, environment in
switch action {
case .response:
state = false
return .none
case .tapped:
state = true
return Effect(value: .response)
.receive(on: DispatchQueue.main)
.eraseToEffect()
}
}
let store = Store(initialState: false, reducer: reducer, environment: ())
let viewStore = ViewStore(store)
XCTAssertNoDifference(viewStore.state, false)
await viewStore.send(.tapped, while: { $0 })
XCTAssertNoDifference(viewStore.state, false)
expectation.fulfill()
}
self.wait(for: [expectation], timeout: 1)
}
func testSuspend() {
let expectation = self.expectation(description: "await")
Task { @MainActor in
enum Action {
case response
case tapped
}
let reducer = Reducer<Bool, Action, Void> { state, action, environment in
switch action {
case .response:
state = false
return .none
case .tapped:
state = true
return Effect(value: .response)
.receive(on: DispatchQueue.main)
.eraseToEffect()
}
}
let store = Store(initialState: false, reducer: reducer, environment: ())
let viewStore = ViewStore(store)
XCTAssertNoDifference(viewStore.state, false)
viewStore.send(.tapped)
XCTAssertNoDifference(viewStore.state, true)
await viewStore.yield(while: { $0 })
XCTAssertNoDifference(viewStore.state, false)
expectation.fulfill()
}
self.wait(for: [expectation], timeout: 1)
}
#endif
}
private struct State: Equatable {
var substate = Substate()
static func == (lhs: Self, rhs: Self) -> Bool {
equalityChecks += 1
return lhs.substate == rhs.substate
}
}
private struct Substate: Equatable {
var name = "Blob"
static func == (lhs: Self, rhs: Self) -> Bool {
subEqualityChecks += 1
return lhs.name == rhs.name
}
}
private var equalityChecks = 0
private var subEqualityChecks = 0