- 3.1 - Using FlagsManager
- 3.2 - Use and Creation of Data Providers
- 3.3 - Firebase Remote Config with FirebaseRemoteProvider
- 3.4 - Modify a feature flag value at runtime
- 3.5 - Listen for flag value updates over time
- 3.6 - Flags Browser & Editor
As you can see above, you must allocate a FlagsLoader
to read values of a feature flag collection. You also need to keep this flags loader instances around; FlagsManager
allows you to collect, keep alive and manage feature flags collection easily.
While you don't need to use it, we suggest it as entry point to query your data.
FlagsManager
is pretty easy to use; consider you want to load a structure for managing several feature flags collection.
Create a single instance of a FlagsLoader
and use the addCollection()
function; this is a silly example in AppDelegate
, but you can do better for sure:
@main
class AppDelegate: UIResponder, UIApplicationDelegate {
// Main manager
public lazy var flagsManager: FlagsManager = {
// Create data providers
let localProvider = LocalProvider()
let firebaseProvider = FirebaseRemoteProvider()
// Create the configuration to manage automatic ff key evaluation
let config = KeyConfiguration(prefix: "myapp_", pathSeparator: "/", keyTransform: .snakeCase)
// Instantiate our manager
let manager = FlagsManager(
providers: [localProvider, firebaseProvider],
keyConfiguration: config
)
// Load some structures
manager.addCollection(MainCollection.self) // main collection
manager.addCollection(ExperimentalFlags.self) // other collection
return manager
}()
// Shortcut
public var appFlags: FlagsLoader<AppFeatureFlags> {
manager.addCollection(AppFeatureFlags.self)
}
}
Now you can forget to keep your collection alive or load them anytime you need them. Just ask for appFlags
:
let cRatingMode = (UIApplication.shared as? AppDelegate)?.appFlags.ratingMode
You can also remove collection from FlagsManager
using the removeCollection()
function.
NOTE: As you may have noticed, only a single collection type can be loaded by a
FlagsManager
: you can't have multiple instances of the same collection type.
RealFlags create an abstract layer of the vertical feature flags implementations. You can use or create a new data provider, which uses your service without altering the structure of your feature flags.
Each data provider must conforms to the FlagsProvider
protocol, which defines two important methods:
// get value
func valueForFlag<Value>(key: FlagKeyPath) -> Value? where Value: FlagProtocol
// set value
func setValue<Value>(_ value: Value?, forFlag key: FlagKeyPath) throws -> Bool where Value: FlagProtocol
One for getting a value and the other to set it.
If your data provider does not provide writable support, you must set the isWritable = false
to inhibit the operation.
RealFlags supports the following built-in providers:
LocalProvider
is a local XML based provider, which support read&write and storage for any types, including all objects conforming to theCodable
protocol. You can instantiate an ephimeral data provider (in-memory), or a file-backed provider to persists data between app restarts.- Any
UserDefaults
instance conforms toFlagsProvider
so you can use it as a data provider. It allows read&write and supportsCodable
objects too. DelegateProvider
is just a forwarder object you can use to attach your own implementation without creating aFlagsProvider
conforming object.
Then, the following remote providers (you can fetch from a separate package):
FirebaseRemoteProvider
supports Firebase Remote Config feature flag service. It is a read-only provider and supports only primitive types (no JSON is supported but you can use your own Strings).
FirebaseRemoteProvider
is the data provider for the Firebase Remote Config service.
You can use it by fetching the additional RealFlagsFirebase
package from SwiftPM or CocoaPods (it will dependeds by Firebase iOS SDK).
Once fetched, remember to configure your Firebase SDK before load the provider.
This is just an example:
import UIKit
import Firebase
@main
class AppDelegate: UIResponder, UIApplicationDelegate {
func application(_ application: UIApplication, didFinishLaunchingWithOptions launchOptions: [UIApplication.LaunchOptionsKey: Any]?) -> Bool {
// Configure Firebase app
FirebaseApp.configure()
// Now you can use
let firebaseProvider = FirebaseRemoteProvider()
firebaseProvider.delegate = self
let manager = FlagsManager(providers: [firebaseProvider])
return true
}
}
You can also use the delegate
to receive important events from the SDK:
extension AppDelegate: FirebaseRemoteConfigProviderDelegate {
func firebaseProvider(_ remote: FirebaseRemoteProvider,
didFetchData: RemoteConfigFetchAndActivateStatus,
error: Error?)
// Data is fetched from SDK
}
}
While you generally don't need to modify a feature flag's value at runtime, sometimes you may need this feature.
Each @Flag
annotated property supports the setValue()
function to alter its value; this is allowed by accessing to the projected value:
appFlags.$ratingMode.setValue("altered value",
providers: [LocalProvider.self])
In addition to the new value it also takes the list of data providers to affect.
Obivously, not all data provider supports writing new values; for example you can't send values to the Firebase Remote Config via FirebaseRemoteProvider
.
However you can store a new value in a LocalProvider
instance.
The method return a boolean indicating the result of the operation.
In this example we're assign a new JSON value to a property:
// JSON type is represented via JSONData object which encapsulate a dictionary.
let json = JSONData(["column": 5, "showDesc": true, "title": true])
appFlags.ui.$layoutAttributes.setValue(json, providers: [LocalProvider.self])
Another way to update a flag's value is by set it using the FlagsProvider
instance itself:
let json = JSONData(["column": 5, "showDesc": true, "title": true])
appFlagsLocalProvider.setValue(json, forFlag: appFlags.ui.$layoutAttributes.keyPath)
Every FlagsProvider
conforming type expose an optional Combine Publisher you can subscribe to to listen for flag value updates.
This Publisher is returned by the following method:
func didUpdateValueForKey() -> AnyPublisher<(FlagKeyPath, (any FlagProtocol)?), Never>?
Currently, only LocalProvider
objects return a Publisher.
Usage example:
var cancellables: Set<AnyCancellable> = []
appFlagsLocalProvider.didUpdateValueForKey()?
.sink { (keyPath, value) in
print("New value for flag \(keyPath.fullPath): \(value?.encoded())")
}
.store(in: &cancellables)
Flags Browser is a small UI tool for displaying and manipulating feature flags.
It's part of the library and you can include it in your product, like inside the developer's mode or similar.
It's really easy to use, just allocate and push our built-in view controller by passing a list of FlagsLoader
or a FlagsManager
; RealFlags takes care to read all the values and show them in a confortable user interface.
If you are using FlagsManager
to keep your flags organized, just pass it to the init:
var flagsManager = FlagsManager(providers: [...])
func showFlagsBrowser() {
flagsBrowser = FlagsBrowserController.create(manager: flagsManager)
present(flagsBrowser, animated: true, completion: nil)
}
Otherwise, you can pass one or more FlagsLoader
that you want to show:
var userFlags: FlagsLoader<UserFlagsCollection> = ...
var expFlags: FlagsLoader<ExperimentalFlagsCollection> = ...
func showFlagsBrowser() {
flagsBrowser = FlagsBrowserController.create(loaders: [userFlags, expFlags])
present(flagsBrowser, animated: true, completion: nil)
}
And you're done! You will get a cool UI with detailed description of each flags along its types and values for each data provider, which allows developers and product owners to check and alter the state of the app!
The following properties of @Flag
properties are used by the Flags Browser to render the UI:
name
: provide the readable name of the property. If not specified, the signature of the property is used automatically.description
: provide a short description of the property which is useful both for devs and product owners.keyPath
: provide the full keypath of the property which is queried to any set data providers. If not specified it will be evaluated automatically (see here).isUILocked
: set it totrue
in order to prevent altering the property from the Flags Browser. By default isfalse
.defaultValue
: shows the fallback value when no value can be obtained from any data provider.metadata
: contains useful settings for flag (see below).
A Flag
instance also include a FlagMetadata
object (via .metadata
) which exposes several additional properties:
uiIcon
: an optional icon to show in Flags Browser. By default you can keep itnil
and the default datatype icon is used (default isnil
).isLocked
: whentrue
you cannot change the value of the flag via Flags Browser (default isfalse
).isInternal
: whentrue
value is not visible in Flags Browser (default isfalse
)