praxis 2.0.pre.16 → 2.0.pre.20

Sign up to get free protection for your applications and to get access to all the features.
Files changed (235) hide show
  1. checksums.yaml +4 -4
  2. data/.rubocop.yml +54 -0
  3. data/.simplecov +3 -1
  4. data/.travis.yml +2 -1
  5. data/CHANGELOG.md +22 -0
  6. data/CONTRIBUTING.md +2 -79
  7. data/Gemfile +5 -1
  8. data/Guardfile +6 -4
  9. data/LICENSE +0 -2
  10. data/MAINTAINERS.md +1 -0
  11. data/README.md +15 -22
  12. data/Rakefile +4 -2
  13. data/bin/praxis +55 -58
  14. data/lib/praxis/action_definition/headers_dsl_compiler.rb +5 -6
  15. data/lib/praxis/action_definition.rb +65 -95
  16. data/lib/praxis/api_definition.rb +21 -29
  17. data/lib/praxis/api_general_info.rb +55 -66
  18. data/lib/praxis/application.rb +15 -32
  19. data/lib/praxis/blueprint.rb +80 -73
  20. data/lib/praxis/bootloader.rb +24 -33
  21. data/lib/praxis/bootloader_stages/environment.rb +5 -10
  22. data/lib/praxis/bootloader_stages/file_loader.rb +3 -6
  23. data/lib/praxis/bootloader_stages/plugin_config_load.rb +4 -6
  24. data/lib/praxis/bootloader_stages/plugin_config_prepare.rb +2 -2
  25. data/lib/praxis/bootloader_stages/plugin_loader.rb +3 -7
  26. data/lib/praxis/bootloader_stages/plugin_setup.rb +3 -3
  27. data/lib/praxis/bootloader_stages/routing.rb +5 -8
  28. data/lib/praxis/bootloader_stages/subgroup_loader.rb +2 -10
  29. data/lib/praxis/bootloader_stages/warn_unloaded_files.rb +15 -19
  30. data/lib/praxis/callbacks.rb +12 -11
  31. data/lib/praxis/collection.rb +11 -14
  32. data/lib/praxis/config.rb +17 -28
  33. data/lib/praxis/config_hash.rb +2 -1
  34. data/lib/praxis/controller.rb +7 -6
  35. data/lib/praxis/dispatcher.rb +34 -42
  36. data/lib/praxis/docs/open_api/info_object.rb +11 -8
  37. data/lib/praxis/docs/open_api/media_type_object.rb +18 -17
  38. data/lib/praxis/docs/open_api/operation_object.rb +7 -4
  39. data/lib/praxis/docs/open_api/parameter_object.rb +17 -14
  40. data/lib/praxis/docs/open_api/paths_object.rb +11 -9
  41. data/lib/praxis/docs/open_api/request_body_object.rb +14 -13
  42. data/lib/praxis/docs/open_api/response_object.rb +24 -18
  43. data/lib/praxis/docs/open_api/responses_object.rb +3 -1
  44. data/lib/praxis/docs/open_api/schema_object.rb +61 -29
  45. data/lib/praxis/docs/open_api/server_object.rb +5 -2
  46. data/lib/praxis/docs/open_api/tag_object.rb +9 -6
  47. data/lib/praxis/docs/open_api_generator.rb +114 -150
  48. data/lib/praxis/endpoint_definition.rb +60 -77
  49. data/lib/praxis/error_handler.rb +2 -2
  50. data/lib/praxis/exception.rb +2 -0
  51. data/lib/praxis/exceptions/config.rb +3 -1
  52. data/lib/praxis/exceptions/config_load.rb +2 -0
  53. data/lib/praxis/exceptions/config_validation.rb +3 -1
  54. data/lib/praxis/exceptions/invalid_configuration.rb +3 -1
  55. data/lib/praxis/exceptions/invalid_response.rb +3 -1
  56. data/lib/praxis/exceptions/invalid_trait.rb +3 -1
  57. data/lib/praxis/exceptions/stage_not_found.rb +3 -1
  58. data/lib/praxis/exceptions/validation.rb +4 -3
  59. data/lib/praxis/extensions/attribute_filtering/active_record_filter_query_builder.rb +187 -131
  60. data/lib/praxis/extensions/attribute_filtering/active_record_patches/5x.rb +18 -13
  61. data/lib/praxis/extensions/attribute_filtering/active_record_patches/6_0.rb +13 -9
  62. data/lib/praxis/extensions/attribute_filtering/active_record_patches/6_1_plus.rb +14 -11
  63. data/lib/praxis/extensions/attribute_filtering/active_record_patches.rb +12 -9
  64. data/lib/praxis/extensions/attribute_filtering/filter_tree_node.rb +8 -5
  65. data/lib/praxis/extensions/attribute_filtering/filtering_params.rb +89 -65
  66. data/lib/praxis/extensions/attribute_filtering/filters_parser.rb +68 -62
  67. data/lib/praxis/extensions/attribute_filtering.rb +3 -1
  68. data/lib/praxis/extensions/field_expansion.rb +6 -4
  69. data/lib/praxis/extensions/field_selection/active_record_query_selector.rb +10 -8
  70. data/lib/praxis/extensions/field_selection/field_selector.rb +91 -92
  71. data/lib/praxis/extensions/field_selection/sequel_query_selector.rb +12 -12
  72. data/lib/praxis/extensions/field_selection.rb +3 -1
  73. data/lib/praxis/extensions/pagination/active_record_pagination_handler.rb +6 -4
  74. data/lib/praxis/extensions/pagination/header_generator.rb +16 -11
  75. data/lib/praxis/extensions/pagination/ordering_params.rb +29 -28
  76. data/lib/praxis/extensions/pagination/pagination_handler.rb +44 -42
  77. data/lib/praxis/extensions/pagination/pagination_params.rb +29 -48
  78. data/lib/praxis/extensions/pagination/sequel_pagination_handler.rb +8 -7
  79. data/lib/praxis/extensions/pagination.rb +10 -15
  80. data/lib/praxis/extensions/rails_compat/request_methods.rb +3 -4
  81. data/lib/praxis/extensions/rails_compat.rb +2 -0
  82. data/lib/praxis/extensions/rendering.rb +12 -12
  83. data/lib/praxis/field_expander.rb +8 -9
  84. data/lib/praxis/file_group.rb +8 -12
  85. data/lib/praxis/finalizable.rb +1 -0
  86. data/lib/praxis/handlers/json.rb +5 -2
  87. data/lib/praxis/handlers/plain.rb +2 -1
  88. data/lib/praxis/handlers/www_form.rb +6 -3
  89. data/lib/praxis/handlers/{xml-sample.rb → xml_sample.rb} +26 -22
  90. data/lib/praxis/mapper/active_model_compat.rb +13 -10
  91. data/lib/praxis/mapper/resource.rb +196 -181
  92. data/lib/praxis/mapper/selector_generator.rb +106 -112
  93. data/lib/praxis/mapper/sequel_compat.rb +70 -67
  94. data/lib/praxis/media_type.rb +2 -2
  95. data/lib/praxis/media_type_identifier.rb +26 -22
  96. data/lib/praxis/middleware_app.rb +18 -15
  97. data/lib/praxis/multipart/parser.rb +46 -51
  98. data/lib/praxis/multipart/part.rb +78 -110
  99. data/lib/praxis/notifications.rb +2 -4
  100. data/lib/praxis/plugin.rb +11 -18
  101. data/lib/praxis/plugin_concern.rb +12 -15
  102. data/lib/praxis/plugins/mapper_plugin.rb +15 -13
  103. data/lib/praxis/plugins/pagination_plugin.rb +8 -6
  104. data/lib/praxis/plugins/rails_plugin.rb +33 -28
  105. data/lib/praxis/renderer.rb +11 -15
  106. data/lib/praxis/request.rb +48 -44
  107. data/lib/praxis/request_stages/action.rb +4 -6
  108. data/lib/praxis/request_stages/load_request.rb +2 -4
  109. data/lib/praxis/request_stages/request_stage.rb +19 -23
  110. data/lib/praxis/request_stages/response.rb +4 -6
  111. data/lib/praxis/request_stages/validate.rb +3 -5
  112. data/lib/praxis/request_stages/validate_params_and_headers.rb +15 -22
  113. data/lib/praxis/request_stages/validate_payload.rb +25 -28
  114. data/lib/praxis/request_superclassing.rb +3 -3
  115. data/lib/praxis/resource_definition.rb +1 -0
  116. data/lib/praxis/response.rb +24 -26
  117. data/lib/praxis/response_definition.rb +77 -122
  118. data/lib/praxis/response_template.rb +11 -15
  119. data/lib/praxis/responses/http.rb +23 -44
  120. data/lib/praxis/responses/internal_server_error.rb +18 -21
  121. data/lib/praxis/responses/multipart_ok.rb +4 -9
  122. data/lib/praxis/responses/validation_error.rb +8 -15
  123. data/lib/praxis/route.rb +8 -10
  124. data/lib/praxis/router/rack.rb +13 -7
  125. data/lib/praxis/router/simple.rb +10 -5
  126. data/lib/praxis/router.rb +27 -34
  127. data/lib/praxis/routing_config.rb +52 -29
  128. data/lib/praxis/simple_media_type.rb +5 -8
  129. data/lib/praxis/stage.rb +17 -25
  130. data/lib/praxis/tasks/api_docs.rb +17 -16
  131. data/lib/praxis/tasks/console.rb +3 -1
  132. data/lib/praxis/tasks/environment.rb +2 -0
  133. data/lib/praxis/tasks/routes.rb +26 -24
  134. data/lib/praxis/tasks.rb +3 -1
  135. data/lib/praxis/trait.rb +37 -46
  136. data/lib/praxis/types/fuzzy_hash.rb +13 -14
  137. data/lib/praxis/types/media_type_common.rb +11 -10
  138. data/lib/praxis/types/multipart_array/part_definition.rb +14 -17
  139. data/lib/praxis/types/multipart_array.rb +100 -115
  140. data/lib/praxis/validation_handler.rb +5 -3
  141. data/lib/praxis/version.rb +3 -1
  142. data/lib/praxis.rb +4 -5
  143. data/praxis.gemspec +22 -21
  144. data/spec/functional_spec.rb +44 -56
  145. data/spec/praxis/action_definition_spec.rb +39 -48
  146. data/spec/praxis/api_definition_spec.rb +45 -47
  147. data/spec/praxis/api_general_info_spec.rb +28 -29
  148. data/spec/praxis/application_spec.rb +18 -14
  149. data/spec/praxis/blueprint_spec.rb +33 -34
  150. data/spec/praxis/bootloader_spec.rb +32 -30
  151. data/spec/praxis/callbacks_spec.rb +37 -37
  152. data/spec/praxis/collection_spec.rb +18 -25
  153. data/spec/praxis/config_hash_spec.rb +5 -4
  154. data/spec/praxis/config_spec.rb +27 -26
  155. data/spec/praxis/controller_spec.rb +8 -9
  156. data/spec/praxis/endpoint_definition_spec.rb +25 -32
  157. data/spec/praxis/extensions/attribute_filtering/active_record_filter_query_builder_spec.rb +221 -106
  158. data/spec/praxis/extensions/attribute_filtering/filter_tree_node_spec.rb +22 -21
  159. data/spec/praxis/extensions/attribute_filtering/filtering_params_spec.rb +112 -60
  160. data/spec/praxis/extensions/attribute_filtering/filters_parser_spec.rb +37 -38
  161. data/spec/praxis/extensions/field_expansion_spec.rb +8 -10
  162. data/spec/praxis/extensions/field_selection/active_record_query_selector_spec.rb +14 -13
  163. data/spec/praxis/extensions/field_selection/field_selector_spec.rb +9 -16
  164. data/spec/praxis/extensions/field_selection/sequel_query_selector_spec.rb +50 -49
  165. data/spec/praxis/extensions/pagination/active_record_pagination_handler_spec.rb +32 -31
  166. data/spec/praxis/extensions/rendering_spec.rb +9 -9
  167. data/spec/praxis/extensions/support/spec_resources_active_model.rb +32 -47
  168. data/spec/praxis/extensions/support/spec_resources_sequel.rb +48 -48
  169. data/spec/praxis/field_expander_spec.rb +6 -5
  170. data/spec/praxis/file_group_spec.rb +3 -1
  171. data/spec/praxis/handlers/json_spec.rb +6 -5
  172. data/spec/praxis/mapper/resource_spec.rb +39 -29
  173. data/spec/praxis/mapper/selector_generator_spec.rb +80 -46
  174. data/spec/praxis/media_type_identifier_spec.rb +13 -10
  175. data/spec/praxis/media_type_spec.rb +12 -12
  176. data/spec/praxis/middleware_app_spec.rb +23 -22
  177. data/spec/praxis/multipart/parser_spec.rb +7 -9
  178. data/spec/praxis/notifications_spec.rb +4 -4
  179. data/spec/praxis/plugin_concern_spec.rb +5 -6
  180. data/spec/praxis/renderer_spec.rb +10 -9
  181. data/spec/praxis/request_spec.rb +38 -41
  182. data/spec/praxis/request_stages/action_spec.rb +14 -15
  183. data/spec/praxis/request_stages/request_stage_spec.rb +30 -41
  184. data/spec/praxis/request_stages/validate_spec.rb +3 -1
  185. data/spec/praxis/response_definition_spec.rb +79 -92
  186. data/spec/praxis/response_spec.rb +35 -40
  187. data/spec/praxis/responses/internal_server_error_spec.rb +6 -9
  188. data/spec/praxis/responses/validation_error_spec.rb +17 -18
  189. data/spec/praxis/route_spec.rb +4 -7
  190. data/spec/praxis/router_spec.rb +69 -79
  191. data/spec/praxis/routing_config_spec.rb +15 -14
  192. data/spec/praxis/stage_spec.rb +56 -53
  193. data/spec/praxis/trait_spec.rb +17 -17
  194. data/spec/praxis/types/fuzzy_hash_spec.rb +11 -9
  195. data/spec/praxis/types/multipart_array/part_definition_spec.rb +3 -2
  196. data/spec/praxis/types/multipart_array_spec.rb +33 -48
  197. data/spec/spec_app/app/concerns/authenticated.rb +5 -5
  198. data/spec/spec_app/app/concerns/basic_api.rb +3 -1
  199. data/spec/spec_app/app/concerns/log_wrapper.rb +5 -3
  200. data/spec/spec_app/app/controllers/base_class.rb +6 -5
  201. data/spec/spec_app/app/controllers/instances.rb +31 -34
  202. data/spec/spec_app/app/controllers/volumes.rb +6 -6
  203. data/spec/spec_app/app/responses/multipart.rb +1 -2
  204. data/spec/spec_app/app/responses/other_response.rb +2 -2
  205. data/spec/spec_app/config/environment.rb +19 -6
  206. data/spec/spec_app/config.ru +4 -3
  207. data/spec/spec_app/design/api.rb +13 -15
  208. data/spec/spec_app/design/media_types/instance.rb +6 -6
  209. data/spec/spec_app/design/media_types/volume.rb +2 -1
  210. data/spec/spec_app/design/media_types/volume_snapshot.rb +2 -1
  211. data/spec/spec_app/design/resources/instances.rb +11 -17
  212. data/spec/spec_app/design/resources/volume_snapshots.rb +4 -5
  213. data/spec/spec_app/design/resources/volumes.rb +4 -5
  214. data/spec/spec_helper.rb +12 -13
  215. data/spec/support/be_deep_equal_matcher.rb +5 -0
  216. data/spec/support/spec_authorization_plugin.rb +7 -12
  217. data/spec/support/spec_blueprints.rb +5 -4
  218. data/spec/support/spec_complex_authentication_plugin.rb +17 -34
  219. data/spec/support/spec_endpoint_definitions.rb +2 -3
  220. data/spec/support/spec_media_types.rb +28 -35
  221. data/spec/support/spec_resources.rb +22 -16
  222. data/spec/support/spec_simple_authentication_plugin.rb +5 -9
  223. data/tasks/loader.thor +4 -2
  224. data/tasks/thor/app.rb +7 -5
  225. data/tasks/thor/example.rb +23 -22
  226. data/tasks/thor/model.rb +7 -7
  227. data/tasks/thor/scaffold.rb +23 -23
  228. data/tasks/thor/templates/generator/example_app/app/v1/resources/user.rb +0 -8
  229. data/tasks/thor/templates/generator/scaffold/implementation/resources/item.rb +1 -2
  230. metadata +72 -84
  231. data/MAINTAINERS +0 -2
  232. data/TODO.md +0 -25
  233. data/spec/praxis/api_resource_spec.rb +0 -0
  234. data/spec/praxis/dispatcher_spec.rb +0 -0
  235. data/spec/spec_app/app/responses/bulk_response.rb +0 -0
