Combine Networking is a convenient and flexible networking layer based on Apple's native Combine and URLSession technologies.
It does not use any third-party dependencies, so there is nothing superfluous in it.
public protocol CNPlugin {
func modifyRequest(_ request: inout URLRequest)
}
An object protocol that can modify a URLRequest. Most often you will use it to add certain headers:
struct AuthPlugin: CNPlugin {
let token: String
init(token: String) {
self.token = token
}
func modifyRequest(_ request: inout URLRequest) {
request.setValue(token, forHTTPHeaderField: "x-api-key")
}
}
You can use the default implementation - CNReachabilityManagerImpl. It is a singleton, which you can get by property shared.
public protocol CNReachabilityManager: AnyObject {
var isInternetConnectionAvailable: Bool { get }
}
Protocol object that monitors the internet connection before running the request.
You can write your own manager that will subscribe to this protocol, or use the default implementation - CNReachabilityManagerImpl.
public protocol CNErrorHandler {
associatedtype ErrorType: CNErrorProtocol
func outputHandling(
_ output: NetworingOutput,
_ retryMethod: @autoclosure @escaping () -> AnyPublisher<Data, ErrorType>
) -> AnyPublisher<Data, ErrorType>
func convert(error: NSError) -> ErrorType
}
Manager protocol for handling errors resulting from Combine Networking.
You can use the standard implementation of CNErrorHandler - CNErrorHandlerImpl. But if you need to handle errors specifically, you will use your own.
It contains:
-
ErrorType associatedtype
The error type that your Combine Networking will work with. You can use your own error type, but it must be signed to CNErrorProtocol, which contains the main errors. -
outputHandling method
A method for processing the response from the server. Gets a server response that you can process the way you want, and a retryMethod block that can be used to retry the request after it has been processed. -
convert method
It often happens that a request to the network fails. For example, while launching the request, the Internet on the device has disappeared.
This method takes such errors as NSError, and serves to convert it into your custom error.
The request builder protocol contains variables that are used to build the request step by step.
public protocol CNRequestBuilder {
var path: String { get }
var query: QueryItems? { get }
var body: Data? { get }
var method: HTTPMethod { get }
var baseURL: URL? { get }
var headerFields: HTTPHeaderFields? { get }
var multipartBody: CNMultipartFormDataModel? { get }
}
// MARK: - Public Methods
extension CNRequestBuilder {
public var baseURL: URL? { nil }
public var headerFields: HTTPHeaderFields? { nil }
public var multipartBody: CNMultipartFormDataModel? { nil }
public func makeRequest(baseURL: URL, plugins: [CNPlugin]) -> URLRequest { ... }
}
It is convenient to use with Enums, which will describe each individual request as a separate enum case.
RequestBuilder includes:
-
path
The path that will be added to the base URL when creating a request. -
query
Query items as keys and values which will be added to the request. If request shouldn't contain them, it should return nil. -
body
Data to be added to the body of the request. If the request should not contain a body - return nil. -
method
HTTP Method which will be added to the reques. -
baseURL
A base URL that can be added to your request. Serves as an exception to the rule. In most cases the base URL will be taken from the CNProvider. Specify this variable only when the base URL of a particular request differs from the base URL injected by Networking Provider. You can ignore this variable when creating an object. By default it returns nil. -
headerFields
Header fields that can be added to the request. Used only if you want to specify additional header fields in addition to those specified in the CNPlugin array. You can ignore this variable when creating an object. By default it returns nil. -
multipartBody
Variable that can return CNMultipartFormDataModel. Used to create multipart form requests. If the variable does not return nil, the body created by the CNMultipartFormDataModel will be added to the request. Do not use this and body variable at the same time. You can ignore this variable when creating an object. By default it returns nil.
A class that serves as a provider to run requests to the server.
CNProvider is an implementation of the CNProviderProtocol. In this section you will learn about the accepted CNProvider parameters. You can learn more about its methods and generic types in CNProviderProtocol.
CNProvider has several initializers. One of them uses the standard implementation of the CNErrorHandler protocol - CNErrorHandlerImpl.
public init(
baseURL: URL,
reachability: CNReachabilityManager = CNReachabilityManagerImpl.shared,
session: URLSession = .shared,
requestBuilder: RequestBuilder.Type,
plugins: [CNPlugin] = [],
decoder: JSONDecoder = JSONDecoder()
) where ErrorHandler == CNErrorHandlerImpl {...}
public required init(
baseURL: URL,
reachability: CNReachabilityManager = CNReachabilityManagerImpl.shared,
session: URLSession = .shared,
errorHandler: ErrorHandler,
requestBuilder: RequestBuilder.Type,
plugins: [CNPlugin] = [],
decoder: JSONDecoder = JSONDecoder()
) {...}
The init takes in:
-
baseURL
Base URL where the request will be made. -
reachability
CNReachabilityManagerImpl by default. -
session
URLSession with which the request will be executed. URLSession.shared by default. -
errorHandler
Responsible for handling errors received in CNProvider. -
requestBuilder
The type of builder the provider will work with. -
plugins
Array with objects for request modification. Most often you will use it to customize request headers. Empty array by default. -
decoder
JSONDecoder with which the object will be decoded.
Protocol for CNProvider.
public protocol CNProviderProtocol {
associatedtype RequestBuilder: CNRequestBuilder
associatedtype ErrorHandler: CNErrorHandler
var baseURL: URL { get }
var reachability: CNReachabilityManager { get }
var session: URLSession { get }
var errorHandler: ErrorHandler { get }
var plugins: [CNPlugin] { get }
var decoder: JSONDecoder { get }
func generalPerform(
_ builder: RequestBuilder
) -> AnyPublisher<Data, ErrorHandler.ErrorType>
func perform<T: Decodable>(
_ builder: RequestBuilder
) -> AnyPublisher<T, ErrorHandler.ErrorType>
func perform<DecodableType: Decodable, Abstraction>(
_ builder: RequestBuilder,
decodableType: DecodableType.Type
) -> AnyPublisher<Abstraction, ErrorHandler.ErrorType>
func perform(
_ builder: RequestBuilder
) -> AnyPublisher<Never, ErrorHandler.ErrorType>
}
Protocol contains associatedtypes:
-
RequestBuilder
Type of object subscribed to the CNRequestBuilder protocol. Responsible for describing and creating the URLRequest object. -
ErrorHandler
The object type subscribed to the CNErrorHandler protocol. Responsible for handling errors received in CNProvider.
You will use the following methods to run the request to the server:
-
generalPerform(_:)
A method that starts a request task. Returns the publisher with the Data, or your custom error, which type is defined in CNErrorHandler. -
perform<T: Decodable>(_:)
A method that starts a request task and decodes the response into an object, relying on a generic parameter. Returns the publisher with the decoded object, or your custom error, which type is defined in CNErrorHandler. -
perform<DecodableType: Decodable, Abstraction>(_:, decodableType:)
A method that starts a request task and decodes the response into an object, relying on a decodableType parameter. Returns the publisher with the generic abstraction object – protocol of the decoded object, or your custom error, which type is defined in CNErrorHandler. -
perform(_:)
A method that starts a request task. Returns the publisher, which can be completed successfully or with your custom error, which type is defined in CNErrorHandler.
Follow these steps to set up your networking layer. Steps marked as optional can be skipped.
- Creating CNPlugin implementations
- Creating a CNErrorProtocol implementation – optional
- Creating a CNErrorHandler implementation – optional
- Creating a CNReachabilityManager implementation – optional
- Creating CNRequestBuilder implementations
- Creating CNProvider implementations
- Creating and using a network client
Create a CNPlugin object for every possible header that can be added to the request.
struct AuthPlugin: CNPlugin {
let token: String
init(token: String) {
self.token = token
}
func modifyRequest(_ request: inout URLRequest) {
request.setValue(token, forHTTPHeaderField: "x-api-key")
}
}
Create your own CNErrorProtocol implementation if the CNError implementation does not suit you. All kinds of errors that your networking can return will be described here.
You can skip this step and use the standard implementation of CNErrorProtocol - CNError.
enum CustomErrorTypeName: CNErrorProtocol {
case badURLError
case clientError
case serverError
// Mandatory errors according to CNErrorProtocol.
case reachabilityError
case decodingError
case unspecifiedError
// -----
var localizedDescription: String {
switch self {
case .badURLError:
return "Bad URL Error. Please try again later."
case .clientError:
return "An error occurred on the client side. Please try again later."
case .serverError:
return "An error occurred on the server side. Please try again later."
case .reachabilityError:
return "Internet connection problem. Please check your internet connection."
case .decodingError:
return "We were unable to identify the data that came from the server. Please try again later."
case .unspecifiedError:
return "For unknown reasons, something went wrong. Please try again later."
}
}
}
Create your own implementation of the CNErrorHandler protocol if the default CNErrorHandlerImpl implementation is not suitable for you. This object will handle the response from the server, and return the type of error you specified (step #2).
You can skip this step and use the standard implementation of CNErrorHandler - CNErrorHandlerImpl.
struct CustomErrorHandlerTypeName: CNErrorHandler {
public func outputHandling(
_ output: NetworingOutput,
_ retryMethod: @autoclosure () -> AnyPublisher<Data, CustomErrorTypeName>
) -> AnyPublisher<Data, CustomErrorTypeName> {
guard let httpResponse = output.response as? HTTPURLResponse else {
return Fail(error: CustomErrorTypeName.unspecifiedError)
.eraseToAnyPublisher()
}
switch httpResponse.statusCode {
case 200...399:
return Just(output.data)
.setFailureType(to: CustomErrorTypeName.self)
.eraseToAnyPublisher()
case 400...499:
return Fail(error: CustomErrorTypeName.clientError)
.eraseToAnyPublisher()
case 500...599:
return Fail(error: CustomErrorTypeName.serverError)
.eraseToAnyPublisher()
default:
return Fail(error: CustomErrorTypeName.unspecifiedError)
.eraseToAnyPublisher()
}
}
public func convert(error: NSError) -> CustomErrorTypeName {
switch error.code {
case NSURLErrorBadURL:
return .badURLError
case NSURLErrorNotConnectedToInternet, NSURLErrorCallIsActive,
NSURLErrorNetworkConnectionLost, NSURLErrorDataNotAllowed:
return .reachabilityError
default: return .unspecifiedError
}
}
}
Create your own implementation of the CNReachabilityManager protocol if the default CNReachabilityManagerImpl implementation is not suitable for you. This object monitors the ability to connect to the network before launching a request to the server.
You can skip this step and use the standard implementation of CNReachabilityManager - CNReachabilityManagerImpl. It is a singleton, which you can get by property shared.
class ReachabilityManager: CNReachabilityManager {
private static let queueLabel = "MyCustomeQueue"
public var isInternetConnectionAvailable: Bool = {
return false
}()
private var connectionMonitor = NWPathMonitor()
public init() {
let queue = DispatchQueue(
label: ReachabilityManager.queueLabel
)
self.connectionMonitor.pathUpdateHandler = { pathUpdateHandler in
self.isInternetConnectionAvailable = pathUpdateHandler.status == .satisfied
}
self.connectionMonitor.start(queue: queue)
}
}
Create an implementation of the CNRequestBuilder as an enum. Try to logically separate the CNRequestBuilder implementation.
enum DogRequestBuilder: CNRequestBuilder {
case searchBy(name: String)
// DogRequestModel – some encodable model
case create(dog: DogRequestModel)
var path: String {
switch self {
case .searchBy:
return "/dog/search"
case .create:
return "/dog/create"
}
}
var query: QueryItems? {
switch self {
case .searchBy(let name):
return ["dog_name": name]
case .create:
return nil
}
}
var body: Data? {
switch self {
case .searchBy(_):
return nil
case .create(let model):
return try? JSONEncoder().encode(model)
}
}
var method: HTTPMethod {
switch self {
case .searchBy(_):
return .get
case .create(let model):
return .post
}
}
}
enum HumanRequestBuilder: CNRequestBuilder {
case search(name: String)
var path: String {
switch self {
case .search: return "/human/search"
}
}
var query: QueryItems? {
switch self {
case .search(let name):
return ["human_name": name]
}
}
var body: Data? { return nil }
var method: HTTPMethod { return .get }
}
The CNProvider implementation takes all the elements you created earlier. It is responsible for launching a request to the network.
...
let baseURL = URL(string: "https://api.example.com/v1")!
let authPlugin = AuthPlugin(token: "my_token")
let reachabilityManager = ReachabilityManager()
let errorHandler = CustomErrorHandlerTypeName()
let provider = CNProvider(
baseURL: baseURL,
reachability: reachabilityManager,
session: URLSession.shared,
errorHandler: errorHandler,
requestBuilder: DogRequestBuilder.self,
plugins: [authPlugin],
decoder: JSONDecoder()
)
let myClientForWork = DogClient(provider)
...
With the CNProvider configured, you can easily create a client to work with the server. If protocols are important to you, you can use CNProviderProtocol.
// Creating a class using CNProviderProtocol
class DogClient<Provider: CNProviderProtocol> where Provider.RequestBuilder == DogRequestBuilder, Provider.ErrorHandler == CustomErrorHandlerTypeName {
private let provider: Provider
init(provider: Provider) {
self.provider = provider
}
}
extension DogClient: DogClientProtocol {
// Response – decodable model or error
func search(name: String) -> AnyPublisher<DogDecodableModel, CustomErrorTypeName> {
provider.perform(
.search(name: name)
)
.receive(on: DispatchQueue.main)
.eraseToAnyPublisher()
}
// Response – decodable model as abstraction or error
func search(name: String) -> AnyPublisher<DogModelProtocol, CustomErrorTypeName> {
provider.perform(
.search(name: name),
decodableType: DogDecodableModel.self
)
.receive(on: DispatchQueue.main)
.eraseToAnyPublisher()
}
// Response – success or error completion
func create(dog: DogRequestModel) -> AnyPublisher<Never, CustomErrorTypeName> {
provider.perform(
.create(dog: dog)
)
.receive(on: DispatchQueue.main)
.eraseToAnyPublisher()
}
}
// Creating a class without using CNProviderProtocol
class HumanClient {
private let provider: CNProvider<HumanRequestBuilder, CNErrorHandlerImpl>
init(with provider: CNProvider<HumanRequestBuilder, CNErrorHandlerImpl>) {
self.provider = provider
}
}
With CNRequestBuilder you can describe multipart data. To do this, create a multipartBody variable inside your CNRequestBuilder.
It is a CNMultipartFormDataModel structure that is initialized with the CNMultipartFormItem array. CNMultipartFormItem is a description of one part of your data.
multipartBody replaces the data variable in CNRequestBuilder, so do not use them simultaneously.
enum HumanRequestBuilder: CNRequestBuilder {
case setHuman(image: Data)
var path: String {
switch self {
case .setHuman: return "/human/image"
}
}
var query: QueryItems? {
switch self {
case .setHuman(_):
return nil
}
}
var body: Data? {
switch self {
case .setHuman(_):
return nil
}
}
var method: HTTPMethod {
switch self {
case .setHuman(_):
return .post
}
}
// MARK: - Multipart Request
var multipartBody: CNMultipartFormDataModel? {
switch self {
case .setHuman(let image):
let multipartItem = CNMultipartFormItem(
name: "human_image_request_key",
fileName: "my_image",
mimeType: "image/jpg", // Optional
data: image
)
return CNMultipartFormDataModel(items: [multipartItem])
}
}
}
Combine Networking available through Swift Package Manager.
in Package.swift
add the following:
dependencies: [
// Dependencies declare other packages that this package depends on.
// .package(url: /* package url */, from: "0.1.0"),
.package(url: "https://github.com/kvasnetskyi/CombineNetworking.git", from: "0.1.0")
],
targets: [
.target(
name: "MyProject",
dependencies: [..., "CombineNetworking"]
)
...
]
- Kvasnetskyi Artem, Savchenko Roman, Kosyi Vlad, CHI Software
Copyright 2021 CHI Software.
Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License.