1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152
|
// This source file is part of the Swift.org open source project
//
// Copyright (c) 2014 - 2020 Apple Inc. and the Swift project authors
// Licensed under Apache License v2.0 with Runtime Library Exception
//
// See https://swift.org/LICENSE.txt for license information
// See https://swift.org/CONTRIBUTORS.txt for the list of Swift project authors
//
import Synchronization
final class DataURLTestDelegate: NSObject, URLSessionTaskDelegate, URLSessionDataDelegate, Sendable {
let expectation: XCTestExpectation?
// This state is setup before running and checked after `expectation`. Unsafe, but would be better with a lock in the future.
nonisolated(unsafe) var callbacks: [String] = []
nonisolated(unsafe) var data: Data?
nonisolated(unsafe) var error: Error?
nonisolated(unsafe) var response: URLResponse?
init(expectation: XCTestExpectation?) {
self.expectation = expectation
}
func urlSession(_ session: URLSession, task: URLSessionTask, didCompleteWithError error: Error?) {
callbacks.append(#function)
self.error = error
expectation?.fulfill()
}
func urlSession(_ session: URLSession, didBecomeInvalidWithError error: Error?) {
callbacks.append(#function)
self.error = error
}
func urlSession(_ session: URLSession, dataTask: URLSessionDataTask, didReceive data: Data) {
callbacks.append(#function)
self.data = data
}
public func urlSession(_ session: URLSession, dataTask: URLSessionDataTask, didReceive response: URLResponse, completionHandler: @escaping (URLSession.ResponseDisposition) -> Void) {
callbacks.append(#function)
self.response = response
completionHandler(.allow)
}
func urlSession(_ session: URLSession, didFailWithError error: Error) {
callbacks.append(#function)
self.error = error
}
}
class TestDataURLProtocol: XCTestCase {
typealias ResponseProperties = (expectedContentLength: Int64, mimeType: String?, textEncodingName: String?)
private func run(with url: URL) -> DataURLTestDelegate {
let expect = expectation(description: url.absoluteString)
let delegate = DataURLTestDelegate(expectation: expect)
let config = URLSessionConfiguration.default
config.timeoutIntervalForRequest = 100000
let session = URLSession(configuration: config, delegate: delegate, delegateQueue: nil)
let task = session.dataTask(with: url)
task.resume()
wait(for: [expect], timeout: 200000)
return delegate
}
func test_validURIs() throws {
let tests: [(String, String, ResponseProperties)] = [
("data:,123", "123", (expectedContentLength: 3, mimeType: "text/plain", textEncodingName: nil)),
("data:;charset=utf-8;base64,8J+RqOKAjfCfkajigI3wn5Gn4oCN8J+Rpw==", "๐จโ๐จโ๐งโ๐ง", (expectedContentLength: 25, mimeType: "text/plain", textEncodingName: "utf-8")),
("data:text/plain;charset=utf-8,%f0%9f%91%a8%e2%80%8d%f0%9f%91%a8%e2%80%8d%f0%9f%91%a7%e2%80%8d%f0%9f%91%a7", "๐จโ๐จโ๐งโ๐ง", (expectedContentLength: 25, mimeType: "text/plain", textEncodingName: "utf-8")),
// utf-16 is utf016BE
("data:;charset=utf-16;base64,2D3caCAN2D3caCAN2D3cZyAN2D3cZw==", "๐จโ๐จโ๐งโ๐ง", (expectedContentLength: 22, mimeType: "text/plain", textEncodingName: "utf-16")),
("data:;charset=utf-16le;base64,Pdho3A0gPdho3A0gPdhn3A0gPdhn3A==", "๐จโ๐จโ๐งโ๐ง", (expectedContentLength: 22, mimeType: "text/plain", textEncodingName: "utf-16le")),
("data:;charset=utf-16be;base64,2D3caCAN2D3caCAN2D3cZyAN2D3cZw==", "๐จโ๐จโ๐งโ๐ง", (expectedContentLength: 22, mimeType: "text/plain", textEncodingName: "utf-16be")),
// ("data:application/json;charset=iso-8859-1;key=value,,123", ",123", (expectedContentLength: 4, mimeType: "application/json", textEncodingName: "iso-8859-1")),
("data:;charset=utf-8;charset=utf-16;image/png,abc", "abc", (expectedContentLength: 3, mimeType: "text/plain", textEncodingName: "utf-8")),
// ("data:a/b;key=value;charset=macroman,blahblah", "blahblah", (expectedContentLength: 8, mimeType: "a/b", textEncodingName: "macroman")),
]
let callbacks = [
"urlSession(_:dataTask:didReceive:completionHandler:)",
"urlSession(_:dataTask:didReceive:)",
"urlSession(_:task:didCompleteWithError:)",
]
let encodings: [String: String.Encoding] = [
"us-ascii": .ascii,
"utf-8": .utf8,
"utf-16": .utf16,
"utf-16be": .utf16BigEndian,
"utf-16le": .utf16LittleEndian,
"utf-32": .utf32,
"utf-32be": .utf32BigEndian,
"utf-32le": .utf32LittleEndian,
"iso-8859-1": .isoLatin1,
]
for (urlString, body, responseProperties) in tests {
let url = try XCTUnwrap(URL(string: urlString))
let delegate = run(with: url)
XCTAssertNil(delegate.error, "\(urlString) returned errors")
XCTAssertNotNil(delegate.data, "\(urlString) had no data")
XCTAssertEqual(delegate.callbacks.count, 3, "\(urlString) has wrong callback count")
XCTAssertEqual(callbacks, delegate.callbacks, "\(urlString) has wrong callbacks")
if let response = delegate.response {
let expectedProperties = responseProperties
XCTAssertEqual(url, response.url)
XCTAssertEqual(urlString, response.url?.absoluteString)
XCTAssertEqual(expectedProperties.expectedContentLength, response.expectedContentLength, "\(urlString) has incorrect content Length")
XCTAssertEqual(expectedProperties.mimeType, response.mimeType, "\(urlString) has incorrect mime type")
XCTAssertEqual(expectedProperties.textEncodingName, response.textEncodingName, "\(urlString) has incorrect encoding")
let encoding = encodings[response.textEncodingName ?? "us-ascii"] ?? .ascii
if let data = delegate.data, let string = String(data: data, encoding: encoding) {
XCTAssertEqual(body, string, "\(urlString) has wrong body string")
} else {
XCTFail("Cant convert data to string for \(urlString)")
}
} else {
XCTFail("\(urlString) missing URLResponse")
}
}
}
func test_invalidURIs() throws {
let tests = [
"data://blah",
"data:%2c123",
"data:application_json;charset=iso-8859-%31;key=value,123,",
"data:appli/cation/json%3bcharset=ISO-8859-1;key=value,,123,"
]
for urlString in tests {
let url = try XCTUnwrap(URL(string: urlString))
let delegate = run(with: url)
XCTAssertNotNil(delegate.error, "Expected errors for \(urlString)")
XCTAssertEqual(delegate.callbacks.count, 1, "Incorrect error count for \(urlString)")
XCTAssertEqual(["urlSession(_:task:didCompleteWithError:)"], delegate.callbacks)
XCTAssertNil(delegate.response, "Unexpected URLResponse for \(urlString)")
}
}
}
|