grape 1.1.0 → 1.7.1

Sign up to get free protection for your applications and to get access to all the features.
Files changed (306) hide show
  1. checksums.yaml +4 -4
  2. data/CHANGELOG.md +370 -44
  3. data/CONTRIBUTING.md +32 -1
  4. data/LICENSE +1 -1
  5. data/README.md +683 -87
  6. data/UPGRADING.md +481 -17
  7. data/grape.gemspec +15 -4
  8. data/lib/grape/api/helpers.rb +2 -0
  9. data/lib/grape/api/instance.rb +279 -0
  10. data/lib/grape/api.rb +144 -176
  11. data/lib/grape/config.rb +34 -0
  12. data/lib/grape/content_types.rb +34 -0
  13. data/lib/grape/cookies.rb +4 -0
  14. data/lib/grape/dry_types.rb +12 -0
  15. data/lib/grape/dsl/api.rb +1 -1
  16. data/lib/grape/dsl/callbacks.rb +21 -1
  17. data/lib/grape/dsl/configuration.rb +1 -1
  18. data/lib/grape/dsl/desc.rb +41 -23
  19. data/lib/grape/dsl/headers.rb +7 -2
  20. data/lib/grape/dsl/helpers.rb +10 -7
  21. data/lib/grape/dsl/inside_route.rb +118 -62
  22. data/lib/grape/dsl/logger.rb +2 -0
  23. data/lib/grape/dsl/middleware.rb +11 -4
  24. data/lib/grape/dsl/parameters.rb +33 -19
  25. data/lib/grape/dsl/request_response.rb +12 -9
  26. data/lib/grape/dsl/routing.rb +22 -13
  27. data/lib/grape/dsl/settings.rb +10 -6
  28. data/lib/grape/dsl/validations.rb +19 -14
  29. data/lib/grape/eager_load.rb +20 -0
  30. data/lib/grape/endpoint.rb +67 -58
  31. data/lib/grape/error_formatter/base.rb +2 -0
  32. data/lib/grape/error_formatter/json.rb +11 -7
  33. data/lib/grape/error_formatter/txt.rb +2 -0
  34. data/lib/grape/error_formatter/xml.rb +4 -6
  35. data/lib/grape/error_formatter.rb +4 -2
  36. data/lib/grape/exceptions/base.rb +23 -16
  37. data/lib/grape/exceptions/empty_message_body.rb +11 -0
  38. data/lib/grape/exceptions/incompatible_option_values.rb +2 -0
  39. data/lib/grape/exceptions/invalid_accept_header.rb +2 -0
  40. data/lib/grape/exceptions/invalid_formatter.rb +2 -0
  41. data/lib/grape/exceptions/invalid_message_body.rb +2 -0
  42. data/lib/grape/exceptions/invalid_response.rb +11 -0
  43. data/lib/grape/exceptions/invalid_version_header.rb +2 -0
  44. data/lib/grape/exceptions/invalid_versioner_option.rb +2 -0
  45. data/lib/grape/exceptions/invalid_with_option_for_represent.rb +2 -0
  46. data/lib/grape/exceptions/method_not_allowed.rb +2 -0
  47. data/lib/grape/exceptions/missing_group_type.rb +10 -1
  48. data/lib/grape/exceptions/missing_mime_type.rb +2 -0
  49. data/lib/grape/exceptions/missing_option.rb +2 -0
  50. data/lib/grape/exceptions/missing_vendor_option.rb +2 -0
  51. data/lib/grape/exceptions/too_many_multipart_files.rb +11 -0
  52. data/lib/grape/exceptions/unknown_options.rb +2 -0
  53. data/lib/grape/exceptions/unknown_parameter.rb +2 -0
  54. data/lib/grape/exceptions/unknown_validator.rb +2 -0
  55. data/lib/grape/exceptions/unsupported_group_type.rb +10 -1
  56. data/lib/grape/exceptions/validation.rb +5 -8
  57. data/lib/grape/exceptions/validation_array_errors.rb +2 -0
  58. data/lib/grape/exceptions/validation_errors.rb +16 -13
  59. data/lib/grape/extensions/active_support/hash_with_indifferent_access.rb +4 -3
  60. data/lib/grape/extensions/deep_mergeable_hash.rb +2 -0
  61. data/lib/grape/extensions/deep_symbolize_hash.rb +2 -0
  62. data/lib/grape/extensions/hash.rb +2 -0
  63. data/lib/grape/extensions/hashie/mash.rb +2 -0
  64. data/lib/grape/formatter/json.rb +3 -0
  65. data/lib/grape/formatter/serializable_hash.rb +4 -1
  66. data/lib/grape/formatter/txt.rb +2 -0
  67. data/lib/grape/formatter/xml.rb +3 -0
  68. data/lib/grape/formatter.rb +5 -3
  69. data/lib/grape/http/headers.rb +50 -18
  70. data/lib/grape/locale/en.yml +11 -8
  71. data/lib/grape/middleware/auth/base.rb +7 -7
  72. data/lib/grape/middleware/auth/dsl.rb +9 -2
  73. data/lib/grape/middleware/auth/strategies.rb +2 -0
  74. data/lib/grape/middleware/auth/strategy_info.rb +2 -0
  75. data/lib/grape/middleware/base.rb +13 -8
  76. data/lib/grape/middleware/error.rb +22 -17
  77. data/lib/grape/middleware/filter.rb +2 -0
  78. data/lib/grape/middleware/formatter.rb +12 -10
  79. data/lib/grape/middleware/globals.rb +2 -0
  80. data/lib/grape/middleware/helpers.rb +12 -0
  81. data/lib/grape/middleware/stack.rb +16 -6
  82. data/lib/grape/middleware/versioner/accept_version_header.rb +5 -5
  83. data/lib/grape/middleware/versioner/header.rb +13 -9
  84. data/lib/grape/middleware/versioner/param.rb +4 -1
  85. data/lib/grape/middleware/versioner/parse_media_type_patch.rb +5 -1
  86. data/lib/grape/middleware/versioner/path.rb +5 -1
  87. data/lib/grape/middleware/versioner.rb +2 -0
  88. data/lib/grape/namespace.rb +14 -2
  89. data/lib/grape/parser/json.rb +3 -1
  90. data/lib/grape/parser/xml.rb +3 -1
  91. data/lib/grape/parser.rb +4 -2
  92. data/lib/grape/path.rb +16 -3
  93. data/lib/grape/presenters/presenter.rb +2 -0
  94. data/lib/grape/request.rb +21 -9
  95. data/lib/grape/router/attribute_translator.rb +41 -8
  96. data/lib/grape/router/pattern.rb +21 -17
  97. data/lib/grape/router/route.rb +15 -29
  98. data/lib/grape/router.rb +36 -29
  99. data/lib/grape/{serve_file → serve_stream}/file_body.rb +3 -1
  100. data/lib/grape/{serve_file → serve_stream}/sendfile_response.rb +3 -1
  101. data/lib/grape/{serve_file/file_response.rb → serve_stream/stream_response.rb} +10 -8
  102. data/lib/grape/types/invalid_value.rb +8 -0
  103. data/lib/grape/util/base_inheritable.rb +43 -0
  104. data/lib/grape/util/cache.rb +20 -0
  105. data/lib/grape/util/endpoint_configuration.rb +8 -0
  106. data/lib/grape/util/env.rb +19 -17
  107. data/lib/grape/util/inheritable_setting.rb +3 -3
  108. data/lib/grape/util/inheritable_values.rb +7 -25
  109. data/lib/grape/util/json.rb +4 -0
  110. data/lib/grape/util/lazy_block.rb +27 -0
  111. data/lib/grape/util/lazy_object.rb +43 -0
  112. data/lib/grape/util/lazy_value.rb +99 -0
  113. data/lib/grape/util/registrable.rb +2 -0
  114. data/lib/grape/util/reverse_stackable_values.rb +10 -35
  115. data/lib/grape/util/stackable_values.rb +21 -34
  116. data/lib/grape/util/strict_hash_configuration.rb +3 -1
  117. data/lib/grape/util/xml.rb +2 -0
  118. data/lib/grape/validations/attributes_doc.rb +58 -0
  119. data/lib/grape/validations/attributes_iterator.rb +16 -6
  120. data/lib/grape/validations/multiple_attributes_iterator.rb +13 -0
  121. data/lib/grape/validations/params_scope.rb +174 -94
  122. data/lib/grape/validations/single_attribute_iterator.rb +24 -0
  123. data/lib/grape/validations/types/array_coercer.rb +63 -0
  124. data/lib/grape/validations/types/build_coercer.rb +47 -49
  125. data/lib/grape/validations/types/custom_type_coercer.rb +30 -51
  126. data/lib/grape/validations/types/custom_type_collection_coercer.rb +10 -25
  127. data/lib/grape/validations/types/dry_type_coercer.rb +72 -0
  128. data/lib/grape/validations/types/file.rb +22 -18
  129. data/lib/grape/validations/types/invalid_value.rb +17 -0
  130. data/lib/grape/validations/types/json.rb +47 -39
  131. data/lib/grape/validations/types/multiple_type_coercer.rb +14 -33
  132. data/lib/grape/validations/types/primitive_coercer.rb +75 -0
  133. data/lib/grape/validations/types/set_coercer.rb +38 -0
  134. data/lib/grape/validations/types/variant_collection_coercer.rb +5 -13
  135. data/lib/grape/validations/types.rb +106 -63
  136. data/lib/grape/validations/validator_factory.rb +8 -11
  137. data/lib/grape/validations/validators/all_or_none_of_validator.rb +16 -0
  138. data/lib/grape/validations/validators/allow_blank_validator.rb +20 -0
  139. data/lib/grape/validations/validators/as_validator.rb +14 -0
  140. data/lib/grape/validations/validators/at_least_one_of_validator.rb +15 -0
  141. data/lib/grape/validations/validators/base.rb +84 -68
  142. data/lib/grape/validations/validators/coerce_validator.rb +75 -0
  143. data/lib/grape/validations/validators/default_validator.rb +51 -0
  144. data/lib/grape/validations/validators/exactly_one_of_validator.rb +17 -0
  145. data/lib/grape/validations/validators/except_values_validator.rb +24 -0
  146. data/lib/grape/validations/validators/multiple_params_base.rb +27 -16
  147. data/lib/grape/validations/validators/mutual_exclusion_validator.rb +16 -0
  148. data/lib/grape/validations/validators/presence_validator.rb +15 -0
  149. data/lib/grape/validations/validators/regexp_validator.rb +16 -0
  150. data/lib/grape/validations/validators/same_as_validator.rb +29 -0
  151. data/lib/grape/validations/validators/values_validator.rb +88 -0
  152. data/lib/grape/validations.rb +18 -6
  153. data/lib/grape/version.rb +3 -1
  154. data/lib/grape.rb +175 -94
  155. data/spec/grape/api/custom_validations_spec.rb +117 -44
  156. data/spec/grape/api/deeply_included_options_spec.rb +4 -4
  157. data/spec/grape/api/defines_boolean_in_params_spec.rb +38 -0
  158. data/spec/grape/api/documentation_spec.rb +59 -0
  159. data/spec/grape/api/inherited_helpers_spec.rb +1 -1
  160. data/spec/grape/api/instance_spec.rb +103 -0
  161. data/spec/grape/api/invalid_format_spec.rb +3 -1
  162. data/spec/grape/api/namespace_parameters_in_route_spec.rb +1 -1
  163. data/spec/grape/api/nested_helpers_spec.rb +1 -1
  164. data/spec/grape/api/optional_parameters_in_route_spec.rb +1 -1
  165. data/spec/grape/api/parameters_modification_spec.rb +2 -2
  166. data/spec/grape/api/patch_method_helpers_spec.rb +1 -1
  167. data/spec/grape/api/recognize_path_spec.rb +2 -2
  168. data/spec/grape/api/required_parameters_in_route_spec.rb +1 -1
  169. data/spec/grape/api/required_parameters_with_invalid_method_spec.rb +1 -1
  170. data/spec/grape/api/routes_with_requirements_spec.rb +59 -0
  171. data/spec/grape/api/shared_helpers_exactly_one_of_spec.rb +10 -16
  172. data/spec/grape/api/shared_helpers_spec.rb +1 -1
  173. data/spec/grape/api_remount_spec.rb +473 -0
  174. data/spec/grape/api_spec.rb +995 -231
  175. data/spec/grape/config_spec.rb +17 -0
  176. data/spec/grape/dsl/callbacks_spec.rb +3 -2
  177. data/spec/grape/dsl/desc_spec.rb +43 -17
  178. data/spec/grape/dsl/headers_spec.rb +40 -10
  179. data/spec/grape/dsl/helpers_spec.rb +6 -5
  180. data/spec/grape/dsl/inside_route_spec.rb +189 -38
  181. data/spec/grape/dsl/logger_spec.rb +17 -19
  182. data/spec/grape/dsl/middleware_spec.rb +11 -2
  183. data/spec/grape/dsl/parameters_spec.rb +3 -1
  184. data/spec/grape/dsl/request_response_spec.rb +8 -7
  185. data/spec/grape/dsl/routing_spec.rb +22 -9
  186. data/spec/grape/dsl/settings_spec.rb +1 -1
  187. data/spec/grape/dsl/validations_spec.rb +1 -16
  188. data/spec/grape/endpoint/declared_spec.rb +846 -0
  189. data/spec/grape/endpoint_spec.rb +136 -577
  190. data/spec/grape/entity_spec.rb +31 -24
  191. data/spec/grape/exceptions/base_spec.rb +81 -0
  192. data/spec/grape/exceptions/body_parse_errors_spec.rb +4 -1
  193. data/spec/grape/exceptions/invalid_accept_header_spec.rb +65 -23
  194. data/spec/grape/exceptions/invalid_formatter_spec.rb +1 -1
  195. data/spec/grape/exceptions/invalid_response_spec.rb +11 -0
  196. data/spec/grape/exceptions/invalid_versioner_option_spec.rb +2 -2
  197. data/spec/grape/exceptions/missing_group_type_spec.rb +21 -0
  198. data/spec/grape/exceptions/missing_mime_type_spec.rb +1 -1
  199. data/spec/grape/exceptions/missing_option_spec.rb +2 -2
  200. data/spec/grape/exceptions/unknown_options_spec.rb +1 -1
  201. data/spec/grape/exceptions/unknown_validator_spec.rb +1 -1
  202. data/spec/grape/exceptions/unsupported_group_type_spec.rb +23 -0
  203. data/spec/grape/exceptions/validation_errors_spec.rb +21 -15
  204. data/spec/grape/exceptions/validation_spec.rb +6 -4
  205. data/spec/grape/extensions/param_builders/hash_spec.rb +8 -8
  206. data/spec/grape/extensions/param_builders/hash_with_indifferent_access_spec.rb +9 -9
  207. data/spec/grape/extensions/param_builders/hashie/mash_spec.rb +9 -9
  208. data/spec/grape/integration/global_namespace_function_spec.rb +2 -2
  209. data/spec/grape/integration/rack_sendfile_spec.rb +14 -10
  210. data/spec/grape/integration/rack_spec.rb +25 -8
  211. data/spec/grape/loading_spec.rb +9 -9
  212. data/spec/grape/middleware/auth/base_spec.rb +2 -1
  213. data/spec/grape/middleware/auth/dsl_spec.rb +19 -10
  214. data/spec/grape/middleware/auth/strategies_spec.rb +62 -22
  215. data/spec/grape/middleware/base_spec.rb +36 -17
  216. data/spec/grape/middleware/error_spec.rb +11 -4
  217. data/spec/grape/middleware/exception_spec.rb +112 -162
  218. data/spec/grape/middleware/formatter_spec.rb +65 -29
  219. data/spec/grape/middleware/globals_spec.rb +8 -5
  220. data/spec/grape/middleware/stack_spec.rb +25 -13
  221. data/spec/grape/middleware/versioner/accept_version_header_spec.rb +3 -2
  222. data/spec/grape/middleware/versioner/header_spec.rb +37 -14
  223. data/spec/grape/middleware/versioner/param_spec.rb +8 -2
  224. data/spec/grape/middleware/versioner/path_spec.rb +6 -2
  225. data/spec/grape/middleware/versioner_spec.rb +2 -2
  226. data/spec/grape/named_api_spec.rb +19 -0
  227. data/spec/grape/parser_spec.rb +10 -6
  228. data/spec/grape/path_spec.rb +53 -53
  229. data/spec/grape/presenters/presenter_spec.rb +8 -7
  230. data/spec/grape/request_spec.rb +29 -3
  231. data/spec/grape/util/inheritable_setting_spec.rb +9 -8
  232. data/spec/grape/util/inheritable_values_spec.rb +5 -3
  233. data/spec/grape/util/reverse_stackable_values_spec.rb +5 -2
  234. data/spec/grape/util/stackable_values_spec.rb +10 -7
  235. data/spec/grape/util/strict_hash_configuration_spec.rb +2 -1
  236. data/spec/grape/validations/attributes_doc_spec.rb +153 -0
  237. data/spec/grape/validations/instance_behaivour_spec.rb +13 -14
  238. data/spec/grape/validations/multiple_attributes_iterator_spec.rb +40 -0
  239. data/spec/grape/validations/params_scope_spec.rb +568 -99
  240. data/spec/grape/validations/single_attribute_iterator_spec.rb +57 -0
  241. data/spec/grape/validations/types/array_coercer_spec.rb +33 -0
  242. data/spec/grape/validations/types/primitive_coercer_spec.rb +150 -0
  243. data/spec/grape/validations/types/set_coercer_spec.rb +32 -0
  244. data/spec/grape/validations/types_spec.rb +44 -45
  245. data/spec/grape/validations/validators/all_or_none_spec.rb +134 -32
  246. data/spec/grape/validations/validators/allow_blank_spec.rb +137 -141
  247. data/spec/grape/validations/validators/at_least_one_of_spec.rb +169 -31
  248. data/spec/grape/validations/validators/coerce_spec.rb +491 -151
  249. data/spec/grape/validations/validators/default_spec.rb +242 -78
  250. data/spec/grape/validations/validators/exactly_one_of_spec.rb +198 -40
  251. data/spec/grape/validations/validators/except_values_spec.rb +6 -5
  252. data/spec/grape/validations/validators/mutual_exclusion_spec.rb +181 -30
  253. data/spec/grape/validations/validators/presence_spec.rb +45 -2
  254. data/spec/grape/validations/validators/regexp_spec.rb +27 -33
  255. data/spec/grape/validations/validators/same_as_spec.rb +57 -0
  256. data/spec/grape/validations/validators/values_spec.rb +227 -180
  257. data/spec/grape/validations_spec.rb +502 -72
  258. data/spec/integration/eager_load/eager_load_spec.rb +15 -0
  259. data/spec/integration/multi_json/json_spec.rb +2 -2
  260. data/spec/integration/multi_xml/xml_spec.rb +2 -2
  261. data/spec/shared/versioning_examples.rb +34 -29
  262. data/spec/spec_helper.rb +31 -5
  263. data/spec/support/basic_auth_encode_helpers.rb +3 -1
  264. data/spec/support/chunks.rb +14 -0
  265. data/spec/support/content_type_helpers.rb +2 -0
  266. data/spec/support/endpoint_faker.rb +2 -0
  267. data/spec/support/file_streamer.rb +2 -0
  268. data/spec/support/integer_helpers.rb +2 -0
  269. data/spec/support/versioned_helpers.rb +8 -8
  270. metadata +111 -61
  271. data/Appraisals +0 -32
  272. data/Dangerfile +0 -2
  273. data/Gemfile +0 -33
  274. data/Gemfile.lock +0 -231
  275. data/Guardfile +0 -10
  276. data/RELEASING.md +0 -111
  277. data/Rakefile +0 -25
  278. data/benchmark/simple.rb +0 -27
  279. data/benchmark/simple_with_type_coercer.rb +0 -22
  280. data/gemfiles/multi_json.gemfile +0 -35
  281. data/gemfiles/multi_xml.gemfile +0 -35
  282. data/gemfiles/rack_1.5.2.gemfile +0 -35
  283. data/gemfiles/rack_edge.gemfile +0 -35
  284. data/gemfiles/rails_3.gemfile +0 -36
  285. data/gemfiles/rails_4.gemfile +0 -35
  286. data/gemfiles/rails_5.gemfile +0 -35
  287. data/gemfiles/rails_edge.gemfile +0 -35
  288. data/lib/grape/extensions/deep_hash_with_indifferent_access.rb +0 -18
  289. data/lib/grape/util/content_types.rb +0 -26
  290. data/lib/grape/validations/types/virtus_collection_patch.rb +0 -16
  291. data/lib/grape/validations/validators/all_or_none.rb +0 -20
  292. data/lib/grape/validations/validators/allow_blank.rb +0 -16
  293. data/lib/grape/validations/validators/as.rb +0 -15
  294. data/lib/grape/validations/validators/at_least_one_of.rb +0 -20
  295. data/lib/grape/validations/validators/coerce.rb +0 -74
  296. data/lib/grape/validations/validators/default.rb +0 -48
  297. data/lib/grape/validations/validators/exactly_one_of.rb +0 -29
  298. data/lib/grape/validations/validators/except_values.rb +0 -20
  299. data/lib/grape/validations/validators/mutual_exclusion.rb +0 -25
  300. data/lib/grape/validations/validators/presence.rb +0 -10
  301. data/lib/grape/validations/validators/regexp.rb +0 -11
  302. data/lib/grape/validations/validators/values.rb +0 -71
  303. data/pkg/grape-0.17.0.gem +0 -0
  304. data/pkg/grape-0.19.0.gem +0 -0
  305. data/spec/grape/dsl/configuration_spec.rb +0 -14
  306. data/spec/grape/validations/attributes_iterator_spec.rb +0 -4
