mirror of
https://github.com/KevinMidboe/linguist.git
synced 2025-10-29 09:40:21 +00:00
696 lines
19 KiB
Perl
696 lines
19 KiB
Perl
package Plack::Request;
|
|
use strict;
|
|
use warnings;
|
|
use 5.008_001;
|
|
our $VERSION = '0.9988';
|
|
$VERSION = eval $VERSION;
|
|
|
|
use HTTP::Headers;
|
|
use Carp ();
|
|
use Hash::MultiValue;
|
|
use HTTP::Body;
|
|
|
|
use Plack::Request::Upload;
|
|
use Plack::TempBuffer;
|
|
use URI;
|
|
use URI::Escape ();
|
|
|
|
sub _deprecated {
|
|
my $alt = shift;
|
|
my $method = (caller(1))[3];
|
|
Carp::carp("$method is deprecated. Use '$alt' instead.");
|
|
}
|
|
|
|
sub new {
|
|
my($class, $env) = @_;
|
|
Carp::croak(q{$env is required})
|
|
unless defined $env && ref($env) eq 'HASH';
|
|
|
|
bless { env => $env }, $class;
|
|
}
|
|
|
|
sub env { $_[0]->{env} }
|
|
|
|
sub address { $_[0]->env->{REMOTE_ADDR} }
|
|
sub remote_host { $_[0]->env->{REMOTE_HOST} }
|
|
sub protocol { $_[0]->env->{SERVER_PROTOCOL} }
|
|
sub method { $_[0]->env->{REQUEST_METHOD} }
|
|
sub port { $_[0]->env->{SERVER_PORT} }
|
|
sub user { $_[0]->env->{REMOTE_USER} }
|
|
sub request_uri { $_[0]->env->{REQUEST_URI} }
|
|
sub path_info { $_[0]->env->{PATH_INFO} }
|
|
sub path { $_[0]->env->{PATH_INFO} || '/' }
|
|
sub script_name { $_[0]->env->{SCRIPT_NAME} }
|
|
sub scheme { $_[0]->env->{'psgi.url_scheme'} }
|
|
sub secure { $_[0]->scheme eq 'https' }
|
|
sub body { $_[0]->env->{'psgi.input'} }
|
|
sub input { $_[0]->env->{'psgi.input'} }
|
|
|
|
sub content_length { $_[0]->env->{CONTENT_LENGTH} }
|
|
sub content_type { $_[0]->env->{CONTENT_TYPE} }
|
|
|
|
sub session { $_[0]->env->{'psgix.session'} }
|
|
sub session_options { $_[0]->env->{'psgix.session.options'} }
|
|
sub logger { $_[0]->env->{'psgix.logger'} }
|
|
|
|
sub cookies {
|
|
my $self = shift;
|
|
|
|
return {} unless $self->env->{HTTP_COOKIE};
|
|
|
|
# HTTP_COOKIE hasn't changed: reuse the parsed cookie
|
|
if ( $self->env->{'plack.cookie.parsed'}
|
|
&& $self->env->{'plack.cookie.string'} eq $self->env->{HTTP_COOKIE}) {
|
|
return $self->env->{'plack.cookie.parsed'};
|
|
}
|
|
|
|
$self->env->{'plack.cookie.string'} = $self->env->{HTTP_COOKIE};
|
|
|
|
my %results;
|
|
my @pairs = grep /=/, split "[;,] ?", $self->env->{'plack.cookie.string'};
|
|
for my $pair ( @pairs ) {
|
|
# trim leading trailing whitespace
|
|
$pair =~ s/^\s+//; $pair =~ s/\s+$//;
|
|
|
|
my ($key, $value) = map URI::Escape::uri_unescape($_), split( "=", $pair, 2 );
|
|
|
|
# Take the first one like CGI.pm or rack do
|
|
$results{$key} = $value unless exists $results{$key};
|
|
}
|
|
|
|
$self->env->{'plack.cookie.parsed'} = \%results;
|
|
}
|
|
|
|
sub query_parameters {
|
|
my $self = shift;
|
|
$self->env->{'plack.request.query'} ||= Hash::MultiValue->new($self->uri->query_form);
|
|
}
|
|
|
|
sub content {
|
|
my $self = shift;
|
|
|
|
unless ($self->env->{'psgix.input.buffered'}) {
|
|
$self->_parse_request_body;
|
|
}
|
|
|
|
my $fh = $self->input or return '';
|
|
my $cl = $self->env->{CONTENT_LENGTH} or return'';
|
|
$fh->read(my($content), $cl, 0);
|
|
$fh->seek(0, 0);
|
|
|
|
return $content;
|
|
}
|
|
|
|
sub raw_body { $_[0]->content }
|
|
|
|
# XXX you can mutate headers with ->headers but it's not written through to the env
|
|
|
|
sub headers {
|
|
my $self = shift;
|
|
if (!defined $self->{headers}) {
|
|
my $env = $self->env;
|
|
$self->{headers} = HTTP::Headers->new(
|
|
map {
|
|
(my $field = $_) =~ s/^HTTPS?_//;
|
|
( $field => $env->{$_} );
|
|
}
|
|
grep { /^(?:HTTP|CONTENT|COOKIE)/i } keys %$env
|
|
);
|
|
}
|
|
$self->{headers};
|
|
}
|
|
|
|
sub content_encoding { shift->headers->content_encoding(@_) }
|
|
sub header { shift->headers->header(@_) }
|
|
sub referer { shift->headers->referer(@_) }
|
|
sub user_agent { shift->headers->user_agent(@_) }
|
|
|
|
sub body_parameters {
|
|
my $self = shift;
|
|
|
|
unless ($self->env->{'plack.request.body'}) {
|
|
$self->_parse_request_body;
|
|
}
|
|
|
|
return $self->env->{'plack.request.body'};
|
|
}
|
|
|
|
# contains body + query
|
|
sub parameters {
|
|
my $self = shift;
|
|
|
|
$self->env->{'plack.request.merged'} ||= do {
|
|
my $query = $self->query_parameters;
|
|
my $body = $self->body_parameters;
|
|
Hash::MultiValue->new($query->flatten, $body->flatten);
|
|
};
|
|
}
|
|
|
|
sub uploads {
|
|
my $self = shift;
|
|
|
|
if ($self->env->{'plack.request.upload'}) {
|
|
return $self->env->{'plack.request.upload'};
|
|
}
|
|
|
|
$self->_parse_request_body;
|
|
return $self->env->{'plack.request.upload'};
|
|
}
|
|
|
|
sub hostname { _deprecated 'remote_host'; $_[0]->remote_host || $_[0]->address }
|
|
sub url_scheme { _deprecated 'scheme'; $_[0]->scheme }
|
|
sub params { _deprecated 'parameters'; shift->parameters(@_) }
|
|
sub query_params { _deprecated 'query_parameters'; shift->query_parameters(@_) }
|
|
sub body_params { _deprecated 'body_parameters'; shift->body_parameters(@_) }
|
|
|
|
sub cookie {
|
|
my $self = shift;
|
|
_deprecated 'cookies';
|
|
|
|
return keys %{ $self->cookies } if @_ == 0;
|
|
|
|
my $name = shift;
|
|
return $self->cookies->{$name};
|
|
}
|
|
|
|
sub param {
|
|
my $self = shift;
|
|
|
|
return keys %{ $self->parameters } if @_ == 0;
|
|
|
|
my $key = shift;
|
|
return $self->parameters->{$key} unless wantarray;
|
|
return $self->parameters->get_all($key);
|
|
}
|
|
|
|
sub upload {
|
|
my $self = shift;
|
|
|
|
return keys %{ $self->uploads } if @_ == 0;
|
|
|
|
my $key = shift;
|
|
return $self->uploads->{$key} unless wantarray;
|
|
return $self->uploads->get_all($key);
|
|
}
|
|
|
|
sub raw_uri {
|
|
my $self = shift;
|
|
_deprecated 'base';
|
|
|
|
my $base = $self->base;
|
|
$base->path_query($self->env->{REQUEST_URI});
|
|
|
|
$base;
|
|
}
|
|
|
|
sub uri {
|
|
my $self = shift;
|
|
|
|
my $base = $self->_uri_base;
|
|
|
|
# We have to escape back PATH_INFO in case they include stuff like
|
|
# ? or # so that the URI parser won't be tricked. However we should
|
|
# preserve '/' since encoding them into %2f doesn't make sense.
|
|
# This means when a request like /foo%2fbar comes in, we recognize
|
|
# it as /foo/bar which is not ideal, but that's how the PSGI PATH_INFO
|
|
# spec goes and we can't do anything about it. See PSGI::FAQ for details.
|
|
# http://github.com/miyagawa/Plack/issues#issue/118
|
|
my $path_escape_class = '^A-Za-z0-9\-\._~/';
|
|
|
|
my $path = URI::Escape::uri_escape($self->env->{PATH_INFO} || '', $path_escape_class);
|
|
$path .= '?' . $self->env->{QUERY_STRING}
|
|
if defined $self->env->{QUERY_STRING} && $self->env->{QUERY_STRING} ne '';
|
|
|
|
$base =~ s!/$!! if $path =~ m!^/!;
|
|
|
|
return URI->new($base . $path)->canonical;
|
|
}
|
|
|
|
sub base {
|
|
my $self = shift;
|
|
URI->new($self->_uri_base)->canonical;
|
|
}
|
|
|
|
sub _uri_base {
|
|
my $self = shift;
|
|
|
|
my $env = $self->env;
|
|
|
|
my $uri = ($env->{'psgi.url_scheme'} || "http") .
|
|
"://" .
|
|
($env->{HTTP_HOST} || (($env->{SERVER_NAME} || "") . ":" . ($env->{SERVER_PORT} || 80))) .
|
|
($env->{SCRIPT_NAME} || '/');
|
|
|
|
return $uri;
|
|
}
|
|
|
|
sub new_response {
|
|
my $self = shift;
|
|
require Plack::Response;
|
|
Plack::Response->new(@_);
|
|
}
|
|
|
|
sub _parse_request_body {
|
|
my $self = shift;
|
|
|
|
my $ct = $self->env->{CONTENT_TYPE};
|
|
my $cl = $self->env->{CONTENT_LENGTH};
|
|
if (!$ct && !$cl) {
|
|
# No Content-Type nor Content-Length -> GET/HEAD
|
|
$self->env->{'plack.request.body'} = Hash::MultiValue->new;
|
|
$self->env->{'plack.request.upload'} = Hash::MultiValue->new;
|
|
return;
|
|
}
|
|
|
|
my $body = HTTP::Body->new($ct, $cl);
|
|
|
|
# HTTP::Body will create temporary files in case there was an
|
|
# upload. Those temporary files can be cleaned up by telling
|
|
# HTTP::Body to do so. It will run the cleanup when the request
|
|
# env is destroyed. That the object will not go out of scope by
|
|
# the end of this sub we will store a reference here.
|
|
$self->env->{'plack.request.http.body'} = $body;
|
|
$body->cleanup(1);
|
|
|
|
my $input = $self->input;
|
|
|
|
my $buffer;
|
|
if ($self->env->{'psgix.input.buffered'}) {
|
|
# Just in case if input is read by middleware/apps beforehand
|
|
$input->seek(0, 0);
|
|
} else {
|
|
$buffer = Plack::TempBuffer->new($cl);
|
|
}
|
|
|
|
my $spin = 0;
|
|
while ($cl) {
|
|
$input->read(my $chunk, $cl < 8192 ? $cl : 8192);
|
|
my $read = length $chunk;
|
|
$cl -= $read;
|
|
$body->add($chunk);
|
|
$buffer->print($chunk) if $buffer;
|
|
|
|
if ($read == 0 && $spin++ > 2000) {
|
|
Carp::croak "Bad Content-Length: maybe client disconnect? ($cl bytes remaining)";
|
|
}
|
|
}
|
|
|
|
if ($buffer) {
|
|
$self->env->{'psgix.input.buffered'} = 1;
|
|
$self->env->{'psgi.input'} = $buffer->rewind;
|
|
} else {
|
|
$input->seek(0, 0);
|
|
}
|
|
|
|
$self->env->{'plack.request.body'} = Hash::MultiValue->from_mixed($body->param);
|
|
|
|
my @uploads = Hash::MultiValue->from_mixed($body->upload)->flatten;
|
|
my @obj;
|
|
while (my($k, $v) = splice @uploads, 0, 2) {
|
|
push @obj, $k, $self->_make_upload($v);
|
|
}
|
|
|
|
$self->env->{'plack.request.upload'} = Hash::MultiValue->new(@obj);
|
|
|
|
1;
|
|
}
|
|
|
|
sub _make_upload {
|
|
my($self, $upload) = @_;
|
|
my %copy = %$upload;
|
|
$copy{headers} = HTTP::Headers->new(%{$upload->{headers}});
|
|
Plack::Request::Upload->new(%copy);
|
|
}
|
|
|
|
1;
|
|
__END__
|
|
|
|
=head1 NAME
|
|
|
|
Plack::Request - Portable HTTP request object from PSGI env hash
|
|
|
|
=head1 SYNOPSIS
|
|
|
|
use Plack::Request;
|
|
|
|
my $app_or_middleware = sub {
|
|
my $env = shift; # PSGI env
|
|
|
|
my $req = Plack::Request->new($env);
|
|
|
|
my $path_info = $req->path_info;
|
|
my $query = $req->param('query');
|
|
|
|
my $res = $req->new_response(200); # new Plack::Response
|
|
$res->finalize;
|
|
};
|
|
|
|
=head1 DESCRIPTION
|
|
|
|
L<Plack::Request> provides a consistent API for request objects across
|
|
web server environments.
|
|
|
|
=head1 CAVEAT
|
|
|
|
Note that this module is intended to be used by Plack middleware
|
|
developers and web application framework developers rather than
|
|
application developers (end users).
|
|
|
|
Writing your web application directly using Plack::Request is
|
|
certainly possible but not recommended: it's like doing so with
|
|
mod_perl's Apache::Request: yet too low level.
|
|
|
|
If you're writing a web application, not a framework, then you're
|
|
encouraged to use one of the web application frameworks that support PSGI (L<http://plackperl.org/#frameworks>),
|
|
or see modules like L<HTTP::Engine> to provide higher level
|
|
Request and Response API on top of PSGI.
|
|
|
|
=head1 METHODS
|
|
|
|
Some of the methods defined in the earlier versions are deprecated in
|
|
version 0.99. Take a look at L</"INCOMPATIBILITIES">.
|
|
|
|
Unless otherwise noted, all methods and attributes are B<read-only>,
|
|
and passing values to the method like an accessor doesn't work like
|
|
you expect it to.
|
|
|
|
=head2 new
|
|
|
|
Plack::Request->new( $env );
|
|
|
|
Creates a new request object.
|
|
|
|
=head1 ATTRIBUTES
|
|
|
|
=over 4
|
|
|
|
=item env
|
|
|
|
Returns the shared PSGI environment hash reference. This is a
|
|
reference, so writing to this environment passes through during the
|
|
whole PSGI request/response cycle.
|
|
|
|
=item address
|
|
|
|
Returns the IP address of the client (C<REMOTE_ADDR>).
|
|
|
|
=item remote_host
|
|
|
|
Returns the remote host (C<REMOTE_HOST>) of the client. It may be
|
|
empty, in which case you have to get the IP address using C<address>
|
|
method and resolve on your own.
|
|
|
|
=item method
|
|
|
|
Contains the request method (C<GET>, C<POST>, C<HEAD>, etc).
|
|
|
|
=item protocol
|
|
|
|
Returns the protocol (HTTP/1.0 or HTTP/1.1) used for the current request.
|
|
|
|
=item request_uri
|
|
|
|
Returns the raw, undecoded request URI path. You probably do B<NOT>
|
|
want to use this to dispatch requests.
|
|
|
|
=item path_info
|
|
|
|
Returns B<PATH_INFO> in the environment. Use this to get the local
|
|
path for the requests.
|
|
|
|
=item path
|
|
|
|
Similar to C<path_info> but returns C</> in case it is empty. In other
|
|
words, it returns the virtual path of the request URI after C<<
|
|
$req->base >>. See L</"DISPATCHING"> for details.
|
|
|
|
=item script_name
|
|
|
|
Returns B<SCRIPT_NAME> in the environment. This is the absolute path
|
|
where your application is hosted.
|
|
|
|
=item scheme
|
|
|
|
Returns the scheme (C<http> or C<https>) of the request.
|
|
|
|
=item secure
|
|
|
|
Returns true or false, indicating whether the connection is secure (https).
|
|
|
|
=item body, input
|
|
|
|
Returns C<psgi.input> handle.
|
|
|
|
=item session
|
|
|
|
Returns (optional) C<psgix.session> hash. When it exists, you can
|
|
retrieve and store per-session data from and to this hash.
|
|
|
|
=item session_options
|
|
|
|
Returns (optional) C<psgix.session.options> hash.
|
|
|
|
=item logger
|
|
|
|
Returns (optional) C<psgix.logger> code reference. When it exists,
|
|
your application is supposed to send the log message to this logger,
|
|
using:
|
|
|
|
$req->logger->({ level => 'debug', message => "This is a debug message" });
|
|
|
|
=item cookies
|
|
|
|
Returns a reference to a hash containing the cookies. Values are
|
|
strings that are sent by clients and are URI decoded.
|
|
|
|
=item query_parameters
|
|
|
|
Returns a reference to a hash containing query string (GET)
|
|
parameters. This hash reference is L<Hash::MultiValue> object.
|
|
|
|
=item body_parameters
|
|
|
|
Returns a reference to a hash containing posted parameters in the
|
|
request body (POST). As with C<query_parameters>, the hash
|
|
reference is a L<Hash::MultiValue> object.
|
|
|
|
=item parameters
|
|
|
|
Returns a L<Hash::MultiValue> hash reference containing (merged) GET
|
|
and POST parameters.
|
|
|
|
=item content, raw_body
|
|
|
|
Returns the request content in an undecoded byte string for POST requests.
|
|
|
|
=item uri
|
|
|
|
Returns an URI object for the current request. The URI is constructed
|
|
using various environment values such as C<SCRIPT_NAME>, C<PATH_INFO>,
|
|
C<QUERY_STRING>, C<HTTP_HOST>, C<SERVER_NAME> and C<SERVER_PORT>.
|
|
|
|
Every time this method is called it returns a new, cloned URI object.
|
|
|
|
=item base
|
|
|
|
Returns an URI object for the base path of current request. This is
|
|
like C<uri> but only contains up to C<SCRIPT_NAME> where your
|
|
application is hosted at.
|
|
|
|
Every time this method is called it returns a new, cloned URI object.
|
|
|
|
=item user
|
|
|
|
Returns C<REMOTE_USER> if it's set.
|
|
|
|
=item headers
|
|
|
|
Returns an L<HTTP::Headers> object containing the headers for the current request.
|
|
|
|
=item uploads
|
|
|
|
Returns a reference to a hash containing uploads. The hash reference
|
|
is a L<Hash::MultiValue> object and values are L<Plack::Request::Upload>
|
|
objects.
|
|
|
|
=item content_encoding
|
|
|
|
Shortcut to $req->headers->content_encoding.
|
|
|
|
=item content_length
|
|
|
|
Shortcut to $req->headers->content_length.
|
|
|
|
=item content_type
|
|
|
|
Shortcut to $req->headers->content_type.
|
|
|
|
=item header
|
|
|
|
Shortcut to $req->headers->header.
|
|
|
|
=item referer
|
|
|
|
Shortcut to $req->headers->referer.
|
|
|
|
=item user_agent
|
|
|
|
Shortcut to $req->headers->user_agent.
|
|
|
|
=item param
|
|
|
|
Returns GET and POST parameters with a CGI.pm-compatible param
|
|
method. This is an alternative method for accessing parameters in
|
|
$req->parameters. Unlike CGI.pm, it does I<not> allow
|
|
setting or modifying query parameters.
|
|
|
|
$value = $req->param( 'foo' );
|
|
@values = $req->param( 'foo' );
|
|
@params = $req->param;
|
|
|
|
=item upload
|
|
|
|
A convenient method to access $req->uploads.
|
|
|
|
$upload = $req->upload('field');
|
|
@uploads = $req->upload('field');
|
|
@fields = $req->upload;
|
|
|
|
for my $upload ( $req->upload('field') ) {
|
|
print $upload->filename;
|
|
}
|
|
|
|
=item new_response
|
|
|
|
my $res = $req->new_response;
|
|
|
|
Creates a new L<Plack::Response> object. Handy to remove dependency on
|
|
L<Plack::Response> in your code for easy subclassing and duck typing
|
|
in web application frameworks, as well as overriding Response
|
|
generation in middlewares.
|
|
|
|
=back
|
|
|
|
=head2 Hash::MultiValue parameters
|
|
|
|
Parameters that can take one or multiple values (i.e. C<parameters>,
|
|
C<query_parameters>, C<body_parameters> and C<uploads>) store the
|
|
hash reference as a L<Hash::MultiValue> object. This means you can use
|
|
the hash reference as a plain hash where values are B<always> scalars
|
|
(B<NOT> array references), so you don't need to code ugly and unsafe
|
|
C<< ref ... eq 'ARRAY' >> anymore.
|
|
|
|
And if you explicitly want to get multiple values of the same key, you
|
|
can call the C<get_all> method on it, such as:
|
|
|
|
my @foo = $req->query_parameters->get_all('foo');
|
|
|
|
You can also call C<get_one> to always get one parameter independent
|
|
of the context (unlike C<param>), and even call C<mixed> (with
|
|
Hash::MultiValue 0.05 or later) to get the I<traditional> hash
|
|
reference,
|
|
|
|
my $params = $req->parameters->mixed;
|
|
|
|
where values are either a scalar or an array reference depending on
|
|
input, so it might be useful if you already have the code to deal with
|
|
that ugliness.
|
|
|
|
=head2 PARSING POST BODY and MULTIPLE OBJECTS
|
|
|
|
The methods to parse request body (C<content>, C<body_parameters> and
|
|
C<uploads>) are carefully coded to save the parsed body in the
|
|
environment hash as well as in the temporary buffer, so you can call
|
|
them multiple times and create Plack::Request objects multiple times
|
|
in a request and they should work safely, and won't parse request body
|
|
more than twice for the efficiency.
|
|
|
|
=head1 DISPATCHING
|
|
|
|
If your application or framework wants to dispatch (or route) actions
|
|
based on request paths, be sure to use C<< $req->path_info >> not C<<
|
|
$req->uri->path >>.
|
|
|
|
This is because C<path_info> gives you the virtual path of the request,
|
|
regardless of how your application is mounted. If your application is
|
|
hosted with mod_perl or CGI scripts, or even multiplexed with tools
|
|
like L<Plack::App::URLMap>, request's C<path_info> always gives you
|
|
the action path.
|
|
|
|
Note that C<path_info> might give you an empty string, in which case
|
|
you should assume that the path is C</>.
|
|
|
|
You will also want to use C<< $req->base >> as a base prefix when
|
|
building URLs in your templates or in redirections. It's a good idea
|
|
for you to subclass Plack::Request and define methods such as:
|
|
|
|
sub uri_for {
|
|
my($self, $path, $args) = @_;
|
|
my $uri = $self->base;
|
|
$uri->path($uri->path . $path);
|
|
$uri->query_form(@$args) if $args;
|
|
$uri;
|
|
}
|
|
|
|
So you can say:
|
|
|
|
my $link = $req->uri_for('/logout', [ signoff => 1 ]);
|
|
|
|
and if C<< $req->base >> is C</app> you'll get the full URI for
|
|
C</app/logout?signoff=1>.
|
|
|
|
=head1 INCOMPATIBILITIES
|
|
|
|
In version 0.99, many utility methods are removed or deprecated, and
|
|
most methods are made read-only.
|
|
|
|
The following methods are deprecated: C<hostname>, C<url_scheme>,
|
|
C<params>, C<query_params>, C<body_params>, C<cookie> and
|
|
C<raw_uri>. They will be removed in the next major release.
|
|
|
|
All parameter-related methods such as C<parameters>,
|
|
C<body_parameters>, C<query_parameters> and C<uploads> now contains
|
|
L<Hash::MultiValue> objects, rather than I<scalar or an array
|
|
reference depending on the user input> which is insecure. See
|
|
L<Hash::MultiValue> for more about this change.
|
|
|
|
C<< $req->path >> method had a bug, where the code and the document
|
|
was mismatching. The document was suggesting it returns the sub
|
|
request path after C<< $req->base >> but the code was always returning
|
|
the absolute URI path. The code is now updated to be an alias of C<<
|
|
$req->path_info >> but returns C</> in case it's empty. If you need
|
|
the older behavior, just call C<< $req->uri->path >> instead.
|
|
|
|
Cookie handling is simplified, and doesn't use L<CGI::Simple::Cookie>
|
|
anymore, which means you B<CAN NOT> set array reference or hash
|
|
reference as a cookie value and expect it be serialized. You're always
|
|
required to set string value, and encoding or decoding them is totally
|
|
up to your application or framework. Also, C<cookies> hash reference
|
|
now returns I<strings> for the cookies rather than CGI::Simple::Cookie
|
|
objects, which means you no longer have to write a wacky code such as:
|
|
|
|
$v = $req->cookie->{foo} ? $req->cookie->{foo}->value : undef;
|
|
|
|
and instead, simply do:
|
|
|
|
$v = $req->cookie->{foo};
|
|
|
|
=head1 AUTHORS
|
|
|
|
Tatsuhiko Miyagawa
|
|
|
|
Kazuhiro Osawa
|
|
|
|
Tokuhiro Matsuno
|
|
|
|
=head1 SEE ALSO
|
|
|
|
L<Plack::Response> L<HTTP::Request>, L<Catalyst::Request>
|
|
|
|
=head1 LICENSE
|
|
|
|
This library is free software; you can redistribute it and/or modify
|
|
it under the same terms as Perl itself.
|
|
|
|
=cut
|