Skip to content

Latest commit

 

History

History
842 lines (586 loc) · 23.9 KB

README.md

File metadata and controls

842 lines (586 loc) · 23.9 KB

CodableRequest

Structured HTTP URLRequest and API client for Swift

Danger Swift Codecov

CodableRequest is a pure Swift framework for building URLRequests using property wrappers. It's a fork of Postie library

Example

Checkout this full example starting at defining the request and the expected response, up to creating a client and sending it to the remote endpoint.

import Foundation
import CodableRequest

// Request contains body data encoded as a JSON
struct MyRequest: JSONRequest {

    // The request body is strongly typed defined
    struct RequestBody: Encodable {
        var someNumberValue: Int
    }

    // Define the response directly inside the request, so every
    // Request-Response are isolated.
    // Also directly define, that the response body shall be decoded
    // from JSON
    struct Response: JSONDecodable {

        // The expected response body structure
        struct Body: Decodable {
            var someNumberValue: Int
        }

        // The expected response body structure, in case we did something wrong
        struct ErrorBody: Decodable {
            var message: String
        }

        // Property wrappers define the purpose
        @ResponseBody<Body> var body
        @ErrorBody<ErrorBody> var errorBody

        // Access specific response headers
        @ResponseHeader<DefaultStrategy> var contentType: String

        // Status codes also have convenience utilities
        @ResponseStatusCode var statusCode

        // Cookies send by the remote
        @Cookies var cookies
    }

    // The `keyEncodingStrategy` determines how to encode a type’s coding keys as JSON keys.
    // The default value return `.convertToSnakeCase` but you can optionally choose to return `.useDefaultKeys` by implementing JSONRequest's protocol requirement as follow:
    // static var keyEncodingStrategy: JSONEncoder.KeyEncodingStrategy {
    //     .useDefaultKeys
    // }
    
    // static var dateEncodingStrategy: JSONEncoder.DateEncodingStrategy {
    //     .iso8601
    // }


    // This property holds the data which will be encoded
    var body: RequestBody

    // Location of our resource with template string
    Path var path = "/profile/{user_id}"

    // Parameter to replace in the template string
    PathParameter(name: "userId") var userId: String

    // HTTP method that shall be used
    @HTTPMethod var method = .post

    // Set request headers using the property naming
    @Header var authorization: String?
    
    // Set multiple instances of HTTPCookie
    @Cookies var cookies
}

// Create a request
var request = MyRequest(body: MyRequest.RequestBody(someNumberValue: 42),
                        userId: "my-user-id")
request.authorization = "Bearer my-oauth-token"

// Create a client
let client = CodableURLSession(
    url: URL(string: "https://example.org")!, 
    retryStrategy: .default
)

// Send the request with Combine
client.sendPublisher(request)
    .sink { result in
        switch result {
        case .failure(let error):
            print("Oh no something went wrong :(")
            print(error)
        case .finished:
            print("Everything worked fine :)")
        }
    } receiveValue: { response in
        // The single response object contains all the interesting data
        print(response.statusCode)
        print(response.body)
        print(response.errorBody)
        print(response.contentType)
    }

Core Concept

The networking layer of Foundation (and with Combine) is already quite advanced. Using URLRequest you can set many different configuration values, e.g. the HTTP Method or Headers.

Unfortunately you still need to manually serialize your payload into Foundation.Data and set it as the request body. Additionally you also have to set Content-Type header, or otherwise the remote won't be able to understand the content.

Also the response needs to be decoded, and even if a few decoders are included, e.g. JSONDecoder, reading and parsing the URLResponse is not intuitive.

Even worse when the response structure differs in case of an error, e.g. instead of

{
  "some": "data"
}

an error object is returned:

{
  "error": {
    "message": "Something went wrong!"
  }
}

This would require to create combined types such as this one:

struct Response: Decodable {
    struct ErrorResponse: Decodable {
        var message: String
    }

    var some: String?
    var error: ErrorResponse?
}

and you would have to use nil-checking (probably in combination with the HTTP Status Code) to see which data is present.

CodableRequest simplifies these use cases. The main idea is defining slim struct types to build the requests, and serialize the associated responses. Configuration of the request is done using property wrappers, e.g. @QueryItem.

Usage

Defining the request

CodableRequest includes a couple of types to build your requests. As a first step, create your Request type, with an associated Response:

import CodableRequest

struct FooRequest: Request  {
    typealias Response = EmptyResponse
}

The default Request type is used for URL requests without any body data. If you want to include payload data, use one of the following ones:

  • PlainRequest
  • JSONRequest
  • FormURLEncodedRequest

All of these expect a body instance variable. For JSONRequest, FormURLEncodedRequest and the type of body is generic but needs to implement the Encodable protocol.

