Swift: UserDefaults protocol

Swift 3 brought a tsunami of changes to the language as well as our codebase, some of you reading this may even still be battling with the migration too. But even with all these changes, we’re still left with some APIs within Foundation that are stringly typed, which is totally fine… Until it’s not.
It’s kind of a love/hate relationship in that we love the flexability that strings within APIs afford us, but we hate that we have to use them because of the inherit consequences they bring if we’re not careful, they’re pretty much the programming equivalent of running with scissors.
We’re given stringly typed APIs because the Foundation framework engineering gods could not predetermine exactly how we intend to use them. So in all their wisdom, power and knowledge, they decided to use strings in some of the APIs because of the unlimited possibilities it creates for us as developers. It’s either that or some type of dark arcane magic.

UserDefaults

Our topic for today is going to be one of the first APIs I became familiar with when learning iOS development . For those that aren’t familiar, it’s simply a persistant data storage for small sets of information, such as a single image or some application settings. Some people like to think of it is as some kind of “Diet Core Data”, however it’s nowhere near as robust, no matter how hard people try to wedge it in as a replacement.

Stringly typed API

UserDefaults.standard.set(true, forKey: “isUserLoggedIn”)
UserDefaults.standard.bool(forKey: "isUserLoggedIn")
A common practice for UserDefaults usage inside an app, it simply allows us to set a value persistantly and retrieve, override or remove it from wherever in our app. But if we’re not careful, we can trip ourselves up with lack of uniformity or context, but most likely typos. In this post, we’re going to transform the generalised nature of UserDefaults and customise it to our own needs.

Use constants

let key = "isUserLoggedIn"
UserDefaults.standard.set(true, forKey: key)
UserDefaults.standard.bool(forKey: key)
I guarantee you will instantly write better code if you follow this one weird trick. If you write a string more than once, convert it into a constant and live by this rule until the end of your days, feel free to thank me in the next life.

Group constants

struct Constants {
    let isUserLoggedIn = "isUserLoggedIn"
}
...
UserDefaults.standard
   .set(true, forKey: Constants().isUserLoggedIn)
UserDefaults.standard
   .bool(forKey: Constants().isUserLoggedIn)
One pattern than can help us maintain uniformity is when group all our important default constants in a single place. So here we’ve made a Constants struct that we can store and reference our defaults from.
Another good tip is to aim to have your property name reflect its own value, especially when working with defaults. Doing this will simplify your code and overall attribute to better uniformity. Copying property names and pasting them inside strings will help you avoid typos.
let isUserLoggedIn = "isUserLoggedIn"

Add Context

struct Constants {
    struct Account
        let isUserLoggedIn = "isUserLoggedIn"
    }
}
...
UserDefaults.standard
   .set(true, forKey: Constants.Account().isUserLoggedIn)
UserDefaults.standard
   .bool(forKey: Constants.Account().isUserLoggedIn)
Just having a Constants struct is totally fine, but we shouldn’t forget about providing context when writing code. A good practice to aim towards is to make your code read more readable to whomever is working with it, including yourself.
Constants().token // Huh?
What does token mean? When someone is trying to distinguish what tokenrelates to, the lack of namespacing context really makes it difficult to someone that is new or unfamiliar to the codebase, or the original author one year into the future.
Constants.Authentication().token // better

Avoid initialization

struct Constants {
    struct Account
        let isUserLoggedIn = "isUserLoggedIn"
    }
    private init() { }
}
Because we never intend, nor want our Constants struct to be initialised, we declare our initializer to be private. This is more of a precautionary step, but I would still recommend doing it. At the very least it will prevent us from accidently declaring instance properties when we only want static. But speaking of static…

Static variables

struct Constants {
    struct Account
        static let isUserLoggedIn = "isUserLoggedIn"
    }
    ...
}
...
UserDefaults.standard
   .set(true, forKey: Constants.Account.isUserLoggedIn)
UserDefaults.standard
   .bool(forKey: Constants.Account.isUserLoggedIn)
You may have noticed that each time we wanted to access the key, we would have to initialize the struct of which it belonged to. Instead of having to do that each time, we should use the static variable declaration.
We use static instead of class because we’ve chosen a struct as our storage type. According to Swift-compiler law, structs cannot use the class property declaration. Also, if you use a static declaration on a property in a class, it is the same as declaring that property final class.
final class name: String
static name: String
// final class == static

Fewer typos with enum cases

enum Constants {
    enum Account : String {
        case isUserLoggedIn
    }
    ...
}
...
UserDefaults.standard
    .set(true, forKey: Constants.Keys.isUserLoggedIn.rawValue)
UserDefaults.standard
    .bool(forKey: Constants.Keys.isUserLoggedIn.rawValue)
Earlier in this post, we talked about making our properties reflect their values for uniformity. Here we’re going to go a step further by automating the process using and enum case instead of static let.
As you may have noticed, we’ve made our Account enum conform to String, which also conforms to RawRepresentable protocol. We’re doing this because if we don’t provide a rawValue for the case, it will default to a reflection of the case. The less manual typing or copy/pasting of strings we have to do, the better.
// Constants.Account.isUserLoggedIn.rawValue == "isUserLoggedIn"
Up until now we’ve managed to do some pretty cool things with UserDefaults, but I would argue that we haven’t done enough awesomethings. The biggest problem is that we’re still working with stringly typed APIs, even though we’ve dressed that string up, it’s still not good enough for our projects.
We’re in this mentality that we can only work with what’s given to us. Swift is an insanely great language and we should be challenging a lot of what we’ve learnt and know from our days writing Objective-C. Let’s go back to the kitchen and sprinkle some syntax sugar on these APIs.

API Goals

