Gitignoring
[catagits/Web-Simple.git] / lib / Web / Dispatch / Predicates.pm
CommitLineData
d63bcdae 1package Web::Dispatch::Predicates;
2
3use strictures 1;
4use base qw(Exporter);
5
5ba2eb68 6our @EXPORT = qw(
ce573717 7 match_and match_or match_not match_method match_path match_path_strip
bc878dde 8 match_extension match_query match_body match_uploads match_true match_false
5ba2eb68 9);
d63bcdae 10
456dc2bb 11sub _matcher { bless shift, 'Web::Dispatch::Matcher' }
481da1e2 12
bc878dde 13sub match_true {
14 _matcher(sub { {} });
15}
16
17sub match_false {
18 _matcher(sub {});
19}
20
d63bcdae 21sub match_and {
22 my @match = @_;
456dc2bb 23 _matcher(sub {
d63bcdae 24 my ($env) = @_;
d96756e8 25 my $my_env = { 'Web::Dispatch.original_env' => $env, %$env };
d63bcdae 26 my $new_env;
27 my @got;
28 foreach my $match (@match) {
29 if (my @this_got = $match->($my_env)) {
4ed4fb42 30 my %change_env = %{shift(@this_got)};
31 @{$my_env}{keys %change_env} = values %change_env;
32 @{$new_env}{keys %change_env} = values %change_env;
33 push @got, @this_got;
d63bcdae 34 } else {
4ed4fb42 35 return;
d63bcdae 36 }
37 }
38 return ($new_env, @got);
481da1e2 39 })
d63bcdae 40}
41
42sub match_or {
43 my @match = @_;
456dc2bb 44 _matcher(sub {
d63bcdae 45 foreach my $try (@match) {
46 if (my @ret = $try->(@_)) {
47 return @ret;
48 }
49 }
50 return;
481da1e2 51 })
d63bcdae 52}
53
ce573717 54sub match_not {
55 my ($match) = @_;
456dc2bb 56 _matcher(sub {
ce573717 57 if (my @discard = $match->($_[0])) {
58 ();
59 } else {
60 ({});
61 }
481da1e2 62 })
ce573717 63}
64
d63bcdae 65sub match_method {
66 my ($method) = @_;
456dc2bb 67 _matcher(sub {
d63bcdae 68 my ($env) = @_;
69 $env->{REQUEST_METHOD} eq $method ? {} : ()
481da1e2 70 })
d63bcdae 71}
72
73sub match_path {
74 my ($re) = @_;
456dc2bb 75 _matcher(sub {
d63bcdae 76 my ($env) = @_;
77 if (my @cap = ($env->{PATH_INFO} =~ /$re/)) {
78 $cap[0] = {}; return @cap;
79 }
80 return;
481da1e2 81 })
d63bcdae 82}
83
84sub match_path_strip {
85 my ($re) = @_;
456dc2bb 86 _matcher(sub {
d63bcdae 87 my ($env) = @_;
88 if (my @cap = ($env->{PATH_INFO} =~ /$re/)) {
89 $cap[0] = {
4ed4fb42 90 SCRIPT_NAME => ($env->{SCRIPT_NAME}||'').$cap[0],
91 PATH_INFO => pop(@cap),
d63bcdae 92 };
93 return @cap;
94 }
95 return;
481da1e2 96 })
d63bcdae 97}
98
5ba2eb68 99sub match_extension {
100 my ($extension) = @_;
101 my $wild = (!$extension or $extension eq '*');
102 my $re = $wild
103 ? qr/\.(\w+)$/
104 : qr/\.(\Q${extension}\E)$/;
456dc2bb 105 _matcher(sub {
5ba2eb68 106 if ($_[0]->{PATH_INFO} =~ $re) {
107 ($wild ? ({}, $1) : {});
108 } else {
109 ();
110 }
481da1e2 111 });
5ba2eb68 112}
113
b6bf9ed3 114sub match_query {
456dc2bb 115 _matcher(_param_matcher(query => $_[0]));
b6bf9ed3 116}
117
118sub match_body {
456dc2bb 119 _matcher(_param_matcher(body => $_[0]));
05aafc1a 120}
121
122sub match_uploads {
456dc2bb 123 _matcher(_param_matcher(uploads => $_[0]));
05aafc1a 124}
125
126sub _param_matcher {
127 my ($type, $spec) = @_;
b6bf9ed3 128 require Web::Dispatch::ParamParser;
05aafc1a 129 my $unpack = Web::Dispatch::ParamParser->can("get_unpacked_${type}_from");
b6bf9ed3 130 sub {
05aafc1a 131 _extract_params($unpack->($_[0]), $spec)
b6bf9ed3 132 };
133}
134
052bdd54 135sub _extract_params {
136 my ($raw, $spec) = @_;
137 foreach my $name (@{$spec->{required}||[]}) {
138 return unless exists $raw->{$name};
139 }
140 my @ret = (
141 {},
142 map {
143 $_->{multi} ? $raw->{$_->{name}}||[] : $raw->{$_->{name}}->[-1]
144 } @{$spec->{positional}||[]}
145 );
146 # separated since 'or' is short circuit
147 my ($named, $star) = ($spec->{named}, $spec->{star});
148 if ($named or $star) {
149 my %kw;
150 if ($star) {
151 @kw{keys %$raw} = (
152 $star->{multi}
153 ? values %$raw
154 : map $_->[-1], values %$raw
155 );
156 }
157 foreach my $n (@{$named||[]}) {
158 next if !$n->{multi} and !exists $raw->{$n->{name}};
159 $kw{$n->{name}} =
160 $n->{multi} ? $raw->{$n->{name}}||[] : $raw->{$n->{name}}->[-1];
161 }
162 push @ret, \%kw;
163 }
164 @ret;
165}
166
d63bcdae 1671;