310 lines
10 KiB
Swift
310 lines
10 KiB
Swift
//
|
|
// Copyright 2021 Signal Messenger, LLC
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
//
|
|
|
|
import CFNetwork
|
|
import Foundation
|
|
import LibSignalClient
|
|
|
|
/// This extension sacrifices Dictionary performance in order to ignore http
|
|
/// header case and should not be generally used. Since the number of http
|
|
/// headers is generally small, this is an acceptable tradeoff for this use case
|
|
/// but may not be for other use cases.
|
|
private extension Dictionary where Key == String {
|
|
subscript(header header: String) -> Value? {
|
|
get {
|
|
if let key = keys.first(where: { $0.caseInsensitiveCompare(header) == .orderedSame }) {
|
|
return self[key]
|
|
}
|
|
return nil
|
|
}
|
|
set {
|
|
if let key = keys.first(where: { $0.caseInsensitiveCompare(header) == .orderedSame }) {
|
|
self[key] = newValue
|
|
} else {
|
|
self[header] = newValue
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
public class HTTPUtils {
|
|
#if TESTABLE_BUILD
|
|
public static func logCurl(for request: URLRequest) {
|
|
guard let httpMethod = request.httpMethod else {
|
|
Logger.debug("attempted to log curl on a request with no http method")
|
|
return
|
|
}
|
|
guard let url = request.url else {
|
|
Logger.debug("attempted to log curl on a request with no url")
|
|
return
|
|
}
|
|
logCurl(
|
|
url: url,
|
|
method: httpMethod,
|
|
headers: HttpHeaders(httpHeaders: request.allHTTPHeaderFields, overwriteOnConflict: true),
|
|
body: request.httpBody,
|
|
)
|
|
}
|
|
|
|
public static func logCurl(for request: TSRequest) {
|
|
logCurl(
|
|
url: request.url,
|
|
method: request.method,
|
|
headers: request.headers,
|
|
body: {
|
|
switch request.body {
|
|
case .data(let bodyData):
|
|
return bodyData
|
|
case .parameters:
|
|
return nil
|
|
}
|
|
}(),
|
|
)
|
|
}
|
|
|
|
public static func logCurl(url: URL, method httpMethod: String, headers: HttpHeaders, body httpBody: Data?) {
|
|
var curlComponents = ["curl", "-v", "-k", "-X", httpMethod]
|
|
|
|
for (header, headerValue) in headers.headers {
|
|
// We don't yet support escaping header values.
|
|
// If these asserts trip, we'll need to add that.
|
|
owsAssertDebug(!header.contains("'"))
|
|
owsAssertDebug(!headerValue.contains("'"))
|
|
|
|
curlComponents.append("-H")
|
|
curlComponents.append("'\(header): \(headerValue)'")
|
|
}
|
|
|
|
if let httpBody, !httpBody.isEmpty {
|
|
let contentType = headers["Content-Type"]
|
|
switch contentType {
|
|
case MimeType.applicationJson.rawValue:
|
|
guard let jsonBody = String(data: httpBody, encoding: .utf8) else {
|
|
Logger.debug("data attached to request as json was not utf8 encoded")
|
|
return
|
|
}
|
|
// We don't yet support escaping JSON.
|
|
// If these asserts trip, we'll need to add that.
|
|
owsAssertDebug(!jsonBody.contains("'"))
|
|
curlComponents.append("--data-ascii")
|
|
curlComponents.append("'\(jsonBody)'")
|
|
case MimeType.applicationXProtobuf.rawValue, "application/x-www-form-urlencoded", "application/vnd.signal-messenger.mrm":
|
|
let filename = "\(UUID().uuidString).tmp"
|
|
var echoBytes = ""
|
|
for byte in httpBody {
|
|
echoBytes.append(String(format: "\\\\x%02X", byte))
|
|
}
|
|
let echoCommand = "echo -n -e \(echoBytes) > \(filename)"
|
|
|
|
Logger.verbose("curl for request: \(echoCommand)")
|
|
curlComponents.append("--data-binary")
|
|
curlComponents.append("@\(filename)")
|
|
default:
|
|
owsFailDebug("Unknown content type: \(contentType ?? "<nil>")")
|
|
}
|
|
|
|
}
|
|
// TODO: Add support for cookies.
|
|
curlComponents.append("\"\(url.absoluteString)\"")
|
|
let curlCommand = curlComponents.joined(separator: " ")
|
|
Logger.verbose("curl for request: \(curlCommand)")
|
|
}
|
|
#endif
|
|
|
|
public static func preprocessMainServiceHTTPError(
|
|
requestUrl: URL,
|
|
responseStatus: Int,
|
|
responseHeaders: HttpHeaders,
|
|
responseData: Data?,
|
|
) async -> OWSHTTPError {
|
|
let httpError: OWSHTTPError
|
|
if responseStatus == 0 {
|
|
httpError = .networkFailure(.invalidResponseStatus)
|
|
} else {
|
|
httpError = .serviceResponse(.init(
|
|
requestUrl: requestUrl,
|
|
responseStatus: responseStatus,
|
|
responseHeaders: responseHeaders,
|
|
responseData: responseData,
|
|
))
|
|
}
|
|
|
|
await applyHTTPError(httpError)
|
|
return httpError
|
|
}
|
|
|
|
public static func applyHTTPError(_ httpError: OWSHTTPError) async {
|
|
if httpError.isNetworkFailureImpl || httpError.isTimeoutImpl {
|
|
OutageDetection.shared.reportConnectionFailure()
|
|
}
|
|
|
|
if httpError.responseStatusCode == AppExpiry.appExpiredStatusCode {
|
|
let appExpiry = DependenciesBridge.shared.appExpiry
|
|
let db = DependenciesBridge.shared.db
|
|
await appExpiry.setHasAppExpiredAtCurrentVersion(db: db)
|
|
}
|
|
}
|
|
|
|
public static func retryDelayNanoSeconds(_ response: HTTPResponse, defaultRetryTime: TimeInterval = 15) -> UInt64 {
|
|
return (response.headers.retryAfterTimeInterval ?? defaultRetryTime).clampedNanoseconds
|
|
}
|
|
}
|
|
|
|
// MARK: -
|
|
|
|
public extension Error {
|
|
var httpRetryAfterDate: Date? {
|
|
guard let httpError = self as? OWSHTTPError else {
|
|
return nil
|
|
}
|
|
|
|
return httpError.responseHeaders?.retryAfterDate
|
|
}
|
|
|
|
var httpResponseData: Data? {
|
|
guard let httpError = self as? OWSHTTPError else {
|
|
return nil
|
|
}
|
|
|
|
return httpError.responseBodyData
|
|
}
|
|
|
|
var httpStatusCode: Int? {
|
|
guard
|
|
let httpError = self as? OWSHTTPError,
|
|
httpError.responseStatusCode > 0
|
|
else {
|
|
return nil
|
|
}
|
|
|
|
return httpError.responseStatusCode
|
|
}
|
|
|
|
var httpResponseHeaders: HttpHeaders? {
|
|
guard let error = self as? OWSHTTPError else {
|
|
return nil
|
|
}
|
|
return error.responseHeaders
|
|
}
|
|
|
|
/// Does this error represent a transient networking issue?
|
|
///
|
|
/// a.k.a. "the internet gave up" (see also `isTimeout`)
|
|
var isNetworkFailure: Bool {
|
|
switch self as any Error {
|
|
case URLError.cannotConnectToHost: return true
|
|
case URLError.networkConnectionLost: return true
|
|
case URLError.dnsLookupFailed: return true
|
|
case URLError.notConnectedToInternet: return true
|
|
case URLError.secureConnectionFailed: return true
|
|
case URLError.cannotLoadFromNetwork: return true
|
|
case URLError.cannotFindHost: return true
|
|
case URLError.badURL: return true
|
|
case POSIXError.EPROTO: return true
|
|
case let httpError as OWSHTTPError: return httpError.isNetworkFailureImpl
|
|
case SignalError.connectionFailed: return true
|
|
case SignalError.ioError: return true
|
|
case SignalError.webSocketError: return true
|
|
case Upload.Error.networkError: return true
|
|
default: return false
|
|
}
|
|
}
|
|
|
|
/// Does this error represent a self-induced timeout?
|
|
///
|
|
/// a.k.a. "we gave up" (see also `isNetworkFailure`)
|
|
var isTimeout: Bool {
|
|
switch self as any Error {
|
|
case URLError.timedOut: return true
|
|
case let httpError as OWSHTTPError: return httpError.isTimeoutImpl
|
|
case GroupsV2Error.timeout: return true
|
|
case PaymentsError.timeout: return true
|
|
case SignalError.connectionTimeoutError: return true
|
|
case Upload.Error.networkTimeout: return true
|
|
default: return false
|
|
}
|
|
}
|
|
|
|
var isNetworkFailureOrTimeout: Bool {
|
|
return isNetworkFailure || isTimeout
|
|
}
|
|
|
|
var is5xxServiceResponse: Bool {
|
|
switch self as? OWSHTTPError {
|
|
case .serviceResponse(let serviceResponse):
|
|
return serviceResponse.is5xx
|
|
case nil, .invalidRequest, .wrappedFailure, .networkFailure:
|
|
return false
|
|
}
|
|
}
|
|
}
|
|
|
|
// MARK: -
|
|
|
|
@inlinable
|
|
public func owsFailDebugUnlessNetworkFailure(
|
|
_ error: Error,
|
|
file: String = #file,
|
|
function: String = #function,
|
|
line: Int = #line,
|
|
) {
|
|
if error.isNetworkFailureOrTimeout {
|
|
// Log but otherwise ignore network failures.
|
|
Logger.warn("Error: \(error)", file: file, function: function, line: line)
|
|
} else {
|
|
owsFailDebug("Error: \(error)", file: file, function: function, line: line)
|
|
}
|
|
}
|
|
|
|
@inlinable
|
|
public func owsFailBetaUnlessNetworkFailure(
|
|
_ error: Error,
|
|
file: String = #file,
|
|
function: String = #function,
|
|
line: Int = #line,
|
|
) {
|
|
if error.isNetworkFailureOrTimeout {
|
|
// Log but otherwise ignore network failures.
|
|
Logger.warn("Error: \(error)", file: file, function: function, line: line)
|
|
} else {
|
|
owsFailBeta("Error: \(error)", file: file, function: function, line: line)
|
|
}
|
|
}
|
|
|
|
// MARK: -
|
|
|
|
extension HttpHeaders {
|
|
|
|
public var retryAfterTimeInterval: TimeInterval? {
|
|
return retryAfterStringValue.flatMap(TimeInterval.init(_:))
|
|
}
|
|
|
|
public var retryAfterDate: Date? {
|
|
guard let retryAfterStringValue else {
|
|
return nil
|
|
}
|
|
|
|
if let date = Date.ows_parseFromHTTPDateString(retryAfterStringValue) {
|
|
return date
|
|
} else if let date = Date.ows_parseFromISO8601String(retryAfterStringValue) {
|
|
return date
|
|
} else if let retryAfterTimeInterval {
|
|
return Date().addingTimeInterval(retryAfterTimeInterval)
|
|
} else {
|
|
owsAssertDebug(
|
|
CurrentAppContext().isRunningTests,
|
|
"Failed to parse retry-after string: \(String(describing: retryAfterStringValue))",
|
|
)
|
|
return nil
|
|
}
|
|
}
|
|
|
|
private var retryAfterStringValue: String? {
|
|
return value(forHeader: "Retry-After")?
|
|
.trimmingCharacters(in: .whitespacesAndNewlines)
|
|
.nilIfEmpty
|
|
}
|
|
}
|