mirror of
https://github.com/pointfreeco/swift-composable-architecture.git
synced 2025-12-20 09:11:33 +01:00
* wip * wip * wip * wip * wip * wip * wip * wip * wip * wip * Silence test warnings * wip * wip * wip * update a bunch of docs * wip * wip * fix * wip * wip * wip * wip * wip * wip * wip * wip * wip * wip * wip * wip * wip * wip * wip * Kill integration tests for now * wip * wip * wip * wip * updating docs for @Reducer macro * replaced more Reducer protocols with @Reducer * Fixed some broken docc references * wip * Some @Reducer docs * more docs * convert some old styles to new style * wip * wip * wip * wip * wip * wip * wip * bump * update tutorials to use body * update tutorials to use DML on destination state enum * Add diagnostic * wip * updated a few more tests * wip * wip * Add another gotcha * wip * wip * wip * fixes * wip * wip * wip * wip * wip * fix * wip * remove for now * wip * wip * updated some docs * migration guides * more migration guide * fix ci * fix * soft deprecate all apis using AnyCasePath * wip * Fix * fix tests * swift-format 509 compatibility * wip * wip * Update Sources/ComposableArchitecture/Macros.swift Co-authored-by: Mateusz Bąk <bakmatthew@icloud.com> * wip * wip * update optional state case study * remove initializer * Don't use @State for BasicsView integration demo * fix tests * remove reduce diagnostics for now * diagnose error not warning * Update Sources/ComposableArchitecture/Macros.swift Co-authored-by: Jesse Tipton <jesse@jessetipton.com> * wip * move integration tests to cron * Revert "move integration tests to cron" This reverts commitf9bdf2f04b. * disable flakey tests on CI * wip * wip * Revert "Revert "move integration tests to cron"" This reverts commit66aafa7327. * fix * wip * fix --------- Co-authored-by: Brandon Williams <mbrandonw@hey.com> Co-authored-by: Mateusz Bąk <bakmatthew@icloud.com> Co-authored-by: Brandon Williams <135203+mbrandonw@users.noreply.github.com> Co-authored-by: Jesse Tipton <jesse@jessetipton.com>
294 lines
7.6 KiB
Swift
294 lines
7.6 KiB
Swift
import Combine
|
|
import ComposableArchitecture
|
|
import XCTest
|
|
|
|
@MainActor
|
|
final class ViewStoreTests: BaseTCATestCase {
|
|
var cancellables: Set<AnyCancellable> = []
|
|
|
|
override func setUpWithError() throws {
|
|
try super.setUpWithError()
|
|
equalityChecks = 0
|
|
subEqualityChecks = 0
|
|
}
|
|
|
|
func testPublisherFirehose() {
|
|
let store = Store<Int, Void>(initialState: 0) {}
|
|
let viewStore = ViewStore(store, observe: { $0 })
|
|
|
|
var emissionCount = 0
|
|
viewStore.publisher
|
|
.sink { _ in emissionCount += 1 }
|
|
.store(in: &self.cancellables)
|
|
|
|
XCTAssertEqual(emissionCount, 1)
|
|
viewStore.send(())
|
|
XCTAssertEqual(emissionCount, 1)
|
|
viewStore.send(())
|
|
XCTAssertEqual(emissionCount, 1)
|
|
viewStore.send(())
|
|
XCTAssertEqual(emissionCount, 1)
|
|
}
|
|
|
|
func testEqualityChecks() {
|
|
let store = Store<State, Void>(initialState: State()) {}
|
|
|
|
let store1 = store.scope(state: { $0 }, action: { $0 })
|
|
let store2 = store1.scope(state: { $0 }, action: { $0 })
|
|
let store3 = store2.scope(state: { $0 }, action: { $0 })
|
|
let store4 = store3.scope(state: { $0 }, action: { $0 })
|
|
|
|
let viewStore1 = ViewStore(store1, observe: { $0 })
|
|
let viewStore2 = ViewStore(store2, observe: { $0 })
|
|
let viewStore3 = ViewStore(store3, observe: { $0 })
|
|
let viewStore4 = ViewStore(store4, observe: { $0 })
|
|
|
|
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)
|
|
|
|
XCTAssertEqual(0, equalityChecks)
|
|
XCTAssertEqual(0, subEqualityChecks)
|
|
viewStore4.send(())
|
|
XCTAssertEqual(4, equalityChecks)
|
|
XCTAssertEqual(4, subEqualityChecks)
|
|
viewStore4.send(())
|
|
XCTAssertEqual(8, equalityChecks)
|
|
XCTAssertEqual(8, subEqualityChecks)
|
|
viewStore4.send(())
|
|
XCTAssertEqual(12, equalityChecks)
|
|
XCTAssertEqual(12, subEqualityChecks)
|
|
viewStore4.send(())
|
|
XCTAssertEqual(16, equalityChecks)
|
|
XCTAssertEqual(16, subEqualityChecks)
|
|
}
|
|
|
|
func testAccessViewStoreStateInPublisherSink() {
|
|
let reducer = Reduce<Int, Void> { count, _ in
|
|
count += 1
|
|
return .none
|
|
}
|
|
|
|
let store = Store(initialState: 0) { reducer }
|
|
let viewStore = ViewStore(store, observe: { $0 })
|
|
|
|
var results: [Int] = []
|
|
|
|
viewStore.publisher
|
|
.sink { _ in results.append(viewStore.state) }
|
|
.store(in: &self.cancellables)
|
|
|
|
viewStore.send(())
|
|
viewStore.send(())
|
|
viewStore.send(())
|
|
|
|
XCTAssertEqual([0, 1, 2, 3], results)
|
|
}
|
|
|
|
func testWillSet() {
|
|
let reducer = Reduce<Int, Void> { count, _ in
|
|
count += 1
|
|
return .none
|
|
}
|
|
|
|
let store = Store(initialState: 0) { reducer }
|
|
let viewStore = ViewStore(store, observe: { $0 })
|
|
|
|
var results: [Int] = []
|
|
|
|
viewStore.objectWillChange
|
|
.sink { _ in results.append(viewStore.state) }
|
|
.store(in: &self.cancellables)
|
|
|
|
viewStore.send(())
|
|
viewStore.send(())
|
|
viewStore.send(())
|
|
|
|
XCTAssertEqual([0, 1, 2], results)
|
|
}
|
|
|
|
func testPublisherOwnsViewStore() {
|
|
let reducer = Reduce<Int, Void> { count, _ in
|
|
count += 1
|
|
return .none
|
|
}
|
|
let store = Store(initialState: 0) { reducer }
|
|
|
|
var results: [Int] = []
|
|
ViewStore(store, observe: { $0 })
|
|
.publisher
|
|
.sink { results.append($0) }
|
|
.store(in: &self.cancellables)
|
|
|
|
ViewStore(store, observe: { $0 }).send(())
|
|
XCTAssertEqual(results, [0, 1])
|
|
}
|
|
|
|
func testStorePublisherSubscriptionOrder() {
|
|
let reducer = Reduce<Int, Void> { count, _ in
|
|
count += 1
|
|
return .none
|
|
}
|
|
let store = Store(initialState: 0) { reducer }
|
|
let viewStore = ViewStore(store, observe: { $0 })
|
|
|
|
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)
|
|
|
|
XCTAssertEqual(results, [0, 1, 2])
|
|
|
|
for _ in 0..<9 {
|
|
viewStore.send(())
|
|
}
|
|
|
|
XCTAssertEqual(results, Array(repeating: [0, 1, 2], count: 10).flatMap { $0 })
|
|
}
|
|
|
|
func testSendWhile() async {
|
|
enum Action {
|
|
case response
|
|
case tapped
|
|
}
|
|
let reducer = Reduce<Bool, Action> { state, action in
|
|
switch action {
|
|
case .response:
|
|
state = false
|
|
return .none
|
|
case .tapped:
|
|
state = true
|
|
return .run { send in await send(.response) }
|
|
}
|
|
}
|
|
|
|
let store = Store(initialState: false) { reducer }
|
|
let viewStore = ViewStore(store, observe: { $0 })
|
|
|
|
XCTAssertEqual(viewStore.state, false)
|
|
await viewStore.send(.tapped, while: { $0 })
|
|
XCTAssertEqual(viewStore.state, false)
|
|
}
|
|
|
|
func testSuspend() {
|
|
let expectation = self.expectation(description: "await")
|
|
Task {
|
|
enum Action {
|
|
case response
|
|
case tapped
|
|
}
|
|
let reducer = Reduce<Bool, Action> { state, action in
|
|
switch action {
|
|
case .response:
|
|
state = false
|
|
return .none
|
|
case .tapped:
|
|
state = true
|
|
return .run { send in await send(.response) }
|
|
}
|
|
}
|
|
|
|
let store = Store(initialState: false) { reducer }
|
|
let viewStore = ViewStore(store, observe: { $0 })
|
|
|
|
XCTAssertEqual(viewStore.state, false)
|
|
_ = { viewStore.send(.tapped) }()
|
|
XCTAssertEqual(viewStore.state, true)
|
|
await viewStore.yield(while: { $0 })
|
|
XCTAssertEqual(viewStore.state, false)
|
|
expectation.fulfill()
|
|
}
|
|
self.wait(for: [expectation], timeout: 1)
|
|
}
|
|
|
|
func testAsyncSend() async throws {
|
|
enum Action {
|
|
case tap
|
|
case response(Int)
|
|
}
|
|
let store = Store(initialState: 0) {
|
|
Reduce<Int, Action> { state, action in
|
|
switch action {
|
|
case .tap:
|
|
return .run { send in
|
|
await send(.response(42))
|
|
}
|
|
case let .response(value):
|
|
state = value
|
|
return .none
|
|
}
|
|
}
|
|
}
|
|
|
|
let viewStore = ViewStore(store, observe: { $0 })
|
|
|
|
XCTAssertEqual(viewStore.state, 0)
|
|
await viewStore.send(.tap).finish()
|
|
XCTAssertEqual(viewStore.state, 42)
|
|
}
|
|
|
|
func testAsyncSendCancellation() async throws {
|
|
enum Action {
|
|
case tap
|
|
case response(Int)
|
|
}
|
|
let store = Store(initialState: 0) {
|
|
Reduce<Int, Action> { state, action in
|
|
switch action {
|
|
case .tap:
|
|
return .run { send in
|
|
try await Task.sleep(nanoseconds: NSEC_PER_SEC)
|
|
await send(.response(42))
|
|
}
|
|
case let .response(value):
|
|
state = value
|
|
return .none
|
|
}
|
|
}
|
|
}
|
|
|
|
let viewStore = ViewStore(store, observe: { $0 })
|
|
|
|
XCTAssertEqual(viewStore.state, 0)
|
|
let task = viewStore.send(.tap)
|
|
task.cancel()
|
|
try await Task.sleep(nanoseconds: NSEC_PER_MSEC)
|
|
XCTAssertEqual(viewStore.state, 0)
|
|
}
|
|
}
|
|
|
|
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
|