@@ -1,23 +1,138 @@
1
- require 'spec_helper'
1
+ # frozen_string_literal: true
2
2
 
3
- describe Grape::Validations::AllowBlankValidator do
4
- module ValidationsSpec
5
- module AllowBlankValidatorSpec
6
- class API < Grape::API
7
- default_format :json
3
+ describe Grape::Validations::Validators::AllowBlankValidator do
4
+ let_it_be(:app) do
5
+ Class.new(Grape::API) do
6
+ default_format :json
8
7
 
9
- params do
8
+ params do
9
+ requires :name, allow_blank: false
10
+ end
11
+ get '/disallow_blank'
12
+
13
+ params do
14
+ optional :name, type: String, allow_blank: false
15
+ end
16
+ get '/opt_disallow_string_blank'
17
+
18
+ params do
19
+ optional :name, allow_blank: false
20
+ end
21
+ get '/disallow_blank_optional_param'
22
+
23
+ params do
24
+ requires :name, allow_blank: true
25
+ end
26
+ get '/allow_blank'
27
+
28
+ params do
29
+ requires :val, type: DateTime, allow_blank: true
30
+ end
31
+ get '/allow_datetime_blank'
32
+
33
+ params do
34
+ requires :val, type: DateTime, allow_blank: false
35
+ end
36
+ get '/disallow_datetime_blank'
37
+
38
+ params do
39
+ requires :val, type: DateTime
40
+ end
41
+ get '/default_allow_datetime_blank'
42
+
43
+ params do
44
+ requires :val, type: Date, allow_blank: true
45
+ end
46
+ get '/allow_date_blank'
47
+
48
+ params do
49
+ requires :val, type: Integer, allow_blank: true
50
+ end
51
+ get '/allow_integer_blank'
52
+
53
+ params do
54
+ requires :val, type: Float, allow_blank: true
55
+ end
56
+ get '/allow_float_blank'
57
+
58
+ params do
59
+ requires :val, type: Integer, allow_blank: true
60
+ end
61
+ get '/allow_integer_blank'
62
+
63
+ params do
64
+ requires :val, type: Symbol, allow_blank: true
65
+ end
66
+ get '/allow_symbol_blank'
67
+
68
+ params do
69
+ requires :val, type: Grape::API::Boolean, allow_blank: true
70
+ end
71
+ get '/allow_boolean_blank'
72
+
73
+ params do
74
+ requires :val, type: Grape::API::Boolean, allow_blank: false
75
+ end
76
+ get '/disallow_boolean_blank'
77
+
78
+ params do
79
+ optional :user, type: Hash do
80
+ requires :name, allow_blank: false
81
+ end
82
+ end
83
+ get '/disallow_blank_required_param_in_an_optional_group'
84
+
85
+ params do
86
+ optional :user, type: Hash do
87
+ requires :name, type: Date, allow_blank: true
88
+ end
89
+ end
90
+ get '/allow_blank_date_param_in_an_optional_group'
91
+
92
+ params do
93
+ optional :user, type: Hash do
94
+ optional :name, allow_blank: false
95
+ requires :age
96
+ end
97
+ end
98
+ get '/disallow_blank_optional_param_in_an_optional_group'
99
+
100
+ params do
101
+ requires :user, type: Hash do
10
102
  requires :name, allow_blank: false