Example:

struct Foo: JSONRequest {

    struct Body: Encodable {}
    typealias Response = EmptyResponse

    var body: Body

}

struct Bar: FormURLEncodedRequest {

    struct Body: Encodable {}
    typealias Response = EmptyResponse

    var body: Body

}

For the PlainRequest the body expects a plain String content. Optionally you can also overwrite the encoding variable with a custom encoding (default is utf8).

Example:

struct Foo: PlainRequest {

    typealias Response = EmptyResponse

    var body: String
    var encoding: String.Encoding = .utf16 // default: .utf8

}

Setting the request HTTP Method

The default HTTP method is GET, but it can be overwritten by adding an instance property with the property wrapper @HTTPMethod:

Example:

struct Request: Encodable {

    typealias Response = EmptyResponse

    @HTTPMethod var method

}

// Usage
var request = Request()
request.method = .post

Note:

As the property name is ignored, it is possible to have multiple properties with this property wrapper, but only the last one will be used.

Setting the request URL path

The default path /, but it can be overwritten by adding an instance property with the property wrapper Path:

Example:

struct Request: Encodable {

    typealias Response = EmptyResponse

    Path var path

}

// Usage
let request = Request(path: "/some-detail-path")

Additionally the request path can contain variables using the mustache syntax, e.g. /path/with/{variable_name}/inside.

To set the variable value, add a new instance property using the PathParameter property wrapper. By default the encoder uses the variable name for encoding, but you can also define a custom name:

struct Request: Encodable {

    typealias Response = EmptyResponse

    Path var path = "/app/{id}/contacts/{cid}"
    PathParameter var id: Int
    PathParameter(name: "cid") var contactId: String

}

// Usage
var request = Request(id: 123)
request.contactId = "ABC456"

// Result:
https://CodableRequest.local/app/123/contacts/ABC456

Note:

As the property name is ignored, it is possible to have multiple properties with this property wrapper, but only the last one will be used. Also you need to require a leading forward slash (/) in the path.

Adding query items to the URL

Multiple query items can be added by adding them as properties using the property wrapper @QueryItem.

Example:

struct Request: Encodable {

    typealias Response = EmptyResponse

    @QueryItem
    var text: String

    @QueryItem(name: "other_text")
    var anotherQuery: String

    @QueryItem
    var optionalText: String?

}

// Usage
var request = Request(text: "foo")
request.anotherQuery = "bar"

// Result query in URL:
?text=foo&other_text=bar

If no custom name is set, the variable name is used. If the query item is optional, and not set (therefore nil), it won't be added to the list.

Supported query value types can be found in QueryItemValue.swift.

Note:

When using an Array as the query item type, every value in the array is appended using the same name. The remote server is then responsible to collect all query items with the same name and merge them into an array.

Example: [1, 2, 3] with name values becomes ?values=1&values=2&values=3

As multiple query items can use the same custom name, they will all be appended to the query. This does not apply to synthesized names, as a Swift type can not have more than one property with the exact same name.

Adding Headers to the request

Multiple headers can be set by adding them as properties using the property wrapper @Header.

Example:

struct Request: Encodable {

    typealias Response = EmptyResponse

    @Header
    var text: String

    @Header(name: "other_text")
    var anotherQuery: String

    @Header
    var optionalText: String?

}

// Usage
var request = Request(text: "foo")
request.anotherQuery = "bar"

// Result query in URL:
?text=foo&other_text=bar

If no custom name is set, the variable name is used. If the header is optional, and not set (therefore nil), it won't be added to the list.

Supported header values types can be found in RequestHeaderValue.swift.

Note:

As multiple query items can use the same custom name, the last one will be used. This does not apply to synthesized names, as a Swift type can not have more than one property with the exact same name.

Defining the response

Every struct implementing Request expects to have an associated Response type implementing the Decodable protocol. In the examples above the EmptyResponse convenience type (which is an empty, decodable type) has been used.

The response structure will be populated with data from either the response body data or metadata.

Parsing the response body

To parse the response data into a Decodable type, add a property with the property wrapper @ResponseBody<BodyType> where BodyType is the response body type.

Example:

struct Request: CodableRequest.Request {
    struct Response: Decodable {
        struct Body: Decodable {
            var value: String
        }

        @ResponseBody<Body> var body
    }
}

To indicate the decoding system which response data format should be expected, conform your response type to one of the following protocols:

  • PlainDecodable
  • JSONDecodable

For JSONDecodable the type of body is generic but needs to implement the Decodable protocol.

Example:

struct Request: CodableRequest.Request {
    struct Response: Decodable {
        struct Body: JSONDecodable {
            var value: String
        }

