stop using Moo as a test package
[catagits/Catalyst-Runtime.git] / t / aggregate / unit_core_component_loading.t
CommitLineData
0756fe3b 1# 2 initial tests, and 6 per component in the loop below
2# (do not forget to update the number of components in test 3 as well)
b94b200c 3# 5 extra tests for the loading options
e1dd56e6 4# One test for components in inner packages
3b5f9ebb 5use Test::More tests => 2 + 6 * 24 + 9 + 1;
0756fe3b 6
7use strict;
8use warnings;
9
10use File::Spec;
11use File::Path;
12
13my $libdir = 'test_trash';
06400669 14local @INC = @INC;
0756fe3b 15unshift(@INC, $libdir);
16
17my $appclass = 'TestComponents';
18my @components = (
19 { type => 'Controller', prefix => 'C', name => 'Bar' },
20 { type => 'Controller', prefix => 'C', name => 'Foo::Bar' },
21 { type => 'Controller', prefix => 'C', name => 'Foo::Foo::Bar' },
22 { type => 'Controller', prefix => 'C', name => 'Foo::Foo::Foo::Bar' },
23 { type => 'Controller', prefix => 'Controller', name => 'Bar::Bar::Bar::Foo' },
24 { type => 'Controller', prefix => 'Controller', name => 'Bar::Bar::Foo' },
25 { type => 'Controller', prefix => 'Controller', name => 'Bar::Foo' },
26 { type => 'Controller', prefix => 'Controller', name => 'Foo' },
27 { type => 'Model', prefix => 'M', name => 'Bar' },
28 { type => 'Model', prefix => 'M', name => 'Foo::Bar' },
29 { type => 'Model', prefix => 'M', name => 'Foo::Foo::Bar' },
30 { type => 'Model', prefix => 'M', name => 'Foo::Foo::Foo::Bar' },
31 { type => 'Model', prefix => 'Model', name => 'Bar::Bar::Bar::Foo' },
32 { type => 'Model', prefix => 'Model', name => 'Bar::Bar::Foo' },
33 { type => 'Model', prefix => 'Model', name => 'Bar::Foo' },
34 { type => 'Model', prefix => 'Model', name => 'Foo' },
35 { type => 'View', prefix => 'V', name => 'Bar' },
36 { type => 'View', prefix => 'V', name => 'Foo::Bar' },
37 { type => 'View', prefix => 'V', name => 'Foo::Foo::Bar' },
38 { type => 'View', prefix => 'V', name => 'Foo::Foo::Foo::Bar' },
39 { type => 'View', prefix => 'View', name => 'Bar::Bar::Bar::Foo' },
40 { type => 'View', prefix => 'View', name => 'Bar::Bar::Foo' },
41 { type => 'View', prefix => 'View', name => 'Bar::Foo' },
42 { type => 'View', prefix => 'View', name => 'Foo' },
43);
44
5d50f369 45sub write_component_file {
b94b200c 46 my ($dir_list, $module_name, $content) = @_;
47
48 my $dir = File::Spec->catdir(@$dir_list);
49 my $file = File::Spec->catfile($dir, $module_name . '.pm');
50
51 mkpath(join(q{/}, @$dir_list) );
52 open(my $fh, '>', $file) or die "Could not open file $file for writing: $!";
53 print $fh $content;
54 close $fh;
55}
56
0756fe3b 57sub make_component_file {
06400669 58 my ($libdir, $appclass, $type, $prefix, $name) = @_;
0756fe3b 59
60 my $compbase = "Catalyst::${type}";
61 my $fullname = "${appclass}::${prefix}::${name}";
62 my @namedirs = split(/::/, $name);
63 my $name_final = pop(@namedirs);
64 my @dir_list = ($libdir, $appclass, $prefix, @namedirs);
0756fe3b 65
b94b200c 66 write_component_file(\@dir_list, $name_final, <<EOF);
0756fe3b 67package $fullname;
f27f2c6d 68use MRO::Compat;
0756fe3b 69use base '$compbase';
70sub COMPONENT {
dbb2d5cd 71 my \$self = shift->next::method(\@_);
0756fe3b 72 no strict 'refs';
73 *{\__PACKAGE__ . "::whoami"} = sub { return \__PACKAGE__; };
74 \$self;
75}
761;
77
78EOF
0756fe3b 79}
80
81foreach my $component (@components) {
06400669 82 make_component_file(
83 $libdir,
84 $appclass,
85 $component->{type},
86 $component->{prefix},
87 $component->{name},
88 );
0756fe3b 89}
90
f27f2c6d 91my $shut_up_deprecated_warnings = q{
0f52a840 92 __PACKAGE__->log(Catalyst::Log->new('fatal'));
f27f2c6d 93};
94
95eval "package $appclass; use Catalyst; $shut_up_deprecated_warnings __PACKAGE__->setup";
0756fe3b 96
97can_ok( $appclass, 'components');
98
99my $complist = $appclass->components;
100
101# the +1 below is for the app class itself
102is(scalar keys %$complist, 24+1, "Correct number of components loaded");
103
104foreach (keys %$complist) {
105
106 # Skip the component which happens to be the app itself
107 next if $_ eq $appclass;
108
109 my $instance = $appclass->component($_);
110 isa_ok($instance, $_);
111 can_ok($instance, 'whoami');
112 is($instance->whoami, $_);
113
114 if($_ =~ /^${appclass}::(?:V|View)::(.*)/) {
115 my $moniker = $1;
116 isa_ok($instance, 'Catalyst::View');
117 can_ok($appclass->view($moniker), 'whoami');
118 is($appclass->view($moniker)->whoami, $_);
119 }
120 elsif($_ =~ /^${appclass}::(?:M|Model)::(.*)/) {
121 my $moniker = $1;
122 isa_ok($instance, 'Catalyst::Model');
123 can_ok($appclass->model($moniker), 'whoami');
124 is($appclass->model($moniker)->whoami, $_);
125 }
126 elsif($_ =~ /^${appclass}::(?:C|Controller)::(.*)/) {
127 my $moniker = $1;
128 isa_ok($instance, 'Catalyst::Controller');
129 can_ok($appclass->controller($moniker), 'whoami');
130 is($appclass->controller($moniker)->whoami, $_);
131 }
132 else {
133 die "Something is wrong with this test, this should"
134 . " have been unreachable";
135 }
136}
137
138rmtree($libdir);
18de900e 139
140# test extra component loading options
141
142$appclass = 'ExtraOptions';
143push @components, { type => 'View', prefix => 'Extra', name => 'Foo' };
144
145foreach my $component (@components) {
06400669 146 make_component_file(
147 $libdir,
148 $appclass,
149 $component->{type},
150 $component->{prefix},
151 $component->{name},
152 );
18de900e 153}
154
3b5f9ebb 155 make_component_file(
156 $libdir,
157 'ExternalExtra',
158 'Controller',
159 'Controller',
160 'FooExternal',
161 );
162
18de900e 163eval qq(
164package $appclass;
165use Catalyst;
f27f2c6d 166$shut_up_deprecated_warnings
18de900e 167__PACKAGE__->config->{ setup_components } = {
3b5f9ebb 168 search_extra => [ '::Extra', 'ExternalExtra::Controller' ],
18de900e 169 except => [ "${appclass}::Controller::Foo" ]
170};
171__PACKAGE__->setup;
172);
173
174can_ok( $appclass, 'components');
175
176$complist = $appclass->components;
177
3b5f9ebb 178is(scalar keys %$complist, 24+2, "Correct number of components loaded");
18de900e 179
2d486591 180ok( !exists $complist->{ "${appclass}::Controller::Foo" }, 'Controller::Foo was skipped' );
181ok( exists $complist->{ "${appclass}::Extra::Foo" }, 'Extra::Foo was loaded' );
18de900e 182
3b5f9ebb 183isa_ok($appclass->controller('FooExternal'), 'Catalyst::Controller', 'ExternalExtra::Controller::FooExternal was loaded');
184
b94b200c 185rmtree($libdir);
186
187$appclass = "ComponentOnce";
188
189write_component_file([$libdir, $appclass, 'Model'], 'TopLevel', <<EOF);
190package ${appclass}::Model::TopLevel;
191use base 'Catalyst::Model';
192sub COMPONENT {
5d50f369 193
dbb2d5cd 194 my \$self = shift->next::method(\@_);
b94b200c 195 no strict 'refs';
196 *{\__PACKAGE__ . "::whoami"} = sub { return \__PACKAGE__; };
e7e4c469 197 *${appclass}::Model::TopLevel::GENERATED::ACCEPT_CONTEXT = sub {
198 return bless {}, 'FooBarBazQuux';
199 };
b94b200c 200 \$self;
201}
202
203package ${appclass}::Model::TopLevel::Nested;
204
205sub COMPONENT { die "COMPONENT called in the wrong order!"; }
206
2071;
208
209EOF
210
211write_component_file([$libdir, $appclass, 'Model', 'TopLevel'], 'Nested', <<EOF);
212package ${appclass}::Model::TopLevel::Nested;
213use base 'Catalyst::Model';
214
ac424253 215my \$called=0;
b94b200c 216no warnings 'redefine';
ac424253 217sub COMPONENT { \$called++;return shift->next::method(\@_); }
218sub called { return \$called };
b94b200c 2191;
220
221EOF
222
223eval "package $appclass; use Catalyst; __PACKAGE__->setup";
224
225is($@, '', "Didn't load component twice");
ac424253 226is($appclass->model('TopLevel::Nested')->called,1, 'COMPONENT called once');
b94b200c 227
e7e4c469 228ok($appclass->model('TopLevel::Generated'), 'Have generated model');
229is(ref($appclass->model('TopLevel::Generated')), 'FooBarBazQuux',
230 'ACCEPT_CONTEXT in generated inner package fired as expected');
231
e1dd56e6 232$appclass = "InnerComponent";
233
234{
235 package InnerComponent::Controller::Test;
236 use base 'Catalyst::Controller';
237}
238
239$INC{'InnerComponent/Controller/Test.pm'} = 1;
240
241eval "package $appclass; use Catalyst; __PACKAGE__->setup";
242
243isa_ok($appclass->controller('Test'), 'Catalyst::Controller');
244
b94b200c 245rmtree($libdir);