praxis 2.0.pre.18 → 2.0.pre.19
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +4 -4
- data/.rubocop.yml +54 -0
- data/.simplecov +3 -1
- data/.travis.yml +2 -1
- data/CHANGELOG.md +6 -0
- data/CONTRIBUTING.md +2 -79
- data/Gemfile +5 -1
- data/Guardfile +6 -4
- data/LICENSE +0 -2
- data/MAINTAINERS.md +1 -0
- data/README.md +15 -22
- data/Rakefile +4 -2
- data/bin/praxis +55 -58
- data/lib/praxis/action_definition/headers_dsl_compiler.rb +5 -6
- data/lib/praxis/action_definition.rb +64 -94
- data/lib/praxis/api_definition.rb +21 -29
- data/lib/praxis/api_general_info.rb +55 -66
- data/lib/praxis/application.rb +15 -32
- data/lib/praxis/blueprint.rb +64 -73
- data/lib/praxis/bootloader.rb +24 -33
- data/lib/praxis/bootloader_stages/environment.rb +5 -10
- data/lib/praxis/bootloader_stages/file_loader.rb +3 -6
- data/lib/praxis/bootloader_stages/plugin_config_load.rb +4 -6
- data/lib/praxis/bootloader_stages/plugin_config_prepare.rb +2 -2
- data/lib/praxis/bootloader_stages/plugin_loader.rb +3 -7
- data/lib/praxis/bootloader_stages/plugin_setup.rb +3 -3
- data/lib/praxis/bootloader_stages/routing.rb +5 -8
- data/lib/praxis/bootloader_stages/subgroup_loader.rb +2 -10
- data/lib/praxis/bootloader_stages/warn_unloaded_files.rb +15 -19
- data/lib/praxis/callbacks.rb +12 -11
- data/lib/praxis/collection.rb +10 -13
- data/lib/praxis/config.rb +17 -28
- data/lib/praxis/config_hash.rb +2 -1
- data/lib/praxis/controller.rb +7 -6
- data/lib/praxis/dispatcher.rb +34 -42
- data/lib/praxis/docs/open_api/info_object.rb +11 -8
- data/lib/praxis/docs/open_api/media_type_object.rb +18 -17
- data/lib/praxis/docs/open_api/operation_object.rb +6 -3
- data/lib/praxis/docs/open_api/parameter_object.rb +17 -14
- data/lib/praxis/docs/open_api/paths_object.rb +11 -9
- data/lib/praxis/docs/open_api/request_body_object.rb +14 -13
- data/lib/praxis/docs/open_api/response_object.rb +24 -18
- data/lib/praxis/docs/open_api/responses_object.rb +3 -1
- data/lib/praxis/docs/open_api/schema_object.rb +16 -16
- data/lib/praxis/docs/open_api/server_object.rb +5 -2
- data/lib/praxis/docs/open_api/tag_object.rb +6 -3
- data/lib/praxis/docs/open_api_generator.rb +92 -95
- data/lib/praxis/endpoint_definition.rb +60 -77
- data/lib/praxis/error_handler.rb +2 -2
- data/lib/praxis/exception.rb +2 -0
- data/lib/praxis/exceptions/config.rb +3 -1
- data/lib/praxis/exceptions/config_load.rb +2 -0
- data/lib/praxis/exceptions/config_validation.rb +3 -1
- data/lib/praxis/exceptions/invalid_configuration.rb +3 -1
- data/lib/praxis/exceptions/invalid_response.rb +3 -1
- data/lib/praxis/exceptions/invalid_trait.rb +3 -1
- data/lib/praxis/exceptions/stage_not_found.rb +3 -1
- data/lib/praxis/exceptions/validation.rb +4 -3
- data/lib/praxis/extensions/attribute_filtering/active_record_filter_query_builder.rb +163 -149
- data/lib/praxis/extensions/attribute_filtering/active_record_patches/5x.rb +18 -13
- data/lib/praxis/extensions/attribute_filtering/active_record_patches/6_0.rb +13 -9
- data/lib/praxis/extensions/attribute_filtering/active_record_patches/6_1_plus.rb +14 -11
- data/lib/praxis/extensions/attribute_filtering/active_record_patches.rb +12 -9
- data/lib/praxis/extensions/attribute_filtering/filter_tree_node.rb +8 -5
- data/lib/praxis/extensions/attribute_filtering/filtering_params.rb +89 -65
- data/lib/praxis/extensions/attribute_filtering/filters_parser.rb +68 -62
- data/lib/praxis/extensions/attribute_filtering.rb +3 -1
- data/lib/praxis/extensions/field_expansion.rb +6 -4
- data/lib/praxis/extensions/field_selection/active_record_query_selector.rb +10 -8
- data/lib/praxis/extensions/field_selection/field_selector.rb +91 -92
- data/lib/praxis/extensions/field_selection/sequel_query_selector.rb +12 -12
- data/lib/praxis/extensions/field_selection.rb +3 -1
- data/lib/praxis/extensions/pagination/active_record_pagination_handler.rb +6 -4
- data/lib/praxis/extensions/pagination/header_generator.rb +16 -11
- data/lib/praxis/extensions/pagination/ordering_params.rb +29 -28
- data/lib/praxis/extensions/pagination/pagination_handler.rb +44 -42
- data/lib/praxis/extensions/pagination/pagination_params.rb +29 -48
- data/lib/praxis/extensions/pagination/sequel_pagination_handler.rb +8 -7
- data/lib/praxis/extensions/pagination.rb +10 -15
- data/lib/praxis/extensions/rails_compat/request_methods.rb +3 -4
- data/lib/praxis/extensions/rails_compat.rb +2 -0
- data/lib/praxis/extensions/rendering.rb +12 -12
- data/lib/praxis/field_expander.rb +8 -9
- data/lib/praxis/file_group.rb +8 -12
- data/lib/praxis/finalizable.rb +1 -0
- data/lib/praxis/handlers/json.rb +5 -2
- data/lib/praxis/handlers/plain.rb +2 -1
- data/lib/praxis/handlers/www_form.rb +6 -3
- data/lib/praxis/handlers/{xml-sample.rb → xml_sample.rb} +26 -22
- data/lib/praxis/mapper/active_model_compat.rb +13 -10
- data/lib/praxis/mapper/resource.rb +171 -180
- data/lib/praxis/mapper/selector_generator.rb +106 -112
- data/lib/praxis/mapper/sequel_compat.rb +70 -67
- data/lib/praxis/media_type.rb +2 -2
- data/lib/praxis/media_type_identifier.rb +26 -22
- data/lib/praxis/middleware_app.rb +18 -15
- data/lib/praxis/multipart/parser.rb +46 -51
- data/lib/praxis/multipart/part.rb +78 -110
- data/lib/praxis/notifications.rb +2 -4
- data/lib/praxis/plugin.rb +11 -18
- data/lib/praxis/plugin_concern.rb +12 -15
- data/lib/praxis/plugins/mapper_plugin.rb +15 -13
- data/lib/praxis/plugins/pagination_plugin.rb +8 -6
- data/lib/praxis/plugins/rails_plugin.rb +33 -28
- data/lib/praxis/renderer.rb +11 -15
- data/lib/praxis/request.rb +46 -47
- data/lib/praxis/request_stages/action.rb +4 -6
- data/lib/praxis/request_stages/load_request.rb +2 -4
- data/lib/praxis/request_stages/request_stage.rb +19 -23
- data/lib/praxis/request_stages/response.rb +4 -6
- data/lib/praxis/request_stages/validate.rb +3 -5
- data/lib/praxis/request_stages/validate_params_and_headers.rb +15 -16
- data/lib/praxis/request_stages/validate_payload.rb +25 -27
- data/lib/praxis/request_superclassing.rb +3 -3
- data/lib/praxis/resource_definition.rb +1 -0
- data/lib/praxis/response.rb +13 -25
- data/lib/praxis/response_definition.rb +77 -122
- data/lib/praxis/response_template.rb +11 -15
- data/lib/praxis/responses/http.rb +23 -44
- data/lib/praxis/responses/internal_server_error.rb +18 -21
- data/lib/praxis/responses/multipart_ok.rb +4 -9
- data/lib/praxis/responses/validation_error.rb +8 -15
- data/lib/praxis/route.rb +8 -10
- data/lib/praxis/router/rack.rb +13 -7
- data/lib/praxis/router/simple.rb +10 -5
- data/lib/praxis/router.rb +27 -34
- data/lib/praxis/routing_config.rb +52 -29
- data/lib/praxis/simple_media_type.rb +5 -8
- data/lib/praxis/stage.rb +17 -25
- data/lib/praxis/tasks/api_docs.rb +15 -15
- data/lib/praxis/tasks/console.rb +3 -1
- data/lib/praxis/tasks/environment.rb +2 -0
- data/lib/praxis/tasks/routes.rb +26 -24
- data/lib/praxis/tasks.rb +3 -1
- data/lib/praxis/trait.rb +37 -46
- data/lib/praxis/types/fuzzy_hash.rb +13 -14
- data/lib/praxis/types/media_type_common.rb +11 -10
- data/lib/praxis/types/multipart_array/part_definition.rb +14 -17
- data/lib/praxis/types/multipart_array.rb +88 -112
- data/lib/praxis/validation_handler.rb +5 -3
- data/lib/praxis/version.rb +3 -1
- data/lib/praxis.rb +4 -5
- data/praxis.gemspec +22 -21
- data/spec/functional_spec.rb +40 -52
- data/spec/praxis/action_definition_spec.rb +36 -46
- data/spec/praxis/api_definition_spec.rb +45 -47
- data/spec/praxis/api_general_info_spec.rb +28 -29
- data/spec/praxis/application_spec.rb +18 -14
- data/spec/praxis/blueprint_spec.rb +33 -34
- data/spec/praxis/bootloader_spec.rb +32 -30
- data/spec/praxis/callbacks_spec.rb +37 -37
- data/spec/praxis/collection_spec.rb +18 -25
- data/spec/praxis/config_hash_spec.rb +5 -4
- data/spec/praxis/config_spec.rb +27 -26
- data/spec/praxis/controller_spec.rb +8 -9
- data/spec/praxis/endpoint_definition_spec.rb +25 -32
- data/spec/praxis/extensions/attribute_filtering/active_record_filter_query_builder_spec.rb +171 -114
- data/spec/praxis/extensions/attribute_filtering/filter_tree_node_spec.rb +22 -21
- data/spec/praxis/extensions/attribute_filtering/filtering_params_spec.rb +112 -60
- data/spec/praxis/extensions/attribute_filtering/filters_parser_spec.rb +37 -38
- data/spec/praxis/extensions/field_expansion_spec.rb +8 -10
- data/spec/praxis/extensions/field_selection/active_record_query_selector_spec.rb +14 -13
- data/spec/praxis/extensions/field_selection/field_selector_spec.rb +9 -16
- data/spec/praxis/extensions/field_selection/sequel_query_selector_spec.rb +50 -49
- data/spec/praxis/extensions/pagination/active_record_pagination_handler_spec.rb +32 -31
- data/spec/praxis/extensions/rendering_spec.rb +9 -9
- data/spec/praxis/extensions/support/spec_resources_active_model.rb +32 -49
- data/spec/praxis/extensions/support/spec_resources_sequel.rb +48 -48
- data/spec/praxis/field_expander_spec.rb +6 -5
- data/spec/praxis/file_group_spec.rb +3 -1
- data/spec/praxis/handlers/json_spec.rb +6 -5
- data/spec/praxis/mapper/resource_spec.rb +27 -30
- data/spec/praxis/mapper/selector_generator_spec.rb +50 -50
- data/spec/praxis/media_type_identifier_spec.rb +13 -10
- data/spec/praxis/media_type_spec.rb +12 -12
- data/spec/praxis/middleware_app_spec.rb +23 -22
- data/spec/praxis/multipart/parser_spec.rb +7 -9
- data/spec/praxis/notifications_spec.rb +4 -4
- data/spec/praxis/plugin_concern_spec.rb +5 -6
- data/spec/praxis/renderer_spec.rb +10 -9
- data/spec/praxis/request_spec.rb +38 -41
- data/spec/praxis/request_stages/action_spec.rb +14 -15
- data/spec/praxis/request_stages/request_stage_spec.rb +30 -41
- data/spec/praxis/request_stages/validate_spec.rb +3 -1
- data/spec/praxis/response_definition_spec.rb +79 -92
- data/spec/praxis/response_spec.rb +28 -39
- data/spec/praxis/responses/internal_server_error_spec.rb +6 -9
- data/spec/praxis/responses/validation_error_spec.rb +17 -18
- data/spec/praxis/route_spec.rb +4 -7
- data/spec/praxis/router_spec.rb +69 -79
- data/spec/praxis/routing_config_spec.rb +15 -14
- data/spec/praxis/stage_spec.rb +56 -53
- data/spec/praxis/trait_spec.rb +17 -18
- data/spec/praxis/types/fuzzy_hash_spec.rb +11 -9
- data/spec/praxis/types/multipart_array/part_definition_spec.rb +3 -2
- data/spec/praxis/types/multipart_array_spec.rb +33 -48
- data/spec/spec_app/app/concerns/authenticated.rb +5 -5
- data/spec/spec_app/app/concerns/basic_api.rb +3 -1
- data/spec/spec_app/app/concerns/log_wrapper.rb +5 -3
- data/spec/spec_app/app/controllers/base_class.rb +6 -5
- data/spec/spec_app/app/controllers/instances.rb +31 -34
- data/spec/spec_app/app/controllers/volumes.rb +6 -6
- data/spec/spec_app/app/responses/multipart.rb +1 -2
- data/spec/spec_app/app/responses/other_response.rb +2 -2
- data/spec/spec_app/config/environment.rb +19 -6
- data/spec/spec_app/config.ru +4 -3
- data/spec/spec_app/design/api.rb +13 -15
- data/spec/spec_app/design/media_types/instance.rb +5 -5
- data/spec/spec_app/design/media_types/volume.rb +2 -1
- data/spec/spec_app/design/media_types/volume_snapshot.rb +2 -1
- data/spec/spec_app/design/resources/instances.rb +9 -15
- data/spec/spec_app/design/resources/volume_snapshots.rb +4 -5
- data/spec/spec_app/design/resources/volumes.rb +4 -5
- data/spec/spec_helper.rb +11 -13
- data/spec/support/be_deep_equal_matcher.rb +5 -0
- data/spec/support/spec_authorization_plugin.rb +7 -12
- data/spec/support/spec_blueprints.rb +2 -1
- data/spec/support/spec_complex_authentication_plugin.rb +17 -34
- data/spec/support/spec_endpoint_definitions.rb +2 -3
- data/spec/support/spec_media_types.rb +28 -35
- data/spec/support/spec_resources.rb +20 -18
- data/spec/support/spec_simple_authentication_plugin.rb +5 -9
- data/tasks/loader.thor +4 -2
- data/tasks/thor/app.rb +7 -5
- data/tasks/thor/example.rb +23 -22
- data/tasks/thor/model.rb +7 -7
- data/tasks/thor/scaffold.rb +23 -23
- data/tasks/thor/templates/generator/example_app/app/v1/resources/user.rb +0 -8
- data/tasks/thor/templates/generator/scaffold/implementation/resources/item.rb +1 -2
- metadata +72 -84
- data/MAINTAINERS +0 -2
- data/TODO.md +0 -25
- data/spec/praxis/api_resource_spec.rb +0 -0
- data/spec/praxis/dispatcher_spec.rb +0 -0
- data/spec/spec_app/app/responses/bulk_response.rb +0 -0
@@ -1,11 +1,12 @@
|
|
1
|
-
|
1
|
+
# frozen_string_literal: true
|
2
2
|
|
3
|
+
require 'spec_helper'
|
3
4
|
|
4
5
|
describe Praxis::Response do
|
5
6
|
let(:spec_status) { 200 }
|
6
7
|
let(:spec_location) { /resources/ }
|
7
8
|
let(:spec_headers) { { 'X-Header' => 'Foobar' } }
|
8
|
-
let(:spec_mime_type) {
|
9
|
+
let(:spec_mime_type) { 'application/vnd.resource' }
|
9
10
|
|
10
11
|
let(:application_vnd_resource_media_type) do
|
11
12
|
Class.new(Praxis::MediaType) do
|
@@ -18,33 +19,33 @@ describe Praxis::Response do
|
|
18
19
|
let(:response_spec) do
|
19
20
|
instance_double(
|
20
21
|
Praxis::ResponseDefinition,
|
21
|
-
:
|
22
|
-
:
|
23
|
-
:
|
24
|
-
:
|
25
|
-
:
|
22
|
+
status: spec_status,
|
23
|
+
location: spec_location,
|
24
|
+
headers: spec_headers,
|
25
|
+
media_type: spec_media_type,
|
26
|
+
name: :ok
|
26
27
|
)
|
27
28
|
end
|
28
29
|
|
29
30
|
let(:action) do
|
30
31
|
instance_double(
|
31
32
|
Praxis::ActionDefinition,
|
32
|
-
:
|
33
|
+
endpoint_definition: config_class
|
33
34
|
)
|
34
35
|
end
|
35
36
|
|
36
37
|
let(:response_status) { 200 }
|
37
|
-
let(:response_headers)
|
38
|
+
let(:response_headers) do
|
38
39
|
{ 'Content-Type' => 'application/vnd.resource+json;type=collection',
|
39
|
-
'X-Header'
|
40
|
-
'Location'
|
41
|
-
|
40
|
+
'X-Header' => 'Foobar',
|
41
|
+
'Location' => '/api/resources/123' }
|
42
|
+
end
|
42
43
|
|
43
44
|
let(:config_media_type) { nil }
|
44
45
|
let(:config_class) do
|
45
46
|
instance_double(
|
46
47
|
Praxis::ResponseDefinition,
|
47
|
-
:
|
48
|
+
media_type: config_media_type
|
48
49
|
)
|
49
50
|
end
|
50
51
|
|
@@ -52,7 +53,7 @@ describe Praxis::Response do
|
|
52
53
|
|
53
54
|
describe '#validate' do
|
54
55
|
before do
|
55
|
-
allow(action).to receive(:responses).and_return({response_spec.name => response_spec })
|
56
|
+
allow(action).to receive(:responses).and_return({ response_spec.name => response_spec })
|
56
57
|
end
|
57
58
|
context 'response spec is not defined' do
|
58
59
|
before :each do
|
@@ -60,26 +61,23 @@ describe Praxis::Response do
|
|
60
61
|
end
|
61
62
|
|
62
63
|
it 'should raise an error' do
|
63
|
-
expect
|
64
|
+
expect do
|
64
65
|
response.validate(action)
|
65
|
-
|
66
|
+
end.to raise_error(Praxis::Exceptions::Validation, /response definition with that name can be found/)
|
66
67
|
end
|
67
68
|
end
|
68
69
|
|
69
70
|
context 'with an existing response spec in the action' do
|
70
|
-
|
71
71
|
it 'should use the spec to validate the response' do
|
72
72
|
expect(response_spec).to receive(:validate).and_return(nil)
|
73
73
|
|
74
74
|
response.validate(action)
|
75
75
|
end
|
76
76
|
end
|
77
|
-
|
78
|
-
|
79
77
|
end
|
80
78
|
|
81
79
|
describe '#encode!' do
|
82
|
-
let(:response_body_structured_data) { [{
|
80
|
+
let(:response_body_structured_data) { [{ 'moo' => 'bah' }] }
|
83
81
|
let(:response_body_entity) { '[{"moo": "bah"}]' }
|
84
82
|
|
85
83
|
context 'given a String body' do
|
@@ -96,31 +94,28 @@ describe Praxis::Response do
|
|
96
94
|
|
97
95
|
it 'encodes using a suitable handler' do
|
98
96
|
response.encode!
|
99
|
-
expect(JSON.
|
97
|
+
expect(JSON.parse(response.body)).to eq(response_body_structured_data)
|
100
98
|
end
|
101
99
|
end
|
102
100
|
|
103
101
|
context 'when no Content-Type is defined' do
|
104
102
|
before { response.body = response_body_structured_data }
|
105
103
|
|
106
|
-
let(:response_headers)
|
107
|
-
{ 'X-Header'
|
108
|
-
'Location'
|
109
|
-
|
104
|
+
let(:response_headers) do
|
105
|
+
{ 'X-Header' => 'Foobar',
|
106
|
+
'Location' => '/api/resources/123' }
|
107
|
+
end
|
110
108
|
it 'still defaults to the default handler' do
|
111
109
|
response.encode!
|
112
|
-
expect(JSON.
|
110
|
+
expect(JSON.parse(response.body)).to eq(response_body_structured_data)
|
113
111
|
end
|
114
112
|
end
|
115
|
-
|
116
|
-
|
117
113
|
end
|
118
114
|
|
119
115
|
context 'multipart responses' do
|
120
|
-
let(:part) { Praxis::MultipartPart.new('not so ok', {'Status' => 400,
|
116
|
+
let(:part) { Praxis::MultipartPart.new('not so ok', { 'Status' => 400, 'Location' => 'somewhere' }) }
|
121
117
|
|
122
118
|
context '#add_part' do
|
123
|
-
|
124
119
|
context 'without a name' do
|
125
120
|
before do
|
126
121
|
response.add_part(part)
|
@@ -140,18 +135,16 @@ describe Praxis::Response do
|
|
140
135
|
context 'with a name' do
|
141
136
|
let(:part_name) { 'a-part' }
|
142
137
|
before do
|
143
|
-
response.add_part(
|
138
|
+
response.add_part(part, part_name)
|
144
139
|
end
|
145
140
|
|
146
141
|
it 'adds the part' do
|
147
142
|
expect(response.parts[part_name]).to be(part)
|
148
143
|
end
|
149
144
|
end
|
150
|
-
|
151
145
|
end
|
152
146
|
|
153
147
|
context '#finish for a multipart response' do
|
154
|
-
|
155
148
|
before do
|
156
149
|
response.add_part(part)
|
157
150
|
response.body = 'a preamble'
|
@@ -172,7 +165,7 @@ describe Praxis::Response do
|
|
172
165
|
end
|
173
166
|
|
174
167
|
it 'sets the headers' do
|
175
|
-
expect(response.headers['Content-Type']).to match(/
|
168
|
+
expect(response.headers['Content-Type']).to match(%r{multipart/form-data})
|
176
169
|
expect(response.headers['Location']).to eq('/api/resources/123')
|
177
170
|
end
|
178
171
|
|
@@ -180,7 +173,7 @@ describe Praxis::Response do
|
|
180
173
|
parser = Praxis::MultipartParser.new(response.headers, response.body)
|
181
174
|
preamble, parts = parser.parse
|
182
175
|
|
183
|
-
expect(preamble).to eq(
|
176
|
+
expect(preamble).to eq('a preamble')
|
184
177
|
expect(parts).to have(1).item
|
185
178
|
|
186
179
|
part_response = parts.first
|
@@ -189,9 +182,5 @@ describe Praxis::Response do
|
|
189
182
|
expect(part_response.body).to eq('not so ok')
|
190
183
|
end
|
191
184
|
end
|
192
|
-
|
193
185
|
end
|
194
|
-
|
195
|
-
|
196
|
-
|
197
186
|
end
|
@@ -1,8 +1,8 @@
|
|
1
|
-
|
1
|
+
# frozen_string_literal: true
|
2
2
|
|
3
|
+
require 'spec_helper'
|
3
4
|
|
4
5
|
describe Praxis::Responses::InternalServerError do
|
5
|
-
|
6
6
|
context 'a basic response' do
|
7
7
|
subject(:response) { Praxis::Responses::InternalServerError.new }
|
8
8
|
it 'always sets the json content type' do
|
@@ -15,12 +15,11 @@ describe Praxis::Responses::InternalServerError do
|
|
15
15
|
end
|
16
16
|
end
|
17
17
|
|
18
|
-
|
19
18
|
context '.format!' do
|
20
19
|
let(:error) { double('error', message: 'error message', backtrace: [1, 2], cause: cause) }
|
21
20
|
|
22
21
|
let(:show_exceptions) { true }
|
23
|
-
let(:config) { double(
|
22
|
+
let(:config) { double('config', show_exceptions: show_exceptions) }
|
24
23
|
|
25
24
|
before do
|
26
25
|
allow(Praxis::Application.instance.config).to receive(:praxis).and_return(config)
|
@@ -30,18 +29,17 @@ describe Praxis::Responses::InternalServerError do
|
|
30
29
|
end
|
31
30
|
|
32
31
|
context 'with show_exceptions true' do
|
33
|
-
|
34
32
|
context 'without a cause' do
|
35
33
|
let(:cause) { nil }
|
36
34
|
it 'it fills message and backtrace' do
|
37
|
-
expect(response.body).to eq({name: error.class.name, message: error.message, backtrace: error.backtrace})
|
35
|
+
expect(response.body).to eq({ name: error.class.name, message: error.message, backtrace: error.backtrace })
|
38
36
|
end
|
39
37
|
end
|
40
38
|
|
41
39
|
context 'with a cause' do
|
42
40
|
let(:cause) { Exception.new('cause message') }
|
43
41
|
it 'it fills message, backtrace and cause' do
|
44
|
-
expect(response.body.keys).to eq([
|
42
|
+
expect(response.body.keys).to eq(%i[name message backtrace cause])
|
45
43
|
expect(response.body[:name]).to eq(error.class.name)
|
46
44
|
expect(response.body[:message]).to eq(error.message)
|
47
45
|
expect(response.body[:backtrace]).to eq(error.backtrace)
|
@@ -55,11 +53,10 @@ describe Praxis::Responses::InternalServerError do
|
|
55
53
|
expect(response.body).to be_nil
|
56
54
|
response.format!
|
57
55
|
end
|
58
|
-
|
59
56
|
end
|
60
57
|
|
61
58
|
context 'its response template' do
|
62
|
-
let(:template){ Praxis::ApiDefinition.instance.responses[:internal_server_error] }
|
59
|
+
let(:template) { Praxis::ApiDefinition.instance.responses[:internal_server_error] }
|
63
60
|
it 'is registered with the ApiDefinition' do
|
64
61
|
expect(template).to be_kind_of(Praxis::ResponseTemplate)
|
65
62
|
end
|
@@ -1,9 +1,9 @@
|
|
1
|
-
|
1
|
+
# frozen_string_literal: true
|
2
2
|
|
3
|
+
require 'spec_helper'
|
3
4
|
|
4
5
|
describe Praxis::Responses::ValidationError do
|
5
|
-
|
6
|
-
let(:summary){ "Something happened here" }
|
6
|
+
let(:summary) { 'Something happened here' }
|
7
7
|
context 'a basic response' do
|
8
8
|
subject(:response) { Praxis::Responses::ValidationError.new(summary: summary) }
|
9
9
|
it 'always sets the json content type' do
|
@@ -18,10 +18,10 @@ describe Praxis::Responses::ValidationError do
|
|
18
18
|
end
|
19
19
|
|
20
20
|
context '.format!' do
|
21
|
-
let(:errors) { [1,2] }
|
22
|
-
let(:cause){ Exception.new(
|
23
|
-
let(:exception_message){
|
24
|
-
let(:exception){ nil }
|
21
|
+
let(:errors) { [1, 2] }
|
22
|
+
let(:cause) { Exception.new('cause message') }
|
23
|
+
let(:exception_message) { 'exception message' }
|
24
|
+
let(:exception) { nil }
|
25
25
|
subject(:response) { Praxis::Responses::ValidationError.new(summary: summary, errors: errors, exception: exception) }
|
26
26
|
before do
|
27
27
|
expect(response.body).to be_nil
|
@@ -29,18 +29,18 @@ describe Praxis::Responses::ValidationError do
|
|
29
29
|
context 'when errors' do
|
30
30
|
it 'it fills the errors key' do
|
31
31
|
response.format!
|
32
|
-
expect(response.body).to eq({name: 'ValidationError', summary: summary, errors: errors})
|
32
|
+
expect(response.body).to eq({ name: 'ValidationError', summary: summary, errors: errors })
|
33
33
|
end
|
34
34
|
end
|
35
35
|
|
36
36
|
context 'without errors but with an exception' do
|
37
37
|
let(:errors) { nil }
|
38
|
-
let(:exception){ double(
|
38
|
+
let(:exception) { double('exception', message: exception_message, cause: cause) }
|
39
39
|
before do
|
40
|
-
|
41
|
-
|
42
|
-
|
43
|
-
|
40
|
+
response.format!
|
41
|
+
expect(response.body.keys).to include(:name, :summary)
|
42
|
+
expect(response.body[:name]).to eq('ValidationError')
|
43
|
+
expect(response.body[:summary]).to eq(summary)
|
44
44
|
end
|
45
45
|
|
46
46
|
context 'that has a message' do
|
@@ -51,14 +51,14 @@ describe Praxis::Responses::ValidationError do
|
|
51
51
|
end
|
52
52
|
|
53
53
|
context 'that does not contain a message' do
|
54
|
-
let(:exception_message){ nil }
|
54
|
+
let(:exception_message) { nil }
|
55
55
|
it 'does not touch the errors key' do
|
56
56
|
expect(response.body).to_not have_key(:errors)
|
57
57
|
end
|
58
58
|
end
|
59
59
|
|
60
60
|
context 'without cause' do
|
61
|
-
let(:cause){ nil}
|
61
|
+
let(:cause) { nil }
|
62
62
|
it 'does not include it in the output' do
|
63
63
|
expect(response.body).to_not have_key(:cause)
|
64
64
|
end
|
@@ -67,15 +67,14 @@ describe Praxis::Responses::ValidationError do
|
|
67
67
|
context 'with a cause' do
|
68
68
|
it 'it fills the cause too' do
|
69
69
|
expect(response.body).to have_key(:cause)
|
70
|
-
expect(response.body[:cause]).to eq({name: cause.class.name, message: cause.message })
|
70
|
+
expect(response.body[:cause]).to eq({ name: cause.class.name, message: cause.message })
|
71
71
|
end
|
72
72
|
end
|
73
73
|
end
|
74
|
-
|
75
74
|
end
|
76
75
|
|
77
76
|
context 'its response template' do
|
78
|
-
let(:template){ Praxis::ApiDefinition.instance.responses[:validation_error] }
|
77
|
+
let(:template) { Praxis::ApiDefinition.instance.responses[:validation_error] }
|
79
78
|
it 'is registered with the ApiDefinition' do
|
80
79
|
expect(template).to be_kind_of(Praxis::ResponseTemplate)
|
81
80
|
end
|
data/spec/praxis/route_spec.rb
CHANGED
@@ -1,7 +1,8 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
1
3
|
require 'spec_helper'
|
2
4
|
|
3
5
|
describe Praxis::Route do
|
4
|
-
|
5
6
|
let(:verb) { 'GET' }
|
6
7
|
let(:path) { '/base/stuff' }
|
7
8
|
let(:prefixed_path) { '/stuff' }
|
@@ -23,15 +24,11 @@ describe Praxis::Route do
|
|
23
24
|
|
24
25
|
context '#describe' do
|
25
26
|
subject(:description) { route.describe }
|
26
|
-
it { should eq({verb:verb, path:path
|
27
|
+
it { should eq({ verb: verb, path: path, version: version }) }
|
27
28
|
|
28
29
|
context 'with options' do
|
29
|
-
let(:options) { {option: 'value'} }
|
30
|
+
let(:options) { { option: 'value' } }
|
30
31
|
its([:options]) { should eq(options) }
|
31
32
|
end
|
32
|
-
|
33
33
|
end
|
34
|
-
|
35
|
-
|
36
|
-
|
37
34
|
end
|
data/spec/praxis/router_spec.rb
CHANGED
@@ -1,26 +1,28 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
1
3
|
require 'spec_helper'
|
2
4
|
|
3
5
|
describe Praxis::Router do
|
4
6
|
describe Praxis::Router::VersionMatcher do
|
5
|
-
let(:endpoint_definition){ double(
|
6
|
-
let(:action){ double(
|
7
|
-
let(:target){ double(
|
8
|
-
let(:args){ {version:
|
9
|
-
subject(:matcher){ Praxis::Router::VersionMatcher.new(target
|
7
|
+
let(:endpoint_definition) { double('endpoint_definition', version_options: { using: %i[header params] }) }
|
8
|
+
let(:action) { double('action', endpoint_definition: endpoint_definition) }
|
9
|
+
let(:target) { double('target', action: action) }
|
10
|
+
let(:args) { { version: '1.0' } }
|
11
|
+
subject(:matcher) { Praxis::Router::VersionMatcher.new(target, **args) }
|
10
12
|
|
11
13
|
context '.initialize' do
|
12
|
-
let(:args){ {} }
|
14
|
+
let(:args) { {} }
|
13
15
|
it 'defaults to no version' do
|
14
|
-
expect(
|
16
|
+
expect(matcher.instance_variable_get(:@version)).to eq('n/a')
|
15
17
|
end
|
16
18
|
end
|
17
19
|
|
18
20
|
context '.call' do
|
19
|
-
let(:env){ {
|
20
|
-
let(:request) {Praxis::Request.new(env)}
|
21
|
+
let(:env) { { 'HTTP_X_API_VERSION' => request_version } }
|
22
|
+
let(:request) { Praxis::Request.new(env) }
|
21
23
|
|
22
24
|
context 'with matching versions' do
|
23
|
-
let(:request_version) {
|
25
|
+
let(:request_version) { '1.0' }
|
24
26
|
it 'calls the target' do
|
25
27
|
expect(target).to receive(:call).with(request)
|
26
28
|
matcher.call(request)
|
@@ -28,67 +30,63 @@ describe Praxis::Router do
|
|
28
30
|
end
|
29
31
|
|
30
32
|
context 'with non-matching versions' do
|
31
|
-
let(:request_version) {
|
33
|
+
let(:request_version) { '4.0' }
|
32
34
|
before do
|
33
35
|
expect { matcher.call(request) }.to throw_symbol(:pass)
|
34
36
|
end
|
35
37
|
it 'does not call the target' do
|
36
|
-
expect(
|
38
|
+
expect(target).not_to receive(:call).with(request)
|
37
39
|
end
|
38
40
|
it 'saves the unmatched version' do
|
39
|
-
expect(
|
41
|
+
expect(request.unmatched_versions).to include(args[:version])
|
40
42
|
end
|
41
43
|
end
|
42
|
-
|
43
44
|
end
|
44
45
|
end
|
45
46
|
|
46
47
|
describe Praxis::Router::RequestRouter do
|
48
|
+
let(:request) { double('request', route_params: '', path: 'path') }
|
49
|
+
let(:callback) { double('callback') }
|
47
50
|
|
48
|
-
|
49
|
-
let(:callback) {double("callback")}
|
51
|
+
subject(:request_router) { Praxis::Router::RequestRouter.new }
|
50
52
|
|
51
|
-
|
52
|
-
|
53
|
-
context ".invoke" do
|
54
|
-
it "update request and call request for callback" do
|
53
|
+
context '.invoke' do
|
54
|
+
it 'update request and call request for callback' do
|
55
55
|
allow(request).to receive(:route_params=)
|
56
56
|
allow(callback).to receive(:call).and_return(1)
|
57
57
|
|
58
|
-
invoke_call = request_router.invoke(callback, request,
|
58
|
+
invoke_call = request_router.invoke(callback, request, 'params', 'pattern')
|
59
59
|
expect(invoke_call).to eq(1)
|
60
60
|
end
|
61
61
|
end
|
62
62
|
|
63
|
-
context
|
64
|
-
it
|
63
|
+
context '.string_for' do
|
64
|
+
it 'returns request path string' do
|
65
65
|
expect(request_router.string_for(request)).to eq('path')
|
66
66
|
end
|
67
67
|
end
|
68
68
|
end
|
69
69
|
|
70
|
-
let(:application) { instance_double('Praxis::Application')}
|
71
|
-
subject(:router) {Praxis::Router.new(application)}
|
70
|
+
let(:application) { instance_double('Praxis::Application') }
|
71
|
+
subject(:router) { Praxis::Router.new(application) }
|
72
72
|
|
73
|
-
context
|
74
|
-
its(:request_class) {should be(Praxis::Request)}
|
73
|
+
context 'attributes' do
|
74
|
+
its(:request_class) { should be(Praxis::Request) }
|
75
75
|
end
|
76
76
|
|
77
|
-
context
|
78
|
-
|
79
|
-
let(:
|
80
|
-
let(:
|
81
|
-
let(:verb_router){ double('verb_router') }
|
77
|
+
context '.add_route' do
|
78
|
+
let(:route) { double('route', options: [1], version: 1, verb: 'verb', path: 'path') }
|
79
|
+
let(:target) { double('target') }
|
80
|
+
let(:verb_router) { double('verb_router') }
|
82
81
|
|
83
|
-
let(:endpoint_definition) { double('endpoint_definition', version_options: {using: [
|
82
|
+
let(:endpoint_definition) { double('endpoint_definition', version_options: { using: %i[header params] }) }
|
84
83
|
let(:action) { double('action', endpoint_definition: endpoint_definition) }
|
85
84
|
let(:target) { double('target', action: action) }
|
86
85
|
|
87
|
-
|
88
86
|
context 'with params or header versioning' do
|
89
87
|
it 'wraps the target with a VersionMatcher' do
|
90
|
-
router.instance_variable_set(
|
91
|
-
expect(verb_router).to receive(:on) do|path, args
|
88
|
+
router.instance_variable_set(:@routes, { 'verb' => verb_router }) # Ugly, but no need to have a reader
|
89
|
+
expect(verb_router).to receive(:on) do |path, args| # .with(route.path, call: "foo")
|
92
90
|
expect(path).to eq(route.path)
|
93
91
|
expect(args).to be_kind_of(Hash)
|
94
92
|
expect(args[:call]).to be_kind_of(Praxis::Router::VersionMatcher)
|
@@ -96,75 +94,72 @@ describe Praxis::Router do
|
|
96
94
|
router.add_route(target, route)
|
97
95
|
end
|
98
96
|
end
|
99
|
-
|
100
97
|
end
|
101
98
|
|
102
|
-
context
|
103
|
-
let(:env){ {
|
99
|
+
context '.call' do
|
100
|
+
let(:env) { { 'PATH_INFO' => request_path_info, 'REQUEST_METHOD' => request_verb } }
|
104
101
|
let(:request_verb) { 'POST' }
|
105
102
|
let(:request_path_info) { '/' }
|
106
|
-
let(:request_version){ nil }
|
107
|
-
let(:request) {Praxis::Request.new(env)}
|
108
|
-
let(:router_response){ 1 }
|
109
|
-
let(:router_response_for_post){
|
110
|
-
let(:router_response_for_wildcard){
|
111
|
-
let(:post_target_router){ double(
|
112
|
-
let(:any_target_router){ double(
|
113
|
-
|
114
|
-
let(:endpoint_definition) { double('endpoint_definition', version_options: {using: [
|
103
|
+
let(:request_version) { nil }
|
104
|
+
let(:request) { Praxis::Request.new(env) }
|
105
|
+
let(:router_response) { 1 }
|
106
|
+
let(:router_response_for_post) { 'POST result' }
|
107
|
+
let(:router_response_for_wildcard) { '* result' }
|
108
|
+
let(:post_target_router) { double('POST target', call: router_response_for_post) }
|
109
|
+
let(:any_target_router) { double('ANY target', call: router_response_for_wildcard) }
|
110
|
+
|
111
|
+
let(:endpoint_definition) { double('endpoint_definition', version_options: { using: %i[header params] }) }
|
115
112
|
let(:action) { double('action', endpoint_definition: endpoint_definition) }
|
116
113
|
let(:target) { double('target', action: action) }
|
117
114
|
|
118
115
|
before do
|
119
116
|
env['HTTP_X_API_VERSION'] = request_version if request_version
|
120
|
-
allow_any_instance_of(Praxis::Router::RequestRouter)
|
121
|
-
to receive(:call).with(request).and_return(router_response)
|
122
|
-
router.add_route(target,double(
|
117
|
+
allow_any_instance_of(Praxis::Router::RequestRouter)
|
118
|
+
.to receive(:call).with(request).and_return(router_response)
|
119
|
+
router.add_route(target, double('route1', verb: 'POST', path: '/', options: {}, version: request_version))
|
123
120
|
# Hijack the callable block in the routes (since there's no way to point back to the registered route object)
|
124
121
|
router.instance_variable_get(:@routes)['POST'] = post_target_router
|
125
|
-
|
126
122
|
end
|
127
123
|
|
128
124
|
context 'for routes without wildcards (a single POST route)' do
|
129
|
-
|
130
125
|
context 'and an incoming POST request' do
|
131
|
-
it
|
126
|
+
it 'finds a match and invokes it the route' do
|
132
127
|
expect(router.call(request)).to eq(router_response_for_post)
|
133
128
|
end
|
134
129
|
end
|
135
130
|
context 'and an incoming PUT request' do
|
136
131
|
let(:request_verb) { 'PUT' }
|
137
|
-
it
|
138
|
-
response_code,
|
132
|
+
it 'does not find a route' do
|
133
|
+
response_code, = router.call(request)
|
139
134
|
expect(response_code).to be(404)
|
140
135
|
end
|
141
136
|
end
|
142
137
|
end
|
143
138
|
|
144
139
|
context 'for routes with wildcards (a POST and a * route)' do
|
145
|
-
let(:endpoint_definition) { double('endpoint_definition', version_options: {using: [
|
140
|
+
let(:endpoint_definition) { double('endpoint_definition', version_options: { using: %i[header params] }) }
|
146
141
|
let(:action) { double('action', endpoint_definition: endpoint_definition) }
|
147
142
|
let(:target) { double('target', action: action) }
|
148
143
|
|
149
144
|
before do
|
150
|
-
router.add_route(target,double(
|
145
|
+
router.add_route(target, double('route2', verb: 'ANY', path: '/*', options: {}, version: request_version))
|
151
146
|
# Hijack the callable block in the routes (since there's no way to point back to the registered route object)
|
152
|
-
router.instance_variable_get(
|
147
|
+
router.instance_variable_get(:@routes)['ANY'] = any_target_router
|
153
148
|
end
|
154
149
|
|
155
150
|
context 'and an incoming PUT request' do
|
156
151
|
let(:request_verb) { 'PUT' }
|
157
|
-
it
|
152
|
+
it 'it can successfully find a match using the wildcard target' do
|
158
153
|
expect(router.call(request)).to eq(router_response_for_wildcard)
|
159
154
|
end
|
160
155
|
end
|
161
156
|
context 'and an incoming POST request' do
|
162
|
-
it 'matches the most specific POST route, rather than the wildcard'do
|
157
|
+
it 'matches the most specific POST route, rather than the wildcard' do
|
163
158
|
expect(router.call(request)).to eq(router_response_for_post)
|
164
159
|
end
|
165
160
|
end
|
166
161
|
context 'and an incoming POST request (but that does not match other route conditions)' do
|
167
|
-
let(:router_response_for_post){ :not_found }
|
162
|
+
let(:router_response_for_post) { :not_found }
|
168
163
|
it 'still matches wildcard verb if that was route conditions-compatible' do
|
169
164
|
expect(post_target_router).to receive(:call).once # try the match cause it's a POST
|
170
165
|
expect(any_target_router).to receive(:call).once # fallback to wildcard upon a not_found
|
@@ -173,13 +168,12 @@ describe Praxis::Router do
|
|
173
168
|
end
|
174
169
|
end
|
175
170
|
|
176
|
-
context
|
171
|
+
context 'when not_found is returned' do
|
177
172
|
let(:request_verb) { 'DELETE' }
|
178
|
-
let(:router_response){ :not_found }
|
179
|
-
|
173
|
+
let(:router_response) { :not_found }
|
180
174
|
|
181
175
|
it 'sets X-Cascade: pass by default' do
|
182
|
-
_, headers,
|
176
|
+
_, headers, = router.call(request)
|
183
177
|
expect(headers).to have_key('X-Cascade')
|
184
178
|
expect(headers['X-Cascade']).to eq('pass')
|
185
179
|
end
|
@@ -191,10 +185,9 @@ describe Praxis::Router do
|
|
191
185
|
end
|
192
186
|
|
193
187
|
it 'does not set X-Cascade: pass' do
|
194
|
-
_, headers,
|
195
|
-
expect(headers).to_not have_key(
|
188
|
+
_, headers, = router.call(request)
|
189
|
+
expect(headers).to_not have_key('X-Cascade')
|
196
190
|
end
|
197
|
-
|
198
191
|
end
|
199
192
|
|
200
193
|
context 'with versioning' do
|
@@ -202,29 +195,28 @@ describe Praxis::Router do
|
|
202
195
|
request.instance_variable_set(:@unmatched_versions, unmatched_versions)
|
203
196
|
end
|
204
197
|
|
205
|
-
context
|
198
|
+
context 'having passed no version in the request' do
|
206
199
|
context 'and no controllers matching the path' do
|
207
200
|
let(:unmatched_versions) { Set.new([]) }
|
208
201
|
|
209
202
|
it 'returns a basic "NotFound" response: 404 status, text/plain content and "NotFound" body' do
|
210
|
-
expect(
|
203
|
+
expect(router.call(request)).to eq([404, { 'Content-Type' => 'text/plain', 'X-Cascade' => 'pass' }, ['NotFound']])
|
211
204
|
end
|
212
205
|
end
|
213
206
|
|
214
207
|
context 'and some controllers matching the path' do
|
215
|
-
let(:unmatched_versions) { Set.new([
|
208
|
+
let(:unmatched_versions) { Set.new(['1.0']) }
|
216
209
|
it 'returns a specific body response noting which request versions would matched if passed in' do
|
217
210
|
_, _, body = router.call(request)
|
218
|
-
expect(
|
211
|
+
expect(body.first).to eq('NotFound. Your request did not specify an API version. Available versions = "1.0".')
|
219
212
|
end
|
220
213
|
end
|
221
214
|
end
|
222
215
|
|
223
|
-
context
|
224
|
-
|
216
|
+
context 'having passed a version in the request' do
|
225
217
|
context 'but having no controllers matching the path part' do
|
226
|
-
let(:request_version) {
|
227
|
-
let(:unmatched_versions) { Set.new([
|
218
|
+
let(:request_version) { '50.0' }
|
219
|
+
let(:unmatched_versions) { Set.new(['1.0', '2.0']) }
|
228
220
|
|
229
221
|
it 'returns a specific body response noting that the version might be wrong (and which could be right)' do
|
230
222
|
code, headers, body = router.call(request)
|
@@ -235,8 +227,6 @@ describe Praxis::Router do
|
|
235
227
|
end
|
236
228
|
end
|
237
229
|
end
|
238
|
-
|
239
|
-
|
240
230
|
end
|
241
231
|
end
|
242
232
|
end
|