Skip to content

Latest commit

 

History

History
97 lines (70 loc) · 5.06 KB

README.md

File metadata and controls

97 lines (70 loc) · 5.06 KB

Clean Architecture for iOS

This is an example of how to architect an iOS application by decoupling completely the UI and the business logic. It's based on the Clean Architecture idea that Uncle Bob has been mentioning for a while. Implementation specific details for iOS are mine.

This example uses UIKit for the UI but it should be easy to replace this UI with a CLI or any other delivery mechanism.

Basic concepts

Entities

You should be familiar with this type of objects. An entity represents a domain-specific concept. Department, Employee, Sale, Return, Flight, etc.

Implementation specific code

Not domain-related code that is required in order to make the system work as expected. Modules that abstract persistence, networking, etc will fall in this category.

Interactors

An interactor represents a Use Case of the app. It is responsible for:

  • Managing the interaction between multiple Entities
  • Keep the application state, given that iOS apps are not service oriented.
  • Define a strict boundary around the business logic.
  • Deal with other implementation specific details like calling the persistence layer, networking to third-party systems, etc.
  • The public API of an interactor is tied to the main thread. Requests are made from the main thread and responses are returned to the main thread.

Delivery mechanism

The specific implementation that the user will use to perform actions on the system and to communicate the state of the system back to the user. Call it GUI, Web, JSON API, CLI, multitouch screen, Rails, etc. Delivery mechanisms talk to interactors via the Boundaries.

Boundaries

These boundaries represent some request to the interactor and some response from it.

In this example, ViewControllers capture a request from the user via the UI, interpret what the user wants and request the interactor to perform that action via the designated boundary. The interactor will then return a response with the result of that action. The ViewController will unpack that response and present it back to the user.

We can find boundaries of multiple types in this example:

  • Method call to the interactor to query the application state.
    • A simple property getter in the interactor.
  • Method call to an interactor to mutate the application state.
    • A method call.
    • A property setter (yeah, method call).
    • A class representing a request that will be passed as an argument to a method of the interactor.

The interactor can respond in different ways, depending on the needs:

  • It can return void in a method call. You should use a query mechanism to find out the result, if you care at all.
  • It can return the result synchronously. (i.e. the getter of a property)
  • It can return synchronously a class representing the response, giving you access to the result of the request, status, and errors.

This last approach is especially interesting because you can include future-like properties in this response. For example, you request to the interactor to login a user, given the username and password coming from the UI using a method call.

LSLoginResponse *response  = [interactor loginWithUsername:username password:password];

This operation is asynchronous in nature because it makes an HTTP request to a backend. In order to avoid passing blocks as arguments for dealing with callbacks, the method call will return synchronously a response object. The asynchronous operation will start right away but you don't know when it will finish. This response object will give you different options:

Asynchronously

The most common way will be to pass a callback after the fact, to get notified of the completion

response.onCompletion = ^{
  // Process response
}

Same with errors

response.onFailure = ^{
  // Process errors
}
Future

You can get the result synchronously, potentially blocking the main thread. This is specially useful if you make the request and you are not going to use the response for a while. In that time the underlying asynchronous request will finish and you can avoid callbacks. If the request is not done when asking for the result, it will block until it gets the result. Another good use of this is for testing!

User *user = response.loggedInUser;

Same applies to errors;

User *user = response.error;
Query the state of the request
  • Check whether the request is done:
BOOL isDone = response.isDone
  • Check whether the request was successful:
BOOL isSuccessful = response.isSuccessful;

Service

A service is a process that runs for the lifetime of the app. Think of analytics frameworks, crash report tools, etc. You set them up at startup time and you forget about them. You create one subclass of LSService for each service and register an instance of each service at startup time. This way the code related to this specific things won't be scattered in your AppDelegate, each one will have its own place to leave.

A service will get notified for changed in the state of the app (most of the methods of the UIApplication Delegate protocol), in order to act consequently.