Skip to content

Commit b911046

Browse files
feat: OpenFeature provider for GO Feature Flag (#1)
1 parent 908cea3 commit b911046

18 files changed

+2078
-0
lines changed

.github/workflows/swift.yaml

+3
Original file line numberDiff line numberDiff line change
@@ -21,11 +21,14 @@ jobs:
2121
run: swift build
2222
- name: Run tests
2323
run: swift test --enable-code-coverage
24+
- name: Prepare Code Coverage
25+
run: xcrun llvm-cov export -format="lcov" .build/debug/go-feature-flag-providerPackageTests.xctest/Contents/MacOS/go-feature-flag-providerPackageTests -instr-profile .build/debug/codecov/default.profdata > info.lcov
2426
- name: Upload coverage reports to Codecov
2527
uses: codecov/codecov-action@v4
2628
with:
2729
fail_ci_if_error: true
2830
token: ${{ secrets.CODECOV_TOKEN }}
31+
files: info.lcov
2932
env:
3033
CODECOV_TOKEN: ${{ secrets.CODECOV_TOKEN }}
3134
lint:

.swiftlint.yml

+75
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,75 @@
1+
# By default, SwiftLint uses a set of sensible default rules you can adjust:
2+
disabled_rules: # rule identifiers turned on by default to exclude from running
3+
- colon
4+
- comma
5+
- control_statement
6+
opt_in_rules: # some rules are turned off by default, so you need to opt-in
7+
- empty_count # find all the available rules by running: `swiftlint rules`
8+
9+
# Alternatively, specify all rules explicitly by uncommenting this option:
10+
# only_rules: # delete `disabled_rules` & `opt_in_rules` if using this
11+
# - empty_parameters
12+
# - vertical_whitespace
13+
14+
analyzer_rules: # rules run by `swiftlint analyze`
15+
- explicit_self
16+
17+
# Case-sensitive paths to include during linting. Directory paths supplied on the
18+
# command line will be ignored.
19+
included:
20+
- Sources
21+
excluded: # case-sensitive paths to ignore during linting. Takes precedence over `included`
22+
- Carthage
23+
- Pods
24+
- Sources/ExcludedFolder
25+
- Sources/ExcludedFile.swift
26+
- Sources/*/ExcludedFile.swift # exclude files with a wildcard
27+
28+
# If true, SwiftLint will not fail if no lintable files are found.
29+
allow_zero_lintable_files: false
30+
31+
# If true, SwiftLint will treat all warnings as errors.
32+
strict: false
33+
34+
# The path to a baseline file, which will be used to filter out detected violations.
35+
baseline: Baseline.json
36+
37+
# The path to save detected violations to as a new baseline.
38+
write_baseline: Baseline.json
39+
40+
# configurable rules can be customized from this configuration file
41+
# binary rules can set their severity level
42+
force_cast: warning # implicitly
43+
force_try:
44+
severity: warning # explicitly
45+
# rules that have both warning and error levels, can set just the warning level
46+
# implicitly
47+
line_length: 120
48+
# they can set both implicitly with an array
49+
type_body_length:
50+
- 300 # warning
51+
- 400 # error
52+
# or they can set both explicitly
53+
file_length:
54+
warning: 500
55+
error: 1200
56+
# naming rules can set warnings/errors for min_length and max_length
57+
# additionally they can set excluded names
58+
type_name:
59+
min_length: 4 # only warning
60+
max_length: # warning and error
61+
warning: 40
62+
error: 50
63+
excluded: iPhone # excluded via string
64+
allowed_symbols: ["_"] # these are allowed in type names
65+
identifier_name:
66+
min_length: # only min_length
67+
error: 4 # only error
68+
excluded: # excluded via string array
69+
- id
70+
- URL
71+
- url
72+
- GlobalAPIKey
73+
- key
74+
- dto
75+
reporter: "xcode" # reporter type (xcode, json, csv, checkstyle, codeclimate, junit, html, emoji, sonarqube, markdown, github-actions-logging, summary)

.swiftpm/xcode/package.xcworkspace/contents.xcworkspacedata

+7
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

Package.resolved

+16
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,16 @@
1+
{
2+
"object": {
3+
"pins": [
4+
{
5+
"package": "OpenFeature",
6+
"repositoryURL": "https://github.com/open-feature/swift-sdk.git",
7+
"state": {
8+
"branch": null,
9+
"revision": "02b033c954766e86d5706bfc8ee5248244c11e77",
10+
"version": "0.1.0"
11+
}
12+
}
13+
]
14+
},
15+
"version": 1
16+
}

Package.swift

+35
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,35 @@
1+
// swift-tools-version: 5.5
2+
// The swift-tools-version declares the minimum version of Swift required to build this package.
3+
4+
import PackageDescription
5+
6+
let package = Package(
7+
name: "go-feature-flag-provider",
8+
platforms: [
9+
.iOS(.v14),
10+
.macOS(.v12)
11+
],
12+
products: [
13+
.library(
14+
name: "go-feature-flag-provider",
15+
targets: ["go-feature-flag-provider"])
16+
],
17+
dependencies: [
18+
.package(url: "https://github.com/open-feature/swift-sdk.git", from: "0.1.0")
19+
],
20+
targets: [
21+
.target(
22+
name: "go-feature-flag-provider",
23+
dependencies: [
24+
.product(name: "OpenFeature", package: "swift-sdk")
25+
],
26+
plugins:[]
27+
),
28+
.testTarget(
29+
name: "go-feature-flag-providerTests",
30+
dependencies: [
31+
"go-feature-flag-provider"
32+
]
33+
)
34+
]
35+
)
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,79 @@
1+
import Foundation
2+
import OpenFeature
3+
4+
class OfrepAPI {
5+
private let networkingService: NetworkingService
6+
private var etag: String = ""
7+
private let options: GoFeatureFlagProviderOptions
8+
9+
init(networkingService: NetworkingService, options: GoFeatureFlagProviderOptions) {
10+
self.networkingService = networkingService
11+
self.options = options
12+
}
13+
14+
func postBulkEvaluateFlags(context: EvaluationContext?) async throws -> (OfrepEvaluationResponse, HTTPURLResponse) {
15+
guard let context = context else {
16+
throw OpenFeatureError.invalidContextError
17+
}
18+
try validateContext(context: context)
19+
20+
guard let url = URL(string: options.endpoint) else {
21+
throw InvalidOptions.invalidEndpoint(message: "endpoint [" + options.endpoint + "] is not valid")
22+
}
23+
let ofrepURL = url.appendingPathComponent("ofrep/v1/evaluate/flags")
24+
var request = URLRequest(url: ofrepURL)
25+
request.httpMethod = "POST"
26+
request.httpBody = try EvaluationRequest.convertEvaluationContext(context: context).asJSONData()
27+
request.setValue(
28+
"application/json",
29+
forHTTPHeaderField: "Content-Type"
30+
)
31+
32+
if etag != "" {
33+
request.setValue(etag, forHTTPHeaderField: "If-None-Match")
34+
}
35+
36+
let (data, response) = try await networkingService.doRequest(for: request)
37+
guard let httpResponse = response as? HTTPURLResponse else {
38+
throw OfrepError.httpResponseCastError
39+
}
40+
41+
if httpResponse.statusCode == 401 {
42+
throw OfrepError.apiUnauthorizedError(response: httpResponse)
43+
}
44+
if httpResponse.statusCode == 403 {
45+
throw OfrepError.forbiddenError(response: httpResponse)
46+
}
47+
if httpResponse.statusCode == 429 {
48+
throw OfrepError.apiTooManyRequestsError(response: httpResponse)
49+
}
50+
if httpResponse.statusCode > 400 {
51+
throw OfrepError.unexpectedResponseError(response: httpResponse)
52+
}
53+
if httpResponse.statusCode == 304 {
54+
return (OfrepEvaluationResponse(flags: [], errorCode: nil, errorDetails: nil), httpResponse)
55+
}
56+
57+
// Store ETag to use it in the next request
58+
if let etagHeaderValue = httpResponse.value(forHTTPHeaderField: "ETag") {
59+
if etagHeaderValue != "" && httpResponse.statusCode == 200 {
60+
etag = etagHeaderValue
61+
}
62+
}
63+
64+
do {
65+
let dto = try JSONDecoder().decode(EvaluationResponseDTO.self, from: data)
66+
let evaluationResponse = OfrepEvaluationResponse.fromEvaluationResponseDTO(dto: dto)
67+
return (evaluationResponse, httpResponse)
68+
} catch {
69+
throw OfrepError.unmarshallError(error: error)
70+
}
71+
}
72+
73+
private func validateContext(context: EvaluationContext) throws {
74+
let targetingKey = context.getTargetingKey()
75+
if targetingKey.isEmpty {
76+
throw OpenFeatureError.targetingKeyMissingError
77+
}
78+
}
79+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,18 @@
1+
//
2+
// File.swift
3+
//
4+
//
5+
// Created by thomas.poignant on 27/06/2024.
6+
//
7+
8+
import Foundation
9+
10+
enum OfrepError: Error {
11+
case httpResponseCastError
12+
case unmarshallError(error: Error)
13+
case apiUnauthorizedError(response: HTTPURLResponse)
14+
case forbiddenError(response: HTTPURLResponse)
15+
case apiTooManyRequestsError(response: HTTPURLResponse)
16+
case unexpectedResponseError(response: HTTPURLResponse)
17+
case waitingRetryLater(date: Date?)
18+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,13 @@
1+
//
2+
// File.swift
3+
//
4+
//
5+
// Created by thomas.poignant on 27/06/2024.
6+
//
7+
8+
import Foundation
9+
10+
enum InvalidOptions: Error {
11+
case invalidEndpoint(message: String)
12+
13+
}

0 commit comments

Comments
 (0)