4c4a7c95a122d4846ec43a8940b6872f85842306
[gitmo/Package-Stash-XS.git] / t / 01-basic.t
1 use strict;
2 use warnings;
3
4 use Test::More;
5 use Test::Fatal;
6
7 use Package::Stash;
8
9 like(exception { Package::Stash->name }, qr/Can't call name as a class method/,
10    q{... can't call name() as a class method});
11
12 {
13     package Foo;
14
15     use constant SOME_CONSTANT => 1;
16 }
17
18 # ----------------------------------------------------------------------
19 ## tests adding a HASH
20
21 my $foo_stash = Package::Stash->new('Foo');
22 ok(!defined($Foo::{foo}), '... the %foo slot has not been created yet');
23 ok(!$foo_stash->has_symbol('%foo'), '... the object agrees');
24 ok(!defined($Foo::{foo}), '... checking doesn\' vivify');
25
26 is(exception {
27     $foo_stash->add_symbol('%foo' => { one => 1 });
28 }, undef, '... created %Foo::foo successfully');
29
30 # ... scalar should NOT be created here
31
32 ok(!$foo_stash->has_symbol('$foo'), '... SCALAR shouldnt have been created too');
33 ok(!$foo_stash->has_symbol('@foo'), '... ARRAY shouldnt have been created too');
34 ok(!$foo_stash->has_symbol('&foo'), '... CODE shouldnt have been created too');
35
36 ok(defined($Foo::{foo}), '... the %foo slot was created successfully');
37 ok($foo_stash->has_symbol('%foo'), '... the meta agrees');
38
39 # check the value ...
40
41 {
42     no strict 'refs';
43     ok(exists ${'Foo::foo'}{one}, '... our %foo was initialized correctly');
44     is(${'Foo::foo'}{one}, 1, '... our %foo was initialized correctly');
45 }
46
47 my $foo = $foo_stash->get_symbol('%foo');
48 is_deeply({ one => 1 }, $foo, '... got the right package variable back');
49
50 # ... make sure changes propogate up
51
52 $foo->{two} = 2;
53
54 {
55     no strict 'refs';
56     is(\%{'Foo::foo'}, $foo_stash->get_symbol('%foo'), '... our %foo is the same as the metas');
57
58     ok(exists ${'Foo::foo'}{two}, '... our %foo was updated correctly');
59     is(${'Foo::foo'}{two}, 2, '... our %foo was updated correctly');
60 }
61
62 # ----------------------------------------------------------------------
63 ## test adding an ARRAY
64
65 ok(!defined($Foo::{bar}), '... the @bar slot has not been created yet');
66
67 is(exception {
68     $foo_stash->add_symbol('@bar' => [ 1, 2, 3 ]);
69 }, undef, '... created @Foo::bar successfully');
70
71 ok(defined($Foo::{bar}), '... the @bar slot was created successfully');
72 ok($foo_stash->has_symbol('@bar'), '... the meta agrees');
73
74 # ... why does this not work ...
75
76 ok(!$foo_stash->has_symbol('$bar'), '... SCALAR shouldnt have been created too');
77 ok(!$foo_stash->has_symbol('%bar'), '... HASH shouldnt have been created too');
78 ok(!$foo_stash->has_symbol('&bar'), '... CODE shouldnt have been created too');
79
80 # check the value itself
81
82 {
83     no strict 'refs';
84     is(scalar @{'Foo::bar'}, 3, '... our @bar was initialized correctly');
85     is(${'Foo::bar'}[1], 2, '... our @bar was initialized correctly');
86 }
87
88 # ----------------------------------------------------------------------
89 ## test adding a SCALAR
90
91 ok(!defined($Foo::{baz}), '... the $baz slot has not been created yet');
92
93 is(exception {
94     $foo_stash->add_symbol('$baz' => 10);
95 }, undef, '... created $Foo::baz successfully');
96
97 ok(defined($Foo::{baz}), '... the $baz slot was created successfully');
98 ok($foo_stash->has_symbol('$baz'), '... the meta agrees');
99
100 ok(!$foo_stash->has_symbol('@baz'), '... ARRAY shouldnt have been created too');
101 ok(!$foo_stash->has_symbol('%baz'), '... HASH shouldnt have been created too');
102 ok(!$foo_stash->has_symbol('&baz'), '... CODE shouldnt have been created too');
103
104 is(${$foo_stash->get_symbol('$baz')}, 10, '... got the right value back');
105
106 {
107     no strict 'refs';
108     ${'Foo::baz'} = 1;
109
110     is(${'Foo::baz'}, 1, '... our $baz was assigned to correctly');
111     is(${$foo_stash->get_symbol('$baz')}, 1, '... the meta agrees');
112 }
113
114 # ----------------------------------------------------------------------
115 ## test adding a CODE
116
117 ok(!defined($Foo::{funk}), '... the &funk slot has not been created yet');
118
119 is(exception {
120     $foo_stash->add_symbol('&funk' => sub { "Foo::funk" });
121 }, undef, '... created &Foo::funk successfully');
122
123 ok(defined($Foo::{funk}), '... the &funk slot was created successfully');
124 ok($foo_stash->has_symbol('&funk'), '... the meta agrees');
125
126 ok(!$foo_stash->has_symbol('$funk'), '... SCALAR shouldnt have been created too');
127 ok(!$foo_stash->has_symbol('@funk'), '... ARRAY shouldnt have been created too');
128 ok(!$foo_stash->has_symbol('%funk'), '... HASH shouldnt have been created too');
129
130 {
131     no strict 'refs';
132     ok(defined &{'Foo::funk'}, '... our &funk exists');
133 }
134
135 is(Foo->funk(), 'Foo::funk', '... got the right value from the function');
136
137 # ----------------------------------------------------------------------
138 ## test multiple slots in the glob
139
140 my $ARRAY = [ 1, 2, 3 ];
141 my $CODE = sub { "Foo::foo" };
142
143 is(exception {
144     $foo_stash->add_symbol('@foo' => $ARRAY);
145 }, undef, '... created @Foo::foo successfully');
146
147 ok($foo_stash->has_symbol('@foo'), '... the @foo slot was added successfully');
148 is($foo_stash->get_symbol('@foo'), $ARRAY, '... got the right values for @Foo::foo');
149
150 is(exception {
151     $foo_stash->add_symbol('&foo' => $CODE);
152 }, undef, '... created &Foo::foo successfully');
153
154 ok($foo_stash->has_symbol('&foo'), '... the meta agrees');
155 is($foo_stash->get_symbol('&foo'), $CODE, '... got the right value for &Foo::foo');
156
157 is(exception {
158     $foo_stash->add_symbol('$foo' => 'Foo::foo');
159 }, undef, '... created $Foo::foo successfully');
160
161 ok($foo_stash->has_symbol('$foo'), '... the meta agrees');
162 my $SCALAR = $foo_stash->get_symbol('$foo');
163 is($$SCALAR, 'Foo::foo', '... got the right scalar value back');
164
165 {
166     no strict 'refs';
167     is(${'Foo::foo'}, 'Foo::foo', '... got the right value from the scalar');
168 }
169
170 is(exception {
171     $foo_stash->remove_symbol('%foo');
172 }, undef, '... removed %Foo::foo successfully');
173
174 ok(!$foo_stash->has_symbol('%foo'), '... the %foo slot was removed successfully');
175 ok($foo_stash->has_symbol('@foo'), '... the @foo slot still exists');
176 ok($foo_stash->has_symbol('&foo'), '... the &foo slot still exists');
177 ok($foo_stash->has_symbol('$foo'), '... the $foo slot still exists');
178
179 is($foo_stash->get_symbol('@foo'), $ARRAY, '... got the right values for @Foo::foo');
180 is($foo_stash->get_symbol('&foo'), $CODE, '... got the right value for &Foo::foo');
181 is($foo_stash->get_symbol('$foo'), $SCALAR, '... got the right value for $Foo::foo');
182
183 {
184     no strict 'refs';
185     ok(!defined(*{"Foo::foo"}{HASH}), '... the %foo slot has been removed successfully');
186     ok(defined(*{"Foo::foo"}{ARRAY}), '... the @foo slot has NOT been removed');
187     ok(defined(*{"Foo::foo"}{CODE}), '... the &foo slot has NOT been removed');
188     ok(defined(${"Foo::foo"}), '... the $foo slot has NOT been removed');
189 }
190
191 is(exception {
192     $foo_stash->remove_symbol('&foo');
193 }, undef, '... removed &Foo::foo successfully');
194
195 ok(!$foo_stash->has_symbol('&foo'), '... the &foo slot no longer exists');
196
197 ok($foo_stash->has_symbol('@foo'), '... the @foo slot still exists');
198 ok($foo_stash->has_symbol('$foo'), '... the $foo slot still exists');
199
200 is($foo_stash->get_symbol('@foo'), $ARRAY, '... got the right values for @Foo::foo');
201 is($foo_stash->get_symbol('$foo'), $SCALAR, '... got the right value for $Foo::foo');
202
203 {
204     no strict 'refs';
205     ok(!defined(*{"Foo::foo"}{HASH}), '... the %foo slot has been removed successfully');
206     ok(!defined(*{"Foo::foo"}{CODE}), '... the &foo slot has now been removed');
207     ok(defined(*{"Foo::foo"}{ARRAY}), '... the @foo slot has NOT been removed');
208     ok(defined(${"Foo::foo"}), '... the $foo slot has NOT been removed');
209 }
210
211 is(exception {
212     $foo_stash->remove_symbol('$foo');
213 }, undef, '... removed $Foo::foo successfully');
214
215 ok(!$foo_stash->has_symbol('$foo'), '... the $foo slot no longer exists');
216
217 ok($foo_stash->has_symbol('@foo'), '... the @foo slot still exists');
218
219 is($foo_stash->get_symbol('@foo'), $ARRAY, '... got the right values for @Foo::foo');
220
221 {
222     no strict 'refs';
223     ok(!defined(*{"Foo::foo"}{HASH}), '... the %foo slot has been removed successfully');
224     ok(!defined(*{"Foo::foo"}{CODE}), '... the &foo slot has now been removed');
225     ok(!defined(${"Foo::foo"}), '... the $foo slot has now been removed');
226     ok(defined(*{"Foo::foo"}{ARRAY}), '... the @foo slot has NOT been removed');
227 }
228
229 # check some errors
230
231 like(exception {
232     $foo_stash->add_symbol('@bar', {})
233 }, qr/HASH.*is not of type ARRAY/, "can't initialize a slot with the wrong type of value");
234
235 like(exception {
236     $foo_stash->add_symbol('bar', [])
237 }, qr/ARRAY.*is not of type IO/, "can't initialize a slot with the wrong type of value");
238
239 like(exception {
240     $foo_stash->add_symbol('$bar', sub { })
241 }, qr/CODE.*is not of type SCALAR/, "can't initialize a slot with the wrong type of value");
242
243 {
244     package Bar;
245     open *foo, '<', $0;
246 }
247
248 like(exception {
249     $foo_stash->add_symbol('$bar', *Bar::foo{IO})
250 }, qr/IO.*is not of type SCALAR/, "can't initialize a slot with the wrong type of value");
251
252 # check compile time manipulation
253
254 {
255     package Baz;
256
257     our $foo = 23;
258     our @foo = "bar";
259     our %foo = (baz => 1);
260     sub foo { }
261     open *foo, '<', $0;
262     BEGIN { Package::Stash->new(__PACKAGE__)->remove_symbol('&foo') }
263 }
264
265 {
266     my $stash = Package::Stash->new('Baz');
267     is(${ $stash->get_symbol('$foo') }, 23, "got \$foo");
268     is_deeply($stash->get_symbol('@foo'), ['bar'], "got \@foo");
269     is_deeply($stash->get_symbol('%foo'), {baz => 1}, "got \%foo");
270     ok(!$stash->has_symbol('&foo'), "got \&foo");
271     is($stash->get_symbol('foo'), *Baz::foo{IO}, "got foo");
272 }
273
274 {
275     package Quux;
276
277     our $foo = 23;
278     our @foo = "bar";
279     our %foo = (baz => 1);
280     sub foo { }
281     open *foo, '<', $0;
282 }
283
284 {
285     my $stash = Package::Stash->new('Quux');
286
287     my %expect = (
288         '$foo' => \23,
289         '@foo' => ["bar"],
290         '%foo' => { baz => 1 },
291         '&foo' => \&Quux::foo,
292         'foo'  => *Quux::foo{IO},
293     );
294
295     for my $sym ( sort keys %expect ) {
296         is_deeply(
297             $stash->get_symbol($sym),
298             $expect{$sym},
299             "got expected value for $sym"
300         );
301     }
302
303     $stash->add_symbol('%bar' => {x => 42});
304
305     $expect{'%bar'} = {x => 42};
306
307     for my $sym ( sort keys %expect ) {
308         is_deeply(
309             $stash->get_symbol($sym),
310             $expect{$sym},
311             "got expected value for $sym"
312         );
313     }
314
315     $stash->add_symbol('%bar' => {x => 43});
316
317     $expect{'%bar'} = {x => 43};
318
319     for my $sym ( sort keys %expect ) {
320         is_deeply(
321             $stash->get_symbol($sym),
322             $expect{$sym},
323             "got expected value for $sym"
324         );
325     }
326 }
327
328 {
329     package Quuux;
330     our $foo;
331     our @foo;
332     our @bar;
333     our %baz;
334     sub baz { }
335     use constant quux => 1;
336     use constant quuux => [];
337     sub quuuux;
338 }
339
340 {
341     my $quuux = Package::Stash->new('Quuux');
342     is_deeply(
343         [sort $quuux->list_all_symbols],
344         [qw(BEGIN bar baz foo quuuux quuux quux)],
345         "list_all_symbols",
346     );
347     { local $TODO = $] < 5.010 ? "undef scalars aren't visible on 5.8" : undef;
348     is_deeply(
349         [sort $quuux->list_all_symbols('SCALAR')],
350         [qw(foo)],
351         "list_all_symbols SCALAR",
352     );
353     }
354     is_deeply(
355         [sort $quuux->list_all_symbols('ARRAY')],
356         [qw(bar foo)],
357         "list_all_symbols ARRAY",
358     );
359     is_deeply(
360         [sort $quuux->list_all_symbols('HASH')],
361         [qw(baz)],
362         "list_all_symbols HASH",
363     );
364     is_deeply(
365         [sort $quuux->list_all_symbols('CODE')],
366         [qw(baz quuuux quuux quux)],
367         "list_all_symbols CODE",
368     );
369 }
370
371 done_testing;