@@ -1,4 +1,5 @@
1
1
  # frozen_string_literal: true
2
+
2
3
  require_relative '../spec_helper'
3
4
 
4
5
  describe Praxis::Renderer do
@@ -30,7 +31,7 @@ describe Praxis::Renderer do
30
31
  address: {
31
32
  state: true,
32
33
  street: true,
33
- resident: { name: true }
34
+ resident: { name: true }
34
35
  },
35
36
  prior_addresses: { name: true },
36
37
  work_address: true,
@@ -45,15 +46,15 @@ describe Praxis::Renderer do
45
46
  subject(:output) { renderer.render(person, fields) }
46
47
 
47
48
  it 'renders existing attributes' do
48
- expect(output.keys).to match_array([:name, :full_name, :alive, :address, :prior_addresses, :metadata, :aliases])
49
+ expect(output.keys).to match_array(%i[name full_name alive address prior_addresses metadata aliases])
49
50
 
50
51
  expect(output[:name]).to eq person.name
51
52
  expect(output[:full_name]).to eq(first: person.full_name.first, last: person.full_name.last)
52
53
  expect(output[:alive]).to be false
53
54
 
54
55
  expect(output[:address]).to eq(state: person.address.state,
55
- street: person.address.street,
56
- resident: { name: person.address.resident.name })
56
+ street: person.address.street,
57
+ resident: { name: person.address.resident.name })
57
58
 
