MultipartFormDataTests.swift 35 KB

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