]> git.vpit.fr Git - perl/modules/Variable-Magic.git/blob - t/14-callbacks.t
More tests for different op types
[perl/modules/Variable-Magic.git] / t / 14-callbacks.t
1 #!perl -T
2
3 use strict;
4 use warnings;
5
6 use Test::More tests => 12 + 9 * (5 + 6 + 5);
7
8 use Variable::Magic qw/wizard cast dispell VMG_OP_INFO_NAME VMG_OP_INFO_OBJECT/;
9
10 my $wiz = eval { wizard get => sub { undef } };
11 is($@, '',             'wizard creation doesn\'t croak');
12 ok(defined $wiz,       'wizard is defined');
13 is(ref $wiz, 'SCALAR', 'wizard is a scalar ref');
14
15 my $n = int rand 1000;
16 my $a = $n;
17
18 my $res = eval { cast $a, $wiz };
19 is($@, '', 'cast doesn\'t croak');
20 ok($res,   'cast is valid');
21
22 my $x;
23 eval {
24  local $SIG{__WARN__} = sub { die };
25  $x = $a
26 };
27 is($@, '', 'callback returning undef doesn\'t warn/croak');
28 is($x, $n, 'callback returning undef fails');
29
30 my @callers;
31 $wiz = wizard get => sub {
32  my @c;
33  my $i = 0;
34  while (@c = caller $i++) {
35   push @callers, [ @c[0, 1, 2] ];
36  }
37 };
38
39 my $b;
40 cast $b, $wiz;
41
42 my $u = $b;
43 is_deeply(\@callers, [
44  [ 'main', $0, __LINE__-2 ],
45 ], 'caller into callback returns the right thing');
46
47 @callers = ();
48 $u = $b;
49 is_deeply(\@callers, [
50  [ 'main', $0, __LINE__-2 ],
51 ], 'caller into callback returns the right thing (second time)');
52
53 {
54  @callers = ();
55  my $u = $b;
56  is_deeply(\@callers, [
57   [ 'main', $0, __LINE__-2 ],
58  ], 'caller into callback into block returns the right thing');
59 }
60
61 @callers = ();
62 eval { my $u = $b };
63 is($@, '', 'caller into callback doesn\'t croak');
64 is_deeply(\@callers, [
65  ([ 'main', $0, __LINE__-3 ]) x 2,
66 ], 'caller into callback into eval returns the right thing');
67
68 sub Variable::Magic::TestPkg::foo { }
69
70 my @tests = (
71  [ 'len', '@c', undef,     'my $x = @c',        [ 'padav',   'B::OP'     ] ],
72  [ 'get', '$c', undef,     '++$c',              [ 'preinc',  'B::UNOP'   ] ],
73  [ 'get', '$c', '$c = 1',  '$c ** 2',           [ 'pow',     'B::BINOP'  ] ],
74  [ 'get', '$c', undef,     'my $x = $c',        [ 'sassign', 'B::BINOP'  ] ],
75  [ 'get', '$c', undef,     '1 if $c',           [ 'and',     'B::LOGOP'  ] ],
76  [ 'set', '$c', undef,     'bless \$c, "main"', [ 'bless',   'B::LISTOP' ] ],
77  [ 'get', '$c', '$c = ""', '$c =~ /x/',         [ 'match',   'B::PMOP'   ] ],
78  [ 'get', '$c', '$c = "Variable::Magic::TestPkg"',
79                            '$c->foo()',    [ 'method_named', 'B::SVOP'   ] ],
80  [ 'get', '$c', '$c = ""', '$c =~ y/x/y/',      [ 'trans',   'B::PVOP'   ] ],
81 );
82
83 for (@tests) {
84  my ($key, $var, $init, $test, $exp) = @$_;
85
86  for my $op_info (VMG_OP_INFO_NAME, VMG_OP_INFO_OBJECT, 3) {
87   our $done;
88   my ($c, @c);
89
90   # We must test for the $op correctness inside the callback because, if we
91   # bring it out, it will go outside of the eval STRING scope, and what it
92   # points to will no longer exist.
93   eval {
94    $wiz = wizard $key => sub {
95     return if $done;
96     my $op = $_[-1];
97     my $desc = "$key magic with op_info == $op_info";
98     if ($op_info == 1) {
99      is $op, $exp->[0], "$desc gets the right op info";
100     } elsif ($op_info == 2) {
101      isa_ok $op, $exp->[1], $desc;
102      is $op->name, $exp->[0], "$desc gets the right op info";
103     } else {
104      is $op, undef, "$desc gets the right op info";
105     }
106     $done = 1;
107     ()
108    }, op_info => $op_info
109   };
110   is $@, '', "$key wizard with op_info == $op_info doesn't croak";
111
112   eval $init if defined $init;
113
114   eval "cast $var, \$wiz";
115   is $@, '', "$key cast with op_info == $op_info doesn't croak";
116
117   local $done = 0;
118   eval $test;
119   is $@, '', "$key magic with op_info == $op_info doesn't croak";
120
121   eval "dispell $var, \$wiz";
122   is $@, '', "$key dispell with op_info == $op_info doesn't croak";
123  }
124 }