58
59
  expected_prior_addresses = prior_addresses.collect { |addr| { name: addr.name } }
59
60
  expect(output[:prior_addresses]).to match_array(expected_prior_addresses)
@@ -118,18 +119,18 @@ describe Praxis::Renderer do
118
119
 
119
120
  context 'rendering stuff that breaks badly' do
120
121
  it 'does not break badly' do
121
- expect{renderer.render(person, {tags: true})}.to_not raise_error
122
+ expect { renderer.render(person, { tags: true }) }.to_not raise_error
122
123
  end
123
124
  end
124
125
 
125
126
  context 'caching rendered objects' do
126
- let(:fields) { {full_name: true} }
127
+ let(:fields) { { full_name: true } }
127
128
  it 'caches and returns identical results for the same field objects' do
128
129
  expect(person).to receive(:full_name).once.and_call_original
129
130
 
130
- render_1 = renderer.render(person, fields)
131
- render_2 = renderer.render(person, fields)
132
- expect(render_1).to be(render_2)
131
+ render1 = renderer.render(person, fields)
132
+ render2 = renderer.render(person, fields)
133
+ expect(render1).to be(render2)
133
134
  end
134
135
  end
135
136
 
@@ -1,9 +1,11 @@
1
+ # frozen_string_literal: true
2
+
1
3
  require 'spec_helper'
