Debian lenny version packages
[pkg-perl] / deb-src / libwww-perl / libwww-perl-5.813 / lib / LWP / Authen / Digest.pm
1 package LWP::Authen::Digest;
2 use strict;
3
4 require Digest::MD5;
5
6 sub authenticate
7 {
8     my($class, $ua, $proxy, $auth_param, $response,
9        $request, $arg, $size) = @_;
10
11     my($user, $pass) = $ua->get_basic_credentials($auth_param->{realm},
12                                                   $request->url, $proxy);
13     return $response unless defined $user and defined $pass;
14
15     my $nc = sprintf "%08X", ++$ua->{authen_md5_nonce_count}{$auth_param->{nonce}};
16     my $cnonce = sprintf "%8x", time;
17
18     my $uri = $request->url->path_query;
19     $uri = "/" unless length $uri;
20
21     my $md5 = Digest::MD5->new;
22
23     my(@digest);
24     $md5->add(join(":", $user, $auth_param->{realm}, $pass));
25     push(@digest, $md5->hexdigest);
26     $md5->reset;
27
28     push(@digest, $auth_param->{nonce});
29
30     if ($auth_param->{qop}) {
31         push(@digest, $nc, $cnonce, ($auth_param->{qop} =~ m|^auth[,;]auth-int$|) ? 'auth' : $auth_param->{qop});
32     }
33
34     $md5->add(join(":", $request->method, $uri));
35     push(@digest, $md5->hexdigest);
36     $md5->reset;
37
38     $md5->add(join(":", @digest));
39     my($digest) = $md5->hexdigest;
40     $md5->reset;
41
42     my %resp = map { $_ => $auth_param->{$_} } qw(realm nonce opaque);
43     @resp{qw(username uri response algorithm)} = ($user, $uri, $digest, "MD5");
44
45     if (($auth_param->{qop} || "") =~ m|^auth([,;]auth-int)?$|) {
46         @resp{qw(qop cnonce nc)} = ("auth", $cnonce, $nc);
47     }
48
49     my(@order) = qw(username realm qop algorithm uri nonce nc cnonce response);
50     if($request->method =~ /^(?:POST|PUT)$/) {
51         $md5->add($request->content);
52         my $content = $md5->hexdigest;
53         $md5->reset;
54         $md5->add(join(":", @digest[0..1], $content));
55         $md5->reset;
56         $resp{"message-digest"} = $md5->hexdigest;
57         push(@order, "message-digest");
58     }
59     push(@order, "opaque");
60     my @pairs;
61     for (@order) {
62         next unless defined $resp{$_};
63         push(@pairs, "$_=" . qq("$resp{$_}"));
64     }
65
66     my $auth_header = $proxy ? "Proxy-Authorization" : "Authorization";
67     my $auth_value  = "Digest " . join(", ", @pairs);
68
69     # Need to check this isn't a repeated fail!
70     my $r = $response;
71     while ($r) {
72         my $u = $r->request->{digest_user_pass};
73         if ($u && $u->[0] eq $user && $u->[1] eq $pass) {
74             # here we know this failed before
75             $response->header("Client-Warning" =>
76                               "Credentials for '$user' failed before");
77             return $response;
78         }
79         $r = $r->previous;
80     }
81
82     my $referral = $request->clone;
83     $referral->header($auth_header => $auth_value);
84     # we shouldn't really do this, but...
85     $referral->{digest_user_pass} = [$user, $pass];
86
87     return $ua->request($referral, $arg, $size, $response);
88 }
89
90 1;