        @ResponseBody<Body> var body
    }
}

For the type PlainDecodable, use it directly, as it is an alias for String.

Example:

struct Request: CodableRequest.Request {
    struct Response: Decodable {
        @ResponseBody<PlainDecodable> var body
    }
}

Response body on error

As mentioned in Core Concept CodableRequest allows defining a body response type when receiving an invalid status code (>=400).

It's usage is exactly the same as with @ResponseBody, but instead you need to use the property wrapper @ErrorBody. Either the @ResponseBody or the @ErrorBody is set, never both at the same time.

The error response body gets set if the response status code is neither a 2XX nor a 3XX status code.

Example:

struct Request: CodableRequest.Request {
    struct Response: Decodable {
        struct ErrorBody: JSONDecodable {
            var message: String
        }
        @ErrorBody<ErrorBody> var errorBody
    }
}

Custom Encoding Strategy

You can define a custom keyEncodingStrategy and dateEncodingStrategy for your JSONRequest to determine how the request body's keys and dates are encoded into JSON keys. This can be useful for handling cases where the backend expects keys in a specific format, such as snake_case or kebab-case.

Example:

extension MyRequest: JSONRequest {     
    static var keyEncodingStrategy: JSONEncoder.KeyEncodingStrategy { .convertToSnakeCase }
    
    ...
}

Custom Decoding Strategies

Similarly, you can specify decoding strategies for handling different date formats or key conventions in JSON responses. These strategies help in parsing complex JSON structures or dates represented in non-standard formats.

Example:

extension MyRequest.Response: JSONDecodable {
    static var keyDecodingStrategy: JSONDecoder.KeyDecodingStrategy { .convertFromSnakeCase } 
    static var dateDecodingStrategy: JSONDecoder.DateDecodingStrategy { .formatted(DateFormatter.iso8601Full) } 
}

Response parsing strategies

As the response body might differ depending on various factors, you can control the parsing using "decoding strategies".

By default if you use ResponseBody to parse the response body, it will use the DefaultBodyStrategy (which expects an HTTP status code 2XX or 3XX).

The same applies to the ResponseErrorBody to parse the response body for a status code of 400 or above, which is using the DefaultErrorBodyStrategy.

For your convenience we added a couple of convenience strategies in ResponseBody and ResponseErrorBody, you can use with the ResponseBodyWrapper and ErrorBodyWrapper.

If you want to implement a custom decoding strategy, all you need to do is define a struct implementing the protocol ResponseBodyDecodingStrategy or ResponseErrorBodyDecodingStrategy.

Example:*

struct CustomBodyDecodingStrategy {
    public static func allowsEmptyContent(for _: Int) -> Bool {
        false
    }

    public static func validate(statusCode: Int) -> Bool {
        // e.g. only decode if the status code is 999
        statusCode == 999
    }
}

struct Request: CodableRequest.Request {
    struct Response: Decodable {
        struct CreatedResponseBody: JSONDecodable {
            ...
        }
        
        @ResponseBody<CreatedResponseBody>.Status201 var createdBody: CreatedResponseBody
        
        struct CustomResponseBody: JSONDecodable {
            ...
        }
        
        @ResponseBodyWrapper<CustomResponseBody, CustomBodyDecodingStrategy> var customBody
    }
}

Note: Due to technical limitations of the Codable protocol in Swift, it is currently not possible to have a non-static/dynamic decoding strategy.

Response headers

Use the property wrapper @ResponseHeader<Strategy> inside the response type.

In the moment, the following decoding strategies are implemented:

  • DefaultHeaderStrategy

Converts the property name into camel-case format (e.g. Content-Type becomes contentType) and compares case-insensitive (e.g. Authorization equals authorization) This strategy expects the response header to be set, otherwise an error will be thrown.

Response from URL requests are always of type String and no casting will be performed. Therefore the only valid property type is String.

  • DefaultHeaderOptionalStrategy

Same as DefaultHeaderStrategy but won't fail if the header can not be found.

Example:

struct Response: Decodable {

    @ResponseHeader<DefaultHeaderStrategy>
    var authorization: String

    @ResponseHeader<DefaultHeaderStrategy>
    var contentType: String

    @ResponseHeader<DefaultHeaderStrategyOptional>
    var optionalValue: String?

}

Response Status

The default HTTP method is GET, but it can be overwritten by adding an instance property with the property wrapper @HTTPMethod:

Example:

struct Response: Decodable {

    @ResponseStatusCode var statusCode

}

Note:

Multiple properties can be declared with this property wrapper. All of them will have the value set.

Nested Responses

To support inheritance, which can be especially useful for pagination, use the property wrapper @NestedResponse to add nested responses.

