File Coverage

blib/lib/Email/MIME/Encode.pm
Criterion Covered Total %
statement 95 95 100.0
branch 29 40 72.5
condition 15 19 78.9
subroutine 17 17 100.0
pod 0 4 0.0
total 156 175 89.1


line stmt bran cond sub pod time code
1 20     20   72847 use v5.12.0;
  20         88  
2 20     20   113 use warnings;
  20         40  
  20         841  
3             package Email::MIME::Encode 1.953;
4             # ABSTRACT: a private helper for MIME header encoding
5              
6 20     20   107 use Carp ();
  20         38  
  20         273  
7 20     20   636 use Encode ();
  20         10183  
  20         329  
8 20     20   7612 use Email::MIME::Header;
  20         51  
  20         636  
9 20     20   7825 use MIME::Base64();
  20         11186  
  20         468  
10 20     20   125 use Module::Runtime ();
  20         38  
  20         335  
11 20     20   95 use Scalar::Util;
  20         39  
  20         4879  
12              
13             our @CARP_NOT;
14              
15             my %no_mime_headers = map { $_ => undef } qw(date message-id in-reply-to references downgraded-message-id downgraded-in-reply-to downgraded-references);
16              
17             sub maybe_mime_encode_header {
18 44     44 0 192 my ($header, $val, $charset) = @_;
19              
20 44         86 $header = lc $header;
21              
22 44         81 my $header_name_length = length($header) + length(": ");
23              
24 44 100 100     181 if (Scalar::Util::blessed($val) && $val->can("as_mime_string")) {
25 2         14 return $val->as_mime_string({
26             charset => $charset,
27             header_name_length => $header_name_length,
28             });
29             }
30              
31             return _object_encode($val, $charset, $header_name_length, $Email::MIME::Header::header_to_class_map{$header})
32 42 100       152 if exists $Email::MIME::Header::header_to_class_map{$header};
33              
34 20         34 my $min_wrap_length = 78 - $header_name_length + 1;
35              
36 20 100 66     36 return $val
37             unless _needs_mime_encode($val) || $val =~ /[^\s]{$min_wrap_length,}/;
38              
39             return $val
40 11 50       37 if exists $no_mime_headers{$header};
41              
42 11         23 return mime_encode($val, $charset, $header_name_length);
43             }
44              
45             sub _needs_mime_encode {
46 100     100   171 my ($val) = @_;
47 100   100 1   1133 return defined $val && $val =~ /(?:\P{ASCII}|=\?|[^\s]{79,}|^\s+|\s+$)/s;
  1         7  
  1         2  
  1         13  
48             }
49              
50             sub _needs_mime_encode_addr {
51 80     80   449 my ($val) = @_;
52 80   66     131 return _needs_mime_encode($val) || ( defined $val && $val =~ /[:;,]/ );
53             }
54              
55             sub _object_encode {
56 22     22   52 my ($val, $charset, $header_name_length, $class) = @_;
57              
58 22         67 local @CARP_NOT = qw(Email::MIME Email::MIME::Header);
59              
60             {
61 22         32 local $@;
  22         29  
62 22 50       40 Carp::croak("Cannot load package '$class': $@") unless eval { Module::Runtime::require_module($class) };
  22         56  
63             }
64              
65 22 50       750 Carp::croak("Class '$class' does not have method 'from_string'") unless $class->can('from_string');
66              
67 22 100       116 my $object = $class->from_string(ref $val eq 'ARRAY' ? @{$val} : $val);
  2         8  
68              
69 22 50       99 Carp::croak("Object from class '$class' does not have method 'as_mime_string'") unless $object->can('as_mime_string');
70              
71 22         95 return $object->as_mime_string({
72             charset => $charset,
73             header_name_length => $header_name_length,
74             });
75             }
76              
77             # XXX this is copied directly out of Courriel::Header
78             # eventually, this should be extracted out into something that could be shared
79             sub mime_encode {
80 36     36 0 182 my ($text, $charset, $header_name_length) = @_;
81              
82 36   100     121 $header_name_length //= 0;
83 36   100     70 $charset //= 'UTF-8';
84              
85 36         105 my $enc_obj = Encode::find_encoding($charset);
86              
87 36         1522 my $head = '=?' . $enc_obj->mime_name() . '?B?';
88 36         2486 my $tail = '?=';
89              
90 36         74 my $mime_length = length($head) + length($tail);
91              
92             # This code is copied from Mail::Message::Field::Full in the Mail-Box
93             # distro.
94 36         137 my $real_length = int( ( 75 - $mime_length ) / 4 ) * 3;
95 36         72 my $first_length = int( ( 75 - $header_name_length - $mime_length ) / 4 ) * 3;
96              
97 36         47 my @result;
98 36         65 my $chunk = q{};
99 36         45 my $first_processed = 0;
100 36         195 while ( length( my $chr = substr( $text, 0, 1, '' ) ) ) {
101 832         1700 my $chr = $enc_obj->encode( $chr, 0 );
102              
103 832 100       1735 if ( length($chunk) + length($chr) > ( $first_processed ? $real_length : $first_length ) ) {
    100          
104 8 50       19 if ( length($chunk) > 0 ) {
105 8         31 push @result, $head . MIME::Base64::encode_base64( $chunk, q{} ) . $tail;
106 8         15 $chunk = q{};
107             }
108 8 50       18 $first_processed = 1
109             unless $first_processed;
110             }
111              
112 832         2294 $chunk .= $chr;
113             }
114              
115 36 50       156 push @result, $head . MIME::Base64::encode_base64( $chunk, q{} ) . $tail
116             if length $chunk;
117              
118 36         195 return join q{ }, @result;
119             }
120              
121             sub maybe_mime_decode_header {
122 37     37 0 81 my ($header, $val) = @_;
123              
124 37         62 $header = lc $header;
125              
126             return _object_decode($val, $Email::MIME::Header::header_to_class_map{$header})
127 37 100       126 if exists $Email::MIME::Header::header_to_class_map{$header};
128              
129             return $val
130 16 100       33 if exists $no_mime_headers{$header};
131              
132 15 100       37 return $val
133             unless $val =~ /=\?/;
134              
135 14         30 return mime_decode($val);
136             }
137              
138             sub _object_decode {
139 21     21   43 my ($string, $class) = @_;
140              
141 21         66 local @CARP_NOT = qw(Email::MIME Email::MIME::Header);
142              
143             {
144 21         29 local $@;
  21         34  
145 21 50       36 Carp::croak("Cannot load package '$class': $@") unless eval { Module::Runtime::require_module($class) };
  21         56  
146             }
147              
148 21 50       753 Carp::croak("Class '$class' does not have method 'from_mime_string'") unless $class->can('from_mime_string');
149              
150 21         65 my $object = $class->from_mime_string($string);
151              
152 21 50       117 Carp::croak("Object from class '$class' does not have method 'as_string'") unless $object->can('as_string');
153              
154 21         53 return $object->as_string();
155             }
156              
157             sub mime_decode {
158 64     64 0 314 my ($text) = @_;
159 64 50       145 return undef unless defined $text;
160              
161             # The eval is to cope with unknown encodings, like Latin-62, or other
162             # nonsense that gets put in there by spammers and weirdos
163             # -- rjbs, 2014-12-04
164 64         98 local $@;
165 64         103 my $result = eval { Encode::decode("MIME-Header", $text) };
  64         154  
166 64   33     26998 return $result // $text;
167             }
168              
169             1;
170              
171             __END__