MultipartFormDataTests.swift 37 KB

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