I have some code that looks like this:
use SomeApp;
use Test::WWW::Mechanize::PSGI;
my $mech = Test::WWW::Mechanize::PSGI->new(
app => sub { SomeApp->run(@_) },
);
$mech->get_ok('/');
However, as soon as get_ok()
is called, I get the following warning:
PSGI error: failed to listen to port 8080: Address already in use at .../5.18.1/HTTP/Server/PSGI.pm line 94.
HTTP::Server::PSGI::setup_listener('HTTP::Server::PSGI=HASH(0x7fe6622fad60)') called at .../5.18.1/HTTP/Server/PSGI.pm line 54
And yes, I'm using that port for something else. From the docs of Test::WWW::Mechanize::PSGI:
This module allows you to test PSGI web applications but does not require a server or issue HTTP requests. Instead, it passes the HTTP request object directly to PSGI.
So in theory, I shouldn't need to specify a port, but I get the above warning and pages fetched return a 500 (they work fine in the browser). What am I missing?
Changing MyApp->run
to MyApp->psgi_app
results in:
Can't call method "request" on an undefined value at .../5.18.1/Test/WWW/Mechanize/PSGI.pm line 47.
This error can be replicated with:
catalyst.pl MyApp
cd MyApp
# run the test program above
Catalyst's run
method would actually run the HTTP server (via Plack/PSGI!) for a development, which is not what you want with testing via PSGI (without running a server). You need: app => MyApp->psgi_app
, without an extra sub
block, since psgi_app
supposedly returns the PSGI app itself.
The error message "Can't call method 'request' on ..." is a common error when your app returns something that is not correct per PSGI spec. The message has been improved a bit on the git master, but it's essentially a user error since you're basically returning sub { $app }
when it expects just $app
.
More documentation on the PSGI support with Catalyst is available with perldoc Catalyst::PSGI
.
Matt Trout mentioned LWP::Protocol::PSGI
as a workaround. It hijacks HTTP to make this work:
use Test::WWW::Mechanize;
use LWP::Protocol::PSGI;
use MyApp;
LWP::Protocol::PSGI->register( MyApp->psgi_app(@_) );
my $mech = Test::WWW::Mechanize->new;
# first GET must be absolute
$mech->get('http://localhost/login');
say $mech->content;
# then we can switch to relative
$mech->get('/login');
say $mech->content;
In short, the above is more or less cargo-culted (as I don't understand why the first version failed), but it's enough for me to move forward.
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With