]> Dogcows Code - chaz/p5-CGI-Ex/blob - t/7_template_00_base.t
CGI::Ex 2.11
[chaz/p5-CGI-Ex] / t / 7_template_00_base.t
1 # -*- Mode: Perl; -*-
2
3 =head1 NAME
4
5 7_template_00_base.t - Test the basic language functionality of CGI::Ex::Template - including many edge cases
6
7 =cut
8
9 use vars qw($module $is_tt);
10 BEGIN {
11 $module = 'CGI::Ex::Template'; #real 0m0.885s #user 0m0.432s #sys 0m0.004s
12 # $module = 'Template'; #real 0m2.133s #user 0m1.108s #sys 0m0.024s
13 $is_tt = $module eq 'Template';
14 };
15
16 use strict;
17 use Test::More tests => ! $is_tt ? 806 : 599;
18 use Data::Dumper qw(Dumper);
19 use constant test_taint => 0 && eval { require Taint::Runtime };
20
21 use_ok($module);
22
23 Taint::Runtime::taint_start() if test_taint;
24
25 ###----------------------------------------------------------------###
26
27 sub process_ok { # process the value and say if it was ok
28 my $str = shift;
29 my $test = shift;
30 my $vars = shift || {};
31 my $conf = local $vars->{'tt_config'} = $vars->{'tt_config'} || [];
32 my $obj = shift || $module->new(@$conf); # new object each time
33 my $out = '';
34 my $line = (caller)[2];
35 delete $vars->{'tt_config'};
36
37 Taint::Runtime::taint(\$str) if test_taint;
38
39 $obj->process(\$str, $vars, \$out);
40 my $ok = ref($test) ? $out =~ $test : $out eq $test;
41 if ($ok) {
42 ok(1, "Line $line \"$str\" => \"$out\"");
43 return $obj;
44 } else {
45 ok(0, "Line $line \"$str\"");
46 warn "# Was:\n$out\n# Should've been:\n$test\n";
47 print $obj->error if $obj->can('error');
48 print Dumper $obj->parse_tree(\$str) if $obj->can('parse_tree');
49 exit;
50 }
51 }
52
53 ###----------------------------------------------------------------###
54
55 ### set up some dummy packages for various tests
56 {
57 package MyTestPlugin::Foo;
58 $INC{'MyTestPlugin/Foo.pm'} = $0;
59 sub load { $_[0] }
60 sub new {
61 my $class = shift;
62 my $context = shift; # note the plugin style object that needs to shift off context
63 my $args = shift || {};
64 return bless $args, $class;
65 }
66 sub bar { my $self = shift; return join('', map {"$_$self->{$_}"} sort keys %$self) }
67 sub seven { 7 }
68 sub many { return 1, 2, 3 }
69 sub echo { my $self = shift; $_[0] }
70 }
71 {
72 package Foo2;
73 $INC{'Foo2.pm'} = $0;
74 use base qw(MyTestPlugin::Foo);
75 use vars qw($AUTOLOAD);
76 sub new {
77 my $class = shift;
78 my $args = shift || {}; # note - no plugin context
79 return bless $args, $class;
80 }
81 sub leave {} # hacks to allow tt to do the plugins passed via PLUGINS
82 sub delocalise {} # hacks to allow tt to do the plugins passed via PLUGINS
83 }
84
85 my $obj = Foo2->new;
86 my $vars;
87
88 ###----------------------------------------------------------------###
89 print "### GET ##############################################################\n";
90
91 process_ok("[% foo %]" => "");
92 process_ok("[% foo %]" => "7", {foo => 7});
93 process_ok("[% foo %]" => "7", {tt_config => [VARIABLES => {foo => 7}]});
94 process_ok("[% foo %]" => "7", {tt_config => [PRE_DEFINE => {foo => 7}]});
95 process_ok("[% foo %][% foo %][% foo %]" => "777", {foo => 7});
96 process_ok("[% foo() %]" => "7", {foo => 7});
97 process_ok("[% foo.bar %]" => "");
98 process_ok("[% foo.bar %]" => "", {foo => {}});
99 process_ok("[% foo.bar %]" => "7", {foo => {bar => 7}});
100 process_ok("[% foo().bar %]" => "7", {foo => {bar => 7}});
101 process_ok("[% foo.0 %]" => "7", {foo => [7, 2, 3]});
102 process_ok("[% foo.10 %]" => "", {foo => [7, 2, 3]});
103 process_ok("[% foo %]" => 7, {foo => sub { 7 }});
104 process_ok("[% foo(7) %]" => 7, {foo => sub { $_[0] }});
105 process_ok("[% foo.length %]" => 1, {foo => sub { 7 }});
106 process_ok("[% foo.0 %]" => 7, {foo => sub { return 7, 2, 3 }});
107 process_ok("[% foo(bar) %]" => 7, {foo => sub { $_[0] }, bar => 7});
108 process_ok("[% foo(bar.baz) %]" => 7,{foo => sub { $_[0] }, bar => {baz => 7}});
109 process_ok("[% foo.seven %]" => 7, {foo => $obj});
110 process_ok("[% foo.seven() %]" => 7, {foo => $obj});
111 process_ok("[% foo.seven.length %]" => 1, {foo => $obj});
112 process_ok("[% foo.echo(7) %]" => 7, {foo => $obj});
113 process_ok("[% foo.many.0 %]" => 1, {foo => $obj});
114 process_ok("[% foo.many.10 %]" => '',{foo => $obj});
115 process_ok("[% foo.nomethod %]" => '',{foo => $obj});
116 process_ok("[% foo.nomethod.0 %]" => '',{foo => $obj});
117
118 process_ok("[% GET foo %]" => "");
119 process_ok("[% GET foo %]" => "7", {foo => 7});
120 process_ok("[% GET foo.bar %]" => "");
121 process_ok("[% GET foo.bar %]" => "", {foo => {}});
122 process_ok("[% GET foo.bar %]" => "7", {foo => {bar => 7}});
123 process_ok("[% GET foo.0 %]" => "7", {foo => [7, 2, 3]});
124 process_ok("[% GET foo %]" => 7, {foo => sub { 7 }});
125 process_ok("[% GET foo(7) %]" => 7, {foo => sub { $_[0] }});
126
127 process_ok("[% \$name %]" => "", {name => 'foo'});
128 process_ok("[% \$name %]" => "7", {name => 'foo', foo => 7});
129 process_ok("[% \$name.bar %]" => "", {name => 'foo'});
130 process_ok("[% \$name.bar %]" => "", {name => 'foo', foo => {}});
131 process_ok("[% \$name.bar %]" => "7", {name => 'foo', foo => {bar => 7}});
132 process_ok("[% \$name().bar %]" => "7", {name => 'foo', foo => {bar => 7}});
133 process_ok("[% \$name.0 %]" => "7", {name => 'foo', foo => [7, 2, 3]});
134 process_ok("[% \$name %]" => 7, {name => 'foo', foo => sub { 7 }});
135 process_ok("[% \$name(7) %]" => 7, {name => 'foo', foo => sub { $_[0] }});
136
137 process_ok("[% GET \$name %]" => "", {name => 'foo'});
138 process_ok("[% GET \$name %]" => "7", {name => 'foo', foo => 7});
139 process_ok("[% GET \$name.bar %]" => "", {name => 'foo'});
140 process_ok("[% GET \$name.bar %]" => "", {name => 'foo', foo => {}});
141 process_ok("[% GET \$name.bar %]" => "7", {name => 'foo', foo => {bar => 7}});
142 process_ok("[% GET \$name.0 %]" => "7", {name => 'foo', foo => [7, 2, 3]});
143 process_ok("[% GET \$name %]" => 7, {name => 'foo', foo => sub { 7 }});
144 process_ok("[% GET \$name(7) %]" => 7, {name => 'foo', foo => sub { $_[0] }});
145
146 process_ok("[% \$name %]" => "", {name => 'foo foo', foo => 7});
147 process_ok("[% GET \$name %]" => "", {name => 'foo foo', foo => 7});
148
149 process_ok("[% \${name} %]" => "", {name => 'foo'});
150 process_ok("[% \${name} %]" => "7", {name => 'foo', foo => 7});
151 process_ok("[% \${name}.bar %]" => "", {name => 'foo'});
152 process_ok("[% \${name}.bar %]" => "", {name => 'foo', foo => {}});
153 process_ok("[% \${name}.bar %]" => "7", {name => 'foo', foo => {bar => 7}});
154 process_ok("[% \${name}().bar %]" => "7", {name => 'foo', foo => {bar => 7}});
155 process_ok("[% \${name}.0 %]" => "7", {name => 'foo', foo => [7, 2, 3]});
156 process_ok("[% \${name} %]" => 7, {name => 'foo', foo => sub { 7 }});
157 process_ok("[% \${name}(7) %]" => 7, {name => 'foo', foo => sub { $_[0] }});
158
159 process_ok("[% GET \${name} %]" => "", {name => 'foo'});
160 process_ok("[% GET \${name} %]" => "7", {name => 'foo', foo => 7});
161 process_ok("[% GET \${name}.bar %]" => "", {name => 'foo'});
162 process_ok("[% GET \${name}.bar %]" => "", {name => 'foo', foo => {}});
163 process_ok("[% GET \${name}.bar %]" => "7", {name => 'foo', foo => {bar => 7}});
164 process_ok("[% GET \${name}.0 %]" => "7", {name => 'foo', foo => [7, 2, 3]});
165 process_ok("[% GET \${name} %]" => 7, {name => 'foo', foo => sub { 7 }});
166 process_ok("[% GET \${name}(7) %]" => 7, {name => 'foo', foo => sub { $_[0] }});
167
168 process_ok("[% \${name} %]" => "", {name => 'foo foo', foo => 7});
169 process_ok("[% GET \${name} %]" => "", {name => 'foo foo', foo => 7});
170 process_ok("[% GET \${'foo'} %]" => 'bar', {foo => 'bar'});
171
172 process_ok("[% foo.\$name %]" => '', {name => 'bar'});
173 process_ok("[% foo.\$name %]" => 7, {name => 'bar', foo => {bar => 7}});
174 process_ok("[% foo.\$name.baz %]" => '', {name => 'bar', bar => {baz => 7}});
175
176 process_ok("[% \"hi\" %]" => 'hi');
177 process_ok("[% \"hi %]" => '');
178 process_ok("[% 'hi' %]" => 'hi');
179 process_ok("[% 'hi %]" => '');
180 process_ok("[% \"\$foo\" %]" => '7', {foo => 7});
181 process_ok("[% \"hi \$foo\" %]" => 'hi 7', {foo => 7});
182 process_ok("[% \"hi \${foo}\" %]" => 'hi 7', {foo => 7});
183 process_ok("[% 'hi \$foo' %]" => 'hi $foo', {foo => 7});
184 process_ok("[% 'hi \${foo}' %]" => 'hi ${foo}', {foo => 7});
185 process_ok("[% 7 %]" => 7);
186 process_ok("[% -7 %]" => -7);
187
188 process_ok("[% \"hi \${foo.seven}\" %]" => 'hi 7', {foo => $obj});
189 process_ok("[% \"hi \${foo.echo(7)}\" %]" => 'hi 7', {foo => $obj});
190
191 process_ok("[% _foo %]2" => '2', {_foo => 1});
192 process_ok("[% \$bar %]2" => '2', {_foo => 1, bar => '_foo'});
193 process_ok("[% __foo %]2" => '2', {__foo => 1});
194
195 process_ok("[% qw/Foo Bar Baz/.0 %]" => 'Foo') if ! $is_tt;
196 process_ok('[% [0..10].-1 %]' => '10') if ! $is_tt;
197 process_ok('[% [0..10].${ 2.3 } %]' => '2') if ! $is_tt;
198
199 process_ok("[% (1 + 2)() %]" => ''); # parse error
200 process_ok("[% (1 + 2) %]" => '3');
201 process_ok("[% (a) %]" => '2', {a => 2});
202 process_ok("[% ('foo') %]" => 'foo');
203 process_ok("[% (a(2)) %]" => '2', {a => sub { $_[0] }});
204
205 ###----------------------------------------------------------------###
206 print "### SET ##############################################################\n";
207
208 process_ok("[% SET foo bar %][% foo %]" => '');
209 process_ok("[% SET foo = 1 %][% foo %]" => '1');
210 process_ok("[% SET foo = 1 bar = 2 %][% foo %][% bar %]" => '12');
211 process_ok("[% SET foo bar = 1 %][% foo %]" => '');
212 process_ok("[% SET foo = 1 ; bar = 1 %][% foo %]" => '1');
213 process_ok("[% SET foo = 1 %][% SET foo %][% foo %]" => '');
214
215 process_ok("[% SET foo = [] %][% foo.0 %]" => "");
216 process_ok("[% SET foo = [1, 2, 3] %][% foo.1 %]" => 2);
217 process_ok("[% SET foo = {} %][% foo.0 %]" => "");
218 process_ok("[% SET foo = {1 => 2} %][% foo.1 %]" => "2") if ! $is_tt;
219 process_ok("[% SET foo = {'1' => 2} %][% foo.1 %]" => "2");
220
221 process_ok("[% SET name = 1 %][% SET foo = name %][% foo %]" => "1");
222 process_ok("[% SET name = 1 %][% SET foo = \$name %][% foo %]" => "");
223 process_ok("[% SET name = 1 %][% SET foo = \${name} %][% foo %]" => "");
224 process_ok("[% SET name = 1 %][% SET foo = \"\$name\" %][% foo %]" => "1");
225 process_ok("[% SET name = 1 foo = name %][% foo %]" => '1');
226 process_ok("[% SET name = 1 %][% SET foo = {\$name => 2} %][% foo.1 %]" => "2");
227 process_ok("[% SET name = 1 %][% SET foo = {\"\$name\" => 2} %][% foo.1 %]" => "2") if ! $is_tt;
228 process_ok("[% SET name = 1 %][% SET foo = {\${name} => 2} %][% foo.1 %]" => "2");
229
230 process_ok("[% SET name = 7 %][% SET foo = {'2' => name} %][% foo.2 %]" => "7");
231 process_ok("[% SET name = 7 %][% SET foo = {'2' => \"\$name\"} %][% foo.2 %]" => "7");
232
233 process_ok("[% SET name = 7 %][% SET foo = [1, name, 3] %][% foo.1 %]" => "7");
234 process_ok("[% SET name = 7 %][% SET foo = [1, \"\$name\", 3] %][% foo.1 %]" => "7");
235
236 process_ok("[% SET foo = { bar => { baz => [0, 7, 2] } } %][% foo.bar.baz.1 %]" => "7");
237
238 process_ok("[% SET foo.bar = 1 %][% foo.bar %]" => '1');
239 process_ok("[% SET foo.bar.baz.bing = 1 %][% foo.bar.baz.bing %]" => '1');
240 process_ok("[% SET foo.bar.2 = 1 %][% foo.bar.2 %] [% foo.bar.size %]" => '1 1');
241 process_ok("[% SET foo.bar = [] %][% SET foo.bar.2 = 1 %][% foo.bar.2 %] [% foo.bar.size %]" => '1 3');
242
243 process_ok("[% SET name = 'two' %][% SET \$name = 3 %][% two %]" => 3);
244 process_ok("[% SET name = 'two' %][% SET \${name} = 3 %][% two %]" => 3);
245 process_ok("[% SET name = 2 %][% SET foo.\$name = 3 %][% foo.2 %]" => 3);
246 process_ok("[% SET name = 2 %][% SET foo.\$name = 3 %][% foo.\$name %]" => 3);
247 process_ok("[% SET name = 2 %][% SET foo.\${name} = 3 %][% foo.2 %]" => 3);
248 process_ok("[% SET name = 2 %][% SET foo.\${name} = 3 %][% foo.2 %]" => 3);
249 process_ok("[% SET name = 'two' %][% SET \$name.foo = 3 %][% two.foo %]" => 3);
250 process_ok("[% SET name = 'two' %][% SET \${name}.foo = 3 %][% two.foo %]" => 3);
251 process_ok("[% SET name = 'two' %][% SET foo.\$name.foo = 3 %][% foo.two.foo %]" => 3);
252 process_ok("[% SET name = 'two' %][% SET foo.\${name}.foo = 3 %][% foo.two.foo %]" => 3);
253
254 process_ok("[% SET foo = [1..10] %][% foo.6 %]" => 7);
255 process_ok("[% SET foo = [10..1] %][% foo.6 %]" => '');
256 process_ok("[% SET foo = [-10..-1] %][% foo.6 %]" => -4);
257 process_ok("[% SET foo = [1..10, 21..30] %][% foo.12 %]" => 23) if ! $is_tt;
258 process_ok("[% SET foo = [..100] bar = 7 %][% bar %][% foo.0 %]" => '');
259 process_ok("[% SET foo = [100..] bar = 7 %][% bar %][% foo.0 %]" => '');
260 process_ok("[% SET foo = ['a'..'z'] %][% foo.6 %]" => 'g');
261 process_ok("[% SET foo = ['z'..'a'] %][% foo.6 %]" => '');
262 process_ok("[% SET foo = ['a'..'z'].reverse %][% foo.6 %]" => 't') if ! $is_tt;
263
264 process_ok("[% foo = 1 %][% foo %]" => '1');
265 process_ok("[% foo = 1 ; bar = 2 %][% foo %][% bar %]" => '12');
266 process_ok("[% foo.bar = 2 %][% foo.bar %]" => '2');
267
268 process_ok('[% a = "a" %]|[% (b = a) %]|[% a %]|[% b %]' => '|a|a|a');
269 process_ok('[% a = "a" %][% (c = (b = a)) %][% a %][% b %][% c %]' => 'aaaa');
270
271 process_ok("[% a = qw{Foo Bar Baz} ; a.2 %]" => 'Baz') if ! $is_tt;
272
273 process_ok("[% foo = 1 bar = 2 %][% foo %][% bar %]" => '12');
274 process_ok("[% foo = 1 bar = 2 %][% foo = 3 bar %][% foo %][% bar %]" => '232') if ! $is_tt;
275 process_ok("[% a = 1 a = a + 2 a %]" => 3) if ! $is_tt;
276
277 process_ok("[% _foo = 1 %][% _foo %]2" => '2');
278 process_ok("[% foo._bar %]2" => '2', {foo => {_bar =>1}});
279
280 ###----------------------------------------------------------------###
281 print "### multiple statements in same tag ##################################\n";
282
283 process_ok("[% foo; %]" => '1', {foo => 1});
284 process_ok("[% GET foo; %]" => '1', {foo => 1});
285 process_ok("[% GET foo; GET foo %]" => '11', {foo => 1});
286
287 ###----------------------------------------------------------------###
288 print "### CALL / DEFAULT ###################################################\n";
289
290 process_ok("[% DEFAULT foo = 7 %][% foo %]" => 7);
291 process_ok("[% SET foo = 5 %][% DEFAULT foo = 7 %][% foo %]" => 5);
292 process_ok("[% DEFAULT foo.bar.baz.bing = 6 %][% foo.bar.baz.bing %]" => 6);
293
294 my $t = 0;
295 process_ok("[% foo %]" => 'hi', {foo => sub {$t++; 'hi'}});
296 process_ok("[% GET foo %]" => 'hi', {foo => sub {$t++; 'hi'}});
297 process_ok("[% CALL foo %]" => '', {foo => sub {$t++; 'hi'}});
298 ok($t == 3, "CALL method actually called var");
299
300 ###----------------------------------------------------------------###
301 print "### scalar vmethods ##################################################\n";
302
303 process_ok("[% n.0 %]" => '7', {n => 7}) if ! $is_tt;
304 process_ok("[% n.chunk(3).join %]" => 'abc def g', {n => 'abcdefg'});
305 process_ok("[% n.chunk(-3).join %]" => 'a bcd efg', {n => 'abcdefg'});
306 process_ok("[% n|collapse %]" => "a b", {n => ' a b '}); # TT2 filter
307 process_ok("[% n.defined %]" => "1", {n => ''});
308 process_ok("[% n.defined %]" => "", {n => undef});
309 process_ok("[% n.defined %]" => "1", {n => '1'});
310 process_ok("[% n|indent %]" => " a\n b", {n => "a\nb"}); # TT2 filter
311 process_ok("[% n|indent(2) %]" => " a\n b", {n => "a\nb"}); # TT2 filter
312 process_ok("[% n|indent('wow ') %]" => "wow a\nwow b", {n => "a\nb"}); # TT2 filter
313 process_ok("[% n.int %]" => "123", {n => "123.234"}) if ! $is_tt;
314 process_ok("[% n.int %]" => "123", {n => "123gggg"}) if ! $is_tt;
315 process_ok("[% n.int %]" => "0", {n => "ff123.234"}) if ! $is_tt;
316 process_ok("[% n.fmt %]" => '7', {n => 7}) if ! $is_tt;
317 process_ok("[% n.fmt('%02d') %]" => '07', {n => 7}) if ! $is_tt;
318 process_ok("[% n.fmt('%0*d', 3) %]" => '007', {n => 7}) if ! $is_tt;
319 process_ok("[% n.fmt('(%s)') %]" => "(a\nb)", {n => "a\nb"}) if ! $is_tt;
320 process_ok("[% n|format('%02d') %]" => '07', {n => 7}); # TT2 filter
321 process_ok("[% n|format('%0*d', 3) %]" => '007', {n => 7}) if ! $is_tt;
322 process_ok("[% n|format('(%s)') %]" => "(a)\n(b)", {n => "a\nb"}); # TT2 filter
323 process_ok("[% n.hash.items.1 %]" => "b", {n => {a => "b"}});
324 process_ok("[% n|html %]" => "&", {n => '&'}); # TT2 filter
325 process_ok("[% n.item %]" => '7', {n => 7});
326 process_ok("[% n|lcfirst %]" => 'fOO', {n => "FOO"}); # TT2 filter
327 process_ok("[% n.length %]" => 3, {n => "abc"});
328 process_ok("[% n.list.0 %]" => 'abc', {n => "abc"});
329 process_ok("[% n|lower %]" => 'abc', {n => "ABC"}); # TT2 filter
330 process_ok("[% n.match('foo').join %]" => '', {n => "bar"});
331 process_ok("[% n.match('foo').join %]" => '1', {n => "foo"});
332 process_ok("[% n.match('foo',1).join %]" => 'foo', {n => "foo"});
333 process_ok("[% n.match('(foo)').join %]" => 'foo', {n => "foo"});
334 process_ok("[% n.match('(foo)').join %]" => 'foo', {n => "foofoo"});
335 process_ok("[% n.match('(foo)',1).join %]" => 'foo foo', {n => "foofoo"});
336 process_ok("[% n.null %]" => '', {n => "abc"});
337 process_ok("[% n.rand %]" => qr{^\d+\.\d+}, {n => "2"}) if ! $is_tt;
338 process_ok("[% n.rand %]" => qr{^\d+\.\d+}, {n => "ab"}) if ! $is_tt;
339 process_ok("[% n.remove('bc') %]" => "a", {n => "abc"});
340 process_ok("[% n.remove('bc') %]" => "aa", {n => "abcabc"});
341 process_ok("[% n.repeat %]" => '1', {n => 1}) if ! $is_tt; # tt2 virtual method defaults to 0
342 process_ok("[% n.repeat(0) %]" => '', {n => 1});
343 process_ok("[% n.repeat(1) %]" => '1', {n => 1});
344 process_ok("[% n.repeat(2) %]" => '11', {n => 1});
345 process_ok("[% n.repeat(2,'|') %]" => '1|1', {n => 1}) if ! $is_tt;
346 process_ok("[% n.replace('foo', 'bar') %]" => 'barbar', {n => 'foofoo'});
347 process_ok("[% n.replace('(foo)', 'bar\$1') %]" => 'barfoobarfoo', {n => 'foofoo'}) if ! $is_tt;
348 process_ok("[% n.replace('foo', 'bar', 0) %]" => 'barfoo', {n => 'foofoo'}) if ! $is_tt;
349 process_ok("[% n.search('foo') %]" => '', {n => "bar"});
350 process_ok("[% n.search('foo') %]" => '1', {n => "foo"});
351 process_ok("[% n.size %]" => '1', {n => "foo"});
352 process_ok("[% n.split.join('|') %]" => "abc", {n => "abc"});
353 process_ok("[% n.split.join('|') %]" => "a|b|c", {n => "a b c"});
354 process_ok("[% n.split.join('|') %]" => "a|b|c", {n => "a b c"});
355 process_ok("[% n.split(u,2).join('|') %]" => "a|b c", {n => "a b c", u => undef}) if ! $is_tt;
356 process_ok("[% n.split(u,2).join('|') %]" => "a| b c", {n => "a b c", u => undef}) if $is_tt;
357 process_ok("[% n.split('/').join('|') %]" => "a|b|c", {n => "a/b/c"});
358 process_ok("[% n.split('/', 2).join('|') %]" => "a|b/c", {n => "a/b/c"});
359 process_ok("[% n.stderr %]" => "", {n => "# testing stderr ... ok\r"});
360 process_ok("[% n|trim %]" => "a b", {n => ' a b '}); # TT2 filter
361 process_ok("[% n|ucfirst %]" => 'Foo', {n => "foo"}); # TT2 filter
362 process_ok("[% n|upper %]" => 'FOO', {n => "foo"}); # TT2 filter
363 process_ok("[% n|uri %]" => 'a%20b', {n => "a b"}); # TT2 filter
364
365 ###----------------------------------------------------------------###
366 print "### list vmethods ####################################################\n";
367
368 process_ok("[% a.defined %]" => '1', {a => [2,3]});
369 process_ok("[% a.defined(1) %]" => '1', {a => [2,3]});
370 process_ok("[% a.defined(3) %]" => '', {a => [2,3]});
371 process_ok("[% a.first %]" => '2', {a => [2..10]});
372 process_ok("[% a.first(3).join %]" => '2 3 4', {a => [2..10]});
373 process_ok("[% a.fmt %]" => '2 3', {a => [2,3]}) if ! $is_tt;
374 process_ok("[% a.fmt('%02d') %]" => '02 03', {a => [2,3]}) if ! $is_tt;
375 process_ok("[% a.fmt('%02d',' ') %]" => '02 03', {a => [2,3]}) if ! $is_tt;
376 process_ok("[% a.fmt('%02d','|') %]" => '02|03', {a => [2,3]}) if ! $is_tt;
377 process_ok("[% a.fmt('%0*d','|', 3) %]" => '002|003', {a => [2,3]}) if ! $is_tt;
378 process_ok("[% a.grep.join %]" => '2 3', {a => [2,3]});
379 process_ok("[% a.grep(2).join %]" => '2', {a => [2,3]});
380 process_ok("[% a.hash.items.join %]" => '2 3', {a => [2,3]});
381 process_ok("[% a.hash(5).items.sort.join %]" => '2 3 5 6', {a => [2,3]});
382 process_ok("[% a.import(5) %]|[% a.join %]" => '|2 3', {a => [2,3]}) if ! $is_tt;
383 process_ok("[% a.import(5) %]|[% a.join %]" => qr{^ARRAY.+|2 3$ }x, {a => [2,3]}) if $is_tt;
384 process_ok("[% a.import([5]) %]|[% a.join %]" => '|2 3 5', {a => [2,3]}) if ! $is_tt;
385 process_ok("[% a.import([5]) %]|[% a.join %]" => qr{ARRAY.+|2 3 5$ }x, {a => [2,3]}) if $is_tt;
386 process_ok("[% a.item %]" => '2', {a => [2,3]});
387 process_ok("[% a.item(1) %]" => '3', {a => [2,3]});
388 process_ok("[% a.join %]" => '2 3', {a => [2,3]});
389 process_ok("[% a.join('|') %]" => '2|3', {a => [2,3]});
390 process_ok("[% a.last %]" => '10', {a => [2..10]});
391 process_ok("[% a.last(3).join %]" => '8 9 10', {a => [2..10]});
392 process_ok("[% a.list.join %]" => '2 3', {a => [2, 3]});
393 process_ok("[% a.max %]" => '1', {a => [2, 3]});
394 process_ok("[% a.merge(5).join %]" => '2 3', {a => [2,3]});
395 process_ok("[% a.merge([5]).join %]" => '2 3 5', {a => [2,3]});
396 process_ok("[% a.merge([5]).null %][% a.join %]" => '2 3', {a => [2,3]});
397 process_ok("[% a.nsort.join %]" => '1 2 3', {a => [2, 3, 1]});
398 process_ok("[% a.nsort('b').0.b %]" => '7', {a => [{b => 23}, {b => 7}]});
399 process_ok("[% a.pop %][% a.join %]" => '32', {a => [2, 3]});
400 process_ok("[% a.push(3) %][% a.join %]" => '2 3 3', {a => [2, 3]});
401 process_ok("[% a.pick %]" => qr{ ^[23]$ }x, {a => [2, 3]}) if ! $is_tt;
402 process_ok("[% a.pick(5).join('') %]" => qr{ ^[23]{5}$ }x, {a => [2, 3]}) if ! $is_tt;
403 process_ok("[% a.reverse.join %]" => '3 2', {a => [2, 3]});
404 process_ok("[% a.shift %][% a.join %]" => '23', {a => [2, 3]});
405 process_ok("[% a.size %]" => '2', {a => [2, 3]});
406 process_ok("[% a.slice.join %]" => '2 3 4 5', {a => [2..5]});
407 process_ok("[% a.slice(2).join %]" => '4 5', {a => [2..5]});
408 process_ok("[% a.slice(0,2).join %]" => '2 3 4', {a => [2..5]});
409 process_ok("[% a.sort.join %]" => '1 2 3', {a => [2, 3, 1]});
410 process_ok("[% a.sort('b').0.b %]" => 'wee', {a => [{b => "wow"}, {b => "wee"}]});
411 process_ok("[% a.splice.join %]|[% a.join %]" => '2 3 4 5|', {a => [2..5]});
412 process_ok("[% a.splice(2).join %]|[% a.join %]" => '4 5|2 3', {a => [2..5]});
413 process_ok("[% a.splice(0,2).join %]|[% a.join %]" => '2 3|4 5', {a => [2..5]});
414 process_ok("[% a.splice(0,2,'hrm').join %]|[% a.join %]" => '2 3|hrm 4 5', {a => [2..5]});
415 process_ok("[% a.unique.join %]" => '2 3', {a => [2,3,3,3,2]});
416 process_ok("[% a.unshift(3) %][% a.join %]" => '3 2 3', {a => [2, 3]});
417
418 ###----------------------------------------------------------------###
419 print "### hash vmethods ####################################################\n";
420
421 process_ok("[% h.defined %]" => "1", {h => {}});
422 process_ok("[% h.defined('a') %]" => "1", {h => {a => 1}});
423 process_ok("[% h.defined('b') %]" => "", {h => {a => 1}});
424 process_ok("[% h.defined('a') %]" => "", {h => {a => undef}});
425 process_ok("[% h.delete('a') %]|[% h.keys.0 %]" => "|b", {h => {a => 1, b=> 2}});
426 process_ok("[% h.delete('a', 'b').join %]|[% h.keys.0 %]" => "|", {h => {a => 1, b=> 2}});
427 process_ok("[% h.delete('a', 'c').join %]|[% h.keys.0 %]" => "|b", {h => {a => 1, b=> 2}});
428 process_ok("[% h.each.sort.join %]" => "1 2 a b", {h => {a => 1, b=> 2}});
429 process_ok("[% h.exists('a') %]" => "1", {h => {a => 1}});
430 process_ok("[% h.exists('b') %]" => "", {h => {a => 1}});
431 process_ok("[% h.exists('a') %]" => "1", {h => {a => undef}});
432 process_ok("[% h.fmt %]" => "b\tB\nc\tC", {h => {b => "B", c => "C"}}) if ! $is_tt;
433 process_ok("[% h.fmt('%s => %s') %]" => "b => B\nc => C", {h => {b => "B", c => "C"}}) if ! $is_tt;
434 process_ok("[% h.fmt('%s => %s', '|') %]" => "b => B|c => C", {h => {b => "B", c => "C"}}) if ! $is_tt;
435 process_ok("[% h.fmt('%*s=>%s', '|', 3) %]" => " b=>B| c=>C", {h => {b => "B", c => "C"}}) if ! $is_tt;
436 process_ok("[% h.fmt('%*s=>%*s', '|', 3, 4) %]" => " b=> B| c=> C", {h => {b => "B", c => "C"}}) if ! $is_tt;
437 process_ok("[% h.hash.fmt %]" => "b\tB\nc\tC", {h => {b => "B", c => "C"}}) if ! $is_tt;
438 process_ok("[% h.import('a') %]|[% h.items.sort.join %]" => "|b B c C", {h => {b => "B", c => "C"}});
439 process_ok("[% h.import({'b' => 'boo'}) %]|[% h.items.sort.join %]" => "|b boo c C", {h => {b => "B", c => "C"}});
440 process_ok("[% h.item('a') %]" => 'A', {h => {a => 'A'}});
441 process_ok("[% h.item('_a') %]" => '', {h => {_a => 'A'}}) if ! $is_tt;
442 process_ok("[% h.items.sort.join %]" => "1 2 a b", {h => {a => 1, b=> 2}});
443 process_ok("[% h.keys.sort.join %]" => "a b", {h => {a => 1, b=> 2}});
444 process_ok("[% h.list('each').sort.join %]" => "1 2 a b", {h => {a => 1, b=> 2}});
445 process_ok("[% h.list('keys').sort.join %]" => "a b", {h => {a => 1, b=> 2}});
446 process_ok("[% h.list('pairs').0.items.sort.join %]" => "1 a key value", {h => {a => 1, b=> 2}});
447 process_ok("[% h.list('values').sort.join %]" => "1 2", {h => {a => 1, b=> 2}});
448 process_ok("[% h.null %]" => "", {h => {}});
449 process_ok("[% h.nsort.join %]" => "b a", {h => {a => 7, b => 2}});
450 process_ok("[% h.pairs.0.items.sort.join %]" => "1 a key value", {h => {a => 1, b=> 2}});
451 process_ok("[% h.size %]" => "2", {h => {a => 1, b=> 2}});
452 process_ok("[% h.sort.join %]" => "b a", {h => {a => "BBB", b => "A"}});
453 process_ok("[% h.values.sort.join %]" => "1 2", {h => {a => 1, b=> 2}});
454
455 ###----------------------------------------------------------------###
456 print "### more virtual methods / filters ###################################\n";
457
458 process_ok("[% [0 .. 10].reverse.1 %]" => 9) if ! $is_tt;
459 process_ok("[% {a => 'A'}.a %]" => 'A') if ! $is_tt;
460 process_ok("[% 'This is a string'.length %]" => 16) if ! $is_tt;
461 process_ok("[% 123.length %]" => 3) if ! $is_tt;
462 process_ok("[% 123.2.length %]" => 5) if ! $is_tt;
463 process_ok("[% -123.2.length %]" => -5) if ! $is_tt; # the - doesn't bind as tight as the dot methods
464 process_ok("[% (-123.2).length %]" => 6) if ! $is_tt;
465 process_ok("[% a = 23; a.0 %]" => 23) if ! $is_tt; # '0' is a scalar_op
466 process_ok('[% 1.rand %]' => qr/^0\.\d+(?:e-?\d+)?$/) if ! $is_tt;
467
468 process_ok("[% n.size %]", => 'SIZE', {n => {size => 'SIZE', a => 'A'}});
469 process_ok("[% n|size %]", => '2', {n => {size => 'SIZE', a => 'A'}}) if ! $is_tt; # tt2 | is alias for FILTER
470
471 process_ok('[% foo | eval %]' => 'baz', {foo => '[% bar %]', bar => 'baz'});
472 process_ok('[% "1" | indent(2) %]' => ' 1');
473
474
475 process_ok("[% n FILTER size %]", => '1', {n => {size => 'SIZE', a => 'A'}}) if ! $is_tt; # tt2 doesn't have size
476
477 process_ok("[% n FILTER repeat %]" => '1', {n => 1});
478 process_ok("[% n FILTER repeat(0) %]" => '', {n => 1});
479 process_ok("[% n FILTER repeat(1) %]" => '1', {n => 1});
480 process_ok("[% n FILTER repeat(2) %]" => '11', {n => 1});
481 process_ok("[% n FILTER repeat(2,'|') %]" => '1|1', {n => 1}) if ! $is_tt;
482
483 process_ok("[% n FILTER echo = repeat(2) %][% n FILTER echo %]" => '1111', {n => 1});
484 process_ok("[% n FILTER echo = repeat(2) %][% n | echo %]" => '1111', {n => 1});
485 process_ok("[% n FILTER echo = repeat(2) %][% n|echo.length %]" => '112', {n => 1}) if ! $is_tt;
486 process_ok("[% n FILTER echo = repeat(2) %][% n FILTER \$foo %]" => '1111', {n => 1, foo => 'echo'});
487 process_ok("[% n FILTER echo = repeat(2) %][% n | \$foo %]" => '1111', {n => 1, foo => 'echo'});
488 process_ok("[% n FILTER echo = repeat(2) %][% n|\$foo.length %]" => '112', {n => 1, foo => 'echo'}) if ! $is_tt;
489
490 process_ok('[% "hi" FILTER $foo %]' => 'hihi', {foo => sub {sub {$_[0]x2}}}); # filter via a passed var
491 process_ok('[% FILTER $foo %]hi[% END %]' => 'hihi', {foo => sub {sub {$_[0]x2}}}); # filter via a passed var
492 process_ok('[% "hi" FILTER foo %]' => 'hihi', {tt_config => [FILTERS => {foo => sub {$_[0]x2}}]});
493 process_ok('[% "hi" FILTER foo %]' => 'hihi', {tt_config => [FILTERS => {foo => [sub {$_[0]x2},0]}]});
494 process_ok('[% "hi" FILTER foo(2) %]' => 'hihi', {tt_config => [FILTERS => {foo => [sub {my$a=$_[1];sub{$_[0]x$a}},1]}]});
495
496 process_ok('[% ["a".."z"].pick %]' => qr/^[a-z]/) if ! $is_tt;
497
498 process_ok("[% ' ' | uri %]" => '%20');
499
500 process_ok('[% "one".fmt %]' => "one") if ! $is_tt;
501 process_ok('[% 2.fmt("%02d") %]' => "02") if ! $is_tt;
502
503 process_ok('[% [1..3].fmt %]' => "1 2 3") if ! $is_tt;
504 process_ok('[% [1..3].fmt("%02d") %]' => '01 02 03') if ! $is_tt;
505 process_ok('[% [1..3].fmt("%s", ", ") %]' => '1, 2, 3') if ! $is_tt;
506
507 process_ok('[% {a => "B", c => "D"}.fmt %]' => "a\tB\nc\tD") if ! $is_tt;
508 process_ok('[% {a => "B", c => "D"}.fmt("%s:%s") %]' => "a:B\nc:D") if ! $is_tt;
509 process_ok('[% {a => "B", c => "D"}.fmt("%s:%s", "; ") %]' => "a:B; c:D") if ! $is_tt;
510
511 process_ok('[% 1|format("%s") %]' => '1') if ! $is_tt;
512 process_ok('[% 1|format("%*s", 6) %]' => ' 1') if ! $is_tt;
513 process_ok('[% 1|format("%-*s", 6) %]' => '1 ') if ! $is_tt;
514
515 process_ok('[% 1.fmt("%-*s", 6) %]' => '1 ') if ! $is_tt;
516 process_ok('[% [1,2].fmt("%-*s", "|", 6) %]' => '1 |2 ') if ! $is_tt;
517 process_ok('[% {1=>2,3=>4}.fmt("%*s:%*s", "|", 3, 3) %]' => ' 1: 2| 3: 4') if ! $is_tt;
518
519 ###----------------------------------------------------------------###
520 print "### virtual objects ##################################################\n";
521
522 process_ok('[% a = "foobar" %][% Text.length(a) %]' => 6) if ! $is_tt;
523 process_ok('[% a = [1 .. 10] %][% List.size(a) %]' => 10) if ! $is_tt;
524 process_ok('[% a = {a=>"A", b=>"B"} ; Hash.size(a) %]' => 2) if ! $is_tt;
525
526 process_ok('[% a = Text.new("This is a string") %][% a.length %]' => 16) if ! $is_tt;
527 process_ok('[% a = List.new("one", "two", "three") %][% a.size %]' => 3) if ! $is_tt;
528 process_ok('[% a = Hash.new("one", "ONE") %][% a.one %]' => 'ONE') if ! $is_tt;
529 process_ok('[% a = Hash.new(one = "ONE") %][% a.one %]' => 'ONE') if ! $is_tt;
530 process_ok('[% a = Hash.new(one => "ONE") %][% a.one %]' => 'ONE') if ! $is_tt;
531
532 process_ok('[% {a => 1, b => 2} | Hash.keys | List.join(", ") %]' => 'a, b') if ! $is_tt;
533
534 ###----------------------------------------------------------------###
535 print "### chomping #########################################################\n";
536
537 process_ok(" [% foo %]" => ' ');
538 process_ok(" [%- foo %]" => '');
539 process_ok("\n[%- foo %]" => '');
540 process_ok("\n [%- foo %]" => '');
541 process_ok("\n\n[%- foo %]" => "\n");
542 process_ok(" \n\n[%- foo %]" => " \n");
543 process_ok(" \n[%- foo %]" => " ") if ! $is_tt;
544 process_ok(" \n \n[%- foo %]" => " \n ") if ! $is_tt;
545
546 process_ok("[% 7 %] " => '7 ');
547 process_ok("[% 7 -%] " => '7 ');
548 process_ok("[% 7 -%]\n" => '7');
549 process_ok("[% 7 -%] \n" => '7');
550 process_ok("[% 7 -%]\n " => '7 ');
551 process_ok("[% 7 -%]\n\n\n" => "7\n\n");
552 process_ok("[% 7 -%] \n " => '7 ');
553
554 ###----------------------------------------------------------------###
555 print "### string operators #################################################\n";
556
557 process_ok('[% a = "foo"; a _ "bar" %]' => 'foobar');
558 process_ok('[% a = "foo"; a ~ "bar" %]' => 'foobar') if ! $is_tt;
559 process_ok('[% a = "foo"; a ~= "bar"; a %]' => 'foobar') if ! $is_tt;
560
561 ###----------------------------------------------------------------###
562 print "### math operators ###################################################\n";
563
564 process_ok("[% 1 + 2 %]" => 3);
565 process_ok("[% 1 + 2 + 3 %]" => 6);
566 process_ok("[% (1 + 2) %]" => 3);
567 process_ok("[% 2 - 1 %]" => 1);
568 process_ok("[% -1 + 2 %]" => 1);
569 process_ok("[% -1+2 %]" => 1);
570 process_ok("[% 2 - 1 %]" => 1);
571 process_ok("[% 2-1 %]" => 1) if ! $is_tt;
572 process_ok("[% 2 - -1 %]" => 3);
573 process_ok("[% 4 * 2 %]" => 8);
574 process_ok("[% 4 / 2 %]" => 2);
575 process_ok("[% 10 / 3 %]" => qr/^3.333/);
576 process_ok("[% 10 div 3 %]" => '3');
577 process_ok("[% 2 ** 3 %]" => 8) if ! $is_tt;
578 process_ok("[% 1 + 2 * 3 %]" => 7);
579 process_ok("[% 3 * 2 + 1 %]" => 7);
580 process_ok("[% (1 + 2) * 3 %]" => 9);
581 process_ok("[% 3 * (1 + 2) %]" => 9);
582 process_ok("[% 1 + 2 ** 3 %]" => 9) if ! $is_tt;
583 process_ok("[% 2 * 2 ** 3 %]" => 16) if ! $is_tt;
584 process_ok("[% SET foo = 1 %][% foo + 2 %]" => 3);
585 process_ok("[% SET foo = 1 %][% (foo + 2) %]" => 3);
586
587 process_ok("[% a = 1; (a += 2) %]" => 3) if ! $is_tt;
588 process_ok("[% a = 1; (a -= 2) %]" => -1) if ! $is_tt;
589 process_ok("[% a = 4; (a /= 2) %]" => 2) if ! $is_tt;
590 process_ok("[% a = 1; (a *= 2) %]" => 2) if ! $is_tt;
591 process_ok("[% a = 3; (a **= 2) %]" => 9) if ! $is_tt;
592 process_ok("[% a = 1; (a %= 2) %]" => 1) if ! $is_tt;
593
594 process_ok('[% a += 1 %]-[% a %]-[% a += 1 %]-[% a %]' => '-1--2') if ! $is_tt;
595 process_ok('[% (a += 1) %]-[% (a += 1) %]' => '1-2') if ! $is_tt;
596
597 process_ok('[% a = 2; a -= 3; a %]' => '-1') if ! $is_tt;
598 process_ok('[% a = 2; a *= 3; a %]' => '6') if ! $is_tt;
599 process_ok('[% a = 2; a /= .5; a %]' => '4') if ! $is_tt;
600 process_ok('[% a = 8; a %= 3; a %]' => '2') if ! $is_tt;
601 process_ok('[% a = 2; a **= 3; a %]' => '8') if ! $is_tt;
602
603 process_ok('[% a = 1 %][% ++a %][% a %]' => '22') if ! $is_tt;
604 process_ok('[% a = 1 %][% a++ %][% a %]' => '12') if ! $is_tt;
605 process_ok('[% a = 1 %][% --a %][% a %]' => '00') if ! $is_tt;
606 process_ok('[% a = 1 %][% a-- %][% a %]' => '10') if ! $is_tt;
607 process_ok('[% a++ FOR [1..3] %]' => '012') if ! $is_tt;
608 process_ok('[% --a FOR [1..3] %]' => '-1-2-3') if ! $is_tt;
609
610 ###----------------------------------------------------------------###
611 print "### boolean operators ################################################\n";
612
613 process_ok("[% 5 && 6 %]" => 6);
614 process_ok("[% 5 || 6 %]" => 5);
615 process_ok("[% 0 || 6 %]" => 6);
616 process_ok("[% 0 && 6 %]" => 0);
617 process_ok("[% 0 && 0 %]" => 0);
618 process_ok("[% 5 && 6 && 7%]" => 7);
619 process_ok("[% 0 || 1 || 2 %]" => 1);
620
621 process_ok("[% 5 + (0 || 5) %]" => 10);
622
623
624 process_ok("[% 1 ? 2 : 3 %]" => '2');
625 process_ok("[% 0 ? 2 : 3 %]" => '3');
626 process_ok("[% 0 ? (1 ? 2 : 3) : 4 %]" => '4');
627 process_ok("[% 0 ? 1 ? 2 : 3 : 4 %]" => '4');
628
629 process_ok("[% t = 1 || 0 ? 3 : 4 %][% t %]" => 3);
630 process_ok("[% t = 0 or 1 ? 3 : 4 %][% t %]" => 3);
631 process_ok("[% t = 1 or 0 ? 3 : 4 %][% t %]" => 1) if ! $is_tt;
632
633 process_ok("[% 0 ? 2 : 3 %]" => '3');
634 process_ok("[% 1 ? 2 : 3 %]" => '2');
635 process_ok("[% 0 ? 1 ? 2 : 3 : 4 %]" => '4');
636 process_ok("[% t = 0 ? 1 ? [1..4] : [2..4] : [3..4] %][% t.0 %]" => '3');
637 process_ok("[% t = 1 || 0 ? 0 : 1 || 2 ? 2 : 3 %][% t %]" => '0');
638 process_ok("[% t = 0 or 0 ? 0 : 1 or 2 ? 2 : 3 %][% t %]" => '1') if ! $is_tt;
639 process_ok("[% t = 0 or 0 ? 0 : 0 or 2 ? 2 : 3 %][% t %]" => '2');
640
641 process_ok("[% 0 ? 1 ? 1 + 2 * 3 : 1 + 2 * 4 : 1 + 2 * 5 %]" => '11');
642
643 ###----------------------------------------------------------------###
644 print "### regex ############################################################\n";
645
646 process_ok("[% /foo/ %]" => '(?-xism:foo)') if ! $is_tt;
647 process_ok("[% /foo %]" => '') if ! $is_tt;
648 process_ok("[% /foo/x %]" => '(?-xism:(?x:foo))') if ! $is_tt;
649 process_ok("[% /foo/xi %]" => '(?-xism:(?xi:foo))') if ! $is_tt;
650 process_ok("[% /foo/xis %]" => '(?-xism:(?xis:foo))') if ! $is_tt;
651 process_ok("[% /foo/xism %]" => '(?-xism:(?xism:foo))') if ! $is_tt;
652 process_ok("[% /foo/e %]" => '') if ! $is_tt;
653 process_ok("[% /foo/g %]" => '') if ! $is_tt;
654 process_ok("[% /foo %]" => '') if ! $is_tt;
655 process_ok("[% /foo**/ %]" => '') if ! $is_tt;
656 process_ok("[% /fo\\/o/ %]" => '(?-xism:fo/o)') if ! $is_tt;
657 process_ok("[% 'foobar'.match(/(f\\w\\w)/).0 %]" => 'foo') if ! $is_tt;
658
659 ###----------------------------------------------------------------###
660 print "### BLOCK / PROCESS / INCLUDE#########################################\n";
661
662 process_ok("[% PROCESS foo %]one" => '');
663 process_ok("[% BLOCK foo %]one" => '');
664 process_ok("[% BLOCK foo %][% END %]one" => 'one');
665 process_ok("[% BLOCK %][% END %]one" => 'one');
666 process_ok("[% BLOCK foo %]hi there[% END %]one" => 'one');
667 process_ok("[% BLOCK foo %][% BLOCK foo %][% END %][% END %]" => '');
668 process_ok("[% BLOCK foo %]hi there[% END %][% PROCESS foo %]" => 'hi there');
669 process_ok("[% PROCESS foo %][% BLOCK foo %]hi there[% END %]" => 'hi there');
670 process_ok("[% BLOCK foo %]hi there[% END %][% PROCESS foo foo %]" => 'hi therehi there') if ! $is_tt;
671 process_ok("[% BLOCK foo %]hi there[% END %][% PROCESS foo, foo %]" => 'hi therehi there') if ! $is_tt;
672 process_ok("[% BLOCK foo %]hi there[% END %][% PROCESS foo + foo %]" => 'hi therehi there');
673 process_ok("[% BLOCK foo %]hi [% one %] there[% END %][% PROCESS foo %]" => 'hi ONE there', {one => 'ONE'});
674 process_ok("[% BLOCK foo %]hi [% IF 1 %]Yes[% END %] there[% END %]<<[% PROCESS foo %]>>" => '<<hi Yes there>>');
675 process_ok("[% BLOCK foo %]hi [% one %] there[% END %][% PROCESS foo one = 'two' %]" => 'hi two there');
676 process_ok("[% BLOCK foo %]hi [% one.two %] there[% END %][% PROCESS foo one.two = 'two' %]" => 'hi two there');
677 process_ok("[% BLOCK foo %]hi [% one.two %] there[% END %][% PROCESS foo + foo one.two = 'two' %]" => 'hi two there'x2);
678 process_ok("[% BLOCK foo %][% BLOCK bar %]hi [% one %] there[% END %][% END %][% PROCESS foo/bar one => 'two' %]" => 'hi two there');
679
680 process_ok("[% BLOCK foo %]hi [% one %] there[% END %][% PROCESS foo one = 'two' %][% one %]" => 'hi two theretwo');
681 process_ok("[% BLOCK foo %]hi [% one %] there[% END %][% INCLUDE foo one = 'two' %][% one %]" => 'hi two there');
682
683 ###----------------------------------------------------------------###
684 print "### IF / UNLESS / ELSIF / ELSE #######################################\n";
685
686 process_ok("[% IF 1 %]Yes[% END %]" => 'Yes');
687 process_ok("[% IF 0 %]Yes[% END %]" => '');
688 process_ok("[% IF 0 %]Yes[% ELSE %]No[% END %]" => 'No');
689 process_ok("[% IF 0 %]Yes[% ELSIF 1 %]No[% END %]" => 'No');
690 process_ok("[% IF 0 %]Yes[% ELSIF 0 %]No[% END %]" => '');
691 process_ok("[% IF 0 %]Yes[% ELSIF 0 %]No[% ELSE %]hmm[% END %]" => 'hmm');
692
693 process_ok("[% UNLESS 1 %]Yes[% END %]" => '');
694 process_ok("[% UNLESS 0 %]Yes[% END %]" => 'Yes');
695 process_ok("[% UNLESS 0 %]Yes[% ELSE %]No[% END %]" => 'Yes');
696 process_ok("[% UNLESS 1 %]Yes[% ELSIF 1 %]No[% END %]" => 'No');
697 process_ok("[% UNLESS 1 %]Yes[% ELSIF 0 %]No[% END %]" => '');
698 process_ok("[% UNLESS 1 %]Yes[% ELSIF 0 %]No[% ELSE %]hmm[% END %]" => 'hmm');
699
700 ###----------------------------------------------------------------###
701 print "### comments #########################################################\n";
702
703 process_ok("[%# one %]" => '', {one => 'ONE'});
704 process_ok("[%#\n one %]" => '', {one => 'ONE'});
705 process_ok("[%-#\n one %]" => '', {one => 'ONE'}) if ! $is_tt;
706 process_ok("[% #\n one %]" => 'ONE', {one => 'ONE'});
707 process_ok("[%# BLOCK one %]" => '');
708 process_ok("[%# BLOCK one %]two" => 'two');
709 process_ok("[%# BLOCK one %]two[% END %]" => '');
710 process_ok("[%# BLOCK one %]two[% END %]three" => '');
711 process_ok("[%
712 #
713 -%]
714 foo" => "foo");
715
716 ###----------------------------------------------------------------###
717 print "### FOREACH / NEXT / LAST ############################################\n";
718
719 process_ok("[% FOREACH foo %]" => '');
720 process_ok("[% FOREACH foo %][% END %]" => '');
721 process_ok("[% FOREACH foo %]bar[% END %]" => '');
722 process_ok("[% FOREACH foo %]bar[% END %]" => 'bar', {foo => 1});
723 process_ok("[% FOREACH f IN foo %]bar[% f %][% END %]" => 'bar1bar2', {foo => [1,2]});
724 process_ok("[% FOREACH f = foo %]bar[% f %][% END %]" => 'bar1bar2', {foo => [1,2]});
725 process_ok("[% FOREACH f = [1,2] %]bar[% f %][% END %]" => 'bar1bar2');
726 process_ok("[% FOREACH f = [1..3] %]bar[% f %][% END %]" => 'bar1bar2bar3');
727 process_ok("[% FOREACH f = [{a=>'A'},{a=>'B'}] %]bar[% f.a %][% END %]" => 'barAbarB');
728 process_ok("[% FOREACH [{a=>'A'},{a=>'B'}] %]bar[% a %][% END %]" => 'barAbarB');
729 process_ok("[% FOREACH [{a=>'A'},{a=>'B'}] %]bar[% a %][% END %][% a %]" => 'barAbarB');
730 process_ok("[% FOREACH f = [1..3] %][% loop.count %]/[% loop.size %] [% END %]" => '1/3 2/3 3/3 ');
731 process_ok("[% FOREACH f = [1..3] %][% IF loop.first %][% f %][% END %][% END %]" => '1');
732 process_ok("[% FOREACH f = [1..3] %][% IF loop.last %][% f %][% END %][% END %]" => '3');
733 process_ok("[% FOREACH f = [1..3] %][% IF loop.first %][% NEXT %][% END %][% f %][% END %]" => '23');
734 process_ok("[% FOREACH f = [1..3] %][% IF loop.first %][% LAST %][% END %][% f %][% END %]" => '');
735 process_ok("[% FOREACH f = [1..3] %][% f %][% IF loop.first %][% NEXT %][% END %][% END %]" => '123');
736 process_ok("[% FOREACH f = [1..3] %][% f %][% IF loop.first %][% LAST %][% END %][% END %]" => '1');
737
738 process_ok('[% a = ["Red", "Blue"] ; FOR [0..3] ; a.${ loop.index % a.size } ; END %]' => 'RedBlueRedBlue') if ! $is_tt;
739
740 ### TT is not consistent in what is localized - well it is documented
741 ### if you set a variable in the FOREACH tag, then nothing in the loop gets localized
742 ### if you don't set a variable - everything gets localized
743 process_ok("[% foo = 1 %][% FOREACH [1..10] %][% foo %][% foo = 2 %][% END %]" => '1222222222');
744 process_ok("[% f = 1 %][% FOREACH i = [1..10] %][% i %][% f = 2 %][% END %][% f %]" => '123456789102');
745 process_ok("[% f = 1 %][% FOREACH [1..10] %][% f = 2 %][% END %][% f %]" => '1');
746 process_ok("[% f = 1 %][% FOREACH f = [1..10] %][% f %][% END %][% f %]" => '1234567891010');
747 process_ok("[% FOREACH [1] %][% SET a = 1 %][% END %][% a %]" => '');
748 process_ok("[% a %][% FOREACH [1] %][% SET a = 1 %][% END %][% a %]" => '');
749 process_ok("[% a = 2 %][% FOREACH [1] %][% SET a = 1 %][% END %][% a %]" => '2');
750 process_ok("[% a = 2 %][% FOREACH [1] %][% a = 1 %][% END %][% a %]" => '2');
751 process_ok("[% a = 2 %][% FOREACH i = [1] %][% a = 1 %][% END %][% a %]" => '1');
752 process_ok("[% FOREACH i = [1] %][% SET a = 1 %][% END %][% a %]" => '1');
753 process_ok("[% f.b = 1 %][% FOREACH f.b = [1..10] %][% f.b %][% END %][% f.b %]" => '1234567891010') if ! $is_tt;
754 process_ok("[% a = 1 %][% FOREACH [{a=>'A'},{a=>'B'}] %]bar[% a %][% END %][% a %]" => 'barAbarB1');
755 process_ok("[% FOREACH [1..3] %][% loop.size %][% END %][% loop.size %]" => '333');
756 process_ok("[% FOREACH i = [1..3] %][% loop.size %][% END %][% loop.size %]" => '333') if ! $is_tt;
757 process_ok("[% FOREACH i = [1..3] %][% loop.size %][% END %][% loop.size %]" => '3331') if $is_tt;
758
759 process_ok('[% FOREACH f = [1..3]; 1; END %]' => '111');
760 process_ok('[% FOREACH f = [1..3]; f; END %]' => '123');
761 process_ok('[% FOREACH f = [1..3]; "$f"; END %]' => '123');
762 process_ok('[% FOREACH f = [1..3]; f + 1; END %]' => '234');
763
764 ###----------------------------------------------------------------###
765 print "### WHILE ############################################################\n";
766
767 process_ok("[% WHILE foo %]" => '');
768 process_ok("[% WHILE foo %][% END %]" => '');
769 process_ok("[% WHILE (foo = foo - 1) %][% END %]" => '');
770 process_ok("[% WHILE (foo = foo - 1) %][% foo %][% END %]" => '21', {foo => 3});
771 process_ok("[% WHILE foo %][% foo %][% foo = foo - 1 %][% END %]" => '321', {foo => 3});
772
773 process_ok("[% WHILE 1 %][% foo %][% foo = foo - 1 %][% LAST IF foo == 1 %][% END %]" => '32', {foo => 3});
774 process_ok("[% f = 10; WHILE f; f = f - 1 ; f ; END %]" => '9876543210');
775 process_ok("[% f = 10; WHILE f; f = f - 1 ; f ; END ; f %]" => '98765432100');
776 process_ok("[% f = 10 a = 2; WHILE f; f = f - 1 ; f ; a=3; END ; a%]" => '98765432103');
777
778 process_ok("[% f = 10; WHILE (g=f); f = f - 1 ; f ; END %]" => '9876543210');
779 process_ok("[% f = 10; WHILE (g=f); f = f - 1 ; f ; END ; f %]" => '98765432100');
780 process_ok("[% f = 10 a = 2; WHILE (g=f); f = f - 1 ; f ; a=3; END ; a%]" => '98765432103');
781 process_ok("[% f = 10 a = 2; WHILE (a=f); f = f - 1 ; f ; a=3; END ; a%]" => '98765432100');
782
783 ###----------------------------------------------------------------###
784 print "### STOP / RETURN / CLEAR ############################################\n";
785
786 process_ok("[% STOP %]" => '');
787 process_ok("One[% STOP %]Two" => 'One');
788 process_ok("[% BLOCK foo %]One[% STOP %]Two[% END %]First[% PROCESS foo %]Last" => 'FirstOne');
789 process_ok("[% FOREACH f = [1..3] %][% f %][% IF loop.first %][% STOP %][% END %][% END %]" => '1');
790 process_ok("[% FOREACH f = [1..3] %][% IF loop.first %][% STOP %][% END %][% f %][% END %]" => '');
791
792 process_ok("[% RETURN %]" => '');
793 process_ok("One[% RETURN %]Two" => 'One');
794 process_ok("[% BLOCK foo %]One[% RETURN %]Two[% END %]First[% PROCESS foo %]Last" => 'FirstOneLast');
795 process_ok("[% FOREACH f = [1..3] %][% f %][% IF loop.first %][% RETURN %][% END %][% END %]" => '1');
796 process_ok("[% FOREACH f = [1..3] %][% IF loop.first %][% RETURN %][% END %][% f %][% END %]" => '');
797
798 process_ok("[% CLEAR %]" => '');
799 process_ok("One[% CLEAR %]Two" => 'Two');
800 process_ok("[% BLOCK foo %]One[% CLEAR %]Two[% END %]First[% PROCESS foo %]Last" => 'FirstTwoLast');
801 process_ok("[% FOREACH f = [1..3] %][% f %][% IF loop.first %][% CLEAR %][% END %][% END %]" => '23');
802 process_ok("[% FOREACH f = [1..3] %][% IF loop.first %][% CLEAR %][% END %][% f %][% END %]" => '123');
803 process_ok("[% FOREACH f = [1..3] %][% f %][% IF loop.last %][% CLEAR %][% END %][% END %]" => '');
804 process_ok("[% FOREACH f = [1..3] %][% IF loop.last %][% CLEAR %][% END %][% f %][% END %]" => '3');
805
806 ###----------------------------------------------------------------###
807 print "### post opererative directives ######################################\n";
808
809 process_ok("[% GET foo IF 1 %]" => '1', {foo => 1});
810 process_ok("[% f FOREACH f = [1..3] %]" => '123');
811
812 process_ok("2[% GET foo IF 1 IF 2 %]" => '21', {foo => 1}) if ! $is_tt;
813 process_ok("2[% GET foo IF 1 IF 0 %]" => '2', {foo => 1}) if ! $is_tt;
814 process_ok("[% f FOREACH f = [1..3] IF 1 %]" => '123') if ! $is_tt;
815 process_ok("[% f FOREACH f = [1..3] IF 0 %]" => '') if ! $is_tt;
816 process_ok("[% f FOREACH f = g FOREACH g = [1..3] %]" => '123') if ! $is_tt;
817 process_ok("[% f FOREACH f = g.a FOREACH g = [{a=>1}, {a=>2}, {a=>3}] %]" => '123') if ! $is_tt;
818 process_ok("[% f FOREACH f = a FOREACH [{a=>1}, {a=>2}, {a=>3}] %]" => '123') if ! $is_tt;
819
820 process_ok("[% FOREACH f = [1..3] IF 1 %]([% f %])[% END %]" => '(1)(2)(3)') if ! $is_tt;
821 process_ok("[% FOREACH f = [1..3] IF 0 %]([% f %])[% END %]" => '') if ! $is_tt;
822
823 process_ok("[% BLOCK bar %][% foo %][% foo = foo - 1 %][% END %][% PROCESS bar WHILE foo %]" => '321', {foo => 3});
824
825 ###----------------------------------------------------------------###
826 print "### capturing ########################################################\n";
827
828 process_ok("[% foo = BLOCK %]Hi[% END %][% foo %][% foo %]" => 'HiHi');
829 process_ok("[% BLOCK foo %]Hi[% END %][% bar = PROCESS foo %]-[% bar %]" => '-Hi');
830 process_ok("[% foo = IF 1 %]Hi[% END %][% foo %]" => 'Hi');
831
832 ###----------------------------------------------------------------###
833 print "### TAGS #############################################################\n";
834
835 process_ok("[% TAGS asp %]<% 1 + 2 %>" => 3);
836 process_ok("[% TAGS default %][% 1 + 2 %]" => 3);
837 process_ok("[% TAGS html %]<!-- 1 + 2 -->" => '3');
838 process_ok("[% TAGS mason %]<% 1 + 2 >" => 3);
839 process_ok("[% TAGS metatext %]%% 1 + 2 %%" => 3);
840 process_ok("[% TAGS php %]<? 1 + 2 ?>" => 3);
841 process_ok("[% TAGS star %][* 1 + 2 *]" => 3);
842 process_ok("[% TAGS template %][% 1 + 2 %]" => 3);
843 process_ok("[% TAGS template1 %][% 1 + 2 %]" => 3);
844 process_ok("[% TAGS template1 %]%% 1 + 2 %%" => 3);
845 process_ok("[% TAGS tt2 %][% 1 + 2 %]" => 3);
846
847 process_ok("[% TAGS html %] <!--- 1 + 2 -->" => '3');
848 process_ok("[% TAGS html %]<!-- 1 + 2 --->" => '3') if ! $is_tt;
849 process_ok("[% TAGS html %]<!-- 1 + 2 --->\n" => '3');
850 process_ok("[% BLOCK foo %][% TAGS html %]<!-- 1 + 2 --><!-- END --><!-- PROCESS foo --> <!-- 1 + 2 -->" => '3 3');
851 process_ok("[% BLOCK foo %][% TAGS html %]<!-- 1 + 2 -->[% END %][% PROCESS foo %] [% 1 + 2 %]" => '');
852
853 process_ok("[% TAGS <!-- --> %]<!-- 1 + 2 -->" => '3');
854
855 process_ok("[% TAGS [<] [>] %][<] 1 + 2 [>]" => 3);
856 process_ok("[% TAGS '[<]' '[>]' %][<] 1 + 2 [>]" => 3) if ! $is_tt;
857 process_ok("[% TAGS /[<]/ /[>]/ %]< 1 + 2 >" => 3) if ! $is_tt;
858 process_ok("[% TAGS ** ** %]** 1 + 2 **" => 3);
859 process_ok("[% TAGS '**' '**' %]** 1 + 2 **" => 3) if ! $is_tt;
860 process_ok("[% TAGS /**/ /**/ %]** 1 + 2 **" => "") if ! $is_tt;
861
862 process_ok("[% TAGS html --><!-- 1 + 2 -->" => '3') if ! $is_tt;
863 process_ok("[% TAGS html ; 7 --><!-- 1 + 2 -->" => '73') if ! $is_tt;
864 process_ok("[% TAGS html ; 7 %]<!-- 1 + 2 -->" => '') if ! $is_tt; # error - the old closing tag must come next
865
866 ###----------------------------------------------------------------###
867 print "### SWITCH / CASE ####################################################\n";
868
869 process_ok("[% SWITCH 1 %][% END %]hi" => 'hi');
870 process_ok("[% SWITCH 1 %][% CASE %]bar[% END %]hi" => 'barhi');
871 process_ok("[% SWITCH 1 %]Pre[% CASE %]bar[% END %]hi" => 'barhi');
872 process_ok("[% SWITCH 1 %][% CASE DEFAULT %]bar[% END %]hi" => 'barhi');
873 process_ok("[% SWITCH 1 %][% CASE 0 %]bar[% END %]hi" => 'hi');
874 process_ok("[% SWITCH 1 %][% CASE 1 %]bar[% END %]hi" => 'barhi');
875 process_ok("[% SWITCH 1 %][% CASE foo %][% CASE 1 %]bar[% END %]hi" => 'barhi');
876 process_ok("[% SWITCH 1 %][% CASE [1..10] %]bar[% END %]hi" => 'barhi');
877 process_ok("[% SWITCH 11 %][% CASE [1..10] %]bar[% END %]hi" => 'hi');
878
879 process_ok("[% SWITCH 1.0 %][% CASE [1..10] %]bar[% END %]hi" => 'barhi');
880 process_ok("[% SWITCH '1.0' %][% CASE [1..10] %]bar[% END %]hi" => 'barhi') if ! $is_tt;
881
882 ###----------------------------------------------------------------###
883 print "### TRY / THROW / CATCH / FINAL ######################################\n";
884
885 process_ok("[% TRY %][% END %]hi" => 'hi');
886 process_ok("[% TRY %]Foo[% END %]hi" => 'Foohi');
887 process_ok("[% TRY %]Foo[% THROW foo 'for fun' %]bar[% END %]hi" => '');
888 process_ok("[% TRY %]Foo[% THROW foo 'for fun' %]bar[% CATCH %][% END %]hi" => 'Foohi') if ! $is_tt;
889 process_ok("[% TRY %]Foo[% THROW foo 'for fun' %]bar[% CATCH %]there[% END %]hi" => 'Footherehi');
890 process_ok("[% TRY %]Foo[% THROW foo 'for fun' %]bar[% CATCH foo %]there[% END %]hi" => 'Footherehi');
891 process_ok("[% TRY %]Foo[% TRY %]Foo[% THROW foo 'for fun' %][% CATCH bar %]one[% END %][% CATCH %]two[% END %]hi" => 'FooFootwohi');
892 process_ok("[% TRY %]Foo[% TRY %]Foo[% THROW foo 'for fun' %][% CATCH bar %]one[% END %][% CATCH s %]two[% END %]hi" => '');
893 process_ok("[% TRY %]Foo[% THROW foo.bar 'for fun' %][% CATCH foo %]one[% CATCH foo.bar %]two[% END %]hi" => 'Footwohi');
894
895 process_ok("[% TRY %]Foo[% FINAL %]Bar[% END %]hi" => 'FooBarhi');
896 process_ok("[% TRY %]Foo[% THROW foo %][% FINAL %]Bar[% CATCH %]one[% END %]hi" => '');
897 process_ok("[% TRY %]Foo[% THROW foo %][% CATCH %]one[% FINAL %]Bar[% END %]hi" => 'FoooneBarhi');
898 process_ok("[% TRY %]Foo[% THROW foo %][% CATCH bar %]one[% FINAL %]Bar[% END %]hi" => '');
899
900 process_ok("[% TRY %][% THROW foo 'bar' %][% CATCH %][% error %][% END %]" => 'foo error - bar');
901 process_ok("[% TRY %][% THROW foo 'bar' %][% CATCH %][% error.type %][% END %]" => 'foo');
902 process_ok("[% TRY %][% THROW foo 'bar' %][% CATCH %][% error.info %][% END %]" => 'bar');
903 process_ok("[% TRY %][% THROW foo %][% CATCH %][% error.type %][% END %]" => 'undef');
904 process_ok("[% TRY %][% THROW foo %][% CATCH %][% error.info %][% END %]" => 'foo');
905
906 ###----------------------------------------------------------------###
907 print "### named args #######################################################\n";
908
909 process_ok("[% foo(bar = 'one', baz = 'two') %]" => "baronebaztwo",
910 {foo=>sub{my $n=$_[-1];join('',map{"$_$n->{$_}"} sort keys %$n)}});
911 process_ok("[%bar='ONE'%][% foo(\$bar = 'one') %]" => "ONEone",
912 {foo=>sub{my $n=$_[-1];join('',map{"$_$n->{$_}"} sort keys %$n)}});
913
914 ###----------------------------------------------------------------###
915 print "### USE ##############################################################\n";
916
917 my @config_p = (PLUGIN_BASE => 'MyTestPlugin', LOAD_PERL => 1);
918 process_ok("[% USE son_of_gun_that_does_not_exist %]one" => '', {tt_config => \@config_p});
919 process_ok("[% USE Foo %]one" => 'one', {tt_config => \@config_p});
920 process_ok("[% USE Foo2 %]one" => 'one', {tt_config => \@config_p});
921 process_ok("[% USE Foo(bar = 'baz') %]one[% Foo.bar %]" => 'onebarbaz', {tt_config => \@config_p});
922 process_ok("[% USE Foo2(bar = 'baz') %]one[% Foo2.bar %]" => 'onebarbaz', {tt_config => \@config_p});
923 process_ok("[% USE Foo(bar = 'baz') %]one[% Foo.bar %]" => 'onebarbaz', {tt_config => \@config_p});
924 process_ok("[% USE d = Foo(bar = 'baz') %]one[% d.bar %]" => 'onebarbaz', {tt_config => \@config_p});
925 process_ok("[% USE d.d = Foo(bar = 'baz') %]one[% d.d.bar %]" => '', {tt_config => \@config_p});
926
927 process_ok("[% USE a(bar = 'baz') %]one[% a.seven %]" => '', {tt_config => [@config_p, PLUGINS => {a=>'Foo'}, ]});
928 process_ok("[% USE a(bar = 'baz') %]one[% a.seven %]" => 'one7', {tt_config => [@config_p, PLUGINS => {a=>'Foo2'},]});
929
930 @config_p = (PLUGIN_BASE => ['NonExistant', 'MyTestPlugin'], LOAD_PERL => 1);
931 process_ok("[% USE Foo %]one" => 'one', {tt_config => \@config_p});
932
933 ###----------------------------------------------------------------###
934 print "### MACRO ############################################################\n";
935
936 process_ok("[% MACRO foo PROCESS bar %][% BLOCK bar %]Hi[% END %][% foo %]" => 'Hi');
937 process_ok("[% MACRO foo BLOCK %]Hi[% END %][% foo %]" => 'Hi');
938 process_ok("[% MACRO foo BLOCK %]Hi[% END %][% foo %]" => 'Hi');
939 process_ok("[% MACRO foo(n) BLOCK %]Hi[% n %][% END %][% foo(2) %]" => 'Hi2');
940 process_ok("[%n=1%][% MACRO foo(n) BLOCK %]Hi[% n %][% END %][% foo(2) %][%n%]" => 'Hi21');
941 process_ok("[%n=1%][% MACRO foo BLOCK %]Hi[% n = 2%][% END %][% foo %][%n%]" => 'Hi1');
942 process_ok("[% MACRO foo(n) FOREACH i=[1..n] %][% i %][% END %][% foo(3) %]" => '123');
943
944 process_ok('[% MACRO f BLOCK %]>[% TRY; f ; CATCH ; "caught" ; END %][% END %][% f %]' => '>>>caught', {tt_config => [MAX_MACRO_RECURSE => 3]}) if ! $is_tt;
945
946 ###----------------------------------------------------------------###
947 print "### DEBUG ############################################################\n";
948
949 process_ok("\n\n[% one %]" => "\n\n\n## input text line 3 : [% one %] ##\nONE", {one=>'ONE', tt_config => ['DEBUG' => 8]});
950 process_ok("[% one %]" => "\n## input text line 1 : [% one %] ##\nONE", {one=>'ONE', tt_config => ['DEBUG' => 8]});
951 process_ok("[% one %]\n\n" => "(1)ONE\n\n", {one=>'ONE', tt_config => ['DEBUG' => 8, 'DEBUG_FORMAT' => '($line)']});
952 process_ok("1\n2\n3[% one %]" => "1\n2\n3(3)ONE", {one=>'ONE', tt_config => ['DEBUG' => 8, 'DEBUG_FORMAT' => '($line)']});
953 process_ok("[% one;\n one %]" => "(1)ONE(2)ONE", {one=>'ONE', tt_config => ['DEBUG' => 8,
954 'DEBUG_FORMAT' => '($line)']}) if ! $is_tt;
955 process_ok("[% DEBUG format '(\$line)' %][% one %]" => qr/\(1\)/, {one=>'ONE', tt_config => ['DEBUG' => 8]});
956
957 process_ok("[% TRY %][% abc %][% CATCH %][% error %][% END %]" => "undef error - abc is undefined\n", {tt_config => ['DEBUG' => 2]});
958 process_ok("[% TRY %][% abc.def %][% CATCH %][% error %][% END %]" => "undef error - def is undefined\n", {abc => {}, tt_config => ['DEBUG' => 2]});
959
960 ###----------------------------------------------------------------###
961 print "### constants ########################################################\n";
962
963 my @config_c = (
964 CONSTANTS => {
965 harry => sub {'do_this_once'},
966 foo => {
967 bar => {baz => 42},
968 bim => 57,
969 },
970 bing => 'baz',
971 bang => 'bim',
972 },
973 VARIABLES => {
974 bam => 'bar',
975 },
976 );
977 process_ok("[% constants.harry %]" => 'do_this_once', {constants => {harry => 'foo'}, tt_config => \@config_c});
978 process_ok("[% constants.harry.length %]" => '12', {tt_config => \@config_c});
979 process_ok("[% SET constants.something = 1 %][% constants.something %]one" => '1one', {tt_config => \@config_c});
980 process_ok("[% SET constants.harry = 1 %][% constants.harry %]one" => 'do_this_onceone', {tt_config => \@config_c});
981 process_ok("[% constants.foo.\${constants.bang} %]" => '57', {tt_config => [@config_c]});
982 process_ok("[% constants.foo.\$bam.\${constants.bing} %]" => '42', {tt_config => [@config_c]}) if ! $is_tt;
983 process_ok("[% bam = 'somethingelse' %][% constants.foo.\$bam.\${constants.bing} %]" => '42', {tt_config => [@config_c]}) if ! $is_tt;
984
985 process_ok('[% constants.${"harry"} %]' => 'do_this_once', {constants => {harry => 'foo'}, tt_config => \@config_c});
986 process_ok('[% ${"constants"}.harry %]' => 'foo', {constants => {harry => 'foo'}, tt_config => \@config_c}) if ! $is_tt;
987 process_ok('[% ${"constants"}.harry %]' => 'do_this_once', {constants => {harry => 'foo'}, tt_config => \@config_c}) if $is_tt;
988 process_ok('[% ${"con${\\"s\\"}tants"}.harry %]' => 'foo', {constants => {harry => 'foo'}, tt_config => \@config_c}) if ! $is_tt;
989
990 ###----------------------------------------------------------------###
991 print "### INTERPOLATE / ANYCASE / TRIM #####################################\n";
992
993 process_ok("Foo \$one Bar" => 'Foo ONE Bar', {one => 'ONE', tt_config => ['INTERPOLATE' => 1]});
994 process_ok("[% PERL %] my \$n=7; print \$n [% END %]" => '7', {tt_config => ['INTERPOLATE' => 1, 'EVAL_PERL' => 1]});
995 process_ok("[% TRY ; PERL %] my \$n=7; print \$n [% END ; END %]" => '7', {tt_config => ['INTERPOLATE' => 1, 'EVAL_PERL' => 1]});
996
997 process_ok("[% GET %]" => '', {GET => 'ONE'});
998 process_ok("[% GET GET %]" => 'ONE', {GET => 'ONE'}) if ! $is_tt;
999 process_ok("[% get one %]" => 'ONE', {one => 'ONE', tt_config => ['ANYCASE' => 1]});
1000 process_ok("[% get %]" => '', {get => 'ONE', tt_config => ['ANYCASE' => 1]});
1001 process_ok("[% get get %]" => 'ONE', {get => 'ONE', tt_config => ['ANYCASE' => 1]}) if ! $is_tt;
1002
1003 process_ok("[% BLOCK foo %]\nhi\n[% END %][% PROCESS foo %]" => "\nhi\n");
1004 process_ok("[% BLOCK foo %]\nhi[% END %][% PROCESS foo %]" => "hi", {tt_config => [TRIM => 1]});
1005 process_ok("[% BLOCK foo %]hi\n[% END %][% PROCESS foo %]" => "hi", {tt_config => [TRIM => 1]});
1006 process_ok("[% BLOCK foo %]hi[% nl %][% END %][% PROCESS foo %]" => "hi", {nl => "\n", tt_config => [TRIM => 1]});
1007 process_ok("[% BLOCK foo %][% nl %]hi[% END %][% PROCESS foo %]" => "hi", {nl => "\n", tt_config => [TRIM => 1]});
1008 process_ok("A[% TRY %]\nhi\n[% END %]" => "A\nhi", {tt_config => [TRIM => 1]});
1009
1010 ###----------------------------------------------------------------###
1011 print "### V1DOLLAR #########################################################\n";
1012
1013 process_ok('[% a %]|[% $a %]|[% ${ a } %]|[% ${ "a" } %]' => 'A|bar|bar|A', {a => 'A', A => 'bar'});
1014 process_ok('[% a %]|[% $a %]|[% ${ a } %]|[% ${ "a" } %]' => 'A|A|bar|A', {a => 'A', A => 'bar', tt_config => [V1DOLLAR => 1]});
1015
1016 $vars = {a => {b => {c=>'Cb'}, B => {c=>'CB'}}, b => 'B', Cb => 'bar', CB => 'Bar'};
1017 process_ok('[% a.b.c %]|[% $a.b.c %]|[% a.$b.c %]|[% ${ a.b.c } %]' => 'Cb||CB|bar', $vars);
1018 process_ok('[% a.b.c %]|[% $a.b.c %]|[% a.$b.c %]|[% ${ a.b.c } %]' => 'Cb|Cb|Cb|bar', {%$vars, tt_config => [V1DOLLAR => 1]});
1019
1020 process_ok('[% "$a" %]|$a|[% "${a}" %]|${a}' => 'A|$a|A|${a}', {a => 'A', A => 'bar'});
1021 process_ok('[% "$a" %]|$a|[% "${a}" %]|${a}' => 'A|$a|A|${a}', {a => 'A', A => 'bar', tt_config => [V1DOLLAR => 1]});
1022 process_ok('[% "$a" %]|$a|[% "${a}" %]|${a}' => 'A|A|A|A', {a => 'A', A => 'bar', tt_config => [INTERPOLATE => 1]});
1023 process_ok('[% "$a" %]|$a|[% "${a}" %]|${a}' => 'A|A|A|A', {a => 'A', A => 'bar', tt_config => [V1DOLLAR => 1, INTERPOLATE => 1]});
1024
1025 process_ok('[% constants.a %]|[% $constants.a %]|[% constants.$a %]' => 'A|A|A', {tt_config => [V1DOLLAR => 1, CONSTANTS => {a => 'A'}]});
1026
1027 ###----------------------------------------------------------------###
1028 print "### V2PIPE ###########################################################\n";
1029
1030 process_ok("[%- BLOCK a %]b is [% b %]
1031 [% END %]
1032 [%- PROCESS a b => 237 | repeat(2) %]" => "b is 237
1033 b is 237\n", {tt_config => [V2PIPE => 1]});
1034
1035 process_ok("[%- BLOCK a %]b is [% b %]
1036 [% END %]
1037 [%- PROCESS a b => 237 | repeat(2) %]" => "b is 237237\n") if ! $is_tt;
1038
1039 ###----------------------------------------------------------------###
1040 print "### configuration ####################################################\n";
1041
1042 process_ok('[% a = 7 %]$a' => 7, {tt_config => ['INTERPOLATE' => 1]});
1043 process_ok('[% a = 7 %]$a' => 7, {tt_config => ['interpolate' => 1]}) if ! $is_tt;
1044
1045 ###----------------------------------------------------------------###
1046 print "### PERL #############################################################\n";
1047
1048 process_ok("[% TRY %][% PERL %][% END %][% CATCH ; error; END %]" => 'perl error - EVAL_PERL not set');
1049 process_ok("[% PERL %] print \"[% one %]\" [% END %]" => 'ONE', {one => 'ONE', tt_config => ['EVAL_PERL' => 1]});
1050 process_ok("[% PERL %] print \$stash->get('one') [% END %]" => 'ONE', {one => 'ONE', tt_config => ['EVAL_PERL' => 1]});
1051 process_ok("[% PERL %] print \$stash->set('a.b.c', 7) [% END %][% a.b.c %]" => '77', {tt_config => ['EVAL_PERL' => 1]});
1052
1053 ###----------------------------------------------------------------###
1054 print "### recursion prevention #############################################\n";
1055
1056 process_ok("[% BLOCK foo %][% PROCESS bar %][% END %][% BLOCK bar %][% PROCESS foo %][% END %][% PROCESS foo %]" => '') if ! $is_tt;
1057
1058 ###----------------------------------------------------------------###
1059 print "### META #############################################################\n";
1060
1061 process_ok("[% template.name %]" => 'input text');
1062 process_ok("[% META foo = 'bar' %][% template.foo %]" => 'bar');
1063 process_ok("[% META name = 'bar' %][% template.name %]" => 'bar');
1064 process_ok("[% META foo = 'bar' %][% component.foo %]" => 'bar');
1065 process_ok("[% META foo = 'bar' %][% component = '' %][% component.foo %]|foo" => '|foo');
1066 process_ok("[% META foo = 'bar' %][% template = '' %][% template.foo %]|foo" => '|foo');
1067
1068 ###----------------------------------------------------------------###
1069 print "### references #######################################################\n";
1070
1071 process_ok("[% a=3; b=\\a; b; a %]" => 33);
1072 process_ok("[% a=3; b=\\a; a=7; b; a %]" => 77);
1073
1074 process_ok("[% a={}; a.1=7; b=\\a.1; b; a.1 %]" => '77');
1075 process_ok("[% a={}; a.1=7; b=\\a.20; a.20=7; b; a.20 %]" => '77');
1076
1077 process_ok("[% a=[]; a.1=7; b=\\a.1; b; a.1 %]" => '77');
1078 process_ok("[% a=[]; a.1=7; b=\\a.20; a.20=7; b; a.20 %]" => '77');
1079
1080 process_ok("[% \\a %]" => qr/^CODE/, {a => sub { return "a sub [@_]" } });
1081 process_ok("[% b=\\a; b %]" => 'a sub []', {a => sub { return "a sub [@_]" } });
1082 process_ok("[% b=\\a(1); b %]" => 'a sub [1]', {a => sub { return "a sub [@_]" } });
1083 process_ok("[% b=\\a; b(2) %]" => 'a sub [2]', {a => sub { return "a sub [@_]" } });
1084 process_ok("[% b=\\a(1); b(2) %]" => 'a sub [1 2]', {a => sub { return "a sub [@_]" } });
1085 process_ok("[% f=\\j.k; j.k=7; f %]" => '7', {j => {k => 3}});
1086
1087 process_ok('[% a = "a" ; f = {a=>"A",b=>"B"} ; foo = \f.$a ; foo %]' => 'A');
1088 process_ok('[% a = "a" ; f = {a=>"A",b=>"B"} ; foo = \f.$a ; a = "b" ; foo %]' => 'A');
1089 process_ok('[% a = "ab" ; f = "abcd"; foo = \f.replace(a, "-AB-") ; a = "cd"; foo %]' => '-AB-cd');
1090 process_ok('[% a = "ab" ; f = "abcd"; foo = \f.replace(a, "-AB-").replace("-AB-", "*") ; a = "cd"; foo %]' => '*cd');
1091
1092 process_ok('[% a = "ab" ; f = "abcd"; foo = \f.replace(a, "-AB-") ; f = "ab"; foo %]' => '-AB-cd');
1093 process_ok('[% a = "ab" ; f = "abcd"; foo = \f.replace(a, "-AB-").replace("-AB-", "*") ; f = "ab"; foo %]' => '*cd');
1094
1095 ###----------------------------------------------------------------###
1096 print "### reserved words ###################################################\n";
1097
1098 $vars = {
1099 GET => 'named_get',
1100 get => 'lower_named_get',
1101 named_get => 'value of named_get',
1102 hold_get => 'GET',
1103 };
1104 process_ok("[% GET %]" => '', $vars);
1105 process_ok("[% GET GET %]" => 'named_get', $vars) if ! $is_tt;
1106 process_ok("[% GET get %]" => 'lower_named_get', $vars);
1107 process_ok("[% GET \${'GET'} %]" => 'bar', {GET => 'bar'});
1108
1109 process_ok("[% GET = 1 %][% GET GET %]" => '', $vars);
1110 process_ok("[% SET GET = 1 %][% GET GET %]" => '1', $vars) if ! $is_tt;
1111
1112 process_ok("[% GET \$hold_get %]" => 'named_get', $vars);
1113 process_ok("[% GET \$GET %]" => 'value of named_get', $vars) if ! $is_tt;
1114 process_ok("[% BLOCK GET %]hi[% END %][% PROCESS GET %]" => 'hi') if ! $is_tt;
1115 process_ok("[% BLOCK foo %]hi[% END %][% PROCESS foo a = GET %]" => 'hi', $vars) if ! $is_tt;
1116 process_ok("[% BLOCK foo %]hi[% END %][% PROCESS foo GET = 1 %]" => '');
1117 process_ok("[% BLOCK foo %]hi[% END %][% PROCESS foo IF GET %]" => 'hi', $vars) if ! $is_tt;
1118
1119 ###----------------------------------------------------------------###
1120 print "### embedded items ###################################################\n";
1121
1122 process_ok('[% " \" " %]' => ' " ');
1123 process_ok('[% " \$foo " %]' => ' $foo ');
1124 process_ok('[% " \${foo} " %]' => ' ${foo} ');
1125 process_ok('[% " \n " %]' => " \n ");
1126 process_ok('[% " \t " %]' => " \t ");
1127 process_ok('[% " \r " %]' => " \r ");
1128
1129 process_ok("[% 'foo\\'bar' %]" => "foo'bar");
1130 process_ok('[% "foo\\"bar" %]' => 'foo"bar');
1131 process_ok('[% qw(foo \)).1 %]' => ')') if ! $is_tt;
1132 process_ok('[% qw|foo \||.1 %]' => '|') if ! $is_tt;
1133
1134 process_ok("[% ' \\' ' %]" => " ' ");
1135 process_ok("[% ' \\r ' %]" => ' \r ');
1136 process_ok("[% ' \\n ' %]" => ' \n ');
1137 process_ok("[% ' \\t ' %]" => ' \t ');
1138 process_ok("[% ' \$foo ' %]" => ' $foo ');
1139
1140 process_ok('[% A = "bar" ; ${ "A" } %]' => 'bar');
1141 process_ok('[% A = "bar" ; "(${ A })" %]' => '(bar)');
1142 process_ok('[% A = "bar" ; ${ {a => "A"}.a } %]' => 'bar') if ! $is_tt;
1143 process_ok('[% A = "bar" ; "(${ {a => \"A\"\\}.a })" %]' => '(A)') if ! $is_tt;
1144 process_ok('[% A = "bar" ; "(${ \\${ {a => \"A\"\\}.a \\} })" %]' => '(bar)') if ! $is_tt;
1145 process_ok('[% A = "bar" %](${ {a => \"A\"\\}.a })' => '(A)', {tt_config => [INTERPOLATE => 1]}) if ! $is_tt;
1146 process_ok('[% A = "bar" %](${ \\${ {a => \"A\"\\}.a \\} })' => '(bar)', {tt_config => [INTERPOLATE => 1]}) if ! $is_tt;
1147
1148 process_ok('[% "[%" %]' => '[%') if ! $is_tt;
1149 process_ok('[% "%]" %]' => '%]') if ! $is_tt;
1150 process_ok('[% a = "[% %]" %][% a %]' => '[% %]') if ! $is_tt;
1151 process_ok('[% "[% 1 + 2 %]" | eval %]' => '3') if ! $is_tt;
1152
1153 process_ok('[% qw([% 1 + 2 %]).join %]' => '[% 1 + 2 %]') if ! $is_tt;
1154 process_ok('[% qw([% 1 + 2 %]).join.eval %]' => '3') if ! $is_tt;
1155
1156 process_ok('[% f = ">[% TRY; f.eval ; CATCH; \'caught\' ; END %]"; f.eval %]' => '>>>>>caught', {tt_config => [MAX_EVAL_RECURSE => 5]}) if ! $is_tt;
1157 process_ok('[% f = ">[% TRY; f.eval ; CATCH; \'foo\' ; END %]"; f.eval;f.eval %]' => '>>foo>>foo', {tt_config => [MAX_EVAL_RECURSE => 2]}) if ! $is_tt;
1158
1159 ###----------------------------------------------------------------###
1160 print "### DUMP #############################################################\n";
1161
1162 if (! $is_tt) {
1163 local $ENV{'REQUEST_METHOD'} = 0;
1164 process_ok("[% DUMP a %]" => "DUMP: File \"input text\" line 1\n a = undef;\n");
1165 process_ok("[% p = DUMP a; p.collapse %]" => 'DUMP: File "input text" line 1 a = undef;');
1166 process_ok("[% p = DUMP a; p.collapse %]" => 'DUMP: File "input text" line 1 a = \'s\';', {a => "s"});
1167 process_ok("[%\n p = DUMP a; p.collapse %]" => 'DUMP: File "input text" line 2 a = \'s\';', {a => "s"});
1168 process_ok("[% p = DUMP a, b; p.collapse %]" => 'DUMP: File "input text" line 1 a, b = [ \'s\', undef ];', {a => "s"});
1169 process_ok("[% p = DUMP a Useqq => 'b'; p.collapse %]" => 'DUMP: File "input text" line 1 a Useqq => \'b\' = [ \'s\', { \'Useqq\' => \'b\' } ];', {a => "s"});
1170 process_ok("[% p = DUMP a; p.collapse %]" => 'DUMP: File "input text" line 1 a = "s";', {a => "s", tt_config => [DUMP => {Useqq => 1}]});
1171 process_ok("[% p = DUMP a; p.collapse %]|foo" => '|foo', {a => "s", tt_config => [DUMP => 0]});
1172 process_ok("[% p = DUMP _a, b; p.collapse %]" => 'DUMP: File "input text" line 1 _a, b = [ undef, \'c\' ];', {_a => "s", b=> "c"});
1173 process_ok("[% p = DUMP {a => 'b'}; p.collapse %]" => 'DUMP: File "input text" line 1 {a => \'b\'} = { \'a\' => \'b\' };');
1174 process_ok("[% p = DUMP _a; p.collapse %]" => 'DUMP: File "input text" line 1 _a = undef;', {_a => "s"});
1175 process_ok("[% p = DUMP a; p.collapse %]" => 'DUMP: File "input text" line 1 a = { \'b\' => \'c\' };', {a => {b => 'c'}});
1176 process_ok("[% p = DUMP a; p.collapse %]" => 'DUMP: File "input text" line 1 a = {};', {a => {_b => 'c'}});
1177 process_ok("[% p = DUMP a; p.collapse %]" => 'DUMP: File "input text" line 1 a = {};', {a => {_b => 'c'}, tt_config => [DUMP => {Sortkeys => 1}]});
1178 process_ok("[% p = DUMP a; p.collapse %]" => 'DUMP: File "input text" line 1 Dump(7)', {a => 7, tt_config => [DUMP => {handler=>sub {"Dump(@_)"}}]});
1179 process_ok("[% p = DUMP a; p.collapse %]" => 'a = \'s\';', {a => "s", tt_config => [DUMP => {header => 0}]});
1180 process_ok("[% p = DUMP a; p.collapse %]" => '<pre>a = &apos;s&apos;; </pre>', {a => "s", tt_config => [DUMP => {header => 0, html => 1}]});
1181 local $ENV{'REQUEST_METHOD'} = 1;
1182 process_ok("[% p = DUMP a; p.collapse %]" => '<pre>a = &apos;s&apos;; </pre>', {a => "s", tt_config => [DUMP => {header => 0}]});
1183 process_ok("[% p = DUMP a; p.collapse %]" => 'a = \'s\';', {a => "s", tt_config => [DUMP => {header => 0, html => 0}]});
1184 local $ENV{'REQUEST_METHOD'} = 0;
1185 process_ok("[% SET global; p = DUMP; p.collapse %]" => "DUMP: File \"input text\" line 1 EntireStash = { 'a' => 'b', 'global' => '' };", {a => 'b', tt_config => [DUMP => {Sortkeys => 1}]});
1186 process_ok("[% SET global; p = DUMP; p.collapse %]" => "DUMP: File \"input text\" line 1 EntireStash = { 'a' => 'b', 'global' => '' };", {a => 'b', tt_config => [DUMP => {Sortkeys => 1, EntireStash => 1}]});
1187 process_ok("[% SET global; p = DUMP; p.collapse %]" => "DUMP: File \"input text\" line 1", {a => 'b', tt_config => [DUMP => {Sortkeys => 1, EntireStash => 0}]});
1188 }
1189
1190 ###----------------------------------------------------------------###
1191 print "### CONFIG ############################################################\n";
1192
1193 if (! $is_tt) {
1194 process_ok("[% CONFIG ANYCASE => 1 %][% get 234 %]" => 234);
1195 process_ok("[% CONFIG anycase => 1 %][% get 234 %]" => 234);
1196 process_ok("[% CONFIG PRE_CHOMP => '-' %]\n[% 234 %]" => 234);
1197 process_ok("[% CONFIG POST_CHOMP => '-' %][% 234 %]\n" => 234);
1198 process_ok("[% CONFIG INTERPOLATE => '-' %]\${ 234 }" => 234);
1199 process_ok("[% CONFIG V1DOLLAR => 1 %][% a = 234 %][% \$a %]" => 234);
1200 process_ok("[% CONFIG V2PIPE => 1 %][% BLOCK a %]b is [% b %][% END %][% PROCESS a b => 234 | repeat(2) %]" => "b is 234b is 234");
1201
1202 process_ok("[% CONFIG BOGUS => 2 %]bar" => '');
1203
1204 process_ok("[% CONFIG ANYCASE %]|[% CONFIG ANYCASE => 1 %][% CONFIG ANYCASE %]" => 'CONFIG ANYCASE = undef|CONFIG ANYCASE = 1');
1205 process_ok("[% CONFIG ANYCASE %]|[% CONFIG ANYCASE => 1 %][% CONFIG ANYCASE %]" => 'CONFIG ANYCASE = undef|CONFIG ANYCASE = 1');
1206
1207 process_ok("[% CONFIG DUMP %]|[% CONFIG DUMP => 0 %][% DUMP %]bar" => 'CONFIG DUMP = undef|bar');
1208 process_ok("[% CONFIG DUMP => {Useqq=>1, header=>0, html=>0} %][% DUMP 'foo' %]" => "'foo' = \"foo\";\n");
1209 }
1210
1211 ###----------------------------------------------------------------###
1212 print "### DONE #############################################################\n";
This page took 0.0848 seconds and 4 git commands to generate.