1 package TestApp::Controller::Action::Chained;
8 use base qw/Catalyst::Controller/;
10 sub begin :Private { }
14 # :Chained('') means what?
18 # Simple parent/child action test
20 sub foo :PathPart('chained/foo') :CaptureArgs(1) :Chained('/') {
21 my ( $self, $c, @args ) = @_;
22 die "missing argument" unless @args;
23 die "more than 1 argument" if @args > 1;
25 sub endpoint :PathPart('end') :Chained('/action/chained/foo') :Args(1) { }
28 # Parent/child test with two args each
30 sub foo2 :PathPart('chained/foo2') :CaptureArgs(2) :Chained('/') { }
31 sub endpoint2 :PathPart('end2') :Chained('/action/chained/foo2') :Args(2) { }
34 # Relative specification of parent action
36 sub bar :PathPart('chained/bar') :Chained('/') :CaptureArgs(0) { }
37 sub finale :PathPart('') :Chained('bar') :Args { }
40 # three chain with concurrent endpoints
42 sub one :PathPart('chained/one') :Chained('/') :CaptureArgs(1) { }
43 sub two :PathPart('two') :Chained('/action/chained/one') :CaptureArgs(2) { }
44 sub three_end :PathPart('three') :Chained('two') :Args(3) { }
45 sub one_end :PathPart('chained/one') :Chained('/') :Args(1) { }
46 sub two_end :PathPart('two') :Chained('one') :Args(2) { }
49 # Dispatch on number of arguments
51 sub multi1 :PathPart('chained/multi') :Chained('/') :Args(1) { }
52 sub multi2 :PathPart('chained/multi') :Chained('/') :Args(2) { }
55 # Roots in an action defined in a higher controller
57 sub higher_root :PathPart('bar') :Chained('/action/chained/foo/higher_root') :Args(1) { }
60 # Controller -> subcontroller -> controller
62 sub pcp1 :PathPart('chained/pcp1') :Chained('/') :CaptureArgs(1) { }
63 sub pcp3 :Chained('/action/chained/foo/pcp2') :Args(1) { }
66 # Dispatch on capture number
68 sub multi_cap1 :PathPart('chained/multi_cap') :Chained('/') :CaptureArgs(1) { }
69 sub multi_cap2 :PathPart('chained/multi_cap') :Chained('/') :CaptureArgs(2) { }
70 sub multi_cap_end1 :PathPart('baz') :Chained('multi_cap1') :Args(0) { }
71 sub multi_cap_end2 :PathPart('baz') :Chained('multi_cap2') :Args(0) { }
74 # Priority: Slurpy args vs. chained actions
76 sub priority_a1 :PathPart('chained/priority_a') :Chained('/') :Args { }
77 sub priority_a2 :PathPart('chained/priority_a') :Chained('/') :CaptureArgs(1) { }
78 sub priority_a2_end :PathPart('end') :Chained('priority_a2') :Args(1) { }
82 # Priority: Fixed args vs. chained actions
84 sub priority_b1 :PathPart('chained/priority_b') :Chained('/') :Args(3) { }
85 sub priority_b2 :PathPart('chained/priority_b') :Chained('/') :CaptureArgs(1) { }
86 sub priority_b2_end :PathPart('end') :Chained('priority_b2') :Args(1) { }
89 # Priority: With no Args()
91 sub priority_c1 :PathPart('chained/priority_c') :Chained('/') :CaptureArgs(1) { }
92 sub priority_c2 :PathPart('') :Chained('priority_c1') { }
93 sub priority_c2_xyz :PathPart('xyz') :Chained('priority_c1') { }
97 # Optional specification of :Args in endpoint
99 sub opt_args :PathPart('chained/opt_args') :Chained('/') { }
102 # Optional PathPart test -> /chained/optpp/*/opt_pathpart/*
104 sub opt_pp_start :Chained('/') :PathPart('chained/optpp') :CaptureArgs(1) { }
105 sub opt_pathpart :Chained('opt_pp_start') :Args(1) { }
108 # Optional Args *and* PathPart -> /chained/optall/*/oa/...
110 sub opt_all_start :Chained('/') :PathPart('chained/optall') :CaptureArgs(1) { }
111 sub oa :Chained('opt_all_start') { }
114 # :Chained is the same as :Chained('/')
116 sub rootdef :Chained :PathPart('chained/rootdef') :Args(1) { }
119 # the ParentChain controller chains to this action by
120 # specifying :Chained('.')
122 sub parentchain :Chained('/') :PathPart('chained/parentchain') :CaptureArgs(1) { }
125 # This is just for a test that a loose end is not callable
127 sub loose :Chained :PathPart('chained/loose') CaptureArgs(1) { }
130 # Forwarding out of the middle of a chain.
132 sub chain_fw_a :Chained :PathPart('chained/chain_fw') :CaptureArgs(1) {
133 $_[1]->forward( '/action/chained/fw_dt_target' );
135 sub chain_fw_b :Chained('chain_fw_a') :PathPart('end') :Args(1) { }
138 # Detaching out of the middle of a chain.
140 sub chain_dt_a :Chained :PathPart('chained/chain_dt') :CaptureArgs(1) {
141 $_[1]->detach( '/action/chained/fw_dt_target' );
143 sub chain_dt_b :Chained('chain_dt_a') :PathPart('end') :Args(1) { }
146 # Target for former forward and chain tests.
148 sub fw_dt_target :Private { }
151 # Test multiple chained actions with no captures
153 sub empty_chain_a : Chained('/') PathPart('chained/empty') CaptureArgs(0) { }
154 sub empty_chain_b : Chained('empty_chain_a') PathPart('') CaptureArgs(0) { }
155 sub empty_chain_c : Chained('empty_chain_b') PathPart('') CaptureArgs(0) { }
156 sub empty_chain_d : Chained('empty_chain_c') PathPart('') CaptureArgs(1) { }
157 sub empty_chain_e : Chained('empty_chain_d') PathPart('') CaptureArgs(0) { }
158 sub empty_chain_f : Chained('empty_chain_e') PathPart('') Args(1) { }
160 sub mult_nopp_base : Chained('/') PathPart('chained/mult_nopp') CaptureArgs(0) { }
161 sub mult_nopp_all : Chained('mult_nopp_base') PathPart('') Args(0) { }
162 sub mult_nopp_new : Chained('mult_nopp_base') PathPart('new') Args(0) { }
163 sub mult_nopp_id : Chained('mult_nopp_base') PathPart('') CaptureArgs(1) { }
164 sub mult_nopp_idall : Chained('mult_nopp_id') PathPart('') Args(0) { }
165 sub mult_nopp_idnew : Chained('mult_nopp_id') PathPart('new') Args(0) { }
168 # Test Choice between branches and early return logic
169 # Declaration order is important for $children->{$*}, since this is first match best.
171 sub cc_base : Chained('/') PathPart('chained/choose_capture') CaptureArgs(0) { }
172 sub cc_link : Chained('cc_base') PathPart('') CaptureArgs(0) { }
173 sub cc_anchor : Chained('cc_link') PathPart('anchor.html') Args(0) { }
174 sub cc_all : Chained('cc_base') PathPart('') Args() { }
176 sub cc_a : Chained('cc_base') PathPart('') CaptureArgs(1) { }
177 sub cc_a_link : Chained('cc_a') PathPart('a') CaptureArgs(0) { }
178 sub cc_a_anchor : Chained('cc_a_link') PathPart('') Args() { }
180 sub cc_b : Chained('cc_base') PathPart('b') CaptureArgs(0) { }
181 sub cc_b_link : Chained('cc_b') PathPart('') CaptureArgs(1) { }
182 sub cc_b_anchor : Chained('cc_b_link') PathPart('anchor.html') Args() { }
185 # Test static paths vs. captures
188 sub apan : Chained('/') CaptureArgs(0) PathPrefix { }
189 sub korv : Chained('apan') CaptureArgs(0) PathPart('') { }
190 sub wurst : Chained('apan') CaptureArgs(1) PathPart('') { }
191 sub static_end : Chained('korv') Args(0) { }
192 sub capture_end : Chained('wurst') Args(0) PathPart('') { }
196 sub view : Chained('/') PathPart('chained') CaptureArgs(1) {}
197 sub star_search : Chained('view') PathPart('search') Args(0) { }
198 sub doc_star : Chained('/') PathPart('chained/doc') Args(1) {}
200 sub return_arg : Chained('view') PathPart('return_arg') Args(1) {}
202 sub return_arg_decoded : Chained('/') PathPart('chained/return_arg_decoded') Args(1) {
204 $c->req->args([ map { decode_entities($_) } @{ $c->req->args }]);
207 sub roundtrip_urifor : Chained('/') PathPart('chained/roundtrip_urifor') CaptureArgs(1) {}
208 sub roundtrip_urifor_end : Chained('roundtrip_urifor') PathPart('') Args(1) {
210 # This should round-trip, always - i.e. the uri you put in should come back out.
211 $c->res->body($c->uri_for($c->action, $c->req->captures, @{$c->req->args}, $c->req->parameters));
212 $c->stash->{no_end} = 1;
217 return if $c->stash->{no_end};
218 my $out = join('; ', map { join(', ', @$_) }
219 ($c->req->captures, $c->req->args));