ParameterEncoderTests.swift 38 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198
  1. //
  2. // ParameterEncoderTests.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 XCTest
  26. final class JSONParameterEncoderTests: BaseTestCase {
  27. func testThatDataIsProperlyEncodedAndProperContentTypeIsSet() throws {
  28. // Given
  29. let encoder = JSONParameterEncoder()
  30. let request = Endpoint().urlRequest
  31. // When
  32. let newRequest = try encoder.encode(TestParameters.default, into: request)
  33. // Then
  34. XCTAssertEqual(newRequest.headers["Content-Type"], "application/json")
  35. XCTAssertEqual(newRequest.httpBody?.asString, "{\"property\":\"property\"}")
  36. }
  37. func testThatDataIsProperlyEncodedButContentTypeIsNotSetIfRequestAlreadyHasAContentType() throws {
  38. // Given
  39. let encoder = JSONParameterEncoder()
  40. var request = Endpoint().urlRequest
  41. request.headers.update(.contentType("type"))
  42. // When
  43. let newRequest = try encoder.encode(TestParameters.default, into: request)
  44. // Then
  45. XCTAssertEqual(newRequest.headers["Content-Type"], "type")
  46. XCTAssertEqual(newRequest.httpBody?.asString, "{\"property\":\"property\"}")
  47. }
  48. func testThatJSONEncoderCanBeCustomized() throws {
  49. // Given
  50. let jsonEncoder = JSONEncoder()
  51. jsonEncoder.outputFormatting = .prettyPrinted
  52. let encoder = JSONParameterEncoder(encoder: jsonEncoder)
  53. let request = Endpoint().urlRequest
  54. // When
  55. let newRequest = try encoder.encode(TestParameters.default, into: request)
  56. // Then
  57. let expected = """
  58. {
  59. "property" : "property"
  60. }
  61. """
  62. XCTAssertEqual(newRequest.httpBody?.asString, expected)
  63. }
  64. func testThatJSONEncoderDefaultWorks() throws {
  65. // Given
  66. let encoder = JSONParameterEncoder.default
  67. let request = Endpoint().urlRequest
  68. // When
  69. let encoded = try encoder.encode(TestParameters.default, into: request)
  70. // Then
  71. let expected = """
  72. {"property":"property"}
  73. """
  74. XCTAssertEqual(encoded.httpBody?.asString, expected)
  75. }
  76. func testThatJSONEncoderPrettyPrintedPrintsPretty() throws {
  77. // Given
  78. let encoder = JSONParameterEncoder.prettyPrinted
  79. let request = Endpoint().urlRequest
  80. // When
  81. let encoded = try encoder.encode(TestParameters.default, into: request)
  82. // Then
  83. let expected = """
  84. {
  85. "property" : "property"
  86. }
  87. """
  88. XCTAssertEqual(encoded.httpBody?.asString, expected)
  89. }
  90. }
  91. final class SortedKeysJSONParameterEncoderTests: BaseTestCase {
  92. func testTestJSONEncoderSortedKeysHasSortedKeys() throws {
  93. guard #available(macOS 10.13, iOS 11.0, tvOS 11.0, watchOS 4.0, *) else { return }
  94. // Given
  95. let encoder = JSONParameterEncoder.sortedKeys
  96. let request = Endpoint().urlRequest
  97. // When
  98. let encoded = try encoder.encode(["z": "z", "a": "a", "p": "p"], into: request)
  99. // Then
  100. let expected = """
  101. {"a":"a","p":"p","z":"z"}
  102. """
  103. XCTAssertEqual(encoded.httpBody?.asString, expected)
  104. }
  105. }
  106. final class URLEncodedFormParameterEncoderTests: BaseTestCase {
  107. func testThatQueryIsBodyEncodedAndProperContentTypeIsSetForPOSTRequest() throws {
  108. // Given
  109. let encoder = URLEncodedFormParameterEncoder()
  110. let request = Endpoint(method: .post).urlRequest
  111. // When
  112. let newRequest = try encoder.encode(TestParameters.default, into: request)
  113. // Then
  114. XCTAssertEqual(newRequest.headers["Content-Type"], "application/x-www-form-urlencoded; charset=utf-8")
  115. XCTAssertEqual(newRequest.httpBody?.asString, "property=property")
  116. }
  117. func testThatQueryIsBodyEncodedButContentTypeIsNotSetWhenRequestAlreadyHasContentType() throws {
  118. // Given
  119. let encoder = URLEncodedFormParameterEncoder()
  120. var request = Endpoint(method: .post).urlRequest
  121. request.headers.update(.contentType("type"))
  122. // When
  123. let newRequest = try encoder.encode(TestParameters.default, into: request)
  124. // Then
  125. XCTAssertEqual(newRequest.headers["Content-Type"], "type")
  126. XCTAssertEqual(newRequest.httpBody?.asString, "property=property")
  127. }
  128. func testThatEncoderCanBeCustomized() throws {
  129. // Given
  130. let urlEncoder = URLEncodedFormEncoder(boolEncoding: .literal)
  131. let encoder = URLEncodedFormParameterEncoder(encoder: urlEncoder)
  132. let request = Endpoint().urlRequest
  133. // When
  134. let newRequest = try encoder.encode(["bool": true], into: request)
  135. // Then
  136. let components = URLComponents(url: newRequest.url!, resolvingAgainstBaseURL: false)
  137. XCTAssertEqual(components?.percentEncodedQuery, "bool=true")
  138. }
  139. func testThatQueryIsInURLWhenDestinationIsURLAndMethodIsPOST() throws {
  140. // Given
  141. let encoder = URLEncodedFormParameterEncoder(destination: .queryString)
  142. let request = Endpoint(method: .post).urlRequest
  143. // When
  144. let newRequest = try encoder.encode(TestParameters.default, into: request)
  145. // Then
  146. let components = URLComponents(url: newRequest.url!, resolvingAgainstBaseURL: false)
  147. XCTAssertEqual(components?.percentEncodedQuery, "property=property")
  148. }
  149. func testThatQueryIsNilWhenEncodableResultsInAnEmptyString() throws {
  150. // Given
  151. let encoder = URLEncodedFormParameterEncoder(destination: .queryString)
  152. let request = Endpoint().urlRequest
  153. // When
  154. let newRequest = try encoder.encode([String: String](), into: request)
  155. // Then
  156. let components = URLComponents(url: newRequest.url!, resolvingAgainstBaseURL: false)
  157. XCTAssertNil(components?.percentEncodedQuery)
  158. }
  159. }
  160. final class URLEncodedFormEncoderTests: BaseTestCase {
  161. func testEncoderCanEncodeDictionary() {
  162. // Given
  163. let encoder = URLEncodedFormEncoder()
  164. let parameters = ["a": "a"]
  165. // When
  166. let result = Result<String, Error> { try encoder.encode(parameters) }
  167. // Then
  168. XCTAssertEqual(result.success, "a=a")
  169. }
  170. func testEncoderCanEncodeDecimal() {
  171. // Given
  172. let encoder = URLEncodedFormEncoder()
  173. let decimal: Decimal = 1.0
  174. let parameters = ["a": decimal]
  175. // When
  176. let result = Result<String, Error> { try encoder.encode(parameters) }
  177. // Then
  178. XCTAssertEqual(result.success, "a=1")
  179. }
  180. func testEncoderCanEncodeDecimalWithHighPrecision() {
  181. // Given
  182. let encoder = URLEncodedFormEncoder()
  183. let decimal: Decimal = 1.123456
  184. let parameters = ["a": decimal]
  185. // When
  186. let result = Result<String, Error> { try encoder.encode(parameters) }
  187. // Then
  188. XCTAssertEqual(result.success, "a=1.123456")
  189. }
  190. func testEncoderCanEncodeDouble() {
  191. // Given
  192. let encoder = URLEncodedFormEncoder()
  193. let parameters = ["a": 1.0]
  194. // When
  195. let result = Result<String, Error> { try encoder.encode(parameters) }
  196. // Then
  197. XCTAssertEqual(result.success, "a=1.0")
  198. }
  199. func testEncoderCanEncodeFloat() {
  200. // Given
  201. let encoder = URLEncodedFormEncoder()
  202. let parameters: [String: Float] = ["a": 1.0]
  203. // When
  204. let result = Result<String, Error> { try encoder.encode(parameters) }
  205. // Then
  206. XCTAssertEqual(result.success, "a=1.0")
  207. }
  208. func testEncoderCanEncodeInt8() {
  209. // Given
  210. let encoder = URLEncodedFormEncoder()
  211. let parameters: [String: Int8] = ["a": 1]
  212. // When
  213. let result = Result<String, Error> { try encoder.encode(parameters) }
  214. // Then
  215. XCTAssertEqual(result.success, "a=1")
  216. }
  217. func testEncoderCanEncodeInt16() {
  218. // Given
  219. let encoder = URLEncodedFormEncoder()
  220. let parameters: [String: Int16] = ["a": 1]
  221. // When
  222. let result = Result<String, Error> { try encoder.encode(parameters) }
  223. // Then
  224. XCTAssertEqual(result.success, "a=1")
  225. }
  226. func testEncoderCanEncodeInt32() {
  227. // Given
  228. let encoder = URLEncodedFormEncoder()
  229. let parameters: [String: Int32] = ["a": 1]
  230. // When
  231. let result = Result<String, Error> { try encoder.encode(parameters) }
  232. // Then
  233. XCTAssertEqual(result.success, "a=1")
  234. }
  235. func testEncoderCanEncodeInt64() {
  236. // Given
  237. let encoder = URLEncodedFormEncoder()
  238. let parameters: [String: Int64] = ["a": 1]
  239. // When
  240. let result = Result<String, Error> { try encoder.encode(parameters) }
  241. // Then
  242. XCTAssertEqual(result.success, "a=1")
  243. }
  244. func testEncoderCanEncodeUInt() {
  245. // Given
  246. let encoder = URLEncodedFormEncoder()
  247. let parameters: [String: UInt] = ["a": 1]
  248. // When
  249. let result = Result<String, Error> { try encoder.encode(parameters) }
  250. // Then
  251. XCTAssertEqual(result.success, "a=1")
  252. }
  253. func testEncoderCanEncodeUInt8() {
  254. // Given
  255. let encoder = URLEncodedFormEncoder()
  256. let parameters: [String: UInt8] = ["a": 1]
  257. // When
  258. let result = Result<String, Error> { try encoder.encode(parameters) }
  259. // Then
  260. XCTAssertEqual(result.success, "a=1")
  261. }
  262. func testEncoderCanEncodeUInt16() {
  263. // Given
  264. let encoder = URLEncodedFormEncoder()
  265. let parameters: [String: UInt16] = ["a": 1]
  266. // When
  267. let result = Result<String, Error> { try encoder.encode(parameters) }
  268. // Then
  269. XCTAssertEqual(result.success, "a=1")
  270. }
  271. func testEncoderCanEncodeUInt32() {
  272. // Given
  273. let encoder = URLEncodedFormEncoder()
  274. let parameters: [String: UInt32] = ["a": 1]
  275. // When
  276. let result = Result<String, Error> { try encoder.encode(parameters) }
  277. // Then
  278. XCTAssertEqual(result.success, "a=1")
  279. }
  280. func testEncoderCanEncodeUInt64() {
  281. // Given
  282. let encoder = URLEncodedFormEncoder()
  283. let parameters: [String: UInt64] = ["a": 1]
  284. // When
  285. let result = Result<String, Error> { try encoder.encode(parameters) }
  286. // Then
  287. XCTAssertEqual(result.success, "a=1")
  288. }
  289. func testThatNestedDictionariesCanHaveBracketKeyPathsByDefault() {
  290. // Given
  291. let encoder = URLEncodedFormEncoder()
  292. let parameters = ["a": ["b": "b"]]
  293. // When
  294. let result = Result<String, Error> { try encoder.encode(parameters) }
  295. // Then
  296. XCTAssertEqual(result.success, "a%5Bb%5D=b")
  297. }
  298. func testThatNestedDictionariesCanHaveExplicitBracketKeyPaths() {
  299. // Given
  300. let encoder = URLEncodedFormEncoder(keyPathEncoding: .brackets)
  301. let parameters = ["a": ["b": "b"]]
  302. // When
  303. let result = Result<String, Error> { try encoder.encode(parameters) }
  304. // Then
  305. XCTAssertEqual(result.success, "a%5Bb%5D=b")
  306. }
  307. func testThatNestedDictionariesCanHaveDottedKeyPaths() {
  308. // Given
  309. let encoder = URLEncodedFormEncoder(keyPathEncoding: .dots)
  310. let parameters = ["a": ["b": "b"]]
  311. // When
  312. let result = Result<String, Error> { try encoder.encode(parameters) }
  313. // Then
  314. XCTAssertEqual(result.success, "a.b=b")
  315. }
  316. func testThatNestedDictionariesCanHaveCustomKeyPaths() {
  317. // Given
  318. let encoder = URLEncodedFormEncoder(keyPathEncoding: .init { "-\($0)" })
  319. let parameters = ["a": ["b": "b"]]
  320. // When
  321. let result = Result<String, Error> { try encoder.encode(parameters) }
  322. // Then
  323. XCTAssertEqual(result.success, "a-b=b")
  324. }
  325. func testThatEncodableStructCanBeEncoded() {
  326. // Given
  327. let encoder = URLEncodedFormEncoder()
  328. let parameters = EncodableStruct()
  329. // When
  330. let result = Result<String, Error> { try encoder.encode(parameters) }
  331. // Then
  332. let expected = "five%5Ba%5D=a&four%5B%5D=1&four%5B%5D=2&four%5B%5D=3&one=one&seven%5Ba%5D=a&six%5Ba%5D%5Bb%5D=b&three=1&two=2"
  333. XCTAssertEqual(result.success, expected)
  334. }
  335. func testThatManuallyEncodableStructCanBeEncoded() {
  336. // Given
  337. let encoder = URLEncodedFormEncoder()
  338. let parameters = ManuallyEncodableStruct()
  339. // When
  340. let result = Result<String, Error> { try encoder.encode(parameters) }
  341. // Then
  342. let expected = "root%5B%5D%5B%5D%5B%5D=1&root%5B%5D%5B%5D%5B%5D=2&root%5B%5D%5B%5D%5B%5D=3&root%5B%5D%5B%5D=1&root%5B%5D%5B%5D=2&root%5B%5D%5B%5D=3&root%5B%5D%5Ba%5D%5Bstring%5D=string"
  343. XCTAssertEqual(result.success, expected)
  344. }
  345. func testThatEncodableClassWithNoInheritanceCanBeEncoded() {
  346. // Given
  347. let encoder = URLEncodedFormEncoder()
  348. let parameters = EncodableSuperclass()
  349. // When
  350. let result = Result<String, Error> { try encoder.encode(parameters) }
  351. // Then
  352. XCTAssertEqual(result.success, "one=one&three=1&two=2")
  353. }
  354. func testThatEncodableSubclassCanBeEncoded() {
  355. // Given
  356. let encoder = URLEncodedFormEncoder()
  357. let parameters = EncodableSubclass()
  358. // When
  359. let result = Result<String, Error> { try encoder.encode(parameters) }
  360. // Then
  361. let expected = "five%5Ba%5D=a&five%5Bb%5D=b&four%5B%5D=1&four%5B%5D=2&four%5B%5D=3&one=one&three=1&two=2"
  362. XCTAssertEqual(result.success, expected)
  363. }
  364. func testThatEncodableSubclassCanBeEncodedInImplementationOrderWhenAlphabetizeKeysIsFalse() {
  365. // Given
  366. let encoder = URLEncodedFormEncoder(alphabetizeKeyValuePairs: false)
  367. let parameters = EncodableStruct()
  368. // When
  369. let result = Result<String, Error> { try encoder.encode(parameters) }
  370. // Then
  371. let expected = "one=one&two=2&three=1&four%5B%5D=1&four%5B%5D=2&four%5B%5D=3&five%5Ba%5D=a&six%5Ba%5D%5Bb%5D=b&seven%5Ba%5D=a"
  372. XCTAssertEqual(result.success, expected)
  373. }
  374. func testThatManuallyEncodableSubclassCanBeEncoded() {
  375. // Given
  376. let encoder = URLEncodedFormEncoder()
  377. let parameters = ManuallyEncodableSubclass()
  378. // When
  379. let result = Result<String, Error> { try encoder.encode(parameters) }
  380. // Then
  381. let expected = "five%5Ba%5D=a&five%5Bb%5D=b&four%5Bfive%5D=2&four%5Bfour%5D=one"
  382. XCTAssertEqual(result.success, expected)
  383. }
  384. func testThatARootArrayCannotBeEncoded() {
  385. // Given
  386. let encoder = URLEncodedFormEncoder()
  387. let parameters = [1]
  388. // When
  389. let result = Result<String, Error> { try encoder.encode(parameters) }
  390. // Then
  391. XCTAssertFalse(result.isSuccess)
  392. }
  393. func testThatARootValueCannotBeEncoded() {
  394. // Given
  395. let encoder = URLEncodedFormEncoder()
  396. let parameters = "string"
  397. // When
  398. let result = Result<String, Error> { try encoder.encode(parameters) }
  399. // Then
  400. XCTAssertFalse(result.isSuccess)
  401. }
  402. func testThatEncodableSuperclassCanBeEncodedWithIndexInBrackets() {
  403. // Given
  404. let encoder = URLEncodedFormEncoder(arrayEncoding: .indexInBrackets)
  405. let parameters = ["foo": [EncodableSuperclass()]]
  406. // When
  407. let result = Result<String, Error> { try encoder.encode(parameters) }
  408. // Then
  409. XCTAssertEqual(result.success, "foo%5B0%5D%5Bone%5D=one&foo%5B0%5D%5Bthree%5D=1&foo%5B0%5D%5Btwo%5D=2")
  410. }
  411. func testThatEncodableSubclassCanBeEncodedWithIndexInBrackets() {
  412. // Given
  413. let encoder = URLEncodedFormEncoder(arrayEncoding: .indexInBrackets)
  414. let parameters = EncodableSubclass()
  415. // When
  416. let result = Result<String, Error> { try encoder.encode(parameters) }
  417. // Then
  418. let expected = "five%5Ba%5D=a&five%5Bb%5D=b&four%5B0%5D=1&four%5B1%5D=2&four%5B2%5D=3&one=one&three=1&two=2"
  419. XCTAssertEqual(result.success, expected)
  420. }
  421. func testThatManuallyEncodableSubclassCanBeEncodedWithIndexInBrackets() {
  422. // Given
  423. let encoder = URLEncodedFormEncoder(arrayEncoding: .indexInBrackets)
  424. let parameters = ManuallyEncodableSubclass()
  425. // When
  426. let result = Result<String, Error> { try encoder.encode(parameters) }
  427. // Then
  428. let expected = "five%5Ba%5D=a&five%5Bb%5D=b&four%5Bfive%5D=2&four%5Bfour%5D=one"
  429. XCTAssertEqual(result.success, expected)
  430. }
  431. func testThatEncodableStructCanBeEncodedWithIndexInBrackets() {
  432. // Given
  433. let encoder = URLEncodedFormEncoder(arrayEncoding: .indexInBrackets)
  434. let parameters = EncodableStruct()
  435. // When
  436. let result = Result<String, Error> { try encoder.encode(parameters) }
  437. // Then
  438. let expected = "five%5Ba%5D=a&four%5B0%5D=1&four%5B1%5D=2&four%5B2%5D=3&one=one&seven%5Ba%5D=a&six%5Ba%5D%5Bb%5D=b&three=1&two=2"
  439. XCTAssertEqual(result.success, expected)
  440. }
  441. func testThatManuallyEncodableStructCanBeEncodedWithIndexInBrackets() {
  442. // Given
  443. let encoder = URLEncodedFormEncoder(arrayEncoding: .indexInBrackets)
  444. let parameters = ManuallyEncodableStruct()
  445. // When
  446. let result = Result<String, Error> { try encoder.encode(parameters) }
  447. // then
  448. let expected = "root%5B0%5D%5B0%5D=1&root%5B0%5D%5B1%5D=2&root%5B0%5D%5B2%5D=3&root%5B1%5D%5Ba%5D%5Bstring%5D=string&root%5B2%5D%5B0%5D%5B0%5D=1&root%5B2%5D%5B0%5D%5B1%5D=2&root%5B2%5D%5B0%5D%5B2%5D=3"
  449. XCTAssertEqual(result.success, expected)
  450. }
  451. func testThatArrayNestedDictionaryIntValueCanBeEncodedWithIndexInBrackets() {
  452. // Given
  453. let encoder = URLEncodedFormEncoder(arrayEncoding: .indexInBrackets)
  454. let parameters = ["foo": [["bar": 2], ["qux": 3], ["quy": 4]]]
  455. // When
  456. let result = Result<String, Error> { try encoder.encode(parameters) }
  457. // Then
  458. XCTAssertEqual(result.success, "foo%5B0%5D%5Bbar%5D=2&foo%5B1%5D%5Bqux%5D=3&foo%5B2%5D%5Bquy%5D=4")
  459. }
  460. func testThatArrayNestedDictionaryStringValueCanBeEncodedWithIndexInBrackets() {
  461. // Given
  462. let encoder = URLEncodedFormEncoder(arrayEncoding: .indexInBrackets)
  463. let parameters = ["foo": [["bar": "2"], ["qux": "3"], ["quy": "4"]]]
  464. // When
  465. let result = Result<String, Error> { try encoder.encode(parameters) }
  466. // Then
  467. XCTAssertEqual(result.success, "foo%5B0%5D%5Bbar%5D=2&foo%5B1%5D%5Bqux%5D=3&foo%5B2%5D%5Bquy%5D=4")
  468. }
  469. func testThatArrayNestedDictionaryBoolValueCanBeEncodedWithIndexInBrackets() {
  470. // Given
  471. let encoder = URLEncodedFormEncoder(arrayEncoding: .indexInBrackets)
  472. let parameters = ["foo": [["bar": true], ["qux": false], ["quy": true]]]
  473. // When
  474. let result = Result<String, Error> { try encoder.encode(parameters) }
  475. // Then
  476. XCTAssertEqual(result.success, "foo%5B0%5D%5Bbar%5D=1&foo%5B1%5D%5Bqux%5D=0&foo%5B2%5D%5Bquy%5D=1")
  477. }
  478. func testThatArraysCanBeEncodedWithoutBrackets() {
  479. // Given
  480. let encoder = URLEncodedFormEncoder(arrayEncoding: .noBrackets)
  481. let parameters = ["array": [1, 2]]
  482. // When
  483. let result = Result<String, Error> { try encoder.encode(parameters) }
  484. // Then
  485. XCTAssertEqual(result.success, "array=1&array=2")
  486. }
  487. func testThatArraysCanBeEncodedWithCustomClosure() {
  488. // Given
  489. let encoder = URLEncodedFormEncoder(arrayEncoding: .custom { key, index in
  490. "\(key).\(index + 1)"
  491. })
  492. let parameters = ["array": [1, 2]]
  493. // When
  494. let result = Result<String, Error> { try encoder.encode(parameters) }
  495. // Then
  496. XCTAssertEqual(result.success, "array.1=1&array.2=2")
  497. }
  498. func testThatBoolsCanBeLiteralEncoded() {
  499. // Given
  500. let encoder = URLEncodedFormEncoder(boolEncoding: .literal)
  501. let parameters = ["bool": true]
  502. // When
  503. let result = Result<String, Error> { try encoder.encode(parameters) }
  504. // Then
  505. XCTAssertEqual(result.success, "bool=true")
  506. }
  507. func testThatDataCanBeEncoded() {
  508. // Given
  509. let encoder = URLEncodedFormEncoder()
  510. let parameters = ["data": Data("data".utf8)]
  511. // When
  512. let result = Result<String, Error> { try encoder.encode(parameters) }
  513. // Then
  514. XCTAssertEqual(result.success, "data=ZGF0YQ%3D%3D")
  515. }
  516. func testThatCustomDataEncodingFailsWhenErrorIsThrown() {
  517. // Given
  518. struct DataEncodingError: Error {}
  519. let encoder = URLEncodedFormEncoder(dataEncoding: .custom { _ in throw DataEncodingError() })
  520. let parameters = ["data": Data("data".utf8)]
  521. // When
  522. let result = Result<String, Error> { try encoder.encode(parameters) }
  523. // Then
  524. XCTAssertTrue(result.isFailure)
  525. XCTAssertTrue(result.failure is DataEncodingError)
  526. }
  527. func testThatDatesCanBeEncoded() {
  528. // Given
  529. let encoder = URLEncodedFormEncoder(dateEncoding: .deferredToDate)
  530. let parameters = ["date": Date(timeIntervalSinceReferenceDate: 123.456)]
  531. // When
  532. let result = Result<String, Error> { try encoder.encode(parameters) }
  533. // Then
  534. XCTAssertEqual(result.success, "date=123.456")
  535. }
  536. func testThatDatesCanBeEncodedAsSecondsSince1970() {
  537. // Given
  538. let encoder = URLEncodedFormEncoder(dateEncoding: .secondsSince1970)
  539. let parameters = ["date": Date(timeIntervalSinceReferenceDate: 123.456)]
  540. // When
  541. let result = Result<String, Error> { try encoder.encode(parameters) }
  542. // Then
  543. XCTAssertEqual(result.success, "date=978307323.456")
  544. }
  545. func testThatDatesCanBeEncodedAsMillisecondsSince1970() {
  546. // Given
  547. let encoder = URLEncodedFormEncoder(dateEncoding: .millisecondsSince1970)
  548. let parameters = ["date": Date(timeIntervalSinceReferenceDate: 123.456)]
  549. // When
  550. let result = Result<String, Error> { try encoder.encode(parameters) }
  551. // Then
  552. XCTAssertEqual(result.success, "date=978307323456.0")
  553. }
  554. func testThatDatesCanBeEncodedAsISO8601Formatted() {
  555. // Given
  556. let encoder = URLEncodedFormEncoder(dateEncoding: .iso8601)
  557. let parameters = ["date": Date(timeIntervalSinceReferenceDate: 123.456)]
  558. // When
  559. let result = Result<String, Error> { try encoder.encode(parameters) }
  560. // Then
  561. XCTAssertEqual(result.success, "date=2001-01-01T00%3A02%3A03Z")
  562. }
  563. func testThatDatesCanBeEncodedAsFormatted() {
  564. // Given
  565. let dateFormatter = DateFormatter()
  566. dateFormatter.dateFormat = "yyyy-MM-dd HH:mm:ss.SSSS"
  567. dateFormatter.timeZone = TimeZone(secondsFromGMT: 0)
  568. let encoder = URLEncodedFormEncoder(dateEncoding: .formatted(dateFormatter))
  569. let parameters = ["date": Date(timeIntervalSinceReferenceDate: 123.456)]
  570. // When
  571. let result = Result<String, Error> { try encoder.encode(parameters) }
  572. // Then
  573. XCTAssertEqual(result.success, "date=2001-01-01%2000%3A02%3A03.4560")
  574. }
  575. func testThatDatesCanBeEncodedAsCustomFormatted() {
  576. // Given
  577. let encoder = URLEncodedFormEncoder(dateEncoding: .custom { "\($0.timeIntervalSinceReferenceDate)" })
  578. let parameters = ["date": Date(timeIntervalSinceReferenceDate: 123.456)]
  579. // When
  580. let result = Result<String, Error> { try encoder.encode(parameters) }
  581. // Then
  582. XCTAssertEqual(result.success, "date=123.456")
  583. }
  584. func testEncoderThrowsErrorWhenCustomDateEncodingFails() {
  585. // Given
  586. struct DateEncodingError: Error {}
  587. let encoder = URLEncodedFormEncoder(dateEncoding: .custom { _ in throw DateEncodingError() })
  588. let parameters = ["date": Date(timeIntervalSinceReferenceDate: 123.456)]
  589. // When
  590. let result = Result<String, Error> { try encoder.encode(parameters) }
  591. // Then
  592. XCTAssertTrue(result.isFailure)
  593. XCTAssertTrue(result.failure is DateEncodingError)
  594. }
  595. func testThatKeysCanBeEncodedIntoSnakeCase() {
  596. // Given
  597. let encoder = URLEncodedFormEncoder(keyEncoding: .convertToSnakeCase)
  598. let parameters = ["oneTwoThree": "oneTwoThree"]
  599. // When
  600. let result = Result<String, Error> { try encoder.encode(parameters) }
  601. // Then
  602. XCTAssertEqual(result.success, "one_two_three=oneTwoThree")
  603. }
  604. func testThatKeysCanBeEncodedIntoKebabCase() {
  605. // Given
  606. let encoder = URLEncodedFormEncoder(keyEncoding: .convertToKebabCase)
  607. let parameters = ["oneTwoThree": "oneTwoThree"]
  608. // When
  609. let result = Result<String, Error> { try encoder.encode(parameters) }
  610. // Then
  611. XCTAssertEqual(result.success, "one-two-three=oneTwoThree")
  612. }
  613. func testThatKeysCanBeEncodedIntoACapitalizedString() {
  614. // Given
  615. let encoder = URLEncodedFormEncoder(keyEncoding: .capitalized)
  616. let parameters = ["oneTwoThree": "oneTwoThree"]
  617. // When
  618. let result = Result<String, Error> { try encoder.encode(parameters) }
  619. // Then
  620. XCTAssertEqual(result.success, "OneTwoThree=oneTwoThree")
  621. }
  622. func testThatKeysCanBeEncodedIntoALowercasedString() {
  623. // Given
  624. let encoder = URLEncodedFormEncoder(keyEncoding: .lowercased)
  625. let parameters = ["oneTwoThree": "oneTwoThree"]
  626. // When
  627. let result = Result<String, Error> { try encoder.encode(parameters) }
  628. // Then
  629. XCTAssertEqual(result.success, "onetwothree=oneTwoThree")
  630. }
  631. func testThatKeysCanBeEncodedIntoAnUppercasedString() {
  632. // Given
  633. let encoder = URLEncodedFormEncoder(keyEncoding: .uppercased)
  634. let parameters = ["oneTwoThree": "oneTwoThree"]
  635. // When
  636. let result = Result<String, Error> { try encoder.encode(parameters) }
  637. // Then
  638. XCTAssertEqual(result.success, "ONETWOTHREE=oneTwoThree")
  639. }
  640. func testThatKeysCanBeCustomEncoded() {
  641. // Given
  642. let encoder = URLEncodedFormEncoder(keyEncoding: .custom { _ in "A" })
  643. let parameters = ["oneTwoThree": "oneTwoThree"]
  644. // When
  645. let result = Result<String, Error> { try encoder.encode(parameters) }
  646. // Then
  647. XCTAssertEqual(result.success, "A=oneTwoThree")
  648. }
  649. func testThatNilCanBeEncodedByDroppingTheKeyByDefault() {
  650. // Given
  651. let encoder = URLEncodedFormEncoder()
  652. let parameters: [String: String?] = ["a": nil]
  653. // When
  654. let result = Result<String, Error> { try encoder.encode(parameters) }
  655. // Then
  656. XCTAssertEqual(result.success, "")
  657. }
  658. func testThatNilCanBeEncodedAsNull() {
  659. // Given
  660. let encoder = URLEncodedFormEncoder(nilEncoding: .null)
  661. let parameters: [String: String?] = ["a": nil]
  662. // When
  663. let result = Result<String, Error> { try encoder.encode(parameters) }
  664. // Then
  665. XCTAssertEqual(result.success, "a=null")
  666. }
  667. func testThatNilCanBeEncodedByDroppingTheKey() {
  668. // Given
  669. let encoder = URLEncodedFormEncoder(nilEncoding: .dropKey)
  670. let parameters: [String: String?] = ["a": nil]
  671. // When
  672. let result = Result<String, Error> { try encoder.encode(parameters) }
  673. // Then
  674. XCTAssertEqual(result.success, "")
  675. }
  676. func testThatNilCanBeEncodedByDroppingTheValue() {
  677. // Given
  678. let encoder = URLEncodedFormEncoder(nilEncoding: .dropValue)
  679. let parameters: [String: String?] = ["a": nil]
  680. // When
  681. let result = Result<String, Error> { try encoder.encode(parameters) }
  682. // Then
  683. XCTAssertEqual(result.success, "a=")
  684. }
  685. func testThatSpacesCanBeEncodedAsPluses() {
  686. // Given
  687. let encoder = URLEncodedFormEncoder(spaceEncoding: .plusReplaced)
  688. let parameters = ["spaces": "replace with spaces"]
  689. // When
  690. let result = Result<String, Error> { try encoder.encode(parameters) }
  691. // Then
  692. XCTAssertEqual(result.success, "spaces=replace+with+spaces")
  693. }
  694. func testThatEscapedCharactersCanBeCustomized() {
  695. // Given
  696. var allowed = CharacterSet.afURLQueryAllowed
  697. allowed.remove(charactersIn: "?/")
  698. let encoder = URLEncodedFormEncoder(allowedCharacters: allowed)
  699. let parameters = ["allowed": "?/"]
  700. // When
  701. let result = Result<String, Error> { try encoder.encode(parameters) }
  702. // Then
  703. XCTAssertEqual(result.success, "allowed=%3F%2F")
  704. }
  705. func testThatUnreservedCharactersAreNotPercentEscaped() {
  706. // Given
  707. let encoder = URLEncodedFormEncoder()
  708. let parameters = ["lowercase": "abcdefghijklmnopqrstuvwxyz",
  709. "numbers": "0123456789",
  710. "uppercase": "ABCDEFGHIJKLMNOPQRSTUVWXYZ"]
  711. // When
  712. let result = Result<String, Error> { try encoder.encode(parameters) }
  713. // Then
  714. let expected = "lowercase=abcdefghijklmnopqrstuvwxyz&numbers=0123456789&uppercase=ABCDEFGHIJKLMNOPQRSTUVWXYZ"
  715. XCTAssertEqual(result.success, expected)
  716. }
  717. func testThatReservedCharactersArePercentEscaped() {
  718. // Given
  719. let encoder = URLEncodedFormEncoder()
  720. let generalDelimiters = ":#[]@"
  721. let subDelimiters = "!$&'()*+,;="
  722. let parameters = ["reserved": "\(generalDelimiters)\(subDelimiters)"]
  723. // When
  724. let result = Result<String, Error> { try encoder.encode(parameters) }
  725. // Then
  726. XCTAssertEqual(result.success, "reserved=%3A%23%5B%5D%40%21%24%26%27%28%29%2A%2B%2C%3B%3D")
  727. }
  728. func testThatIllegalASCIICharactersArePercentEscaped() {
  729. // Given
  730. let encoder = URLEncodedFormEncoder()
  731. let parameters = ["illegal": " \"#%<>[]\\^`{}|"]
  732. // When
  733. let result = Result<String, Error> { try encoder.encode(parameters) }
  734. // Then
  735. XCTAssertEqual(result.success, "illegal=%20%22%23%25%3C%3E%5B%5D%5C%5E%60%7B%7D%7C")
  736. }
  737. func testThatAmpersandsInKeysAndValuesArePercentEscaped() {
  738. // Given
  739. let encoder = URLEncodedFormEncoder()
  740. let parameters = ["foo&bar": "baz&qux", "foobar": "bazqux"]
  741. // When
  742. let result = Result<String, Error> { try encoder.encode(parameters) }
  743. // Then
  744. XCTAssertEqual(result.success, "foo%26bar=baz%26qux&foobar=bazqux")
  745. }
  746. func testThatQuestionMarksInKeysAndValuesAreNotPercentEscaped() {
  747. // Given
  748. let encoder = URLEncodedFormEncoder()
  749. let parameters = ["?foo?": "?bar?"]
  750. // When
  751. let result = Result<String, Error> { try encoder.encode(parameters) }
  752. // Then
  753. XCTAssertEqual(result.success, "?foo?=?bar?")
  754. }
  755. func testThatSlashesInKeysAndValuesAreNotPercentEscaped() {
  756. // Given
  757. let encoder = URLEncodedFormEncoder()
  758. let parameters = ["foo": "/bar/baz/qux"]
  759. // When
  760. let result = Result<String, Error> { try encoder.encode(parameters) }
  761. // Then
  762. XCTAssertEqual(result.success, "foo=/bar/baz/qux")
  763. }
  764. func testThatSpacesInKeysAndValuesArePercentEscaped() {
  765. // Given
  766. let encoder = URLEncodedFormEncoder()
  767. let parameters = [" foo ": " bar "]
  768. // When
  769. let result = Result<String, Error> { try encoder.encode(parameters) }
  770. // Then
  771. XCTAssertEqual(result.success, "%20foo%20=%20bar%20")
  772. }
  773. func testThatPlusesInKeysAndValuesArePercentEscaped() {
  774. // Given
  775. let encoder = URLEncodedFormEncoder()
  776. let parameters = ["+foo+": "+bar+"]
  777. // When
  778. let result = Result<String, Error> { try encoder.encode(parameters) }
  779. // Then
  780. XCTAssertEqual(result.success, "%2Bfoo%2B=%2Bbar%2B")
  781. }
  782. func testThatPercentsInKeysAndValuesArePercentEscaped() {
  783. // Given
  784. let encoder = URLEncodedFormEncoder()
  785. let parameters = ["percent%": "%25"]
  786. // When
  787. let result = Result<String, Error> { try encoder.encode(parameters) }
  788. // Then
  789. XCTAssertEqual(result.success, "percent%25=%2525")
  790. }
  791. func testThatNonLatinCharactersArePercentEscaped() {
  792. // Given
  793. let encoder = URLEncodedFormEncoder()
  794. let parameters = ["french": "français",
  795. "japanese": "日本語",
  796. "arabic": "العربية",
  797. "emoji": "😃"]
  798. // When
  799. let result = Result<String, Error> { try encoder.encode(parameters) }
  800. // Then
  801. let expectedParameterValues = ["arabic=%D8%A7%D9%84%D8%B9%D8%B1%D8%A8%D9%8A%D8%A9",
  802. "emoji=%F0%9F%98%83",
  803. "french=fran%C3%A7ais",
  804. "japanese=%E6%97%A5%E6%9C%AC%E8%AA%9E"].joined(separator: "&")
  805. XCTAssertEqual(result.success, expectedParameterValues)
  806. }
  807. func testStringWithThousandsOfChineseCharactersIsPercentEscaped() {
  808. // Given
  809. let encoder = URLEncodedFormEncoder()
  810. let repeatedCount = 2000
  811. let parameters = ["chinese": String(repeating: "一二三四五六七八九十", count: repeatedCount)]
  812. // When
  813. let result = Result<String, Error> { try encoder.encode(parameters) }
  814. // Then
  815. let escaped = String(repeating: "%E4%B8%80%E4%BA%8C%E4%B8%89%E5%9B%9B%E4%BA%94%E5%85%AD%E4%B8%83%E5%85%AB%E4%B9%9D%E5%8D%81",
  816. count: repeatedCount)
  817. let expected = "chinese=\(escaped)"
  818. XCTAssertEqual(result.success, expected)
  819. }
  820. }
  821. final class StaticParameterEncoderInstanceTests: BaseTestCase {
  822. func takeParameterEncoder(_ parameterEncoder: ParameterEncoder) {
  823. _ = parameterEncoder
  824. }
  825. func testThatJSONParameterEncoderCanBeCreatedStaticallyFromProtocol() {
  826. // Given, When, Then
  827. takeParameterEncoder(.json())
  828. }
  829. func testThatURLEncodedFormParameterEncoderCanBeCreatedStaticallyFromProtocol() {
  830. // Given, When, Then
  831. takeParameterEncoder(.urlEncodedForm())
  832. }
  833. }
  834. private struct EncodableStruct: Encodable {
  835. let one = "one"
  836. let two = 2
  837. let three = true
  838. let four = [1, 2, 3]
  839. let five = ["a": "a"]
  840. let six = ["a": ["b": "b"]]
  841. let seven = NestedEncodableStruct()
  842. }
  843. private struct NestedEncodableStruct: Encodable {
  844. let a = "a"
  845. }
  846. private class EncodableSuperclass: Encodable {
  847. let one = "one"
  848. let two = 2
  849. let three = true
  850. }
  851. private final class EncodableSubclass: EncodableSuperclass {
  852. let four = [1, 2, 3]
  853. let five = ["a": "a", "b": "b"]
  854. private enum CodingKeys: String, CodingKey {
  855. case four, five
  856. }
  857. override func encode(to encoder: Encoder) throws {
  858. try super.encode(to: encoder)
  859. var container = encoder.container(keyedBy: CodingKeys.self)
  860. try container.encode(four, forKey: .four)
  861. try container.encode(five, forKey: .five)
  862. }
  863. }
  864. private final class ManuallyEncodableSubclass: EncodableSuperclass {
  865. let four = [1, 2, 3]
  866. let five = ["a": "a", "b": "b"]
  867. private enum CodingKeys: String, CodingKey {
  868. case four, five
  869. }
  870. override func encode(to encoder: Encoder) throws {
  871. var keyedContainer = encoder.container(keyedBy: CodingKeys.self)
  872. try keyedContainer.encode(four, forKey: .four)
  873. try keyedContainer.encode(five, forKey: .five)
  874. let superEncoder = keyedContainer.superEncoder()
  875. var superContainer = superEncoder.container(keyedBy: CodingKeys.self)
  876. try superContainer.encode(one, forKey: .four)
  877. let keyedSuperEncoder = keyedContainer.superEncoder(forKey: .four)
  878. var superKeyedContainer = keyedSuperEncoder.container(keyedBy: CodingKeys.self)
  879. try superKeyedContainer.encode(two, forKey: .five)
  880. var unkeyedContainer = keyedContainer.nestedUnkeyedContainer(forKey: .four)
  881. let unkeyedSuperEncoder = unkeyedContainer.superEncoder()
  882. var keyedUnkeyedSuperContainer = unkeyedSuperEncoder.container(keyedBy: CodingKeys.self)
  883. try keyedUnkeyedSuperContainer.encode(one, forKey: .four)
  884. }
  885. }
  886. private struct ManuallyEncodableStruct: Encodable {
  887. let a = ["string": "string"]
  888. let b = [1, 2, 3]
  889. private enum RootKey: String, CodingKey {
  890. case root
  891. }
  892. private enum TypeKeys: String, CodingKey {
  893. case a, b
  894. }
  895. func encode(to encoder: Encoder) throws {
  896. var container = encoder.container(keyedBy: RootKey.self)
  897. var nestedKeyedContainer = container.nestedContainer(keyedBy: TypeKeys.self, forKey: .root)
  898. try nestedKeyedContainer.encode(a, forKey: .a)
  899. var nestedUnkeyedContainer = container.nestedUnkeyedContainer(forKey: .root)
  900. try nestedUnkeyedContainer.encode(b)
  901. var nestedUnkeyedKeyedContainer = nestedUnkeyedContainer.nestedContainer(keyedBy: TypeKeys.self)
  902. try nestedUnkeyedKeyedContainer.encode(a, forKey: .a)
  903. var nestedUnkeyedUnkeyedContainer = nestedUnkeyedContainer.nestedUnkeyedContainer()
  904. try nestedUnkeyedUnkeyedContainer.encode(b)
  905. }
  906. }
  907. private struct FailingOptionalStruct: Encodable {
  908. enum TestedContainer {
  909. case keyed, unkeyed
  910. }
  911. enum CodingKeys: String, CodingKey { case a }
  912. let testedContainer: TestedContainer
  913. func encode(to encoder: Encoder) throws {
  914. var container = encoder.container(keyedBy: CodingKeys.self)
  915. switch testedContainer {
  916. case .keyed:
  917. var nested = container.nestedContainer(keyedBy: CodingKeys.self, forKey: .a)
  918. try nested.encodeNil(forKey: .a)
  919. case .unkeyed:
  920. var nested = container.nestedUnkeyedContainer(forKey: .a)
  921. try nested.encodeNil()
  922. }
  923. }
  924. }