cleanup and more tests;
[gitmo/Class-MOP.git] / lib / Class / MOP / Package.pm
CommitLineData
2243a22b 1
2package Class::MOP::Package;
3
4use strict;
5use warnings;
6
7use Scalar::Util 'blessed';
6d5355c3 8use Carp 'confess';
2243a22b 9
9d6dce77 10our $VERSION = '0.02';
2243a22b 11
12# introspection
13
14sub meta {
15 require Class::MOP::Class;
16 Class::MOP::Class->initialize(blessed($_[0]) || $_[0]);
17}
18
6d5355c3 19# creation ...
20
21sub initialize {
9d6dce77 22 my $class = shift;
23 my $package_name = shift;
24 # we hand-construct the class
25 # until we can bootstrap it
a5e51f0b 26 no strict 'refs';
27 return bless {
28 '$:package' => $package_name,
29 '%:namespace' => \%{$package_name . '::'},
30 } => $class;
6d5355c3 31}
32
33# Attributes
34
35# NOTE:
36# all these attribute readers will be bootstrapped
37# away in the Class::MOP bootstrap section
38
a5e51f0b 39sub name { $_[0]->{'$:package'} }
40sub namespace { $_[0]->{'%:namespace'} }
6d5355c3 41
a5e51f0b 42# utility methods
6d5355c3 43
c0cbf4d9 44{
45 my %SIGIL_MAP = (
46 '$' => 'SCALAR',
47 '@' => 'ARRAY',
48 '%' => 'HASH',
49 '&' => 'CODE',
50 );
6d5355c3 51
a5e51f0b 52 sub _deconstruct_variable_name {
53 my ($self, $variable) = @_;
54
c0cbf4d9 55 (defined $variable)
56 || confess "You must pass a variable name";
a5e51f0b 57
c0cbf4d9 58 my ($sigil, $name) = ($variable =~ /^(.)(.*)$/);
a5e51f0b 59
c0cbf4d9 60 (defined $sigil)
61 || confess "The variable name must include a sigil";
a5e51f0b 62
c0cbf4d9 63 (exists $SIGIL_MAP{$sigil})
a5e51f0b 64 || confess "I do not recognize that sigil '$sigil'";
65
66 return ($name, $sigil, $SIGIL_MAP{$sigil});
c0cbf4d9 67 }
a5e51f0b 68}
6d5355c3 69
a5e51f0b 70# Class attributes
6d5355c3 71
c46b802b 72# ... these functions have to touch the symbol table itself,.. yuk
73
a5e51f0b 74sub add_package_symbol {
75 my ($self, $variable, $initial_value) = @_;
6d5355c3 76
a5e51f0b 77 my ($name, $sigil, $type) = $self->_deconstruct_variable_name($variable);
6d5355c3 78
a5e51f0b 79 no strict 'refs';
c20522bd 80 no warnings 'redefine', 'misc';
c46b802b 81 *{$self->name . '::' . $name} = ref $initial_value ? $initial_value : \$initial_value;
82}
83
84sub remove_package_glob {
85 my ($self, $name) = @_;
86 no strict 'refs';
87 delete ${$self->name . '::'}{$name};
a5e51f0b 88}
6d5355c3 89
c46b802b 90# ... these functions deal with stuff on the namespace level
91
a5e51f0b 92sub has_package_symbol {
93 my ($self, $variable) = @_;
94
95 my ($name, $sigil, $type) = $self->_deconstruct_variable_name($variable);
96
c20522bd 97 return 0 unless exists $self->namespace->{$name};
a5e51f0b 98 defined *{$self->namespace->{$name}}{$type} ? 1 : 0;
99}
100
101sub get_package_symbol {
102 my ($self, $variable) = @_;
103
104 my ($name, $sigil, $type) = $self->_deconstruct_variable_name($variable);
105
c20522bd 106 $self->add_package_symbol($variable)
107 unless exists $self->namespace->{$name};
108 return *{$self->namespace->{$name}}{$type};
a5e51f0b 109}
110
111sub remove_package_symbol {
112 my ($self, $variable) = @_;
113
114 my ($name, $sigil, $type) = $self->_deconstruct_variable_name($variable);
115
c46b802b 116 # FIXME:
117 # no doubt this is grossly inefficient and
118 # could be done much easier and faster in XS
119
120 my ($scalar, $array, $hash, $code);
a5e51f0b 121 if ($type eq 'SCALAR') {
c46b802b 122 $array = $self->get_package_symbol('@' . $name) if $self->has_package_symbol('@' . $name);
123 $hash = $self->get_package_symbol('%' . $name) if $self->has_package_symbol('%' . $name);
124 $code = $self->get_package_symbol('&' . $name) if $self->has_package_symbol('&' . $name);
a5e51f0b 125 }
126 elsif ($type eq 'ARRAY') {
c46b802b 127 $scalar = $self->get_package_symbol('$' . $name) if $self->has_package_symbol('$' . $name);
128 $hash = $self->get_package_symbol('%' . $name) if $self->has_package_symbol('%' . $name);
129 $code = $self->get_package_symbol('&' . $name) if $self->has_package_symbol('&' . $name);
a5e51f0b 130 }
131 elsif ($type eq 'HASH') {
c46b802b 132 $scalar = $self->get_package_symbol('$' . $name) if $self->has_package_symbol('$' . $name);
133 $array = $self->get_package_symbol('@' . $name) if $self->has_package_symbol('@' . $name);
134 $code = $self->get_package_symbol('&' . $name) if $self->has_package_symbol('&' . $name);
a5e51f0b 135 }
136 elsif ($type eq 'CODE') {
c46b802b 137 $scalar = $self->get_package_symbol('$' . $name) if $self->has_package_symbol('$' . $name);
138 $array = $self->get_package_symbol('@' . $name) if $self->has_package_symbol('@' . $name);
139 $hash = $self->get_package_symbol('%' . $name) if $self->has_package_symbol('%' . $name);
a5e51f0b 140 }
141 else {
142 confess "This should never ever ever happen";
7f436b8c 143 }
c46b802b 144
145 $self->remove_package_glob($name);
146
147 $self->add_package_symbol(('$' . $name) => $scalar) if defined $scalar;
148 $self->add_package_symbol(('@' . $name) => $array) if defined $array;
149 $self->add_package_symbol(('%' . $name) => $hash) if defined $hash;
150 $self->add_package_symbol(('&' . $name) => $code) if defined $code;
9d6dce77 151}
c0cbf4d9 152
9d6dce77 153sub list_all_package_symbols {
154 my ($self) = @_;
a5e51f0b 155 return keys %{$self->namespace};
6d5355c3 156}
157
2243a22b 1581;
159
160__END__
161
162=pod
163
164=head1 NAME
165
166Class::MOP::Package - Package Meta Object
167
168=head1 SYNOPSIS
169
170=head1 DESCRIPTION
171
172=head1 METHODS
173
174=over 4
175
176=item B<meta>
177
6d5355c3 178=item B<initialize>
179
180=item B<name>
181
a5e51f0b 182=item B<namespace>
183
58d75218 184=item B<add_package_symbol>
6d5355c3 185
58d75218 186=item B<get_package_symbol>
6d5355c3 187
58d75218 188=item B<has_package_symbol>
6d5355c3 189
58d75218 190=item B<remove_package_symbol>
6d5355c3 191
c46b802b 192=item B<remove_package_glob>
193
9d6dce77 194=item B<list_all_package_symbols>
195
2243a22b 196=back
197
1a09d9cc 198=head1 AUTHORS
2243a22b 199
200Stevan Little E<lt>stevan@iinteractive.comE<gt>
201
1a09d9cc 202Yuval Kogman E<lt>nothingmuch@woobling.comE<gt>
203
2243a22b 204=head1 COPYRIGHT AND LICENSE
205
206Copyright 2006 by Infinity Interactive, Inc.
207
208L<http://www.iinteractive.com>
209
210This library is free software; you can redistribute it and/or modify
211it under the same terms as Perl itself.
212
213=cut