The civilized way to write REST API clients for iOS / macOS
The elegant way to write iOS / macOS REST clients
Drastically simplifies app code by providing a client-side cache of observable models for RESTful resources.
swift-*
branches for legacy support)Want your app to talk to a remote API? Welcome to your state nightmare!
You need to display response data whenever it arrives. Unless the requesting screen is no longer visible. Unless some other currently visible bit of UI happens to need the same data. Or is about to need it.
You should show a loading indicator (but watch out for race conditions that leave it stuck spinning forever), display user-friendly errors (but not redundantly — no modal alert dogpiles!), give users a retry mechanism … and hide all of that when a subsequent request succeeds.
Be sure to avoid redundant requests — and redundant response deserialization. Deserialization should be on a background thread, of course. Oh, and remember not to retain your ViewController / model / helper thingy by accident in your callback closures. Unless you’re supposed to.
Naturally you’ll want to rewrite all of this from scratch in a slightly different ad hoc way for every project you create.
What could possibly go wrong?
Siesta ends this headache by providing a resource-centric alternative to the familiar request-centric approach.
Siesta provides an app-wide observable model of a RESTful resource’s state. This model answers three basic questions:
…and broadcasts notifications whenever the answers to these questions change.
Siesta handles all the transitions and corner cases to deliver these answers wrapped up with a pretty bow on top, letting you focus on your logic and UI.
URLSession
by default, or Alamofire, or inject your own custom adapter).This project started as helper code we wrote out of practical need on several Bust Out Solutions projects. When we found ourselves copying the code between projects, we knew it was time to open source it.
For the open source transition, we took the time to rewrite our code in Swift — and rethink it in Swift, embracing the language to make the API as clean as the concepts.
Siesta’s code is therefore both old and new: battle-tested on the App Store, then reincarnated in a Swifty green field.
Make the default thing the right thing most of the time.
Make the right thing easy all of the time.
Build from need. Don’t invent solutions in search of problems.
Design the API with these goals:
…in that order of priority.
Siesta requires Swift 5.3+ and Xcode 12+. (Use the swift-*
branches branches if you are still on an older version.)
In Xcode:
https://github.com/bustoutsolutions/siesta
in the URL field and click Next.Please note that Xcode will show all of Siesta’s optional and test-only dependencies, including Quick, Nimble, and Alamofire. Don’t worry: these won’t actually be bundled into your app (except Alamofire, if you use it).
In your Podfile
:
pod 'Siesta', '~> 1.0'
If you want to use the UI helpers:
pod 'Siesta/UI', '~> 1.0'
If you want to use Alamofire as your networking provider instead of Foundation’s URLSession
:
pod 'Siesta/Alamofire', '~> 1.0'
(You’ll also need to pass an Alamofire.Manager
when you configure your Siesta.Service
. See the API docs for more info.)
In your Cartfile
:
github "bustoutsolutions/siesta" ~> 1.0
Follow the Carthage instructions to add Siesta.framework
to your project. If you want to use the UI helpers, you will also need to add SiestaUI.framework
to your project as well.
As of this writing, there is one additional step you need to follow that isn’t in the Carthage docs:
$(PROJECT_DIR)/Carthage/Build/iOS/
(In-depth discussion of Carthage in recent Xcode versions is here.)
The code in Extensions/
is not part of the Siesta.framework
that Carthage builds. (This includes optional integrations for other libraries, such as Alamofire.) You will need to include those source files in your project manually if you want to use them.
Clone Siesta as a submodule into the directory of your choice, in this case Libraries/Siesta:
git submodule add https://github.com/bustoutsolutions/siesta.git Libraries/Siesta
git submodule update --init
Drag Siesta.xcodeproj
into your project tree as a subproject.
Under your project’s Build Phases, expand Target Dependencies. Click the + button and add Siesta.
Expand the Link Binary With Libraries phase. Click the + button and add Siesta.
Click the + button in the top left corner to add a Copy Files build phase. Set the directory to Frameworks. Click the + button and add Siesta.
If you want to use the UI helpers, you will need to repeat steps 3–5 for SiestaUI
.
Please let us know about it, even if you eventually figure it out. Knowing where people get stuck will help improve these instructions!
Make a shared service instance for the REST API you want to use:
let MyAPI = Service(baseURL: "https://api.example.com")
Now register your view controller — or view, internal glue class, reactive signal/sequence, anything you like — to receive notifications whenever a particular resource’s state changes:
override func viewDidLoad() {
super.viewDidLoad()
MyAPI.resource("/profile").addObserver(self)
}
Use those notifications to populate your UI:
func resourceChanged(_ resource: Resource, event: ResourceEvent) {
nameLabel.text = resource.jsonDict["name"] as? String
colorLabel.text = resource.jsonDict["favoriteColor"] as? String
errorLabel.text = resource.latestError?.userMessage
}
Or if you don’t like delegates, Siesta supports closure observers:
MyAPI.resource("/profile").addObserver(owner: self) {
[weak self] resource, _ in
self?.nameLabel.text = resource.jsonDict["name"] as? String
self?.colorLabel.text = resource.jsonDict["favoriteColor"] as? String
self?.errorLabel.text = resource.latestError?.userMessage
}
Note that no actual JSON parsing occurs when we invoke jsonDict
. The JSON has already been parsed off the main thread, in a GCD queue — and unlike other frameworks, it is only parsed once no matter how many observers there are.
Of course, you probably don’t want to work with raw JSON in all your controllers. You can configure Siesta to automatically turn raw responses into models:
MyAPI.configureTransformer("/profile") { // Path supports wildcards
UserProfile(json: $0.content) // Create models however you like
}
…and now your observers see models instead of JSON:
MyAPI.resource("/profile").addObserver(owner: self) {
[weak self] resource, _ in
self?.showProfile(resource.typedContent()) // Response now contains UserProfile instead of JSON
}
func showProfile(profile: UserProfile?) {
...
}
Trigger a staleness-aware, redundant-request-suppressing load when the view appears:
override func viewWillAppear(_ animated: Bool) {
MyAPI.resource("/profile").loadIfNeeded()
}
…and you have a networked UI.
Add a loading indicator:
MyAPI.resource("/profile").addObserver(owner: self) {
[weak self] resource, event in
self?.activityIndicator.isHidden = !resource.isLoading
}
…or better yet, use Siesta’s prebaked ResourceStatusOverlay
view to get an activity indicator, a nicely formatted error message, and a retry button for free:
class ProfileViewController: UIViewController, ResourceObserver {
@IBOutlet weak var nameLabel, colorLabel: UILabel!
@IBOutlet weak var statusOverlay: ResourceStatusOverlay!
override func viewDidLoad() {
super.viewDidLoad()
MyAPI.resource("/profile")
.addObserver(self)
.addObserver(statusOverlay)
}
override func viewWillAppear(_ animated: Bool) {
super.viewWillAppear(animated)
MyAPI.resource("/profile").loadIfNeeded()
}
func resourceChanged(_ resource: Resource, event: ResourceEvent) {
nameLabel.text = resource.jsonDict["name"] as? String
colorLabel.text = resource.jsonDict["favoriteColor"] as? String
}
}
Note that this example is not toy code. Together with its storyboard, this small class is a fully armed and operational REST-backed user interface.
Take a look at AFNetworking’s venerable UIImageView
extension for asynchronously loading and caching remote images on demand. Seriously, go skim that code and digest all the cool things it does. Take a few minutes. I’ll wait. I’m a README. I’m not going anywhere.
Got it? Good.
Here’s how you implement the same functionality using Siesta:
class RemoteImageView: UIImageView {
static var imageCache: Service = Service()
var placeholderImage: UIImage?
var imageURL: URL? {
get { return imageResource?.url }
set { imageResource = RemoteImageView.imageCache.resource(absoluteURL: newValue) }
}
var imageResource: Resource? {
willSet {
imageResource?.removeObservers(ownedBy: self)
imageResource?.cancelLoadIfUnobserved(afterDelay: 0.05)
}
didSet {
imageResource?.loadIfNeeded()
imageResource?.addObserver(owner: self) { [weak self] _,_ in
self?.image = self?.imageResource?.typedContent(
ifNone: self?.placeholderImage)
}
}
}
}
A thumbnail of both versions, for your code comparing pleasure:
The same functionality. Yes, really.
(Well, OK, they’re not exactly identical. The Siesta version has more robust caching behavior, and will automatically update an image everywhere it is displayed if it’s refreshed.)
There’s a more featureful version of RemoteImageView
already included with Siesta — but the UI freebies aren’t the point. “Less code” isn’t even the point. The point is that Siesta gives you an elegant abstraction that solves problems you actually have, making your code simpler and less brittle.
Popular REST / networking frameworks have different primary goals:
Which one is right for your project? It depends on your needs and your tastes.
Siesta has robust functionality, but does not attempt to solve everything. In particular, Moya and RestKit address complementary / alternative concerns, while Alamofire and AFNetworking provide more robust low-level HTTP support. Further complicating a comparison, some frameworks are built on top of others. When you use Moya, for example, you’re also signing up for Alamofire. Siesta uses URLSession by default, but can also stack on top of Alamofire if you want to use its SSL trust management features. Combinations abound.
With all that in mind, here is a capabilities comparison¹:
Siesta | Alamofire | RestKit | Moya | AFNetworking | URLSession | |
---|---|---|---|---|---|---|
HTTP requests | ✓ | ✓ | ✓ | ✓ | ✓ | ✓ |
Async response callbacks | ✓ | ✓ | ✓ | ✓ | ✓ | ✓ |
Observable in-memory cache | ✓ | |||||
Prevents redundant requests | ✓ | |||||
Prevents redundant parsing | ✓ | |||||
Parsing for common formats | ✓ | ✓ | ✓ | |||
Route-based parsing | ✓ | ✓ | ||||
Content-type-based parsing | ✓ | |||||
File upload/download tasks | ✓ | ~ | ✓ | ✓ | ||
Object model mapping | ✓ | |||||
Core data integration | ✓ | |||||
Hides HTTP | ✓ | |||||
UI helpers | ✓ | ✓ | ||||
Primary language | Swift | Swift | Obj-C | Swift | Obj-C | Obj-C |
Nontrivial lines of code² | 2609 | 3980 | 13220 | 1178 | 3936 | ? |
Built on top of | any (injectable) | URLSession | AFNetworking | Alamofire | NSURLSession / NSURLConnection | Apple guts |
1. Disclaimer: table compiled by Siesta’s non-omniscient author. Corrections / additions? Please submit a PR.
2. “Trivial” means lines containing only whitespace, comments, parens, semicolons, and braces.
Despite this capabilities list, Siesta is a relatively lean codebase — smaller than Alamofire, and 5.5x lighter than RestKit.
It’s not just the features. Siesta solves a different problem than other REST frameworks.
Other frameworks essentially view HTTP as a form of RPC. New information arrives only in responses that are coupled to requests — the return values of asynchronous functions.
Siesta puts the the “ST” back in “REST”, embracing the notion of state transfer as an architectural principle, and decoupling the act of observing state from the act of transferring it.
If that approach sounds appealing, give Siesta a try.
This repo includes a simple example project. To download the example project, install its dependencies, and run it locally:
pod try Siesta
(Note that there’s no need to download/clone Siesta locally first; this command does that for you.)To ask for help, please post a question on Stack Overflow and tag it with siesta-swift
. (Be sure to include that tag. It triggers a notification to the Siesta core team.) This is preferable to filing an issue because other people may have the same question as you, and Stack Overflow answers are more discoverable than closed issues.
Things that belong on Stack Overflow:
For a bug, feature request, or cool idea, please file a Github issue. Things that belong in Github issues:
Unsure which to choose? If you’re proposing a change to Siesta, use Github issues. If you’re asking a question that doesn’t change the project, and thus will remain valid even after you get an answer, then use Stack Overflow.
Keep in mind that Siesta is maintained by volunteers. Please be patient if you don’t immediately get an answer to your question; we all have jobs, families, obligations, and lives beyond this project.
Please be excellent to one another and follow our code of conduct.