# Rack
class Hello
def call(env)
return [
200,
{ “Content-Type” => ”text/plain” },
[“Hello World”]
]
end
end
# PSGI
my $app = sub {
my $env = shift;
return [
200,
[ ‘Content-Type’, ‘text/plain’ ],
[ ‘Hello World’ ],
];
};
PSGI application
code reference
$app = sub {...};
my $app = sub {
my $env = shift;
return [ $status, $header, $body ];
};
environment hash
$env: CGI-like env variables
+ psgi.input, psgi.errors etc.
my $app = sub {
my $env = shift;
return [ $status, $header, $body ];
};
Response
array ref with three elements
status code, headers (array ref)
and body (IO-like or array ref)
my $app = sub {
my $env = shift;
return [ $status, $header, $body ];
};
$body
IO::Handle-like
getline() and close()
IO::Handle::Util
Easily turns perl code ref into a IO::Handle
Streaming interface
my $app = sub {
my $env = shift;
return sub {
my $respond = shift;
# You could do some event loop
# to delay response (e.g. Comet)
$respond->([ $status, $header, $body ]);
};
};
my $app = sub {
my $env = shift;
return sub {
my $respond = shift;
my $w = $respond->([ $status, $header ]);
$w->write($body);
$w->write($body);
...
$w->close;
};
};
Streaming Interface
Originally designed for non-blocking servers
Now available for most servers incl. CGI, Apache
HTTP::Server::PSGI
Reference PSGI web server
bundled in Plack
Very fast
3000 QPS on standalone
15000 QPS with prefork :)
Plack::Handler
Connects PSGI apps to Web servers
CGI, FastCGI, Apache, Standalone
Utilities
Plackup
Run PSGI app instantly from CLI
(inspired by rackup)
Plack::Runner
plackup backend
Use this to make CLI for your web app
Middleware
my $app = sub {
my $env = shift;
return [ $status, $header, $body ];
};
my $mw = sub {
my $env = shift;
# do something with $env
my $res = $app->($env);
# do something with $res;
return $res;
};
Plack::App::URLMap
Multiplex multiple apps
Integrated with Builder DSL
use CatApp;
use CGIApp;
my $c1 = sub { CatApp->run };
my $c2 = sub { CGIApp->run_psgi };
use Plack::Builder;
builder {
mount “/cat” => $c1;
mount “/cgi-app” => builder {
enable “StackTrace”;
$c2;
};
}
CGI::PSGI
Easy migration from CGI.pm
CGI::Emulate::PSGI
CGI::Compile
Easiest migration from CGI scripts (like Registry)
Plack::Request
like libapreq (Apache::Request)
wrapper APIs for middleware developers
Plack::Test
Unified interface to write tests
with Mock HTTP and Live HTTP
use Plack::Test;
use HTTP::Request::Common;
my $app = sub {
my $env = shift;
return [ $status, $header, $body ];
};
test_psgi app => $app, client => sub {
my $cb = shift;
my $req = GET “http://localhost/foo”;
my $res = $cb->($req);
# test $res;
};
use Plack::Test;
use HTTP::Request::Common;
$Plack::Test::Impl = “Server”;
my $app = sub {
my $env = shift;
return [ $status, $header, $body ];
};
test_psgi app => $app, client => sub {
my $cb = shift;
my $req = GET “http://localhost/foo”;
my $res = $cb->($req);
# test $res;
};
Merge Plack::Request
Becomes a library for middleware writers
Make it work better when created multiple times
Summary
• PSGI is an interface, Plack is the code.
• We have many (pretty fast) PSGI servers.
• We have adapters and tools for most web
frameworks.
• Use it!