From 8f9d89b5eeaaef672df9ec352f941986ad164b0a Mon Sep 17 00:00:00 2001 From: fumoboy007 <2100868+fumoboy007@users.noreply.github.com> Date: Wed, 3 Jan 2024 13:00:35 -0800 Subject: [PATCH] Initial commit. --- .github/workflows/documentation.yml | 47 ++++++++++ .github/workflows/tests.yml | 35 ++++++++ .gitignore | 9 ++ LICENSE.md | 9 ++ Package.resolved | 50 +++++++++++ Package.swift | 61 +++++++++++++ README.md | 37 ++++++++ .../RetryHTTPRequestUsingURLSession.swift | 16 ++++ .../CustomHTTPResponseInterpretation.swift | 25 ++++++ .../DefaultHTTPResponseInterpretation.swift | 22 +++++ ...HTTPApplicationErrorWithResponseBody.swift | 40 +++++++++ .../Retrying HTTP Requests/BasicRetry.swift | 24 ++++++ .../ConfigureRetryBehavior.swift | 30 +++++++ .../ReuseRetryConfiguration.swift | 39 +++++++++ .../UseFakeClockType.swift | 86 +++++++++++++++++++ .../Error/HTTPApplicationError.swift | 65 ++++++++++++++ .../Error/HTTPApplicationErrorProtocol.swift | 45 ++++++++++ .../HTTPErrorHandling.md | 33 +++++++ ...operability With Popular HTTP Libraries.md | 9 ++ .../Interpreting HTTP Responses.md | 13 +++ .../Retrying HTTP Requests.md | 17 ++++ .../HTTPResponse+Interpretation.swift | 82 ++++++++++++++++++ .../HTTPResponseStatus+Interpretation.swift | 42 +++++++++ .../Retry/HTTPRequest+RetryableRequest.swift | 57 ++++++++++++ .../HTTPRequestMethod+IsIdempotent.swift | 50 +++++++++++ .../DummyFile.swift | 21 +++++ .../Fakes/ClockFake.swift | 66 ++++++++++++++ .../Fakes/ErrorFake.swift | 24 ++++++ .../Fakes/HTTPRequest+Fakes.swift | 30 +++++++ .../Fakes/HTTPResponse+Fakes.swift | 31 +++++++ .../HTTPRequestRetryTests.swift | 68 +++++++++++++++ .../HTTPResponseInterpretationTests.swift | 80 +++++++++++++++++ 32 files changed, 1263 insertions(+) create mode 100644 .github/workflows/documentation.yml create mode 100644 .github/workflows/tests.yml create mode 100644 .gitignore create mode 100644 LICENSE.md create mode 100644 Package.resolved create mode 100644 Package.swift create mode 100644 README.md create mode 100644 Snippets/Interoperability With Popular HTTP Libraries/RetryHTTPRequestUsingURLSession.swift create mode 100644 Snippets/Interpreting HTTP Responses/CustomHTTPResponseInterpretation.swift create mode 100644 Snippets/Interpreting HTTP Responses/DefaultHTTPResponseInterpretation.swift create mode 100644 Snippets/Interpreting HTTP Responses/HTTPApplicationErrorWithResponseBody.swift create mode 100644 Snippets/Retrying HTTP Requests/BasicRetry.swift create mode 100644 Snippets/Retrying HTTP Requests/ConfigureRetryBehavior.swift create mode 100644 Snippets/Retrying HTTP Requests/ReuseRetryConfiguration.swift create mode 100644 Snippets/Retrying HTTP Requests/UseFakeClockType.swift create mode 100644 Sources/HTTPErrorHandling/Error/HTTPApplicationError.swift create mode 100644 Sources/HTTPErrorHandling/Error/HTTPApplicationErrorProtocol.swift create mode 100644 Sources/HTTPErrorHandling/HTTPErrorHandling.docc/HTTPErrorHandling.md create mode 100644 Sources/HTTPErrorHandling/HTTPErrorHandling.docc/Interoperability With Popular HTTP Libraries.md create mode 100644 Sources/HTTPErrorHandling/HTTPErrorHandling.docc/Interpreting HTTP Responses.md create mode 100644 Sources/HTTPErrorHandling/HTTPErrorHandling.docc/Retrying HTTP Requests.md create mode 100644 Sources/HTTPErrorHandling/Interpretation/HTTPResponse+Interpretation.swift create mode 100644 Sources/HTTPErrorHandling/Interpretation/HTTPResponseStatus+Interpretation.swift create mode 100644 Sources/HTTPErrorHandling/Retry/HTTPRequest+RetryableRequest.swift create mode 100644 Sources/HTTPErrorHandling/Retry/HTTPRequestMethod+IsIdempotent.swift create mode 100644 Sources/_AdditionalSnippetDependencies/DummyFile.swift create mode 100644 Tests/HTTPErrorHandlingTests/Fakes/ClockFake.swift create mode 100644 Tests/HTTPErrorHandlingTests/Fakes/ErrorFake.swift create mode 100644 Tests/HTTPErrorHandlingTests/Fakes/HTTPRequest+Fakes.swift create mode 100644 Tests/HTTPErrorHandlingTests/Fakes/HTTPResponse+Fakes.swift create mode 100644 Tests/HTTPErrorHandlingTests/HTTPRequestRetryTests.swift create mode 100644 Tests/HTTPErrorHandlingTests/HTTPResponseInterpretationTests.swift diff --git a/.github/workflows/documentation.yml b/.github/workflows/documentation.yml new file mode 100644 index 0000000..8bb8a14 --- /dev/null +++ b/.github/workflows/documentation.yml @@ -0,0 +1,47 @@ +name: Documentation + +on: + push: + branches: + - main + +permissions: + contents: read + pages: write + id-token: write + +# Allow only one concurrent deployment, skipping runs queued between the run in-progress and latest queued. +# However, do NOT cancel in-progress runs as we want to allow these production deployments to complete. +concurrency: + group: "pages" + cancel-in-progress: false + +jobs: + publish: + name: Publish Documentation + # TODO: Use `macos-latest` after the macOS 13 image graduates to GA. + # https://github.com/actions/runner-images/issues/7508#issuecomment-1718206371 + runs-on: macos-13 + + environment: + name: github-pages + url: ${{ steps.deployment.outputs.page_url }} + + steps: + - name: Set up GitHub Pages + uses: actions/configure-pages@v3 + - uses: maxim-lobanov/setup-xcode@v1 + with: + xcode-version: latest-stable + - name: Print Swift compiler version + run: "swift --version" + - uses: actions/checkout@v3 + - name: Generate documentation + run: "swift package generate-documentation --target HTTPErrorHandling --disable-indexing --include-extended-types --transform-for-static-hosting --hosting-base-path swift-http-error-handling" + - name: Upload documentation + uses: actions/upload-pages-artifact@v2 + with: + path: ".build/plugins/Swift-DocC/outputs/HTTPErrorHandling.doccarchive" + - name: Deploy to GitHub Pages + id: deployment + uses: actions/deploy-pages@v2 diff --git a/.github/workflows/tests.yml b/.github/workflows/tests.yml new file mode 100644 index 0000000..df7389e --- /dev/null +++ b/.github/workflows/tests.yml @@ -0,0 +1,35 @@ +name: Tests + +on: [push] + +# TODO: Add Windows job after Swift is added to the Windows images [1] or after +# `swift-actions/setup-swift` supports Swift 5.9+ on Windows [2]. +# 1. https://github.com/actions/runner-images/issues/8281 +# 2. https://github.com/swift-actions/setup-swift/pull/470#issuecomment-1718406382 +jobs: + test-macos: + name: Run Tests on macOS + # TODO: Use `macos-latest` after the macOS 13 image graduates to GA. + # https://github.com/actions/runner-images/issues/7508#issuecomment-1718206371 + runs-on: macos-13 + + steps: + - uses: maxim-lobanov/setup-xcode@v1 + with: + xcode-version: latest-stable + - name: Print Swift compiler version + run: "swift --version" + - uses: actions/checkout@v3 + - name: Run tests + run: "swift test --parallel" + + test-linux: + name: Run Tests on Linux + runs-on: ubuntu-latest + + steps: + - name: Print Swift compiler version + run: "swift --version" + - uses: actions/checkout@v3 + - name: Run tests + run: "swift test --parallel" diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..3b29812 --- /dev/null +++ b/.gitignore @@ -0,0 +1,9 @@ +.DS_Store +/.build +/Packages +/*.xcodeproj +xcuserdata/ +DerivedData/ +.swiftpm/config/registries.json +.swiftpm/xcode/package.xcworkspace/contents.xcworkspacedata +.netrc diff --git a/LICENSE.md b/LICENSE.md new file mode 100644 index 0000000..fae353c --- /dev/null +++ b/LICENSE.md @@ -0,0 +1,9 @@ +# MIT License + +Copyright © 2024 Darren Mo. + +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. diff --git a/Package.resolved b/Package.resolved new file mode 100644 index 0000000..924c8d1 --- /dev/null +++ b/Package.resolved @@ -0,0 +1,50 @@ +{ + "pins" : [ + { + "identity" : "swift-docc-plugin", + "kind" : "remoteSourceControl", + "location" : "https://github.com/apple/swift-docc-plugin.git", + "state" : { + "revision" : "26ac5758409154cc448d7ab82389c520fa8a8247", + "version" : "1.3.0" + } + }, + { + "identity" : "swift-docc-symbolkit", + "kind" : "remoteSourceControl", + "location" : "https://github.com/apple/swift-docc-symbolkit", + "state" : { + "revision" : "b45d1f2ed151d057b54504d653e0da5552844e34", + "version" : "1.0.0" + } + }, + { + "identity" : "swift-http-types", + "kind" : "remoteSourceControl", + "location" : "https://github.com/apple/swift-http-types.git", + "state" : { + "revision" : "1827dc94bdab2eb5f2fc804e9b0cb43574282566", + "version" : "1.0.2" + } + }, + { + "identity" : "swift-log", + "kind" : "remoteSourceControl", + "location" : "https://github.com/apple/swift-log.git", + "state" : { + "revision" : "532d8b529501fb73a2455b179e0bbb6d49b652ed", + "version" : "1.5.3" + } + }, + { + "identity" : "swift-retry", + "kind" : "remoteSourceControl", + "location" : "https://github.com/fumoboy007/swift-retry.git", + "state" : { + "revision" : "bd3f43ed9deaa0d296928e8bf0f02dcbd935fe14", + "version" : "0.1.3" + } + } + ], + "version" : 2 +} diff --git a/Package.swift b/Package.swift new file mode 100644 index 0000000..e64160f --- /dev/null +++ b/Package.swift @@ -0,0 +1,61 @@ +// swift-tools-version: 5.9 + +import PackageDescription + +let package = Package( + name: "swift-http-error-handling", + platforms: [ + .visionOS(.v1), + .macOS(.v13), + .macCatalyst(.v16), + .iOS(.v16), + .tvOS(.v16), + .watchOS(.v9), + ], + products: [ + .library( + name: "HTTPErrorHandling", + targets: [ + "HTTPErrorHandling", + ] + ), + // According to SE-0356, Swift Package Manager does not yet officially support snippet-only dependencies. + // This library product and the corresponding target work around that limitation. The product name is + // prefixed with an underscore to convey that the product was not meant to be externally visible. + .library( + name: "_AdditionalSnippetDependencies", + targets: [ + "_AdditionalSnippetDependencies", + ] + ), + ], + dependencies: [ + .package(url: "https://github.com/apple/swift-docc-plugin.git", from: "1.3.0"), + .package(url: "https://github.com/apple/swift-http-types.git", from: "1.0.2"), + .package(url: "https://github.com/apple/swift-log.git", from: "1.5.3"), + .package(url: "https://github.com/fumoboy007/swift-retry.git", .upToNextMinor(from: "0.1.3")), + ], + targets: [ + .target( + name: "HTTPErrorHandling", + dependencies: [ + .product(name: "DMRetry", package: "swift-retry"), + .product(name: "HTTPTypes", package: "swift-http-types"), + ] + ), + .testTarget( + name: "HTTPErrorHandlingTests", + dependencies: [ + "HTTPErrorHandling", + .product(name: "HTTPTypes", package: "swift-http-types"), + ] + ), + .target( + name: "_AdditionalSnippetDependencies", + dependencies: [ + .product(name: "HTTPTypesFoundation", package: "swift-http-types"), + .product(name: "Logging", package: "swift-log"), + ] + ), + ] +) diff --git a/README.md b/README.md new file mode 100644 index 0000000..d20052d --- /dev/null +++ b/README.md @@ -0,0 +1,37 @@ +# swift-http-error-handling + +Interpret HTTP responses and handle failures in Swift. + +![Swift 5.9](https://img.shields.io/badge/swift-v5.9-%23F05138) +![Linux, visionOS 1, macOS 13, iOS 16, tvOS 16, watchOS 9](https://img.shields.io/badge/platform-Linux%20%7C%20visionOS%201%20%7C%20macOS%2013%20%7C%20iOS%2016%20%7C%20tvOS%2016%20%7C%20watchOS%209-blue) +![MIT License](https://img.shields.io/github/license/fumoboy007/swift-http-error-handling) +![Automated Tests Workflow Status](https://img.shields.io/github/actions/workflow/status/fumoboy007/swift-http-error-handling/tests.yml?event=push&label=tests) + +## Overview + +In the HTTP protocol, a client sends a request to a server and the server sends a response back to the client. The response contains a [status code](https://httpwg.org/specs/rfc9110.html#overview.of.status.codes) to help the client interpret the response. + +HTTP libraries like `Foundation` pass the response through to the caller without interpreting the response as a success or failure. `HTTPErrorHandling` can help the caller interpret HTTP responses and handle failures. + +The module works with any HTTP library that is compatible with Swift’s [standard HTTP request and response types](https://github.com/apple/swift-http-types). The module can be used on its own in code that directly uses an HTTP library, or the module can be used as a building block by higher-level networking libraries. + +## Example Usage + +```swift +import Foundation +import HTTPErrorHandling +import HTTPTypes +import HTTPTypesFoundation + +let request = HTTPRequest(method: .get, + scheme: "https", + authority: "example.com", + path: "/") + +try await request.retry { request in + let (_, response) = try await URLSession.shared.data(for: request) + try response.throwIfFailed() +} +``` + +See the [documentation](https://fumoboy007.github.io/swift-http-error-handling/documentation/httperrorhandling/) for more examples. diff --git a/Snippets/Interoperability With Popular HTTP Libraries/RetryHTTPRequestUsingURLSession.swift b/Snippets/Interoperability With Popular HTTP Libraries/RetryHTTPRequestUsingURLSession.swift new file mode 100644 index 0000000..851b60a --- /dev/null +++ b/Snippets/Interoperability With Popular HTTP Libraries/RetryHTTPRequestUsingURLSession.swift @@ -0,0 +1,16 @@ +// Retry an HTTP request using `URLSession`. + +import Foundation +import HTTPErrorHandling +import HTTPTypes +import HTTPTypesFoundation + +let request = HTTPRequest(method: .get, + scheme: "https", + authority: "example.com", + path: "/") + +try await request.retry { request in + let (_, response) = try await URLSession.shared.data(for: request) + try response.throwIfFailed() +} diff --git a/Snippets/Interpreting HTTP Responses/CustomHTTPResponseInterpretation.swift b/Snippets/Interpreting HTTP Responses/CustomHTTPResponseInterpretation.swift new file mode 100644 index 0000000..cd1b7db --- /dev/null +++ b/Snippets/Interpreting HTTP Responses/CustomHTTPResponseInterpretation.swift @@ -0,0 +1,25 @@ +// Customize the interpretation of the HTTP response status. + +// snippet.hide + +import HTTPErrorHandling +import HTTPTypes + +let request = HTTPRequest(method: .post, + scheme: "https", + authority: "example.com", + path: "/") + +// snippet.show + +let response = try await perform(request) +try response.throwIfFailed( + successStatuses: [.created], + transientFailureStatuses: .commonTransientFailureStatuses.union([.conflict]) +) + +// snippet.hide + +func perform(_ request: HTTPRequest) async throws -> HTTPResponse { + return HTTPResponse(status: .ok) +} diff --git a/Snippets/Interpreting HTTP Responses/DefaultHTTPResponseInterpretation.swift b/Snippets/Interpreting HTTP Responses/DefaultHTTPResponseInterpretation.swift new file mode 100644 index 0000000..300d327 --- /dev/null +++ b/Snippets/Interpreting HTTP Responses/DefaultHTTPResponseInterpretation.swift @@ -0,0 +1,22 @@ +// Throw an error if the HTTP response represents a failure. + +// snippet.hide + +import HTTPErrorHandling +import HTTPTypes + +let request = HTTPRequest(method: .get, + scheme: "https", + authority: "example.com", + path: "/") + +// snippet.show + +let response = try await perform(request) +try response.throwIfFailed() + +// snippet.hide + +func perform(_ request: HTTPRequest) async throws -> HTTPResponse { + return HTTPResponse(status: .ok) +} diff --git a/Snippets/Interpreting HTTP Responses/HTTPApplicationErrorWithResponseBody.swift b/Snippets/Interpreting HTTP Responses/HTTPApplicationErrorWithResponseBody.swift new file mode 100644 index 0000000..ece0542 --- /dev/null +++ b/Snippets/Interpreting HTTP Responses/HTTPApplicationErrorWithResponseBody.swift @@ -0,0 +1,40 @@ +// Attach the response body to ``HTTPApplicationError`` and access it later. + +// snippet.hide + +import Foundation +import HTTPErrorHandling +import HTTPTypes + +let request = HTTPRequest(method: .get, + scheme: "https", + authority: "example.com", + path: "/") + +// snippet.show + +do { + let (responseBody, response) = try await perform(request) + try await response.throwIfFailed { + return try await deserializeFailureDetails(from: responseBody) + } +} catch let error as HTTPApplicationError { + let failureDetails = error.responseBody + doSomething(with: failureDetails) +} + +// snippet.hide + +func perform(_ request: HTTPRequest) async throws -> (Data, HTTPResponse) { + return (Data(), HTTPResponse(status: .ok)) +} + +struct MyFailureDetails { +} + +func deserializeFailureDetails(from responseBody: Data) async throws -> MyFailureDetails { + return MyFailureDetails() +} + +func doSomething(with failureDetails: MyFailureDetails) { +} diff --git a/Snippets/Retrying HTTP Requests/BasicRetry.swift b/Snippets/Retrying HTTP Requests/BasicRetry.swift new file mode 100644 index 0000000..f606277 --- /dev/null +++ b/Snippets/Retrying HTTP Requests/BasicRetry.swift @@ -0,0 +1,24 @@ +// Retry an HTTP request. + +// snippet.hide + +import HTTPErrorHandling +import HTTPTypes + +let request = HTTPRequest(method: .get, + scheme: "https", + authority: "example.com", + path: "/") + +// snippet.show + +try await request.retry { request in + let response = try await perform(request) + try response.throwIfFailed() +} + +// snippet.hide + +func perform(_ request: HTTPRequest) async throws -> HTTPResponse { + return HTTPResponse(status: .ok) +} diff --git a/Snippets/Retrying HTTP Requests/ConfigureRetryBehavior.swift b/Snippets/Retrying HTTP Requests/ConfigureRetryBehavior.swift new file mode 100644 index 0000000..a1f6c4f --- /dev/null +++ b/Snippets/Retrying HTTP Requests/ConfigureRetryBehavior.swift @@ -0,0 +1,30 @@ +// Configure the retry behavior. + +// snippet.hide + +import HTTPErrorHandling +import HTTPTypes +import Logging + +let request = HTTPRequest(method: .get, + scheme: "https", + authority: "example.com", + path: "/") + +// snippet.show + +try await request.retry(maxAttempts: 5, + backoff: .default(baseDelay: .milliseconds(500), + maxDelay: .seconds(10)), + logger: myLogger) { request in + let response = try await perform(request) + try response.throwIfFailed() +} + +// snippet.hide + +let myLogger = Logger(label: "Example Code") + +func perform(_ request: HTTPRequest) async throws -> HTTPResponse { + return HTTPResponse(status: .ok) +} diff --git a/Snippets/Retrying HTTP Requests/ReuseRetryConfiguration.swift b/Snippets/Retrying HTTP Requests/ReuseRetryConfiguration.swift new file mode 100644 index 0000000..8d045e4 --- /dev/null +++ b/Snippets/Retrying HTTP Requests/ReuseRetryConfiguration.swift @@ -0,0 +1,39 @@ +// Encapsulate retry behavior in a ``RetryConfiguration`` instance. + +// snippet.hide + +import HTTPErrorHandling +import HTTPTypes +import Logging +import Retry + +let request = HTTPRequest(method: .get, + scheme: "https", + authority: "example.com", + path: "/") + +// snippet.show + +extension RetryConfiguration { + static let standard = RetryConfiguration() + + static let highTolerance = ( + Self.standard + .withMaxAttempts(10) + .withBackoff(.default(baseDelay: .seconds(1), + maxDelay: nil)) + ) +} + +try await request.retry(with: .highTolerance.withLogger(myLogger)) { request in + let response = try await perform(request) + try response.throwIfFailed() +} + +// snippet.hide + +let myLogger = Logger(label: "Example Code") + +func perform(_ request: HTTPRequest) async throws -> HTTPResponse { + return HTTPResponse(status: .ok) +} diff --git a/Snippets/Retrying HTTP Requests/UseFakeClockType.swift b/Snippets/Retrying HTTP Requests/UseFakeClockType.swift new file mode 100644 index 0000000..a52951c --- /dev/null +++ b/Snippets/Retrying HTTP Requests/UseFakeClockType.swift @@ -0,0 +1,86 @@ +// Use a fake `Clock` type for deterministic and efficient automated tests. + +// snippet.hide + +import Foundation +import HTTPErrorHandling +import HTTPTypes +import XCTest + +// snippet.show + +final class MyServiceImplementation where ClockType.Duration == Duration { + private let clock: ClockType + + init(clock: ClockType) { + self.clock = clock + } + + func reliablyPerform(_ request: HTTPRequest) async throws { + try await request.retry(clock: clock) { request in + let response = try await perform(request) + try response.throwIfFailed() + } + } +} + +final class MyServiceImplementationTests: XCTestCase { + func testReliablyPerformRequest_succeeds() async throws { + let myService = MyServiceImplementation(clock: ClockFake()) + try await myService.reliablyPerform(requestFake) + } +} + +// snippet.hide + +class ClockFake: Clock, @unchecked Sendable { + typealias Instant = ContinuousClock.Instant + + private let realClock = ContinuousClock() + + private let lock = NSLock() + + init() { + self._now = realClock.now + } + + private var _now: Instant + var now: Instant { + lock.lock() + defer { + lock.unlock() + } + + return _now + } + + var minimumResolution: Duration { + return realClock.minimumResolution + } + + func sleep(until deadline: Instant, + tolerance: Duration?) async throws { + // Refactored into a non-async method so that `NSLock.lock` and `NSLock.unlock` can be used. + // Cannot use the async-safe `NSLock.withLocking` method until the following change is released: + // https://github.com/apple/swift-corelibs-foundation/pull/4736 + sleep(until: deadline) + } + + private func sleep(until deadline: Instant) { + lock.lock() + defer { + lock.unlock() + } + + _now = max(deadline, _now) + } +} + +let requestFake = HTTPRequest(method: .get, + scheme: "https", + authority: "example.com", + path: "/") + +func perform(_ request: HTTPRequest) async throws -> HTTPResponse { + return HTTPResponse(status: .ok) +} diff --git a/Sources/HTTPErrorHandling/Error/HTTPApplicationError.swift b/Sources/HTTPErrorHandling/Error/HTTPApplicationError.swift new file mode 100644 index 0000000..e81960b --- /dev/null +++ b/Sources/HTTPErrorHandling/Error/HTTPApplicationError.swift @@ -0,0 +1,65 @@ +// MIT License +// +// Copyright © 2024 Darren Mo. +// +// 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. + +import HTTPTypes + +/// A concrete error type that contains details about an HTTP application failure. +public struct HTTPApplicationError: Error { + /// The response, which represents a failed request. + public let response: HTTPResponse + + /// The response body. + /// + /// The server may return additional information about the failure in the response body. + /// + /// - SeeAlso: ``HTTPApplicationErrorProtocol/anyResponseBody`` + public let responseBody: ResponseBodyType + + /// Whether the failure is transient. + /// + /// If the failure is transient, a subsequent attempt may succeed. If the failure is not transient, subsequent + /// attempts will never succeed. + public let isTransient: Bool + + /// Initializes the error. + /// + /// - Parameters: + /// - response: The response, which represents a failed request. + /// - responseBody: The response body. The server may return additional information about the + /// failure in the response body. + /// - isTransient: Whether the failure is transient. If the failure is transient, a subsequent + /// attempt may succeed. If the failure is not transient, subsequent attempts will never succeed. + public init(response: HTTPResponse, + responseBody: ResponseBodyType, + isTransient: Bool) { + self.response = response + self.responseBody = responseBody + + self.isTransient = isTransient + } +} + +extension HTTPApplicationError: HTTPApplicationErrorProtocol { + public var anyResponseBody: Any { + return responseBody + } +} diff --git a/Sources/HTTPErrorHandling/Error/HTTPApplicationErrorProtocol.swift b/Sources/HTTPErrorHandling/Error/HTTPApplicationErrorProtocol.swift new file mode 100644 index 0000000..4cbfd18 --- /dev/null +++ b/Sources/HTTPErrorHandling/Error/HTTPApplicationErrorProtocol.swift @@ -0,0 +1,45 @@ +// MIT License +// +// Copyright © 2024 Darren Mo. +// +// 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. + +import HTTPTypes + +/// A protocol that all specialized ``HTTPApplicationError`` types conform to. +/// +/// ``HTTPApplicationError`` has a generic type parameter. This protocol can be used to match any +/// ``HTTPApplicationError`` instance, regardless of its generic type parameter. +public protocol HTTPApplicationErrorProtocol: Error { + /// The response, which represents a failed request. + var response: HTTPResponse { get } + + /// The response body. + /// + /// The server may return additional information about the failure in the response body. The property type + /// is `Any` because the response body type is application-specific. Consider using + /// ``HTTPApplicationError/responseBody`` when the response body type is known. + var anyResponseBody: Any { get } + + /// Whether the failure is transient. + /// + /// If the failure is transient, a subsequent attempt may succeed. If the failure is not transient, subsequent + /// attempts will never succeed. + var isTransient: Bool { get } +} diff --git a/Sources/HTTPErrorHandling/HTTPErrorHandling.docc/HTTPErrorHandling.md b/Sources/HTTPErrorHandling/HTTPErrorHandling.docc/HTTPErrorHandling.md new file mode 100644 index 0000000..5d7b542 --- /dev/null +++ b/Sources/HTTPErrorHandling/HTTPErrorHandling.docc/HTTPErrorHandling.md @@ -0,0 +1,33 @@ +# ``HTTPErrorHandling`` + +Interpret HTTP responses and handle failures. + +## Overview + +In the HTTP protocol, a client sends a request to a server and the server sends a response back to the client. The response contains a [status code](https://httpwg.org/specs/rfc9110.html#overview.of.status.codes) to help the client interpret the response. + +HTTP libraries like `Foundation` pass the response through to the caller without interpreting the response as a success or failure. `HTTPErrorHandling` can help the caller interpret HTTP responses and handle failures. + +The module works with any HTTP library that is compatible with Swift’s [standard HTTP request and response types](https://github.com/apple/swift-http-types). The module can be used on its own in code that directly uses an HTTP library, or the module can be used as a building block by higher-level networking libraries. + +## Topics + +### Examples + +- +- +- + +### Representing an HTTP Application Failure + +- ``HTTPApplicationError`` +- ``HTTPApplicationErrorProtocol`` + +### Interpreting HTTP Responses + +- ``HTTPTypes/HTTPResponse`` +- ``Swift/Set`` + +### Retrying HTTP Requests + +- ``HTTPTypes/HTTPRequest`` diff --git a/Sources/HTTPErrorHandling/HTTPErrorHandling.docc/Interoperability With Popular HTTP Libraries.md b/Sources/HTTPErrorHandling/HTTPErrorHandling.docc/Interoperability With Popular HTTP Libraries.md new file mode 100644 index 0000000..9c10239 --- /dev/null +++ b/Sources/HTTPErrorHandling/HTTPErrorHandling.docc/Interoperability With Popular HTTP Libraries.md @@ -0,0 +1,9 @@ +# Interoperability With Popular HTTP Libraries + +## Foundation + +@Snippet(path: "swift-http-error-handling/Snippets/Interoperability With Popular HTTP Libraries/RetryHTTPRequestUsingURLSession") + +## AsyncHTTPClient + +TODO: Add an example after `AsyncHTTPClient` [adopts](https://github.com/swift-server/async-http-client/issues/708#issuecomment-1871343870) `swift-http-types`. diff --git a/Sources/HTTPErrorHandling/HTTPErrorHandling.docc/Interpreting HTTP Responses.md b/Sources/HTTPErrorHandling/HTTPErrorHandling.docc/Interpreting HTTP Responses.md new file mode 100644 index 0000000..c1a3815 --- /dev/null +++ b/Sources/HTTPErrorHandling/HTTPErrorHandling.docc/Interpreting HTTP Responses.md @@ -0,0 +1,13 @@ +# Interpreting HTTP Responses + +## Default Interpretation + +@Snippet(path: "swift-http-error-handling/Snippets/Interpreting HTTP Responses/DefaultHTTPResponseInterpretation") + +## Custom Interpretation + +@Snippet(path: "swift-http-error-handling/Snippets/Interpreting HTTP Responses/CustomHTTPResponseInterpretation") + +## Attaching the Response Body to the Error + +@Snippet(path: "swift-http-error-handling/Snippets/Interpreting HTTP Responses/HTTPApplicationErrorWithResponseBody") diff --git a/Sources/HTTPErrorHandling/HTTPErrorHandling.docc/Retrying HTTP Requests.md b/Sources/HTTPErrorHandling/HTTPErrorHandling.docc/Retrying HTTP Requests.md new file mode 100644 index 0000000..c133300 --- /dev/null +++ b/Sources/HTTPErrorHandling/HTTPErrorHandling.docc/Retrying HTTP Requests.md @@ -0,0 +1,17 @@ +# Retrying HTTP Requests + +## Basic Retry + +@Snippet(path: "swift-http-error-handling/Snippets/Retrying HTTP Requests/BasicRetry") + +## Configuring the Retry Behavior + +@Snippet(path: "swift-http-error-handling/Snippets/Retrying HTTP Requests/ConfigureRetryBehavior") + +## Reusing a Configuration + +@Snippet(path: "swift-http-error-handling/Snippets/Retrying HTTP Requests/ReuseRetryConfiguration") + +## Using a Fake Clock Type For Automated Tests + +@Snippet(path: "swift-http-error-handling/Snippets/Retrying HTTP Requests/UseFakeClockType") diff --git a/Sources/HTTPErrorHandling/Interpretation/HTTPResponse+Interpretation.swift b/Sources/HTTPErrorHandling/Interpretation/HTTPResponse+Interpretation.swift new file mode 100644 index 0000000..01950b5 --- /dev/null +++ b/Sources/HTTPErrorHandling/Interpretation/HTTPResponse+Interpretation.swift @@ -0,0 +1,82 @@ +// MIT License +// +// Copyright © 2024 Darren Mo. +// +// 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. + +import HTTPTypes + +/// Adds methods to interpret the response and throw ``HTTPApplicationError`` if the response +/// represents an application failure. +extension HTTPResponse { + /// Throws ``HTTPApplicationError`` if the response represents an application failure. + /// + /// - Parameters: + /// - successStatuses: The HTTP response statuses that indicate success. + /// - transientFailuresStatuses: The HTTP response statuses that indicate the failure is transient. + /// If the failure is transient, a subsequent attempt may succeed. If the failure is not transient, subsequent + /// attempts will never succeed. + /// + /// - SeeAlso: ``throwIfFailed(successStatuses:transientFailureStatuses:makeResponseBody:)`` + public func throwIfFailed( + successStatuses: Set = .commonSuccessStatuses, + transientFailureStatuses: Set = .commonTransientFailureStatuses + ) throws { + precondition(successStatuses.isDisjoint(with: transientFailureStatuses)) + + guard successStatuses.contains(status) else { + let isTransientFailure = transientFailureStatuses.contains(status) + + throw HTTPApplicationError(response: self, + responseBody: (), + isTransient: isTransientFailure) + } + } + + /// Throws ``HTTPApplicationError`` if the response represents an application failure. The error instance + /// includes the response body. + /// + /// - Parameters: + /// - successStatuses: The HTTP response statuses that indicate success. + /// - transientFailuresStatuses: The HTTP response statuses that indicate the failure is transient. + /// If the failure is transient, a subsequent attempt may succeed. If the failure is not transient, subsequent + /// attempts will never succeed. + /// - makeResponseBody: A closure that returns the response body, which can be accessed via + /// ``HTTPApplicationError/responseBody``. The closure is `async` because the response + /// body may not yet have been fully received. + /// + /// - SeeAlso: ``throwIfFailed(successStatuses:transientFailureStatuses:)`` + public func throwIfFailed( + successStatuses: Set = .commonSuccessStatuses, + transientFailureStatuses: Set = .commonTransientFailureStatuses, + @_inheritActorContext @_implicitSelfCapture makeResponseBody: () async throws -> ResponseBodyType + ) async throws { + precondition(successStatuses.isDisjoint(with: transientFailureStatuses)) + + guard successStatuses.contains(status) else { + let responseBody = try await makeResponseBody() + + let isTransientFailure = transientFailureStatuses.contains(status) + + throw HTTPApplicationError(response: self, + responseBody: responseBody, + isTransient: isTransientFailure) + } + } +} diff --git a/Sources/HTTPErrorHandling/Interpretation/HTTPResponseStatus+Interpretation.swift b/Sources/HTTPErrorHandling/Interpretation/HTTPResponseStatus+Interpretation.swift new file mode 100644 index 0000000..1ade54f --- /dev/null +++ b/Sources/HTTPErrorHandling/Interpretation/HTTPResponseStatus+Interpretation.swift @@ -0,0 +1,42 @@ +// MIT License +// +// Copyright © 2024 Darren Mo. +// +// 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. + +import HTTPTypes + +/// Adds static properties to access common sets of HTTP response statuses. +extension Set where Element == HTTPResponse.Status { + /// Statuses that are commonly used to indicate success. + public static let commonSuccessStatuses = Set((200..<300).map(HTTPResponse.Status.init)) + + /// Statuses that are commonly used to indicate transient failures. + /// + /// If the failure is transient, a subsequent attempt may succeed. If the failure is not transient, subsequent + /// attempts will never succeed. + public static let commonTransientFailureStatuses: Set = [ + .requestTimeout, + .tooManyRequests, + .internalServerError, + .badGateway, + .serviceUnavailable, + .gatewayTimeout + ] +} diff --git a/Sources/HTTPErrorHandling/Retry/HTTPRequest+RetryableRequest.swift b/Sources/HTTPErrorHandling/Retry/HTTPRequest+RetryableRequest.swift new file mode 100644 index 0000000..22b55da --- /dev/null +++ b/Sources/HTTPErrorHandling/Retry/HTTPRequest+RetryableRequest.swift @@ -0,0 +1,57 @@ +// MIT License +// +// Copyright © 2024 Darren Mo. +// +// 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. + +import HTTPTypes +import Retry + +/// Adds `RetryableRequest` conformance to `HTTPRequest`. +/// +/// - Important: The retry methods accept a closure that attempts the request. The closure must interpret the response +/// and throw ``HTTPApplicationError`` when the response represents a failure. Calling +/// ``HTTPTypes/HTTPResponse/throwIfFailed(successStatuses:transientFailureStatuses:)`` +/// is a convenient way to do so. +/// +/// - SeeAlso: [`Retry`](https://fumoboy007.github.io/swift-retry/documentation/retry/) +extension HTTPRequest: RetryableRequest { + public var isIdempotent: Bool { + return method.isIdempotent + } + + public func unsafeRetryIgnoringIdempotency( + with configuration: RetryConfiguration, + @_inheritActorContext @_implicitSelfCapture operation: (Self) async throws -> ReturnType + ) async throws -> ReturnType { + let configuration = configuration.withShouldRetry { error in + switch error { + case let error as any HTTPApplicationErrorProtocol: + return error.isTransient + + default: + return configuration.shouldRetry(error) + } + } + + return try await Retry.retry(with: configuration) { + return try await operation(self) + } + } +} diff --git a/Sources/HTTPErrorHandling/Retry/HTTPRequestMethod+IsIdempotent.swift b/Sources/HTTPErrorHandling/Retry/HTTPRequestMethod+IsIdempotent.swift new file mode 100644 index 0000000..b0b6887 --- /dev/null +++ b/Sources/HTTPErrorHandling/Retry/HTTPRequestMethod+IsIdempotent.swift @@ -0,0 +1,50 @@ +// MIT License +// +// Copyright © 2024 Darren Mo. +// +// 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. + +import HTTPTypes + +extension HTTPRequest.Method { + /// Determines whether the HTTP method is idempotent. + /// + /// From the [definition](https://httpwg.org/specs/rfc9110.html#idempotent.methods): + /// > A request method is considered idempotent if the intended effect on the server of multiple + /// > identical requests with that method is the same as the effect for a single such request. + public var isIdempotent: Bool { + switch self { + case .get, + .head, + .options, + .trace, + .put, + .delete: + return true + + case .post, + .patch, + .connect: + return false + + default: + return false + } + } +} diff --git a/Sources/_AdditionalSnippetDependencies/DummyFile.swift b/Sources/_AdditionalSnippetDependencies/DummyFile.swift new file mode 100644 index 0000000..ae95e30 --- /dev/null +++ b/Sources/_AdditionalSnippetDependencies/DummyFile.swift @@ -0,0 +1,21 @@ +// MIT License +// +// Copyright © 2024 Darren Mo. +// +// 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. diff --git a/Tests/HTTPErrorHandlingTests/Fakes/ClockFake.swift b/Tests/HTTPErrorHandlingTests/Fakes/ClockFake.swift new file mode 100644 index 0000000..d8dab0c --- /dev/null +++ b/Tests/HTTPErrorHandlingTests/Fakes/ClockFake.swift @@ -0,0 +1,66 @@ +// MIT License +// +// Copyright © 2024 Darren Mo. +// +// 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. + +import Foundation + +class ClockFake: Clock, @unchecked Sendable { + typealias Instant = ContinuousClock.Instant + + private let realClock = ContinuousClock() + + private let lock = NSLock() + + init() { + self._now = realClock.now + } + + private var _now: Instant + var now: Instant { + lock.lock() + defer { + lock.unlock() + } + + return _now + } + + var minimumResolution: Duration { + return realClock.minimumResolution + } + + func sleep(until deadline: Instant, + tolerance: Duration?) async throws { + // Refactored into a non-async method so that `NSLock.lock` and `NSLock.unlock` can be used. + // Cannot use the async-safe `NSLock.withLocking` method until the following change is released: + // https://github.com/apple/swift-corelibs-foundation/pull/4736 + sleep(until: deadline) + } + + private func sleep(until deadline: Instant) { + lock.lock() + defer { + lock.unlock() + } + + _now = max(deadline, _now) + } +} diff --git a/Tests/HTTPErrorHandlingTests/Fakes/ErrorFake.swift b/Tests/HTTPErrorHandlingTests/Fakes/ErrorFake.swift new file mode 100644 index 0000000..61556ed --- /dev/null +++ b/Tests/HTTPErrorHandlingTests/Fakes/ErrorFake.swift @@ -0,0 +1,24 @@ +// MIT License +// +// Copyright © 2024 Darren Mo. +// +// 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. + +struct ErrorFake: Error { +} diff --git a/Tests/HTTPErrorHandlingTests/Fakes/HTTPRequest+Fakes.swift b/Tests/HTTPErrorHandlingTests/Fakes/HTTPRequest+Fakes.swift new file mode 100644 index 0000000..8bf32f7 --- /dev/null +++ b/Tests/HTTPErrorHandlingTests/Fakes/HTTPRequest+Fakes.swift @@ -0,0 +1,30 @@ +// MIT License +// +// Copyright © 2024 Darren Mo. +// +// 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. + +import HTTPTypes + +extension HTTPRequest { + static let fake = HTTPRequest(method: .get, + scheme: "https", + authority: "example.com", + path: "/") +} diff --git a/Tests/HTTPErrorHandlingTests/Fakes/HTTPResponse+Fakes.swift b/Tests/HTTPErrorHandlingTests/Fakes/HTTPResponse+Fakes.swift new file mode 100644 index 0000000..614e6fd --- /dev/null +++ b/Tests/HTTPErrorHandlingTests/Fakes/HTTPResponse+Fakes.swift @@ -0,0 +1,31 @@ +// MIT License +// +// Copyright © 2024 Darren Mo. +// +// 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. + +import HTTPTypes + +extension HTTPResponse { + static let successFake = HTTPResponse(status: .ok) + static let failureFake = nonTransientFailureFake + + static let transientFailureFake = HTTPResponse(status: .tooManyRequests) + static let nonTransientFailureFake = HTTPResponse(status: .badRequest) +} diff --git a/Tests/HTTPErrorHandlingTests/HTTPRequestRetryTests.swift b/Tests/HTTPErrorHandlingTests/HTTPRequestRetryTests.swift new file mode 100644 index 0000000..83a6e7a --- /dev/null +++ b/Tests/HTTPErrorHandlingTests/HTTPRequestRetryTests.swift @@ -0,0 +1,68 @@ +// MIT License +// +// Copyright © 2024 Darren Mo. +// +// 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. + +import HTTPErrorHandling + +import HTTPTypes +import XCTest + +final class HTTPRequestRetryTests: XCTestCase { + func testTransientHTTPApplicationError_isRetried() async throws { + var attemptCount = 0 + + try await HTTPRequest.fake.retry(clock: ClockFake()) { request in + attemptCount += 1 + + if attemptCount == 1 { + throw HTTPApplicationError(response: HTTPResponse(status: .tooManyRequests), + responseBody: (), + isTransient: true) + } else { + // Success. + } + } + + XCTAssertEqual(attemptCount, 2) + } + + func testNonTransientHTTPApplicationError_notRetried() async throws { + var attemptCount = 0 + + do { + try await HTTPRequest.fake.retry(clock: ClockFake()) { request in + attemptCount += 1 + + if attemptCount == 1 { + throw HTTPApplicationError(response: HTTPResponse(status: .badRequest), + responseBody: (), + isTransient: false) + } else { + // Success. + } + } + } catch is HTTPApplicationError { + // Expected. + } + + XCTAssertEqual(attemptCount, 1) + } +} diff --git a/Tests/HTTPErrorHandlingTests/HTTPResponseInterpretationTests.swift b/Tests/HTTPErrorHandlingTests/HTTPResponseInterpretationTests.swift new file mode 100644 index 0000000..aa087f9 --- /dev/null +++ b/Tests/HTTPErrorHandlingTests/HTTPResponseInterpretationTests.swift @@ -0,0 +1,80 @@ +// MIT License +// +// Copyright © 2024 Darren Mo. +// +// 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. + +import HTTPErrorHandling + +import HTTPTypes +import XCTest + +final class HTTPResponseInterpretationTests: XCTestCase { + func testSuccessResponse_noError() throws { + try HTTPResponse.successFake.throwIfFailed() + } + + func testFailureResponse_throwsError() throws { + let response = HTTPResponse.failureFake + + do { + try response.throwIfFailed() + } catch let error as HTTPApplicationError { + XCTAssertEqual(error.response, response) + } + } + + func testTransientFailureResponse_throwsTransientError() throws { + do { + try HTTPResponse.transientFailureFake.throwIfFailed() + } catch let error as HTTPApplicationError { + XCTAssertTrue(error.isTransient) + } + } + + func testNonTransientFailureResponse_throwsNonTransientError() throws { + do { + try HTTPResponse.nonTransientFailureFake.throwIfFailed() + } catch let error as HTTPApplicationError { + XCTAssertFalse(error.isTransient) + } + } + + func testAttachResponseBodyToError_errorHasResponseBody() async throws { + let responseBody = Data() + + do { + try await HTTPResponse.failureFake.throwIfFailed() { + return responseBody + } + } catch let error as HTTPApplicationError { + XCTAssertEqual(error.responseBody, responseBody) + } + } + + func testFailureToCreateResponseBody_throwsResponseBodyCreationError() async throws { + do { + try await HTTPResponse.failureFake.throwIfFailed() { + throw ErrorFake() + } + } catch is ErrorFake { + // Expected. + } + } +}