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.

  • OS: iOS 8+, macOS/OS X 10.11+
  • Languages: Written in Swift, supports apps in both Swift and Objective-C
  • Tool requirements: Xcode 7, Swift 2.0
  • License: MIT
  • Status: Solid code, already in use on the App Store, but still classified as “beta” so we can gather feedback before locking in the API for the official 1.0 release. Please kick the tires, file issues, and send pull requests. Be bold!

Contents

  • What ‘s It For?
  • Features
  • Origin
  • Design Philosophy
  • Installation
  • Basic Usage
  • Comparison With Other Frameworks
  • Documentation

    • User Guide
    • API Docs
    • Specs
  • Examples
  • Contributing and Getting Help

What ‘s It For?

The Problem

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), The display user-friendly Errors (but not redundantly 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?

The Solution

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:

  • What is the latest data for this resource, if any?
  • Did the latest request result in an error?
  • Is there a request in progress?

… 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.

Features

  • Decouples view, model, and controller lifecycle from network request lifecycle
  • Decouples request initiation from request configuration
  • Eliminates error-prone state tracking logic
  • Eliminates redundant network requests
  • Unified handling for all errors: encoding, network, server-side, and parsing
  • Highly extensible, multithreaded response deserialization
  • Transparent built-in parsing (which you can turn off) for JSON, text, and images
  • Smooth progress reporting that accounts for upload, download, and latency
  • Transparent Etag / If-Modified-Since handling
  • Prebaked UI helpers for loading & error handling, remote images
  • Debug-friendly, customizable logging
  • Written in Swift with a great Swift-centric API, but…
  • … also works great from Objective-C thanks to a compatibility layer.
  • Lightweight. Won’t achieve sentience and attempt to destroy you.
  • Robust regression tests
  • Documentation and more documentation

What it doesn ‘t do

  • It Doesn ‘t reinvent networking. Siesta delegates network operations to your library of choice (NSURLSession by default, or Alamofire, or inject your own custom adapter).
  • It doesn’t hide HTTP. On the contrary, Siesta strives to expose the full richness of HTTP while providing conveniences to simplify common usage patterns. You can devise an abstraction layer to suit your own particular needs, or work directly with Siesta’s nice APIs for requests and response entities.
  • It doesn’t do automatic response to model mapping. This means that Siesta doesn’t constrain your response models, or force you to have any at all. Add a response transformer to output models of whatever flavor you prefer, or work directly with parsed JSON.

Origin

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-tasted on the App Store, then reincarnated in a Swifty green field.

Design Philosophy

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:

  1. Make client code easy to read.
  2. Make client code easy to write.
  3. Keep the API clean.
  4. Keep the implementation tidy.

… in that order of priority.


Installation

Siesta requires Swift 2.0, so make sure you have Xcode 7.

CocoaPods

In your Podfile:

Pod 'Siesta', '> = 1.0 beta. 7'Copy the code

If you want to use Alamofire as your networking provider instead of NSURLSession:

pod 'Siesta/Alamofire'
Copy the code

You’ll also need to pass an alamofire. Manager when You configure your siesta.service. See the API docs for more info.

Carthage

In your Cartfile:

Making "bustoutsolutions/siesta" "1.0 beta. 7"Copy the code

Follow the Carthage instructions to add Siesta.framework to your project.

As of this writing, there is one additional step you need to follow for Xcode 7 that isn’t in the Carthage docs:

  • Build settings → Framework search paths → $(PROJECT_DIR)/Carthage/Build/iOS/

(In-depth discussion of Carthage on XC7 is here.)

The code in Extensions/ is not part of the Siesta.framework that Carthage builds. (This currently includes only Alamofire support.) You will need to include those source files in your project manually if you want to use them.

Git Submodule

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 --initCopy the code

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.


Basic Usage

Make a shared service instance for the REST API you want to use:

let MyAPI = Service(baseURL: "https://api.example.com")Copy the code

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)
}Copy the code

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 }Copy the code

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 }Copy the code

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
}Copy the code

… 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?) {
    ...
}Copy the code

Trigger a staleness-aware, redundant-request-suppressing load when the view appears:

override func viewWillAppear(animated: Bool) {
    MyAPI.resource("/profile").loadIfNeeded()
}Copy the code

… and you have a networked UI.

Add a loading indicator:

MyAPI.resource("/profile").addObserver(owner: self) { [weak self] resource, event in self? .activityIndicator.hidden = ! resource.isLoading }Copy the code

… 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
    }
}Copy the code

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.

Your socks still on?

Take a look at AFNetworking’s venerable UIImageView extension for asynchronously loading and caching remote images on demand. Seriously, 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: NSURL? { get { return imageResource? .url } set { imageResource = RemoteImageView.imageCache.resource(url: 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 = imageResource? .contentAsType(ifNone: placeholderImage) } } } }Copy the code

A thumbnail of both versions, for your code comparing pleasure:

The same functionality. Yes, really.

(Well, OK, they ‘re notexactly 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 “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.

Comparison With Other Frameworks

Popular REST / networking frameworks have different primary goals:

  • NSURLSession is Apple’s standard iOS HTTP library (and is all most projects need).
  • Siesta untangles state problems with an observable resource cache.
  • Alamofire provides a Swifty, modern-feeling wrapper for NSURLSession.
  • Moya wraps Alamofire to hide HTTP URLs and parameters.
  • RestKit couples HTTP with JSON ↔ object model Core Data mapping.
  • AFNetworking is a modern-feeling Obj-C wrapper for Apple’s network APIs, plus a suite of related utilities.

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.

With all that in mind, here is a capabilities comparison

Siesta Alamofire RestKit Moya AFNetworking NSURLSession
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
Nontrivial lines of code squared 2069 1943 10651 639 4029
Built on top of any (injectable) NSURLSession AFNetworking Alamofire NSURLSession / NSURLConnection

1. Disclaimer: Table compiled by Siesta’s non-omniscient author. Corrections/additions? 2. “Trivial” means lines containing only whitespace, comments, parens, semicolons, and braces.

Despite this capabilities list, Siesta is a relatively small codebase — about the same size as Alamofire, And 5.5x smaller than RestKit.

What sets Siesta apart?

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 “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.


Documentation

  • User Guide
  • API documentation
  • Specs

Examples

This repo includes a simple example project. Use Carthage to build its dependencies.

Contributing and Getting Help

To report a bug, file an issue.

To point out anything incorrect or confusing in the documentation, file an issue.

To submit a feature request / cool idea, file an issue.

To get help, post your question to Stack Overflow and tag it with siesta-swift. (Be sure to include the tag. It triggers a notification.)

Pull Requests

Want to do something instead of just talking about it? Fantastic! Be bold.

  • If you have trouble building or testing the project, Please file an issue about it — even if you resolve the problem. This will help improve the docs.
  • If you are proposing a design change or nontrivial new functionality, please float your idea as an issue first so you don’t end up doing wasted work.
  • Please follow the formatting conventions of the existing code. Yes, including Paul’s idiosyncratic taste in brace placement.
  • Please make sure the tests pass locally before pushing (cmd-U in Xcode).
  • Alice: Don’t be discouraged. Alice: Nit-picking is not a sign of bad work; It’s a sign of interest!
  • If you want to update the user guide, note that the guide is generated from the Docs directory in master, so that’s where you should make your change. Do not submit pull requests against gh-pages.