2
4
 
3
5
  describe Praxis::Request do
4
- let(:path) { '/instances/1?junk=foo&api_version=1.0' }
6
+ let(:path) { '/instances/1?junk=foo&api_version=1.0' }
5
7
  let(:rack_input) { StringIO.new('{"something": "given"}') }
6
- let(:env_content_type){ 'application/json' }
8
+ let(:env_content_type) { 'application/json' }
7
9
  let(:env) do
8
10
  env = Rack::MockRequest.env_for(path)
9
11
  env['rack.input'] = rack_input
@@ -17,9 +19,9 @@ describe Praxis::Request do
17
19
 
18
20
  let(:context) do
19
21
  {
20
- params: [Attributor::AttributeResolver::ROOT_PREFIX, "params".freeze],
21
- headers: [Attributor::AttributeResolver::ROOT_PREFIX, "headers".freeze],
22
- payload: [Attributor::AttributeResolver::ROOT_PREFIX, "payload".freeze]
22
+ params: [Attributor::ROOT_PREFIX, 'params'],
23
+ headers: [Attributor::ROOT_PREFIX, 'headers'],
24
+ payload: [Attributor::ROOT_PREFIX, 'payload']
23
25
  }.freeze
24
26
  end
25
27
 
@@ -29,9 +31,9 @@ describe Praxis::Request do
29
31
  request
