convert all uses of Test::Exception to Test::Fatal.
[gitmo/Moose.git] / t / 050_metaclasses / 012_moose_exporter.t
1 #!/usr/bin/perl
2
3 use strict;
4 use warnings;
5
6 use Test::More;
7 use Test::Fatal;
8
9 use Test::Requires {
10     'Test::Output' => '0.01', # skip all if not installed
11 };
12
13 {
14     package HasOwnImmutable;
15
16     use Moose;
17
18     no Moose;
19
20     ::stderr_is( sub { eval q[sub make_immutable { return 'foo' }] },
21                   '',
22                   'no warning when defining our own make_immutable sub' );
23 }
24
25 {
26     is( HasOwnImmutable->make_immutable(), 'foo',
27         'HasOwnImmutable->make_immutable does not get overwritten' );
28 }
29
30 {
31     package MooseX::Empty;
32
33     use Moose ();
34     Moose::Exporter->setup_import_methods( also => 'Moose' );
35 }
36
37 {
38     package WantsMoose;
39
40     MooseX::Empty->import();
41
42     sub foo { 1 }
43
44     ::can_ok( 'WantsMoose', 'has' );
45     ::can_ok( 'WantsMoose', 'with' );
46     ::can_ok( 'WantsMoose', 'foo' );
47
48     MooseX::Empty->unimport();
49 }
50
51 {
52     # Note: it's important that these methods be out of scope _now_,
53     # after unimport was called. We tried a
54     # namespace::clean(0.08)-based solution, but had to abandon it
55     # because it cleans the namespace _later_ (when the file scope
56     # ends).
57     ok( ! WantsMoose->can('has'),  'WantsMoose::has() has been cleaned' );
58     ok( ! WantsMoose->can('with'), 'WantsMoose::with() has been cleaned' );
59     can_ok( 'WantsMoose', 'foo' );
60
61     # This makes sure that Moose->init_meta() happens properly
62     isa_ok( WantsMoose->meta(), 'Moose::Meta::Class' );
63     isa_ok( WantsMoose->new(), 'Moose::Object' );
64
65 }
66
67 {
68     package MooseX::Sugar;
69
70     use Moose ();
71
72     sub wrapped1 {
73         my $meta = shift;
74         return $meta->name . ' called wrapped1';
75     }
76
77     Moose::Exporter->setup_import_methods(
78         with_meta => ['wrapped1'],
79         also      => 'Moose',
80     );
81 }
82
83 {
84     package WantsSugar;
85
86     MooseX::Sugar->import();
87
88     sub foo { 1 }
89
90     ::can_ok( 'WantsSugar', 'has' );
91     ::can_ok( 'WantsSugar', 'with' );
92     ::can_ok( 'WantsSugar', 'wrapped1' );
93     ::can_ok( 'WantsSugar', 'foo' );
94     ::is( wrapped1(), 'WantsSugar called wrapped1',
95           'wrapped1 identifies the caller correctly' );
96
97     MooseX::Sugar->unimport();
98 }
99
100 {
101     ok( ! WantsSugar->can('has'),  'WantsSugar::has() has been cleaned' );
102     ok( ! WantsSugar->can('with'), 'WantsSugar::with() has been cleaned' );
103     ok( ! WantsSugar->can('wrapped1'), 'WantsSugar::wrapped1() has been cleaned' );
104     can_ok( 'WantsSugar', 'foo' );
105 }
106
107 {
108     package MooseX::MoreSugar;
109
110     use Moose ();
111
112     sub wrapped2 {
113         my $caller = shift->name;
114         return $caller . ' called wrapped2';
115     }
116
117     sub as_is1 {
118         return 'as_is1';
119     }
120
121     Moose::Exporter->setup_import_methods(
122         with_meta => ['wrapped2'],
123         as_is     => ['as_is1'],
124         also      => 'MooseX::Sugar',
125     );
126 }
127
128 {
129     package WantsMoreSugar;
130
131     MooseX::MoreSugar->import();
132
133     sub foo { 1 }
134
135     ::can_ok( 'WantsMoreSugar', 'has' );
136     ::can_ok( 'WantsMoreSugar', 'with' );
137     ::can_ok( 'WantsMoreSugar', 'wrapped1' );
138     ::can_ok( 'WantsMoreSugar', 'wrapped2' );
139     ::can_ok( 'WantsMoreSugar', 'as_is1' );
140     ::can_ok( 'WantsMoreSugar', 'foo' );
141     ::is( wrapped1(), 'WantsMoreSugar called wrapped1',
142           'wrapped1 identifies the caller correctly' );
143     ::is( wrapped2(), 'WantsMoreSugar called wrapped2',
144           'wrapped2 identifies the caller correctly' );
145     ::is( as_is1(), 'as_is1',
146           'as_is1 works as expected' );
147
148     MooseX::MoreSugar->unimport();
149 }
150
151 {
152     ok( ! WantsMoreSugar->can('has'),  'WantsMoreSugar::has() has been cleaned' );
153     ok( ! WantsMoreSugar->can('with'), 'WantsMoreSugar::with() has been cleaned' );
154     ok( ! WantsMoreSugar->can('wrapped1'), 'WantsMoreSugar::wrapped1() has been cleaned' );
155     ok( ! WantsMoreSugar->can('wrapped2'), 'WantsMoreSugar::wrapped2() has been cleaned' );
156     ok( ! WantsMoreSugar->can('as_is1'), 'WantsMoreSugar::as_is1() has been cleaned' );
157     can_ok( 'WantsMoreSugar', 'foo' );
158 }
159
160 {
161     package My::Metaclass;
162     use Moose;
163     BEGIN { extends 'Moose::Meta::Class' }
164
165     package My::Object;
166     use Moose;
167     BEGIN { extends 'Moose::Object' }
168
169     package HasInitMeta;
170
171     use Moose ();
172
173     sub init_meta {
174         shift;
175         return Moose->init_meta( @_,
176                                  metaclass  => 'My::Metaclass',
177                                  base_class => 'My::Object',
178                                );
179     }
180
181     Moose::Exporter->setup_import_methods( also => 'Moose' );
182 }
183
184 {
185     package NewMeta;
186
187     HasInitMeta->import();
188 }
189
190 {
191     isa_ok( NewMeta->meta(), 'My::Metaclass' );
192     isa_ok( NewMeta->new(), 'My::Object' );
193 }
194
195 {
196     package MooseX::CircularAlso;
197
198     use Moose ();
199
200     my $error;
201     ::ok($error = ::exception
202         {
203             Moose::Exporter->setup_import_methods(
204                 also => [ 'Moose', 'MooseX::CircularAlso' ],
205             );
206         },
207         'a circular reference in also dies with an error'
208     );
209
210     ::like(
211         $error,
212         qr/\QCircular reference in 'also' parameter to Moose::Exporter between MooseX::CircularAlso and MooseX::CircularAlso/,
213         'got the expected error from circular reference in also'
214     );
215 }
216
217 {
218     package MooseX::NoAlso;
219
220     use Moose ();
221
222     my $error;
223     ::ok($error = ::exception
224         {
225             Moose::Exporter->setup_import_methods(
226                 also => [ 'NoSuchThing' ],
227             );
228         },
229         'a package which does not use Moose::Exporter in also dies with an error'
230     );
231
232     ::like(
233         $error,
234         qr/\QPackage in also (NoSuchThing) does not seem to use Moose::Exporter (is it loaded?) at /,
235         'got the expected error from a reference in also to a package which is not loaded'
236     );
237 }
238
239 {
240     package MooseX::NotExporter;
241
242     use Moose ();
243
244     my $error;
245     ::ok($error = ::exception
246         {
247             Moose::Exporter->setup_import_methods(
248                 also => [ 'Moose::Meta::Method' ],
249             );
250         },
251         'a package which does not use Moose::Exporter in also dies with an error'
252     );
253
254     ::like(
255         $error,
256         qr/\QPackage in also (Moose::Meta::Method) does not seem to use Moose::Exporter at /,
257         'got the expected error from a reference in also to a package which does not use Moose::Exporter'
258     );
259 }
260
261 {
262     package MooseX::OverridingSugar;
263
264     use Moose ();
265
266     sub has {
267         my $caller = shift->name;
268         return $caller . ' called has';
269     }
270
271     Moose::Exporter->setup_import_methods(
272         with_meta => ['has'],
273         also      => 'Moose',
274     );
275 }
276
277 {
278     package WantsOverridingSugar;
279
280     MooseX::OverridingSugar->import();
281
282     ::can_ok( 'WantsOverridingSugar', 'has' );
283     ::can_ok( 'WantsOverridingSugar', 'with' );
284     ::is( has('foo'), 'WantsOverridingSugar called has',
285           'has from MooseX::OverridingSugar is called, not has from Moose' );
286
287     MooseX::OverridingSugar->unimport();
288 }
289
290 {
291     ok( ! WantsSugar->can('has'),  'WantsSugar::has() has been cleaned' );
292     ok( ! WantsSugar->can('with'), 'WantsSugar::with() has been cleaned' );
293 }
294
295 {
296     package NonExistentExport;
297
298     use Moose ();
299
300     ::stderr_like {
301         Moose::Exporter->setup_import_methods(
302             also => ['Moose'],
303             with_meta => ['does_not_exist'],
304         );
305     } qr/^Trying to export undefined sub NonExistentExport::does_not_exist/,
306       "warns when a non-existent method is requested to be exported";
307 }
308
309 {
310     package WantsNonExistentExport;
311
312     NonExistentExport->import;
313
314     ::ok(!__PACKAGE__->can('does_not_exist'),
315          "undefined subs do not get exported");
316 }
317
318 {
319     package AllOptions;
320     use Moose ();
321     use Moose::Deprecated -api_version => '0.88';
322     use Moose::Exporter;
323
324     Moose::Exporter->setup_import_methods(
325         also        => ['Moose'],
326         with_meta   => [ 'with_meta1', 'with_meta2' ],
327         with_caller => [ 'with_caller1', 'with_caller2' ],
328         as_is       => ['as_is1'],
329     );
330
331     sub with_caller1 {
332         return @_;
333     }
334
335     sub with_caller2 (&) {
336         return @_;
337     }
338
339     sub as_is1 {2}
340
341     sub with_meta1 {
342         return @_;
343     }
344
345     sub with_meta2 (&) {
346         return @_;
347     }
348 }
349
350 {
351     package UseAllOptions;
352
353     AllOptions->import();
354 }
355
356 {
357     can_ok( 'UseAllOptions', $_ )
358         for qw( with_meta1 with_meta2 with_caller1 with_caller2 as_is1 );
359
360     {
361         my ( $caller, $arg1 ) = UseAllOptions::with_caller1(42);
362         is( $caller, 'UseAllOptions', 'with_caller wrapped sub gets the right caller' );
363         is( $arg1, 42, 'with_caller wrapped sub returns argument it was passed' );
364     }
365
366     {
367         my ( $meta, $arg1 ) = UseAllOptions::with_meta1(42);
368         isa_ok( $meta, 'Moose::Meta::Class', 'with_meta first argument' );
369         is( $arg1, 42, 'with_meta1 returns argument it was passed' );
370     }
371
372     is(
373         prototype( UseAllOptions->can('with_caller2') ),
374         prototype( AllOptions->can('with_caller2') ),
375         'using correct prototype on with_meta function'
376     );
377
378     is(
379         prototype( UseAllOptions->can('with_meta2') ),
380         prototype( AllOptions->can('with_meta2') ),
381         'using correct prototype on with_meta function'
382     );
383 }
384
385 {
386     package UseAllOptions;
387     AllOptions->unimport();
388 }
389
390 {
391     ok( ! UseAllOptions->can($_), "UseAllOptions::$_ has been unimported" )
392         for qw( with_meta1 with_meta2 with_caller1 with_caller2 as_is1 );
393 }
394
395 done_testing;