This library contains Swift3 networking utilities for use in both Qvik's internal and customer projects. They are released with the MIT license.
- 1.0.0
- Swift3 / 1.0.0 release
- Dropped dependencies to SwiftKeychain / SwiftGif
- 0.10.0
- Swift3 support
- 0.9.2
- Added MockRemoteService
- Added Double.clamp() and CGFloat.clamp()
- 0.9.1
- Fixed Swift3 deprecation warnings
- Added features + performance boost to preview thumbnails (CachedImageView)
- Added top-level JSON-array support to rest call parsing (BaseRemoteService + RemoteResponse)
- 0.9.0
- Adopted AlamoFire 3; getting ready for 1.0
- 0.1.4
- Moved the podspec to point to Github
- 0.1.0
- Added JPEG Thumbnails support.
- 0.0.14
- ImageCache: GIF passthrough support.
- 0.0.13
- Some rework to further boost robustness of ImageCache
- Added put/remove image functionality to ImageCache
- 0.0.12
- Added .NetworkTimeout support
- 0.0.11
- Added missing response parsing for non-200 responses
- 0.0.10
- Changes for changed CryptoSwift API compliance
- 0.0.9
- New features to CachedImageView
- 0.0.8
- Added autoreleasepool {} to control memory usage with writing image data
- 0.0.7
- Using a dedicated logger instance instead of default (shared) one
- Performance increase to image cache; added data passthrough writes when applicable
- Made the ImageCache instance configurable for CachedImageView
- 0.0.6
- Made CachedImageView inherit from QvikImageView
- 0.0.5
- Added image changed -callback to CachedImageView
- 0.0.4
- Added BaseRemoteService
- 0.0.3
- Initial version
To use the library in your projects, add the following (or what ever suits your needs) to your Podfile:
use_frameworks!
source 'https://github.com/qvik/qvik-podspecs.git'
pod 'QvikNetwork'
And the following to your source:
import QvikNetwork
The library has dependencies to the following external modules:
The library may emit logging for errors, and if you tell it to, debug stuff. Enable debug logging as such:
QvikNetwork.logLevel = .debug // Or .info, .verbose
This chapter introduces the library's classes.
The purpose of DownloadManager (and the related classes DownloadGroup and Download) is to provide easy-to-use API for downloading files over HTTP(S), typically dynamic application resources such as images and data files. DownloadManager also provides a way of grouping downloads together for tracking the overall download progress instead of adding this code to the application logic. Downloading multiple items at once is common feature in modern mobile applications. DownloadManager and the related classes are thread safe.
Example usage:
let downloadUrls = ["http://example.com/1", "http://example.com/2", "http://example.com/3"]
let downloadManager = DownloadManager.sharedInstance()
let downloadGroup = downloadManager.createGroup()
downloadGroup.progressCallback = { [weak self] totalBytesRead, progress in
updateProgressUI(progress)
}
downloadGroup.completionCallback = { [weak self] numErrors in
log.debug("All downloads completed, \(numErrors) errors")
updateCompletedUI()
}
for url in downloadUrls {
let download = downloadGroup.download(url)
download.progressCallback = { [weak self] totalBytesRead, totalBytesExpectedToRead in
let progress = Double(totalBytesRead) / Double(totalBytesExpectedToRead)
log.debug("Download progress for url \(url): \(progress)")
}
}
A very fast, disk-backed in-memory image cache, featuring configurable storage location, file format (JPEG/PNG) and disk storage max age etc. It can also be configured to automatically downscale any downloaded images to a maximum size (retaining aspect ratio) before insertion to the in-memory storage and disk. ImageCache is thread safe.
Example usage:
func imageLoaded(notification: Notification) {
if let imageUrl = notification.userInfo?[ImageCache.urlParam] as? String {
// Check if it is my image
if imageUrl == self.imageUrl {
// Now the image is in the cache - get it
let image = ImageCache.sharedInstance().getImage(url: imageUrl, loadPolicy: .memory)
// TODO use the image for something
}
}
}
// ..
let imageCache = ImageCache.sharedInstance()
NotificationCenter.default.addObserver(self, #selector(imageLoaded), name: NSNotification.Name(rawValue: ImageCache.cacheImageLoadedNotification), object: nil)
// This will trigger a fetch over the network if not found in cache
guard let image = imageCache.getImage(url: imageUrl, loadPolicy: .Network) else {
log.debug("Requested image not found in in-memory cache; waiting for a load from disk or network")
// .. wait for imageLoaded() to get called
}
A UIImageView that is backed up by ImageCache; features automatic image retrieval from the cache (and hence from the network if not found locally). Also supports JPEG Thumbnails (see below) as well as a fade-in effect.
Sample usage:
@IBOutlet private weak var imageView: CachedImageView!
func populateView(message: Message) {
// Will automatically trigger load from cache/network and populate the image when done
imageView.imageUrl = message.imageUrl
// Will display this image while the other one loads
imageView.placeholderImage = smallerImage
// Control how fast the fade-in animation for async loaded images is
imageView.imageFadeInDuration = 0.5
}
This is a superclass for a Remote API service built on top of the Alamofire HTTP library. It provides response parsing and wraps all the Alamofire APIs, reducing boiler plate from the actual implementation. For full feature set, see the code.
Example implementation:
class RemoteService: BaseRemoteService {
private static let singletonInstance = RemoteService()
class func sharedInstance() -> RemoteService {
return singletonInstance
}
/// Attempts to verify an email address
func verifyEmailAddress(email: String, completionCallback: (RemoteResponse -> Void)) {
let url = "\(baseUrl)/api/verifyemail"
let params = ["email": email]
request(.POST, url, parameters: params, encoding: .JSON, headers: nil) { response in
completionCallback(response)
}
}
// TODO Other methods ..
init() {
let additionalHeaders = [
"X-My-API-Key": "asldasdaksdjhakdjhasdkjh",
"X-My-Api-Version": "1",
"X-My-Client": getClientId()
]
super.init(backgroundSessionId: "com.example.MyApp", additionalHeaders: additionalHeaders, timeout: 5)
}
}
You like to unit test your REST interface? With MockRemoteService
you can! Set up your interface to use BaseRemoteService
with an has-a relationship instead of a is-a relationship, instantiate MockRemoteService
and use it instead of BaseRemoteService
. For example like this:
class TestRemoteService {
let remoteImpl: BaseRemoteService
let baseUrl = "http://www.site.com"
func list(callback: ((RemoteResponse) -> Void)) {
let url = "\(baseUrl)/list"
remoteImpl.request(.GET, url, parameters: nil, callback: callback)
}
func update(name name: String, age: Int, married: Bool, callback: ((RemoteResponse) -> Void)) {
let url = "\(baseUrl)/update"
let params: [String: AnyObject] = ["name": name, "age": age, "married": married]
remoteImpl.request(.POST, url, parameters: params, callback: callback)
}
init(remoteImpl: BaseRemoteService) {
self.remoteImpl = remoteImpl
}
}
And build test cases like this:
func testPathAndParamMapping() {
let mockService = MockRemoteService()
// Create a precondition that /update always fails if params age = 17, married = true are set
mockService.addOperationMappingForPath("/update", mapping: (failureProbability: 1.0, params: ["age": 17, "married": true], successResponse: ["status": "ok"], failureResponse: ["status": "failed"], failureError: .ServerError))
// Set default success response content
mockService.successResponse = ["status": "ok"]
let remoteService = TestRemoteService(remoteImpl: mockService)
let mustSucceed = expectationWithDescription("Must succeed with these params")
let mustFail = expectationWithDescription("Must fail with these params")
// Start a request that should succeed
remoteService.update(name: "Leslie", age: 18, married: true) { response in
if response.success {
// Also check that proper success response content is in place
if let status = response.parsedJson?["status"] as? String where status == "ok" {
mustSucceed.fulfill()
}
}
}
// Start a request that should fail
remoteService.update(name: "Leslie", age: 17, married: true) { response in
if !response.success {
// Also check that proper failure content is in place
if let status = response.parsedJson?["status"] as? String where status == "failed" {
mustFail.fulfill()
}
}
}
waitForExpectationsWithTimeout(1.0, handler: nil)
}
}
JPEG thumbnails can be used for fast image previews. To understand the concept, read the links mentioned in the JpegThumbnails.swift
source file.
/// Use a thumbnail:
thumbnailImageView.image = jpegThumbnailDataToImage(data: thumbnailData, maxSize: self.frame.size)
/// Or, benefit from background threading etc. in CachedImageView:
myCachedImageView.thumbnailDominantColor = myImageObject.thumbDominantColor
myCachedImageView.thumbnailData = myImageObject.thumbData
Contributions to this library are welcomed. Any contributions have to meet the following criteria:
- Meaningfulness. Discuss whether what you are about to contribute indeed belongs to this library in the first place before submitting a pull request.
- Code style. Follow our Swift style guide 100%.
- Stability. No code in the library must ever crash; never place *assert()*s or implicit optional unwrapping in library methods.
- Testing. Create a test app for testing the functionality of your classes and/or provide unit tests if appropriate.
- Logging. All code in the library must use the common logging handle (see QvikNetwork.swift) and sensible log levels.
This project uses Swiftlint. No code in the library must generate Swiftlint warnings or errors.
Guide for setting up Swiftlint
The library is distributed with the MIT License. Make sure all your source files contain the license header at the start of the file:
// The MIT License (MIT)
//
// Copyright (c) 2015-2016 Qvik (www.qvik.fi)
//
// Permission is hereby granted, free of charge, to any person obtaining a copy
// of this software and associated documentation files (the "Software"), to deal
// in the Software without restriction, including without limitation the rights
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
// copies of the Software, and to permit persons to whom the Software is
// furnished to do so, subject to the following conditions:
//
// The above copyright notice and this permission notice shall be included in all
// copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
// SOFTWARE.
All merges to the master branch go through a Merge ('pull') Request and MUST meet the above criteria.
In other words, follow the following procedure to submit your code into the library:
- Clone the library repository
- Create a feature branch for your code
- Code it, clean it up, test it thoroughly
- Make sure all your methods meant to be public are defined as public
- Push your branch
- Create a merge request
As a contributor you do not need to do this; we'll update the pod whenever needed by projects.
- Update QvikNetwork.podspec and set s.version to match the upcoming tag
- Commit all your changes, merge all pending accepted Merge ('pull') Requests
- Create a new tag following Semantic Versioning; eg.
git tag -a 1.2.0 -m "Your tag comment"
git push --tags
pod repo push qvik-podspecs QvikNetwork.podspec
Unless already set up, you might do the following steps to set up the pod repo:
pod repo add qvik-podspecs https://github.com/qvik/qvik-podspecs.git
Any questions? Contact [email protected].