30
32
  end
31
33
 
32
- its(:verb) { should eq("GET") }
34
+ its(:verb) { should eq('GET') }
33
35
  its(:path) { should eq('/instances/1') }
34
- its(:raw_params) { should eq({'junk' => 'foo'}) }
36
+ its(:raw_params) { should eq({ 'junk' => 'foo' }) }
35
37
  its(:version) { should eq('1.0') }
36
38
 
37
39
  context 'path versioning' do
@@ -42,26 +44,24 @@ describe Praxis::Request do
42
44
  allow(
43
45
  Praxis::ApiDefinition.instance.info
44
46
  ).to receive(:base_path).and_return('/api/v:api_version')
45
-
46
47
  end
47
48
 
48
49
  let(:path) { '/api/v2.0/instances/1' }
49
50
 
50
-
51
51
  its(:version) { should eq '2.0' }
52
- #its('class.path_version_prefix'){ should eq("/v") }
53
- #its(:path_version_matcher){ should be_kind_of(Regexp) }
54
- #it 'uses a "/v*" default matcher with a "version" named capture' do
52
+ # its('class.path_version_prefix'){ should eq("/v") }
53
+ # its(:path_version_matcher){ should be_kind_of(Regexp) }
54
+ # it 'uses a "/v*" default matcher with a "version" named capture' do
55
55
  # match = subject.path_version_matcher.match("/v5.5/something")
56
56
  # expect(match).to_not be(nil)
57
57
  # expect(match['version']).to eq("5.5")
58
- #end
58
+ # end
59
59
  end
60
60
 
61
61
  context 'loading api version' do
62
- #let(:request) { Praxis::Request.new(env) }
63
- subject(:version){ request.version }
64
- let(:versioning_scheme){ Praxis::Request::VERSION_USING_DEFAULTS }
62
+ # let(:request) { Praxis::Request.new(env) }
63
+ subject(:version) { request.version }
64
+ let(:versioning_scheme) { Praxis::Request::VERSION_USING_DEFAULTS }
65
65
  before do