UserDefaults.standard.set(true, forKey: .isUserLoggedIn) 
// #APIGoals
For the rest of this talk, we’re going to strive to create better APIs to work with when we’re interacting with UserDefaults that suit our needs instead of the general populus, and what better way than to do so than making extensions with protocols.

BoolUserDefaultable

protocol BoolUserDefaultable {
    associatedType BoolDefaultKey : RawRepresentable
}
Let’s begin with making protocols for boolean UserDefaults, a simple protocol with out any variables or conforming functions. However, we do provide an associatedType named BoolDefaultKey which conforms to RawRepresentable, You’ll understand why this is next.

Extension

extension BoolUserDefaultable 
    where BoolDefaultKey.RawValue == String { ... }
If we plan to abide by Crusty’s Laws of Protocols, we would declare a protocol extension. But we’re also applying a where clause that constraints the extension to only apply if the associatedType’s RawValue was of type String.
With every protocol, there is an equal and corresponding protocol extension — Crusty’s Third Law

UserDefault Setter

// BoolUserDefaultable extension
static func set(_ value: Bool, forKey key: BoolDefaultKey) {
    let key = key.rawValue
    UserDefaults.standard.set(value, forKey: key)
}
static func bool(forKey key: BoolDefaultKey) -> Bool {
    let key = key.rawValue
    return UserDefaults.standard.bool(forKey: key)
}
Yep, that’s a simple API wrapper around standard UserDefaults API. We’re doing this because it’s much more readable to pass in a simple enum caseinstead of a string which is key-pathed.
UserDefaults.set(false, 
    forKey: Aint.Nobody.Got.Time.For.this.rawValue)

Conformance

extension UserDefaults : BoolUserDefaultSettable {
    enum BoolDefaultKey : String {
        case isUserLoggedIn
    }
}
Yep, we’re going to extend UserDefaults to with a conformity to BoolDefaultSettable and provide an associated type named BoolDefaultKey which conforms to RawRepresentable (String)
// Setter
UserDefaults.set(true, forKey: .isUserLoggedIn)
// Getter
UserDefaults.bool(forKey: .isUserLoggedIn)
Again we’re challening the norms of working with provided APIs and instead defining our own. This is because when we extended UserDefaults, we lost context with our own API. If this was any key other than .isUserLoggedIn, would we understand what it related to?
UserDefaults.set(true, forKey: .isAccepted) 
// Huh? isAccepted for what?
That key is so ambiguous it can mean a whole range of things. Providing context is always beneficial, even when it doesn’t seem like it.
It’s better to have it and not need it, than it is to need it and not have it.
Not to worry though, adding context is simple enough. We simply create a namespace for the key. In this case, we created an Account namespace which houses the isUserLoggedIn key.
struct Account : BoolUserDefaultSettable {
    enum BoolDefaultKey : String {
        case isUserLoggedIn
    }
    ...
}
...
Account.set(true, forKey: .isUserLoggedIn)

Collisions

ley account = Account.BoolDefaultKey.isUserLoggedIn.rawValue
let default = UserDefaults.BoolDefaultKey.isUserLoggedIn.rawValue
// account == default
// "isUserLoggedIn" == "isUserLoggedIn"
Having two separate types conforming to the same protocol and providing the same key case is definitely a possibility, as programmers this will definitely keep us up at night if we don’t solve it before we ship. We cannot take the risk of having one key changing the value of another. So instead we should be namespacing our keys.

Namespacing

protocol KeyNamespaceable { }
Of course we make a protocol for this, we’re Swift developers. Protocols are usually our first attempt at solving each problem we face. If protocols were chocolate sauce, we’d put it on everything, even steak. That’s how much we love making protocols.
extension KeyNamespaceable { 
  func namespace<T>(_ key: T) -> String where T: RawRepresentable {
        return "\(Self.self).\(key.rawValue)"
  }
}
A simple function that does some string interpolation that combines two objects and separates them with a full stop; the name of the class, and the rawValue of they key. We also relied on generics to allow our function take in a generic type for the key arguement if it conforms to RawRepresentable.
protocol BoolUserDefaultSettable : KeyNamespaceable
After creating our namespacing protocol, we revisit our earlier BoolUserDefaultSettable protocol and make it conform to KeyNamespaceable and modify the original extension to take advantage of the functionality.
// BoolUserDefaultable extension
static func set(_ value: Bool, forKey key: BoolDefaultKey) {
    let key = namespace(key)
    UserDefaults.standard.set(value, forKey: key)
}
static func bool(forKey key: BoolDefaultKey) -> Bool {
    let key = namespace(key)
    return UserDefaults.standard.bool(forKey: key)
}
...
ley account = namespace(Account.BoolDefaultKey.isUserLoggedIn)
let default = namespace(UserDefaults.BoolDefaultKey.isUserLoggedIn)
// account != default
// "Account.isUserLoggedIn" != "UserDefaults.isUserLoggedIn"

Context

Because we created this protocol, we felt liberated from using the UserDefaults API and became perhaps became a little too intoxicated with protocol power. In doing that, we created context for our keys by moving them into housing that made sense when we were reading code:
Account.set(true, forKey: .isUserLoggedIn)
But we also lost context because the API no longer makes complete sense. At first glance, there is nothing about this code that informs me that this boolean is going to be stored persistently, or even inside UserDefaults. So to bring everything full circle, we’re going to extend UserDefaults and place our default types within:
extension UserDefaults {
    struct Account : BoolUserDefaultSettable { ... }
}
...
UserDefaults.Account.set(true, forKey: .isUserLoggedIn)
UserDefaults.Account.bool(forKey: .isUserLoggedIn)

Comments

Popular posts from this blog

Download .dmg file of Xcode

How to setup Xcode Swift Project to use LLVM C APIs