1
0
Fork 0
mirror of https://github.com/rails/rails.git synced 2022-11-09 12:12:34 -05:00
rails--rails/actionpack/test/journey/path/pattern_test.rb
Daniel Colson 8eec6ee962
Move Path::Pattern factories into test helper
`Pattern#from_sting` was introduced in bb207ea7 to support creating
patterns from strings in tests (removing a conditional in the initialize
method that had been there only for the sake of those tests).

`Pattern#build` was introduced in 947ebe9a, and was similarly used only
in tests.

My understanding is that [`Mapping#path`][path] is the only place where
we initialize a `Path::Pattern` in library code.

Since these two methods appear to be used only in tests, this
commit moves them out of the class and into a test helper.

My reasoning for doing this is that I am doing some performance work
that may involve a modification to how `Path::Pattern`s get initialized,
and I will have more confidence in that work if I know for sure that
these methods are test helpers that can be modified freely.

[path]: https://github.com/rails/rails/blob/master/actionpack/lib/action_dispatch/routing/mapper.rb#L192-L194
2020-07-27 12:23:33 -04:00

329 lines
11 KiB
Ruby

# frozen_string_literal: true
require "abstract_unit"
require "support/path_helper"
module ActionDispatch
module Journey
module Path
class TestPattern < ActiveSupport::TestCase
include PathHelper
SEPARATORS = ["/", ".", "?"].join
x = /.+/
{
"/:controller(/:action)" => %r{\A/(#{x})(?:/([^/.?]+))?\Z},
"/:controller/foo" => %r{\A/(#{x})/foo\Z},
"/:controller/:action" => %r{\A/(#{x})/([^/.?]+)\Z},
"/:controller" => %r{\A/(#{x})\Z},
"/:controller(/:action(/:id))" => %r{\A/(#{x})(?:/([^/.?]+)(?:/([^/.?]+))?)?\Z},
"/:controller/:action.xml" => %r{\A/(#{x})/([^/.?]+)\.xml\Z},
"/:controller.:format" => %r{\A/(#{x})\.([^/.?]+)\Z},
"/:controller(.:format)" => %r{\A/(#{x})(?:\.([^/.?]+))?\Z},
"/:controller/*foo" => %r{\A/(#{x})/(.+)\Z},
"/:controller/*foo/bar" => %r{\A/(#{x})/(.+)/bar\Z},
"/:foo|*bar" => %r{\A/(?:([^/.?]+)|(.+))\Z},
}.each do |path, expected|
define_method(:"test_to_regexp_#{Regexp.escape(path)}") do
path = build_path(
path,
{ controller: /.+/ },
SEPARATORS,
true
)
assert_equal(expected, path.to_regexp)
end
end
{
"/:controller(/:action)" => %r{\A/(#{x})(?:/([^/.?]+))?(?:\b|\Z|/)},
"/:controller/foo" => %r{\A/(#{x})/foo(?:\b|\Z|/)},
"/:controller/:action" => %r{\A/(#{x})/([^/.?]+)(?:\b|\Z|/)},
"/:controller" => %r{\A/(#{x})(?:\b|\Z|/)},
"/:controller(/:action(/:id))" => %r{\A/(#{x})(?:/([^/.?]+)(?:/([^/.?]+))?)?(?:\b|\Z|/)},
"/:controller/:action.xml" => %r{\A/(#{x})/([^/.?]+)\.xml(?:\b|\Z|/)},
"/:controller.:format" => %r{\A/(#{x})\.([^/.?]+)(?:\b|\Z|/)},
"/:controller(.:format)" => %r{\A/(#{x})(?:\.([^/.?]+))?(?:\b|\Z|/)},
"/:controller/*foo" => %r{\A/(#{x})/(.+)(?:\b|\Z|/)},
"/:controller/*foo/bar" => %r{\A/(#{x})/(.+)/bar(?:\b|\Z|/)},
"/:foo|*bar" => %r{\A/(?:([^/.?]+)|(.+))(?:\b|\Z|/)},
}.each do |path, expected|
define_method(:"test_to_non_anchored_regexp_#{Regexp.escape(path)}") do
path = build_path(
path,
{ controller: /.+/ },
SEPARATORS,
false
)
assert_equal(expected, path.to_regexp)
end
end
{
"/:controller(/:action)" => %w{ controller action },
"/:controller/foo" => %w{ controller },
"/:controller/:action" => %w{ controller action },
"/:controller" => %w{ controller },
"/:controller(/:action(/:id))" => %w{ controller action id },
"/:controller/:action.xml" => %w{ controller action },
"/:controller.:format" => %w{ controller format },
"/:controller(.:format)" => %w{ controller format },
"/:controller/*foo" => %w{ controller foo },
"/:controller/*foo/bar" => %w{ controller foo },
}.each do |path, expected|
define_method(:"test_names_#{Regexp.escape(path)}") do
path = build_path(
path,
{ controller: /.+/ },
SEPARATORS,
true
)
assert_equal(expected, path.names)
end
end
def test_to_regexp_with_extended_group
path = build_path(
"/page/:name",
{ name: /
#ROFL
(tender|love
#MAO
)/x },
SEPARATORS,
true
)
assert_match(path, "/page/tender")
assert_match(path, "/page/love")
assert_no_match(path, "/page/loving")
end
def test_optional_names
[
["/:foo(/:bar(/:baz))", %w{ bar baz }],
["/:foo(/:bar)", %w{ bar }],
["/:foo(/:bar)/:lol(/:baz)", %w{ bar baz }],
].each do |pattern, list|
path = path_from_string pattern
assert_equal list.sort, path.optional_names.sort
end
end
def test_to_regexp_match_non_optional
path = build_path(
"/:name",
{ name: /\d+/ },
SEPARATORS,
true
)
assert_match(path, "/123")
assert_no_match(path, "/")
end
def test_to_regexp_with_group
path = build_path(
"/page/:name",
{ name: /(tender|love)/ },
SEPARATORS,
true
)
assert_match(path, "/page/tender")
assert_match(path, "/page/love")
assert_no_match(path, "/page/loving")
end
def test_ast_sets_regular_expressions
requirements = { name: /(tender|love)/, value: /./ }
path = build_path(
"/page/:name/:value",
requirements,
SEPARATORS,
true
)
nodes = path.ast.grep(Nodes::Symbol)
assert_equal 2, nodes.length
nodes.each do |node|
assert_equal requirements[node.to_sym], node.regexp
end
end
def test_match_data_with_group
path = build_path(
"/page/:name",
{ name: /(tender|love)/ },
SEPARATORS,
true
)
match = path.match "/page/tender"
assert_equal "tender", match[1]
assert_equal 2, match.length
end
def test_match_data_with_multi_group
path = build_path(
"/page/:name/:id",
{ name: /t(((ender|love)))()/ },
SEPARATORS,
true
)
match = path.match "/page/tender/10"
assert_equal "tender", match[1]
assert_equal "10", match[2]
assert_equal 3, match.length
assert_equal %w{ tender 10 }, match.captures
end
def test_star_with_custom_re
z = /\d+/
path = build_path(
"/page/*foo",
{ foo: z },
SEPARATORS,
true
)
assert_equal(%r{\A/page/(#{z})\Z}, path.to_regexp)
end
def test_insensitive_regexp_with_group
path = build_path(
"/page/:name/aaron",
{ name: /(tender|love)/i },
SEPARATORS,
true
)
assert_match(path, "/page/TENDER/aaron")
assert_match(path, "/page/loVE/aaron")
assert_no_match(path, "/page/loVE/AAron")
end
def test_to_regexp_with_strexp
path = build_path("/:controller", {}, SEPARATORS, true)
x = %r{\A/([^/.?]+)\Z}
assert_equal(x.source, path.source)
end
def test_to_regexp_defaults
path = path_from_string "/:controller(/:action(/:id))"
expected = %r{\A/([^/.?]+)(?:/([^/.?]+)(?:/([^/.?]+))?)?\Z}
assert_equal expected, path.to_regexp
end
def test_failed_match
path = path_from_string "/:controller(/:action(/:id(.:format)))"
uri = "content"
assert_not path =~ uri
end
def test_match_controller
path = path_from_string "/:controller(/:action(/:id(.:format)))"
uri = "/content"
match = path =~ uri
assert_equal %w{ controller action id format }, match.names
assert_equal "content", match[1]
assert_nil match[2]
assert_nil match[3]
assert_nil match[4]
end
def test_match_controller_action
path = path_from_string "/:controller(/:action(/:id(.:format)))"
uri = "/content/list"
match = path =~ uri
assert_equal %w{ controller action id format }, match.names
assert_equal "content", match[1]
assert_equal "list", match[2]
assert_nil match[3]
assert_nil match[4]
end
def test_match_controller_action_id
path = path_from_string "/:controller(/:action(/:id(.:format)))"
uri = "/content/list/10"
match = path =~ uri
assert_equal %w{ controller action id format }, match.names
assert_equal "content", match[1]
assert_equal "list", match[2]
assert_equal "10", match[3]
assert_nil match[4]
end
def test_match_literal
path = path_from_string "/books(/:action(.:format))"
uri = "/books"
match = path =~ uri
assert_equal %w{ action format }, match.names
assert_nil match[1]
assert_nil match[2]
end
def test_match_literal_with_action
path = path_from_string "/books(/:action(.:format))"
uri = "/books/list"
match = path =~ uri
assert_equal %w{ action format }, match.names
assert_equal "list", match[1]
assert_nil match[2]
end
def test_match_literal_with_action_and_format
path = path_from_string "/books(/:action(.:format))"
uri = "/books/list.rss"
match = path =~ uri
assert_equal %w{ action format }, match.names
assert_equal "list", match[1]
assert_equal "rss", match[2]
end
def test_named_captures
path = path_from_string "/books(/:action(.:format))"
uri = "/books/list.rss"
match = path =~ uri
named_captures = { "action" => "list", "format" => "rss" }
assert_equal named_captures, match.named_captures
end
def test_requirements_for_missing_keys_check
name_regex = /test/
path = build_path(
"/page/:name",
{ name: name_regex },
SEPARATORS,
true
)
transformed_regex = path.requirements_for_missing_keys_check[:name]
assert_not_nil transformed_regex
assert_equal(transformed_regex, /\A#{name_regex}\Z/)
end
def test_requirements_for_missing_keys_check_memoization
name_regex = /test/
path = build_path(
"/page/:name",
{ name: name_regex },
SEPARATORS,
true
)
first_call = path.requirements_for_missing_keys_check[:name]
second_call = path.requirements_for_missing_keys_check[:name]
assert_equal(first_call.object_id, second_call.object_id)
end
end
end
end
end