MultipartFormDataTests.swift 37 KB

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