X-Git-Url: http://git.shadowcat.co.uk/gitweb/gitweb.cgi?p=catagits%2FCatalyst-Runtime.git;a=blobdiff_plain;f=lib%2FCatalyst%2FEngine.pm;h=70f49fb488d8f1162c5ae48c56bf27658a14df55;hp=948f28ff76edbc62e4eaa3b6c79c554ef841c138;hb=199731fb710c6a165793f055f85de60539039dfe;hpb=6adc45cf93c50a080f1f32bad475fd2ab5844887 diff --git a/lib/Catalyst/Engine.pm b/lib/Catalyst/Engine.pm index 948f28f..70f49fb 100644 --- a/lib/Catalyst/Engine.pm +++ b/lib/Catalyst/Engine.pm @@ -10,10 +10,11 @@ use HTML::Entities; use HTTP::Headers; use Plack::Loader; use Catalyst::EngineLoader; -use Encode 2.21 'decode_utf8'; +use Encode 2.21 'decode_utf8', 'encode', 'decode'; use Plack::Request::Upload; use Hash::MultiValue; use namespace::clean -except => 'meta'; +use utf8; # Amount of data to read from input on each pass our $CHUNKSIZE = 64 * 1024; @@ -129,7 +130,6 @@ sub finalize_body { # There's no body... $body = []; } - $res->_response_cb->([ $res->status, \@headers, $body]); $res->_clear_response_cb; @@ -159,11 +159,11 @@ sub finalize_body { } else { - # Case where body was set afgter calling ->write. We'd prefer not to + # Case where body was set after calling ->write. We'd prefer not to # support this, but I can see some use cases with the way most of the - # views work. - - $self->write($c, $body ); + # views work. Since body has already been encoded, we need to do + # an 'unencoded_write' here. + $self->unencoded_write( $c, $body ); } } @@ -573,63 +573,50 @@ process the query string and extract query parameters. sub prepare_query_parameters { my ($self, $c) = @_; my $env = $c->request->env; + my $do_not_decode_query = $c->config->{do_not_decode_query}; - if(my $query_obj = $env->{'plack.request.query'}) { - $c->request->query_parameters( - $c->request->_use_hash_multivalue ? - $query_obj->clone : - $query_obj->as_hashref_mixed); - return; + my $old_encoding; + if(my $new = $c->config->{default_query_encoding}) { + $old_encoding = $c->encoding; + $c->encoding($new); } + my $check = $c->config->{do_not_check_query_encoding} ? undef :$c->_encode_check; + my $decoder = sub { + my $str = shift; + return $str if $do_not_decode_query; + return $c->_handle_param_unicode_decoding($str, $check); + }; + my $query_string = exists $env->{QUERY_STRING} ? $env->{QUERY_STRING} : ''; - # Check for keywords (no = signs) - # (yes, index() is faster than a regex :)) - if ( index( $query_string, '=' ) < 0 ) { - my $keywords = $self->unescape_uri($query_string); - $keywords = decode_utf8 $keywords; - $c->request->query_keywords($keywords); - return; - } - - my %query; + $query_string =~ s/\A[&;]+//; - # replace semi-colons - $query_string =~ s/;/&/g; + my @unsplit_pairs = split /[&;]+/, $query_string; + my $p = Hash::MultiValue->new(); - my @params = grep { length $_ } split /&/, $query_string; + my $is_first_pair = 1; + for my $pair (@unsplit_pairs) { + my ($name, $value) + = map { defined $_ ? $decoder->($self->unescape_uri($_)) : $_ } + ( split /=/, $pair, 2 )[0,1]; # slice forces two elements - for my $item ( @params ) { + if ($is_first_pair) { + # If the first pair has no equal sign, then it means the isindex + # flag is set. + $c->request->query_keywords($name) unless defined $value; - my ($param, $value) - = map { decode_utf8($self->unescape_uri($_)) } - split( /=/, $item, 2 ); - - unless(defined $param) { - $param = $self->unescape_uri($item); - $param = decode_utf8 $param; + $is_first_pair = 0; } - if ( exists $query{$param} ) { - if ( ref $query{$param} ) { - push @{ $query{$param} }, $value; - } - else { - $query{$param} = [ $query{$param}, $value ]; - } - } - else { - $query{$param} = $value; - } + $p->add( $name => $value ); } - $c->request->query_parameters( - $c->request->_use_hash_multivalue ? - Hash::MultiValue->from_mixed(\%query) : - \%query); + + $c->encoding($old_encoding) if $old_encoding; + $c->request->query_parameters( $c->request->_use_hash_multivalue ? $p : $p->mixed ); } =head2 $self->prepare_read($c) @@ -673,8 +660,8 @@ sub prepare_uploads { my $uploads = $request->_body->upload; my $parameters = $request->parameters; foreach my $name (keys %$uploads) { - $name = $c->_handle_unicode_decoding($name) if $enc; my $files = $uploads->{$name}; + $name = $c->_handle_unicode_decoding($name) if $enc; my @uploads; for my $upload (ref $files eq 'ARRAY' ? @$files : ($files)) { my $headers = HTTP::Headers->new( %{ $upload->{headers} } ); @@ -723,6 +710,20 @@ sub write { $c->response->write($buffer); } +=head2 $self->unencoded_write($c, $buffer) + +Writes the buffer to the client without encoding. Necessary for +already encoded buffers. Used when a $c->write has been done +followed by $c->res->body. + +=cut + +sub unencoded_write { + my ( $self, $c, $buffer ) = @_; + + $c->response->unencoded_write($buffer); +} + =head2 $self->read($c, [$maxlength]) Reads from the input stream by calling C<< $self->read_chunk >>.