MultipartFormDataTests.swift 36 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942
  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. 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 = 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 = 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 = 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 = 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 = 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 = 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 = 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 testWritingEncodedStreamBodyPartToDisk() {
  483. // Given
  484. let fileURL = temporaryFileURL
  485. let multipartFormData = MultipartFormData()
  486. let unicornImageURL = url(forResource: "unicorn", withExtension: "png")
  487. let unicornDataLength = UInt64((try! Data(contentsOf: unicornImageURL)).count)
  488. let unicornStream = InputStream(url: unicornImageURL)!
  489. multipartFormData.append(unicornStream,
  490. withLength: unicornDataLength,
  491. name: "unicorn",
  492. fileName: "unicorn.png",
  493. mimeType: "image/png")
  494. var encodingError: Error?
  495. // When
  496. do {
  497. try multipartFormData.writeEncodedData(to: fileURL)
  498. } catch {
  499. encodingError = error
  500. }
  501. // Then
  502. XCTAssertNil(encodingError, "encoding error should be nil")
  503. if let fileData = try? Data(contentsOf: fileURL) {
  504. let boundary = multipartFormData.boundary
  505. var expectedFileData = Data()
  506. expectedFileData.append(BoundaryGenerator.boundaryData(boundaryType: .initial, boundaryKey: boundary))
  507. expectedFileData.append(Data((
  508. "Content-Disposition: form-data; name=\"unicorn\"; filename=\"unicorn.png\"\(crlf)" +
  509. "Content-Type: image/png\(crlf)\(crlf)").utf8
  510. )
  511. )
  512. expectedFileData.append(try! Data(contentsOf: unicornImageURL))
  513. expectedFileData.append(BoundaryGenerator.boundaryData(boundaryType: .final, boundaryKey: boundary))
  514. XCTAssertEqual(fileData, expectedFileData, "file data should match expected file data")
  515. } else {
  516. XCTFail("file data should not be nil")
  517. }
  518. }
  519. func testWritingMultipleEncodedStreamBodyPartsToDisk() {
  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. let rainbowImageURL = url(forResource: "rainbow", withExtension: "jpg")
  527. let rainbowDataLength = UInt64((try! Data(contentsOf: rainbowImageURL)).count)
  528. let rainbowStream = InputStream(url: rainbowImageURL)!
  529. multipartFormData.append(unicornStream,
  530. withLength: unicornDataLength,
  531. name: "unicorn",
  532. fileName: "unicorn.png",
  533. mimeType: "image/png")
  534. multipartFormData.append(rainbowStream,
  535. withLength: rainbowDataLength,
  536. name: "rainbow",
  537. fileName: "rainbow.jpg",
  538. mimeType: "image/jpeg")
  539. var encodingError: Error?
  540. // When
  541. do {
  542. try multipartFormData.writeEncodedData(to: fileURL)
  543. } catch {
  544. encodingError = error
  545. }
  546. // Then
  547. XCTAssertNil(encodingError, "encoding error should be nil")
  548. if let fileData = try? Data(contentsOf: fileURL) {
  549. let boundary = multipartFormData.boundary
  550. var expectedFileData = Data()
  551. expectedFileData.append(BoundaryGenerator.boundaryData(boundaryType: .initial, boundaryKey: boundary))
  552. expectedFileData.append(Data((
  553. "Content-Disposition: form-data; name=\"unicorn\"; filename=\"unicorn.png\"\(crlf)" +
  554. "Content-Type: image/png\(crlf)\(crlf)").utf8
  555. )
  556. )
  557. expectedFileData.append(try! Data(contentsOf: unicornImageURL))
  558. expectedFileData.append(BoundaryGenerator.boundaryData(boundaryType: .encapsulated, boundaryKey: boundary))
  559. expectedFileData.append(Data((
  560. "Content-Disposition: form-data; name=\"rainbow\"; filename=\"rainbow.jpg\"\(crlf)" +
  561. "Content-Type: image/jpeg\(crlf)\(crlf)").utf8
  562. )
  563. )
  564. expectedFileData.append(try! Data(contentsOf: rainbowImageURL))
  565. expectedFileData.append(BoundaryGenerator.boundaryData(boundaryType: .final, boundaryKey: boundary))
  566. XCTAssertEqual(fileData, expectedFileData, "file data should match expected file data")
  567. } else {
  568. XCTFail("file data should not be nil")
  569. }
  570. }
  571. func testWritingMultipleEncodedBodyPartsWithVaryingTypesToDisk() {
  572. // Given
  573. let fileURL = temporaryFileURL
  574. let multipartFormData = MultipartFormData()
  575. let loremData = Data("Lorem ipsum.".utf8)
  576. let unicornImageURL = url(forResource: "unicorn", withExtension: "png")
  577. let rainbowImageURL = url(forResource: "rainbow", withExtension: "jpg")
  578. let rainbowDataLength = UInt64((try! Data(contentsOf: rainbowImageURL)).count)
  579. let rainbowStream = InputStream(url: rainbowImageURL)!
  580. multipartFormData.append(loremData, withName: "lorem")
  581. multipartFormData.append(unicornImageURL, withName: "unicorn")
  582. multipartFormData.append(rainbowStream,
  583. withLength: rainbowDataLength,
  584. name: "rainbow",
  585. fileName: "rainbow.jpg",
  586. mimeType: "image/jpeg")
  587. var encodingError: Error?
  588. // When
  589. do {
  590. try multipartFormData.writeEncodedData(to: fileURL)
  591. } catch {
  592. encodingError = error
  593. }
  594. // Then
  595. XCTAssertNil(encodingError, "encoding error should be nil")
  596. if let fileData = try? Data(contentsOf: fileURL) {
  597. let boundary = multipartFormData.boundary
  598. var expectedFileData = Data()
  599. expectedFileData.append(BoundaryGenerator.boundaryData(boundaryType: .initial, boundaryKey: boundary))
  600. expectedFileData.append(Data(
  601. "Content-Disposition: form-data; name=\"lorem\"\(crlf)\(crlf)".utf8
  602. )
  603. )
  604. expectedFileData.append(loremData)
  605. expectedFileData.append(BoundaryGenerator.boundaryData(boundaryType: .encapsulated, boundaryKey: boundary))
  606. expectedFileData.append(Data((
  607. "Content-Disposition: form-data; name=\"unicorn\"; filename=\"unicorn.png\"\(crlf)" +
  608. "Content-Type: image/png\(crlf)\(crlf)").utf8
  609. )
  610. )
  611. expectedFileData.append(try! Data(contentsOf: unicornImageURL))
  612. expectedFileData.append(BoundaryGenerator.boundaryData(boundaryType: .encapsulated, boundaryKey: boundary))
  613. expectedFileData.append(Data((
  614. "Content-Disposition: form-data; name=\"rainbow\"; filename=\"rainbow.jpg\"\(crlf)" +
  615. "Content-Type: image/jpeg\(crlf)\(crlf)").utf8
  616. )
  617. )
  618. expectedFileData.append(try! Data(contentsOf: rainbowImageURL))
  619. expectedFileData.append(BoundaryGenerator.boundaryData(boundaryType: .final, boundaryKey: boundary))
  620. XCTAssertEqual(fileData, expectedFileData, "file data should match expected file data")
  621. } else {
  622. XCTFail("file data should not be nil")
  623. }
  624. }
  625. }
  626. // MARK: -
  627. class MultipartFormDataFailureTestCase: BaseTestCase {
  628. func testThatAppendingFileBodyPartWithInvalidLastPathComponentReturnsError() {
  629. // Given
  630. let fileURL = NSURL(string: "")! as URL
  631. let multipartFormData = MultipartFormData()
  632. multipartFormData.append(fileURL, withName: "empty_data")
  633. var encodingError: Error?
  634. // When
  635. do {
  636. _ = try multipartFormData.encode()
  637. } catch {
  638. encodingError = error
  639. }
  640. // Then
  641. XCTAssertNotNil(encodingError, "encoding error should not be nil")
  642. XCTAssertEqual(encodingError?.asAFError?.isBodyPartFilenameInvalid, true)
  643. XCTAssertEqual(encodingError?.asAFError?.url, fileURL)
  644. }
  645. func testThatAppendingFileBodyPartThatIsNotFileURLReturnsError() {
  646. // Given
  647. let fileURL = URL(string: "https://example.com/image.jpg")!
  648. let multipartFormData = MultipartFormData()
  649. multipartFormData.append(fileURL, withName: "empty_data")
  650. var encodingError: Error?
  651. // When
  652. do {
  653. _ = try multipartFormData.encode()
  654. } catch {
  655. encodingError = error
  656. }
  657. // Then
  658. XCTAssertNotNil(encodingError, "encoding error should not be nil")
  659. XCTAssertEqual(encodingError?.asAFError?.isBodyPartURLInvalid, true)
  660. XCTAssertEqual(encodingError?.asAFError?.url, fileURL)
  661. }
  662. func testThatAppendingFileBodyPartThatIsNotReachableReturnsError() {
  663. // Given
  664. let filePath = (NSTemporaryDirectory() as NSString).appendingPathComponent("does_not_exist.jpg")
  665. let fileURL = URL(fileURLWithPath: filePath)
  666. let multipartFormData = MultipartFormData()
  667. multipartFormData.append(fileURL, withName: "empty_data")
  668. var encodingError: Error?
  669. // When
  670. do {
  671. _ = try multipartFormData.encode()
  672. } catch {
  673. encodingError = error
  674. }
  675. // Then
  676. XCTAssertNotNil(encodingError, "encoding error should not be nil")
  677. XCTAssertEqual(encodingError?.asAFError?.isBodyPartFileNotReachableWithError, true)
  678. XCTAssertEqual(encodingError?.asAFError?.url, fileURL)
  679. }
  680. func testThatAppendingFileBodyPartThatIsDirectoryReturnsError() {
  681. // Given
  682. let directoryURL = URL(fileURLWithPath: NSTemporaryDirectory(), isDirectory: true)
  683. let multipartFormData = MultipartFormData()
  684. multipartFormData.append(directoryURL, withName: "empty_data", fileName: "empty", mimeType: "application/octet")
  685. var encodingError: Error?
  686. // When
  687. do {
  688. _ = try multipartFormData.encode()
  689. } catch {
  690. encodingError = error
  691. }
  692. // Then
  693. XCTAssertNotNil(encodingError, "encoding error should not be nil")
  694. XCTAssertEqual(encodingError?.asAFError?.isBodyPartFileIsDirectory, true)
  695. XCTAssertEqual(encodingError?.asAFError?.url, directoryURL)
  696. }
  697. func testThatWritingEncodedDataToExistingFileURLFails() {
  698. // Given
  699. let fileURL = temporaryFileURL
  700. var writerError: Error?
  701. do {
  702. try "dummy data".write(to: fileURL, atomically: true, encoding: .utf8)
  703. } catch {
  704. writerError = error
  705. }
  706. let multipartFormData = MultipartFormData()
  707. let data = Data("Lorem ipsum dolor sit amet.".utf8)
  708. multipartFormData.append(data, withName: "data")
  709. var encodingError: Error?
  710. // When
  711. do {
  712. try multipartFormData.writeEncodedData(to: fileURL)
  713. } catch {
  714. encodingError = error
  715. }
  716. // Then
  717. XCTAssertNil(writerError, "writer error should be nil")
  718. XCTAssertNotNil(encodingError, "encoding error should not be nil")
  719. XCTAssertEqual(encodingError?.asAFError?.isOutputStreamFileAlreadyExists, true)
  720. }
  721. func testThatWritingEncodedDataToBadURLFails() {
  722. // Given
  723. let fileURL = URL(string: "/this/is/not/a/valid/url")!
  724. let multipartFormData = MultipartFormData()
  725. let data = Data("Lorem ipsum dolor sit amet.".utf8)
  726. multipartFormData.append(data, withName: "data")
  727. var encodingError: Error?
  728. // When
  729. do {
  730. try multipartFormData.writeEncodedData(to: fileURL)
  731. } catch {
  732. encodingError = error
  733. }
  734. // Then
  735. XCTAssertNotNil(encodingError, "encoding error should not be nil")
  736. XCTAssertEqual(encodingError?.asAFError?.isOutputStreamURLInvalid, true)
  737. }
  738. func testThatStreamBodyPartHasUnexpectedLength() {
  739. // Given
  740. let multipartFormData = MultipartFormData()
  741. let data = Data("Lorem ipsum dolor sit amet.".utf8)
  742. multipartFormData.append(data, withName: "data")
  743. var firstError: Error?
  744. var secondError: Error?
  745. // When
  746. do {
  747. _ = try multipartFormData.encode()
  748. } catch {
  749. firstError = error
  750. }
  751. do {
  752. _ = try multipartFormData.encode()
  753. } catch {
  754. secondError = error
  755. }
  756. XCTAssertNil(firstError, "firstError should be nil")
  757. XCTAssertNotNil(secondError, "secondError should not be nil")
  758. XCTAssertEqual(secondError?.asAFError?.isInputStreamReadFailed, true)
  759. XCTAssert(secondError?.asAFError?.underlyingError is AFError.UnexpectedInputStreamLength)
  760. }
  761. }