While decoding the flat HTTP response will be applied recursively to all nested responses, therefore it is possible, that different nested responses access different values of the original HTTP response.

Example:

struct PaginatedResponse<NestedRequest: Request>: Decodable {

    /// Header which indicates how many more elements are available
    @ResponseHeader<DefaultHeaderStrategy> var totalElements

    @NestedResponse var nested: NestedRequest
}

struct ListRequest: Request {

    typealias Response = PaginatedResponse<ListResponse>

    struct ListResponse: Decodable {
        // see other examples
    }
}

HTTP API Client

The easiest way of sending CodableRequest requests, is using the CodableURLSession which takes care of encoding requests, and decoding responses.

All it takes to create a client, is the URL which is used as a base for all requests. Afterwards you can just send the requests, either using Async-Await, Combine publishers, or classic callbacks.

Additionally the CodableURLSession provides the option of setting a session provider, which encapsulates the default URLSession by a protocol. This allows to create networking clients which can be mocked (perfect for unit testing).

Async Await

Example:

let url: URL = ...
let client = CodableURLSession(baseURL: url)

// ... create request ...

try {
    let response = try await client.send(request)
    // process response
    print(response)
} catch {
    // handle error
}

You can also specify queues to send and reveive requests on:

let url: URL = ...
let client = CodableURLSession(baseURL: url)

// ... create request ...

try {
    let response = try await client.send(request, on: .global(qos: .default), receiveOn: .main)
    
    /*  
        You can also specify queues to send and reveive on  
            try await client.send(request, on: .global(qos: .default), receiveOn: .main) 
    */
    
    // process response
    print(response)
} catch {
    // handle error
}

Combine

Example:

let url: URL = ...
let client = CodableURLSession(baseURL: url)

// ... create request ...

client.send(request)
    .sink(receiveCompletion: { completion in
        switch completion {
        case .failure(let error):
            // handle error
            break
        case .finished:
            break
        }
    }, receiveValue: { response in
        // process response
        print(response)
    })
    .store(in: &cancellables)

Callback

Example:

let url: URL = ...
let client = CodableURLSession(baseURL: url)

// ... create request ...

client.send(request) { result in
    switch result {
    case .failure(let error):
        // handle error
        break
    case .finished(let response):
        // process response
        break
    }
}

Cookies

By default the cookies of requests and responses are handled by the session used by the CodableURLSession. If you want to explicitly set the request cookies, use Cookies, and to access the response cookies use ResponseCookies.

Example:

struct MyRequest: Request {
    struct Response: Decodable {
        // List of HTTPCookie parsed from the `Set-Cookie` headers of the response
        @ResponseCookies var cookies
    }

    // List of HTTPCookie to be set in the request as `Cookie` headers
    @Cookies var cookies
}

Encoding & Decoding

The RequestEncoder is responsible to turn an encodable Request into an URLRequest. It requires an URL in the initializer, as CodableRequest requests are relative requests.

Example:

// A request as explained above
let request: Request = ...

// Create a request encoder
let url = URL(string: "http://techprimate.com")
let encoder = RequestEncoder(baseURL: url)

// Encode request
let urlRequest: URLRequest
do {
    let urlRequest = try encoder.encode(request)
    // continue with url request
    ...
} catch {
    // Handle error
    ...
}

As its contrarity component, the RequestDecoder is responsible to turn a tuple of (data: Data, response: HTTPURLResponse) into a given type Response.

Example:

// Data received from the URL session task
let response: HTTPURLResponse = ...
let data: Data = ...

// Create decoder
let decoder = ResponseDecoder()
do {
    let decoded = try decoder.decode(Response.self, from: (data, response)))
    // continue with decoded response
    ...
} catch{
    // Handle error
    ...
}

Combine Support

RequestEncoder conforms to TopLevelEncoder and RequestDecoder conforms to TopLevelDecoder. This means both encoders can be used in a Combine pipeline.

Example:

let request = Request()
let session = URLSession.shared

let url = URL(string: "https://techprimate.com")!
let encodedRequest = try RequestEncoder(baseURL: url).encode(request)

// Send request using the given URL session provider
return session
    .dataTaskPublisher(for: encodedRequest)
    .tryMap { (data: Data, response: URLResponse) in
        guard let response = response as? HTTPURLResponse else {
            fatalError("handle non HTTP url responses")
        }
        return (data: data, response: response)
    }
    .decode(type: Request.Response.self, decoder: ResponseDecoder())
    .sink(receiveCompletion: { result in
        // handle result
    }, receiveValue: { decoded in
        // do something with decoded response
    })

License

MIT License

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.

CodableRequest is created and maintained by Alex Nazarov. Based on Postie, created by Philip Niedertscheider at kula.app and all the amazing contributors.