MultipartFormDataTests.swift 36 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964
  1. //
  2. // MultipartFormDataTests.swift
  3. //
  4. // Copyright (c) 2014-2018 Alamofire Software Foundation (http://alamofire.org/)
  5. //
  6. // Permission is hereby granted, free of charge, to any person obtaining a copy
  7. // of this software and associated documentation files (the "Software"), to deal
  8. // in the Software without restriction, including without limitation the rights
  9. // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
  10. // copies of the Software, and to permit persons to whom the Software is
  11. // furnished to do so, subject to the following conditions:
  12. //
  13. // The above copyright notice and this permission notice shall be included in
  14. // all copies or substantial portions of the Software.
  15. //
  16. // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
  17. // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
  18. // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
  19. // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
  20. // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
  21. // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
  22. // THE SOFTWARE.
  23. //
  24. import Alamofire
  25. import Foundation
  26. import XCTest
  27. struct EncodingCharacters {
  28. static let crlf = "\r\n"
  29. }
  30. struct BoundaryGenerator {
  31. enum BoundaryType {
  32. case initial, encapsulated, final
  33. }
  34. static func boundary(forBoundaryType boundaryType: BoundaryType, boundaryKey: String) -> String {
  35. let boundary: String
  36. switch boundaryType {
  37. case .initial:
  38. boundary = "--\(boundaryKey)\(EncodingCharacters.crlf)"
  39. case .encapsulated:
  40. boundary = "\(EncodingCharacters.crlf)--\(boundaryKey)\(EncodingCharacters.crlf)"
  41. case .final:
  42. boundary = "\(EncodingCharacters.crlf)--\(boundaryKey)--\(EncodingCharacters.crlf)"
  43. }
  44. return boundary
  45. }
  46. static func boundaryData(boundaryType: BoundaryType, boundaryKey: String) -> Data {
  47. return BoundaryGenerator.boundary(
  48. forBoundaryType: boundaryType,
  49. boundaryKey: boundaryKey
  50. ).data(using: .utf8, allowLossyConversion: false)!
  51. }
  52. }
  53. private func temporaryFileURL() -> URL { return BaseTestCase.testDirectoryURL.appendingPathComponent(UUID().uuidString) }
  54. // MARK: -
  55. class MultipartFormDataPropertiesTestCase: BaseTestCase {
  56. func testThatContentTypeContainsBoundary() {
  57. // Given
  58. let multipartFormData = MultipartFormData()
  59. // When
  60. let boundary = multipartFormData.boundary
  61. // Then
  62. let expectedContentType = "multipart/form-data; boundary=\(boundary)"
  63. XCTAssertEqual(multipartFormData.contentType, expectedContentType, "contentType should match expected value")
  64. }
  65. func testThatContentLengthMatchesTotalBodyPartSize() {
  66. // Given
  67. let multipartFormData = MultipartFormData()
  68. let data1 = "Lorem ipsum dolor sit amet.".data(using: .utf8, allowLossyConversion: false)!
  69. let data2 = "Vim at integre alterum.".data(using: .utf8, allowLossyConversion: false)!
  70. // When
  71. multipartFormData.append(data1, withName: "data1")
  72. multipartFormData.append(data2, withName: "data2")
  73. // Then
  74. let expectedContentLength = UInt64(data1.count + data2.count)
  75. XCTAssertEqual(multipartFormData.contentLength, expectedContentLength, "content length should match expected value")
  76. }
  77. }
  78. // MARK: -
  79. class MultipartFormDataEncodingTestCase: BaseTestCase {
  80. let crlf = EncodingCharacters.crlf
  81. func testEncodingDataBodyPart() {
  82. // Given
  83. let multipartFormData = MultipartFormData()
  84. let data = "Lorem ipsum dolor sit amet.".data(using: .utf8, allowLossyConversion: false)!
  85. multipartFormData.append(data, withName: "data")
  86. var encodedData: Data?
  87. // When
  88. do {
  89. encodedData = try multipartFormData.encode()
  90. } catch {
  91. // No-op
  92. }
  93. // Then
  94. XCTAssertNotNil(encodedData, "encoded data should not be nil")
  95. if let encodedData = encodedData {
  96. let boundary = multipartFormData.boundary
  97. let expectedData = (
  98. BoundaryGenerator.boundary(forBoundaryType: .initial, boundaryKey: boundary) +
  99. "Content-Disposition: form-data; name=\"data\"\(crlf)\(crlf)" +
  100. "Lorem ipsum dolor sit amet." +
  101. BoundaryGenerator.boundary(forBoundaryType: .final, boundaryKey: boundary)
  102. ).data(using: .utf8, allowLossyConversion: false)!
  103. XCTAssertEqual(encodedData, expectedData, "encoded data should match expected data")
  104. }
  105. }
  106. func testEncodingMultipleDataBodyParts() {
  107. // Given
  108. let multipartFormData = MultipartFormData()
  109. let frenchData = Data("français".utf8)
  110. let japaneseData = Data("日本語".utf8)
  111. let emojiData = Data("😃👍🏻🍻🎉".utf8)
  112. multipartFormData.append(frenchData, withName: "french")
  113. multipartFormData.append(japaneseData, withName: "japanese", mimeType: "text/plain")
  114. multipartFormData.append(emojiData, withName: "emoji", mimeType: "text/plain")
  115. var encodedData: Data?
  116. // When
  117. do {
  118. encodedData = try multipartFormData.encode()
  119. } catch {
  120. // No-op
  121. }
  122. // Then
  123. XCTAssertNotNil(encodedData, "encoded data should not be nil")
  124. if let encodedData = encodedData {
  125. let boundary = multipartFormData.boundary
  126. let expectedString = (
  127. BoundaryGenerator.boundary(forBoundaryType: .initial, boundaryKey: boundary) +
  128. "Content-Disposition: form-data; name=\"french\"\(crlf)\(crlf)" +
  129. "français" +
  130. BoundaryGenerator.boundary(forBoundaryType: .encapsulated, boundaryKey: boundary) +
  131. "Content-Type: text/plain\(crlf)" +
  132. "Content-Disposition: form-data; name=\"japanese\"\(crlf)\(crlf)" +
  133. "日本語" +
  134. BoundaryGenerator.boundary(forBoundaryType: .encapsulated, boundaryKey: boundary) +
  135. "Content-Type: text/plain\(crlf)" +
  136. "Content-Disposition: form-data; name=\"emoji\"\(crlf)\(crlf)" +
  137. "😃👍🏻🍻🎉" +
  138. BoundaryGenerator.boundary(forBoundaryType: .final, boundaryKey: boundary)
  139. )
  140. let expectedData = Data(expectedString.utf8)
  141. XCTAssertEqual(encodedData, expectedData, "encoded data should match expected data")
  142. }
  143. }
  144. func testEncodingFileBodyPart() {
  145. // Given
  146. let multipartFormData = MultipartFormData()
  147. let unicornImageURL = url(forResource: "unicorn", withExtension: "png")
  148. multipartFormData.append(unicornImageURL, withName: "unicorn")
  149. var encodedData: Data?
  150. // When
  151. do {
  152. encodedData = try multipartFormData.encode()
  153. } catch {
  154. // No-op
  155. }
  156. // Then
  157. XCTAssertNotNil(encodedData, "encoded data should not be nil")
  158. if let encodedData = encodedData {
  159. let boundary = multipartFormData.boundary
  160. var expectedData = Data()
  161. expectedData.append(BoundaryGenerator.boundaryData(boundaryType: .initial, boundaryKey: boundary))
  162. expectedData.append(Data((
  163. "Content-Type: image/png\(crlf)" +
  164. "Content-Disposition: form-data; name=\"unicorn\"; filename=\"unicorn.png\"\(crlf)\(crlf)").utf8
  165. )
  166. )
  167. expectedData.append(try! Data(contentsOf: unicornImageURL))
  168. expectedData.append(BoundaryGenerator.boundaryData(boundaryType: .final, boundaryKey: boundary))
  169. XCTAssertEqual(encodedData, expectedData, "data should match expected data")
  170. }
  171. }
  172. func testEncodingMultipleFileBodyParts() {
  173. // Given
  174. let multipartFormData = MultipartFormData()
  175. let unicornImageURL = url(forResource: "unicorn", withExtension: "png")
  176. let rainbowImageURL = url(forResource: "rainbow", withExtension: "jpg")
  177. multipartFormData.append(unicornImageURL, withName: "unicorn")
  178. multipartFormData.append(rainbowImageURL, withName: "rainbow")
  179. var encodedData: Data?
  180. // When
  181. do {
  182. encodedData = try multipartFormData.encode()
  183. } catch {
  184. // No-op
  185. }
  186. // Then
  187. XCTAssertNotNil(encodedData, "encoded data should not be nil")
  188. if let encodedData = encodedData {
  189. let boundary = multipartFormData.boundary
  190. var expectedData = Data()
  191. expectedData.append(BoundaryGenerator.boundaryData(boundaryType: .initial, boundaryKey: boundary))
  192. expectedData.append(Data((
  193. "Content-Type: image/png\(crlf)" +
  194. "Content-Disposition: form-data; name=\"unicorn\"; filename=\"unicorn.png\"\(crlf)\(crlf)").utf8
  195. )
  196. )
  197. expectedData.append(try! Data(contentsOf: unicornImageURL))
  198. expectedData.append(BoundaryGenerator.boundaryData(boundaryType: .encapsulated, boundaryKey: boundary))
  199. expectedData.append(Data((
  200. "Content-Type: image/jpeg\(crlf)" +
  201. "Content-Disposition: form-data; name=\"rainbow\"; filename=\"rainbow.jpg\"\(crlf)\(crlf)").utf8
  202. )
  203. )
  204. expectedData.append(try! Data(contentsOf: rainbowImageURL))
  205. expectedData.append(BoundaryGenerator.boundaryData(boundaryType: .final, boundaryKey: boundary))
  206. XCTAssertEqual(encodedData, expectedData, "data should match expected data")
  207. }
  208. }
  209. func testEncodingStreamBodyPart() {
  210. // Given
  211. let multipartFormData = MultipartFormData()
  212. let unicornImageURL = url(forResource: "unicorn", withExtension: "png")
  213. let unicornDataLength = UInt64((try! Data(contentsOf: unicornImageURL)).count)
  214. let unicornStream = InputStream(url: unicornImageURL)!
  215. multipartFormData.append(
  216. unicornStream,
  217. withLength: unicornDataLength,
  218. name: "unicorn",
  219. fileName: "unicorn.png",
  220. mimeType: "image/png"
  221. )
  222. var encodedData: Data?
  223. // When
  224. do {
  225. encodedData = try multipartFormData.encode()
  226. } catch {
  227. // No-op
  228. }
  229. // Then
  230. XCTAssertNotNil(encodedData, "encoded data should not be nil")
  231. if let encodedData = encodedData {
  232. let boundary = multipartFormData.boundary
  233. var expectedData = Data()
  234. expectedData.append(BoundaryGenerator.boundaryData(boundaryType: .initial, boundaryKey: boundary))
  235. expectedData.append(Data((
  236. "Content-Type: image/png\(crlf)" +
  237. "Content-Disposition: form-data; name=\"unicorn\"; filename=\"unicorn.png\"\(crlf)\(crlf)").utf8
  238. )
  239. )
  240. expectedData.append(try! Data(contentsOf: unicornImageURL))
  241. expectedData.append(BoundaryGenerator.boundaryData(boundaryType: .final, boundaryKey: boundary))
  242. XCTAssertEqual(encodedData, expectedData, "data should match expected data")
  243. }
  244. }
  245. func testEncodingMultipleStreamBodyParts() {
  246. // Given
  247. let multipartFormData = MultipartFormData()
  248. let unicornImageURL = url(forResource: "unicorn", withExtension: "png")
  249. let unicornDataLength = UInt64((try! Data(contentsOf: unicornImageURL)).count)
  250. let unicornStream = InputStream(url: unicornImageURL)!
  251. let rainbowImageURL = url(forResource: "rainbow", withExtension: "jpg")
  252. let rainbowDataLength = UInt64((try! Data(contentsOf: rainbowImageURL)).count)
  253. let rainbowStream = InputStream(url: rainbowImageURL)!
  254. multipartFormData.append(
  255. unicornStream,
  256. withLength: unicornDataLength,
  257. name: "unicorn",
  258. fileName: "unicorn.png",
  259. mimeType: "image/png"
  260. )
  261. multipartFormData.append(
  262. rainbowStream,
  263. withLength: rainbowDataLength,
  264. name: "rainbow",
  265. fileName: "rainbow.jpg",
  266. mimeType: "image/jpeg"
  267. )
  268. var encodedData: Data?
  269. // When
  270. do {
  271. encodedData = try multipartFormData.encode()
  272. } catch {
  273. // No-op
  274. }
  275. // Then
  276. XCTAssertNotNil(encodedData, "encoded data should not be nil")
  277. if let encodedData = encodedData {
  278. let boundary = multipartFormData.boundary
  279. var expectedData = Data()
  280. expectedData.append(BoundaryGenerator.boundaryData(boundaryType: .initial, boundaryKey: boundary))
  281. expectedData.append(Data((
  282. "Content-Type: image/png\(crlf)" +
  283. "Content-Disposition: form-data; name=\"unicorn\"; filename=\"unicorn.png\"\(crlf)\(crlf)").utf8
  284. )
  285. )
  286. expectedData.append(try! Data(contentsOf: unicornImageURL))
  287. expectedData.append(BoundaryGenerator.boundaryData(boundaryType: .encapsulated, boundaryKey: boundary))
  288. expectedData.append(Data((
  289. "Content-Type: image/jpeg\(crlf)" +
  290. "Content-Disposition: form-data; name=\"rainbow\"; filename=\"rainbow.jpg\"\(crlf)\(crlf)").utf8
  291. )
  292. )
  293. expectedData.append(try! Data(contentsOf: rainbowImageURL))
  294. expectedData.append(BoundaryGenerator.boundaryData(boundaryType: .final, boundaryKey: boundary))
  295. XCTAssertEqual(encodedData, expectedData, "data should match expected data")
  296. }
  297. }
  298. func testEncodingMultipleBodyPartsWithVaryingTypes() {
  299. // Given
  300. let multipartFormData = MultipartFormData()
  301. let loremData = Data("Lorem ipsum.".utf8)
  302. let unicornImageURL = url(forResource: "unicorn", withExtension: "png")
  303. let rainbowImageURL = url(forResource: "rainbow", withExtension: "jpg")
  304. let rainbowDataLength = UInt64((try! Data(contentsOf: rainbowImageURL)).count)
  305. let rainbowStream = InputStream(url: rainbowImageURL)!
  306. multipartFormData.append(loremData, withName: "lorem")
  307. multipartFormData.append(unicornImageURL, withName: "unicorn")
  308. multipartFormData.append(
  309. rainbowStream,
  310. withLength: rainbowDataLength,
  311. name: "rainbow",
  312. fileName: "rainbow.jpg",
  313. mimeType: "image/jpeg"
  314. )
  315. var encodedData: Data?
  316. // When
  317. do {
  318. encodedData = try multipartFormData.encode()
  319. } catch {
  320. // No-op
  321. }
  322. // Then
  323. XCTAssertNotNil(encodedData, "encoded data should not be nil")
  324. if let encodedData = encodedData {
  325. let boundary = multipartFormData.boundary
  326. var expectedData = Data()
  327. expectedData.append(BoundaryGenerator.boundaryData(boundaryType: .initial, boundaryKey: boundary))
  328. expectedData.append(Data(
  329. "Content-Disposition: form-data; name=\"lorem\"\(crlf)\(crlf)".utf8
  330. )
  331. )
  332. expectedData.append(loremData)
  333. expectedData.append(BoundaryGenerator.boundaryData(boundaryType: .encapsulated, boundaryKey: boundary))
  334. expectedData.append(Data((
  335. "Content-Type: image/png\(crlf)" +
  336. "Content-Disposition: form-data; name=\"unicorn\"; filename=\"unicorn.png\"\(crlf)\(crlf)").utf8
  337. )
  338. )
  339. expectedData.append(try! Data(contentsOf: unicornImageURL))
  340. expectedData.append(BoundaryGenerator.boundaryData(boundaryType: .encapsulated, boundaryKey: boundary))
  341. expectedData.append(Data((
  342. "Content-Type: image/jpeg\(crlf)" +
  343. "Content-Disposition: form-data; name=\"rainbow\"; filename=\"rainbow.jpg\"\(crlf)\(crlf)").utf8
  344. )
  345. )
  346. expectedData.append(try! Data(contentsOf: rainbowImageURL))
  347. expectedData.append(BoundaryGenerator.boundaryData(boundaryType: .final, boundaryKey: boundary))
  348. XCTAssertEqual(encodedData, expectedData, "data should match expected data")
  349. }
  350. }
  351. }
  352. // MARK: -
  353. class MultipartFormDataWriteEncodedDataToDiskTestCase: BaseTestCase {
  354. let crlf = EncodingCharacters.crlf
  355. func testWritingEncodedDataBodyPartToDisk() {
  356. // Given
  357. let fileURL = temporaryFileURL()
  358. let multipartFormData = MultipartFormData()
  359. let data = "Lorem ipsum dolor sit amet.".data(using: .utf8, allowLossyConversion: false)!
  360. multipartFormData.append(data, withName: "data")
  361. var encodingError: Error?
  362. // When
  363. do {
  364. try multipartFormData.writeEncodedData(to: fileURL)
  365. } catch {
  366. encodingError = error
  367. }
  368. // Then
  369. XCTAssertNil(encodingError, "encoding error should be nil")
  370. if let fileData = try? Data(contentsOf: fileURL) {
  371. let boundary = multipartFormData.boundary
  372. let expectedFileData = (
  373. BoundaryGenerator.boundary(forBoundaryType: .initial, boundaryKey: boundary) +
  374. "Content-Disposition: form-data; name=\"data\"\(crlf)\(crlf)" +
  375. "Lorem ipsum dolor sit amet." +
  376. BoundaryGenerator.boundary(forBoundaryType: .final, boundaryKey: boundary)
  377. ).data(using: .utf8, allowLossyConversion: false)!
  378. XCTAssertEqual(fileData, expectedFileData, "file data should match expected file data")
  379. } else {
  380. XCTFail("file data should not be nil")
  381. }
  382. }
  383. func testWritingMultipleEncodedDataBodyPartsToDisk() {
  384. // Given
  385. let fileURL = temporaryFileURL()
  386. let multipartFormData = MultipartFormData()
  387. let frenchData = "français".data(using: .utf8, allowLossyConversion: false)!
  388. let japaneseData = "日本語".data(using: .utf8, allowLossyConversion: false)!
  389. let emojiData = "😃👍🏻🍻🎉".data(using: .utf8, allowLossyConversion: false)!
  390. multipartFormData.append(frenchData, withName: "french")
  391. multipartFormData.append(japaneseData, withName: "japanese")
  392. multipartFormData.append(emojiData, withName: "emoji")
  393. var encodingError: Error?
  394. // When
  395. do {
  396. try multipartFormData.writeEncodedData(to: fileURL)
  397. } catch {
  398. encodingError = error
  399. }
  400. // Then
  401. XCTAssertNil(encodingError, "encoding error should be nil")
  402. if let fileData = try? Data(contentsOf: fileURL) {
  403. let boundary = multipartFormData.boundary
  404. let expectedFileData = (
  405. BoundaryGenerator.boundary(forBoundaryType: .initial, boundaryKey: boundary) +
  406. "Content-Disposition: form-data; name=\"french\"\(crlf)\(crlf)" +
  407. "français" +
  408. BoundaryGenerator.boundary(forBoundaryType: .encapsulated, boundaryKey: boundary) +
  409. "Content-Disposition: form-data; name=\"japanese\"\(crlf)\(crlf)" +
  410. "日本語" +
  411. BoundaryGenerator.boundary(forBoundaryType: .encapsulated, boundaryKey: boundary) +
  412. "Content-Disposition: form-data; name=\"emoji\"\(crlf)\(crlf)" +
  413. "😃👍🏻🍻🎉" +
  414. BoundaryGenerator.boundary(forBoundaryType: .final, boundaryKey: boundary)
  415. ).data(using: .utf8, allowLossyConversion: false)!
  416. XCTAssertEqual(fileData, expectedFileData, "file data should match expected file data")
  417. } else {
  418. XCTFail("file data should not be nil")
  419. }
  420. }
  421. func testWritingEncodedFileBodyPartToDisk() {
  422. // Given
  423. let fileURL = temporaryFileURL()
  424. let multipartFormData = MultipartFormData()
  425. let unicornImageURL = url(forResource: "unicorn", withExtension: "png")
  426. multipartFormData.append(unicornImageURL, withName: "unicorn")
  427. var encodingError: Error?
  428. // When
  429. do {
  430. try multipartFormData.writeEncodedData(to: fileURL)
  431. } catch {
  432. encodingError = error
  433. }
  434. // Then
  435. XCTAssertNil(encodingError, "encoding error should be nil")
  436. if let fileData = try? Data(contentsOf: fileURL) {
  437. let boundary = multipartFormData.boundary
  438. var expectedFileData = Data()
  439. expectedFileData.append(BoundaryGenerator.boundaryData(boundaryType: .initial, boundaryKey: boundary))
  440. expectedFileData.append(Data((
  441. "Content-Type: image/png\(crlf)" +
  442. "Content-Disposition: form-data; name=\"unicorn\"; filename=\"unicorn.png\"\(crlf)\(crlf)").utf8
  443. )
  444. )
  445. expectedFileData.append(try! Data(contentsOf: unicornImageURL))
  446. expectedFileData.append(BoundaryGenerator.boundaryData(boundaryType: .final, boundaryKey: boundary))
  447. XCTAssertEqual(fileData, expectedFileData, "file data should match expected file data")
  448. } else {
  449. XCTFail("file data should not be nil")
  450. }
  451. }
  452. func testWritingMultipleEncodedFileBodyPartsToDisk() {
  453. // Given
  454. let fileURL = temporaryFileURL()
  455. let multipartFormData = MultipartFormData()
  456. let unicornImageURL = url(forResource: "unicorn", withExtension: "png")
  457. let rainbowImageURL = url(forResource: "rainbow", withExtension: "jpg")
  458. multipartFormData.append(unicornImageURL, withName: "unicorn")
  459. multipartFormData.append(rainbowImageURL, withName: "rainbow")
  460. var encodingError: Error?
  461. // When
  462. do {
  463. try multipartFormData.writeEncodedData(to: fileURL)
  464. } catch {
  465. encodingError = error
  466. }
  467. // Then
  468. XCTAssertNil(encodingError, "encoding error should be nil")
  469. if let fileData = try? Data(contentsOf: fileURL) {
  470. let boundary = multipartFormData.boundary
  471. var expectedFileData = Data()
  472. expectedFileData.append(BoundaryGenerator.boundaryData(boundaryType: .initial, boundaryKey: boundary))
  473. expectedFileData.append(Data((
  474. "Content-Type: image/png\(crlf)" +
  475. "Content-Disposition: form-data; name=\"unicorn\"; filename=\"unicorn.png\"\(crlf)\(crlf)").utf8
  476. )
  477. )
  478. expectedFileData.append(try! Data(contentsOf: unicornImageURL))
  479. expectedFileData.append(BoundaryGenerator.boundaryData(boundaryType: .encapsulated, boundaryKey: boundary))
  480. expectedFileData.append(Data((
  481. "Content-Type: image/jpeg\(crlf)" +
  482. "Content-Disposition: form-data; name=\"rainbow\"; filename=\"rainbow.jpg\"\(crlf)\(crlf)").utf8
  483. )
  484. )
  485. expectedFileData.append(try! Data(contentsOf: rainbowImageURL))
  486. expectedFileData.append(BoundaryGenerator.boundaryData(boundaryType: .final, boundaryKey: boundary))
  487. XCTAssertEqual(fileData, expectedFileData, "file data should match expected file data")
  488. } else {
  489. XCTFail("file data should not be nil")
  490. }
  491. }
  492. func testWritingEncodedStreamBodyPartToDisk() {
  493. // Given
  494. let fileURL = temporaryFileURL()
  495. let multipartFormData = MultipartFormData()
  496. let unicornImageURL = url(forResource: "unicorn", withExtension: "png")
  497. let unicornDataLength = UInt64((try! Data(contentsOf: unicornImageURL)).count)
  498. let unicornStream = InputStream(url: unicornImageURL)!
  499. multipartFormData.append(
  500. unicornStream,
  501. withLength: unicornDataLength,
  502. name: "unicorn",
  503. fileName: "unicorn.png",
  504. mimeType: "image/png"
  505. )
  506. var encodingError: Error?
  507. // When
  508. do {
  509. try multipartFormData.writeEncodedData(to: fileURL)
  510. } catch {
  511. encodingError = error
  512. }
  513. // Then
  514. XCTAssertNil(encodingError, "encoding error should be nil")
  515. if let fileData = try? Data(contentsOf: fileURL) {
  516. let boundary = multipartFormData.boundary
  517. var expectedFileData = Data()
  518. expectedFileData.append(BoundaryGenerator.boundaryData(boundaryType: .initial, boundaryKey: boundary))
  519. expectedFileData.append(Data((
  520. "Content-Type: image/png\(crlf)" +
  521. "Content-Disposition: form-data; name=\"unicorn\"; filename=\"unicorn.png\"\(crlf)\(crlf)").utf8
  522. )
  523. )
  524. expectedFileData.append(try! Data(contentsOf: unicornImageURL))
  525. expectedFileData.append(BoundaryGenerator.boundaryData(boundaryType: .final, boundaryKey: boundary))
  526. XCTAssertEqual(fileData, expectedFileData, "file data should match expected file data")
  527. } else {
  528. XCTFail("file data should not be nil")
  529. }
  530. }
  531. func testWritingMultipleEncodedStreamBodyPartsToDisk() {
  532. // Given
  533. let fileURL = temporaryFileURL()
  534. let multipartFormData = MultipartFormData()
  535. let unicornImageURL = url(forResource: "unicorn", withExtension: "png")
  536. let unicornDataLength = UInt64((try! Data(contentsOf: unicornImageURL)).count)
  537. let unicornStream = InputStream(url: unicornImageURL)!
  538. let rainbowImageURL = url(forResource: "rainbow", withExtension: "jpg")
  539. let rainbowDataLength = UInt64((try! Data(contentsOf: rainbowImageURL)).count)
  540. let rainbowStream = InputStream(url: rainbowImageURL)!
  541. multipartFormData.append(
  542. unicornStream,
  543. withLength: unicornDataLength,
  544. name: "unicorn",
  545. fileName: "unicorn.png",
  546. mimeType: "image/png"
  547. )
  548. multipartFormData.append(
  549. rainbowStream,
  550. withLength: rainbowDataLength,
  551. name: "rainbow",
  552. fileName: "rainbow.jpg",
  553. mimeType: "image/jpeg"
  554. )
  555. var encodingError: Error?
  556. // When
  557. do {
  558. try multipartFormData.writeEncodedData(to: fileURL)
  559. } catch {
  560. encodingError = error
  561. }
  562. // Then
  563. XCTAssertNil(encodingError, "encoding error should be nil")
  564. if let fileData = try? Data(contentsOf: fileURL) {
  565. let boundary = multipartFormData.boundary
  566. var expectedFileData = Data()
  567. expectedFileData.append(BoundaryGenerator.boundaryData(boundaryType: .initial, boundaryKey: boundary))
  568. expectedFileData.append(Data((
  569. "Content-Type: image/png\(crlf)" +
  570. "Content-Disposition: form-data; name=\"unicorn\"; filename=\"unicorn.png\"\(crlf)\(crlf)").utf8
  571. )
  572. )
  573. expectedFileData.append(try! Data(contentsOf: unicornImageURL))
  574. expectedFileData.append(BoundaryGenerator.boundaryData(boundaryType: .encapsulated, boundaryKey: boundary))
  575. expectedFileData.append(Data((
  576. "Content-Type: image/jpeg\(crlf)" +
  577. "Content-Disposition: form-data; name=\"rainbow\"; filename=\"rainbow.jpg\"\(crlf)\(crlf)").utf8
  578. )
  579. )
  580. expectedFileData.append(try! Data(contentsOf: rainbowImageURL))
  581. expectedFileData.append(BoundaryGenerator.boundaryData(boundaryType: .final, boundaryKey: boundary))
  582. XCTAssertEqual(fileData, expectedFileData, "file data should match expected file data")
  583. } else {
  584. XCTFail("file data should not be nil")
  585. }
  586. }
  587. func testWritingMultipleEncodedBodyPartsWithVaryingTypesToDisk() {
  588. // Given
  589. let fileURL = temporaryFileURL()
  590. let multipartFormData = MultipartFormData()
  591. let loremData = Data("Lorem ipsum.".utf8)
  592. let unicornImageURL = url(forResource: "unicorn", withExtension: "png")
  593. let rainbowImageURL = url(forResource: "rainbow", withExtension: "jpg")
  594. let rainbowDataLength = UInt64((try! Data(contentsOf: rainbowImageURL)).count)
  595. let rainbowStream = InputStream(url: rainbowImageURL)!
  596. multipartFormData.append(loremData, withName: "lorem")
  597. multipartFormData.append(unicornImageURL, withName: "unicorn")
  598. multipartFormData.append(
  599. rainbowStream,
  600. withLength: rainbowDataLength,
  601. name: "rainbow",
  602. fileName: "rainbow.jpg",
  603. mimeType: "image/jpeg"
  604. )
  605. var encodingError: Error?
  606. // When
  607. do {
  608. try multipartFormData.writeEncodedData(to: fileURL)
  609. } catch {
  610. encodingError = error
  611. }
  612. // Then
  613. XCTAssertNil(encodingError, "encoding error should be nil")
  614. if let fileData = try? Data(contentsOf: fileURL) {
  615. let boundary = multipartFormData.boundary
  616. var expectedFileData = Data()
  617. expectedFileData.append(BoundaryGenerator.boundaryData(boundaryType: .initial, boundaryKey: boundary))
  618. expectedFileData.append(Data(
  619. "Content-Disposition: form-data; name=\"lorem\"\(crlf)\(crlf)".utf8
  620. )
  621. )
  622. expectedFileData.append(loremData)
  623. expectedFileData.append(BoundaryGenerator.boundaryData(boundaryType: .encapsulated, boundaryKey: boundary))
  624. expectedFileData.append(Data((
  625. "Content-Type: image/png\(crlf)" +
  626. "Content-Disposition: form-data; name=\"unicorn\"; filename=\"unicorn.png\"\(crlf)\(crlf)").utf8
  627. )
  628. )
  629. expectedFileData.append(try! Data(contentsOf: unicornImageURL))
  630. expectedFileData.append(BoundaryGenerator.boundaryData(boundaryType: .encapsulated, boundaryKey: boundary))
  631. expectedFileData.append(Data((
  632. "Content-Type: image/jpeg\(crlf)" +
  633. "Content-Disposition: form-data; name=\"rainbow\"; filename=\"rainbow.jpg\"\(crlf)\(crlf)").utf8
  634. )
  635. )
  636. expectedFileData.append(try! Data(contentsOf: rainbowImageURL))
  637. expectedFileData.append(BoundaryGenerator.boundaryData(boundaryType: .final, boundaryKey: boundary))
  638. XCTAssertEqual(fileData, expectedFileData, "file data should match expected file data")
  639. } else {
  640. XCTFail("file data should not be nil")
  641. }
  642. }
  643. }
  644. // MARK: -
  645. class MultipartFormDataFailureTestCase: BaseTestCase {
  646. func testThatAppendingFileBodyPartWithInvalidLastPathComponentReturnsError() {
  647. // Given
  648. let fileURL = NSURL(string: "")! as URL
  649. let multipartFormData = MultipartFormData()
  650. multipartFormData.append(fileURL, withName: "empty_data")
  651. var encodingError: Error?
  652. // When
  653. do {
  654. _ = try multipartFormData.encode()
  655. } catch {
  656. encodingError = error
  657. }
  658. // Then
  659. XCTAssertNotNil(encodingError, "encoding error should not be nil")
  660. if let error = encodingError as? AFError {
  661. XCTAssertTrue(error.isBodyPartFilenameInvalid)
  662. let expectedFailureReason = "The URL provided does not have a valid filename: \(fileURL)"
  663. XCTAssertEqual(error.localizedDescription, expectedFailureReason, "failure reason does not match expected value")
  664. } else {
  665. XCTFail("Error should be AFError.")
  666. }
  667. }
  668. func testThatAppendingFileBodyPartThatIsNotFileURLReturnsError() {
  669. // Given
  670. let fileURL = URL(string: "https://example.com/image.jpg")!
  671. let multipartFormData = MultipartFormData()
  672. multipartFormData.append(fileURL, withName: "empty_data")
  673. var encodingError: Error?
  674. // When
  675. do {
  676. _ = try multipartFormData.encode()
  677. } catch {
  678. encodingError = error
  679. }
  680. // Then
  681. XCTAssertNotNil(encodingError, "encoding error should not be nil")
  682. if let error = encodingError as? AFError {
  683. XCTAssertTrue(error.isBodyPartURLInvalid)
  684. let expectedFailureReason = "The URL provided is not a file URL: \(fileURL)"
  685. XCTAssertEqual(error.localizedDescription, expectedFailureReason, "error failure reason does not match expected value")
  686. } else {
  687. XCTFail("Error should be AFError.")
  688. }
  689. }
  690. func testThatAppendingFileBodyPartThatIsNotReachableReturnsError() {
  691. // Given
  692. let filePath = (NSTemporaryDirectory() as NSString).appendingPathComponent("does_not_exist.jpg")
  693. let fileURL = URL(fileURLWithPath: filePath)
  694. let multipartFormData = MultipartFormData()
  695. multipartFormData.append(fileURL, withName: "empty_data")
  696. var encodingError: Error?
  697. // When
  698. do {
  699. _ = try multipartFormData.encode()
  700. } catch {
  701. encodingError = error
  702. }
  703. // Then
  704. XCTAssertNotNil(encodingError, "encoding error should not be nil")
  705. if let error = encodingError as? AFError {
  706. XCTAssertTrue(error.isBodyPartFileNotReachableWithError)
  707. } else {
  708. XCTFail("Error should be AFError.")
  709. }
  710. }
  711. func testThatAppendingFileBodyPartThatIsDirectoryReturnsError() {
  712. // Given
  713. let directoryURL = URL(fileURLWithPath: NSTemporaryDirectory(), isDirectory: true)
  714. let multipartFormData = MultipartFormData()
  715. multipartFormData.append(directoryURL, withName: "empty_data", fileName: "empty", mimeType: "application/octet")
  716. var encodingError: Error?
  717. // When
  718. do {
  719. _ = try multipartFormData.encode()
  720. } catch {
  721. encodingError = error
  722. }
  723. // Then
  724. XCTAssertNotNil(encodingError, "encoding error should not be nil")
  725. if let error = encodingError as? AFError {
  726. XCTAssertTrue(error.isBodyPartFileIsDirectory)
  727. let expectedFailureReason = "The URL provided is a directory: \(directoryURL)"
  728. XCTAssertEqual(error.localizedDescription, expectedFailureReason, "error failure reason does not match expected value")
  729. } else {
  730. XCTFail("Error should be AFError.")
  731. }
  732. }
  733. func testThatWritingEncodedDataToExistingFileURLFails() {
  734. // Given
  735. let fileURL = temporaryFileURL()
  736. var writerError: Error?
  737. do {
  738. try "dummy data".write(to: fileURL, atomically: true, encoding: String.Encoding.utf8)
  739. } catch {
  740. writerError = error
  741. }
  742. let multipartFormData = MultipartFormData()
  743. let data = "Lorem ipsum dolor sit amet.".data(using: .utf8, allowLossyConversion: false)!
  744. multipartFormData.append(data, withName: "data")
  745. var encodingError: Error?
  746. // When
  747. do {
  748. try multipartFormData.writeEncodedData(to: fileURL)
  749. } catch {
  750. encodingError = error
  751. }
  752. // Then
  753. XCTAssertNil(writerError, "writer error should be nil")
  754. XCTAssertNotNil(encodingError, "encoding error should not be nil")
  755. if let encodingError = encodingError as? AFError {
  756. XCTAssertTrue(encodingError.isOutputStreamFileAlreadyExists)
  757. }
  758. }
  759. func testThatWritingEncodedDataToBadURLFails() {
  760. // Given
  761. let fileURL = URL(string: "/this/is/not/a/valid/url")!
  762. let multipartFormData = MultipartFormData()
  763. let data = "Lorem ipsum dolor sit amet.".data(using: .utf8, allowLossyConversion: false)!
  764. multipartFormData.append(data, withName: "data")
  765. var encodingError: Error?
  766. // When
  767. do {
  768. try multipartFormData.writeEncodedData(to: fileURL)
  769. } catch {
  770. encodingError = error
  771. }
  772. // Then
  773. XCTAssertNotNil(encodingError, "encoding error should not be nil")
  774. if let encodingError = encodingError as? AFError {
  775. XCTAssertTrue(encodingError.isOutputStreamURLInvalid)
  776. }
  777. }
  778. }