new match and match captutres for http methods, plus tests, docs
[catagits/Catalyst-Runtime.git] / t / lib / TestApp / Controller / Action / Chained.pm
CommitLineData
5882c86e 1package TestApp::Controller::Action::Chained;
2
3use strict;
4use warnings;
5
372b0c1b 6use HTML::Entities;
7
ca5d34df 8use base qw/Catalyst::Controller/;
5882c86e 9
10sub begin :Private { }
11
12#
13# TODO
0e853a7f 14# :Chained('') means what?
5882c86e 15#
16
17#
18# Simple parent/child action test
19#
62605319 20sub foo :PathPart('chained/foo') :CaptureArgs(1) :Chained('/') {
21 my ( $self, $c, @args ) = @_;
22 die "missing argument" unless @args;
760d121e 23 die "more than 1 argument: got @args" if @args > 1;
62605319 24}
5882c86e 25sub endpoint :PathPart('end') :Chained('/action/chained/foo') :Args(1) { }
26
27#
28# Parent/child test with two args each
29#
1c34f703 30sub foo2 :PathPart('chained/foo2') :CaptureArgs(2) :Chained('/') { }
5882c86e 31sub endpoint2 :PathPart('end2') :Chained('/action/chained/foo2') :Args(2) { }
32
33#
34# Relative specification of parent action
35#
1c34f703 36sub bar :PathPart('chained/bar') :Chained('/') :CaptureArgs(0) { }
5882c86e 37sub finale :PathPart('') :Chained('bar') :Args { }
38
39#
40# three chain with concurrent endpoints
41#
1c34f703 42sub one :PathPart('chained/one') :Chained('/') :CaptureArgs(1) { }
43sub two :PathPart('two') :Chained('/action/chained/one') :CaptureArgs(2) { }
5882c86e 44sub three_end :PathPart('three') :Chained('two') :Args(3) { }
45sub one_end :PathPart('chained/one') :Chained('/') :Args(1) { }
46sub two_end :PathPart('two') :Chained('one') :Args(2) { }
47
48#
49# Dispatch on number of arguments
50#
51sub multi1 :PathPart('chained/multi') :Chained('/') :Args(1) { }
52sub multi2 :PathPart('chained/multi') :Chained('/') :Args(2) { }
53
54#
55# Roots in an action defined in a higher controller
56#
57sub higher_root :PathPart('bar') :Chained('/action/chained/foo/higher_root') :Args(1) { }
58
59#
60# Controller -> subcontroller -> controller
61#
1c34f703 62sub pcp1 :PathPart('chained/pcp1') :Chained('/') :CaptureArgs(1) { }
5882c86e 63sub pcp3 :Chained('/action/chained/foo/pcp2') :Args(1) { }
64
65#
66# Dispatch on capture number
67#
1c34f703 68sub multi_cap1 :PathPart('chained/multi_cap') :Chained('/') :CaptureArgs(1) { }
69sub multi_cap2 :PathPart('chained/multi_cap') :Chained('/') :CaptureArgs(2) { }
5882c86e 70sub multi_cap_end1 :PathPart('baz') :Chained('multi_cap1') :Args(0) { }
71sub multi_cap_end2 :PathPart('baz') :Chained('multi_cap2') :Args(0) { }
72
73#
74# Priority: Slurpy args vs. chained actions
75#
76sub priority_a1 :PathPart('chained/priority_a') :Chained('/') :Args { }
1c34f703 77sub priority_a2 :PathPart('chained/priority_a') :Chained('/') :CaptureArgs(1) { }
5882c86e 78sub priority_a2_end :PathPart('end') :Chained('priority_a2') :Args(1) { }
79
b7ce908f 80
5882c86e 81#
82# Priority: Fixed args vs. chained actions
83#
84sub priority_b1 :PathPart('chained/priority_b') :Chained('/') :Args(3) { }
1c34f703 85sub priority_b2 :PathPart('chained/priority_b') :Chained('/') :CaptureArgs(1) { }
5882c86e 86sub priority_b2_end :PathPart('end') :Chained('priority_b2') :Args(1) { }
87
88#
b7ce908f 89# Priority: With no Args()
90#
91sub priority_c1 :PathPart('chained/priority_c') :Chained('/') :CaptureArgs(1) { }
92sub priority_c2 :PathPart('') :Chained('priority_c1') { }
93sub priority_c2_xyz :PathPart('xyz') :Chained('priority_c1') { }
94
95
96#
5882c86e 97# Optional specification of :Args in endpoint
98#
99sub opt_args :PathPart('chained/opt_args') :Chained('/') { }
100
101#
102# Optional PathPart test -> /chained/optpp/*/opt_pathpart/*
103#
1c34f703 104sub opt_pp_start :Chained('/') :PathPart('chained/optpp') :CaptureArgs(1) { }
5882c86e 105sub opt_pathpart :Chained('opt_pp_start') :Args(1) { }
106
107#
108# Optional Args *and* PathPart -> /chained/optall/*/oa/...
109#
1c34f703 110sub opt_all_start :Chained('/') :PathPart('chained/optall') :CaptureArgs(1) { }
5882c86e 111sub oa :Chained('opt_all_start') { }
112
0e853a7f 113#
114# :Chained is the same as :Chained('/')
115#
116sub rootdef :Chained :PathPart('chained/rootdef') :Args(1) { }
117
118#
119# the ParentChain controller chains to this action by
120# specifying :Chained('.')
121#
1c34f703 122sub parentchain :Chained('/') :PathPart('chained/parentchain') :CaptureArgs(1) { }
0e853a7f 123
2349aeea 124#
125# This is just for a test that a loose end is not callable
126#
1c34f703 127sub loose :Chained :PathPart('chained/loose') CaptureArgs(1) { }
2349aeea 128
129#
130# Forwarding out of the middle of a chain.
131#
1c34f703 132sub chain_fw_a :Chained :PathPart('chained/chain_fw') :CaptureArgs(1) {
2349aeea 133 $_[1]->forward( '/action/chained/fw_dt_target' );
134}
135sub chain_fw_b :Chained('chain_fw_a') :PathPart('end') :Args(1) { }
136
137#
138# Detaching out of the middle of a chain.
139#
1c34f703 140sub chain_dt_a :Chained :PathPart('chained/chain_dt') :CaptureArgs(1) {
2349aeea 141 $_[1]->detach( '/action/chained/fw_dt_target' );
142}
143sub chain_dt_b :Chained('chain_dt_a') :PathPart('end') :Args(1) { }
144
145#
146# Target for former forward and chain tests.
147#
148sub fw_dt_target :Private { }
149
8b13f357 150#
151# Test multiple chained actions with no captures
152#
153sub empty_chain_a : Chained('/') PathPart('chained/empty') CaptureArgs(0) { }
154sub empty_chain_b : Chained('empty_chain_a') PathPart('') CaptureArgs(0) { }
155sub empty_chain_c : Chained('empty_chain_b') PathPart('') CaptureArgs(0) { }
156sub empty_chain_d : Chained('empty_chain_c') PathPart('') CaptureArgs(1) { }
157sub empty_chain_e : Chained('empty_chain_d') PathPart('') CaptureArgs(0) { }
158sub empty_chain_f : Chained('empty_chain_e') PathPart('') Args(1) { }
159
f505df49 160sub mult_nopp_base : Chained('/') PathPart('chained/mult_nopp') CaptureArgs(0) { }
161sub mult_nopp_all : Chained('mult_nopp_base') PathPart('') Args(0) { }
162sub mult_nopp_new : Chained('mult_nopp_base') PathPart('new') Args(0) { }
163sub mult_nopp_id : Chained('mult_nopp_base') PathPart('') CaptureArgs(1) { }
164sub mult_nopp_idall : Chained('mult_nopp_id') PathPart('') Args(0) { }
165sub mult_nopp_idnew : Chained('mult_nopp_id') PathPart('new') Args(0) { }
166
055abad1 167sub mult_nopp2_base : Chained('/') PathPart('chained/mult_nopp2') CaptureArgs(0) { }
168sub mult_nopp2_nocap : Chained('mult_nopp2_base') PathPart('') CaptureArgs(0) { }
169sub mult_nopp2_action : Chained('mult_nopp2_nocap') PathPart('action') CaptureArgs(0) { }
170sub mult_nopp2_action_default : Chained('mult_nopp2_action') PathPart('') Args(0) { }
171sub mult_nopp2_action_with_arg : Chained('mult_nopp2_action') PathPart('') Args(1) { }
172sub mult_nopp2_load : Chained('mult_nopp2_base') PathPart('') CaptureArgs(1) { }
173sub mult_nopp2_view : Chained('mult_nopp2_load') PathPart('') Args(0) { }
174
6365b527 175#
176# Test Choice between branches and early return logic
177# Declaration order is important for $children->{$*}, since this is first match best.
178#
179sub cc_base : Chained('/') PathPart('chained/choose_capture') CaptureArgs(0) { }
180sub cc_link : Chained('cc_base') PathPart('') CaptureArgs(0) { }
181sub cc_anchor : Chained('cc_link') PathPart('anchor.html') Args(0) { }
182sub cc_all : Chained('cc_base') PathPart('') Args() { }
183
184sub cc_a : Chained('cc_base') PathPart('') CaptureArgs(1) { }
185sub cc_a_link : Chained('cc_a') PathPart('a') CaptureArgs(0) { }
186sub cc_a_anchor : Chained('cc_a_link') PathPart('') Args() { }
187
188sub cc_b : Chained('cc_base') PathPart('b') CaptureArgs(0) { }
189sub cc_b_link : Chained('cc_b') PathPart('') CaptureArgs(1) { }
190sub cc_b_anchor : Chained('cc_b_link') PathPart('anchor.html') Args() { }
191
2f381252 192#
193# Test static paths vs. captures
194#
195
196sub apan : Chained('/') CaptureArgs(0) PathPrefix { }
197sub korv : Chained('apan') CaptureArgs(0) PathPart('') { }
198sub wurst : Chained('apan') CaptureArgs(1) PathPart('') { }
199sub static_end : Chained('korv') Args(0) { }
200sub capture_end : Chained('wurst') Args(0) PathPart('') { }
201
ef5e5ed1 202
203# */search vs doc/*
204sub view : Chained('/') PathPart('chained') CaptureArgs(1) {}
205sub star_search : Chained('view') PathPart('search') Args(0) { }
206sub doc_star : Chained('/') PathPart('chained/doc') Args(1) {}
207
06f97d3f 208sub return_arg : Chained('view') PathPart('return_arg') Args(1) {}
209
372b0c1b 210sub return_arg_decoded : Chained('/') PathPart('chained/return_arg_decoded') Args(1) {
211 my ($self, $c) = @_;
212 $c->req->args([ map { decode_entities($_) } @{ $c->req->args }]);
213}
214
36c67dc1 215sub roundtrip_urifor : Chained('/') PathPart('chained/roundtrip_urifor') CaptureArgs(1) {}
216sub roundtrip_urifor_end : Chained('roundtrip_urifor') PathPart('') Args(1) {
217 my ($self, $c) = @_;
218 # This should round-trip, always - i.e. the uri you put in should come back out.
219 $c->res->body($c->uri_for($c->action, $c->req->captures, @{$c->req->args}, $c->req->parameters));
220 $c->stash->{no_end} = 1;
221}
2c527b91 222
ca5d34df 223sub match_captures : Chained('/') PathPart('chained/match_captures') CaptureArgs(1) ActionClass('+TestApp::Action::TestMatchCaptures') {
1279064a 224 my ($self, $c) = @_;
225 $c->res->header( 'X-TestAppActionTestMatchCapturesHasRan', 'yes');
226}
227
228sub match_captures_end : Chained('match_captures') PathPart('bar') Args(0) { }
229
5882c86e 230sub end :Private {
231 my ($self, $c) = @_;
81e75875 232 return if $c->stash->{no_end};
5882c86e 233 my $out = join('; ', map { join(', ', @$_) }
234 ($c->req->captures, $c->req->args));
235 $c->res->body($out);
236}
237
2381;