accessors
[gitmo/Package-Stash-XS.git] / lib / Package / Stash.pm
CommitLineData
e94260da 1package Package::Stash;
f10f6217 2use strict;
3use warnings;
988beb41 4# ABSTRACT: routines for manipulating stashes
f10f6217 5
6use Carp qw(confess);
7use Scalar::Util qw(reftype);
dc378b60 8use Symbol;
59017825 9
10use XSLoader;
11XSLoader::load(
12 __PACKAGE__,
13 # we need to be careful not to touch $VERSION at compile time, otherwise
14 # DynaLoader will assume it's set and check against it, which will cause
15 # fail when being run in the checkout without dzil having set the actual
16 # $VERSION
17 exists $Package::Stash::{VERSION}
18 ? ${ $Package::Stash::{VERSION} } : (),
19);
20
dc378b60 21# before 5.12, assigning to the ISA glob would make it lose its magical ->isa
22# powers
23use constant BROKEN_ISA_ASSIGNMENT => ($] < 5.012);
f4979588 24
f4979588 25=head1 SYNOPSIS
26
e94260da 27 my $stash = Package::Stash->new('Foo');
683542f5 28 $stash->add_package_symbol('%foo', {bar => 1});
29 # $Foo::foo{bar} == 1
30 $stash->has_package_symbol('$foo') # false
31 my $namespace = $stash->namespace;
32 *{ $namespace->{foo} }{HASH} # {bar => 1}
f4979588 33
34=head1 DESCRIPTION
35
683542f5 36Manipulating stashes (Perl's symbol tables) is occasionally necessary, but
37incredibly messy, and easy to get wrong. This module hides all of that behind a
38simple API.
39
56a29840 40NOTE: Most methods in this class require a variable specification that includes
41a sigil. If this sigil is absent, it is assumed to represent the IO slot.
42
988beb41 43=method new $package_name
683542f5 44
e94260da 45Creates a new C<Package::Stash> object, for the package given as the only
683542f5 46argument.
f4979588 47
988beb41 48=method name
683542f5 49
50Returns the name of the package that this object represents.
51
988beb41 52=method namespace
683542f5 53
54Returns the raw stash itself.
55
56=cut
57
f10f6217 58{
59 my %SIGIL_MAP = (
60 '$' => 'SCALAR',
61 '@' => 'ARRAY',
62 '%' => 'HASH',
63 '&' => 'CODE',
56a29840 64 '' => 'IO',
f10f6217 65 );
66
67 sub _deconstruct_variable_name {
68 my ($self, $variable) = @_;
69
56a29840 70 (defined $variable && length $variable)
f10f6217 71 || confess "You must pass a variable name";
72
73 my $sigil = substr($variable, 0, 1, '');
74
56a29840 75 if (exists $SIGIL_MAP{$sigil}) {
76 return ($variable, $sigil, $SIGIL_MAP{$sigil});
77 }
78 else {
79 return ("${sigil}${variable}", '', $SIGIL_MAP{''});
80 }
f10f6217 81 }
82}
83
988beb41 84=method add_package_symbol $variable $value %opts
683542f5 85
86Adds a new package symbol, for the symbol given as C<$variable>, and optionally
87gives it an initial value of C<$value>. C<$variable> should be the name of
88variable including the sigil, so
89
e94260da 90 Package::Stash->new('Foo')->add_package_symbol('%foo')
683542f5 91
92will create C<%Foo::foo>.
93
c61010aa 94Valid options (all optional) are C<filename>, C<first_line_num>, and
95C<last_line_num>.
96
97C<$opts{filename}>, C<$opts{first_line_num}>, and C<$opts{last_line_num}> can
98be used to indicate where the symbol should be regarded as having been defined.
4ada57e0 99Currently these values are only used if the symbol is a subroutine ('C<&>'
c61010aa 100sigil) and only if C<$^P & 0x10> is true, in which case the special C<%DB::sub>
101hash is updated to record the values of C<filename>, C<first_line_num>, and
102C<last_line_num> for the subroutine. If these are not passed, their values are
103inferred (as much as possible) from C<caller> information.
4ada57e0 104
105This is especially useful for debuggers and profilers, which use C<%DB::sub> to
106determine where the source code for a subroutine can be found. See
107L<http://perldoc.perl.org/perldebguts.html#Debugger-Internals> for more
108information about C<%DB::sub>.
109
683542f5 110=cut
111
3634ce60 112sub _valid_for_type {
113 my $self = shift;
114 my ($value, $type) = @_;
115 if ($type eq 'HASH' || $type eq 'ARRAY'
116 || $type eq 'IO' || $type eq 'CODE') {
117 return reftype($value) eq $type;
118 }
119 else {
120 my $ref = reftype($value);
121 return !defined($ref) || $ref eq 'SCALAR' || $ref eq 'REF' || $ref eq 'LVALUE';
122 }
123}
124
f10f6217 125sub add_package_symbol {
640de369 126 my ($self, $variable, $initial_value, %opts) = @_;
f10f6217 127
128 my ($name, $sigil, $type) = ref $variable eq 'HASH'
129 ? @{$variable}{qw[name sigil type]}
130 : $self->_deconstruct_variable_name($variable);
131
4ada57e0 132 my $pkg = $self->name;
133
3634ce60 134 if (@_ > 2) {
135 $self->_valid_for_type($initial_value, $type)
136 || confess "$initial_value is not of type $type";
3634ce60 137
4ada57e0 138 # cheap fail-fast check for PERLDBf_SUBLINE and '&'
139 if ($^P and $^P & 0x10 && $sigil eq '&') {
640de369 140 my $filename = $opts{filename};
141 my $first_line_num = $opts{first_line_num};
4ada57e0 142
640de369 143 (undef, $filename, $first_line_num) = caller
4ada57e0 144 if not defined $filename;
640de369 145
146 my $last_line_num = $opts{last_line_num} || ($first_line_num ||= 0);
4ada57e0 147
148 # http://perldoc.perl.org/perldebguts.html#Debugger-Internals
640de369 149 $DB::sub{$pkg . '::' . $name} = "$filename:$first_line_num-$last_line_num";
4ada57e0 150 }
151 }
f10f6217 152
153 no strict 'refs';
154 no warnings 'redefine', 'misc', 'prototype';
155 *{$pkg . '::' . $name} = ref $initial_value ? $initial_value : \$initial_value;
156}
157
988beb41 158=method remove_package_glob $name
683542f5 159
160Removes all package variables with the given name, regardless of sigil.
161
162=cut
163
f10f6217 164sub remove_package_glob {
165 my ($self, $name) = @_;
166 no strict 'refs';
167 delete ${$self->name . '::'}{$name};
168}
169
170# ... these functions deal with stuff on the namespace level
171
988beb41 172=method has_package_symbol $variable
683542f5 173
174Returns whether or not the given package variable (including sigil) exists.
175
176=cut
177
f10f6217 178sub has_package_symbol {
179 my ($self, $variable) = @_;
180
181 my ($name, $sigil, $type) = ref $variable eq 'HASH'
182 ? @{$variable}{qw[name sigil type]}
183 : $self->_deconstruct_variable_name($variable);
184
185 my $namespace = $self->namespace;
186
187 return unless exists $namespace->{$name};
188
189 my $entry_ref = \$namespace->{$name};
190 if (reftype($entry_ref) eq 'GLOB') {
f7543739 191 # XXX: assigning to any typeglob slot also initializes the SCALAR slot,
192 # and saying that an undef scalar variable doesn't exist is probably
193 # vaguely less surprising than a scalar variable popping into existence
194 # without anyone defining it
195 if ($type eq 'SCALAR') {
196 return defined ${ *{$entry_ref}{$type} };
197 }
198 else {
199 return defined *{$entry_ref}{$type};
200 }
f10f6217 201 }
202 else {
203 # a symbol table entry can be -1 (stub), string (stub with prototype),
204 # or reference (constant)
205 return $type eq 'CODE';
206 }
207}
208
988beb41 209=method get_package_symbol $variable
683542f5 210
211Returns the value of the given package variable (including sigil).
212
213=cut
214
f10f6217 215sub get_package_symbol {
e55803fc 216 my ($self, $variable, %opts) = @_;
f10f6217 217
218 my ($name, $sigil, $type) = ref $variable eq 'HASH'
219 ? @{$variable}{qw[name sigil type]}
220 : $self->_deconstruct_variable_name($variable);
221
222 my $namespace = $self->namespace;
223
7486ccf3 224 if (!exists $namespace->{$name}) {
dc378b60 225 if ($opts{vivify}) {
226 if ($type eq 'ARRAY') {
227 if (BROKEN_ISA_ASSIGNMENT) {
228 $self->add_package_symbol(
229 $variable,
230 $name eq 'ISA' ? () : ([])
231 );
232 }
233 else {
234 $self->add_package_symbol($variable, []);
235 }
236 }
237 elsif ($type eq 'HASH') {
238 $self->add_package_symbol($variable, {});
239 }
240 elsif ($type eq 'SCALAR') {
241 $self->add_package_symbol($variable);
242 }
243 elsif ($type eq 'IO') {
244 $self->add_package_symbol($variable, Symbol::geniosym);
245 }
246 elsif ($type eq 'CODE') {
247 confess "Don't know how to vivify CODE variables";
248 }
249 else {
250 confess "Unknown type $type in vivication";
251 }
5d3589c8 252 }
e55803fc 253 else {
dc378b60 254 if ($type eq 'CODE') {
255 # this effectively "de-vivifies" the code slot. if we don't do
256 # this, referencing the coderef at the end of this function
257 # will cause perl to auto-vivify a stub coderef in the slot,
258 # which isn't what we want
259 $self->add_package_symbol($variable);
260 }
e55803fc 261 }
30d1a098 262 }
f10f6217 263
264 my $entry_ref = \$namespace->{$name};
265
266 if (ref($entry_ref) eq 'GLOB') {
267 return *{$entry_ref}{$type};
268 }
269 else {
270 if ($type eq 'CODE') {
271 no strict 'refs';
272 return \&{ $self->name . '::' . $name };
273 }
274 else {
275 return undef;
276 }
277 }
278}
279
988beb41 280=method get_or_add_package_symbol $variable
e55803fc 281
282Like C<get_package_symbol>, except that it will return an empty hashref or
283arrayref if the variable doesn't exist.
284
285=cut
286
287sub get_or_add_package_symbol {
288 my $self = shift;
289 $self->get_package_symbol(@_, vivify => 1);
290}
291
988beb41 292=method remove_package_symbol $variable
683542f5 293
294Removes the package variable described by C<$variable> (which includes the
295sigil); other variables with the same name but different sigils will be
296untouched.
297
298=cut
299
f10f6217 300sub remove_package_symbol {
301 my ($self, $variable) = @_;
302
303 my ($name, $sigil, $type) = ref $variable eq 'HASH'
304 ? @{$variable}{qw[name sigil type]}
305 : $self->_deconstruct_variable_name($variable);
306
307 # FIXME:
9a3d1390 308 # no doubt this is grossly inefficient and
f10f6217 309 # could be done much easier and faster in XS
310
b1a00d0e 311 my ($scalar_desc, $array_desc, $hash_desc, $code_desc, $io_desc) = (
f10f6217 312 { sigil => '$', type => 'SCALAR', name => $name },
313 { sigil => '@', type => 'ARRAY', name => $name },
314 { sigil => '%', type => 'HASH', name => $name },
315 { sigil => '&', type => 'CODE', name => $name },
b1a00d0e 316 { sigil => '', type => 'IO', name => $name },
f10f6217 317 );
318
b1a00d0e 319 my ($scalar, $array, $hash, $code, $io);
f10f6217 320 if ($type eq 'SCALAR') {
321 $array = $self->get_package_symbol($array_desc) if $self->has_package_symbol($array_desc);
322 $hash = $self->get_package_symbol($hash_desc) if $self->has_package_symbol($hash_desc);
323 $code = $self->get_package_symbol($code_desc) if $self->has_package_symbol($code_desc);
b1a00d0e 324 $io = $self->get_package_symbol($io_desc) if $self->has_package_symbol($io_desc);
f10f6217 325 }
326 elsif ($type eq 'ARRAY') {
42fa5cfc 327 $scalar = $self->get_package_symbol($scalar_desc);
f10f6217 328 $hash = $self->get_package_symbol($hash_desc) if $self->has_package_symbol($hash_desc);
329 $code = $self->get_package_symbol($code_desc) if $self->has_package_symbol($code_desc);
b1a00d0e 330 $io = $self->get_package_symbol($io_desc) if $self->has_package_symbol($io_desc);
f10f6217 331 }
332 elsif ($type eq 'HASH') {
42fa5cfc 333 $scalar = $self->get_package_symbol($scalar_desc);
f10f6217 334 $array = $self->get_package_symbol($array_desc) if $self->has_package_symbol($array_desc);
335 $code = $self->get_package_symbol($code_desc) if $self->has_package_symbol($code_desc);
b1a00d0e 336 $io = $self->get_package_symbol($io_desc) if $self->has_package_symbol($io_desc);
f10f6217 337 }
338 elsif ($type eq 'CODE') {
42fa5cfc 339 $scalar = $self->get_package_symbol($scalar_desc);
f10f6217 340 $array = $self->get_package_symbol($array_desc) if $self->has_package_symbol($array_desc);
341 $hash = $self->get_package_symbol($hash_desc) if $self->has_package_symbol($hash_desc);
b1a00d0e 342 $io = $self->get_package_symbol($io_desc) if $self->has_package_symbol($io_desc);
343 }
344 elsif ($type eq 'IO') {
42fa5cfc 345 $scalar = $self->get_package_symbol($scalar_desc);
b1a00d0e 346 $array = $self->get_package_symbol($array_desc) if $self->has_package_symbol($array_desc);
347 $hash = $self->get_package_symbol($hash_desc) if $self->has_package_symbol($hash_desc);
348 $code = $self->get_package_symbol($code_desc) if $self->has_package_symbol($code_desc);
f10f6217 349 }
350 else {
351 confess "This should never ever ever happen";
352 }
353
354 $self->remove_package_glob($name);
355
42fa5cfc 356 $self->add_package_symbol($scalar_desc => $scalar);
f10f6217 357 $self->add_package_symbol($array_desc => $array) if defined $array;
358 $self->add_package_symbol($hash_desc => $hash) if defined $hash;
359 $self->add_package_symbol($code_desc => $code) if defined $code;
b1a00d0e 360 $self->add_package_symbol($io_desc => $io) if defined $io;
f10f6217 361}
362
988beb41 363=method list_all_package_symbols $type_filter
683542f5 364
365Returns a list of package variable names in the package, without sigils. If a
366C<type_filter> is passed, it is used to select package variables of a given
367type, where valid types are the slots of a typeglob ('SCALAR', 'CODE', 'HASH',
d1f721b3 368etc). Note that if the package contained any C<BEGIN> blocks, perl will leave
369an empty typeglob in the C<BEGIN> slot, so this will show up if no filter is
370used (and similarly for C<INIT>, C<END>, etc).
683542f5 371
372=cut
373
f10f6217 374sub list_all_package_symbols {
375 my ($self, $type_filter) = @_;
376
377 my $namespace = $self->namespace;
378 return keys %{$namespace} unless defined $type_filter;
379
380 # NOTE:
9a3d1390 381 # or we can filter based on
f10f6217 382 # type (SCALAR|ARRAY|HASH|CODE)
383 if ($type_filter eq 'CODE') {
384 return grep {
25c87f5c 385 # any non-typeglob in the symbol table is a constant or stub
386 ref(\$namespace->{$_}) ne 'GLOB'
387 # regular subs are stored in the CODE slot of the typeglob
d1f721b3 388 || defined(*{$namespace->{$_}}{CODE})
389 } keys %{$namespace};
390 }
391 elsif ($type_filter eq 'SCALAR') {
392 return grep {
393 ref(\$namespace->{$_}) eq 'GLOB'
394 && defined(${*{$namespace->{$_}}{'SCALAR'}})
395 } keys %{$namespace};
396 }
397 else {
398 return grep {
399 ref(\$namespace->{$_}) eq 'GLOB'
400 && defined(*{$namespace->{$_}}{$type_filter})
f10f6217 401 } keys %{$namespace};
f10f6217 402 }
403}
f4979588 404
0992f4ec 405=head1 BUGS
406
407No known bugs.
408
409Please report any bugs through RT: email
410C<bug-package-stash at rt.cpan.org>, or browse to
411L<http://rt.cpan.org/NoAuth/ReportBug.html?Queue=Package-Stash>.
412
f4979588 413=head1 SEE ALSO
414
f4979588 415=over 4
416
988beb41 417=item * L<Class::MOP::Package>
f4979588 418
988beb41 419This module is a factoring out of code that used to live here
f4979588 420
421=back
422
0992f4ec 423=head1 SUPPORT
424
425You can find this documentation for this module with the perldoc command.
426
427 perldoc Package::Stash
428
429You can also look for information at:
430
431=over 4
432
433=item * AnnoCPAN: Annotated CPAN documentation
434
435L<http://annocpan.org/dist/Package-Stash>
436
437=item * CPAN Ratings
438
439L<http://cpanratings.perl.org/d/Package-Stash>
440
441=item * RT: CPAN's request tracker
442
443L<http://rt.cpan.org/NoAuth/Bugs.html?Dist=Package-Stash>
444
445=item * Search CPAN
446
447L<http://search.cpan.org/dist/Package-Stash>
448
449=back
450
451=head1 AUTHOR
452
453Jesse Luehrs <doy at tozt dot net>
454
455Mostly copied from code from L<Class::MOP::Package>, by Stevan Little and the
456Moose Cabal.
457
f4979588 458=cut
459
4601;