more tests
[gitmo/Class-MOP.git] / t / 010_self_introspection.t
1 use strict;
2 use warnings;
3
4 use Test::More;
5 use Test::Exception;
6
7 use Class::MOP;
8 use Class::MOP::Class;
9 use Class::MOP::Package;
10 use Class::MOP::Module;
11
12 {
13     my $class = Class::MOP::Class->initialize('Foo');
14     is($class->meta, Class::MOP::Class->meta, '... instance and class both lead to the same meta');
15 }
16
17 my $class_mop_class_meta = Class::MOP::Class->meta();
18 isa_ok($class_mop_class_meta, 'Class::MOP::Class');
19
20 my $class_mop_package_meta = Class::MOP::Package->meta();
21 isa_ok($class_mop_package_meta, 'Class::MOP::Package');
22
23 my $class_mop_module_meta = Class::MOP::Module->meta();
24 isa_ok($class_mop_module_meta, 'Class::MOP::Module');
25
26 my @class_mop_package_methods = qw(
27     _new
28
29     initialize reinitialize
30
31     name
32     namespace
33
34     add_package_symbol get_package_symbol has_package_symbol remove_package_symbol
35     list_all_package_symbols get_all_package_symbols remove_package_glob
36
37     _deconstruct_variable_name
38
39     get_method_map
40 );
41
42 my @class_mop_module_methods = qw(
43     _new
44
45     _instantiate_module
46
47     version authority identifier create
48 );
49
50 my @class_mop_class_methods = qw(
51     _new
52
53     is_pristine
54
55     initialize create
56
57     update_package_cache_flag
58     reset_package_cache_flag
59
60     create_anon_class is_anon_class
61
62     instance_metaclass get_meta_instance
63     create_meta_instance _create_meta_instance
64     new_object clone_object
65     construct_instance _construct_instance
66     construct_class_instance _construct_class_instance
67     clone_instance _clone_instance
68     rebless_instance rebless_instance_back rebless_instance_away
69     check_metaclass_compatibility _check_metaclass_compatibility
70     _check_class_metaclass_compatibility _check_single_metaclass_compatibility
71     _class_metaclass_is_compatible _single_metaclass_is_compatible
72     _fix_metaclass_incompatibility _fix_class_metaclass_incompatibility
73     _fix_single_metaclass_incompatibility _base_metaclasses
74     _can_fix_class_metaclass_incompatibility_by_subclassing
75     _can_fix_single_metaclass_incompatibility_by_subclassing
76     _can_fix_metaclass_incompatibility_by_subclassing
77     _can_fix_metaclass_incompatibility
78
79     add_meta_instance_dependencies remove_meta_instance_dependencies update_meta_instance_dependencies
80     add_dependent_meta_instance remove_dependent_meta_instance
81     invalidate_meta_instances invalidate_meta_instance
82
83     superclasses subclasses direct_subclasses class_precedence_list
84     linearized_isa _superclasses_updated
85
86     alias_method get_all_method_names get_all_methods compute_all_applicable_methods
87         find_method_by_name find_all_methods_by_name find_next_method_by_name
88
89         add_before_method_modifier add_after_method_modifier add_around_method_modifier
90
91     _attach_attribute
92     _post_add_attribute
93     remove_attribute
94     find_attribute_by_name
95     get_all_attributes
96
97     compute_all_applicable_attributes
98     get_attribute_map
99
100     is_mutable is_immutable make_mutable make_immutable
101     _initialize_immutable _install_inlined_code _inlined_methods
102     _add_inlined_method _inline_accessors _inline_constructor
103     _inline_destructor _immutable_options _rebless_as_immutable
104     _rebless_as_mutable _remove_inlined_code
105
106     _immutable_metaclass
107     immutable_trait immutable_options
108     constructor_name constructor_class destructor_class
109
110     DESTROY
111 );
112
113 # check the class ...
114
115 is_deeply([ sort $class_mop_class_meta->get_method_list ], [ sort @class_mop_class_methods ], '... got the correct method list for class');
116
117 foreach my $method_name (sort @class_mop_class_methods) {
118     ok($class_mop_class_meta->has_method($method_name), '... Class::MOP::Class->has_method(' . $method_name . ')');
119     {
120         no strict 'refs';
121         is($class_mop_class_meta->get_method($method_name)->body,
122            \&{'Class::MOP::Class::' . $method_name},
123            '... Class::MOP::Class->get_method(' . $method_name . ') == &Class::MOP::Class::' . $method_name);
124     }
125 }
126
127 ## check the package ....
128
129 is_deeply([ sort $class_mop_package_meta->get_method_list ], [ sort @class_mop_package_methods ], '... got the correct method list for package');
130
131 foreach my $method_name (sort @class_mop_package_methods) {
132     ok($class_mop_package_meta->has_method($method_name), '... Class::MOP::Package->has_method(' . $method_name . ')');
133     {
134         no strict 'refs';
135         is($class_mop_package_meta->get_method($method_name)->body,
136            \&{'Class::MOP::Package::' . $method_name},
137            '... Class::MOP::Package->get_method(' . $method_name . ') == &Class::MOP::Package::' . $method_name);
138     }
139 }
140
141 ## check the module ....
142
143 is_deeply([ sort $class_mop_module_meta->get_method_list ], [ sort @class_mop_module_methods ], '... got the correct method list for module');
144
145 foreach my $method_name (sort @class_mop_module_methods) {
146     ok($class_mop_module_meta->has_method($method_name), '... Class::MOP::Module->has_method(' . $method_name . ')');
147     {
148         no strict 'refs';
149         is($class_mop_module_meta->get_method($method_name)->body,
150            \&{'Class::MOP::Module::' . $method_name},
151            '... Class::MOP::Module->get_method(' . $method_name . ') == &Class::MOP::Module::' . $method_name);
152     }
153 }
154
155
156 # check for imported functions which are not methods
157
158 foreach my $non_method_name (qw(
159     confess
160     blessed
161     subname
162     svref_2object
163     )) {
164     ok(!$class_mop_class_meta->has_method($non_method_name), '... NOT Class::MOP::Class->has_method(' . $non_method_name . ')');
165 }
166
167 # check for the right attributes
168
169 my @class_mop_package_attributes = (
170     'package',
171     'namespace',
172 );
173
174 my @class_mop_module_attributes = (
175     'version',
176     'authority'
177 );
178
179 my @class_mop_class_attributes = (
180     'superclasses',
181     'instance_metaclass',
182     'immutable_trait',
183     'constructor_name',
184     'constructor_class',
185     'destructor_class',
186 );
187
188 # check class
189
190 is_deeply(
191     [ sort $class_mop_class_meta->get_attribute_list ],
192     [ sort @class_mop_class_attributes ],
193     '... got the right list of attributes'
194 );
195
196 is_deeply(
197     [ sort keys %{$class_mop_class_meta->_attribute_map} ],
198     [ sort @class_mop_class_attributes ],
199     '... got the right list of attributes');
200
201 foreach my $attribute_name (sort @class_mop_class_attributes) {
202     ok($class_mop_class_meta->has_attribute($attribute_name), '... Class::MOP::Class->has_attribute(' . $attribute_name . ')');
203     isa_ok($class_mop_class_meta->get_attribute($attribute_name), 'Class::MOP::Attribute');
204 }
205
206 # check module
207
208 is_deeply(
209     [ sort $class_mop_package_meta->get_attribute_list ],
210     [ sort @class_mop_package_attributes ],
211     '... got the right list of attributes');
212
213 is_deeply(
214     [ sort keys %{$class_mop_package_meta->_attribute_map} ],
215     [ sort @class_mop_package_attributes ],
216     '... got the right list of attributes');
217
218 foreach my $attribute_name (sort @class_mop_package_attributes) {
219     ok($class_mop_package_meta->has_attribute($attribute_name), '... Class::MOP::Package->has_attribute(' . $attribute_name . ')');
220     isa_ok($class_mop_package_meta->get_attribute($attribute_name), 'Class::MOP::Attribute');
221 }
222
223 # check package
224
225 is_deeply(
226     [ sort $class_mop_module_meta->get_attribute_list ],
227     [ sort @class_mop_module_attributes ],
228     '... got the right list of attributes');
229
230 is_deeply(
231     [ sort keys %{$class_mop_module_meta->_attribute_map} ],
232     [ sort @class_mop_module_attributes ],
233     '... got the right list of attributes');
234
235 foreach my $attribute_name (sort @class_mop_module_attributes) {
236     ok($class_mop_module_meta->has_attribute($attribute_name), '... Class::MOP::Module->has_attribute(' . $attribute_name . ')');
237     isa_ok($class_mop_module_meta->get_attribute($attribute_name), 'Class::MOP::Attribute');
238 }
239
240 ## check the attributes themselves
241
242 # ... package
243
244 ok($class_mop_package_meta->get_attribute('package')->has_reader, '... Class::MOP::Class package has a reader');
245 is(ref($class_mop_package_meta->get_attribute('package')->reader), 'HASH', '... Class::MOP::Class package\'s a reader is { name => sub { ... } }');
246
247 ok($class_mop_package_meta->get_attribute('package')->has_init_arg, '... Class::MOP::Class package has a init_arg');
248 is($class_mop_package_meta->get_attribute('package')->init_arg, 'package', '... Class::MOP::Class package\'s a init_arg is package');
249
250 # ... class, but inherited from HasMethods
251 ok($class_mop_class_meta->find_attribute_by_name('method_metaclass')->has_reader, '... Class::MOP::Class method_metaclass has a reader');
252 is_deeply($class_mop_class_meta->find_attribute_by_name('method_metaclass')->reader,
253    { 'method_metaclass' => \&Class::MOP::Mixin::HasMethods::method_metaclass },
254    '... Class::MOP::Class method_metaclass\'s a reader is &method_metaclass');
255
256 ok($class_mop_class_meta->find_attribute_by_name('method_metaclass')->has_init_arg, '... Class::MOP::Class method_metaclass has a init_arg');
257 is($class_mop_class_meta->find_attribute_by_name('method_metaclass')->init_arg,
258   'method_metaclass',
259   '... Class::MOP::Class method_metaclass\'s init_arg is method_metaclass');
260
261 ok($class_mop_class_meta->find_attribute_by_name('method_metaclass')->has_default, '... Class::MOP::Class method_metaclass has a default');
262 is($class_mop_class_meta->find_attribute_by_name('method_metaclass')->default,
263    'Class::MOP::Method',
264   '... Class::MOP::Class method_metaclass\'s a default is Class::MOP:::Method');
265
266 ok($class_mop_class_meta->find_attribute_by_name('wrapped_method_metaclass')->has_reader, '... Class::MOP::Class wrapped_method_metaclass has a reader');
267 is_deeply($class_mop_class_meta->find_attribute_by_name('wrapped_method_metaclass')->reader,
268    { 'wrapped_method_metaclass' => \&Class::MOP::Mixin::HasMethods::wrapped_method_metaclass },
269    '... Class::MOP::Class wrapped_method_metaclass\'s a reader is &wrapped_method_metaclass');
270
271 ok($class_mop_class_meta->find_attribute_by_name('wrapped_method_metaclass')->has_init_arg, '... Class::MOP::Class wrapped_method_metaclass has a init_arg');
272 is($class_mop_class_meta->find_attribute_by_name('wrapped_method_metaclass')->init_arg,
273   'wrapped_method_metaclass',
274   '... Class::MOP::Class wrapped_method_metaclass\'s init_arg is wrapped_method_metaclass');
275
276 ok($class_mop_class_meta->find_attribute_by_name('method_metaclass')->has_default, '... Class::MOP::Class method_metaclass has a default');
277 is($class_mop_class_meta->find_attribute_by_name('method_metaclass')->default,
278    'Class::MOP::Method',
279   '... Class::MOP::Class method_metaclass\'s a default is Class::MOP:::Method');
280
281
282 # ... class, but inherited from HasAttributes
283
284 ok($class_mop_class_meta->find_attribute_by_name('attributes')->has_reader, '... Class::MOP::Class attributes has a reader');
285 is_deeply($class_mop_class_meta->find_attribute_by_name('attributes')->reader,
286    { '_attribute_map' => \&Class::MOP::Mixin::HasAttributes::_attribute_map },
287    '... Class::MOP::Class attributes\'s a reader is &_attribute_map');
288
289 ok($class_mop_class_meta->find_attribute_by_name('attributes')->has_init_arg, '... Class::MOP::Class attributes has a init_arg');
290 is($class_mop_class_meta->find_attribute_by_name('attributes')->init_arg,
291   'attributes',
292   '... Class::MOP::Class attributes\'s a init_arg is attributes');
293
294 ok($class_mop_class_meta->find_attribute_by_name('attributes')->has_default, '... Class::MOP::Class attributes has a default');
295 is_deeply($class_mop_class_meta->find_attribute_by_name('attributes')->default('Foo'),
296          {},
297          '... Class::MOP::Class attributes\'s a default of {}');
298
299 ok($class_mop_class_meta->find_attribute_by_name('attribute_metaclass')->has_reader, '... Class::MOP::Class attribute_metaclass has a reader');
300 is_deeply($class_mop_class_meta->find_attribute_by_name('attribute_metaclass')->reader,
301    { 'attribute_metaclass' => \&Class::MOP::Mixin::HasAttributes::attribute_metaclass },
302   '... Class::MOP::Class attribute_metaclass\'s a reader is &attribute_metaclass');
303
304 ok($class_mop_class_meta->find_attribute_by_name('attribute_metaclass')->has_init_arg, '... Class::MOP::Class attribute_metaclass has a init_arg');
305 is($class_mop_class_meta->find_attribute_by_name('attribute_metaclass')->init_arg,
306    'attribute_metaclass',
307    '... Class::MOP::Class attribute_metaclass\'s a init_arg is attribute_metaclass');
308
309 ok($class_mop_class_meta->find_attribute_by_name('attribute_metaclass')->has_default, '... Class::MOP::Class attribute_metaclass has a default');
310 is($class_mop_class_meta->find_attribute_by_name('attribute_metaclass')->default,
311   'Class::MOP::Attribute',
312   '... Class::MOP::Class attribute_metaclass\'s a default is Class::MOP:::Attribute');
313
314 # check the values of some of the methods
315
316 is($class_mop_class_meta->name, 'Class::MOP::Class', '... Class::MOP::Class->name');
317 is($class_mop_class_meta->version, $Class::MOP::Class::VERSION, '... Class::MOP::Class->version');
318
319 ok($class_mop_class_meta->has_package_symbol('$VERSION'), '... Class::MOP::Class->has_package_symbol($VERSION)');
320 is(${$class_mop_class_meta->get_package_symbol('$VERSION')},
321    $Class::MOP::Class::VERSION,
322    '... Class::MOP::Class->get_package_symbol($VERSION)');
323
324 is_deeply(
325     [ $class_mop_class_meta->superclasses ],
326     [ qw/Class::MOP::Module Class::MOP::Mixin::HasAttributes Class::MOP::Mixin::HasMethods/ ],
327     '... Class::MOP::Class->superclasses == [ Class::MOP::Module ]');
328
329 is_deeply(
330     [ $class_mop_class_meta->class_precedence_list ],
331     [ qw/
332         Class::MOP::Class
333         Class::MOP::Module
334         Class::MOP::Package
335         Class::MOP::Object
336         Class::MOP::Mixin::HasAttributes
337         Class::MOP::Mixin
338         Class::MOP::Mixin::HasMethods
339         Class::MOP::Mixin
340     / ],
341     '... Class::MOP::Class->class_precedence_list == [ Class::MOP::Class Class::MOP::Module Class::MOP::Package ]');
342
343 is($class_mop_class_meta->attribute_metaclass, 'Class::MOP::Attribute', '... got the right value for attribute_metaclass');
344 is($class_mop_class_meta->method_metaclass, 'Class::MOP::Method', '... got the right value for method_metaclass');
345 is($class_mop_class_meta->instance_metaclass, 'Class::MOP::Instance', '... got the right value for instance_metaclass');
346
347 done_testing;