ViewVC Help
View File | Revision Log | Show Annotations | Download File
/cvs/AnyEvent-HTTP/HTTP.pm
(Generate patch)

Comparing AnyEvent-HTTP/HTTP.pm (file contents):
Revision 1.70 by root, Fri Dec 31 20:31:47 2010 UTC vs.
Revision 1.77 by root, Sat Jan 1 19:13:41 2011 UTC

122 122
123If the server sends a header multiple times, then their contents will be 123If the server sends a header multiple times, then their contents will be
124joined together with a comma (C<,>), as per the HTTP spec. 124joined together with a comma (C<,>), as per the HTTP spec.
125 125
126If an internal error occurs, such as not being able to resolve a hostname, 126If an internal error occurs, such as not being able to resolve a hostname,
127then C<$data> will be C<undef>, C<< $headers->{Status} >> will be C<59x> 127then C<$data> will be C<undef>, C<< $headers->{Status} >> will be
128(usually C<599>) and the C<Reason> pseudo-header will contain an error 128C<590>-C<599> and the C<Reason> pseudo-header will contain an error
129message. 129message. Currently the following status codes are used:
130
131=over 4
132
133=item 595 - errors during connection etsbalishment, proxy handshake.
134
135=item 596 - errors during TLS negotiation, request sending and header processing.
136
137=item 597 - errors during body receive or processing.
138
139=item 598 - user aborted request in C<on_header> or C<on_body>.
140
141=item 599 - other, usually nonretryable, errors (garbled URL etc.).
142
143=back
130 144
131A typical callback might look like this: 145A typical callback might look like this:
132 146
133 sub { 147 sub {
134 my ($body, $hdr) = @_; 148 my ($body, $hdr) = @_;
364 push @{ $CO_SLOT{$_[0]}[1] }, $_[1]; 378 push @{ $CO_SLOT{$_[0]}[1] }, $_[1];
365 379
366 _slot_schedule $_[0]; 380 _slot_schedule $_[0];
367} 381}
368 382
383# extract cookies from jar
384sub cookie_jar_extract($$$$) {
385 my ($jar, $uscheme, $uhost, $upath) = @_;
386
387 %$jar = () if $jar->{version} != 1;
388
389 my @cookies;
390
391 while (my ($chost, $paths) = each %$jar) {
392 next unless ref $paths;
393
394 if ($chost =~ /^\./) {
395 next unless $chost eq substr $uhost, -length $chost;
396 } elsif ($chost =~ /\./) {
397 next unless $chost eq $uhost;
398 } else {
399 next;
400 }
401
402 while (my ($cpath, $cookies) = each %$paths) {
403 next unless $cpath eq substr $upath, 0, length $cpath;
404
405 while (my ($cookie, $kv) = each %$cookies) {
406 next if $uscheme ne "https" && exists $kv->{secure};
407
408 if (exists $kv->{expires}) {
409 if (AE::now > parse_date ($kv->{expires})) {
410 delete $cookies->{$cookie};
411 next;
412 }
413 }
414
415 my $value = $kv->{value};
416
417 if ($value =~ /[=;,[:space:]]/) {
418 $value =~ s/([\\"])/\\$1/g;
419 $value = "\"$value\"";
420 }
421
422 push @cookies, "$cookie=$value";
423 }
424 }
425 }
426
427 \@cookies
428}
429
430# parse set_cookie header into jar
431sub cookie_jar_set_cookie($$$) {
432 my ($jar, $set_cookie, $uhost) = @_;
433
434 for ($set_cookie) {
435 # parse NAME=VALUE
436 my @kv;
437
438 while (
439 m{
440 \G\s*
441 (?:
442 expires \s*=\s* ([A-Z][a-z][a-z],\ [^,;]+)
443 | ([^=;,[:space:]]+) \s*=\s* (?: "((?:[^\\"]+|\\.)*)" | ([^=;,[:space:]]*) )
444 )
445 }gcxsi
446 ) {
447 my $name = $2;
448 my $value = $4;
449
450 unless (defined $name) {
451 # expires
452 $name = "expires";
453 $value = $1;
454 } elsif (!defined $value) {
455 # quoted
456 $value = $3;
457 $value =~ s/\\(.)/$1/gs;
458 }
459
460 push @kv, lc $name, $value;
461
462 last unless /\G\s*;/gc;
463 }
464
465 last unless @kv;
466
467 my $name = shift @kv;
468 my %kv = (value => shift @kv, @kv);
469
470 $kv{expires} ||= format_date (AE::now + $kv{"max-age"})
471 if exists $kv{"max-age"};
472
473 my $cdom;
474 my $cpath = (delete $kv{path}) || "/";
475
476 if (exists $kv{domain}) {
477 $cdom = delete $kv{domain};
478
479 $cdom =~ s/^\.?/./; # make sure it starts with a "."
480
481 next if $cdom =~ /\.$/;
482
483 # this is not rfc-like and not netscape-like. go figure.
484 my $ndots = $cdom =~ y/.//;
485 next if $ndots < ($cdom =~ /\.[^.][^.]\.[^.][^.]$/ ? 3 : 2);
486 } else {
487 $cdom = $uhost;
488 }
489
490 # store it
491 $jar->{version} = 1;
492 $jar->{$cdom}{$cpath}{$name} = \%kv;
493
494 redo if /\G\s*,/gc;
495 }
496}
497
369# continue to parse $_ for headers and place them into the arg 498# continue to parse $_ for headers and place them into the arg
370sub parse_hdr() { 499sub parse_hdr() {
371 my %hdr; 500 my %hdr;
372 501
373 # things seen, not parsed: 502 # things seen, not parsed:
448 577
449 $upath =~ s%^/?%/%; 578 $upath =~ s%^/?%/%;
450 579
451 # cookie processing 580 # cookie processing
452 if (my $jar = $arg{cookie_jar}) { 581 if (my $jar = $arg{cookie_jar}) {
453 %$jar = () if $jar->{version} != 1; 582 my $cookies = cookie_jar_extract $jar, $uscheme, $uhost, $upath;
454
455 my @cookie;
456
457 while (my ($chost, $paths) = each %$jar) {
458 if ($chost =~ /^\./) {
459 next unless $chost eq substr $uhost, -length $chost;
460 } elsif ($chost =~ /\./) {
461 next unless $chost eq $uhost;
462 } else {
463 next;
464 }
465 583
466 while (my ($cpath, $cookies) = each %$paths) {
467 next unless $cpath eq substr $upath, 0, length $cpath;
468
469 while (my ($cookie, $kv) = each %$cookies) {
470 next if $uscheme ne "https" && exists $kv->{secure};
471
472 if (exists $kv->{expires}) {
473 if (AE::now > parse_date ($kv->{expires})) {
474 delete $cookies->{$cookie};
475 next;
476 }
477 }
478
479 my $value = $kv->{value};
480 $value =~ s/([\\"])/\\$1/g;
481 push @cookie, "$cookie=\"$value\"";
482 }
483 }
484 }
485
486 $hdr{cookie} = join "; ", @cookie 584 $hdr{cookie} = join "; ", @$cookies
487 if @cookie; 585 if @$cookies;
488 } 586 }
489 587
490 my ($rhost, $rport, $rscheme, $rpath); # request host, port, path 588 my ($rhost, $rport, $rscheme, $rpath); # request host, port, path
491 589
492 if ($proxy) { 590 if ($proxy) {
516 _get_slot $uhost, sub { 614 _get_slot $uhost, sub {
517 $state{slot_guard} = shift; 615 $state{slot_guard} = shift;
518 616
519 return unless $state{connect_guard}; 617 return unless $state{connect_guard};
520 618
619 my $ae_error = 595; # connecting
620
521 my $connect_cb = sub { 621 my $connect_cb = sub {
522 $state{fh} = shift 622 $state{fh} = shift
523 or do { 623 or do {
524 my $err = "$!"; 624 my $err = "$!";
525 %state = (); 625 %state = ();
526 return $cb->(undef, { @pseudo, Status => 599, Reason => $err }); 626 return $cb->(undef, { @pseudo, Status => $ae_error, Reason => $err });
527 }; 627 };
528
529 pop; # free memory, save a tree
530 628
531 return unless delete $state{connect_guard}; 629 return unless delete $state{connect_guard};
532 630
533 # get handle 631 # get handle
534 $state{handle} = new AnyEvent::Handle 632 $state{handle} = new AnyEvent::Handle
537 tls_ctx => $arg{tls_ctx}, 635 tls_ctx => $arg{tls_ctx},
538 # these need to be reconfigured on keepalive handles 636 # these need to be reconfigured on keepalive handles
539 timeout => $timeout, 637 timeout => $timeout,
540 on_error => sub { 638 on_error => sub {
541 %state = (); 639 %state = ();
542 $cb->(undef, { @pseudo, Status => 599, Reason => $_[2] }); 640 $cb->(undef, { @pseudo, Status => $ae_error, Reason => $_[2] });
543 }, 641 },
544 on_eof => sub { 642 on_eof => sub {
545 %state = (); 643 %state = ();
546 $cb->(undef, { @pseudo, Status => 599, Reason => "Unexpected end-of-file" }); 644 $cb->(undef, { @pseudo, Status => $ae_error, Reason => "Unexpected end-of-file" });
547 }, 645 },
548 ; 646 ;
549 647
550 # limit the number of persistent connections 648 # limit the number of persistent connections
551 # keepalive not yet supported 649 # keepalive not yet supported
559 657
560 $state{handle}->starttls ("connect") if $rscheme eq "https"; 658 $state{handle}->starttls ("connect") if $rscheme eq "https";
561 659
562 # handle actual, non-tunneled, request 660 # handle actual, non-tunneled, request
563 my $handle_actual_request = sub { 661 my $handle_actual_request = sub {
662 $ae_error = 596; # request phase
663
564 $state{handle}->starttls ("connect") if $uscheme eq "https" && !exists $state{handle}{tls}; 664 $state{handle}->starttls ("connect") if $uscheme eq "https" && !exists $state{handle}{tls};
565 665
566 # send request 666 # send request
567 $state{handle}->push_write ( 667 $state{handle}->push_write (
568 "$method $rpath HTTP/1.1\015\012" 668 "$method $rpath HTTP/1.1\015\012"
579 # status line and headers 679 # status line and headers
580 $state{read_response} = sub { 680 $state{read_response} = sub {
581 for ("$_[1]") { 681 for ("$_[1]") {
582 y/\015//d; # weed out any \015, as they show up in the weirdest of places. 682 y/\015//d; # weed out any \015, as they show up in the weirdest of places.
583 683
584 /^HTTP\/([0-9\.]+) \s+ ([0-9]{3}) (?: \s+ ([^\012]*) )? \012/igxc 684 /^HTTP\/0*([0-9\.]+) \s+ ([0-9]{3}) (?: \s+ ([^\012]*) )? \012/gxci
585 or return (%state = (), $cb->(undef, { @pseudo, Status => 599, Reason => "Invalid server response" })); 685 or return (%state = (), $cb->(undef, { @pseudo, Status => 599, Reason => "Invalid server response" }));
586 686
587 # 100 Continue handling 687 # 100 Continue handling
588 # should not happen as we don't send expect: 100-continue, 688 # should not happen as we don't send expect: 100-continue,
589 # but we handle it just in case. 689 # but we handle it just in case.
624 724
625 if ($recurse) { 725 if ($recurse) {
626 my $status = $hdr{Status}; 726 my $status = $hdr{Status};
627 727
628 # industry standard is to redirect POST as GET for 728 # industry standard is to redirect POST as GET for
629 # 301, 302 and 303, in contrast to http/1.0 and 1.1. 729 # 301, 302 and 303, in contrast to HTTP/1.0 and 1.1.
630 # also, the UA should ask the user for 301 and 307 and POST, 730 # also, the UA should ask the user for 301 and 307 and POST,
631 # industry standard seems to be to simply follow. 731 # industry standard seems to be to simply follow.
632 # we go with the industry standard. 732 # we go with the industry standard.
633 if ($status == 301 or $status == 302 or $status == 303) { 733 if ($status == 301 or $status == 302 or $status == 303) {
634 # HTTP/1.1 is unclear on how to mutate the method 734 # HTTP/1.1 is unclear on how to mutate the method
638 $redirect = 1; 738 $redirect = 1;
639 } 739 }
640 } 740 }
641 741
642 my $finish = sub { # ($data, $err_status, $err_reason[, $keepalive]) 742 my $finish = sub { # ($data, $err_status, $err_reason[, $keepalive])
643 my $keepalive = pop; 743 my $may_keep_alive = $_[3];
644 744
645 $state{handle}->destroy if $state{handle}; 745 $state{handle}->destroy if $state{handle};
646 %state = (); 746 %state = ();
647 747
648 if (defined $_[1]) { 748 if (defined $_[1]) {
650 $hdr{OrigReason} = $hdr{Reason}; $hdr{Reason} = $_[2]; 750 $hdr{OrigReason} = $hdr{Reason}; $hdr{Reason} = $_[2];
651 } 751 }
652 752
653 # set-cookie processing 753 # set-cookie processing
654 if ($arg{cookie_jar}) { 754 if ($arg{cookie_jar}) {
655 for ($hdr{"set-cookie"}) { 755 cookie_jar_set_cookie $arg{cookie_jar}, $hdr{"set-cookie"}, $uhost;
656 # parse NAME=VALUE
657 my @kv;
658
659 while (
660 m{
661 \G\s*
662 (?:
663 expires \s*=\s* ([A-Z][a-z][a-z],\ [^,;]+)
664 | ([^=;,[:space:]]+) \s*=\s* (?: "((?:[^\\"]+|\\.)*)" | ([^=;,[:space:]]*) )
665 )
666 }gcxsi
667 ) {
668 my $name = $2;
669 my $value = $4;
670
671 unless (defined $name) {
672 # expires
673 $name = "expires";
674 $value = $1;
675 } elsif (!defined $value) {
676 # quoted
677 $value = $3;
678 $value =~ s/\\(.)/$1/gs;
679 }
680
681 push @kv, lc $name, $value;
682
683 last unless /\G\s*;/gc;
684 }
685
686 last unless @kv;
687
688 my $name = shift @kv;
689 my %kv = (value => shift @kv, @kv);
690
691 $kv{expires} ||= format_date (AE::now + $kv{"max-age"})
692 if exists $kv{"max-age"};
693
694 my $cdom;
695 my $cpath = (delete $kv{path}) || "/";
696
697 if (exists $kv{domain}) {
698 $cdom = delete $kv{domain};
699
700 $cdom =~ s/^\.?/./; # make sure it starts with a "."
701
702 next if $cdom =~ /\.$/;
703
704 # this is not rfc-like and not netscape-like. go figure.
705 my $ndots = $cdom =~ y/.//;
706 next if $ndots < ($cdom =~ /\.[^.][^.]\.[^.][^.]$/ ? 3 : 2);
707 } else {
708 $cdom = $uhost;
709 }
710
711 # store it
712 $arg{cookie_jar}{version} = 1;
713 $arg{cookie_jar}{$cdom}{$cpath}{$name} = \%kv;
714
715 redo if /\G\s*,/gc;
716 }
717 } 756 }
718 757
719 if ($redirect && exists $hdr{location}) { 758 if ($redirect && exists $hdr{location}) {
720 # we ignore any errors, as it is very common to receive 759 # we ignore any errors, as it is very common to receive
721 # Content-Length != 0 but no actual body 760 # Content-Length != 0 but no actual body
728 $cb); 767 $cb);
729 } else { 768 } else {
730 $cb->($_[0], \%hdr); 769 $cb->($_[0], \%hdr);
731 } 770 }
732 }; 771 };
772
773 $ae_error = 597; # body phase
733 774
734 my $len = $hdr{"content-length"}; 775 my $len = $hdr{"content-length"};
735 776
736 if (!$redirect && $arg{on_header} && !$arg{on_header}(\%hdr)) { 777 if (!$redirect && $arg{on_header} && !$arg{on_header}(\%hdr)) {
737 $finish->(undef, 598 => "Request cancelled by on_header"); 778 $finish->(undef, 598 => "Request cancelled by on_header");
759 } elsif ($hdr{"transfer-encoding"} =~ /\bchunked\b/i) { 800 } elsif ($hdr{"transfer-encoding"} =~ /\bchunked\b/i) {
760 my $cl = 0; 801 my $cl = 0;
761 my $body = undef; 802 my $body = undef;
762 my $on_body = $arg{on_body} || sub { $body .= shift; 1 }; 803 my $on_body = $arg{on_body} || sub { $body .= shift; 1 };
763 804
764 $_[0]->on_error (sub { $finish->(undef, 599 => $_[2]) });
765
766 my $read_chunk; $read_chunk = sub { 805 my $read_chunk; $read_chunk = sub {
767 $_[1] =~ /^([0-9a-fA-F]+)/ 806 $_[1] =~ /^([0-9a-fA-F]+)/
768 or $finish->(undef, 599 => "Garbled chunked transfer encoding"); 807 or $finish->(undef, $ae_error => "Garbled chunked transfer encoding");
769 808
770 my $len = hex $1; 809 my $len = hex $1;
771 810
772 if ($len) { 811 if ($len) {
773 $cl += $len; 812 $cl += $len;
776 $on_body->($_[1], \%hdr) 815 $on_body->($_[1], \%hdr)
777 or return $finish->(undef, 598 => "Request cancelled by on_body"); 816 or return $finish->(undef, 598 => "Request cancelled by on_body");
778 817
779 $_[0]->push_read (line => sub { 818 $_[0]->push_read (line => sub {
780 length $_[1] 819 length $_[1]
781 and return $finish->(undef, 599 => "Garbled chunked transfer encoding"); 820 and return $finish->(undef, $ae_error => "Garbled chunked transfer encoding");
782 $_[0]->push_read (line => $read_chunk); 821 $_[0]->push_read (line => $read_chunk);
783 }); 822 });
784 }); 823 });
785 } else { 824 } else {
786 $hdr{"content-length"} ||= $cl; 825 $hdr{"content-length"} ||= $cl;
789 if (length $_[1]) { 828 if (length $_[1]) {
790 for ("$_[1]") { 829 for ("$_[1]") {
791 y/\015//d; # weed out any \015, as they show up in the weirdest of places. 830 y/\015//d; # weed out any \015, as they show up in the weirdest of places.
792 831
793 my $hdr = parse_hdr 832 my $hdr = parse_hdr
794 or return $finish->(undef, 599 => "Garbled response trailers"); 833 or return $finish->(undef, $ae_error => "Garbled response trailers");
795 834
796 %hdr = (%hdr, %$hdr); 835 %hdr = (%hdr, %$hdr);
797 } 836 }
798 } 837 }
799 838
803 }; 842 };
804 843
805 $_[0]->push_read (line => $read_chunk); 844 $_[0]->push_read (line => $read_chunk);
806 845
807 } elsif ($arg{on_body}) { 846 } elsif ($arg{on_body}) {
808 $_[0]->on_error (sub { $finish->(undef, 599 => $_[2]) });
809
810 if ($len) { 847 if ($len) {
811 $_[0]->on_read (sub { 848 $_[0]->on_read (sub {
812 $len -= length $_[0]{rbuf}; 849 $len -= length $_[0]{rbuf};
813 850
814 $arg{on_body}(delete $_[0]{rbuf}, \%hdr) 851 $arg{on_body}(delete $_[0]{rbuf}, \%hdr)
828 } 865 }
829 } else { 866 } else {
830 $_[0]->on_eof (undef); 867 $_[0]->on_eof (undef);
831 868
832 if ($len) { 869 if ($len) {
833 $_[0]->on_error (sub { $finish->(undef, 599 => $_[2]) });
834 $_[0]->on_read (sub { 870 $_[0]->on_read (sub {
835 $finish->((substr delete $_[0]{rbuf}, 0, $len, ""), undef, undef, 1) 871 $finish->((substr delete $_[0]{rbuf}, 0, $len, ""), undef, undef, 1)
836 if $len <= length $_[0]{rbuf}; 872 if $len <= length $_[0]{rbuf};
837 }); 873 });
838 } else { 874 } else {
839 $_[0]->on_error (sub { 875 $_[0]->on_error (sub {
840 ($! == Errno::EPIPE || !$!) 876 ($! == Errno::EPIPE || !$!)
841 ? $finish->(delete $_[0]{rbuf}) 877 ? $finish->(delete $_[0]{rbuf})
842 : $finish->(undef, 599 => $_[2]); 878 : $finish->(undef, $ae_error => $_[2]);
843 }); 879 });
844 $_[0]->on_read (sub { }); 880 $_[0]->on_read (sub { });
845 } 881 }
846 } 882 }
847 } 883 }

Diff Legend

Removed lines
+ Added lines
< Changed lines
> Changed lines