use Mojo::Base -strict;
BEGIN {
$ENV{PLACK_ENV} = undef;
$ENV{MOJO_MODE} = 'development';
$ENV{MOJO_NO_IPV6} = 1;
$ENV{MOJO_REACTOR} = 'Mojo::Reactor::Poll';
}
use Test::More;
use FindBin;
use lib "$FindBin::Bin/lib";
use File::Spec::Functions 'catdir';
use Mojo::Date;
use Mojolicious;
use Test::Mojo;
# Missing config file
{
eval { Test::Mojo->new('MojoliciousConfigTest')->app };
like $@, qr/mojolicious_config_test.conf" missing/, 'right error';
local $ENV{MOJO_MODE} = 'whatever';
is(Test::Mojo->new('MojoliciousConfigTest')->app->config->{it},
'works', 'right result');
}
# Mode detection
{
local $ENV{MOJO_MODE} = undef;
local $ENV{PLACK_ENV} = 'something';
is(Test::Mojo->new('MojoliciousTest')->app->mode, 'something', 'right mode');
}
{
local $ENV{MOJO_MODE} = 'else';
local $ENV{PLACK_ENV} = 'something';
is(Test::Mojo->new('MojoliciousTest')->app->mode, 'else', 'right mode');
}
my $t = Test::Mojo->new('MojoliciousTest');
# Application is already available
is $t->app->routes->find('something')->to_string, '/test4/:something',
'right pattern';
is $t->app->routes->find('test3')->pattern->defaults->{namespace},
'MojoliciousTestController', 'right namespace';
is $t->app->routes->find('withblock')->pattern->defaults->{controller}, 'foo',
'right controller';
is ref $t->app->routes->find('something'), 'Mojolicious::Routes::Route',
'right class';
is ref $t->app->routes->find('something')->root, 'Mojolicious::Routes',
'right class';
is $t->app->sessions->cookie_domain, '.example.com', 'right domain';
is $t->app->sessions->cookie_path, '/bar', 'right path';
is_deeply $t->app->commands->namespaces,
[qw(Mojolicious::Command MojoliciousTest::Command)], 'right namespaces';
is $t->app, $t->app->commands->app, 'applications are equal';
is $t->app->static->file('hello.txt')->slurp,
"Hello Mojo from a development static file!\n", 'right content';
is $t->app->moniker, 'mojolicious_test', 'right moniker';
is $t->app->secrets->[0], $t->app->moniker, 'secret defaults to moniker';
is $t->app->renderer->template_handler(
{template => 'foo/bar/index', format => 'html'}), 'epl', 'right handler';
is $t->app->build_controller->req->url, '', 'no URL';
is $t->app->build_controller->render_to_string('does_not_exist'), undef,
'no result';
# Missing methods and functions (AUTOLOAD)
eval { $t->app->missing };
like $@,
qr/^Can't locate object method "missing" via package "MojoliciousTest"/,
'right error';
eval { Mojolicious::missing() };
like $@, qr/^Undefined subroutine &Mojolicious::missing called/, 'right error';
my $c = $t->app->build_controller;
eval { $c->missing };
like $@, qr/^Can't locate object method "missing" via package "@{[ref $c]}"/,
'right error';
eval { Mojolicious::Controller::missing() };
like $@, qr/^Undefined subroutine &Mojolicious::Controller::missing called/,
'right error';
eval { $t->app->routes->missing };
like $@,
qr/^Can't locate object method "missing" via package "Mojolicious::Routes"/,
'right error';
eval { Mojolicious::Route::missing() };
like $@, qr/^Undefined subroutine &Mojolicious::Route::missing called/,
'right error';
# Hidden controller attributes and methods
$t->app->routes->hide('bar');
ok !$t->app->routes->is_hidden('foo'), 'not hidden';
ok $t->app->routes->is_hidden('bar'), 'is hidden';
ok $t->app->routes->is_hidden('_foo'), 'is hidden';
ok $t->app->routes->is_hidden('AUTOLOAD'), 'is hidden';
ok $t->app->routes->is_hidden('DESTROY'), 'is hidden';
ok $t->app->routes->is_hidden('FOO_BAR'), 'is hidden';
ok $t->app->routes->is_hidden('app'), 'is hidden';
ok $t->app->routes->is_hidden('attr'), 'is hidden';
ok $t->app->routes->is_hidden('continue'), 'is hidden';
ok $t->app->routes->is_hidden('cookie'), 'is hidden';
ok $t->app->routes->is_hidden('finish'), 'is hidden';
ok $t->app->routes->is_hidden('flash'), 'is hidden';
ok $t->app->routes->is_hidden('handler'), 'is hidden';
ok $t->app->routes->is_hidden('has'), 'is hidden';
ok $t->app->routes->is_hidden('helpers'), 'is hidden';
ok $t->app->routes->is_hidden('match'), 'is hidden';
ok $t->app->routes->is_hidden('new'), 'is hidden';
ok $t->app->routes->is_hidden('on'), 'is hidden';
ok $t->app->routes->is_hidden('param'), 'is hidden';
ok $t->app->routes->is_hidden('redirect_to'), 'is hidden';
ok $t->app->routes->is_hidden('render'), 'is hidden';
ok $t->app->routes->is_hidden('render_exception'), 'is hidden';
ok $t->app->routes->is_hidden('render_later'), 'is hidden';
ok $t->app->routes->is_hidden('render_maybe'), 'is hidden';
ok $t->app->routes->is_hidden('render_not_found'), 'is hidden';
ok $t->app->routes->is_hidden('render_static'), 'is hidden';
ok $t->app->routes->is_hidden('render_to_string'), 'is hidden';
ok $t->app->routes->is_hidden('rendered'), 'is hidden';
ok $t->app->routes->is_hidden('req'), 'is hidden';
ok $t->app->routes->is_hidden('res'), 'is hidden';
ok $t->app->routes->is_hidden('respond_to'), 'is hidden';
ok $t->app->routes->is_hidden('send'), 'is hidden';
ok $t->app->routes->is_hidden('session'), 'is hidden';
ok $t->app->routes->is_hidden('signed_cookie'), 'is hidden';
ok $t->app->routes->is_hidden('stash'), 'is hidden';
ok $t->app->routes->is_hidden('tap'), 'is hidden';
ok $t->app->routes->is_hidden('tx'), 'is hidden';
ok $t->app->routes->is_hidden('url_for'), 'is hidden';
ok $t->app->routes->is_hidden('validation'), 'is hidden';
ok $t->app->routes->is_hidden('write'), 'is hidden';
ok $t->app->routes->is_hidden('write_chunk'), 'is hidden';
# Unknown hooks
ok !$t->app->plugins->emit_chain('does_not_exist'), 'hook has been emitted';
ok !!$t->app->plugins->emit_hook('does_not_exist'), 'hook has been emitted';
ok !!$t->app->plugins->emit_hook_reverse('does_not_exist'),
'hook has been emitted';
# Replaced helper
my $log = '';
my $cb = $t->app->log->on(message => sub { $log .= pop });
$t->app->helper(replaced_helper => sub { });
$t->app->helper(replaced_helper => sub { });
like $log, qr/Helper "replaced_helper" already exists, replacing\./,
'right message';
$t->app->log->unsubscribe(message => $cb);
# Custom hooks
my $custom;
$t->app->hook('custom_hook' => sub { $custom += shift });
$t->app->plugins->emit_hook(custom_hook => 1);
is $custom, 1, 'hook has been emitted';
$t->app->plugins->emit_hook_reverse(custom_hook => 2);
is $custom, 3, 'hook has been emitted again';
$t->app->hook('custom_chain' => sub { return shift->() * 2 });
$t->app->hook('custom_chain' => sub { return pop });
is $t->app->plugins->emit_chain(custom_chain => 4), 8, 'hook has been emitted';
# MojoliciousTest::Command::test_command (with abbreviation)
is $t->app->start(qw(test_command --to)), 'works too!', 'right result';
# Plugin::Test::SomePlugin2::register (security violation)
$t->get_ok('/plugin-test-some_plugin2/register')->status_isnt(500)
->status_is(404)->header_is(Server => 'Mojolicious (Perl)')
->content_unlike(qr/Something/)->content_like(qr/Page not found/);
# Plugin::Test::SomePlugin2::register (security violation again)
$log = '';
$cb = $t->app->log->on(message => sub { $log .= pop });
$t->get_ok('/plugin-test-some_plugin2/register')->status_isnt(500)
->status_is(404)->header_is(Server => 'Mojolicious (Perl)')
->content_unlike(qr/Something/)->content_like(qr/Page not found/);
like $log,
qr/Class "MojoliciousTest::Plugin::Test::SomePlugin2" is not a controller\./,
'right message';
$t->app->log->unsubscribe(message => $cb);
# Foo::fun
my $url = $t->ua->server->url;
$url->path('/fun/time');
$t->get_ok($url => {'X-Test' => 'Hi there!'})->status_isnt(404)
->status_is(200)->header_isnt('X-Bender' => 'Bite my shiny metal ass!')
->header_unlike('X-Bender' => qr/shiny metal/)
->header_is('X-Bender' => undef)->header_is(Server => 'Mojolicious (Perl)')
->header_unlike(Server => qr/Bender/)->content_isnt('Have')
->content_is('Have fun!');
# Foo::baz (missing action without template)
$log = '';
$cb = $t->app->log->on(message => sub { $log .= pop });
$t->get_ok('/foo/baz')->status_is(404)
->header_is(Server => 'Mojolicious (Perl)')->content_unlike(qr/Something/)
->content_like(qr/Page not found/);
like $log, qr/Action not found in controller\./, 'right message';
$t->app->log->unsubscribe(message => $cb);
# Foo::render (action not allowed)
$log = '';
$cb = $t->app->log->on(message => sub { $log .= pop });
$t->get_ok('/foo/render')->status_is(404)
->header_is(Server => 'Mojolicious (Perl)')
->content_like(qr/Page not found/);
like $log, qr/Action "render" is not allowed\./, 'right message';
$t->app->log->unsubscribe(message => $cb);
# Foo::yada (action-less template)
$t->get_ok('/foo/yada')->status_is(200)
->header_is(Server => 'Mojolicious (Perl)')
->content_like(qr/look ma! no action!/);
# SyntaxError::foo (syntax error in controller)
$t->get_ok('/syntax_error/foo')->status_is(500)
->header_is(Server => 'Mojolicious (Perl)')
->content_like(qr/Missing right curly/);
# Foo::syntaxerror (syntax error in template)
$log = '';
$cb = $t->app->log->on(message => sub { $log .= pop });
$t->get_ok('/foo/syntaxerror')->status_is(500)
->header_is(Server => 'Mojolicious (Perl)')
->content_like(qr/Missing right curly/);
like $log, qr/Rendering template "syntaxerror.html.epl"\./, 'right message';
like $log, qr/Missing right curly/, 'right message';
like $log, qr/Template "exception.development.html.ep" not found\./,
'right message';
like $log, qr/Rendering template "exception.html.epl"\./, 'right message';
like $log, qr/500 Internal Server Error/, 'right message';
$t->app->log->unsubscribe(message => $cb);
# Exceptional::this_one_dies (action dies)
$t->get_ok('/exceptional/this_one_dies')->status_is(500)
->header_is(Server => 'Mojolicious (Perl)')->content_is("doh!\n\n");
# Exceptional::this_one_might_die (bridge dies)
$t->get_ok('/exceptional_too/this_one_dies')->status_is(500)
->header_is(Server => 'Mojolicious (Perl)')->content_is("double doh!\n\n");
# Exceptional::this_one_dies (action behind bridge dies)
$t->get_ok('/exceptional_too/this_one_dies' => {'X-DoNotDie' => 1})
->status_is(500)->header_is(Server => 'Mojolicious (Perl)')
->content_is("doh!\n\n");
# Exceptional::this_one_does_not_exist (action does not exist)
$t->get_ok('/exceptional/this_one_does_not_exist')->status_is(404)
->header_is(Server => 'Mojolicious (Perl)')
->content_like(qr/Page not found/);
# Exceptional::this_one_does_not_exist (action behind bridge does not exist)
$t->get_ok('/exceptional_too/this_one_does_not_exist' => {'X-DoNotDie' => 1})
->status_is(404)->header_is(Server => 'Mojolicious (Perl)')
->content_like(qr/Page not found/);
# Foo::fun
$t->get_ok('/fun/time' => {'X-Test' => 'Hi there!'})->status_is(200)
->header_is('X-Bender' => undef)->header_is(Server => 'Mojolicious (Perl)')
->content_is('Have fun!');
# Foo::fun
$url = $t->ua->server->url;
$log = '';
$cb = $t->app->log->on(message => sub { $log .= pop });
$url->path('/fun/time');
$t->get_ok($url => {'X-Test' => 'Hi there!'})->status_is(200)
->header_is('X-Bender' => undef)->header_is(Server => 'Mojolicious (Perl)')
->content_is('Have fun!');
like $log,
qr!Rendering cached template "foo/fun\.html\.ep" from DATA section\.!,
'right message';
$t->app->log->unsubscribe(message => $cb);
# Foo::fun
$t->get_ok('/happy/fun/time' => {'X-Test' => 'Hi there!'})->status_is(200)
->header_is('X-Bender' => undef)->header_is(Server => 'Mojolicious (Perl)')
->content_is('Have fun!');
# Foo::test
$t->get_ok('/foo/test' => {'X-Test' => 'Hi there!'})->status_is(200)
->header_is('X-Bender' => 'Bite my shiny metal ass!')
->header_is(Server => 'Mojolicious (Perl)')->content_like(qr!/bar/test!);
# Foo::index
$t->get_ok('/foo' => {'X-Test' => 'Hi there!'})->status_is(200)
->header_is(Server => 'Mojolicious (Perl)')
->content_like(qr|<body>\s+23\nHello Mojo from the template /foo! He|);
# Foo::Bar::index
$t->get_ok('/foo-bar' => {'X-Test' => 'Hi there!'})->status_is(200)
->header_is(Server => 'Mojolicious (Perl)')
->content_like(qr|Hello Mojo from the other template /foo-bar!|);
# Foo::something
$t->get_ok('/somethingtest' => {'X-Test' => 'Hi there!'})->status_is(200)
->header_is(Server => 'Mojolicious (Perl)')->content_is('/test4/42');
# Foo::url_for_missing
$t->get_ok('/something_missing' => {'X-Test' => 'Hi there!'})->status_is(200)
->header_is(Server => 'Mojolicious (Perl)')->content_is('does_not_exist');
# Foo::templateless
$t->get_ok('/foo/templateless' => {'X-Test' => 'Hi there!'})->status_is(200)
->header_is(Server => 'Mojolicious (Perl)')
->content_like(qr/Hello Mojo from a templateless renderer!/);
# Foo::withlayout
$t->get_ok('/foo/withlayout' => {'X-Test' => 'Hi there!'})->status_is(200)
->header_is(Server => 'Mojolicious (Perl)')
->content_like(qr/Same old in green Seems to work!/);
# Foo::withBlock
$t->get_ok('/withblock.txt' => {'X-Test' => 'Hi there!'})->status_is(200)
->header_is(Server => 'Mojolicious (Perl)')->content_type_isnt('text/html')
->content_type_is('text/plain;charset=UTF-8')
->content_like(qr/Hello Baerbel\.\s+Hello Wolfgang\./);
# MojoliciousTest2::Foo::test
$t->get_ok('/test2' => {'X-Test' => 'Hi there!'})->status_is(200)
->header_is('X-Bender' => 'Bite my shiny metal ass!')
->header_is(Server => 'Mojolicious (Perl)')->content_like(qr!/test2!);
# MojoliciousTestController::index
$t->get_ok('/test3' => {'X-Test' => 'Hi there!'})->status_is(200)
->header_is('X-Bender' => 'Bite my shiny metal ass!')
->header_is(Server => 'Mojolicious (Perl)')
->content_like(qr/No class works!/);
# MojoliciousTestController::index (only namespace)
$t->get_ok('/test5' => {'X-Test' => 'Hi there!'})->status_is(200)
->header_is('X-Bender' => 'Bite my shiny metal ass!')
->header_is(Server => 'Mojolicious (Perl)')->content_is('/test5');
# MojoliciousTestController::index (no namespace)
$t->get_ok('/test6' => {'X-Test' => 'Hi there!'})->status_is(200)
->header_is('X-Bender' => 'Bite my shiny metal ass!')
->header_is(Server => 'Mojolicious (Perl)')->content_is('/test6');
# MojoliciousTest::Foo::Bar::test (controller class shortcut)
$t->get_ok('/test7' => {'X-Test' => 'Hi there!'})->status_is(200)
->header_is(Server => 'Mojolicious (Perl)')->content_is("Class works!\n");
# MojoliciousTest::Foo::Bar::test (controller class)
$t->get_ok('/test8' => {'X-Test' => 'Hi there!'})->status_is(200)
->header_is(Server => 'Mojolicious (Perl)')->content_is("Class works!\n");
# MojoliciousTest3::Bar::index (controller class in development namespace)
$t->get_ok('/test9')->status_is(200)
->header_is(Server => 'Mojolicious (Perl)')
->content_is('Development namespace works!');
# MojoliciousTest3::Baz::index (controller class precedence)
$t->get_ok('/test10')->status_is(200)
->header_is(Server => 'Mojolicious (Perl)')
->content_is('Development namespace has high precedence!');
# 404
$t->get_ok('/' => {'X-Test' => 'Hi there!'})->status_is(404)
->header_is(Server => 'Mojolicious (Perl)')
->content_like(qr/Page not found/);
# Static file /another/file (no extension)
$t->get_ok('/another/file')->status_is(200)
->header_is(Server => 'Mojolicious (Perl)')
->content_type_is('text/plain;charset=UTF-8')
->content_like(qr/Hello Mojolicious!/);
# Static directory /another
$log = '';
$cb = $t->app->log->on(message => sub { $log .= pop });
$t->get_ok('/another')->status_is(404)
->header_is(Server => 'Mojolicious (Perl)');
like $log, qr/Controller "MojoliciousTest::Another" does not exist\./,
'right message';
$t->app->log->unsubscribe(message => $cb);
# Check Last-Modified header for static files
my $path = catdir($FindBin::Bin, 'public_dev', 'hello.txt');
my $size = (stat $path)[7];
my $mtime = Mojo::Date->new((stat $path)[9])->to_string;
# Static file /hello.txt
$t->get_ok('/hello.txt')->status_is(200)
->header_is(Server => 'Mojolicious (Perl)')
->header_is('Last-Modified' => $mtime)->header_like('ETag' => qr/^"\w+"$/)
->header_is('Content-Length' => $size)
->content_type_is('text/plain;charset=UTF-8')
->content_like(qr/Hello Mojo from a development static file!/);
# Try to access a file which is not under the web root via path
# traversal
$t->get_ok('/../../mojolicious/secret.txt')->status_is(404)
->header_is(Server => 'Mojolicious (Perl)')
->content_like(qr/Page not found/);
# Check If-Modified-Since
$t->get_ok('/hello.txt' => {'If-Modified-Since' => $mtime})->status_is(304)
->header_is(Server => 'Mojolicious (Perl)')->content_is('');
# Check If-None-Match
my $etag = $t->tx->res->headers->etag;
$t->get_ok('/hello.txt' => {'If-None-Match' => $etag})->status_is(304)
->header_is(Server => 'Mojolicious (Perl)')->content_is('');
# Check If-None-Match and If-Last-Modified
$t->get_ok(
'/hello.txt' => {'If-None-Match' => $etag, 'If-Last-Modified' => $mtime})
->status_is(304)->header_is(Server => 'Mojolicious (Perl)')->content_is('');
# Bad If-None-Match with correct If-Modified-Since
$t->get_ok(
'/hello.txt' => {'If-None-Match' => '"123"', 'If-Modified-Since' => $mtime})
->status_is(200)->header_is(Server => 'Mojolicious (Perl)')
->content_like(qr/Hello Mojo from a development static file!/);
# Bad If-Modified-Since with correct If-None-Match
$t->get_ok('/hello.txt' =>
{'If-Modified-Since' => Mojo::Date->new(23), 'If-None-Match' => $etag})
->status_is(200)->header_is(Server => 'Mojolicious (Perl)')
->content_like(qr/Hello Mojo from a development static file!/);
# Embedded development static file
$t->get_ok('/some/static/file.txt')->status_is(200)
->header_is(Server => 'Mojolicious (Perl)')
->content_is("Development static file with high precedence.\n");
# Embedded development template
$t->get_ok('/just/some/template')->status_is(200)
->header_is(Server => 'Mojolicious (Perl)')
->content_is("Development template with high precedence.\n");
# Check develpment mode log level
my $app = Mojolicious->new;
is $app->log->level, 'debug', 'right log level';
# Make sure we can override attributes with constructor arguments
$app = MojoliciousTest->new({mode => 'test'});
is $app->mode, 'test', 'right mode';
# Persistent error
$app = MojoliciousTest->new;
my $tx = $t->ua->build_tx(GET => '/foo');
$app->handler($tx);
is $tx->res->code, 200, 'right status';
like $tx->res->body, qr|Hello Mojo from the template /foo! Hello World!|,
'right content';
$tx = $t->ua->build_tx(GET => '/foo/willdie');
$app->handler($tx);
is $tx->res->code, 500, 'right status';
like $tx->res->body, qr/Foo\.pm/, 'right content';
$tx = $t->ua->build_tx(GET => '/foo');
$app->handler($tx);
is $tx->res->code, 200, 'right status';
like $tx->res->body, qr|Hello Mojo from the template /foo! Hello World!|,
'right content';
$t = Test::Mojo->new('SingleFileTestApp');
# SingleFileTestApp::Foo::index
$t->get_ok('/foo')->status_is(200)->header_is(Server => 'Mojolicious (Perl)')
->content_like(qr/Same old in green Seems to work!/);
# SingleFileTestApp (helper)
$t->get_ok('/helper')->status_is(200)
->header_is(Server => 'Mojolicious (Perl)')->content_is('Welcome aboard!');
# PluginWithEmbeddedApp (lite app in plugin)
$t->get_ok('/plugin/foo')->status_is(200)
->header_is(Server => 'Mojolicious (Perl)')->content_is('plugin works!');
# SingleFileTestApp::Foo::conf (config file)
$t->get_ok('/foo/conf')->status_is(200)
->header_is(Server => 'Mojolicious (Perl)')->content_is('works!');
# SingleFileTestApp::Foo::data_template
$t->get_ok('/foo/data_template')->status_is(200)
->header_is(Server => 'Mojolicious (Perl)')->content_is("23 works!\n");
# SingleFileTestApp::Foo::data_template
$t->get_ok('/foo/data_template2')->status_is(200)
->header_is(Server => 'Mojolicious (Perl)')
->content_is("This one works too!\n");
# SingleFileTestApp::Foo::data_static
$t->get_ok('/foo/data_static')->status_is(200)
->header_is(Server => 'Mojolicious (Perl)')
->content_is("And this one... ALL GLORY TO THE HYPNOTOAD!\n");
# SingleFileTestApp::Foo::routes
$t->get_ok('/foo/routes')->status_is(200)
->header_is('X-Bender' => 'Bite my shiny metal ass!')
->header_is(Server => 'Mojolicious (Perl)')->content_is('/foo/routes');
# SingleFileTestApp::Redispatch::handler
$log = '';
$cb = $t->app->log->on(message => sub { $log .= pop });
$t->get_ok('/redispatch')->status_is(200)
->header_is(Server => 'Mojolicious (Perl)')->content_is('Redispatch!');
like $log, qr/Routing to application "SingleFileTestApp::Redispatch"\./,
'right message';
$t->app->log->unsubscribe(message => $cb);
# SingleFileTestApp::Redispatch::render
$t->get_ok('/redispatch/render')->status_is(200)
->header_is(Server => 'Mojolicious (Perl)')->content_is('Render!');
# SingleFileTestApp::Redispatch::handler (targeting an existing method)
$t->get_ok('/redispatch/secret')->status_is(200)
->header_is(Server => 'Mojolicious (Perl)')->content_is('Redispatch!');
# SingleFileTestApp::Redispatch::secret
$t->get_ok('/redispatch/secret?rly=1')->status_is(200)
->header_is(Server => 'Mojolicious (Perl)')->content_is('Secret!');
$t = Test::Mojo->new('MojoliciousTest');
# MojoliciousTestController::Foo::plugin_upper_case
$t->get_ok('/plugin/upper_case')->status_is(200)
->header_is(Server => 'Mojolicious (Perl)')->content_is('WELCOME aboard!');
# MojoliciousTestController::Foo::plugin_camel_case
$t->get_ok('/plugin/camel_case')->status_is(200)
->header_is(Server => 'Mojolicious (Perl)')->content_is('Welcome aboard!');
# MojoliciousTestController::Foo::stage2
$t->get_ok('/staged' => {'X-Pass' => 1})->status_is(200)
->header_is(Server => 'Mojolicious (Perl)')->content_is('Welcome aboard!');
# MojoliciousTestController::Foo::stage1
$t->get_ok('/staged')->status_is(200)
->header_is(Server => 'Mojolicious (Perl)')->content_is('Go away!');
# MojoliciousTestController::Foo::suspended
$log = '';
$cb = $t->app->log->on(message => sub { $log .= pop });
$t->get_ok('/suspended')->status_is(200)
->header_is(Server => 'Mojolicious (Perl)')
->header_is('X-Suspended' => '0, 1, 1, 2')->content_is('Have fun!');
like $log, qr!GET "/suspended"\.!, 'right message';
like $log,
qr/Routing to controller "MojoliciousTest::Foo" and action "suspended"\./,
'right message';
like $log, qr/Routing to controller "MojoliciousTest::Foo" and action "fun"\./,
'right message';
like $log, qr!Rendering template "foo/fun.html.ep" from DATA section\.!,
'right message';
like $log, qr/200 OK/, 'right message';
$t->app->log->unsubscribe(message => $cb);
# MojoliciousTest::Foo::longpoll
my $stash;
$t->app->plugins->once(before_dispatch => sub { $stash = shift->stash });
$t->get_ok('/longpoll')->status_is(200)
->header_is(Server => 'Mojolicious (Perl)')->content_is('Poll!');
ok $stash->{finished}, 'finish event has been emitted';
ok $stash->{destroyed}, 'controller has been destroyed';
# MojoliciousTest::Foo::config
$t->get_ok('/stash_config')->status_is(200)
->header_is(Server => 'Mojolicious (Perl)')->content_is('123');
# Shortcuts to controller#action
$t->get_ok('/shortcut/ctrl-act')->status_is(200)
->header_is(Server => 'Mojolicious (Perl)')->content_is('ctrl-act');
$t->get_ok('/shortcut/ctrl')->status_is(200)
->header_is(Server => 'Mojolicious (Perl)')->content_is('ctrl');
$t->get_ok('/shortcut/act')->status_is(200)
->header_is(Server => 'Mojolicious (Perl)')->content_is('act');
# Session with domain
$t->get_ok('/foo/session')->status_is(200)
->header_like('Set-Cookie' => qr/; domain=\.example\.com/)
->header_like('Set-Cookie' => qr!; path=/bar!)
->content_is('Bender rockzzz!');
# Mixed formats
$t->get_ok('/rss.xml')->status_is(200)->content_type_is('application/rss+xml')
->content_like(qr!<\?xml version="1.0" encoding="UTF-8"\?><rss />!);
done_testing();