This is a live mirror of the Perl 5 development currently hosted at https://github.com/perl/perl5
Perl_hv_placeholders_get() actually takes a const HV *hv.
[perl5.git] / lib / constant.t
CommitLineData
6515510f 1#!./perl -T
54310121 2
3BEGIN {
6515510f
AT
4 if ($ENV{PERL_CORE}) {
5 chdir 't' if -d 't';
6 @INC = '../lib';
7 }
54310121 8}
9
9f1b1f2d 10use warnings;
69e7dc3c 11use vars qw{ @warnings $fagwoosh $putt $kloong};
54310121 12BEGIN { # ...and save 'em for later
13 $SIG{'__WARN__'} = sub { push @warnings, @_ }
14}
803b07a7 15END { print STDERR @warnings }
54310121 16
54310121 17
18use strict;
6515510f 19use Test::More tests => 97;
10a0e555
HS
20my $TB = Test::More->builder;
21
22BEGIN { use_ok('constant'); }
54310121 23
54310121 24use constant PI => 4 * atan2 1, 1;
25
10a0e555
HS
26ok defined PI, 'basic scalar constant';
27is substr(PI, 0, 7), '3.14159', ' in substr()';
54310121 28
29sub deg2rad { PI * $_[0] / 180 }
30
31my $ninety = deg2rad 90;
32
10a0e555 33cmp_ok abs($ninety - 1.5707), '<', 0.0001, ' in math expression';
54310121 34
35use constant UNDEF1 => undef; # the right way
36use constant UNDEF2 => ; # the weird way
37use constant 'UNDEF3' ; # the 'short' way
38use constant EMPTY => ( ) ; # the right way for lists
39
10a0e555
HS
40is UNDEF1, undef, 'right way to declare an undef';
41is UNDEF2, undef, ' weird way';
42is UNDEF3, undef, ' short way';
43
44# XXX Why is this way different than the other ones?
54310121 45my @undef = UNDEF1;
10a0e555
HS
46is @undef, 1;
47is $undef[0], undef;
48
54310121 49@undef = UNDEF2;
10a0e555 50is @undef, 0;
54310121 51@undef = UNDEF3;
10a0e555 52is @undef, 0;
54310121 53@undef = EMPTY;
10a0e555 54is @undef, 0;
54310121 55
56use constant COUNTDOWN => scalar reverse 1, 2, 3, 4, 5;
57use constant COUNTLIST => reverse 1, 2, 3, 4, 5;
58use constant COUNTLAST => (COUNTLIST)[-1];
59
10a0e555 60is COUNTDOWN, '54321';
54310121 61my @cl = COUNTLIST;
10a0e555
HS
62is @cl, 5;
63is COUNTDOWN, join '', @cl;
64is COUNTLAST, 1;
65is((COUNTLIST)[1], 4);
54310121 66
67use constant ABC => 'ABC';
10a0e555 68is "abc${\( ABC )}abc", "abcABCabc";
54310121 69
9d116dd7 70use constant DEF => 'D', 'E', chr ord 'F';
10a0e555 71is "d e f @{[ DEF ]} d e f", "d e f D E F d e f";
54310121 72
73use constant SINGLE => "'";
74use constant DOUBLE => '"';
75use constant BACK => '\\';
76my $tt = BACK . SINGLE . DOUBLE ;
10a0e555 77is $tt, q(\\'");
54310121 78
79use constant MESS => q('"'\\"'"\\);
10a0e555
HS
80is MESS, q('"'\\"'"\\);
81is length(MESS), 8;
54310121 82
83use constant TRAILING => '12 cats';
84{
6515510f 85 local $^W;
10a0e555 86 cmp_ok TRAILING, '==', 12;
54310121 87}
10a0e555 88is TRAILING, '12 cats';
54310121 89
c1b0f331 90use constant LEADING => " \t1234";
10a0e555
HS
91cmp_ok LEADING, '==', 1234;
92is LEADING, " \t1234";
54310121 93
94use constant ZERO1 => 0;
95use constant ZERO2 => 0.0;
96use constant ZERO3 => '0.0';
10a0e555
HS
97is ZERO1, '0';
98is ZERO2, '0';
99is ZERO3, '0.0';
54310121 100
101{
102 package Other;
103 use constant PI => 3.141;
104}
105
10a0e555
HS
106cmp_ok(abs(PI - 3.1416), '<', 0.0001);
107is Other::PI, 3.141;
54310121 108
109use constant E2BIG => $! = 7;
10a0e555 110cmp_ok E2BIG, '==', 7;
54310121 111# This is something like "Arg list too long", but the actual message
112# text may vary, so we can't test much better than this.
10a0e555 113cmp_ok length(E2BIG), '>', 6;
54310121 114
10a0e555 115is @warnings, 0 or diag join "\n", "unexpected warning", @warnings;
54310121 116@warnings = (); # just in case
117undef &PI;
10a0e555
HS
118ok @warnings && ($warnings[0] =~ /Constant sub.* undefined/) or
119 diag join "\n", "unexpected warning", @warnings;
120shift @warnings;
54310121 121
10a0e555 122is @warnings, 0, "unexpected warning";
779c5bc9 123
10a0e555
HS
124my $curr_test = $TB->current_test;
125use constant CSCALAR => \"ok 37\n";
126use constant CHASH => { foo => "ok 38\n" };
127use constant CARRAY => [ undef, "ok 39\n" ];
779c5bc9
GS
128use constant CCODE => sub { "ok $_[0]\n" };
129
6515510f
AT
130my $output = $TB->output ;
131print $output ${+CSCALAR};
132print $output CHASH->{foo};
133print $output CARRAY->[1];
134print $output CCODE->($curr_test+4);
10a0e555
HS
135
136$TB->current_test($curr_test+4);
137
779c5bc9 138eval q{ CCODE->{foo} };
10a0e555
HS
139ok scalar($@ =~ /^Constant is not a HASH/);
140
83763826
GS
141
142# Allow leading underscore
143use constant _PRIVATE => 47;
10a0e555 144is _PRIVATE, 47;
83763826
GS
145
146# Disallow doubled leading underscore
147eval q{
148 use constant __DISALLOWED => "Oops";
149};
10a0e555 150like $@, qr/begins with '__'/;
83763826
GS
151
152# Check on declared() and %declared. This sub should be EXACTLY the
153# same as the one quoted in the docs!
154sub declared ($) {
155 use constant 1.01; # don't omit this!
156 my $name = shift;
157 $name =~ s/^::/main::/;
158 my $pkg = caller;
159 my $full_name = $name =~ /::/ ? $name : "${pkg}::$name";
160 $constant::declared{$full_name};
161}
162
10a0e555
HS
163ok declared 'PI';
164ok $constant::declared{'main::PI'};
83763826 165
10a0e555
HS
166ok !declared 'PIE';
167ok !$constant::declared{'main::PIE'};
83763826
GS
168
169{
170 package Other;
171 use constant IN_OTHER_PACK => 42;
10a0e555
HS
172 ::ok ::declared 'IN_OTHER_PACK';
173 ::ok $constant::declared{'Other::IN_OTHER_PACK'};
174 ::ok ::declared 'main::PI';
175 ::ok $constant::declared{'main::PI'};
83763826
GS
176}
177
10a0e555
HS
178ok declared 'Other::IN_OTHER_PACK';
179ok $constant::declared{'Other::IN_OTHER_PACK'};
d3a7d8c7
GS
180
181@warnings = ();
182eval q{
9f1b1f2d 183 no warnings;
6515510f 184 #local $^W if $] < 5.006;
d3a7d8c7
GS
185 use warnings 'constant';
186 use constant 'BEGIN' => 1 ;
187 use constant 'INIT' => 1 ;
188 use constant 'CHECK' => 1 ;
189 use constant 'END' => 1 ;
190 use constant 'DESTROY' => 1 ;
191 use constant 'AUTOLOAD' => 1 ;
192 use constant 'STDIN' => 1 ;
193 use constant 'STDOUT' => 1 ;
194 use constant 'STDERR' => 1 ;
195 use constant 'ARGV' => 1 ;
196 use constant 'ARGVOUT' => 1 ;
197 use constant 'ENV' => 1 ;
198 use constant 'INC' => 1 ;
199 use constant 'SIG' => 1 ;
83b99c4f 200 use constant 'UNITCHECK' => 1;
d3a7d8c7
GS
201};
202
10a0e555
HS
203my @Expected_Warnings =
204 (
205 qr/^Constant name 'BEGIN' is a Perl keyword at/,
206 qr/^Constant subroutine BEGIN redefined at/,
207 qr/^Constant name 'INIT' is a Perl keyword at/,
208 qr/^Constant name 'CHECK' is a Perl keyword at/,
209 qr/^Constant name 'END' is a Perl keyword at/,
210 qr/^Constant name 'DESTROY' is a Perl keyword at/,
211 qr/^Constant name 'AUTOLOAD' is a Perl keyword at/,
212 qr/^Constant name 'STDIN' is forced into package main:: a/,
213 qr/^Constant name 'STDOUT' is forced into package main:: at/,
214 qr/^Constant name 'STDERR' is forced into package main:: at/,
215 qr/^Constant name 'ARGV' is forced into package main:: at/,
216 qr/^Constant name 'ARGVOUT' is forced into package main:: at/,
217 qr/^Constant name 'ENV' is forced into package main:: at/,
218 qr/^Constant name 'INC' is forced into package main:: at/,
219 qr/^Constant name 'SIG' is forced into package main:: at/,
83b99c4f 220 qr/^Constant name 'UNITCHECK' is a Perl keyword at/,
10a0e555 221);
6515510f 222
83b99c4f
NC
223unless ($] > 5.009) {
224 # Remove the UNITCHECK warning
225 pop @Expected_Warnings;
226 # But keep the count the same
227 push @Expected_Warnings, qr/^$/;
228 push @warnings, "";
229}
230
6515510f
AT
231# when run under "make test"
232if (@warnings == 16) {
233 push @warnings, "";
234 push @Expected_Warnings, qr/^$/;
235}
236# when run directly: perl -wT -Ilib t/constant.t
237elsif (@warnings == 17) {
238 splice @Expected_Warnings, 1, 0,
239 qr/^Prototype mismatch: sub main::BEGIN \(\) vs none at/;
240}
241# when run directly under 5.6.2: perl -wT -Ilib t/constant.t
242elsif (@warnings == 15) {
243 splice @Expected_Warnings, 1, 1;
244 push @warnings, "", "";
245 push @Expected_Warnings, qr/^$/, qr/^$/;
246}
247else {
248 my $rule = " -" x 20;
249 diag "/!\\ unexpected case: ", scalar @warnings, " warnings\n$rule\n";
250 diag map { " $_" } @warnings;
251 diag $rule, $/;
252}
253
254is @warnings, 17;
255
10a0e555
HS
256for my $idx (0..$#warnings) {
257 like $warnings[$idx], $Expected_Warnings[$idx];
258}
6515510f 259
d3a7d8c7 260@warnings = ();
c7206c54
CT
261
262
263use constant {
264 THREE => 3,
265 FAMILY => [ qw( John Jane Sally ) ],
266 AGES => { John => 33, Jane => 28, Sally => 3 },
267 RFAM => [ [ qw( John Jane Sally ) ] ],
268 SPIT => sub { shift },
c7206c54
CT
269};
270
10a0e555
HS
271is @{+FAMILY}, THREE;
272is @{+FAMILY}, @{RFAM->[0]};
273is FAMILY->[2], RFAM->[0]->[2];
274is AGES->{FAMILY->[1]}, 28;
275is THREE**3, SPIT->(@{+FAMILY}**3);
5b673cda
AS
276
277# Allow name of digits/underscores only if it begins with underscore
278{
279 use warnings FATAL => 'constant';
280 eval q{
281 use constant _1_2_3 => 'allowed';
282 };
283 ok( $@ eq '' );
284}
69e7dc3c
NC
285
286sub slotch ();
287
288{
289 my @warnings;
290 local $SIG{'__WARN__'} = sub { push @warnings, @_ };
291 eval 'use constant slotch => 3; 1' or die $@;
292
293 is ("@warnings", "", "No warnings if a prototype exists");
294
295 my $value = eval 'slotch';
296 is ($@, '');
297 is ($value, 3);
298}
299
300sub zit;
301
302{
303 my @warnings;
304 local $SIG{'__WARN__'} = sub { push @warnings, @_ };
305 eval 'use constant zit => 4; 1' or die $@;
306
6515510f 307 # empty prototypes are reported differently in different versions
13e592d2 308 my $no_proto = $] < 5.008004 ? "" : ": none";
6515510f 309
69e7dc3c 310 is(scalar @warnings, 1, "1 warning");
6515510f 311 like ($warnings[0], qr/^Prototype mismatch: sub main::zit$no_proto vs \(\)/,
69e7dc3c
NC
312 "about the prototype mismatch");
313
314 my $value = eval 'zit';
315 is ($@, '');
316 is ($value, 4);
317}
318
319$fagwoosh = 'geronimo';
320$putt = 'leutwein';
321$kloong = 'schlozhauer';
322
323{
324 my @warnings;
325 local $SIG{'__WARN__'} = sub { push @warnings, @_ };
326 eval 'use constant fagwoosh => 5; 1' or die $@;
327
328 is ("@warnings", "", "No warnings if the typeglob exists already");
329
330 my $value = eval 'fagwoosh';
331 is ($@, '');
332 is ($value, 5);
333
334 my @value = eval 'fagwoosh';
335 is ($@, '');
336 is_deeply (\@value, [5]);
337
338 eval 'use constant putt => 6, 7; 1' or die $@;
339
340 is ("@warnings", "", "No warnings if the typeglob exists already");
341
342 @value = eval 'putt';
343 is ($@, '');
344 is_deeply (\@value, [6, 7]);
345
346 eval 'use constant "klong"; 1' or die $@;
347
348 is ("@warnings", "", "No warnings if the typeglob exists already");
349
350 $value = eval 'klong';
351 is ($@, '');
352 is ($value, undef);
353
354 @value = eval 'klong';
355 is ($@, '');
356 is_deeply (\@value, []);
357}