11
103
  end
12
- get '/disallow_blank'
104
+ end
105
+ get '/disallow_blank_required_param_in_a_required_group'
106
+
107
+ params do
108
+ requires :user, type: Hash do
109
+ requires :name, allow_blank: false
110
+ end
111
+ end
112
+ get '/disallow_string_value_in_a_required_hash_group'
113
+
114
+ params do
115
+ requires :user, type: Hash do
116
+ optional :name, allow_blank: false
117
+ end
118
+ end
119
+ get '/disallow_blank_optional_param_in_a_required_group'
120
+
121
+ params do
122
+ optional :user, type: Hash do
123
+ optional :name, allow_blank: false
124
+ end
125
+ end
126
+ get '/disallow_string_value_in_an_optional_hash_group'
13
127
 
128
+ resources :custom_message do
14
129
  params do
15
- optional :name, type: String, allow_blank: false
130
+ requires :name, allow_blank: { value: false, message: 'has no value' }
16
131
  end
17
- get '/opt_disallow_string_blank'
132
+ get
18
133
 
19
134
  params do
20
- optional :name, allow_blank: false
135
+ optional :name, allow_blank: { value: false, message: 'has no value' }
21
136
  end
22
137
  get '/disallow_blank_optional_param'
23
138
 
@@ -32,7 +147,7 @@ describe Grape::Validations::AllowBlankValidator do
32
147
  get '/allow_datetime_blank'