66
66
  allow(
67
67
  Praxis::Application.instance
@@ -69,18 +69,18 @@ describe Praxis::Request do
69
69
  end
70
70
 
71
71
  context 'using X-Api-Header' do
72
- let(:env){ {'HTTP_X_API_VERSION' => "5.0", "PATH_INFO" => "/something"} }
73
- let(:versioning_scheme){ :header }
72
+ let(:env) { { 'HTTP_X_API_VERSION' => '5.0', 'PATH_INFO' => '/something' } }
73
+ let(:versioning_scheme) { :header }
74
74
  it { should eq('5.0') }
75
75
  end
76
76
  context 'using query param' do
77
77
  let(:env) { Rack::MockRequest.env_for('/instances/1?junk=foo&api_version=5.0') }
78
- let(:versioning_scheme){ :params }
78
+ let(:versioning_scheme) { :params }
79
79
  it { should eq('5.0') }
80
80
  end
81
81
  context 'using path (with the default pattern matcher)' do
82
- let(:env){ Rack::MockRequest.env_for('/v5.0/instances/1?junk=foo') }
83
- let(:versioning_scheme){ :path }
82
+ let(:env) { Rack::MockRequest.env_for('/v5.0/instances/1?junk=foo') }
83
+ let(:versioning_scheme) { :path }
84
84
 
85
85
  before do
86
86
  allow(
@@ -92,21 +92,21 @@ describe Praxis::Request do
92
92
 
93
93
  context 'using a method that it is not allowed in the definition' do
94
94
  context 'allowing query param but passing it through a header' do
95
- let(:env){ {'HTTP_X_API_VERSION' => "5.0", "PATH_INFO" => "/something"} }
96
- let(:versioning_scheme){ :params }
95
+ let(:env) { { 'HTTP_X_API_VERSION' => '5.0', 'PATH_INFO' => '/something' } }
96
+ let(:versioning_scheme) { :params }
97
97
  it { should eq('n/a') }
98
98
  end
99
99
  context 'allowing header but passing it through param' do
100
100
  let(:env) { Rack::MockRequest.env_for('/instances/1?junk=foo&api_version=5.0') }
101
- let(:versioning_scheme){ :header }
101
+ let(:versioning_scheme) { :header }
102
102
  it { should eq('n/a') }
103
103
  end
104
104
  end
105
105
 
106
106
  context 'using defaults' do
107
- subject(:version){ request.version }
107
+ subject(:version) { request.version }
108
108
  context 'would succeed if passed through the header' do
109
- let(:env){ {'HTTP_X_API_VERSION' => "5.0", "PATH_INFO" => "/something"} }
109
+ let(:env) { { 'HTTP_X_API_VERSION' => '5.0', 'PATH_INFO' => '/something' } }
110
110
  it { should eq('5.0') }
111
111
  end
112
112
  context 'would succeed if passed through params' do
@@ -126,9 +126,9 @@ describe Praxis::Request do
126
126
  form_data
127
127
  end
128
128
 
129
- let(:rack_input) {
129
+ let(:rack_input) do
130
130
  StringIO.new(form.body.to_s)
131
- }
131
+ end
132
132
 
133
133
  let(:env) do
134
134
  env = Rack::MockRequest.env_for('/instances/1?junk=foo&api_version=1.0')
@@ -138,21 +138,18 @@ describe Praxis::Request do
138
138
  env
139
139
  end
140
140
 
141
- #its(:multipart?) { should be(true) }
141
+ # its(:multipart?) { should be(true) }
142
142
 
143
143
  it 'works' do
144
- #p request #.body
145
- #p request #.parts
144
+ # p request #.body
145
+ # p request #.parts
146
146
  end
147
-
148
-
149
147
  end
150
148
 
151
149
  context '#load_headers' do
152
-
153
150
  it 'is done preserving the original case' do
154
151
  request.load_headers(context[:headers])
155
- expect(request.headers).to eq({"Authorization" => "Secret"})
152
+ expect(request.headers).to eq({ 'Authorization' => 'Secret' })
156
153
  end
157
154
 
158
155
  it 'performs it using the memoized rack keys from the action (Hacky but...performance is important)' do
@@ -161,7 +158,7 @@ describe Praxis::Request do
161
158
  end
162
159
  end
163
160
 
164
- context "performs request validation" do
161
+ context 'performs request validation' do
165
162
  before(:each) do
166
163
  request.load_headers(context[:headers])
167
164
  request.load_params(context[:params])
@@ -205,17 +202,17 @@ describe Praxis::Request do
205
202
  end
206
203
 
207
204
  context '#load_payload' do
208
- let(:load_context){ context[:payload] }
209
- let(:parsed_result){ double("parsed") }
205
+ let(:load_context) { context[:payload] }
206
+ let(:parsed_result) { double('parsed') }
210
207
 
211
208
  after do
212
- expect(request.action.payload).to receive(:load).with(parsed_result, load_context, content_type: request.content_type.to_s )
213
- request.load_payload( load_context )
209
+ expect(request.action.payload).to receive(:load).with(parsed_result, load_context, content_type: request.content_type.to_s)
210
+ request.load_payload(load_context)
214
211
  end
215
212
 
216
213
  context 'that is json encoded' do
217
214
  let(:rack_input) { StringIO.new('{"one":1,"sub_hash":{"first":"hello"}}') }
218
- let(:env_content_type){ 'application/json' }
215
+ let(:env_content_type) { 'application/json' }
219
216
  let(:parsed_result) { Praxis::Handlers::JSON.new.parse(request.raw_payload) }
220
217
 
221
218
  it 'decodes using the JSON handler' do end
@@ -1,17 +1,18 @@
1
+ # frozen_string_literal: true
2
+
1
3
  require 'spec_helper'
2
4
 
3
5
  describe Praxis::RequestStages::Action do
4
-
5
6
  let(:controller) do
6
7
  Class.new do
7
8
  include Praxis::Controller
8
9
  end.new(request)
9
10
  end
10
11
 
11
- let(:action) { double("action", name: "foo") }
12
- let(:response){ Praxis::Responses::Ok.new }
13
- let(:app){ double("App", controller: controller, action:action, request: request)}
14
- let(:action_stage){ Praxis::RequestStages::Action.new(action.name,app) }
12
+ let(:action) { double('action', name: 'foo') }
13
+ let(:response) { Praxis::Responses::Ok.new }
14
+ let(:app) { double('App', controller: controller, action: action, request: request) }
15
+ let(:action_stage) { Praxis::RequestStages::Action.new(action.name, app) }
15
16
 
16
17
  let(:request) do
17
18
  env = Rack::MockRequest.env_for('/instances/1?cloud_id=1&api_version=1.0')
@@ -23,19 +24,17 @@ describe Praxis::RequestStages::Action do
23
24
  request
24
25
  end
25
26
 
26
-
27
27
  context '.execute' do
28
28
  before do
29
29
  expect(controller).to receive(action_stage.name) do |args|
30
30
  if args
31
31
  expect(args).to eq({})
32
32
  else
33
- expect(args).to eq(nil)
33
+ expect(args).to eq(nil)
34
34
  end
35
35
  end.and_return(controller_response)
36
-
37
36
  end
38
- let(:controller_response){ controller.response }
37
+ let(:controller_response) { controller.response }
39
38
 
40
39
  it 'always call the right controller method' do
41
40
  action_stage.execute
@@ -47,28 +46,28 @@ describe Praxis::RequestStages::Action do
47
46
  end
48
47
 
49
48
  it 'sends the right ActiveSupport::Notification' do
50
- expect(ActiveSupport::Notifications).to receive(:instrument).with('praxis.request_stage.execute', {controller: an_instance_of(controller.class)}).and_call_original
49
+ expect(ActiveSupport::Notifications).to receive(:instrument).with('praxis.request_stage.execute', { controller: an_instance_of(controller.class) }).and_call_original
51
50
  action_stage.execute
52
51
  end
53
52
 
54
53
  context 'if the controller method returns a string' do
55
- let(:controller_response){ "this is the body"}
54
+ let(:controller_response) { 'this is the body' }
56
55
  it 'sets the response body with it (and save the request too)' do
57
56
  action_stage.execute
58
- expect(controller.response.body).to eq("this is the body")
57
+ expect(controller.response.body).to eq('this is the body')
59
58
  end
60
59
  end
61
60
  context 'if the controller method returns a response object' do
62
- let(:controller_response){ Praxis::Responses::Created.new }
61
+ let(:controller_response) { Praxis::Responses::Created.new }
63
62
  it 'set that response in the controller' do
64
63
  action_stage.execute
65
64
  expect(controller.response).to eq(controller_response)
66
65
  end
67
66
  end
68
67
  context 'if the controller method returns neither a string or a response' do
69
- let(:controller_response){ nil }
68
+ let(:controller_response) { nil }
70
69
  it 'an error is raised ' do
71
- expect{ action_stage.execute }.to raise_error(/Only Response objects or Strings allowed/)
70
+ expect { action_stage.execute }.to raise_error(/Only Response objects or Strings allowed/)
72
71
  end
73
72
  end
74
73
  end
@@ -1,7 +1,8 @@
1
+ # frozen_string_literal: true
2
+
1
3
  require 'spec_helper'
2
4
 
3
5
  describe Praxis::RequestStages::RequestStage do
4
-
5
6
  let(:controller_class) do
6
7
  Class.new do
7
8
  include Praxis::Controller
@@ -11,15 +12,14 @@ describe Praxis::RequestStages::RequestStage do
11
12
  let(:stage_class) { Class.new(Praxis::RequestStages::RequestStage) }
12
13
 
13
14
  let(:request) { Praxis::Request.new({}) }
14
- let(:controller){ controller_class.new(request) }
15
+ let(:controller) { controller_class.new(request) }
15
16
 
17
+ let(:action) { instance_double('Praxis::ActionDefinition') }
18
+ let(:context) { double('context', controller: controller, action: action) }
16
19
 
17
- let(:action){ instance_double("Praxis::ActionDefinition") }
18
- let(:context){ double("context", controller: controller , action: action) }
19
-
20
- let(:substage_1) { instance_double('Praxis::RequestStage') }
21
- let(:substage_2) { instance_double('Praxis::RequestStage') }
22
- let(:substage_3) { instance_double('Praxis::RequestStage') }
20
+ let(:substage1) { instance_double('Praxis::RequestStage') }
21
+ let(:substage2) { instance_double('Praxis::RequestStage') }
22
+ let(:substage3) { instance_double('Praxis::RequestStage') }
23
23
 
24
24
  let(:before_callbacks) { double('before_callbacks') }
25
25
  let(:after_callbacks) { double('after_callbacks') }
@@ -28,19 +28,18 @@ describe Praxis::RequestStages::RequestStage do
28
28
 
29
29
  subject(:stage) { stage_class.new(:action, context) }
30
30
 
31
-
32
31
  before do
33
32
  # clear any pre-existing callbacks that may have been added by plugins
34
- controller_class.before_callbacks = Hash.new
35
- controller_class.after_callbacks = Hash.new
33
+ controller_class.before_callbacks = ({})
34
+ controller_class.after_callbacks = ({})
36
35
  end
37
36
 
38
37
  context 'for an abstract stage' do
39
38
  subject(:stage) { Praxis::RequestStages::RequestStage.new(:action, context) }
40
39
  it 'raises NotImplementedError for undefined #execute' do
41
- expect {
40
+ expect do
42
41
  stage.execute
43
- }.to raise_error(NotImplementedError,/Subclass must implement Stage#execute/)
42
+ end.to raise_error(NotImplementedError, /Subclass must implement Stage#execute/)
44
43
  end
45
44
  end
46
45
 
@@ -57,16 +56,16 @@ describe Praxis::RequestStages::RequestStage do
57
56
  end
58
57
  end
59
58
 
60
- context "#execute" do
59
+ context '#execute' do
61
60
  before do
62
- stage.stages.push(substage_1, substage_2, substage_3)
61
+ stage.stages.push(substage1, substage2, substage3)
63
62
  end
64
63
 
65
64
  context 'when all stages succeed' do
66
- it "runs them all and returns nil" do
67
- expect(substage_1).to receive(:run).once
68
- expect(substage_2).to receive(:run).once
69
- expect(substage_3).to receive(:run).once
65
+ it 'runs them all and returns nil' do
66
+ expect(substage1).to receive(:run).once
67
+ expect(substage2).to receive(:run).once
68
+ expect(substage3).to receive(:run).once
70
69
  expect(stage.execute).to be(nil)
71
70
  end
72
71
  end
@@ -74,12 +73,12 @@ describe Praxis::RequestStages::RequestStage do
74
73
  context 'when one stage returns a Response' do
75
74
  let(:response) { Praxis::Responses::Ok.new }
76
75
  before do
77
- expect(substage_1).to receive(:run).once
78
- expect(substage_2).to receive(:run).once.and_return(response)
76
+ expect(substage1).to receive(:run).once
77
+ expect(substage2).to receive(:run).once.and_return(response)
79
78
  end
80
79
 
81
- it "runs no further stages after that" do
82
- expect(substage_3).to_not receive(:run)
80
+ it 'runs no further stages after that' do
81
+ expect(substage3).to_not receive(:run)
83
82
  stage.execute
84
83
  end
85
84
 
@@ -88,12 +87,10 @@ describe Praxis::RequestStages::RequestStage do
88
87
 
89
88
  expect(controller.response).to be(response)
90
89
  end
91
-
92
90
  end
93
91
  end
94
92
 
95
- context "#run" do
96
-
93
+ context '#run' do
97
94
  let(:before_callbacks) { double('before_callbacks') }
98
95
  let(:after_callbacks) { double('after_callbacks') }
99
96
 
@@ -113,14 +110,13 @@ describe Praxis::RequestStages::RequestStage do
113
110
  allow(controller_class).to receive(:after_callbacks).once.and_return(controller_after_callbacks)
114
111
  end
115
112
 
116
- it "sets up and executes callbacks" do
113
+ it 'sets up and executes callbacks' do
117
114
  expect(stage).to receive(:execute)
118
115
  expect(stage).to receive(:execute_callbacks).once.with(before_callbacks)
119
116
  expect(stage).to receive(:execute_callbacks).once.with(after_callbacks)
120
117
  expect(stage).to receive(:execute_controller_callbacks).once.with(controller_before_callbacks)
121
118
  expect(stage).to receive(:execute_controller_callbacks).once.with(controller_after_callbacks)
122
119
  end
123
-
124
120
  end
125
121
 
126
122
  context 'when the before execute_controller_callbacks return a Response' do
@@ -131,7 +127,6 @@ describe Praxis::RequestStages::RequestStage do
131
127
  expect(stage).to receive(:execute_controller_callbacks).once.and_return(response)
132
128
  end
133
129
 
134
-
135
130
  it 'does not call "execute"' do
136
131
  expect(stage).to_not receive(:execute)
137
132
  end
@@ -140,22 +135,20 @@ describe Praxis::RequestStages::RequestStage do
140
135
  expect(stage).to_not receive(:after_callbacks)
141
136
  expect(controller_class).to_not receive(:after_callbacks)
142
137
  end
143
-
144
138
  end
145
139
 
146
140
  context 'with substages' do
147
141
  before do
148
- stage.stages.push(substage_1, substage_2, substage_3)
142
+ stage.stages.push(substage1, substage2, substage3)
149
143
  end
150
144
 
151
-
152
145
  context 'when one returns a Response' do
153
146
  let(:response) { Praxis::Responses::Unauthorized.new }
154
147
 
155
148
  before do
156
- expect(substage_1).to receive(:run).once
157
- expect(substage_2).to receive(:run).once.and_return(response)
158
- expect(substage_3).to_not receive(:run)
149
+ expect(substage1).to receive(:run).once
150
+ expect(substage2).to receive(:run).once.and_return(response)
151
+ expect(substage3).to_not receive(:run)
159
152
  end
160
153
 
161
154
  it 'runs no after callbacks (including from the controller) ' do
@@ -165,14 +158,10 @@ describe Praxis::RequestStages::RequestStage do
165
158
 
166
159
  it 'assigns controller.response' do
167
160
  # twice, because we do it once in #execute, and again in #run...
168
- expect(controller).to receive(:response=).
169
- with(response).twice.and_call_original
161
+ expect(controller).to receive(:response=)
162
+ .with(response).twice.and_call_original
170
163
  end
171
-
172
164
  end
173
165
  end
174
166
  end
175
-
176
-
177
-
178
167
  end
@@ -1,3 +1,5 @@
1
+ # frozen_string_literal: true
2
+
1
3
  require 'spec_helper'
2
4
 
3
5
  describe Praxis::RequestStages::Validate do
@@ -19,7 +21,7 @@ describe Praxis::RequestStages::Validate do
19
21
 
20
22
  let(:request) do
21
23
  r = Praxis::Request.new(env)
22
- r.route_params = {id: 1}
24
+ r.route_params = { id: 1 }
23
25
  r.action = action
24
26
  r
25
27
  end