actionpack 5.2.8.1 → 6.0.0.beta1
Sign up to get free protection for your applications and to get access to all the features.
Potentially problematic release.
This version of actionpack might be problematic. Click here for more details.
- checksums.yaml +4 -4
- data/CHANGELOG.md +109 -482
- data/MIT-LICENSE +1 -1
- data/README.rdoc +1 -1
- data/lib/abstract_controller/base.rb +4 -2
- data/lib/abstract_controller/caching/fragments.rb +6 -21
- data/lib/abstract_controller/callbacks.rb +12 -0
- data/lib/abstract_controller/collector.rb +1 -1
- data/lib/abstract_controller/helpers.rb +2 -2
- data/lib/abstract_controller/railties/routes_helpers.rb +1 -1
- data/lib/action_controller/api.rb +2 -1
- data/lib/action_controller/base.rb +2 -7
- data/lib/action_controller/caching.rb +1 -1
- data/lib/action_controller/log_subscriber.rb +8 -5
- data/lib/action_controller/metal/conditional_get.rb +9 -3
- data/lib/action_controller/metal/data_streaming.rb +5 -6
- data/lib/action_controller/metal/default_headers.rb +17 -0
- data/lib/action_controller/metal/exceptions.rb +22 -1
- data/lib/action_controller/metal/flash.rb +5 -5
- data/lib/action_controller/metal/force_ssl.rb +17 -57
- data/lib/action_controller/metal/head.rb +1 -1
- data/lib/action_controller/metal/helpers.rb +1 -2
- data/lib/action_controller/metal/http_authentication.rb +21 -22
- data/lib/action_controller/metal/implicit_render.rb +2 -12
- data/lib/action_controller/metal/instrumentation.rb +3 -5
- data/lib/action_controller/metal/live.rb +28 -26
- data/lib/action_controller/metal/mime_responds.rb +13 -2
- data/lib/action_controller/metal/params_wrapper.rb +18 -14
- data/lib/action_controller/metal/redirecting.rb +32 -11
- data/lib/action_controller/metal/rendering.rb +1 -1
- data/lib/action_controller/metal/request_forgery_protection.rb +32 -97
- data/lib/action_controller/metal/strong_parameters.rb +57 -34
- data/lib/action_controller/metal/url_for.rb +1 -1
- data/lib/action_controller/metal.rb +2 -2
- data/lib/action_controller/railties/helpers.rb +1 -1
- data/lib/action_controller/renderer.rb +15 -2
- data/lib/action_controller/test_case.rb +5 -9
- data/lib/action_controller.rb +1 -0
- data/lib/action_dispatch/http/cache.rb +14 -10
- data/lib/action_dispatch/http/content_disposition.rb +45 -0
- data/lib/action_dispatch/http/content_security_policy.rb +17 -8
- data/lib/action_dispatch/http/filter_parameters.rb +8 -6
- data/lib/action_dispatch/http/filter_redirect.rb +1 -1
- data/lib/action_dispatch/http/headers.rb +1 -1
- data/lib/action_dispatch/http/mime_negotiation.rb +7 -10
- data/lib/action_dispatch/http/mime_type.rb +1 -5
- data/lib/action_dispatch/http/parameter_filter.rb +5 -79
- data/lib/action_dispatch/http/parameters.rb +13 -3
- data/lib/action_dispatch/http/request.rb +10 -13
- data/lib/action_dispatch/http/response.rb +14 -14
- data/lib/action_dispatch/http/upload.rb +5 -0
- data/lib/action_dispatch/http/url.rb +81 -81
- data/lib/action_dispatch/journey/formatter.rb +1 -1
- data/lib/action_dispatch/journey/nfa/simulator.rb +0 -2
- data/lib/action_dispatch/journey/nodes/node.rb +9 -8
- data/lib/action_dispatch/journey/path/pattern.rb +3 -4
- data/lib/action_dispatch/journey/router/utils.rb +10 -10
- data/lib/action_dispatch/journey/router.rb +0 -3
- data/lib/action_dispatch/journey/scanner.rb +11 -4
- data/lib/action_dispatch/journey/visitors.rb +1 -1
- data/lib/action_dispatch/middleware/callbacks.rb +2 -4
- data/lib/action_dispatch/middleware/cookies.rb +49 -70
- data/lib/action_dispatch/middleware/debug_exceptions.rb +32 -58
- data/lib/action_dispatch/middleware/debug_locks.rb +5 -5
- data/lib/action_dispatch/middleware/debug_view.rb +50 -0
- data/lib/action_dispatch/middleware/exception_wrapper.rb +36 -7
- data/lib/action_dispatch/middleware/executor.rb +1 -1
- data/lib/action_dispatch/middleware/flash.rb +1 -1
- data/lib/action_dispatch/middleware/host_authorization.rb +103 -0
- data/lib/action_dispatch/middleware/remote_ip.rb +6 -8
- data/lib/action_dispatch/middleware/request_id.rb +2 -2
- data/lib/action_dispatch/middleware/session/abstract_store.rb +0 -14
- data/lib/action_dispatch/middleware/session/cache_store.rb +6 -11
- data/lib/action_dispatch/middleware/session/cookie_store.rb +11 -27
- data/lib/action_dispatch/middleware/ssl.rb +8 -8
- data/lib/action_dispatch/middleware/stack.rb +2 -2
- data/lib/action_dispatch/middleware/static.rb +5 -6
- data/lib/action_dispatch/middleware/templates/rescues/_source.html.erb +4 -2
- data/lib/action_dispatch/middleware/templates/rescues/_trace.html.erb +45 -35
- data/lib/action_dispatch/middleware/templates/rescues/blocked_host.html.erb +7 -0
- data/lib/action_dispatch/middleware/templates/rescues/blocked_host.text.erb +5 -0
- data/lib/action_dispatch/middleware/templates/rescues/diagnostics.html.erb +20 -2
- data/lib/action_dispatch/middleware/templates/rescues/invalid_statement.html.erb +4 -4
- data/lib/action_dispatch/middleware/templates/rescues/invalid_statement.text.erb +2 -2
- data/lib/action_dispatch/middleware/templates/rescues/missing_exact_template.html.erb +19 -0
- data/lib/action_dispatch/middleware/templates/rescues/missing_exact_template.text.erb +3 -0
- data/lib/action_dispatch/middleware/templates/rescues/missing_template.html.erb +2 -2
- data/lib/action_dispatch/middleware/templates/rescues/routing_error.html.erb +1 -1
- data/lib/action_dispatch/middleware/templates/rescues/template_error.html.erb +2 -2
- data/lib/action_dispatch/middleware/templates/routes/_table.html.erb +3 -0
- data/lib/action_dispatch/railtie.rb +1 -0
- data/lib/action_dispatch/request/session.rb +8 -6
- data/lib/action_dispatch/routing/inspector.rb +99 -50
- data/lib/action_dispatch/routing/mapper.rb +36 -29
- data/lib/action_dispatch/routing/polymorphic_routes.rb +7 -12
- data/lib/action_dispatch/routing/route_set.rb +11 -12
- data/lib/action_dispatch/routing/url_for.rb +1 -0
- data/lib/action_dispatch/routing.rb +3 -2
- data/lib/action_dispatch/system_testing/test_helpers/screenshot_helper.rb +3 -3
- data/lib/action_dispatch/testing/assertions/response.rb +2 -3
- data/lib/action_dispatch/testing/assertions/routing.rb +7 -2
- data/lib/action_dispatch/testing/integration.rb +11 -5
- data/lib/action_dispatch/testing/test_process.rb +2 -2
- data/lib/action_dispatch/testing/test_response.rb +4 -32
- data/lib/action_dispatch.rb +7 -6
- data/lib/action_pack/gem_version.rb +4 -4
- data/lib/action_pack.rb +1 -1
- metadata +25 -23
@@ -1,7 +1,7 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
3
|
require "delegate"
|
4
|
-
require "
|
4
|
+
require "io/console/size"
|
5
5
|
|
6
6
|
module ActionDispatch
|
7
7
|
module Routing
|
@@ -61,11 +61,11 @@ module ActionDispatch
|
|
61
61
|
@routes = routes
|
62
62
|
end
|
63
63
|
|
64
|
-
def format(formatter, filter =
|
64
|
+
def format(formatter, filter = {})
|
65
65
|
routes_to_display = filter_routes(normalize_filter(filter))
|
66
66
|
routes = collect_routes(routes_to_display)
|
67
67
|
if routes.none?
|
68
|
-
formatter.no_routes(collect_routes(@routes))
|
68
|
+
formatter.no_routes(collect_routes(@routes), filter)
|
69
69
|
return formatter.result
|
70
70
|
end
|
71
71
|
|
@@ -81,12 +81,12 @@ module ActionDispatch
|
|
81
81
|
end
|
82
82
|
|
83
83
|
private
|
84
|
-
|
85
84
|
def normalize_filter(filter)
|
86
|
-
if filter
|
85
|
+
if filter[:controller]
|
87
86
|
{ controller: /#{filter[:controller].underscore.sub(/_?controller\z/, "")}/ }
|
88
|
-
elsif filter
|
89
|
-
{ controller: /#{filter}/, action: /#{filter
|
87
|
+
elsif filter[:grep]
|
88
|
+
{ controller: /#{filter[:grep]}/, action: /#{filter[:grep]}/,
|
89
|
+
verb: /#{filter[:grep]}/, name: /#{filter[:grep]}/, path: /#{filter[:grep]}/ }
|
90
90
|
end
|
91
91
|
end
|
92
92
|
|
@@ -126,62 +126,111 @@ module ActionDispatch
|
|
126
126
|
end
|
127
127
|
end
|
128
128
|
|
129
|
-
|
130
|
-
|
131
|
-
|
132
|
-
|
129
|
+
module ConsoleFormatter
|
130
|
+
class Base
|
131
|
+
def initialize
|
132
|
+
@buffer = []
|
133
|
+
end
|
133
134
|
|
134
|
-
|
135
|
-
|
136
|
-
|
135
|
+
def result
|
136
|
+
@buffer.join("\n")
|
137
|
+
end
|
137
138
|
|
138
|
-
|
139
|
-
|
140
|
-
end
|
139
|
+
def section_title(title)
|
140
|
+
end
|
141
141
|
|
142
|
-
|
143
|
-
|
144
|
-
end
|
142
|
+
def section(routes)
|
143
|
+
end
|
145
144
|
|
146
|
-
|
147
|
-
|
148
|
-
end
|
145
|
+
def header(routes)
|
146
|
+
end
|
149
147
|
|
150
|
-
|
151
|
-
|
152
|
-
|
153
|
-
|
154
|
-
|
148
|
+
def no_routes(routes, filter)
|
149
|
+
@buffer <<
|
150
|
+
if routes.none?
|
151
|
+
<<~MESSAGE
|
152
|
+
You don't have any routes defined!
|
153
|
+
|
154
|
+
Please add some routes in config/routes.rb.
|
155
|
+
MESSAGE
|
156
|
+
elsif filter.key?(:controller)
|
157
|
+
"No routes were found for this controller."
|
158
|
+
elsif filter.key?(:grep)
|
159
|
+
"No routes were found for this grep pattern."
|
160
|
+
end
|
155
161
|
|
156
|
-
|
157
|
-
MESSAGE
|
158
|
-
else
|
159
|
-
"No routes were found for this controller"
|
162
|
+
@buffer << "For more information about routes, see the Rails guide: https://guides.rubyonrails.org/routing.html."
|
160
163
|
end
|
161
|
-
@buffer << "For more information about routes, see the Rails guide: http://guides.rubyonrails.org/routing.html."
|
162
164
|
end
|
163
165
|
|
164
|
-
|
165
|
-
def
|
166
|
-
|
167
|
-
|
166
|
+
class Sheet < Base
|
167
|
+
def section_title(title)
|
168
|
+
@buffer << "\n#{title}:"
|
169
|
+
end
|
168
170
|
|
169
|
-
|
170
|
-
|
171
|
-
|
171
|
+
def section(routes)
|
172
|
+
@buffer << draw_section(routes)
|
173
|
+
end
|
174
|
+
|
175
|
+
def header(routes)
|
176
|
+
@buffer << draw_header(routes)
|
172
177
|
end
|
173
178
|
|
174
|
-
|
175
|
-
|
179
|
+
private
|
180
|
+
|
181
|
+
def draw_section(routes)
|
182
|
+
header_lengths = ["Prefix", "Verb", "URI Pattern"].map(&:length)
|
183
|
+
name_width, verb_width, path_width = widths(routes).zip(header_lengths).map(&:max)
|
184
|
+
|
185
|
+
routes.map do |r|
|
186
|
+
"#{r[:name].rjust(name_width)} #{r[:verb].ljust(verb_width)} #{r[:path].ljust(path_width)} #{r[:reqs]}"
|
187
|
+
end
|
188
|
+
end
|
189
|
+
|
190
|
+
def draw_header(routes)
|
191
|
+
name_width, verb_width, path_width = widths(routes)
|
192
|
+
|
193
|
+
"#{"Prefix".rjust(name_width)} #{"Verb".ljust(verb_width)} #{"URI Pattern".ljust(path_width)} Controller#Action"
|
194
|
+
end
|
195
|
+
|
196
|
+
def widths(routes)
|
197
|
+
[routes.map { |r| r[:name].length }.max || 0,
|
198
|
+
routes.map { |r| r[:verb].length }.max || 0,
|
199
|
+
routes.map { |r| r[:path].length }.max || 0]
|
200
|
+
end
|
201
|
+
end
|
176
202
|
|
177
|
-
|
203
|
+
class Expanded < Base
|
204
|
+
def section_title(title)
|
205
|
+
@buffer << "\n#{"[ #{title} ]"}"
|
178
206
|
end
|
179
207
|
|
180
|
-
def
|
181
|
-
|
182
|
-
routes.map { |r| r[:verb].length }.max || 0,
|
183
|
-
routes.map { |r| r[:path].length }.max || 0]
|
208
|
+
def section(routes)
|
209
|
+
@buffer << draw_expanded_section(routes)
|
184
210
|
end
|
211
|
+
|
212
|
+
private
|
213
|
+
|
214
|
+
def draw_expanded_section(routes)
|
215
|
+
routes.map.each_with_index do |r, i|
|
216
|
+
<<~MESSAGE.chomp
|
217
|
+
#{route_header(index: i + 1)}
|
218
|
+
Prefix | #{r[:name]}
|
219
|
+
Verb | #{r[:verb]}
|
220
|
+
URI | #{r[:path]}
|
221
|
+
Controller#Action | #{r[:reqs]}
|
222
|
+
MESSAGE
|
223
|
+
end
|
224
|
+
end
|
225
|
+
|
226
|
+
def route_header(index:)
|
227
|
+
console_width = IO.console_size.second
|
228
|
+
header_prefix = "--[ Route #{index} ]"
|
229
|
+
dash_remainder = [console_width - header_prefix.size, 0].max
|
230
|
+
|
231
|
+
"#{header_prefix}#{'-' * dash_remainder}"
|
232
|
+
end
|
233
|
+
end
|
185
234
|
end
|
186
235
|
|
187
236
|
class HtmlTableFormatter
|
@@ -203,16 +252,16 @@ module ActionDispatch
|
|
203
252
|
end
|
204
253
|
|
205
254
|
def no_routes(*)
|
206
|
-
@buffer <<
|
255
|
+
@buffer << <<~MESSAGE
|
207
256
|
<p>You don't have any routes defined!</p>
|
208
257
|
<ul>
|
209
258
|
<li>Please add some routes in <tt>config/routes.rb</tt>.</li>
|
210
259
|
<li>
|
211
260
|
For more information about routes, please see the Rails guide
|
212
|
-
<a href="
|
261
|
+
<a href="https://guides.rubyonrails.org/routing.html">Rails Routing from the Outside In</a>.
|
213
262
|
</li>
|
214
263
|
</ul>
|
215
|
-
|
264
|
+
MESSAGE
|
216
265
|
end
|
217
266
|
|
218
267
|
def result
|
@@ -50,7 +50,19 @@ module ActionDispatch
|
|
50
50
|
|
51
51
|
private
|
52
52
|
def constraint_args(constraint, request)
|
53
|
-
|
53
|
+
arity = if constraint.respond_to?(:arity)
|
54
|
+
constraint.arity
|
55
|
+
else
|
56
|
+
constraint.method(:call).arity
|
57
|
+
end
|
58
|
+
|
59
|
+
if arity < 1
|
60
|
+
[]
|
61
|
+
elsif arity == 1
|
62
|
+
[request]
|
63
|
+
else
|
64
|
+
[request.path_parameters, request]
|
65
|
+
end
|
54
66
|
end
|
55
67
|
end
|
56
68
|
|
@@ -148,17 +160,8 @@ module ActionDispatch
|
|
148
160
|
end
|
149
161
|
|
150
162
|
def make_route(name, precedence)
|
151
|
-
|
152
|
-
|
153
|
-
path,
|
154
|
-
conditions,
|
155
|
-
required_defaults,
|
156
|
-
defaults,
|
157
|
-
request_method,
|
158
|
-
precedence,
|
159
|
-
@internal)
|
160
|
-
|
161
|
-
route
|
163
|
+
Journey::Route.new(name, application, path, conditions, required_defaults,
|
164
|
+
defaults, request_method, precedence, @internal)
|
162
165
|
end
|
163
166
|
|
164
167
|
def application
|
@@ -279,7 +282,7 @@ module ActionDispatch
|
|
279
282
|
|
280
283
|
def verify_regexp_requirements(requirements)
|
281
284
|
requirements.each do |requirement|
|
282
|
-
if requirement.source
|
285
|
+
if ANCHOR_CHARACTERS_REGEX.match?(requirement.source)
|
283
286
|
raise ArgumentError, "Regexp anchor characters are not allowed in routing requirements: #{requirement.inspect}"
|
284
287
|
end
|
285
288
|
|
@@ -308,8 +311,8 @@ module ActionDispatch
|
|
308
311
|
def check_controller_and_action(path_params, controller, action)
|
309
312
|
hash = check_part(:controller, controller, path_params, {}) do |part|
|
310
313
|
translate_controller(part) {
|
311
|
-
message = "'#{part}' is not a supported controller name. This can lead to potential routing problems."
|
312
|
-
message << " See
|
314
|
+
message = +"'#{part}' is not a supported controller name. This can lead to potential routing problems."
|
315
|
+
message << " See https://guides.rubyonrails.org/routing.html#specifying-a-controller-to-use"
|
313
316
|
|
314
317
|
raise ArgumentError, message
|
315
318
|
}
|
@@ -333,7 +336,7 @@ module ActionDispatch
|
|
333
336
|
end
|
334
337
|
|
335
338
|
def split_to(to)
|
336
|
-
if to
|
339
|
+
if /#/.match?(to)
|
337
340
|
to.split("#")
|
338
341
|
else
|
339
342
|
[]
|
@@ -342,7 +345,7 @@ module ActionDispatch
|
|
342
345
|
|
343
346
|
def add_controller_module(controller, modyoule)
|
344
347
|
if modyoule && !controller.is_a?(Regexp)
|
345
|
-
if
|
348
|
+
if %r{\A/}.match?(controller)
|
346
349
|
controller[1..-1]
|
347
350
|
else
|
348
351
|
[modyoule, controller].compact.join("/")
|
@@ -553,10 +556,10 @@ module ActionDispatch
|
|
553
556
|
#
|
554
557
|
# match 'json_only', constraints: { format: 'json' }, via: :get
|
555
558
|
#
|
556
|
-
# class
|
559
|
+
# class PermitList
|
557
560
|
# def matches?(request) request.remote_ip == '1.2.3.4' end
|
558
561
|
# end
|
559
|
-
# match 'path', to: 'c#a', constraints:
|
562
|
+
# match 'path', to: 'c#a', constraints: PermitList.new, via: :get
|
560
563
|
#
|
561
564
|
# See <tt>Scoping#constraints</tt> for more examples with its scope
|
562
565
|
# equivalent.
|
@@ -611,7 +614,7 @@ module ActionDispatch
|
|
611
614
|
end
|
612
615
|
|
613
616
|
raise ArgumentError, "A rack application must be specified" unless app.respond_to?(:call)
|
614
|
-
raise ArgumentError,
|
617
|
+
raise ArgumentError, <<~MSG unless path
|
615
618
|
Must be called with mount point
|
616
619
|
|
617
620
|
mount SomeRackApp, at: "some_route"
|
@@ -644,7 +647,7 @@ module ActionDispatch
|
|
644
647
|
|
645
648
|
# Query if the following named route was already defined.
|
646
649
|
def has_named_route?(name)
|
647
|
-
@set.named_routes.key?
|
650
|
+
@set.named_routes.key?(name)
|
648
651
|
end
|
649
652
|
|
650
653
|
private
|
@@ -668,7 +671,7 @@ module ActionDispatch
|
|
668
671
|
|
669
672
|
script_namer = ->(options) do
|
670
673
|
prefix_options = options.slice(*_route.segment_keys)
|
671
|
-
prefix_options[:relative_url_root] = ""
|
674
|
+
prefix_options[:relative_url_root] = ""
|
672
675
|
|
673
676
|
if options[:_recall]
|
674
677
|
prefix_options.reverse_merge!(options[:_recall].slice(*_route.segment_keys))
|
@@ -1159,10 +1162,16 @@ module ActionDispatch
|
|
1159
1162
|
end
|
1160
1163
|
|
1161
1164
|
def actions
|
1165
|
+
if @except
|
1166
|
+
available_actions - Array(@except).map(&:to_sym)
|
1167
|
+
else
|
1168
|
+
available_actions
|
1169
|
+
end
|
1170
|
+
end
|
1171
|
+
|
1172
|
+
def available_actions
|
1162
1173
|
if @only
|
1163
1174
|
Array(@only).map(&:to_sym)
|
1164
|
-
elsif @except
|
1165
|
-
default_actions - Array(@except).map(&:to_sym)
|
1166
1175
|
else
|
1167
1176
|
default_actions
|
1168
1177
|
end
|
@@ -1588,7 +1597,7 @@ module ActionDispatch
|
|
1588
1597
|
when Symbol
|
1589
1598
|
options[:action] = to
|
1590
1599
|
when String
|
1591
|
-
if to
|
1600
|
+
if /#/.match?(to)
|
1592
1601
|
options[:to] = to
|
1593
1602
|
else
|
1594
1603
|
options[:controller] = to
|
@@ -1914,7 +1923,7 @@ module ActionDispatch
|
|
1914
1923
|
|
1915
1924
|
default_action = options.delete(:action) || @scope[:action]
|
1916
1925
|
|
1917
|
-
if
|
1926
|
+
if /^[\w\-\/]+$/.match?(action)
|
1918
1927
|
default_action ||= action.tr("-", "_") unless action.include?("/")
|
1919
1928
|
else
|
1920
1929
|
action = nil
|
@@ -1934,9 +1943,7 @@ module ActionDispatch
|
|
1934
1943
|
end
|
1935
1944
|
|
1936
1945
|
def match_root_route(options)
|
1937
|
-
|
1938
|
-
args = ["/", { as: name, via: :get }.merge!(options)]
|
1939
|
-
|
1946
|
+
args = ["/", { as: :root, via: :get }.merge(options)]
|
1940
1947
|
match(*args)
|
1941
1948
|
end
|
1942
1949
|
end
|
@@ -120,8 +120,7 @@ module ActionDispatch
|
|
120
120
|
opts
|
121
121
|
end
|
122
122
|
|
123
|
-
# Returns the path component of a URL for the given record.
|
124
|
-
# <tt>polymorphic_url</tt> with <tt>routing_type: :path</tt>.
|
123
|
+
# Returns the path component of a URL for the given record.
|
125
124
|
def polymorphic_path(record_or_hash_or_array, options = {})
|
126
125
|
if Hash === record_or_hash_or_array
|
127
126
|
options = record_or_hash_or_array.merge(options)
|
@@ -182,8 +181,8 @@ module ActionDispatch
|
|
182
181
|
CACHE[type].fetch(action) { build action, type }
|
183
182
|
end
|
184
183
|
|
185
|
-
def self.url; CACHE["url"
|
186
|
-
def self.path; CACHE["path"
|
184
|
+
def self.url; CACHE["url"][nil]; end
|
185
|
+
def self.path; CACHE["path"][nil]; end
|
187
186
|
|
188
187
|
def self.build(action, type)
|
189
188
|
prefix = action ? "#{action}_" : ""
|
@@ -288,12 +287,10 @@ module ActionDispatch
|
|
288
287
|
|
289
288
|
args = []
|
290
289
|
|
291
|
-
route = record_list.map
|
290
|
+
route = record_list.map { |parent|
|
292
291
|
case parent
|
293
|
-
when Symbol
|
292
|
+
when Symbol, String
|
294
293
|
parent.to_s
|
295
|
-
when String
|
296
|
-
raise(ArgumentError, "Please use symbols for polymorphic route arguments.")
|
297
294
|
when Class
|
298
295
|
args << parent
|
299
296
|
parent.model_name.singular_route_key
|
@@ -301,14 +298,12 @@ module ActionDispatch
|
|
301
298
|
args << parent.to_model
|
302
299
|
parent.to_model.model_name.singular_route_key
|
303
300
|
end
|
304
|
-
|
301
|
+
}
|
305
302
|
|
306
303
|
route <<
|
307
304
|
case record
|
308
|
-
when Symbol
|
305
|
+
when Symbol, String
|
309
306
|
record.to_s
|
310
|
-
when String
|
311
|
-
raise(ArgumentError, "Please use symbols for polymorphic route arguments.")
|
312
307
|
when Class
|
313
308
|
@key_strategy.call record.model_name
|
314
309
|
else
|
@@ -2,7 +2,6 @@
|
|
2
2
|
|
3
3
|
require "action_dispatch/journey"
|
4
4
|
require "active_support/core_ext/object/to_query"
|
5
|
-
require "active_support/core_ext/hash/slice"
|
6
5
|
require "active_support/core_ext/module/redefine_method"
|
7
6
|
require "active_support/core_ext/module/remove_method"
|
8
7
|
require "active_support/core_ext/array/extract_options"
|
@@ -36,7 +35,7 @@ module ActionDispatch
|
|
36
35
|
if @raise_on_name_error
|
37
36
|
raise
|
38
37
|
else
|
39
|
-
|
38
|
+
[404, { "X-Cascade" => "pass" }, []]
|
40
39
|
end
|
41
40
|
end
|
42
41
|
|
@@ -91,11 +90,11 @@ module ActionDispatch
|
|
91
90
|
|
92
91
|
def clear!
|
93
92
|
@path_helpers.each do |helper|
|
94
|
-
@path_helpers_module.
|
93
|
+
@path_helpers_module.remove_method helper
|
95
94
|
end
|
96
95
|
|
97
96
|
@url_helpers.each do |helper|
|
98
|
-
@url_helpers_module.
|
97
|
+
@url_helpers_module.remove_method helper
|
99
98
|
end
|
100
99
|
|
101
100
|
@routes.clear
|
@@ -109,8 +108,8 @@ module ActionDispatch
|
|
109
108
|
url_name = :"#{name}_url"
|
110
109
|
|
111
110
|
if routes.key? key
|
112
|
-
@path_helpers_module.
|
113
|
-
@url_helpers_module.
|
111
|
+
@path_helpers_module.undef_method path_name
|
112
|
+
@url_helpers_module.undef_method url_name
|
114
113
|
end
|
115
114
|
routes[key] = route
|
116
115
|
define_url_helper @path_helpers_module, route, path_name, route.defaults, name, PATH
|
@@ -154,13 +153,13 @@ module ActionDispatch
|
|
154
153
|
url_name = :"#{name}_url"
|
155
154
|
|
156
155
|
@path_helpers_module.module_eval do
|
157
|
-
|
156
|
+
redefine_method(path_name) do |*args|
|
158
157
|
helper.call(self, args, true)
|
159
158
|
end
|
160
159
|
end
|
161
160
|
|
162
161
|
@url_helpers_module.module_eval do
|
163
|
-
|
162
|
+
redefine_method(url_name) do |*args|
|
164
163
|
helper.call(self, args, false)
|
165
164
|
end
|
166
165
|
end
|
@@ -246,7 +245,7 @@ module ActionDispatch
|
|
246
245
|
missing_keys << missing_key
|
247
246
|
}
|
248
247
|
constraints = Hash[@route.requirements.merge(params).sort_by { |k, v| k.to_s }]
|
249
|
-
message = "No route matches #{constraints.inspect}"
|
248
|
+
message = +"No route matches #{constraints.inspect}"
|
250
249
|
message << ", missing required keys: #{missing_keys.sort.inspect}"
|
251
250
|
|
252
251
|
raise ActionController::UrlGenerationError, message
|
@@ -378,7 +377,7 @@ module ActionDispatch
|
|
378
377
|
@prepend = []
|
379
378
|
@disable_clear_and_finalize = false
|
380
379
|
@finalized = false
|
381
|
-
@env_key = "ROUTES_#{object_id}_SCRIPT_NAME"
|
380
|
+
@env_key = "ROUTES_#{object_id}_SCRIPT_NAME"
|
382
381
|
|
383
382
|
@set = Journey::Routes.new
|
384
383
|
@router = Journey::Router.new @set
|
@@ -585,7 +584,7 @@ module ActionDispatch
|
|
585
584
|
"You may have defined two routes with the same name using the `:as` option, or " \
|
586
585
|
"you may be overriding a route already defined by a resource with the same naming. " \
|
587
586
|
"For the latter, you can restrict the routes created with `resources` as explained here: \n" \
|
588
|
-
"
|
587
|
+
"https://guides.rubyonrails.org/routing.html#restricting-the-routes-created"
|
589
588
|
end
|
590
589
|
|
591
590
|
route = @set.add_route(name, mapping)
|
@@ -730,7 +729,7 @@ module ActionDispatch
|
|
730
729
|
# Remove leading slashes from controllers
|
731
730
|
def normalize_controller!
|
732
731
|
if controller
|
733
|
-
if controller.start_with?("/"
|
732
|
+
if controller.start_with?("/")
|
734
733
|
@options[:controller] = controller[1..-1]
|
735
734
|
else
|
736
735
|
@options[:controller] = controller
|
@@ -133,6 +133,7 @@ module ActionDispatch
|
|
133
133
|
# <tt>ActionDispatch::Http::URL.tld_length</tt>, which in turn defaults to 1.
|
134
134
|
# * <tt>:port</tt> - Optionally specify the port to connect to.
|
135
135
|
# * <tt>:anchor</tt> - An anchor name to be appended to the path.
|
136
|
+
# * <tt>:params</tt> - The query parameters to be appended to the path.
|
136
137
|
# * <tt>:trailing_slash</tt> - If true, adds a trailing slash, as in "/archive/2009/"
|
137
138
|
# * <tt>:script_name</tt> - Specifies application path relative to domain root. If provided, prepends application path.
|
138
139
|
#
|
@@ -243,8 +243,9 @@ module ActionDispatch
|
|
243
243
|
#
|
244
244
|
# rails routes
|
245
245
|
#
|
246
|
-
# Target specific
|
247
|
-
#
|
246
|
+
# Target a specific controller with <tt>-c</tt>, or grep routes
|
247
|
+
# using <tt>-g</tt>. Useful in conjunction with <tt>--expanded</tt>
|
248
|
+
# which displays routes vertically.
|
248
249
|
module Routing
|
249
250
|
extend ActiveSupport::Autoload
|
250
251
|
|
@@ -43,7 +43,7 @@ module ActionDispatch
|
|
43
43
|
end
|
44
44
|
|
45
45
|
def image_path
|
46
|
-
@image_path ||= absolute_image_path.
|
46
|
+
@image_path ||= absolute_image_path.to_s
|
47
47
|
end
|
48
48
|
|
49
49
|
def absolute_image_path
|
@@ -65,7 +65,7 @@ module ActionDispatch
|
|
65
65
|
end
|
66
66
|
|
67
67
|
def display_image
|
68
|
-
message = "[Screenshot]: #{image_path}\n"
|
68
|
+
message = +"[Screenshot]: #{image_path}\n"
|
69
69
|
|
70
70
|
case output_type
|
71
71
|
when "artifact"
|
@@ -80,7 +80,7 @@ module ActionDispatch
|
|
80
80
|
end
|
81
81
|
|
82
82
|
def inline_base64(path)
|
83
|
-
Base64.
|
83
|
+
Base64.strict_encode64(path)
|
84
84
|
end
|
85
85
|
|
86
86
|
def failed?
|
@@ -79,9 +79,8 @@ module ActionDispatch
|
|
79
79
|
end
|
80
80
|
|
81
81
|
def generate_response_message(expected, actual = @response.response_code)
|
82
|
-
"Expected response to be a <#{code_with_name(expected)}>,"\
|
83
|
-
" but was a <#{code_with_name(actual)}>"
|
84
|
-
.dup.concat(location_if_redirected).concat(response_body_if_short)
|
82
|
+
(+"Expected response to be a <#{code_with_name(expected)}>,"\
|
83
|
+
" but was a <#{code_with_name(actual)}>").concat(location_if_redirected).concat(response_body_if_short)
|
85
84
|
end
|
86
85
|
|
87
86
|
def response_body_if_short
|
@@ -9,6 +9,11 @@ module ActionDispatch
|
|
9
9
|
module Assertions
|
10
10
|
# Suite of assertions to test routes generated by \Rails and the handling of requests made to them.
|
11
11
|
module RoutingAssertions
|
12
|
+
def setup # :nodoc:
|
13
|
+
@routes ||= nil
|
14
|
+
super
|
15
|
+
end
|
16
|
+
|
12
17
|
# Asserts that the routing of the given +path+ was handled correctly and that the parsed options (given in the +expected_options+ hash)
|
13
18
|
# match +path+. Basically, it asserts that \Rails recognizes the route given by +expected_options+.
|
14
19
|
#
|
@@ -78,7 +83,7 @@ module ActionDispatch
|
|
78
83
|
# # Asserts that the generated route gives us our custom route
|
79
84
|
# assert_generates "changesets/12", { controller: 'scm', action: 'show_diff', revision: "12" }
|
80
85
|
def assert_generates(expected_path, options, defaults = {}, extras = {}, message = nil)
|
81
|
-
if
|
86
|
+
if %r{://}.match?(expected_path)
|
82
87
|
fail_on(URI::InvalidURIError, message) do
|
83
88
|
uri = URI.parse(expected_path)
|
84
89
|
expected_path = uri.path.to_s.empty? ? "/" : uri.path
|
@@ -189,7 +194,7 @@ module ActionDispatch
|
|
189
194
|
|
190
195
|
request = ActionController::TestRequest.create @controller.class
|
191
196
|
|
192
|
-
if
|
197
|
+
if %r{://}.match?(path)
|
193
198
|
fail_on(URI::InvalidURIError, msg) do
|
194
199
|
uri = URI.parse(path)
|
195
200
|
request.env["rack.url_scheme"] = uri.scheme || "http"
|
@@ -50,10 +50,11 @@ module ActionDispatch
|
|
50
50
|
|
51
51
|
# Follow a single redirect response. If the last response was not a
|
52
52
|
# redirect, an exception will be raised. Otherwise, the redirect is
|
53
|
-
# performed on the location header.
|
54
|
-
|
53
|
+
# performed on the location header. Any arguments are passed to the
|
54
|
+
# underlying call to `get`.
|
55
|
+
def follow_redirect!(**args)
|
55
56
|
raise "not a redirect! #{status} #{status_message}" unless redirect?
|
56
|
-
get(response.location)
|
57
|
+
get(response.location, **args)
|
57
58
|
status
|
58
59
|
end
|
59
60
|
end
|
@@ -189,6 +190,12 @@ module ActionDispatch
|
|
189
190
|
# merged into the Rack env hash.
|
190
191
|
# - +env+: Additional env to pass, as a Hash. The headers will be
|
191
192
|
# merged into the Rack env hash.
|
193
|
+
# - +xhr+: Set to `true` if you want to make and Ajax request.
|
194
|
+
# Adds request headers characteristic of XMLHttpRequest e.g. HTTP_X_REQUESTED_WITH.
|
195
|
+
# The headers will be merged into the Rack env hash.
|
196
|
+
# - +as+: Used for encoding the request with different content type.
|
197
|
+
# Supports `:json` by default and will set the approriate request headers.
|
198
|
+
# The headers will be merged into the Rack env hash.
|
192
199
|
#
|
193
200
|
# This method is rarely used directly. Use +#get+, +#post+, or other standard
|
194
201
|
# HTTP methods in integration tests. +#process+ is only required when using a
|
@@ -210,7 +217,7 @@ module ActionDispatch
|
|
210
217
|
method = :post
|
211
218
|
end
|
212
219
|
|
213
|
-
if
|
220
|
+
if %r{://}.match?(path)
|
214
221
|
path = build_expanded_path(path) do |location|
|
215
222
|
https! URI::HTTPS === location if location.scheme
|
216
223
|
|
@@ -401,7 +408,6 @@ module ActionDispatch
|
|
401
408
|
super
|
402
409
|
end
|
403
410
|
end
|
404
|
-
ruby2_keywords(:method_missing) if respond_to?(:ruby2_keywords, true)
|
405
411
|
end
|
406
412
|
end
|
407
413
|
|
@@ -8,12 +8,12 @@ module ActionDispatch
|
|
8
8
|
module FixtureFile
|
9
9
|
# Shortcut for <tt>Rack::Test::UploadedFile.new(File.join(ActionDispatch::IntegrationTest.fixture_path, path), type)</tt>:
|
10
10
|
#
|
11
|
-
# post :change_avatar, avatar: fixture_file_upload('files/spongebob.png', 'image/png')
|
11
|
+
# post :change_avatar, params: { avatar: fixture_file_upload('files/spongebob.png', 'image/png') }
|
12
12
|
#
|
13
13
|
# To upload binary files on Windows, pass <tt>:binary</tt> as the last parameter.
|
14
14
|
# This will not affect other platforms:
|
15
15
|
#
|
16
|
-
# post :change_avatar, avatar: fixture_file_upload('files/spongebob.png', 'image/png', :binary)
|
16
|
+
# post :change_avatar, params: { avatar: fixture_file_upload('files/spongebob.png', 'image/png', :binary) }
|
17
17
|
def fixture_file_upload(path, mime_type = nil, binary = false)
|
18
18
|
if self.class.respond_to?(:fixture_path) && self.class.fixture_path &&
|
19
19
|
!File.exist?(path)
|