-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathDependencyKey.swift
242 lines (226 loc) · 9.42 KB
/
DependencyKey.swift
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
/// A key for accessing dependencies.
///
/// Types conform to this protocol to extend ``DependencyValues`` with custom dependencies. It is
/// similar to SwiftUI's `EnvironmentKey` protocol, which is used to add values to
/// `EnvironmentValues`.
///
/// `DependencyKey` has one main requirement, ``liveValue``, which must return a default value for
/// your dependency that is used when the application is run in a simulator or device. If the
/// ``liveValue`` is accessed while your feature runs in tests a test failure will be
/// triggered.
///
/// To add a `UserClient` dependency that can fetch and save user values can be done like so:
///
/// ```swift
/// // The user client dependency.
/// struct UserClient {
/// var fetchUser: (User.ID) async throws -> User
/// var saveUser: (User) async throws -> Void
/// }
/// // Conform to DependencyKey to provide a live implementation of
/// // the interface.
/// extension UserClient: DependencyKey {
/// static let liveValue = Self(
/// fetchUser: { /* Make request to fetch user */ },
/// saveUser: { /* Make request to save user */ }
/// )
/// }
/// // Register the dependency within DependencyValues.
/// extension DependencyValues {
/// var userClient: UserClient {
/// get { self[UserClient.self] }
/// set { self[UserClient.self] = newValue }
/// }
/// }
/// ```
///
/// When a dependency is first accessed its value is cached so that it will not be requested again.
/// This means if your `liveValue` is implemented as a computed property instead of a `static let`,
/// then it will only be called a single time:
///
/// ```swift
/// extension UserClient: DependencyKey {
/// static var liveValue: Self {
/// // Only called once when dependency is first accessed.
/// return Self(/* ... */)
/// }
/// }
/// ```
///
/// `DependencyKey` inherits from ``TestDependencyKey``, which has two other overridable
/// requirements: ``TestDependencyKey/testValue``, which should return a default value for the
/// purpose of testing, and ``TestDependencyKey/previewValue-8u2sy``, which can return a default
/// value suitable for Xcode previews. When left unimplemented, these endpoints will return the
/// ``liveValue``, instead.
///
/// If you plan on separating your interface from your live implementation, conform to
/// ``TestDependencyKey`` in your interface module, and conform to `DependencyKey` in your
/// implementation module.
///
/// See the <doc:LivePreviewTest> article for more information.
public protocol DependencyKey: TestDependencyKey {
/// The live value for the dependency key.
///
/// This is the value used by default when running the application in a simulator or on a device.
/// Using a live dependency in a test context will lead to a test failure as you should mock your
/// dependencies for tests.
///
/// To automatically supply a test dependency in a test context, consider implementing the
/// ``testValue-535kh`` requirement.
static var liveValue: Value { get }
// NB: The associated type and requirements of TestDependencyKey are repeated in this protocol
// due to a Swift compiler bug that prevents it from inferring the associated type in
// in the base protocol. See this issue for more information:
// https://github.com/apple/swift/issues/61077
/// The associated type representing the type of the dependency key's value.
associatedtype Value = Self
/// The preview value for the dependency key.
///
/// This value is automatically used when the associated dependency value is accessed from an
/// Xcode preview, as well as when the current ``DependencyValues/context`` is set to
/// ``DependencyContext/preview``:
///
/// ```swift
/// withDependencies {
/// $0.context = .preview
/// } operation: {
/// // Dependencies accessed here default to their "preview" value
/// }
/// ```
static var previewValue: Value { get }
/// The test value for the dependency key.
///
/// This value is automatically used when the associated dependency value is accessed from an
/// XCTest run, as well as when the current ``DependencyValues/context`` is set to
/// ``DependencyContext/test``:
///
/// ```swift
/// withDependencies {
/// $0.context = .test
/// } operation: {
/// // Dependencies accessed here default to their "test" value
/// }
/// ```
static var testValue: Value { get }
}
/// A key for accessing test dependencies.
///
/// This protocol lives one layer below ``DependencyKey`` and allows you to separate a dependency's
/// interface from its live implementation.
///
/// ``TestDependencyKey`` has one main requirement, ``testValue``, which must return a default value
/// for the purposes of testing, and one optional requirement, ``previewValue-8u2sy``, which can
/// return a default value suitable for Xcode previews, or the ``testValue``, if left unimplemented.
///
/// See ``DependencyKey`` to define a static, default value for the live application.
public protocol TestDependencyKey {
/// The associated type representing the type of the dependency key's value.
associatedtype Value = Self
// NB: This associated type should be constrained to `Sendable` when this bug is fixed:
// https://github.com/apple/swift/issues/60649
/// The preview value for the dependency key.
///
/// This value is automatically used when the associated dependency value is accessed from an
/// Xcode preview, as well as when the current ``DependencyValues/context`` is set to
/// ``DependencyContext/preview``:
///
/// ```swift
/// withDependencies {
/// $0.context = .preview
/// } operation: {
/// // Dependencies accessed here default to their "preview" value
/// }
/// ```
static var previewValue: Value { get }
/// The test value for the dependency key.
///
/// This value is automatically used when the associated dependency value is accessed from an
/// XCTest run, as well as when the current ``DependencyValues/context`` is set to
/// ``DependencyContext/test``:
///
/// ```swift
/// withDependencies {
/// $0.context = .test
/// } operation: {
/// // Dependencies accessed here default to their "test" value
/// }
/// ```
static var testValue: Value { get }
}
extension DependencyKey {
/// A default implementation that provides the ``liveValue`` to Xcode previews.
///
/// You may provide your own default `previewValue` in your conformance to ``TestDependencyKey``,
/// which will take precedence over this implementation.
public static var previewValue: Value { Self.liveValue }
/// A default implementation that provides the ``previewValue`` to XCTest runs (or ``liveValue``,
/// if no preview value is implemented), but will trigger a test failure when accessed.
///
/// To prevent test failures, explicitly override the dependency in any tests in which it is
/// accessed:
///
/// ```swift
/// func testFeatureThatUsesMyDependency() {
/// withDependencies {
/// $0.myDependency = .mock // Override dependency
/// } operation: {
/// // Test feature with dependency overridden
/// }
/// }
/// ```
///
/// You may provide your own default `testValue` in your conformance to ``TestDependencyKey``,
/// which will take precedence over this implementation.
public static var testValue: Value {
guard !DependencyValues.isSetting
else { return Self.previewValue }
var dependencyDescription = ""
if let fileID = DependencyValues.currentDependency.fileID,
let line = DependencyValues.currentDependency.line
{
dependencyDescription.append(
"""
Location:
\(fileID):\(line)
"""
)
}
dependencyDescription.append(
Self.self == Value.self
? """
Dependency:
\(typeName(Value.self))
"""
: """
Key:
\(typeName(Self.self))
Value:
\(typeName(Value.self))
"""
)
XCTFail(
"""
\(DependencyValues.currentDependency.name.map { "@Dependency(\\.\($0))" } ?? "A dependency") \
has no test implementation, but was accessed from a test context:
\(dependencyDescription)
Dependencies registered with the library are not allowed to use their default, live \
implementations when run from tests.
To fix, override \
\(DependencyValues.currentDependency.name.map { "'\($0)'" } ?? "the dependency") with a test \
value. If you are using the Composable Architecture, mutate the 'dependencies' property on \
your 'TestStore'. Otherwise, use 'withDependencies' to define a scope for the override. \
If you'd like to provide a default value for all tests, implement the 'testValue' \
requirement of the 'DependencyKey' protocol.
"""
)
return Self.previewValue
}
}
extension TestDependencyKey {
/// A default implementation that provides the
/// <doc:/documentation/Dependencies/TestDependencyKey/testValue> to Xcode previews.
///
/// You may provide your own default `previewValue` in your conformance to ``TestDependencyKey``,
/// which will take precedence over this implementation.
public static var previewValue: Value { Self.testValue }
}