MultipartFormDataTests.swift 36 KB

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