33
148
 
34
149
  params do
35
- requires :val, type: DateTime, allow_blank: false
150
+ requires :val, type: DateTime, allow_blank: { value: false, message: 'has no value' }
36
151
  end
37
152
  get '/disallow_datetime_blank'
38
153
 
@@ -67,18 +182,18 @@ describe Grape::Validations::AllowBlankValidator do
67
182
  get '/allow_symbol_blank'
68
183
 
69
184
  params do
70
- requires :val, type: Boolean, allow_blank: true
185
+ requires :val, type: Grape::API::Boolean, allow_blank: true
71
186
  end
72
187
  get '/allow_boolean_blank'
73
188
 
74
189
  params do
75
- requires :val, type: Boolean, allow_blank: false
190
+ requires :val, type: Grape::API::Boolean, allow_blank: { value: false, message: 'has no value' }
76
191
  end
77
192
  get '/disallow_boolean_blank'
78
193
 
79
194
  params do
80
195
  optional :user, type: Hash do
81
- requires :name, allow_blank: false
196
+ requires :name, allow_blank: { value: false, message: 'has no value' }
82
197
  end
83
198
  end
84
199
  get '/disallow_blank_required_param_in_an_optional_group'
@@ -92,7 +207,7 @@ describe Grape::Validations::AllowBlankValidator do
92
207
 
