5 use B qw/svref_2object/;
7 # we test the pure-perl versions only, but allow overrides
8 # from the accessor_xs test-umbrella
9 # Also make sure a rogue envvar will not interfere with
13 $Class::Accessor::Grouped::USE_XS = 0
14 unless defined $Class::Accessor::Grouped::USE_XS;
16 $use_xs = $Class::Accessor::Grouped::USE_XS;
19 require AccessorGroupsSubclass;
21 my $test_accessors = {
27 # even though this accessor is declared as simple it will *not* be
28 # reinstalled due to the runtime 'around'
29 forced_class => 'AccessorGroups',
38 custom_field => 'lr1;field',
41 custom_field => "lr2'field",
43 fieldname_torture => {
45 custom_field => join ('', map { chr($_) } (0..255) ),
50 AccessorGroupsSubclass
54 my $obj = $class->new;
56 for my $name (sort keys %$test_accessors) {
57 my $alias = "_${name}_accessor";
58 my $field = $test_accessors->{$name}{custom_field} || $name;
59 my $extra = $test_accessors->{$name}{has_extra};
60 my $origin_class = 'AccessorGroupsParent';
62 if ( $class eq 'AccessorGroupsParent' ) {
63 next if $name eq 'runtime_around'; # implemented in the AG subclass
66 elsif ($name eq 'fieldname_torture') {
67 $field = reverse $field;
68 $origin_class = 'AccessorGroups';
71 can_ok($obj, $name, $alias);
72 ok(!$obj->can($field), "field for $name is not a method on $class")
78 for my $meth ($name, $alias) {
79 my $cv = svref_2object( $init_shims->{$meth} = $obj->can($meth) );
80 is($cv->GV->NAME, $meth, "initial ${class}::$meth accessor is named");
83 $test_accessors->{$name}{forced_class} || $origin_class,
84 "initial ${class}::$meth origin class correct",
88 is($obj->$name, undef, "${class}::$name begins undef");
89 is($obj->$alias, undef, "${class}::$alias begins undef");
92 is($obj->$name('a'), 'a', "${class}::$name setter RV correct");
93 is($obj->$name, 'a', "${class}::$name getter correct");
94 is($obj->{$field}, $extra ? 'a Extra tackled on' : 'a', "${class}::$name corresponding field correct");
96 # alias gets same as name
97 is($obj->$alias, 'a', "${class}::$alias getter correct after ${class}::$name setter");
100 is($obj->$alias('b'), 'b', "${class}::$alias setter RV correct");
101 is($obj->$alias, 'b', "${class}::$alias getter correct");
102 is($obj->{$field}, $extra ? 'b Extra tackled on' : 'b', "${class}::$alias corresponding field still correct");
104 # alias gets same as name
105 is($obj->$name, 'b', "${class}::$name getter correct after ${class}::$alias setter");
107 for my $meth ($name, $alias) {
108 my $resolved = $obj->can($meth);
110 my $cv = svref_2object($resolved);
111 is($cv->GV->NAME, $meth, "$meth accessor is named after operations");
113 $cv->GV->STASH->NAME,
114 # XS deferred subs install into each caller, not into the original parent
115 $test_accessors->{$name}{forced_class} || (
116 ($use_xs and $test_accessors->{$name}{is_simple})
120 "${class}::$meth origin class correct after operations",
123 # just simple for now
124 if ($use_xs and $test_accessors->{$name}{is_simple} and ! $test_accessors->{$name}{forced_class}) {
125 ok ($resolved != $init_shims->{$meth}, "$meth was replaced with a resolved version");
126 if ($class eq 'AccessorGroupsParent') {
127 ok ($cv->XSUB, "${class}::$meth is an XSUB");
130 ok (!$cv->XSUB, "${class}::$meth is *not* an XSUB (due to get_simple overrides)");