Post is a simple global messaging service. Students will review MVC principles and work with NSURLSession, JSON parsing, and closures to build an app that lists and submits posts to a global feed.
Post is a single view application, with the main view being a list of all posts from the global feed listed in reverse-chronological order. The user can add posts via an alert controller presented after tapping an Add (+) bar button.
Students who complete this project independently are able to:
- use NSURLSession to make asynchronous GET HTTP requests
- parse JSON data and generate model objects from youb requests
- use closures to execute code when an asyncronous task is complete
- use UIRefreshControl to reload data for a table view
- use NSURLSession to make asynchronous POST HTTP requests
- build custom table views that support paging through network requests
- use NSURLSession to make asynchronous GET HTTP requests
- parse JSON data and generate model objects from youb requests
- use closures to execute code when an asyncronous task is complete
- build custom table views that support paging through network requests
Build the network controller, post model object, post controller, and post list view controller. Add some polish to the view controller allow the user to reload posts, and know when network requests are happening. Focus on the network requests and serializing the data to display in the post list view controller.
Create a NetworkController class. This will have methods to build the different URLs you might want and it should have a method to return NSData from a URL.
The NetworkController will be responsible for building URLs and executing HTTP requests. Build the NetworkController to support different HTTP methods (GET, PUT, POST, PATCH, DELETE), and keep things generic such that you could use the same NetworkController in future projects if desired.
It is good practice to write reusable code. Even when you do not plan to reuse the class in future projects, it will help you keep the roles of your types properly separated. In this specific case, it is good practice for the NetworkController to not know anything about the project's model or controller types.
- Write a
StringtypedenumcalledHTTPMethod. You will use this enum to classify our HTTP requests as GET, PUT, POST, PATCH, or DELETE requests. Add cases for each.- example:
case Get = "GET"
- example:
- Write a function signature
performRequestForURLthat will take anNSURL, anHTTPMethod, an optional[String: String]dictionary of URL parameters, an optionalNSDatarequest body, and an optional completion closure. The completion closure should include aNSData?data parameter and anNSError?error parameter, and the should returnVoid.- note: At this point, it is OK if you do not understand why you are including each parameter. Spend some time contemplating each parameter and why you would include it in this function. For example: An HTTP request is made up of a URL, and an HTTP Method. Certain requests need URL parameters. Certain POST or PUT requests can carry a body. The completion closure is included so you know when the request is complete.
- Add the following
urlFromURLParametersfunction to yourNetworkControllerclass. This function takes a base URL, URL parameters, and returns a completed URL with the parameters in place.- example: To perform a Google Search, you use the URL
https://google.com/search?q=test. 'q' and 'test' are URL parameters, with 'q' being the name, and 'test' beging the value. This function will take the base URLhttps://google.com/searchand a[String: String]dictionary["q":"test"], and return the URLhttps://google.com/search?q=test
- example: To perform a Google Search, you use the URL
static func urlFromURLParameters(url: NSURL, urlParameters: [String: String]?) -> NSURL {
let components = NSURLComponents(URL: url, resolvingAgainstBaseURL: true)
components?.queryItems = urlParameters?.flatMap({NSURLQueryItem(name: $0.0, value: $0.1)})
if let url = components?.URL {
return url
} else {
fatalError("URL optional is nil")
}
}- Implement the
performRequestforURLfunction.- Use the
urlFromURLParametersto get a requestURL. - Creating a new
NSMutableURLRequest, set the HTTP method, set the body. - Generate and start the data task.
- Calling the completion when the data task completes.
- Use the
This method will make the network call and call the completion closer with the NSData? result. If successful, NSData? will contain the response, if unsuccessful, NSData? will be nil. The class or function that calls this function will need to handle nil data.
- Use a Playground to test your network controller method with a sample endpoint from the Post API to see if you are getting data returned.
As of iOS 9, Apple is boosting security and requiring developers to use the secure HTTPS protocol and requires the server to use the proper TLS Certificate version. The Post API does support HTTPS but does not use the correct TLS Certificate version. So for this app, you will need to turn off the App Transport Security feature.
- Open your
Info.plistfile and add a key-value pair to your Info.plist. This key-value pair should be:App Transport Security Settings : [Allow Arbitrary Loads : YES].
Create a Post model type that will hold the information of a post to display to the user.
Create a model object that will represent the Post objects that are listed in the feed. This model object will be generated locally, but also must be generated from JSON dictionaries.
- Create a
Post.swiftfile and define a newPoststruct. - Go to a sample endpoint of the Post API API and see what JSON (information) you will get back for each post.
- Using this information, add properties on
Post.let username: Stringlet text: Stringlet timestamp: NSTimeIntervallet identifier: NSUUID- note: UUID stands for Universal Unique Identifier, the
Foundationframework comes with anNSUUIDclass that helps generate and initializeNSUUIDs from strings. Review the Class Reference to review how to create a new UUID, extract the string value, and initalize anNSUUIDfrom aString.
- Create a memberwise initializer that takes parameters for the
usernameandtext. Optionally, add parameters for theidentifierandtimestamp, but set default values for them.- note: This memberwise initializer will only be used locally to generate new model objects. You can safely assume that initializing a new
Postmodel object will require a new UUID for theidentifier, and will use the.timeIntervalSince1970from the current date for thetimestamp.
- note: This memberwise initializer will only be used locally to generate new model objects. You can safely assume that initializing a new
- Create a failable initializer function with a parameter of a JSON dictionary (
[String: AnyObject]) and an identifier (String). This is the method you will use to initialize yourPostobject from a JSON dictionary. Remember to use a sample endpoint to inspect the JSON you will get back and the keys you will use to get each piece of data. Guard against required values, and returnnilif all required information for a post is not available.- note: The failable initializer is extremely important in this project. Bad users of the API could post improperly formatted JSON that could otherwise break your model objects or cause your application to crash. Make sure you always guard for the required information.
- note: Remember to avoid using magic strings. Create private keys to help you unwrap your JSON.
There is one more computed property you will add to the Post type called queryTimestamp, but you will discuss that in context in a step below.
Create a PostController class. This class will use the NetworkController to fetch data, and will serialize the results into Post objects. This class will be used by the view controllers to fetch Post objects through completion closures.
Because you will only use one View Controller in this project, there is no reason to make this controller a singleton or shared controller. To learn more about when singletons may not be the best tool, review this article on Singleton Abuse. The key takeaway for now is that singletons aren't always the right tool for the job, and you should carefully consider if is is the best pattern for accessing data in your project.
- Add a static constant
baseURLfor thePostControllerto know the base URL for the /posts/ subdirectory. This URL will be used to build other URLs throughout the app. - Add a static constant
endpointfor thePostControllerto know where to fetchPostobjects from. Use thebaseURLandurlByAppending...functions to build the correct endpoint for the API to fetch posts. This URL will be passed to ourNetworkControllerwhen you want to fetch posts. - Add a
postsproperty that will hold thePostobjects that you pull and serialize from the API. - Add a method
fetchPoststhat provides an array ofPostobjects through an optional completion closure.- This method should call the
NetworkControllerperformRequestforURLmethod to get the NSData at the endpoint URL. - In the closure of the
performRequestforURL, guard for data andpostDictionariesby serializing the JSON usingNSJsonSerialization. You will need to use thetry?keyword to useNSJSONSerializationto serialize theNSData. Note that the API returns a Dictionary with UUID Strings as the keys, and the [String: AnyObject] representation as the value for each key. If the guard fails, print an error message, run the optional completion, and return from the function.- note: The syntax for this function can be very difficult. Do your best to complete this without looking at the solution code.
- note: You must run your closure even if the data is unavailable, otherwise the object that calls this function will be waiting for a response forever.
- note: You can use
NSString(data: data!, encoding: NSUTF8StringEncoding)to capture and print a readable representation of the returned data.
- If the NSData can be serialized, initialize the
Postobjects and call the completion closure with the populated array. (Hint: Use a for loop or.flatMap()to iterate through the dictionaries and initialize a new array ofPostobjects.)- note: Because Dictionaries are unsorted, the
[Post]you generated in this step will not be in any logical order.
- note: Because Dictionaries are unsorted, the
- Use the
.sort()function to sort the posts by thetimestampproperty in reverse chronological order. - Set
self.poststo the sorted posts. - Unwrap the completion and run the closure with the sorted posts as the parameter.
- Because a primary use case for
fetchPostswill be to reload the user interface when the request is finished, use Grand Central Dispatch to force the completion closure to run on the main thread. (Hint:dispatch_async)
- This method should call the
There are many different patterns and techniques to serialize JSON data into Model objects. Feel free to experiment with different techniques to get at the [String: AnyObject] dictionaries within the NSData returned from the NSURLSessionDataTask.
At this point you should be able to pull the Post data from the API and serialize a list of Post objects. Test this functionality with a Playground or in your App Delegate by trying to print the results for a state to the console.
- Because you will always want to
fetchPosts()whenever you initialize thePostController, add aninit()function and callfetchPosts(). This will start the call to fetch posts and assign them to the.postsproperty. - Other classes that use the
PostControllerwill be interested to know whenever thepostsproperty is updated. Implement the delegate pattern to allow thePostControllerto notify an observer of updates to thepostsproperty.- Add a
PostControllerDelegateprotocol with apostsUpdatedfunction that with a parameter that will be used to pass an array of the posts. - Add an optional, youak
delegatevariable that conforms to thePostControllerDelegateprotocol. - Use the
didSetfunction on thepostsvariable to call thepostsUpdatedfunction on the delegate.
- Add a
Build a view that lists all posts. Implement dynamic height for the cells so that messages are not truncated. Include a Refresh Control that allows the user to 'pull to refresh' to load new, recent posts. Conform to the PostController delegate protocol to reload the tableview when the PostController serializes new posts from the API.
- Add a
UITableViewControlleras your root view controller in Main.storyboard and embed it into aUINavigationController - Create an
PostListTableViewControllerfile as a subclass ofUITableViewControllerand set the class of your root view controller scene - Add a
postControllerproperty and set it to an instance of thePostController - Implement the UITableViewDataSource functions using the included
postController.postsarray - Set the
cell.textLabelto the message, and thecell.detailTextLabelto the author and post date.- note: It may also help to temporarily add the
indexPath.rowto thecell.detailTextLabelto quickly determine if the posts are showing up where you expect them to be.
- note: It may also help to temporarily add the
Adopt the PostControllerDelegate protocol to observe the .posts property on the PostController to reload the tableview when there are new results.
- Adopt the
PostControllerDelegateprotocol - Implement the required
postsUpdatedfunction by reloading the table view
The length of the text on each Post is variable. Add support for dynamic resizing cells to your tableview so messages are not truncated.
- Set the
tableView.estimatedRowHeightin theviewDidLoad()function - Set the
tableView.rowHeighttoUITableViewAutomaticDimension - Update the
textLabelanddetailTextLabelon the Post List storyboard scene to support multiple lines by setting the.numberOfLinesto 0
Add a UIRefreshControl to the table view to support the 'pull to refresh' gesture.
- Add the
UIRefreshControlobject to the table view on the storyboard scene - Add an IBAction from the
UIRefreshControlin yourPostListTableViewControllerclass file - Implement the IBaction by telling the
postControllerto fetch new posts - Tell the
UIRefreshControlto end refreshing when thefetchPostsis complete.
It is good practice to let the user know that a network request is processing. This is most commonly done using the Network Activity Indicator in the status bar.
- Look up the documentation for
networkActivityIndicatorVisibleproperty onUIApplicationto turn on the indicator when when fetching new posts - Turn it off when the network call is complete
- Use a computed
.dateproperty,NSDateComponents andNSDateFormatters to display thePostdate in the correct time zone - Make your table view more efficient by inserting cells for new posts instead of reloading the entire tableview
- use NSURLSession to make asynchronous POST HTTP requests
- build custom table views that support paging through network requests
Build functionality to allow the user to submit new posts to the feed. Make the network requests more efficient by adding paging functionality to the Post Controller. Update the table view to support paging.
Note that you already built the NetworkController to support PUT requests. Review the code and NSURLSession to review how you wrote the code that supports both GET, PUT, and other HTTP requests.
Update the Post type with an endpoint and a jsonValue that will be used to send Post objects to the API.
- Add an
endpointcomputed property. Append the.identifierstring value and '.json' to thePostController.baseURLto return the URL for the HTTP request. - Add a
jsonValuecomputed property that returns a[String: AnyObject]representation of thePostobject.- note: Remember to use the correct keys. This will be the same Dictionary that you use in the failable initializer.
- Add a
jsonDatacomputed property as a convenient accessor that uses NSJSONSerialization to get anNSData?representation of thejsonValuedictionary.- note: This will be used when you set the HTTP Body on the
NSMutableURLRequest, which requires NSData?, not a [String: AnyObject]
- note: This will be used when you set the HTTP Body on the
Update your PostController to initialize a new Post and use the NetworkController to post it to the API.
- Add an
addPostfunction that takesStringusername andStringtext parameters. - Implement the
addPostfunction:- Initialize the
Postwith the memberwise initializer - Capture the request URL from the
Post's.endpointproperty. - Call the
NetworkControllerperformRequestforURLfunction. Passing the request URL, .PUT HTTP method, and thepost.jsonDatafor the body. - Check for errors. Check the included API documentation for details on catching errors from the Post API.
- note: You can use
NSString(data: data!, encoding: NSUTF8StringEncoding)to capture and print a readable representation of the returned data. Because of the quirks of this specific API, you will want to check this string to see if the returned data indicates an error.
- note: You can use
- If there are no errors, log the success and the response to the console.
- After posting to the API, call
fetchPoststo load the newPostand any other newPostobjects from the server.
- Initialize the
- Add a +
UIBarButtonItemto thePostListTableViewControllerscene in the storyboard - Add an IBAction to the
PostListTableViewControllerclass file - Write a
presentNewPostAlert()function that initializes aUIAlertController.- Add a
userNameTextFieldand amessageTextFieldthat the user will use to create their message. - Add a 'Post' alert action that guards for username and message text, and uses the
PostControllerto add a post with the username and text.
- Add a
- Write a
presentErrorAlert()function that initializes aUIAlertControllerthat says the user is missing information and should try again. Call the function if the user doesn't include include text in theuserNameTextFieldormessageTextField - Call the
presentNewPostAlert()function from the IBaction from the +UIBarButtonItem
You may have noticed that the network request to load the global feed can take multiple seconds to run. As more students build this project and submit more messages, the data returned from the PostController will get larger and larger. When you are working with hundreds of objects, this is not a problem. But once you start dealing with thousands, tens of thousands, or more, things will start lowing down considerably.
Additionally, consider that the user is unlikely to scroll all the way to the first message in the global feed if there are thousands of posts. We can be more efficient by not loading it in the first place.
To avoid the inefficiency of loading data that will never be displayed, many APIs support querying or paging. The Post API you are using for this project supports paging. We will implement paging on the PostController, and add support on the Post List Scene to load new posts as the user scrolls.
Update the PostController to fetch a limited number of Post objects from the API by using the URL parameters detailed in the API documentation.
Consider that there are two use cases for using the fetchPosts function:
- Load a fresh list of
Postobjects for when the user wants to see the latest posts. - Add the next set (or 'page') of posts to the already fetched posts for when the user wants to see older posts than the ones already loaded.
So you must update the fetchPosts function to support both of these use cases.
- Add a Bool
resetparameter to the beginning of thefetchPosts()function. Assign a default value oftrue.- This value will be used to determine whether you should replace the
.postsproperty, or append to the end of it
- This value will be used to determine whether you should replace the
- Review the API Documentation to determine what URL parameters you need to pass to fetch a subset of posts.
- note: Experiment with the URL parameters using PostMan, Paw, or your youb browser.
- Consider the following concepts. Attempt to implement the different ways you have considered. Continue to the next step after 10 minutes.
- Consider how you can get the range of timestamps for the request
- Consider how many
Postdictionaries you want returned in the request - Use a whiteboard to draw out scenarios and potential sorting and filtering mechaninisms to get the data you want
- Use the following logic to generate the URL parameters to get the desired subset of
PostJSON. This can be complex, but think through it before using the included sample code below.- You want to order the posts in reverse chronological order.
- Request the posts ordered by
timestampto put them in chronological order (orderBy). - Specify that you want the list to at the
timestampof the least recentPostyou have already fetched (or at the current date if you haven't posted any). Specify that you want the posts at the end of that ordered list (endAt). - Specify that you want the last 15 posts (
limitToLast).
- Request the posts ordered by
- You want to order the posts in reverse chronological order.
- Determine the necessary
timestampfor your query based on whether you are resetting the list (where you would want to use the current time), or appending to the list (where you would want to use the time of the earlier fetchedPost).
let queryEndInterval = reset ? NSDate().timeIntervalSince1970 : posts.last?.timestamp ?? NSDate().timeIntervalSince1970- Build a
[String: String]Dictionary literal of the URL Parameters you want to use.
let urlParameters = [
"orderBy": "\"timestamp\"",
"endAt": "\(queryEndInterval)",
"limitToLast": "15",
]- Pass the URL parameters in the
performRequestForURLfunction. - Replace the
self.posts = sortedPostswith logic that uses theresetparameter to to determine whether you should replaceself.postsor append toself.posts.- note: If you want to reset the list, you want to replace, otherwise, you want to append.
Add paging functionality to the List View by adding logic that checks for when the user has scrolled to the end of the table view, and calls the updated fetchPosts function with the correct parameters.
- Review the
UITableViewDelegateProtocol Reference to find a function that could be used to determine when the user has scrolled to the bottom of the table view.- note: Move on to the next step after reviewing for potential solutions to implement this feature.
- Add and implement the
willDisplayCellfunction- Check if the indexPath.row of the cell parameter is greater than the number of posts currently loaded on the
postController - If so, call the
fetchPostsfunction with reset set to false - In the completion closure, reload the tableview if the returned [Post] is not empty
- Check if the indexPath.row of the cell parameter is greater than the number of posts currently loaded on the
Review the newly implemented paging feature. Scroll through the posts on the feed. Pay special attention to any abnormalities (unordered posts, repeated posts, empty posts, etc).
You will notice that there is a repeated post where every new fetch occurred. If you review the API documentation, you'll find that our endAt query parameter is inclusive, meaning that it will include any posts that match the exact timestamp of the last post. So each time we run the fetchPosts function, the API will return a duplicate of the last post.
We can fix this bug by adjusting the timestamp we use for the query by a single digit.
- Add a computed property
queryTimestampto thePosttype that returns anNSTimeIntervaladjusted by 0.000001 from theself.timestamp - Update the
queryEndIntervalvariable in thefetchPostsfunction to use theposts.last?.queryTimestampinstead of the regulartimestamp
- Any app that displays user submitted content is required to provide a way to report and hide content, or it will be rejected during App Review. Add reporting functionality to the project.
- Update the user interface to cue to the user that a post is new.
- Make your table view more efficient by inserting cells for new posts instead of reloading the entire tableview.
- Implement streaming with web sockets.
Please refer to CONTRIBUTING.md.
© DevMountain LLC, 2015. Unauthorized use and/or duplication of this material without express and written permission from DevMountain, LLC is strictly prohibited. Excerpts and links may be used, provided that full and clear credit is given to DevMountain with appropriate and specific direction to the original content.