Add a test that Person->full_name actually uses the predicate method
[gitmo/moose-presentations.git] / moose-class / exercises / t / lib / MooseClass / Tests.pm
1 package MooseClass::Tests;
2
3 use strict;
4 use warnings;
5
6 use Lingua::EN::Inflect qw( A PL_N );
7 use Test::More 'no_plan';
8
9 sub tests01 {
10     my %p = (
11         person_attr_count   => 2,
12         employee_attr_count => 3,
13         @_,
14     );
15
16     local $Test::Builder::Level = $Test::Builder::Level + 1;
17
18     has_meta('Person');
19
20     check_isa( 'Person', ['Moose::Object'] );
21
22     count_attrs( 'Person', $p{person_attr_count} );
23
24     has_rw_attr( 'Person', $_ ) for qw( first_name last_name );
25
26     has_method( 'Person', 'full_name' );
27
28     no_droppings('Person');
29     is_immutable('Person');
30
31     person01();
32
33     has_meta('Employee');
34
35     check_isa( 'Employee', [ 'Person', 'Moose::Object' ] );
36
37     count_attrs( 'Employee', $p{employee_attr_count} );
38
39     has_rw_attr( 'Employee', $_ ) for qw( title salary );
40     has_ro_attr( 'Employee', 'ssn' );
41
42     has_overridden_method( 'Employee', 'full_name' );
43
44     employee01();
45 }
46
47 sub tests02 {
48     tests01( person_attr_count => 3, @_ );
49
50     local $Test::Builder::Level = $Test::Builder::Level + 1;
51
52     no_droppings($_) for qw( Printable HasAccount );
53
54     does_role( 'Person', $_ ) for qw( Printable HasAccount );
55     has_method( 'Person', $_ ) for qw( as_string deposit withdraw );
56     has_rw_attr( 'Person', 'balance' );
57
58     does_role( 'Employee', $_ ) for qw( Printable HasAccount );
59
60     person02();
61     employee02();
62 }
63
64 sub tests03 {
65     {
66         local $Test::Builder::Level = $Test::Builder::Level + 1;
67
68         has_meta('Person');
69         has_meta('Employee');
70
71         has_rw_attr( 'Person', 'title' );
72
73         has_rw_attr( 'Employee', 'title' );
74         has_rw_attr( 'Employee', 'salary_level' );
75         has_ro_attr( 'Employee', 'salary' );
76
77         has_method( 'Employee', '_build_salary' );
78     }
79
80     ok( ! Employee->meta->has_method('full_name'),
81         'Employee no longer implements a full_name method' );
82
83     my $person_title_attr = Person->meta->get_attribute('title');
84     ok( !$person_title_attr->is_required, 'title is not required in Person' );
85     is( $person_title_attr->predicate, 'has_title',
86         'Person title attr has a has_title predicate' );
87     is( $person_title_attr->clearer, 'clear_title',
88         'Person title attr has a clear_title clearer' );
89
90     my $balance_attr = Person->meta->get_attribute('balance');
91     is( $balance_attr->default, 100, 'balance defaults to 100' );
92
93     my $employee_title_attr = Employee->meta->get_attribute('title');
94     is( $employee_title_attr->default, 'Worker',
95         'title defaults to Worker in Employee' );
96
97     my $salary_level_attr = Employee->meta->get_attribute('salary_level');
98     is( $salary_level_attr->default, 1, 'salary_level defaults to 1' );
99
100     my $salary_attr = Employee->meta->get_attribute('salary');
101     ok( !$salary_attr->init_arg,   'no init_arg for salary attribute' );
102     ok( $salary_attr->has_builder, 'salary attr has a builder' );
103
104     person03();
105     employee03();
106 }
107
108 sub tests04 {
109     {
110         local $Test::Builder::Level = $Test::Builder::Level + 1;
111
112         no_droppings('OutputsXML');
113
114         does_role( 'Person', 'OutputsXML' );
115     }
116
117     ok( scalar OutputsXML->meta->get_around_method_modifiers('as_xml'),
118         'OutputsXML has an around modifier for as_xml' );
119
120     isa_ok( Employee->meta->get_method('as_xml'),
121             'Moose::Meta::Method::Augmented', 'as_xml is augmented in Employee' );
122
123     person04();
124     employee04();
125 }
126
127 sub tests06 {
128     {
129         local $Test::Builder::Level = $Test::Builder::Level + 1;
130
131         has_meta('BankAccount');
132         no_droppings('BankAccount');
133
134         has_rw_attr( 'BankAccount', 'balance' );
135         has_rw_attr( 'BankAccount', 'owner' );
136         has_ro_attr( 'BankAccount', 'history' );
137     }
138
139     my $person_meta = Person->meta;
140     ok( ! $person_meta->has_attribute('balance'),
141         'Person class does not have a balance attribute' );
142
143     my $deposit_meth = $person_meta->get_method('deposit');
144     isa_ok( $deposit_meth, 'Moose::Meta::Method::Delegation' );
145
146     my $withdraw_meth = $person_meta->get_method('withdraw');
147     isa_ok( $withdraw_meth, 'Moose::Meta::Method::Delegation' );
148
149     my $ba_meta = BankAccount->meta;
150     ok( $ba_meta->get_attribute('owner')->is_weak_ref,
151         'owner attribute is a weak ref' );
152
153     person06();
154 }
155
156
157 sub has_meta {
158     my $class = shift;
159
160     ok( $class->can('meta'), "$class has a meta() method" )
161         or BAIL_OUT("Cannot run tests against a class without a meta!");
162 }
163
164 sub check_isa {
165     my $class   = shift;
166     my $parents = shift;
167
168     my @isa = $class->meta->linearized_isa;
169     shift @isa;    # returns $class as the first entry
170
171     my $count = scalar @{$parents};
172     my $noun = PL_N( 'parent', $count );
173
174     is( scalar @isa, $count, "$class has $count $noun" );
175
176     for ( my $i = 0; $i < @{$parents}; $i++ ) {
177         is( $isa[$i], $parents->[$i], "parent[$i] is $parents->[$i]" );
178     }
179 }
180
181 sub count_attrs {
182     my $class = shift;
183     my $count = shift;
184
185     my $noun = PL_N( 'attribute', $count );
186     is( scalar $class->meta->get_attribute_list, $count,
187         "$class defines $count $noun" );
188 }
189
190 sub has_rw_attr {
191     my $class = shift;
192     my $name  = shift;
193
194     my $articled = A($name);
195     ok( $class->meta->has_attribute($name),
196         "$class has $articled attribute" );
197
198     my $attr = $class->meta->get_attribute($name);
199
200     is( $attr->get_read_method, $name,
201         "$name attribute has a reader accessor - $name()" );
202     is( $attr->get_write_method, $name,
203         "$name attribute has a writer accessor - $name()" );
204 }
205
206 sub has_ro_attr {
207     my $class = shift;
208     my $name  = shift;
209
210     my $articled = A($name);
211     ok( $class->meta->has_attribute($name),
212         "$class has $articled attribute" );
213
214     my $attr = $class->meta->get_attribute($name);
215
216     is( $attr->get_read_method, $name,
217         "$name attribute has a reader accessor - $name()" );
218     is( $attr->get_write_method, undef,
219         "$name attribute does not have a writer" );
220 }
221
222 sub has_method {
223     my $class = shift;
224     my $name  = shift;
225
226     my $articled = A($name);
227     ok( $class->meta->has_method($name), "$class has $articled method" );
228 }
229
230 sub has_overridden_method {
231     my $class = shift;
232     my $name  = shift;
233
234     my $articled = A($name);
235     ok( $class->meta->has_method($name), "$class has $articled method" );
236
237     my $meth = $class->meta->get_method($name);
238     isa_ok( $meth, 'Moose::Meta::Method::Overridden' );
239 }
240
241 sub no_droppings {
242     my $class = shift;
243
244     ok( !$class->can('has'), "no Moose droppings in $class" );
245 }
246
247 sub is_immutable {
248     my $class = shift;
249
250     ok( $class->meta->is_immutable, "$class has been made immutable" );
251 }
252
253 sub does_role {
254     my $class = shift;
255     my $role  = shift;
256
257     ok( $class->meta->does_role($role), "$class does the $role role" );
258 }
259
260 sub person01 {
261     my $person = Person->new(
262         first_name => 'Bilbo',
263         last_name  => 'Baggins',
264     );
265
266     is( $person->full_name, 'Bilbo Baggins',
267         'full_name() is correctly implemented' );
268
269     $person = Person->new( [ qw( Lisa Smith ) ] );
270     is( $person->first_name, 'Lisa', 'set first_name from two-arg arrayref' );
271     is( $person->last_name, 'Smith', 'set last_name from two-arg arrayref' );
272
273     eval { Person->new( sub { 'foo' } ) };
274     like( $@, qr/\QSingle parameters to new() must be a HASH ref/,
275           'Person constructor still rejects bad parameters' );
276 }
277
278 sub employee01 {
279     my $employee = Employee->new(
280         first_name => 'Amanda',
281         last_name  => 'Palmer',
282         title      => 'Singer',
283     );
284
285     is(        $employee->full_name, 'Amanda Palmer (Singer)',        'full_name() is properly overriden in Employee'    );
286 }
287
288 sub person02 {
289     my $person = Person->new(
290         first_name => 'Bilbo',
291         last_name  => 'Baggins',
292         balance    => 0,
293     );
294
295     is( $person->as_string, 'Bilbo Baggins',
296         'as_string() is correctly implemented' );
297
298     account_tests($person);
299 }
300
301 sub employee02 {
302     my $employee = Employee->new(
303         first_name => 'Amanda',
304         last_name  => 'Palmer',
305         title      => 'Singer',
306         balance    => 0,
307     );
308
309     is( $employee->as_string, 'Amanda Palmer (Singer)',
310         'as_string() uses overridden full_name method in Employee' );
311
312     account_tests($employee);
313 }
314
315 sub person03 {
316     my $person = Person->new(
317         first_name => 'Bilbo',
318         last_name  => 'Baggins',
319     );
320
321     is( $person->full_name, 'Bilbo Baggins',
322         'full_name() is correctly implemented for a Person without a title' );
323     ok( !$person->has_title,
324         'Person has_title predicate is working correctly (returns false)' );
325
326     $person->title('Ringbearer');
327     ok( $person->has_title, 'Person has_title predicate is working correctly (returns true)' );
328
329     Person->meta->make_mutable;
330     my $called = 0;
331     Person->meta->add_after_method_modifier( 'has_title' => sub { $called++ } );
332     is( $person->full_name, 'Bilbo Baggins (Ringbearer)',
333         'full_name() is correctly implemented for a Person with a title' );
334     ok( $called, 'full_name in person uses the predicate for the title attribute' );
335
336     $person->clear_title;
337     ok( !$person->has_title, 'Person clear_title method cleared the title' );
338
339     account_tests( $person, 100 );
340 }
341
342 sub employee03 {
343     my $employee = Employee->new(
344         first_name   => 'Jimmy',
345         last_name    => 'Foo',
346         salary_level => 3,
347         salary       => 42,
348     );
349
350     is( $employee->salary, 30000,
351         'salary is calculated from salary_level, and salary passed to constructor is ignored' );
352 }
353
354
355 sub person04 {
356     my $person = Person->new(
357         first_name => 'Bilbo',
358         last_name  => 'Baggins',
359     );
360
361     my $xml = <<'EOF';
362 <?xml version="1.0" encoding="UTF-8"?>
363 <Person>
364 <first_name>Bilbo</first_name>
365 <last_name>Baggins</last_name>
366 <title></title>
367 </Person>
368 EOF
369
370     is( $person->as_xml, $xml, 'Person outputs expected XML' );
371 }
372
373 sub employee04 {
374     my $employee = Employee->new(
375         first_name   => 'Jimmy',
376         last_name    => 'Foo',
377         ssn          => '123-99-4567',
378         salary_level => 3,
379     );
380
381     my $xml = <<'EOF';
382 <?xml version="1.0" encoding="UTF-8"?>
383 <Employee>
384 <first_name>Jimmy</first_name>
385 <last_name>Foo</last_name>
386 <title>Worker</title>
387 <salary>30000</salary>
388 <salary_level>3</salary_level>
389 <ssn>123-99-4567</ssn>
390 </Employee>
391 EOF
392
393     is( $employee->as_xml, $xml, 'Employee outputs expected XML' );
394 }
395
396 sub person06 {
397     my $person = Person->new(
398         first_name => 'Bilbo',
399         last_name  => 'Baggins',
400     );
401
402     isa_ok( $person->account, 'BankAccount' );
403     is( $person->account->owner, $person,
404         'owner of bank account is person that created account' );
405
406     $person->deposit(10);
407     is_deeply( $person->account->history, [ 100, 10 ],
408                'deposit was recorded in account history' );
409
410     $person->withdraw(15);
411     is_deeply( $person->account->history, [ 100, 10, -15 ],
412                'withdrawal was recorded in account history' );
413 }
414
415 sub account_tests {
416     local $Test::Builder::Level = $Test::Builder::Level + 1;
417
418     my $person = shift;
419     my $base_amount = shift || 0;
420
421     $person->deposit(50);
422     eval { $person->withdraw( 75 + $base_amount ) };
423     like( $@, qr/\QBalance cannot be negative/,
424           'cannot withdraw more than is in our balance' );
425
426     $person->withdraw( 23 );
427
428     is( $person->balance, 27 + $base_amount,
429         'balance is 27 (+ starting balance) after deposit of 50 and withdrawal of 23' );
430 }
431
432 1;