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