93
208
  params do
94
209
  optional :user, type: Hash do
95
- optional :name, allow_blank: false
210
+ optional :name, allow_blank: { value: false, message: 'has no value' }
96
211
  requires :age
97
212
  end
98
213
  end
@@ -100,156 +215,35 @@ describe Grape::Validations::AllowBlankValidator do
100
215
 
101
216
  params do
102
217
  requires :user, type: Hash do
103
- requires :name, allow_blank: false
218
+ requires :name, allow_blank: { value: false, message: 'has no value' }
104
219
  end
105
220
  end
106
221
  get '/disallow_blank_required_param_in_a_required_group'
107
222
 
108
223
  params do
109
224
  requires :user, type: Hash do
110
- requires :name, allow_blank: false
225
+ requires :name, allow_blank: { value: false, message: 'has no value' }
111
226
  end
112
227
  end
113
228
  get '/disallow_string_value_in_a_required_hash_group'
114
229
 
115
230
  params do
116
231
  requires :user, type: Hash do
117
- optional :name, allow_blank: false
232
+ optional :name, allow_blank: { value: false, message: 'has no value' }
118
233
  end
119
234
  end
120
235
  get '/disallow_blank_optional_param_in_a_required_group'
121
236
 
122
237
  params do
123
238
  optional :user, type: Hash do
