File Coverage

blib/lib/LWP/Authen/Digest.pm
Criterion Covered Total %
statement 49 49 100.0
branch 10 14 71.4
condition 4 5 80.0
subroutine 3 3 100.0
pod 0 1 0.0
total 66 72 91.6


line stmt bran cond sub pod time code
1             package LWP::Authen::Digest;
2              
3 1     1   10 use strict;
  1         12  
  1         27  
4 1     1   4 use base 'LWP::Authen::Basic';
  1         2  
  1         573  
5              
6             our $VERSION = '6.34';
7              
8             require Digest::MD5;
9              
10             sub auth_header {
11 8     8 0 14 my($class, $user, $pass, $request, $ua, $h) = @_;
12              
13 8         12 my $auth_param = $h->{auth_param};
14              
15 8         26 my $nc = sprintf "%08X", ++$ua->{authen_md5_nonce_count}{$auth_param->{nonce}};
16 8         15 my $cnonce = sprintf "%8x", time;
17              
18 8         20 my $uri = $request->uri->path_query;
19 8 50       120 $uri = "/" unless length $uri;
20              
21 8         40 my $md5 = Digest::MD5->new;
22              
23 8         12 my(@digest);
24 8         31 $md5->add(join(":", $user, $auth_param->{realm}, $pass));
25 8         52 push(@digest, $md5->hexdigest);
26 8         24 $md5->reset;
27              
28 8         11 push(@digest, $auth_param->{nonce});
29              
30 8 50       15 if ($auth_param->{qop}) {
31 8 50       19 push(@digest, $nc, $cnonce, ($auth_param->{qop} =~ m|^auth[,;]auth-int$|) ? 'auth' : $auth_param->{qop});
32             }
33              
34 8         20 $md5->add(join(":", $request->method, $uri));
35 8         92 push(@digest, $md5->hexdigest);
36 8         18 $md5->reset;
37              
38 8         22 $md5->add(join(":", @digest));
39 8         20 my($digest) = $md5->hexdigest;
40 8         17 $md5->reset;
41              
42 8         9 my %resp = map { $_ => $auth_param->{$_} } qw(realm nonce opaque);
  24         50  
43 8         25 @resp{qw(username uri response algorithm)} = ($user, $uri, $digest, "MD5");
44              
45 8 50 50     33 if (($auth_param->{qop} || "") =~ m|^auth([,;]auth-int)?$|) {
46 8         22 @resp{qw(qop cnonce nc)} = ("auth", $cnonce, $nc);
47             }
48              
49 8         16 my(@order) = qw(username realm qop algorithm uri nonce nc cnonce response);
50 8 100       27 if($request->method =~ /^(?:POST|PUT)$/) {
51 2         21 $md5->add($request->content);
52 2         25 my $content = $md5->hexdigest;
53 2         9 $md5->reset;
54 2         8 $md5->add(join(":", @digest[0..1], $content));
55 2         4 $md5->reset;
56 2         4 $resp{"message-digest"} = $md5->hexdigest;
57 2         5 push(@order, "message-digest");
58             }
59 8         54 push(@order, "opaque");
60 8         9 my @pairs;
61 8         10 for (@order) {
62 82 100       125 next unless defined $resp{$_};
63              
64             # RFC2617 says that qop-value and nc-value should be unquoted.
65 74 100 100     160 if ( $_ eq 'qop' || $_ eq 'nc' ) {
66 16         27 push(@pairs, "$_=" . $resp{$_});
67             }
68             else {
69 58         121 push(@pairs, "$_=" . qq("$resp{$_}"));
70             }
71             }
72              
73 8         26 my $auth_value = "Digest " . join(", ", @pairs);
74 8         56 return $auth_value;
75             }
76              
77             1;