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