| line |
stmt |
bran |
cond |
sub |
pod |
time |
code |
|
1
|
|
|
|
|
|
|
package WWW::Postmark; |
|
2
|
|
|
|
|
|
|
|
|
3
|
|
|
|
|
|
|
# ABSTRACT: API for the Postmark mail service for web applications. |
|
4
|
|
|
|
|
|
|
|
|
5
|
3
|
|
|
3
|
|
41019
|
use strict; |
|
|
3
|
|
|
|
|
5
|
|
|
|
3
|
|
|
|
|
73
|
|
|
6
|
3
|
|
|
3
|
|
10
|
use warnings; |
|
|
3
|
|
|
|
|
5
|
|
|
|
3
|
|
|
|
|
63
|
|
|
7
|
|
|
|
|
|
|
|
|
8
|
3
|
|
|
3
|
|
10
|
use Carp; |
|
|
3
|
|
|
|
|
5
|
|
|
|
3
|
|
|
|
|
164
|
|
|
9
|
3
|
|
|
3
|
|
1458
|
use Email::Valid; |
|
|
3
|
|
|
|
|
271316
|
|
|
|
3
|
|
|
|
|
97
|
|
|
10
|
3
|
|
|
3
|
|
1916
|
use HTTP::Tiny; |
|
|
3
|
|
|
|
|
22236
|
|
|
|
3
|
|
|
|
|
102
|
|
|
11
|
3
|
|
|
3
|
|
1334
|
use JSON::MaybeXS qw/encode_json decode_json/; |
|
|
3
|
|
|
|
|
16939
|
|
|
|
3
|
|
|
|
|
151
|
|
|
12
|
3
|
|
|
3
|
|
15
|
use File::Basename; |
|
|
3
|
|
|
|
|
4
|
|
|
|
3
|
|
|
|
|
153
|
|
|
13
|
3
|
|
|
3
|
|
1252
|
use File::MimeInfo; |
|
|
3
|
|
|
|
|
9902
|
|
|
|
3
|
|
|
|
|
163
|
|
|
14
|
3
|
|
|
3
|
|
1266
|
use MIME::Base64 qw/encode_base64/; |
|
|
3
|
|
|
|
|
1399
|
|
|
|
3
|
|
|
|
|
2767
|
|
|
15
|
|
|
|
|
|
|
|
|
16
|
|
|
|
|
|
|
our $VERSION = "1.000001"; |
|
17
|
|
|
|
|
|
|
$VERSION = eval $VERSION; |
|
18
|
|
|
|
|
|
|
|
|
19
|
|
|
|
|
|
|
my $ua = HTTP::Tiny->new(timeout => 45); |
|
20
|
|
|
|
|
|
|
|
|
21
|
|
|
|
|
|
|
=encoding utf-8 |
|
22
|
|
|
|
|
|
|
|
|
23
|
|
|
|
|
|
|
=head1 NAME |
|
24
|
|
|
|
|
|
|
|
|
25
|
|
|
|
|
|
|
WWW::Postmark - API for the Postmark mail service for web applications. |
|
26
|
|
|
|
|
|
|
|
|
27
|
|
|
|
|
|
|
=head1 SYNOPSIS |
|
28
|
|
|
|
|
|
|
|
|
29
|
|
|
|
|
|
|
use WWW::Postmark; |
|
30
|
|
|
|
|
|
|
|
|
31
|
|
|
|
|
|
|
my $api = WWW::Postmark->new('api_token'); |
|
32
|
|
|
|
|
|
|
|
|
33
|
|
|
|
|
|
|
# or, if you want to use SSL |
|
34
|
|
|
|
|
|
|
my $api = WWW::Postmark->new('api_token', 1); |
|
35
|
|
|
|
|
|
|
|
|
36
|
|
|
|
|
|
|
# send an email |
|
37
|
|
|
|
|
|
|
$api->send(from => 'me@domain.tld', to => 'you@domain.tld, them@domain.tld', |
|
38
|
|
|
|
|
|
|
subject => 'an email message', body => "hi guys, what's up?"); |
|
39
|
|
|
|
|
|
|
|
|
40
|
|
|
|
|
|
|
=head1 DESCRIPTION |
|
41
|
|
|
|
|
|
|
|
|
42
|
|
|
|
|
|
|
The WWW::Postmark module provides a simple API for the Postmark web service, |
|
43
|
|
|
|
|
|
|
that provides email sending facilities for web applications. Postmark is |
|
44
|
|
|
|
|
|
|
located at L. It is a paid service that charges |
|
45
|
|
|
|
|
|
|
according the amount of emails you send, and requires signing up in order |
|
46
|
|
|
|
|
|
|
to receive an API token. |
|
47
|
|
|
|
|
|
|
|
|
48
|
|
|
|
|
|
|
You can send emails either through HTTP or HTTPS with SSL encryption. You |
|
49
|
|
|
|
|
|
|
can send your emails to multiple recipients at once (but there's a 20 |
|
50
|
|
|
|
|
|
|
recipients limit). If WWW::Postmark receives a successful response from |
|
51
|
|
|
|
|
|
|
the Postmark service, it will return a true value; otherwise it will die. |
|
52
|
|
|
|
|
|
|
|
|
53
|
|
|
|
|
|
|
To make it clear, Postmark is not an email marketing service for sending |
|
54
|
|
|
|
|
|
|
email campaigns or newsletters to multiple subscribers at once. It's meant |
|
55
|
|
|
|
|
|
|
for sending emails from web applications in response to certain events, |
|
56
|
|
|
|
|
|
|
like someone signing up to your website. |
|
57
|
|
|
|
|
|
|
|
|
58
|
|
|
|
|
|
|
Postmark provides a test API token that doesn't really send the emails. |
|
59
|
|
|
|
|
|
|
The token is 'POSTMARK_API_TEST', and you can use it for testing purposes |
|
60
|
|
|
|
|
|
|
(the tests in this distribution use this token). |
|
61
|
|
|
|
|
|
|
|
|
62
|
|
|
|
|
|
|
Besides sending emails, this module also provides support for Postmark's |
|
63
|
|
|
|
|
|
|
spam score API, which allows you to get a SpamAssassin report for an email |
|
64
|
|
|
|
|
|
|
message. See documentation for the C method for more info. |
|
65
|
|
|
|
|
|
|
|
|
66
|
|
|
|
|
|
|
=head1 METHODS |
|
67
|
|
|
|
|
|
|
|
|
68
|
|
|
|
|
|
|
=head2 new( [ $api_token, $use_ssl] ) |
|
69
|
|
|
|
|
|
|
|
|
70
|
|
|
|
|
|
|
Creates a new instance of this class, with a Postmark API token that you've |
|
71
|
|
|
|
|
|
|
received from the Postmark app. By default, requests are made through HTTP; |
|
72
|
|
|
|
|
|
|
if you want to send them with SSL encryption, pass a true value for |
|
73
|
|
|
|
|
|
|
C<$use_ssl>. |
|
74
|
|
|
|
|
|
|
|
|
75
|
|
|
|
|
|
|
If you do not provide an API token, you will only be able to use Postmark's |
|
76
|
|
|
|
|
|
|
spam score API (you will not be able to send emails). |
|
77
|
|
|
|
|
|
|
|
|
78
|
|
|
|
|
|
|
Note that in order to use SSL, C requires certain dependencies |
|
79
|
|
|
|
|
|
|
to be installed. See L for more information. |
|
80
|
|
|
|
|
|
|
|
|
81
|
|
|
|
|
|
|
=cut |
|
82
|
|
|
|
|
|
|
|
|
83
|
|
|
|
|
|
|
sub new { |
|
84
|
3
|
|
|
3
|
1
|
197843
|
my ($class, $token, $use_ssl) = @_; |
|
85
|
|
|
|
|
|
|
|
|
86
|
3
|
100
|
|
|
|
513
|
carp "You have not provided a Postmark API token, you will not be able to send emails." |
|
87
|
|
|
|
|
|
|
unless $token; |
|
88
|
|
|
|
|
|
|
|
|
89
|
3
|
|
50
|
|
|
25
|
$use_ssl ||= 0; |
|
90
|
3
|
50
|
|
|
|
11
|
$use_ssl = 1 if $use_ssl; |
|
91
|
|
|
|
|
|
|
|
|
92
|
3
|
|
|
|
|
31
|
bless { token => $token, use_ssl => $use_ssl }, $class; |
|
93
|
|
|
|
|
|
|
} |
|
94
|
|
|
|
|
|
|
|
|
95
|
|
|
|
|
|
|
=head2 send( %params ) |
|
96
|
|
|
|
|
|
|
|
|
97
|
|
|
|
|
|
|
Receives a hash representing the email message that should be sent and |
|
98
|
|
|
|
|
|
|
attempts to send it through the Postmark service. If the message was |
|
99
|
|
|
|
|
|
|
successfully sent, a hash reference of Postmark's response is returned |
|
100
|
|
|
|
|
|
|
(refer to L); |
|
101
|
|
|
|
|
|
|
otherwise, this method will croak with an approriate error message (see |
|
102
|
|
|
|
|
|
|
L"DIAGNOSTICS"> for a full list). |
|
103
|
|
|
|
|
|
|
|
|
104
|
|
|
|
|
|
|
The following keys are required when using this method: |
|
105
|
|
|
|
|
|
|
|
|
106
|
|
|
|
|
|
|
=over |
|
107
|
|
|
|
|
|
|
|
|
108
|
|
|
|
|
|
|
=item * from |
|
109
|
|
|
|
|
|
|
|
|
110
|
|
|
|
|
|
|
The email address of the sender. Either pass the email address itself |
|
111
|
|
|
|
|
|
|
in the format 'mail_address@domain.tld' or also provide a name, like |
|
112
|
|
|
|
|
|
|
'My Name '. |
|
113
|
|
|
|
|
|
|
|
|
114
|
|
|
|
|
|
|
=item * to |
|
115
|
|
|
|
|
|
|
|
|
116
|
|
|
|
|
|
|
The email address(es) of the recipient(s). You can use both formats as in |
|
117
|
|
|
|
|
|
|
'to', but here you can give multiple addresses. Use a comma to separate |
|
118
|
|
|
|
|
|
|
them. Note, however, that Postmark limits this to 20 recipients and sending |
|
119
|
|
|
|
|
|
|
will fail if you attempt to send to more than 20 addresses. |
|
120
|
|
|
|
|
|
|
|
|
121
|
|
|
|
|
|
|
=item * subject |
|
122
|
|
|
|
|
|
|
|
|
123
|
|
|
|
|
|
|
The subject of your message. |
|
124
|
|
|
|
|
|
|
|
|
125
|
|
|
|
|
|
|
=item * body |
|
126
|
|
|
|
|
|
|
|
|
127
|
|
|
|
|
|
|
The body of your message. This could be plain text, or HTML. If you want |
|
128
|
|
|
|
|
|
|
to send HTML, be sure to open with '' and close with ''. This |
|
129
|
|
|
|
|
|
|
module will look for these tags in order to find out whether you're sending |
|
130
|
|
|
|
|
|
|
a text message or an HTML message. |
|
131
|
|
|
|
|
|
|
|
|
132
|
|
|
|
|
|
|
Since version 0.3, however, you can explicitly specify the type of your |
|
133
|
|
|
|
|
|
|
message, and also send both plain text and HTML. To do so, use the C |
|
134
|
|
|
|
|
|
|
and/or C attributes. Their presence will override C. |
|
135
|
|
|
|
|
|
|
|
|
136
|
|
|
|
|
|
|
=item * html |
|
137
|
|
|
|
|
|
|
|
|
138
|
|
|
|
|
|
|
Instead of using C you can also specify the HTML content directly. |
|
139
|
|
|
|
|
|
|
|
|
140
|
|
|
|
|
|
|
=item * text |
|
141
|
|
|
|
|
|
|
|
|
142
|
|
|
|
|
|
|
... or the plain text part of the email. |
|
143
|
|
|
|
|
|
|
|
|
144
|
|
|
|
|
|
|
=back |
|
145
|
|
|
|
|
|
|
|
|
146
|
|
|
|
|
|
|
You can optionally supply the following parameters as well: |
|
147
|
|
|
|
|
|
|
|
|
148
|
|
|
|
|
|
|
=over |
|
149
|
|
|
|
|
|
|
|
|
150
|
|
|
|
|
|
|
=item * cc, bcc |
|
151
|
|
|
|
|
|
|
|
|
152
|
|
|
|
|
|
|
Same rules as the 'to' parameter. |
|
153
|
|
|
|
|
|
|
|
|
154
|
|
|
|
|
|
|
=item * tag |
|
155
|
|
|
|
|
|
|
|
|
156
|
|
|
|
|
|
|
Can be used to label your mail messages according to different categories, |
|
157
|
|
|
|
|
|
|
so you can analyze statistics of your mail sendings through the Postmark service. |
|
158
|
|
|
|
|
|
|
|
|
159
|
|
|
|
|
|
|
=item * attachments |
|
160
|
|
|
|
|
|
|
|
|
161
|
|
|
|
|
|
|
An array-ref with paths of files to attach to the email. C will |
|
162
|
|
|
|
|
|
|
automatically determine the MIME types of these files and encode their contents |
|
163
|
|
|
|
|
|
|
to base64 as Postmark requires. |
|
164
|
|
|
|
|
|
|
|
|
165
|
|
|
|
|
|
|
=item * reply_to |
|
166
|
|
|
|
|
|
|
|
|
167
|
|
|
|
|
|
|
Will force recipients of your email to send their replies to this mail |
|
168
|
|
|
|
|
|
|
address when replying to your email. |
|
169
|
|
|
|
|
|
|
|
|
170
|
|
|
|
|
|
|
=item * track_opens |
|
171
|
|
|
|
|
|
|
|
|
172
|
|
|
|
|
|
|
Set to a true value to enable Postmark's open tracking functionality. |
|
173
|
|
|
|
|
|
|
|
|
174
|
|
|
|
|
|
|
=back |
|
175
|
|
|
|
|
|
|
|
|
176
|
|
|
|
|
|
|
=cut |
|
177
|
|
|
|
|
|
|
|
|
178
|
|
|
|
|
|
|
sub send { |
|
179
|
8
|
|
|
8
|
1
|
42304
|
my ($self, %params) = @_; |
|
180
|
|
|
|
|
|
|
|
|
181
|
|
|
|
|
|
|
# do we have an API token? |
|
182
|
|
|
|
|
|
|
croak "You have not provided a Postmark API token, you cannot send emails" |
|
183
|
8
|
100
|
|
|
|
119
|
unless $self->{token}; |
|
184
|
|
|
|
|
|
|
|
|
185
|
|
|
|
|
|
|
# make sure there's a from address |
|
186
|
|
|
|
|
|
|
croak "You must provide a valid 'from' address in the format 'address\@domain.tld', or 'Your Name '." |
|
187
|
7
|
50
|
33
|
|
|
115
|
unless $params{from} && Email::Valid->address($params{from}); |
|
188
|
|
|
|
|
|
|
|
|
189
|
|
|
|
|
|
|
# make sure there's at least on to address |
|
190
|
|
|
|
|
|
|
croak $self->_recipient_error('to') |
|
191
|
7
|
50
|
|
|
|
3258
|
unless $params{to}; |
|
192
|
|
|
|
|
|
|
|
|
193
|
|
|
|
|
|
|
# validate all 'to' addresses |
|
194
|
7
|
|
|
|
|
27
|
$self->_validate_recipients('to', $params{to}); |
|
195
|
|
|
|
|
|
|
|
|
196
|
|
|
|
|
|
|
# make sure there's a subject |
|
197
|
|
|
|
|
|
|
croak "You must provide a mail subject." |
|
198
|
7
|
50
|
|
|
|
18
|
unless $params{subject}; |
|
199
|
|
|
|
|
|
|
|
|
200
|
|
|
|
|
|
|
# make sure there's a mail body |
|
201
|
|
|
|
|
|
|
croak "You must provide a mail body." |
|
202
|
7
|
50
|
66
|
|
|
89
|
unless $params{body} or $params{html} or $params{text}; |
|
|
|
|
66
|
|
|
|
|
|
203
|
|
|
|
|
|
|
|
|
204
|
|
|
|
|
|
|
# if cc and/or bcc are provided, validate them |
|
205
|
6
|
100
|
|
|
|
14
|
if ($params{cc}) { |
|
206
|
1
|
|
|
|
|
3
|
$self->_validate_recipients('cc', $params{cc}); |
|
207
|
|
|
|
|
|
|
} |
|
208
|
6
|
50
|
|
|
|
13
|
if ($params{bcc}) { |
|
209
|
0
|
|
|
|
|
0
|
$self->_validate_recipients('bcc', $params{bcc}); |
|
210
|
|
|
|
|
|
|
} |
|
211
|
|
|
|
|
|
|
|
|
212
|
|
|
|
|
|
|
# if reply_to is provided, validate it |
|
213
|
6
|
50
|
|
|
|
13
|
if ($params{reply_to}) { |
|
214
|
|
|
|
|
|
|
croak "You must provide a valid reply-to address, in the format 'address\@domain.tld', or 'Some Name '." |
|
215
|
0
|
0
|
|
|
|
0
|
unless Email::Valid->address($params{reply_to}); |
|
216
|
|
|
|
|
|
|
} |
|
217
|
|
|
|
|
|
|
|
|
218
|
|
|
|
|
|
|
# parse the body param, unless html or text are present |
|
219
|
6
|
50
|
66
|
|
|
26
|
unless ($params{html} || $params{text}) { |
|
220
|
5
|
|
|
|
|
10
|
my $body = delete $params{body}; |
|
221
|
5
|
100
|
66
|
|
|
28
|
if ($body =~ m/^\/i && $body =~ m!\$!i) { |
|
222
|
|
|
|
|
|
|
# this is an HTML message |
|
223
|
2
|
|
|
|
|
4
|
$params{html} = $body; |
|
224
|
|
|
|
|
|
|
} else { |
|
225
|
|
|
|
|
|
|
# this is a test message |
|
226
|
3
|
|
|
|
|
6
|
$params{text} = $body; |
|
227
|
|
|
|
|
|
|
} |
|
228
|
|
|
|
|
|
|
} |
|
229
|
|
|
|
|
|
|
|
|
230
|
|
|
|
|
|
|
# all's well, let's try an send this |
|
231
|
|
|
|
|
|
|
|
|
232
|
|
|
|
|
|
|
# create the message data structure |
|
233
|
|
|
|
|
|
|
my $msg = { |
|
234
|
|
|
|
|
|
|
From => $params{from}, |
|
235
|
|
|
|
|
|
|
To => $params{to}, |
|
236
|
|
|
|
|
|
|
Subject => $params{subject}, |
|
237
|
6
|
|
|
|
|
24
|
}; |
|
238
|
|
|
|
|
|
|
|
|
239
|
6
|
100
|
|
|
|
18
|
$msg->{HtmlBody} = $params{html} if $params{html}; |
|
240
|
6
|
100
|
|
|
|
18
|
$msg->{TextBody} = $params{text} if $params{text}; |
|
241
|
6
|
100
|
|
|
|
13
|
$msg->{Cc} = $params{cc} if $params{cc}; |
|
242
|
6
|
50
|
|
|
|
14
|
$msg->{Bcc} = $params{bcc} if $params{bcc}; |
|
243
|
6
|
50
|
|
|
|
16
|
$msg->{Tag} = $params{tag} if $params{tag}; |
|
244
|
6
|
50
|
|
|
|
12
|
$msg->{ReplyTo} = $params{reply_to} if $params{reply_to}; |
|
245
|
6
|
50
|
|
|
|
9
|
$msg->{TrackOpens} = 1 if $params{track_opens}; |
|
246
|
|
|
|
|
|
|
|
|
247
|
6
|
100
|
66
|
|
|
30
|
if ($params{attachments} && ref $params{attachments} eq 'ARRAY') { |
|
248
|
|
|
|
|
|
|
# for every file, we need to determine its MIME type and |
|
249
|
|
|
|
|
|
|
# create a base64 representation of its content |
|
250
|
1
|
|
|
|
|
1
|
foreach (@{$params{attachments}}) { |
|
|
1
|
|
|
|
|
4
|
|
|
251
|
2
|
|
|
|
|
358
|
my ($buf, $content); |
|
252
|
|
|
|
|
|
|
|
|
253
|
2
|
|
33
|
|
|
85
|
open FILE, $_ |
|
254
|
|
|
|
|
|
|
|| croak "Failed opening attachment $_: $!"; |
|
255
|
|
|
|
|
|
|
|
|
256
|
2
|
|
|
|
|
27
|
while (read FILE, $buf, 60*57) { |
|
257
|
8
|
|
|
|
|
103
|
$content .= encode_base64($buf); |
|
258
|
|
|
|
|
|
|
} |
|
259
|
|
|
|
|
|
|
|
|
260
|
2
|
|
|
|
|
46
|
close FILE; |
|
261
|
|
|
|
|
|
|
|
|
262
|
2
|
|
100
|
|
|
3
|
push(@{$msg->{Attachments} ||= []}, { |
|
|
2
|
|
|
|
|
86
|
|
|
263
|
|
|
|
|
|
|
Name => basename($_), |
|
264
|
|
|
|
|
|
|
ContentType => mimetype($_), |
|
265
|
|
|
|
|
|
|
Content => $content |
|
266
|
|
|
|
|
|
|
}); |
|
267
|
|
|
|
|
|
|
} |
|
268
|
|
|
|
|
|
|
} |
|
269
|
|
|
|
|
|
|
|
|
270
|
|
|
|
|
|
|
# create and send the request |
|
271
|
|
|
|
|
|
|
my $res = $ua->request( |
|
272
|
|
|
|
|
|
|
'POST', |
|
273
|
|
|
|
|
|
|
'http' . ($self->{use_ssl} ? 's' : '') . '://api.postmarkapp.com/email', |
|
274
|
|
|
|
|
|
|
{ |
|
275
|
|
|
|
|
|
|
headers => { |
|
276
|
|
|
|
|
|
|
'Accept' => 'application/json', |
|
277
|
|
|
|
|
|
|
'Content-Type' => 'application/json', |
|
278
|
|
|
|
|
|
|
'X-Postmark-Server-Token' => $self->{token}, |
|
279
|
|
|
|
|
|
|
}, |
|
280
|
6
|
100
|
|
|
|
307
|
content => encode_json($msg), |
|
281
|
|
|
|
|
|
|
} |
|
282
|
|
|
|
|
|
|
); |
|
283
|
|
|
|
|
|
|
|
|
284
|
|
|
|
|
|
|
# analyze the response |
|
285
|
6
|
100
|
|
|
|
1890573
|
if ($res->{success}) { |
|
286
|
|
|
|
|
|
|
# woooooooooooooeeeeeeeeeeee |
|
287
|
5
|
|
|
|
|
173
|
return decode_json($res->{content}); |
|
288
|
|
|
|
|
|
|
} else { |
|
289
|
1
|
50
|
|
|
|
10
|
if ($msg->{Attachments}) { |
|
290
|
0
|
|
|
|
|
0
|
print STDERR $res->{content}; |
|
291
|
|
|
|
|
|
|
} |
|
292
|
1
|
|
|
|
|
6
|
croak "Failed sending message: ".$self->_analyze_response($res); |
|
293
|
|
|
|
|
|
|
} |
|
294
|
|
|
|
|
|
|
} |
|
295
|
|
|
|
|
|
|
|
|
296
|
|
|
|
|
|
|
=head2 spam_score( $raw_email, [ $options ] ) |
|
297
|
|
|
|
|
|
|
|
|
298
|
|
|
|
|
|
|
Use Postmark's SpamAssassin API to determine the spam score of an email |
|
299
|
|
|
|
|
|
|
message. You need to provide the raw email text to this method, with all |
|
300
|
|
|
|
|
|
|
headers intact. If C<$options> is 'long' (the default), this method |
|
301
|
|
|
|
|
|
|
will return a hash-ref with a 'report' key, containing the full |
|
302
|
|
|
|
|
|
|
SpamAssasin report, and a 'score' key, containing the spam score. If |
|
303
|
|
|
|
|
|
|
C<$options> is 'short', only the spam score will be returned (directly, not |
|
304
|
|
|
|
|
|
|
in a hash-ref). |
|
305
|
|
|
|
|
|
|
|
|
306
|
|
|
|
|
|
|
If the API returns an error, this method will croak. |
|
307
|
|
|
|
|
|
|
|
|
308
|
|
|
|
|
|
|
Note that the SpamAssassin API is currently HTTP only, there is no HTTPS |
|
309
|
|
|
|
|
|
|
interface, so the C option to the C method is ignored here. |
|
310
|
|
|
|
|
|
|
|
|
311
|
|
|
|
|
|
|
For more information about this API, go to L. |
|
312
|
|
|
|
|
|
|
|
|
313
|
|
|
|
|
|
|
=cut |
|
314
|
|
|
|
|
|
|
|
|
315
|
|
|
|
|
|
|
sub spam_score { |
|
316
|
2
|
|
|
2
|
1
|
1594
|
my ($self, $raw_email, $options) = @_; |
|
317
|
|
|
|
|
|
|
|
|
318
|
2
|
50
|
|
|
|
9
|
croak 'You must provide the raw email text to spam_score().' |
|
319
|
|
|
|
|
|
|
unless $raw_email; |
|
320
|
|
|
|
|
|
|
|
|
321
|
2
|
|
100
|
|
|
7
|
$options ||= 'long'; |
|
322
|
|
|
|
|
|
|
|
|
323
|
2
|
|
|
|
|
51
|
my $res = $ua->request( |
|
324
|
|
|
|
|
|
|
'POST', |
|
325
|
|
|
|
|
|
|
'http://spamcheck.postmarkapp.com/filter', |
|
326
|
|
|
|
|
|
|
{ |
|
327
|
|
|
|
|
|
|
headers => { |
|
328
|
|
|
|
|
|
|
'Accept' => 'application/json', |
|
329
|
|
|
|
|
|
|
'Content-Type' => 'application/json', |
|
330
|
|
|
|
|
|
|
}, |
|
331
|
|
|
|
|
|
|
content => encode_json({ |
|
332
|
|
|
|
|
|
|
email => $raw_email, |
|
333
|
|
|
|
|
|
|
options => $options, |
|
334
|
|
|
|
|
|
|
}), |
|
335
|
|
|
|
|
|
|
} |
|
336
|
|
|
|
|
|
|
); |
|
337
|
|
|
|
|
|
|
|
|
338
|
|
|
|
|
|
|
# analyze the response |
|
339
|
2
|
50
|
|
|
|
65079998
|
if ($res->{success}) { |
|
340
|
|
|
|
|
|
|
# doesn't mean we have succeeded, an error may have been returned |
|
341
|
2
|
|
|
|
|
41
|
my $ret = decode_json($res->{content}); |
|
342
|
2
|
50
|
|
|
|
47
|
if ($ret->{success}) { |
|
343
|
2
|
100
|
|
|
|
40
|
return $options eq 'long' ? $ret : $ret->{score}; |
|
344
|
|
|
|
|
|
|
} else { |
|
345
|
0
|
|
|
|
|
0
|
croak "Postmark spam score API returned error: ".$ret->{message}; |
|
346
|
|
|
|
|
|
|
} |
|
347
|
|
|
|
|
|
|
} else { |
|
348
|
0
|
|
|
|
|
0
|
croak "Failed determining spam score: $res->{content}"; |
|
349
|
|
|
|
|
|
|
} |
|
350
|
|
|
|
|
|
|
} |
|
351
|
|
|
|
|
|
|
|
|
352
|
|
|
|
|
|
|
################################## |
|
353
|
|
|
|
|
|
|
## INTERNAL METHODS ## |
|
354
|
|
|
|
|
|
|
################################## |
|
355
|
|
|
|
|
|
|
|
|
356
|
|
|
|
|
|
|
sub _validate_recipients { |
|
357
|
8
|
|
|
8
|
|
10
|
my ($self, $field, $param) = @_; |
|
358
|
|
|
|
|
|
|
|
|
359
|
|
|
|
|
|
|
# split all addresses |
|
360
|
8
|
|
|
|
|
24
|
my @ads = split(/, ?/, $param); |
|
361
|
|
|
|
|
|
|
|
|
362
|
|
|
|
|
|
|
# make sure there are no more than twenty |
|
363
|
8
|
50
|
|
|
|
19
|
croak $self->_recipient_error($field) |
|
364
|
|
|
|
|
|
|
if scalar @ads > 20; |
|
365
|
|
|
|
|
|
|
|
|
366
|
|
|
|
|
|
|
# validate them |
|
367
|
8
|
|
|
|
|
16
|
foreach (@ads) { |
|
368
|
11
|
50
|
|
|
|
676
|
croak $self->_recipient_error($field) |
|
369
|
|
|
|
|
|
|
unless Email::Valid->address($_); |
|
370
|
|
|
|
|
|
|
} |
|
371
|
|
|
|
|
|
|
|
|
372
|
|
|
|
|
|
|
# all's well |
|
373
|
8
|
|
|
|
|
1790
|
return 1; |
|
374
|
|
|
|
|
|
|
} |
|
375
|
|
|
|
|
|
|
|
|
376
|
|
|
|
|
|
|
sub _recipient_error { |
|
377
|
0
|
|
|
0
|
|
0
|
my ($self, $field) = @_; |
|
378
|
|
|
|
|
|
|
|
|
379
|
0
|
|
|
|
|
0
|
return "You must provide a valid '$field' address or addresses, in the format 'address\@domain.tld', or 'Some Name '. If you're sending to multiple addresses, separate them with commas. You can send up to 20 maximum addresses."; |
|
380
|
|
|
|
|
|
|
} |
|
381
|
|
|
|
|
|
|
|
|
382
|
|
|
|
|
|
|
sub _analyze_response { |
|
383
|
1
|
|
|
1
|
|
3
|
my ($self, $res) = @_; |
|
384
|
|
|
|
|
|
|
|
|
385
|
|
|
|
|
|
|
return $res->{status} == 401 ? 'Missing or incorrect API Key header.' : |
|
386
|
|
|
|
|
|
|
$res->{status} == 422 ? $self->_extract_error($res->{content}) : |
|
387
|
1
|
0
|
|
|
|
204
|
$res->{status} == 500 ? 'Postmark service error. The service might be down.' : |
|
|
|
0
|
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
388
|
|
|
|
|
|
|
"Unknown HTTP error code $res->{status}."; |
|
389
|
|
|
|
|
|
|
} |
|
390
|
|
|
|
|
|
|
|
|
391
|
|
|
|
|
|
|
sub _extract_error { |
|
392
|
0
|
|
|
0
|
|
|
my ($self, $content) = @_; |
|
393
|
|
|
|
|
|
|
|
|
394
|
0
|
|
|
|
|
|
my $msg = decode_json($content); |
|
395
|
|
|
|
|
|
|
|
|
396
|
0
|
|
|
|
|
|
my %errors = ( |
|
397
|
|
|
|
|
|
|
10 => 'Bad or missing API token', |
|
398
|
|
|
|
|
|
|
300 => 'Invalid email request', |
|
399
|
|
|
|
|
|
|
400 => 'Sender signature not found', |
|
400
|
|
|
|
|
|
|
401 => 'Sender signature not confirmed', |
|
401
|
|
|
|
|
|
|
402 => 'Invalid JSON', |
|
402
|
|
|
|
|
|
|
403 => 'Incompatible JSON', |
|
403
|
|
|
|
|
|
|
405 => 'Not allowed to send', |
|
404
|
|
|
|
|
|
|
406 => 'Inactive recipient', |
|
405
|
|
|
|
|
|
|
409 => 'JSON required', |
|
406
|
|
|
|
|
|
|
410 => 'Too many batch messages', |
|
407
|
|
|
|
|
|
|
411 => 'Forbidden attachment type' |
|
408
|
|
|
|
|
|
|
); |
|
409
|
|
|
|
|
|
|
|
|
410
|
0
|
|
0
|
|
|
|
my $code_msg = $errors{$msg->{ErrorCode}} || "Unknown Postmark error code $msg->{ErrorCode}"; |
|
411
|
|
|
|
|
|
|
|
|
412
|
0
|
|
|
|
|
|
return $code_msg . ': '. $msg->{Message}; |
|
413
|
|
|
|
|
|
|
} |
|
414
|
|
|
|
|
|
|
|
|
415
|
|
|
|
|
|
|
=head1 DIAGNOSTICS |
|
416
|
|
|
|
|
|
|
|
|
417
|
|
|
|
|
|
|
The following exceptions are thrown by this module: |
|
418
|
|
|
|
|
|
|
|
|
419
|
|
|
|
|
|
|
=over |
|
420
|
|
|
|
|
|
|
|
|
421
|
|
|
|
|
|
|
=item C<< "You have not provided a Postmark API token, you cannot send emails" >> |
|
422
|
|
|
|
|
|
|
|
|
423
|
|
|
|
|
|
|
This means you haven't provided the C subroutine your Postmark API token. |
|
424
|
|
|
|
|
|
|
Using the Postmark API requires an API token, received when registering to their |
|
425
|
|
|
|
|
|
|
service via their website. |
|
426
|
|
|
|
|
|
|
|
|
427
|
|
|
|
|
|
|
=item C<< "You must provide a mail subject." >> |
|
428
|
|
|
|
|
|
|
|
|
429
|
|
|
|
|
|
|
This error means you haven't given the C method a subject for your email |
|
430
|
|
|
|
|
|
|
message. Messages sent with this module must have a subject. |
|
431
|
|
|
|
|
|
|
|
|
432
|
|
|
|
|
|
|
=item C<< "You must provide a mail body." >> |
|
433
|
|
|
|
|
|
|
|
|
434
|
|
|
|
|
|
|
This error means you haven't given the C method a body for your email |
|
435
|
|
|
|
|
|
|
message. Messages sent with this module must have content. |
|
436
|
|
|
|
|
|
|
|
|
437
|
|
|
|
|
|
|
=item C<< "You must provide a valid 'from' address in the format 'address\@domain.tld', or 'Your Name '." >> |
|
438
|
|
|
|
|
|
|
|
|
439
|
|
|
|
|
|
|
This error means the address (or one of the addresses) you're trying to send |
|
440
|
|
|
|
|
|
|
an email to with the C method is not a valid email address (in the sense |
|
441
|
|
|
|
|
|
|
that it I be an email address, not in the sense that the email address does not |
|
442
|
|
|
|
|
|
|
exist (For example, "asdf" is not a valid email address). |
|
443
|
|
|
|
|
|
|
|
|
444
|
|
|
|
|
|
|
=item C<< "You must provide a valid reply-to address, in the format 'address\@domain.tld', or 'Some Name '." >> |
|
445
|
|
|
|
|
|
|
|
|
446
|
|
|
|
|
|
|
This error, when providing the C parameter to the C method, |
|
447
|
|
|
|
|
|
|
means the C value is not a valid email address. |
|
448
|
|
|
|
|
|
|
|
|
449
|
|
|
|
|
|
|
=item C<< "You must provide a valid '%s' address or addresses, in the format 'address\@domain.tld', or 'Some Name '. If you're sending to multiple addresses, separate them with commas. You can send up to 20 maximum addresses." >> |
|
450
|
|
|
|
|
|
|
|
|
451
|
|
|
|
|
|
|
Like the above two error messages, but for other email fields such as C and C. |
|
452
|
|
|
|
|
|
|
|
|
453
|
|
|
|
|
|
|
=item C<< "Failed sending message: %s" >> |
|
454
|
|
|
|
|
|
|
|
|
455
|
|
|
|
|
|
|
This error is thrown when sending an email fails. The error message should |
|
456
|
|
|
|
|
|
|
include the actual reason for the failure. Usually, the error is returned by |
|
457
|
|
|
|
|
|
|
the Postmark API. For a list of errors returned by Postmark and their meaning, |
|
458
|
|
|
|
|
|
|
take a look at L. |
|
459
|
|
|
|
|
|
|
|
|
460
|
|
|
|
|
|
|
=item C<< "Unknown Postmark error code %s" >> |
|
461
|
|
|
|
|
|
|
|
|
462
|
|
|
|
|
|
|
This means Postmark returned an error code that this module does not |
|
463
|
|
|
|
|
|
|
recognize. The error message should include the error code. If you find |
|
464
|
|
|
|
|
|
|
that error code in L, |
|
465
|
|
|
|
|
|
|
it probably means this is a new error code this module does not know about yet, |
|
466
|
|
|
|
|
|
|
so please open an appropriate bug report. |
|
467
|
|
|
|
|
|
|
|
|
468
|
|
|
|
|
|
|
=item C<< "Unknown HTTP error code %s." >> |
|
469
|
|
|
|
|
|
|
|
|
470
|
|
|
|
|
|
|
This means the Postmark API returned an unexpected HTTP status code. The error |
|
471
|
|
|
|
|
|
|
message should include the status code returned. |
|
472
|
|
|
|
|
|
|
|
|
473
|
|
|
|
|
|
|
=item C<< "Failed opening attachment %s: %s" >> |
|
474
|
|
|
|
|
|
|
|
|
475
|
|
|
|
|
|
|
This error means C was unable to open a file attachment you have |
|
476
|
|
|
|
|
|
|
supplied for reading. This could be due to permission problem or the file not |
|
477
|
|
|
|
|
|
|
existing. The full error message should detail the exact cause. |
|
478
|
|
|
|
|
|
|
|
|
479
|
|
|
|
|
|
|
=item C<< "You must provide the raw email text to spam_score()." >> |
|
480
|
|
|
|
|
|
|
|
|
481
|
|
|
|
|
|
|
This error means you haven't passed the C method the |
|
482
|
|
|
|
|
|
|
requried raw email text. |
|
483
|
|
|
|
|
|
|
|
|
484
|
|
|
|
|
|
|
=item C<< "Postmark spam score API returned error: %s" >> |
|
485
|
|
|
|
|
|
|
|
|
486
|
|
|
|
|
|
|
This error means the spam score API failed parsing your raw email |
|
487
|
|
|
|
|
|
|
text. The error message should include the actual reason for the failure. |
|
488
|
|
|
|
|
|
|
This would be an I API error. I API errors will |
|
489
|
|
|
|
|
|
|
be thrown with the next error message. |
|
490
|
|
|
|
|
|
|
|
|
491
|
|
|
|
|
|
|
=item C<< "Failed determining spam score: %s" >> |
|
492
|
|
|
|
|
|
|
|
|
493
|
|
|
|
|
|
|
This error means the spam score API returned an HTTP error. The error |
|
494
|
|
|
|
|
|
|
message should include the actual error message returned. |
|
495
|
|
|
|
|
|
|
|
|
496
|
|
|
|
|
|
|
=back |
|
497
|
|
|
|
|
|
|
|
|
498
|
|
|
|
|
|
|
=head1 CONFIGURATION AND ENVIRONMENT |
|
499
|
|
|
|
|
|
|
|
|
500
|
|
|
|
|
|
|
C requires no configuration files or environment variables. |
|
501
|
|
|
|
|
|
|
|
|
502
|
|
|
|
|
|
|
=head1 DEPENDENCIES |
|
503
|
|
|
|
|
|
|
|
|
504
|
|
|
|
|
|
|
C B on the following CPAN modules: |
|
505
|
|
|
|
|
|
|
|
|
506
|
|
|
|
|
|
|
=over |
|
507
|
|
|
|
|
|
|
|
|
508
|
|
|
|
|
|
|
=item * L |
|
509
|
|
|
|
|
|
|
|
|
510
|
|
|
|
|
|
|
=item * L |
|
511
|
|
|
|
|
|
|
|
|
512
|
|
|
|
|
|
|
=item * L |
|
513
|
|
|
|
|
|
|
|
|
514
|
|
|
|
|
|
|
=item * L |
|
515
|
|
|
|
|
|
|
|
|
516
|
|
|
|
|
|
|
=item * L |
|
517
|
|
|
|
|
|
|
|
|
518
|
|
|
|
|
|
|
=item * L |
|
519
|
|
|
|
|
|
|
|
|
520
|
|
|
|
|
|
|
=back |
|
521
|
|
|
|
|
|
|
|
|
522
|
|
|
|
|
|
|
C recommends L for parsing JSON (the Postmark API |
|
523
|
|
|
|
|
|
|
is JSON based). If installed, L will automatically load L |
|
524
|
|
|
|
|
|
|
or L. For SSL support, L and L will also be |
|
525
|
|
|
|
|
|
|
needed. |
|
526
|
|
|
|
|
|
|
|
|
527
|
|
|
|
|
|
|
=head1 INCOMPATIBILITIES WITH OTHER MODULES |
|
528
|
|
|
|
|
|
|
|
|
529
|
|
|
|
|
|
|
None reported. |
|
530
|
|
|
|
|
|
|
|
|
531
|
|
|
|
|
|
|
=head1 BUGS AND LIMITATIONS |
|
532
|
|
|
|
|
|
|
|
|
533
|
|
|
|
|
|
|
No bugs have been reported. |
|
534
|
|
|
|
|
|
|
|
|
535
|
|
|
|
|
|
|
Please report any bugs or feature requests to |
|
536
|
|
|
|
|
|
|
C, or through the web interface at |
|
537
|
|
|
|
|
|
|
L. |
|
538
|
|
|
|
|
|
|
|
|
539
|
|
|
|
|
|
|
=head1 AUTHOR |
|
540
|
|
|
|
|
|
|
|
|
541
|
|
|
|
|
|
|
Ido Perlmuter |
|
542
|
|
|
|
|
|
|
|
|
543
|
|
|
|
|
|
|
With help from: Casimir Loeber. |
|
544
|
|
|
|
|
|
|
|
|
545
|
|
|
|
|
|
|
=head1 LICENSE AND COPYRIGHT |
|
546
|
|
|
|
|
|
|
|
|
547
|
|
|
|
|
|
|
Copyright 2017 Ido Perlmuter |
|
548
|
|
|
|
|
|
|
|
|
549
|
|
|
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License"); |
|
550
|
|
|
|
|
|
|
you may not use this file except in compliance with the License. |
|
551
|
|
|
|
|
|
|
You may obtain a copy of the License at |
|
552
|
|
|
|
|
|
|
|
|
553
|
|
|
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0 |
|
554
|
|
|
|
|
|
|
|
|
555
|
|
|
|
|
|
|
Unless required by applicable law or agreed to in writing, software |
|
556
|
|
|
|
|
|
|
distributed under the License is distributed on an "AS IS" BASIS, |
|
557
|
|
|
|
|
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
|
558
|
|
|
|
|
|
|
See the License for the specific language governing permissions and |
|
559
|
|
|
|
|
|
|
limitations under the License. |
|
560
|
|
|
|
|
|
|
|
|
561
|
|
|
|
|
|
|
=cut |
|
562
|
|
|
|
|
|
|
|
|
563
|
|
|
|
|
|
|
1; |
|
564
|
|
|
|
|
|
|
__END__ |