124
- optional :name, allow_blank: false
125
- end
126
- end
127
- get '/disallow_string_value_in_an_optional_hash_group'
128
-
129
- resources :custom_message do
130
- params do
131
- requires :name, allow_blank: { value: false, message: 'has no value' }
132
- end
133
- get
134
-
135
- params do
136
239
  optional :name, allow_blank: { value: false, message: 'has no value' }
137
240
  end
138
- get '/disallow_blank_optional_param'
139
-
140
- params do
141
- requires :name, allow_blank: true
142
- end
143
- get '/allow_blank'
144
-
145
- params do
146
- requires :val, type: DateTime, allow_blank: true
147
- end
148
- get '/allow_datetime_blank'
149
-
150
- params do
151
- requires :val, type: DateTime, allow_blank: { value: false, message: 'has no value' }
152
- end
153
- get '/disallow_datetime_blank'
154
-
155
- params do
156
- requires :val, type: DateTime
157
- end
158
- get '/default_allow_datetime_blank'
159
-
160
- params do
161
- requires :val, type: Date, allow_blank: true
162
- end
163
- get '/allow_date_blank'
164
-
165
- params do
166
- requires :val, type: Integer, allow_blank: true
167
- end
168
- get '/allow_integer_blank'
169
-
170
- params do
171
- requires :val, type: Float, allow_blank: true
172
- end
173
- get '/allow_float_blank'
174
-
175
- params do
176
- requires :val, type: Integer, allow_blank: true
177
- end
178
- get '/allow_integer_blank'
179
-
180
- params do
181
- requires :val, type: Symbol, allow_blank: true
182
- end
183
- get '/allow_symbol_blank'
184
-
185
- params do
186
- requires :val, type: Boolean, allow_blank: true
187
- end
188
- get '/allow_boolean_blank'
189
-
190
- params do
191
- requires :val, type: Boolean, allow_blank: { value: false, message: 'has no value' }
192
- end
193
- get '/disallow_boolean_blank'
194
-
195
- params do
196
- optional :user, type: Hash do
197
- requires :name, allow_blank: { value: false, message: 'has no value' }
198
- end
199
- end
200
- get '/disallow_blank_required_param_in_an_optional_group'
201
-
202
- params do
203
- optional :user, type: Hash do
204
- requires :name, type: Date, allow_blank: true
205
- end
206
- end
207
- get '/allow_blank_date_param_in_an_optional_group'
208
-
209
- params do
210
- optional :user, type: Hash do
211
- optional :name, allow_blank: { value: false, message: 'has no value' }
212
- requires :age
213
- end
214
- end
215
- get '/disallow_blank_optional_param_in_an_optional_group'
216
-
217
- params do
218
- requires :user, type: Hash do
219
- requires :name, allow_blank: { value: false, message: 'has no value' }
220
- end
221
- end
222
- get '/disallow_blank_required_param_in_a_required_group'
223
-
224
- params do
225
- requires :user, type: Hash do
226
- requires :name, allow_blank: { value: false, message: 'has no value' }
227
- end
228
- end
229
- get '/disallow_string_value_in_a_required_hash_group'
230
-
231
- params do
232
- requires :user, type: Hash do
233
- optional :name, allow_blank: { value: false, message: 'has no value' }
234
- end
235
- end
236
- get '/disallow_blank_optional_param_in_a_required_group'
237
-
238
- params do
239
- optional :user, type: Hash do
240
- optional :name, allow_blank: { value: false, message: 'has no value' }
241
- end
242
- end
243
- get '/disallow_string_value_in_an_optional_hash_group'
244
241
  end
