-package Web::Simple::DispatchParser;
-
-use strict;
-use warnings FATAL => 'all';
+package Web::Dispatch::Parser;
sub DEBUG () { 0 }
BEGIN {
- if ($ENV{WEB_SIMPLE_DISPATCHPARSER_DEBUG}) {
+ if ($ENV{WEB_DISPATCH_PARSER_DEBUG}) {
no warnings 'redefine';
*DEBUG = sub () { 1 }
}
}
-sub diag { if (DEBUG) { warn $_[0] } }
+use Sub::Quote;
+use Web::Dispatch::Predicates;
+use Moo;
-sub new { bless({}, ref($_[0])||$_[0]) }
+has _cache => (
+ is => 'lazy', default => quote_sub q{ {} }
+);
+
+sub diag { if (DEBUG) { warn $_[0] } }
sub _blam {
my ($self, $error) = @_;
${hat} here\n";
}
-sub parse_dispatch_specification {
+sub parse {
my ($self, $spec) = @_;
- return $self->_parse_spec($spec);
+ return $self->_cache->{$spec} ||= $self->_parse_spec($spec);
}
sub _parse_spec {
$self->_blam("No closing ) found for opening (");
}
return $match[0] if (@match == 1);
- return sub {
- my $env = { %{$_[0]} };
- my $new_env;
- my @got;
- foreach my $match (@match) {
- if (my @this_got = $match->($env)) {
- my %change_env = %{shift(@this_got)};
- @{$env}{keys %change_env} = values %change_env;
- @{$new_env}{keys %change_env} = values %change_env;
- push @got, @this_got;
- } else {
- return;
- }
- }
- return ($new_env, @got);
- };
+ return match_and(@match);
}
}
last PARSE if (pos == length);
last PARSE unless /\G\|/gc; # give up when next thing isn't |
} until (pos == length) }; # accept trailing whitespace
- return sub {
- foreach my $try (@match) {
- if (my @ret = $try->(@_)) {
- return @ret;
- }
- }
- return;
- };
+ return match_or(@match);
};
}
return;
sub _http_method_match {
my ($self, $str, $method) = @_;
- sub { shift->{REQUEST_METHOD} eq $method ? {} : () };
+ match_method($method);
}
sub _url_path_match {
push @path, $self->_url_path_segment_match($_)
or $self->_blam("Couldn't parse path match segment");
}
- my $re = '^()'.join('/','',@path).($full_path ? '$' : '(/.*)$');
+ my $re = '^('.join('/','',@path).')'.($full_path ? '$' : '(/.*)$');
$re = qr/$re/;
if ($full_path) {
- return sub {
- if (my @cap = (shift->{PATH_INFO} =~ /$re/)) {
- $cap[0] = {}; return @cap;
- }
- return ();
- };
+ return match_path($re);
}
- return sub {
- if (my @cap = (shift->{PATH_INFO} =~ /$re/)) {
- $cap[0] = { PATH_INFO => pop(@cap) }; return @cap;
- }
- return ();
- };
+ return match_path_strip($re);
}
return;
}
use Test::More qw(no_plan);
-use Web::Simple::DispatchParser;
+use Web::Dispatch::Parser;
-my $dp = Web::Simple::DispatchParser->new;
+my $dp = Web::Dispatch::Parser->new;
{
- my $get = $dp->parse_dispatch_specification('GET');
+ my $get = $dp->parse('GET');
is_deeply(
[ $get->({ REQUEST_METHOD => 'GET' }) ],
}
ok(
- !eval { $dp->parse_dispatch_specification('GET POST'); 1; },
+ !eval { $dp->parse('GET POST'); 1; },
"Don't yet allow two methods"
);
{
- my $html = $dp->parse_dispatch_specification('.html');
+ my $html = $dp->parse('.html');
is_deeply(
[ $html->({ PATH_INFO => '/foo/bar.html' }) ],
}
{
- my $any_ext = $dp->parse_dispatch_specification('.*');
+ my $any_ext = $dp->parse('.*');
is_deeply(
[ $any_ext->({ PATH_INFO => '/foo/bar.html' }) ],
}
{
- my $slash = $dp->parse_dispatch_specification('/');
+ my $slash = $dp->parse('/');
is_deeply(
[ $slash->({ PATH_INFO => '/' }) ],
}
{
- my $post = $dp->parse_dispatch_specification('/post/*');
+ my $post = $dp->parse('/post/*');
is_deeply(
[ $post->({ PATH_INFO => '/post/one' }) ],
}
{
- my $combi = $dp->parse_dispatch_specification('GET+/post/*');
+ my $combi = $dp->parse('GET+/post/*');
is_deeply(
[ $combi->({ PATH_INFO => '/post/one', REQUEST_METHOD => 'GET' }) ],
}
{
- my $or = $dp->parse_dispatch_specification('GET|POST');
+ my $or = $dp->parse('GET|POST');
foreach my $meth (qw(GET POST)) {
}
{
- my $or = $dp->parse_dispatch_specification('GET|POST|DELETE');
+ my $or = $dp->parse('GET|POST|DELETE');
foreach my $meth (qw(GET POST DELETE)) {
}
{
- my $nest = $dp->parse_dispatch_specification('(GET+/foo)|POST');
+ my $nest = $dp->parse('(GET+/foo)|POST');
is_deeply(
[ $nest->({ PATH_INFO => '/foo', REQUEST_METHOD => 'GET' }) ],
{
local $@;
ok(
- !eval { $dp->parse_dispatch_specification('/foo+(GET'); 1 },
+ !eval { $dp->parse('/foo+(GET'); 1 },
'Death with missing closing )'
);
my $err = q{
}
{
- my $not = $dp->parse_dispatch_specification('!.html+.*');
+ my $not = $dp->parse('!.html+.*');
is_deeply(
[ $not->({ PATH_INFO => '/foo.xml' }) ],
}
{
- my $sub = $dp->parse_dispatch_specification('/foo/*/...');
+ my $sub = $dp->parse('/foo/*/...');
is_deeply(
[ $sub->({ PATH_INFO => '/foo/1/bar' }) ],
- [ { PATH_INFO => '/bar' }, 1 ],
+ [ { PATH_INFO => '/bar', SCRIPT_NAME => '/foo/1' }, 1 ],
'/foo/*/... matches /foo/1/bar and strips to /bar'
);
is_deeply(
[ $sub->({ PATH_INFO => '/foo/1/' }) ],
- [ { PATH_INFO => '/' }, 1 ],
+ [ { PATH_INFO => '/', SCRIPT_NAME => '/foo/1' }, 1 ],
'/foo/*/... matches /foo/1/bar and strips to /'
);
);
foreach my $lose ('?foo=','?:foo=','?@foo=','?:@foo=') {
- my $foo = $dp->parse_dispatch_specification($lose);
+ my $foo = $dp->parse($lose);
is_deeply(
[ $foo->({ QUERY_STRING => '' }) ],
[ '?:@bar=&*' => { %all_single, bar => [ qw(BAR1 BAR2) ] } ],
) {
my ($spec, @res) = @$win;
- my $match = $dp->parse_dispatch_specification($spec);
+ my $match = $dp->parse($spec);
#use Data::Dump::Streamer; warn Dump($match);
is_deeply(
[ $match->({ QUERY_STRING => $q }) ],
#
foreach my $lose2 ('/foo/bar/+?foo=','/foo/bar/+?:foo=','/foo/bar/+?@foo=','/foo/bar/+?:@foo=') {
- my $foo = $dp->parse_dispatch_specification($lose2);
+ my $foo = $dp->parse($lose2);
is_deeply(
[ $foo->({ PATH_INFO => '/foo/bar/', QUERY_STRING => '' }) ],
[ '/foo/bar/+?:@bar=&*' => { %all_single, bar => [ qw(BAR1 BAR2) ] } ],
) {
my ($spec, @res) = @$win2;
- my $match = $dp->parse_dispatch_specification($spec);
+ my $match = $dp->parse($spec);
# use Data::Dump::Streamer; warn Dump($match);
is_deeply(
[ $match->({ PATH_INFO => '/foo/bar/', QUERY_STRING => $q }) ],
#
foreach my $lose3 ('/foo/bar+?foo=','/foo/bar+?:foo=','/foo/bar+?@foo=','/foo/bar+?:@foo=') {
- my $foo = $dp->parse_dispatch_specification($lose3);
+ my $foo = $dp->parse($lose3);
is_deeply(
[ $foo->({ PATH_INFO => '/foo/bar', QUERY_STRING => '' }) ],
[ '/foo/bar+?:@bar=&*' => { %all_single, bar => [ qw(BAR1 BAR2) ] } ],
) {
my ($spec, @res) = @$win3;
- my $match = $dp->parse_dispatch_specification($spec);
+ my $match = $dp->parse($spec);
# use Data::Dump::Streamer; warn Dump($match);
is_deeply(
[ $match->({ PATH_INFO => '/foo/bar', QUERY_STRING => $q }) ],