Skip to content

Commit b6457f6

Browse files
feat!: Spec v0.8 adherence (#46)
<!-- Please use this template for your pull request. --> <!-- Please use the sections that you need and delete other sections --> ## Spec v0.8 adherence - Created "ProviderStatus.swift" according to [Specification](https://github.com/open-feature/spec/blob/634f2b52f990232b0b1c54c38f422cbff2c507bc/specification/types.md#provider-status). - ProviderState is now maintained by the SDK, and not by the Provider implementations - Align "ProviderEvent.swift" with [Specification](https://github.com/open-feature/spec/blob/634f2b52f990232b0b1c54c38f422cbff2c507bc/specification/types.md#provider-events) - Basic events like `Ready`, `Error` or `Fatal` are emitted by the SDK and are not expected by the Provider implementations. - Makes `initialize` and `onContextChange` protocols throwing, allowing for the OpenFeatureAPI layer to detect cases where `Error` events should be emitted - Makes `initialize` and `onContextChange` protocols async for an easier Provider implementation (in most cases) ### Missing parts - From the [Specification](https://openfeature.dev/specification/sections/events#conditional-requirement-5342): `It's possible that the on context changed function is invoked simultaneously or in quick succession; in this case the SDK will only run the PROVIDER_CONTEXT_CHANGED handlers after all reentrant invocations have terminated, and the last to terminate was successful (terminated normally)` - This is not part of this PR: subsequent calls are serialized and executed in sequence, each returning from its execution even if more same functions are queued - Handling of [spontaneous Provider events](https://openfeature.dev/specification/sections/events#requirement-535) missing in this PR ## Changes unrelated to Spec v0.8 - Added `setEvaluationContextAndWait` to provide the same better ergonomics that `setProviderAndWait` already offers - Evaluation returns early if provider state is not "ready" - Refactor `afterAll` hook to adhere to latest conventions ## Notes on backwards compatibility - A backwards incompatible aspect of this change is that the `initialize()` implementation needs to throw in case of errors, rather than emitting the `ERROR` event. The latter is now responsibility of the SDK. This is also valid for `onContextChange`. - This change in semantics is not enforced with code, and it might be easy for the Provider implementer to make mistakes when adopting this new version of the SDK (e.g. forget to remove `ERROR` emission from the Provider side) ## Example Adoption The Confidence OpenFeature Provider adopting this version of the SDK is in the works here: spotify/confidence-sdk-swift#184 --------- Signed-off-by: Fabrizio Demaria <[email protected]> Co-authored-by: Michael Beemer <[email protected]>
1 parent b900743 commit b6457f6

24 files changed

+505
-307
lines changed

README.md

+9-4
Original file line numberDiff line numberDiff line change
@@ -14,7 +14,7 @@
1414
<p align="center" class="github-badges">
1515
<!-- TODO: update this with the version of the SDK your implementation supports -->
1616

17-
<a href="https://github.com/open-feature/spec/releases/tag/v0.7.0">
17+
<a href="https://github.com/open-feature/spec/releases/tag/v0.8.0">
1818
<img alt="Specification" src="https://img.shields.io/static/v1?label=specification&message=v0.7.0&color=yellow&style=for-the-badge" />
1919
</a>
2020
<!-- x-release-please-start-version -->
@@ -90,11 +90,12 @@ Task {
9090
## 🌟 Features
9191

9292

93-
| Status | Features | Description |
94-
| ------ | ------------------------------- | ---------------------------------------------------------------------------------------------------------------------------------- |
93+
| Status | Features | Description |
94+
| ------ | ------------------------------- | ---------------------------------------------------------------------------------------------------------------------------------- |
9595
|| [Providers](#providers) | Integrate with a commercial, open source, or in-house feature management tool. |
9696
|| [Targeting](#targeting) | Contextually-aware flag evaluation using [evaluation context](https://openfeature.dev/docs/reference/concepts/evaluation-context). |
9797
|| [Hooks](#hooks) | Add functionality to various stages of the flag evaluation life-cycle. |
98+
|| [Tracking](#tracking) | Associate user actions with feature flag evaluations. |
9899
|| [Logging](#logging) | Integrate with popular logging packages. |
99100
|| [Named clients](#named-clients) | Utilize multiple providers in a single application. |
100101
|| [Eventing](#eventing) | React to state changes in the provider or flag management system. |
@@ -153,6 +154,10 @@ _ = client.getValue(
153154
defaultValue: false,
154155
options: FlagEvaluationOptions(hooks: [ExampleHook()]))
155156
```
157+
### Tracking
158+
159+
Tracking is not yet available in the iOS SDK.
160+
156161
### Logging
157162

158163
Logging customization is not yet available in the iOS SDK.
@@ -242,7 +247,7 @@ class BooleanHook: Hook {
242247
// do something
243248
}
244249

245-
func finallyAfter<HookValue>(ctx: HookContext<HookValue>, hints: [String: Any]) {
250+
func finally<HookValue>(ctx: HookContext<HookValue>, hints: [String: Any]) {
246251
// do something
247252
}
248253
}

Sources/OpenFeature/EventHandler.swift

+6-11
Original file line numberDiff line numberDiff line change
@@ -2,29 +2,24 @@ import Combine
22
import Foundation
33

44
public class EventHandler: EventSender, EventPublisher {
5-
private let eventState: CurrentValueSubject<ProviderEvent, Never>
5+
private let lastSentEvent = PassthroughSubject<ProviderEvent?, Never>()
66

7-
convenience init() {
8-
self.init(.notReady)
7+
public init() {
98
}
109

11-
public init(_ state: ProviderEvent) {
12-
eventState = CurrentValueSubject<ProviderEvent, Never>(state)
13-
}
14-
15-
public func observe() -> AnyPublisher<ProviderEvent, Never> {
16-
return eventState.eraseToAnyPublisher()
10+
public func observe() -> AnyPublisher<ProviderEvent?, Never> {
11+
return lastSentEvent.eraseToAnyPublisher()
1712
}
1813

1914
public func send(
2015
_ event: ProviderEvent
2116
) {
22-
eventState.send(event)
17+
lastSentEvent.send(event)
2318
}
2419
}
2520

2621
public protocol EventPublisher {
27-
func observe() -> AnyPublisher<ProviderEvent, Never>
22+
func observe() -> AnyPublisher<ProviderEvent?, Never>
2823
}
2924

3025
public protocol EventSender {

Sources/OpenFeature/Hook.swift

+8-2
Original file line numberDiff line numberDiff line change
@@ -11,7 +11,11 @@ public protocol Hook {
1111

1212
func error<HookValue>(ctx: HookContext<HookValue>, error: Error, hints: [String: Any])
1313

14-
func finallyAfter<HookValue>(ctx: HookContext<HookValue>, hints: [String: Any])
14+
func finally<HookValue>(
15+
ctx: HookContext<HookValue>,
16+
details: FlagEvaluationDetails<HookValue>,
17+
hints: [String: Any]
18+
)
1519

1620
func supportsFlagValueType(flagValueType: FlagValueType) -> Bool
1721
}
@@ -31,7 +35,9 @@ extension Hook {
3135
// Default implementation
3236
}
3337

34-
public func finallyAfter<HookValue>(ctx: HookContext<HookValue>, hints: [String: Any]) {
38+
public func finally<HookValue>(
39+
ctx: HookContext<HookValue>, details: FlagEvaluationDetails<HookValue>, hints: [String: Any]
40+
) {
3541
// Default implementation
3642
}
3743

Sources/OpenFeature/HookSupport.swift

+20-16
Original file line numberDiff line numberDiff line change
@@ -4,20 +4,12 @@ import os
44
class HookSupport {
55
var logger = Logger()
66

7-
func errorHooks<T>(
8-
flagValueType: FlagValueType, hookCtx: HookContext<T>, error: Error, hooks: [any Hook], hints: [String: Any]
9-
) {
10-
hooks
11-
.filter { $0.supportsFlagValueType(flagValueType: flagValueType) }
12-
.forEach { $0.error(ctx: hookCtx, error: error, hints: hints) }
13-
}
14-
15-
func afterAllHooks<T>(
16-
flagValueType: FlagValueType, hookCtx: HookContext<T>, hooks: [any Hook], hints: [String: Any]
17-
) {
7+
func beforeHooks<T>(flagValueType: FlagValueType, hookCtx: HookContext<T>, hooks: [any Hook], hints: [String: Any])
8+
{
189
hooks
10+
.reversed()
1911
.filter { $0.supportsFlagValueType(flagValueType: flagValueType) }
20-
.forEach { $0.finallyAfter(ctx: hookCtx, hints: hints) }
12+
.forEach { $0.before(ctx: hookCtx, hints: hints) }
2113
}
2214

2315
func afterHooks<T>(
@@ -32,11 +24,23 @@ class HookSupport {
3224
.forEach { $0.after(ctx: hookCtx, details: details, hints: hints) }
3325
}
3426

35-
func beforeHooks<T>(flagValueType: FlagValueType, hookCtx: HookContext<T>, hooks: [any Hook], hints: [String: Any])
36-
{
27+
func errorHooks<T>(
28+
flagValueType: FlagValueType, hookCtx: HookContext<T>, error: Error, hooks: [any Hook], hints: [String: Any]
29+
) {
3730
hooks
38-
.reversed()
3931
.filter { $0.supportsFlagValueType(flagValueType: flagValueType) }
40-
.forEach { $0.before(ctx: hookCtx, hints: hints) }
32+
.forEach { $0.error(ctx: hookCtx, error: error, hints: hints) }
33+
}
34+
35+
func finallyHooks<T>(
36+
flagValueType: FlagValueType,
37+
hookCtx: HookContext<T>,
38+
details: FlagEvaluationDetails<T>,
39+
hooks: [any Hook],
40+
hints: [String: Any]
41+
) {
42+
hooks
43+
.filter { $0.supportsFlagValueType(flagValueType: flagValueType) }
44+
.forEach { $0.finally(ctx: hookCtx, details: details, hints: hints) }
4145
}
4246
}

Sources/OpenFeature/OpenFeatureAPI.swift

+129-45
Original file line numberDiff line numberDiff line change
@@ -4,52 +4,107 @@ import Foundation
44
/// A global singleton which holds base configuration for the OpenFeature library.
55
/// Configuration here will be shared across all ``Client``s.
66
public class OpenFeatureAPI {
7-
private var _provider: FeatureProvider? {
8-
get {
9-
providerSubject.value
10-
}
11-
set {
12-
providerSubject.send(newValue)
13-
}
14-
}
15-
private var _context: EvaluationContext?
7+
private let eventHandler = EventHandler()
8+
private let queue = DispatchQueue(label: "com.openfeature.providerDescriptor.queue")
9+
10+
private(set) var providerSubject = CurrentValueSubject<FeatureProvider?, Never>(nil)
11+
private(set) var evaluationContext: EvaluationContext?
12+
private(set) var providerStatus: ProviderStatus = .notReady
1613
private(set) var hooks: [any Hook] = []
17-
private var providerSubject = CurrentValueSubject<FeatureProvider?, Never>(nil)
1814

1915
/// The ``OpenFeatureAPI`` singleton
2016
static public let shared = OpenFeatureAPI()
2117

2218
public init() {
2319
}
2420

25-
public func setProvider(provider: FeatureProvider) {
26-
self.setProvider(provider: provider, initialContext: nil)
21+
/**
22+
Set provider and calls its `initialize` in a background thread.
23+
Readiness can be determined from `getState` or listening for `ready` event.
24+
*/
25+
public func setProvider(provider: FeatureProvider, initialContext: EvaluationContext?) {
26+
queue.async {
27+
Task {
28+
await self.setProviderInternal(provider: provider, initialContext: initialContext)
29+
}
30+
}
2731
}
2832

29-
public func setProvider(provider: FeatureProvider, initialContext: EvaluationContext?) {
30-
self._provider = provider
31-
if let context = initialContext {
32-
self._context = context
33+
/**
34+
Set provider and calls its `initialize`.
35+
This async function returns when the `initialize` from the provider is completed.
36+
*/
37+
public func setProviderAndWait(provider: FeatureProvider, initialContext: EvaluationContext?) async {
38+
await withCheckedContinuation { continuation in
39+
queue.async {
40+
Task {
41+
await self.setProviderInternal(provider: provider, initialContext: initialContext)
42+
continuation.resume()
43+
}
44+
}
3345
}
34-
provider.initialize(initialContext: self._context)
46+
}
47+
48+
/**
49+
Set provider and calls its `initialize` in a background thread.
50+
Readiness can be determined from `getState` or listening for `ready` event.
51+
*/
52+
public func setProvider(provider: FeatureProvider) {
53+
setProvider(provider: provider, initialContext: nil)
54+
}
55+
56+
/**
57+
Set provider and calls its `initialize`.
58+
This async function returns when the `initialize` from the provider is completed.
59+
*/
60+
public func setProviderAndWait(provider: FeatureProvider) async {
61+
await setProviderAndWait(provider: provider, initialContext: nil)
3562
}
3663

3764
public func getProvider() -> FeatureProvider? {
38-
return self._provider
65+
return self.providerSubject.value
3966
}
4067

4168
public func clearProvider() {
42-
self._provider = nil
69+
queue.sync {
70+
self.providerSubject.send(nil)
71+
self.providerStatus = .notReady
72+
}
4373
}
4474

75+
/**
76+
Set evaluation context and calls the provider's `onContextSet` in a background thread.
77+
Readiness can be determined from `getState` or listening for `contextChanged` event.
78+
*/
4579
public func setEvaluationContext(evaluationContext: EvaluationContext) {
46-
let oldContext = self._context
47-
self._context = evaluationContext
48-
getProvider()?.onContextSet(oldContext: oldContext, newContext: evaluationContext)
80+
queue.async {
81+
Task {
82+
await self.updateContext(evaluationContext: evaluationContext)
83+
}
84+
}
85+
}
86+
87+
/**
88+
Set evaluation context and calls the provider's `onContextSet`.
89+
This async function returns when the `onContextSet` from the provider is completed.
90+
*/
91+
public func setEvaluationContextAndWait(evaluationContext: EvaluationContext) async {
92+
await withCheckedContinuation { continuation in
93+
queue.async {
94+
Task {
95+
await self.updateContext(evaluationContext: evaluationContext)
96+
continuation.resume()
97+
}
98+
}
99+
}
49100
}
50101

51102
public func getEvaluationContext() -> EvaluationContext? {
52-
return self._context
103+
return self.evaluationContext
104+
}
105+
106+
public func getProviderStatus() -> ProviderStatus {
107+
return self.providerStatus
53108
}
54109

55110
public func getProviderMetadata() -> ProviderMetadata? {
@@ -72,43 +127,72 @@ public class OpenFeatureAPI {
72127
self.hooks.removeAll()
73128
}
74129

75-
public func observe() -> AnyPublisher<ProviderEvent, Never> {
130+
public func observe() -> AnyPublisher<ProviderEvent?, Never> {
76131
return providerSubject.map { provider in
77132
if let provider = provider {
78133
return provider.observe()
134+
.merge(with: self.eventHandler.observe())
135+
.eraseToAnyPublisher()
79136
} else {
80-
return Empty<ProviderEvent, Never>()
137+
return Empty<ProviderEvent?, Never>()
81138
.eraseToAnyPublisher()
82139
}
83140
}
84141
.switchToLatest()
85142
.eraseToAnyPublisher()
86143
}
87-
}
88144

89-
extension OpenFeatureAPI {
90-
public func setProviderAndWait(provider: FeatureProvider) async {
91-
await setProviderAndWait(provider: provider, initialContext: nil)
145+
internal func getState() -> OpenFeatureState {
146+
return queue.sync {
147+
OpenFeatureState(
148+
provider: providerSubject.value,
149+
evaluationContext: evaluationContext,
150+
providerStatus: providerStatus)
151+
}
92152
}
93153

94-
public func setProviderAndWait(provider: FeatureProvider, initialContext: EvaluationContext?) async {
95-
let task = Task {
96-
var holder: [AnyCancellable] = []
97-
await withCheckedContinuation { continuation in
98-
let stateObserver = provider.observe().sink {
99-
if $0 == .ready || $0 == .error {
100-
continuation.resume()
101-
holder.removeAll()
102-
}
103-
}
104-
stateObserver.store(in: &holder)
105-
setProvider(provider: provider, initialContext: initialContext)
154+
private func setProviderInternal(provider: FeatureProvider, initialContext: EvaluationContext? = nil) async {
155+
self.providerStatus = .notReady
156+
self.providerSubject.send(provider)
157+
158+
if let initialContext = initialContext {
159+
self.evaluationContext = initialContext
160+
}
161+
162+
do {
163+
try await provider.initialize(initialContext: initialContext)
164+
self.providerStatus = .ready
165+
self.eventHandler.send(.ready)
166+
} catch {
167+
switch error {
168+
case OpenFeatureError.providerFatalError:
169+
self.providerStatus = .fatal
170+
self.eventHandler.send(.error(errorCode: .providerFatal))
171+
default:
172+
self.providerStatus = .error
173+
self.eventHandler.send(.error(message: error.localizedDescription))
106174
}
107175
}
108-
await withTaskCancellationHandler {
109-
await task.value
110-
} onCancel: {
111-
task.cancel()
176+
}
177+
178+
private func updateContext(evaluationContext: EvaluationContext) async {
179+
do {
180+
let oldContext = self.evaluationContext
181+
self.evaluationContext = evaluationContext
182+
self.providerStatus = .reconciling
183+
eventHandler.send(.reconciling)
184+
try await self.providerSubject.value?.onContextSet(oldContext: oldContext, newContext: evaluationContext)
185+
self.providerStatus = .ready
186+
eventHandler.send(.contextChanged)
187+
} catch {
188+
self.providerStatus = .error
189+
eventHandler.send(.error(message: error.localizedDescription))
112190
}
113191
}
192+
193+
struct OpenFeatureState {
194+
let provider: FeatureProvider?
195+
let evaluationContext: EvaluationContext?
196+
let providerStatus: ProviderStatus
197+
}
114198
}

0 commit comments

Comments
 (0)