242
+ get '/disallow_string_value_in_an_optional_hash_group'
245
243
  end
246
244
  end
247
245
  end
248
246
 
249
- def app
250
- ValidationsSpec::AllowBlankValidatorSpec::API
251
- end
252
-
253
247
  context 'invalid input' do
254
248
  it 'refuses empty string' do
255
249
  get '/disallow_blank', name: ''
@@ -287,10 +281,12 @@ describe Grape::Validations::AllowBlankValidator do
287
281
  get '/custom_message', name: ''
288
282
  expect(last_response.body).to eq('{"error":"name has no value"}')
289
283
  end
284
+
290
285
  it 'refuses empty string for an optional param' do
291
286
  get '/custom_message/disallow_blank_optional_param', name: ''
292
287
  expect(last_response.body).to eq('{"error":"name has no value"}')
293
288
  end
289
+
294
290
  it 'refuses only whitespaces' do
295
291
  get '/custom_message', name: ' '
296
292
  expect(last_response.body).to eq('{"error":"name has no value"}')
@@ -1,66 +1,204 @@
1
- require 'spec_helper'
1
+ # frozen_string_literal: true
2
2
 
3
- describe Grape::Validations::AtLeastOneOfValidator do
4
- describe '#validate!' do
5
- let(:scope) do
6
- Struct.new(:opts) do
7
- def params(arg)
8
- arg
3
+ describe Grape::Validations::Validators::AtLeastOneOfValidator do
4
+ let_it_be(:app) do
5
+ Class.new(Grape::API) do
6
+ rescue_from Grape::Exceptions::ValidationErrors do |e|
7
+ error!(e.errors.transform_keys! { |key| key.join(',') }, 400)
8
+ end
9
+
10
+ params do
11
+ optional :beer, :wine, :grapefruit
12
+ at_least_one_of :beer, :wine, :grapefruit
13
+ end
14
+ post do
15
+ end
16
+
17
+ params do
18
+ optional :beer, :wine, :grapefruit, :other
19
+ at_least_one_of :beer, :wine, :grapefruit
20
+ end
21
+ post 'mixed-params' do
22
+ end
23
+
24
+ params do
25
+ optional :beer, :wine, :grapefruit
26
+ at_least_one_of :beer, :wine, :grapefruit, message: 'you should choose something'
27
+ end
28
+ post '/custom-message' do
29
+ end
30
+
31
+ params do
32
+ requires :item, type: Hash do
33
+ optional :beer, :wine, :grapefruit
34
+ at_least_one_of :beer, :wine, :grapefruit, message: 'fail'
9
35
  end
36
+ end
37
+ post '/nested-hash' do
38
+ end
10
39
 
11
- def required?; end
40
+ params do
41
+ requires :items, type: Array do
42
+ optional :beer, :wine, :grapefruit
43
+ at_least_one_of :beer, :wine, :grapefruit, message: 'fail'
44
+ end
45
+ end
46
+ post '/nested-array' do
47
+ end
48
+
49
+ params do
50
+ requires :items, type: Array do
51
+ requires :nested_items, type: Array do
52
+ optional :beer, :wine, :grapefruit
53
+ at_least_one_of :beer, :wine, :grapefruit, message: 'fail'
54
+ end
55
+ end
56
+ end
57
+ post '/deeply-nested-array' do
12
58
  end
13
59
  end
14
- let(:at_least_one_of_params) { %i[beer wine grapefruit] }
15
- let(:validator) { described_class.new(at_least_one_of_params, {}, false, scope.new) }
60
+ end
61
+
62
+ describe '#validate!' do
63
+ subject(:validate) { post path, params }
16
64
 
17
65
  context 'when all restricted params are present' do
66
+ let(:path) { '/' }
18
67
  let(:params) { { beer: true, wine: true, grapefruit: true } }
19
68
 
20
- it 'does not raise a validation exception' do
21
- expect(validator.validate!(params)).to eql params
69
+ it 'does not return a validation error' do
70
+ validate
71
+ expect(last_response.status).to eq 201
22
72
  end
23
73
 
24
74
  context 'mixed with other params' do
25
- let(:mixed_params) { params.merge!(other: true, andanother: true) }
75
+ let(:path) { '/mixed-params' }
76
+ let(:params) { { beer: true, wine: true, grapefruit: true, other: true } }
26
77
 
27
- it 'does not raise a validation exception' do
28
- expect(validator.validate!(mixed_params)).to eql mixed_params
78
+ it 'does not return a validation error' do
79
+ validate
80
+ expect(last_response.status).to eq 201
29
81
  end
30
82
  end
31
83
  end
32
84
 
33
85
  context 'when a subset of restricted params are present' do
86
+ let(:path) { '/' }
34
87
  let(:params) { { beer: true, grapefruit: true } }
35
88
 
36
- it 'does not raise a validation exception' do
37
- expect(validator.validate!(params)).to eql params
89
+ it 'does not return a validation error' do
90
+ validate
91
+ expect(last_response.status).to eq 201
38
92
  end
39
93
  end
40
94
 
41
- context 'when params keys come as strings' do
42
- let(:params) { { 'beer' => true, 'grapefruit' => true } }
95
+ context 'when none of the restricted params is selected' do
96
+ let(:path) { '/' }
97
+ let(:params) { { other: true } }
43
98
 
44
- it 'does not raise a validation exception' do
45
- expect(validator.validate!(params)).to eql params
99
+ it 'returns a validation error' do
100
+ validate
101
+ expect(last_response.status).to eq 400
102
+ expect(JSON.parse(last_response.body)).to eq(
103
+ 'beer,wine,grapefruit' => ['are missing, at least one parameter must be provided']
104
+ )
46
105
  end
47
- end
48
106
 
49
- context 'when none of the restricted params is selected' do
50
- let(:params) { { somethingelse: true } }
107
+ context 'when custom message is specified' do
108
+ let(:path) { '/custom-message' }
51
109
 
52
- it 'raises a validation exception' do
53
- expect do
54
- validator.validate! params
55
- end.to raise_error(Grape::Exceptions::Validation)
110
+ it 'returns a validation error' do
111
+ validate
112
+ expect(last_response.status).to eq 400
113
+ expect(JSON.parse(last_response.body)).to eq(
114
+ 'beer,wine,grapefruit' => ['you should choose something']
115
+ )
116
+ end
56
117
  end
57
118
  end
58
119
 
59
120
  context 'when exactly one of the restricted params is selected' do
60
- let(:params) { { beer: true, somethingelse: true } }
121
+ let(:path) { '/' }
122
+ let(:params) { { beer: true } }
61
123
 
62
- it 'does not raise a validation exception' do
63
- expect(validator.validate!(params)).to eql params
124
+ it 'does not return a validation error' do
125
+ validate
126
+ expect(last_response.status).to eq 201
127
+ end
128
+ end
129
+
130
+ context 'when restricted params are nested inside hash' do
131
+ let(:path) { '/nested-hash' }
132
+
133
+ context 'when at least one of them is present' do
134
+ let(:params) { { item: { beer: true, wine: true } } }
135
+
136
+ it 'does not return a validation error' do
137
+ validate
138
+ expect(last_response.status).to eq 201
139
+ end
140
+ end
141
+
142
+ context 'when none of them are present' do
143
+ let(:params) { { item: { other: true } } }
144
+
145
+ it 'returns a validation error with full names of the params' do
146
+ validate
147
+ expect(last_response.status).to eq 400
148
+ expect(JSON.parse(last_response.body)).to eq(
149
+ 'item[beer],item[wine],item[grapefruit]' => ['fail']
150
+ )
151
+ end
152
+ end
153
+ end
154
+
155
+ context 'when restricted params are nested inside array' do
156
+ let(:path) { '/nested-array' }
157
+
158
+ context 'when at least one of them is present' do
159
+ let(:params) { { items: [{ beer: true, wine: true }, { grapefruit: true }] } }
160
+
161
+ it 'does not return a validation error' do
162
+ validate
163
+ expect(last_response.status).to eq 201
164
+ end
165
+ end
166
+
167
+ context 'when none of them are present' do
168
+ let(:params) { { items: [{ beer: true, other: true }, { other: true }] } }
169
+
170
+ it 'returns a validation error with full names of the params' do
171
+ validate
172
+ expect(last_response.status).to eq 400
173
+ expect(JSON.parse(last_response.body)).to eq(
174
+ 'items[1][beer],items[1][wine],items[1][grapefruit]' => ['fail']
175
+ )
176
+ end
177
+ end
178
+ end
179
+
180
+ context 'when restricted params are deeply nested' do
181
+ let(:path) { '/deeply-nested-array' }
182
+
183
+ context 'when at least one of them is present' do
184
+ let(:params) { { items: [{ nested_items: [{ wine: true }] }] } }
185
+
186
+ it 'does not return a validation error' do
187
+ validate
188
+ expect(last_response.status).to eq 201
189
+ end
190
+ end
191
+
192
+ context 'when none of them are present' do
193
+ let(:params) { { items: [{ nested_items: [{ other: true }] }] } }
194
+
195
+ it 'returns a validation error with full names of the params' do
196
+ validate
197
+ expect(last_response.status).to eq 400
198
+ expect(JSON.parse(last_response.body)).to eq(
199
+ 'items[0][nested_items][0][beer],items[0][nested_items][0][wine],items[0][nested_items][0][grapefruit]' => ['fail']
200
+ )
201
+ end
64
202
  end
65
203
  end
66
204
  end