line |
stmt |
bran |
cond |
sub |
pod |
time |
code |
1
|
|
|
|
|
|
|
package Mojolicious::Plugin::PayPal; |
2
|
|
|
|
|
|
|
|
3
|
|
|
|
|
|
|
=head1 NAME |
4
|
|
|
|
|
|
|
|
5
|
|
|
|
|
|
|
Mojolicious::Plugin::PayPal - Make payments using PayPal |
6
|
|
|
|
|
|
|
|
7
|
|
|
|
|
|
|
=head1 VERSION |
8
|
|
|
|
|
|
|
|
9
|
|
|
|
|
|
|
0.06 |
10
|
|
|
|
|
|
|
|
11
|
|
|
|
|
|
|
=head1 DESCRIPTION |
12
|
|
|
|
|
|
|
|
13
|
|
|
|
|
|
|
L is a plugin for the L web |
14
|
|
|
|
|
|
|
framework which allow you to do payments using L. |
15
|
|
|
|
|
|
|
|
16
|
|
|
|
|
|
|
This module is EXPERIMENTAL. The API can change at any time. Let me know |
17
|
|
|
|
|
|
|
if you are using it. |
18
|
|
|
|
|
|
|
|
19
|
|
|
|
|
|
|
See also L. |
20
|
|
|
|
|
|
|
|
21
|
|
|
|
|
|
|
=head1 SYNOPSIS |
22
|
|
|
|
|
|
|
|
23
|
|
|
|
|
|
|
use Mojolicious::Lite; |
24
|
|
|
|
|
|
|
|
25
|
|
|
|
|
|
|
plugin PayPal => { |
26
|
|
|
|
|
|
|
secret => '...', |
27
|
|
|
|
|
|
|
client_id => '...', |
28
|
|
|
|
|
|
|
}; |
29
|
|
|
|
|
|
|
|
30
|
|
|
|
|
|
|
# register a payment and send the visitor to PayPal payment terminal |
31
|
|
|
|
|
|
|
post '/checkout' => sub { |
32
|
|
|
|
|
|
|
my $self = shift->render_later; |
33
|
|
|
|
|
|
|
my %payment = ( |
34
|
|
|
|
|
|
|
amount => $self->param('amount'), |
35
|
|
|
|
|
|
|
description => 'Some description', |
36
|
|
|
|
|
|
|
); |
37
|
|
|
|
|
|
|
|
38
|
|
|
|
|
|
|
Mojo::IOLoop->delay( |
39
|
|
|
|
|
|
|
sub { |
40
|
|
|
|
|
|
|
my ($delay) = @_; |
41
|
|
|
|
|
|
|
$self->paypal(register => \%payment, $delay->begin); |
42
|
|
|
|
|
|
|
}, |
43
|
|
|
|
|
|
|
sub { |
44
|
|
|
|
|
|
|
my ($delay, $res) = @_; |
45
|
|
|
|
|
|
|
return $self->render(text => "Ooops!", status => $res->code) unless $res->code == 302; |
46
|
|
|
|
|
|
|
# store $res->param('transaction_id'); |
47
|
|
|
|
|
|
|
$self->redirect_to($res->headers->location); |
48
|
|
|
|
|
|
|
}, |
49
|
|
|
|
|
|
|
); |
50
|
|
|
|
|
|
|
}; |
51
|
|
|
|
|
|
|
|
52
|
|
|
|
|
|
|
# after redirected back from PayPal payment terminal |
53
|
|
|
|
|
|
|
get '/checkout' => sub { |
54
|
|
|
|
|
|
|
my $self = shift->render_later; |
55
|
|
|
|
|
|
|
|
56
|
|
|
|
|
|
|
Mojo::IOLoop->delay( |
57
|
|
|
|
|
|
|
sub { |
58
|
|
|
|
|
|
|
my ($delay) = @_; |
59
|
|
|
|
|
|
|
$self->paypal(process => {}, $delay->begin); |
60
|
|
|
|
|
|
|
}, |
61
|
|
|
|
|
|
|
sub { |
62
|
|
|
|
|
|
|
my ($delay, $res) = @_; |
63
|
|
|
|
|
|
|
return $self->render(text => $res->param("message"), status => $res->code) unless $res->code == 200; |
64
|
|
|
|
|
|
|
return $self->render(text => "yay!"); |
65
|
|
|
|
|
|
|
}, |
66
|
|
|
|
|
|
|
); |
67
|
|
|
|
|
|
|
}; |
68
|
|
|
|
|
|
|
|
69
|
|
|
|
|
|
|
|
70
|
|
|
|
|
|
|
=head2 Transaction ID mapper |
71
|
|
|
|
|
|
|
|
72
|
|
|
|
|
|
|
You should provide a L. Here is an example code on how to do that: |
73
|
|
|
|
|
|
|
|
74
|
|
|
|
|
|
|
$app->paypal->transaction_id_mapper(sub { |
75
|
|
|
|
|
|
|
my ($self, $token, $transaction_id, $cb) = @_; |
76
|
|
|
|
|
|
|
|
77
|
|
|
|
|
|
|
if($transaction_id) { |
78
|
|
|
|
|
|
|
eval { My::DB->store_transaction_id($token => $transaction_id); }; |
79
|
|
|
|
|
|
|
$self->$cb($@, $transaction_id); |
80
|
|
|
|
|
|
|
} |
81
|
|
|
|
|
|
|
else { |
82
|
|
|
|
|
|
|
my $transaction_id = eval { My::DB->get_transaction_id($token)); }; |
83
|
|
|
|
|
|
|
$self->$cb($@, $transaction_id); |
84
|
|
|
|
|
|
|
} |
85
|
|
|
|
|
|
|
}); |
86
|
|
|
|
|
|
|
|
87
|
|
|
|
|
|
|
=cut |
88
|
|
|
|
|
|
|
|
89
|
2
|
|
|
2
|
|
1815
|
use Mojo::Base 'Mojolicious::Plugin'; |
|
2
|
|
|
|
|
3
|
|
|
2
|
|
|
|
|
15
|
|
90
|
2
|
|
|
2
|
|
411
|
use Mojo::JSON 'j'; |
|
2
|
|
|
|
|
4
|
|
|
2
|
|
|
|
|
101
|
|
91
|
2
|
|
|
2
|
|
10
|
use Mojo::UserAgent; |
|
2
|
|
|
|
|
2
|
|
|
2
|
|
|
|
|
24
|
|
92
|
2
|
|
50
|
2
|
|
79
|
use constant DEBUG => $ENV{MOJO_PAYPAL_DEBUG} || 0; |
|
2
|
|
|
|
|
4
|
|
|
2
|
|
|
|
|
6484
|
|
93
|
|
|
|
|
|
|
|
94
|
|
|
|
|
|
|
our $VERSION = '0.06'; |
95
|
|
|
|
|
|
|
|
96
|
|
|
|
|
|
|
=head1 ATTRIBUTES |
97
|
|
|
|
|
|
|
|
98
|
|
|
|
|
|
|
=head2 base_url |
99
|
|
|
|
|
|
|
|
100
|
|
|
|
|
|
|
$str = $self->base_url; |
101
|
|
|
|
|
|
|
|
102
|
|
|
|
|
|
|
This is the location to PayPal payment solution. Will be set to |
103
|
|
|
|
|
|
|
L if the mojolicious application mode is |
104
|
|
|
|
|
|
|
"production" or L. |
105
|
|
|
|
|
|
|
|
106
|
|
|
|
|
|
|
=head2 client_id |
107
|
|
|
|
|
|
|
|
108
|
|
|
|
|
|
|
$str = $self->client_id; |
109
|
|
|
|
|
|
|
|
110
|
|
|
|
|
|
|
The value used as username when fetching the the access token. |
111
|
|
|
|
|
|
|
This can be found in "Applications tab" in the PayPal Developer site. |
112
|
|
|
|
|
|
|
|
113
|
|
|
|
|
|
|
=head2 currency_code |
114
|
|
|
|
|
|
|
|
115
|
|
|
|
|
|
|
$str = $self->currency_code; |
116
|
|
|
|
|
|
|
|
117
|
|
|
|
|
|
|
The currency code. Default is "USD". |
118
|
|
|
|
|
|
|
|
119
|
|
|
|
|
|
|
=head2 transaction_id_mapper |
120
|
|
|
|
|
|
|
|
121
|
|
|
|
|
|
|
$code = $self->transaction_id_mapper; |
122
|
|
|
|
|
|
|
|
123
|
|
|
|
|
|
|
Holds a code used to find the transaction ID, after user has been redirected |
124
|
|
|
|
|
|
|
back from PayPal terminal page. |
125
|
|
|
|
|
|
|
|
126
|
|
|
|
|
|
|
NOTE! The default callback provided by this module does not scale and will |
127
|
|
|
|
|
|
|
not work in a multi-process environment, such as running under C |
128
|
|
|
|
|
|
|
or using a load balancer. You should therefor provide your own backend |
129
|
|
|
|
|
|
|
solution. See L for example code. |
130
|
|
|
|
|
|
|
|
131
|
|
|
|
|
|
|
=head2 secret |
132
|
|
|
|
|
|
|
|
133
|
|
|
|
|
|
|
$str = $self->secret; |
134
|
|
|
|
|
|
|
|
135
|
|
|
|
|
|
|
The value used as password when fetching the the access token. |
136
|
|
|
|
|
|
|
This can be found in "Applications tab" in the PayPal Developer site. |
137
|
|
|
|
|
|
|
|
138
|
|
|
|
|
|
|
=cut |
139
|
|
|
|
|
|
|
|
140
|
|
|
|
|
|
|
has base_url => 'https://api.sandbox.paypal.com'; |
141
|
|
|
|
|
|
|
has client_id => 'dummy_client'; |
142
|
|
|
|
|
|
|
has currency_code => 'USD'; |
143
|
|
|
|
|
|
|
has transaction_id_mapper => undef; |
144
|
|
|
|
|
|
|
has secret => 'dummy_secret'; |
145
|
|
|
|
|
|
|
has _ua => sub { Mojo::UserAgent->new; }; |
146
|
|
|
|
|
|
|
|
147
|
|
|
|
|
|
|
=head1 HELPERS |
148
|
|
|
|
|
|
|
|
149
|
|
|
|
|
|
|
=head2 paypal |
150
|
|
|
|
|
|
|
|
151
|
|
|
|
|
|
|
$self = $c->paypal; |
152
|
|
|
|
|
|
|
$c = $c->paypal($method => @args); |
153
|
|
|
|
|
|
|
|
154
|
|
|
|
|
|
|
Returns this instance unless any args have been given or calls one of the |
155
|
|
|
|
|
|
|
available L instead. C<$method> need to be without "_payment" at |
156
|
|
|
|
|
|
|
the end. Example: |
157
|
|
|
|
|
|
|
|
158
|
|
|
|
|
|
|
$c->paypal(register => { ... }, sub { |
159
|
|
|
|
|
|
|
my ($c, $res) = @_; |
160
|
|
|
|
|
|
|
# ... |
161
|
|
|
|
|
|
|
}); |
162
|
|
|
|
|
|
|
|
163
|
|
|
|
|
|
|
=head1 METHODS |
164
|
|
|
|
|
|
|
|
165
|
|
|
|
|
|
|
=head2 process_payment |
166
|
|
|
|
|
|
|
|
167
|
|
|
|
|
|
|
$self = $self->process_payment( |
168
|
|
|
|
|
|
|
$c, |
169
|
|
|
|
|
|
|
{ |
170
|
|
|
|
|
|
|
token => $str, # default to $c->param("token") |
171
|
|
|
|
|
|
|
payer_id => $str, # default to $c->param("PayerID") |
172
|
|
|
|
|
|
|
}, |
173
|
|
|
|
|
|
|
sub { |
174
|
|
|
|
|
|
|
my ($self, $res) = @_; |
175
|
|
|
|
|
|
|
}, |
176
|
|
|
|
|
|
|
); |
177
|
|
|
|
|
|
|
|
178
|
|
|
|
|
|
|
This is used to process the payment after a user has been redirected back |
179
|
|
|
|
|
|
|
from the PayPal terminal. |
180
|
|
|
|
|
|
|
|
181
|
|
|
|
|
|
|
See L |
182
|
|
|
|
|
|
|
for details. |
183
|
|
|
|
|
|
|
|
184
|
|
|
|
|
|
|
=cut |
185
|
|
|
|
|
|
|
|
186
|
|
|
|
|
|
|
sub process_payment { |
187
|
1
|
|
|
1
|
1
|
3
|
my ($self, $c, $args, $cb) = @_; |
188
|
1
|
|
|
|
|
3
|
my %body; |
189
|
|
|
|
|
|
|
|
190
|
1
|
50
|
33
|
|
|
12
|
$args->{cancel} //= $c->param('return_url') ? 0 : 1; |
191
|
1
|
50
|
33
|
|
|
427
|
$args->{token} ||= $c->param('token') or return $self->$cb($self->_error('token missing in input')); |
192
|
1
|
50
|
33
|
|
|
71
|
$args->{payer_id} ||= $c->param('PayerID') or return $self->$cb($self->_error('PayerID missing in input')); |
193
|
|
|
|
|
|
|
|
194
|
1
|
|
|
|
|
66
|
%body = ( payer_id => $args->{payer_id} ); |
195
|
|
|
|
|
|
|
|
196
|
|
|
|
|
|
|
Mojo::IOLoop->delay( |
197
|
|
|
|
|
|
|
sub { |
198
|
1
|
|
|
1
|
|
47
|
my ($delay) = @_; |
199
|
1
|
|
|
|
|
26
|
$self->transaction_id_mapper->($self, $args->{token}, undef, $delay->begin); |
200
|
|
|
|
|
|
|
}, |
201
|
|
|
|
|
|
|
sub { |
202
|
1
|
|
|
1
|
|
119
|
my ($delay, $err, $transaction_id) = @_; |
203
|
1
|
50
|
|
|
|
5
|
return $self->$cb($self->_error($err)) if $err; |
204
|
1
|
|
|
|
|
7
|
my $url = $self->_url("/v1/payments/payment/$transaction_id/execute"); |
205
|
1
|
|
|
|
|
6
|
$delay->pass($transaction_id); |
206
|
1
|
|
|
|
|
29
|
$self->_make_request_with_token(post => $url, j(\%body), $delay->begin); |
207
|
|
|
|
|
|
|
}, |
208
|
|
|
|
|
|
|
sub { |
209
|
1
|
|
|
1
|
|
28
|
my ($delay, $transaction_id, $tx) = @_; |
210
|
1
|
|
|
|
|
4
|
my $res = Mojolicious::Plugin::NetsPayment::Res->new($tx->res); |
211
|
|
|
|
|
|
|
|
212
|
1
|
50
|
|
|
|
27
|
$res->code(0) unless $res->code; |
213
|
1
|
|
|
|
|
11
|
$res->param(transaction_id => $transaction_id); |
214
|
|
|
|
|
|
|
|
215
|
1
|
50
|
|
|
|
121
|
if ($args->{cancel}) { |
216
|
0
|
|
|
|
|
0
|
$res->param(message => 'Payment cancelled.'); |
217
|
0
|
|
|
|
|
0
|
$res->param(source => $self->base_url); |
218
|
0
|
|
|
|
|
0
|
$res->code(205); |
219
|
0
|
|
|
|
|
0
|
return $self->$cb($res); |
220
|
|
|
|
|
|
|
} |
221
|
|
|
|
|
|
|
|
222
|
1
|
|
|
|
|
2
|
local $@; |
223
|
|
|
|
|
|
|
eval { |
224
|
1
|
|
|
|
|
5
|
my $json = $res->json; |
225
|
1
|
|
|
|
|
1840
|
my $token; |
226
|
|
|
|
|
|
|
|
227
|
1
|
50
|
|
|
|
7
|
$json->{id} or die 'No transaction ID in response from PayPal'; |
228
|
1
|
50
|
|
|
|
7
|
$json->{state} eq 'approved' or die $json->{state}; |
229
|
|
|
|
|
|
|
|
230
|
1
|
50
|
|
|
|
2
|
while(my($key, $value) = each %{ $json->{payer}{payer_info} || {} }) { |
|
5
|
|
|
|
|
205
|
|
231
|
4
|
|
|
|
|
14
|
$res->param("payer_$key" => $value); |
232
|
|
|
|
|
|
|
} |
233
|
|
|
|
|
|
|
|
234
|
1
|
|
|
|
|
5
|
$res->param(payer_id => $args->{payer_id}); |
235
|
1
|
|
|
|
|
49
|
$res->param(state => $json->{state}); |
236
|
1
|
|
|
|
|
48
|
$res->param(transaction_id => $json->{id}); |
237
|
1
|
|
|
|
|
51
|
$res->code(200); |
238
|
1
|
|
|
|
|
8
|
$self->$cb($res); |
239
|
1
|
|
|
|
|
1240
|
1; |
240
|
1
|
50
|
|
|
|
2
|
} or do { |
241
|
0
|
|
|
|
|
0
|
warn "[MOJO_PAYPAL] ! $@" if DEBUG; |
242
|
0
|
|
|
|
|
0
|
$self->$cb($self->_extract_error($res, $@)); |
243
|
|
|
|
|
|
|
}; |
244
|
|
|
|
|
|
|
}, |
245
|
1
|
|
|
|
|
16
|
); |
246
|
|
|
|
|
|
|
|
247
|
1
|
|
|
|
|
95
|
$self; |
248
|
|
|
|
|
|
|
} |
249
|
|
|
|
|
|
|
|
250
|
|
|
|
|
|
|
=head2 register_payment |
251
|
|
|
|
|
|
|
|
252
|
|
|
|
|
|
|
$self = $self->register_payment( |
253
|
|
|
|
|
|
|
$c, |
254
|
|
|
|
|
|
|
{ |
255
|
|
|
|
|
|
|
amount => $num, # 99.90, not 9990 |
256
|
|
|
|
|
|
|
redirect_url => $str, # default to current request URL |
257
|
|
|
|
|
|
|
# ... |
258
|
|
|
|
|
|
|
}, |
259
|
|
|
|
|
|
|
sub { |
260
|
|
|
|
|
|
|
my ($self, $res) = @_; |
261
|
|
|
|
|
|
|
}, |
262
|
|
|
|
|
|
|
); |
263
|
|
|
|
|
|
|
|
264
|
|
|
|
|
|
|
The L method is used to send the required payment details |
265
|
|
|
|
|
|
|
to PayPal which will later be approved by the user after being redirected |
266
|
|
|
|
|
|
|
to the PayPal terminal page. |
267
|
|
|
|
|
|
|
|
268
|
|
|
|
|
|
|
Useful C<$res> values: |
269
|
|
|
|
|
|
|
|
270
|
|
|
|
|
|
|
=over 4 |
271
|
|
|
|
|
|
|
|
272
|
|
|
|
|
|
|
=item * $res->code |
273
|
|
|
|
|
|
|
|
274
|
|
|
|
|
|
|
Set to 302 on success. |
275
|
|
|
|
|
|
|
|
276
|
|
|
|
|
|
|
=item * $res->param("transaction_id") |
277
|
|
|
|
|
|
|
|
278
|
|
|
|
|
|
|
Only set on success. An ID identifying this transaction. Generated by PayPal. |
279
|
|
|
|
|
|
|
|
280
|
|
|
|
|
|
|
=item * $res->headers->location |
281
|
|
|
|
|
|
|
|
282
|
|
|
|
|
|
|
Only set on success. This holds a URL to the PayPal terminal page, which |
283
|
|
|
|
|
|
|
you will redirect the user to after storing the transaction ID and other |
284
|
|
|
|
|
|
|
customer related details. |
285
|
|
|
|
|
|
|
|
286
|
|
|
|
|
|
|
=back |
287
|
|
|
|
|
|
|
|
288
|
|
|
|
|
|
|
=cut |
289
|
|
|
|
|
|
|
|
290
|
|
|
|
|
|
|
sub register_payment { |
291
|
2
|
|
|
2
|
1
|
5
|
my ($self, $c, $args, $cb) = @_; |
292
|
2
|
|
|
|
|
9
|
my $register_url = $self->_url('/v1/payments/payment'); |
293
|
2
|
|
|
|
|
9
|
my $redirect_url = Mojo::URL->new($args->{redirect_url} = $c->req->url->to_abs); |
294
|
2
|
|
|
|
|
1055
|
my %body; |
295
|
|
|
|
|
|
|
|
296
|
2
|
100
|
|
|
|
13
|
$args->{amount} or return $self->$cb($self->_error('amount missing in input')); |
297
|
|
|
|
|
|
|
|
298
|
|
|
|
|
|
|
%body = ( |
299
|
|
|
|
|
|
|
intent => 'sale', |
300
|
|
|
|
|
|
|
redirect_urls => { |
301
|
|
|
|
|
|
|
return_url => $redirect_url->query(return_url => 1)->to_abs, |
302
|
|
|
|
|
|
|
cancel_url => $redirect_url->to_abs, |
303
|
|
|
|
|
|
|
}, |
304
|
|
|
|
|
|
|
payer => { |
305
|
|
|
|
|
|
|
payment_method => 'paypal', |
306
|
|
|
|
|
|
|
}, |
307
|
|
|
|
|
|
|
transactions => [ |
308
|
|
|
|
|
|
|
{ |
309
|
|
|
|
|
|
|
description => $args->{description} || '', |
310
|
|
|
|
|
|
|
amount => { |
311
|
|
|
|
|
|
|
total => $args->{amount}, |
312
|
1
|
|
50
|
|
|
6
|
currency => $args->{currency_code} || $self->currency_code, |
|
|
|
33
|
|
|
|
|
313
|
|
|
|
|
|
|
}, |
314
|
|
|
|
|
|
|
}, |
315
|
|
|
|
|
|
|
], |
316
|
|
|
|
|
|
|
); |
317
|
|
|
|
|
|
|
|
318
|
|
|
|
|
|
|
Mojo::IOLoop->delay( |
319
|
|
|
|
|
|
|
sub { |
320
|
1
|
|
|
1
|
|
50
|
my ($delay) = @_; |
321
|
1
|
|
|
|
|
6
|
$self->_make_request_with_token(post => $register_url, j(\%body), $delay->begin); |
322
|
|
|
|
|
|
|
}, |
323
|
|
|
|
|
|
|
sub { |
324
|
1
|
|
|
1
|
|
26
|
my ($delay, $tx) = @_; |
325
|
1
|
|
|
|
|
5
|
my $res = Mojolicious::Plugin::NetsPayment::Res->new($tx->res); |
326
|
|
|
|
|
|
|
|
327
|
1
|
50
|
|
|
|
28
|
$res->code(0) unless $res->code; |
328
|
|
|
|
|
|
|
|
329
|
1
|
|
|
|
|
9
|
local $@; |
330
|
|
|
|
|
|
|
eval { |
331
|
1
|
|
|
|
|
9
|
my $json = $res->json; |
332
|
1
|
|
|
|
|
1134
|
my $token; |
333
|
|
|
|
|
|
|
|
334
|
1
|
50
|
|
|
|
7
|
$json->{id} or die 'No transaction ID in response from PayPal'; |
335
|
1
|
50
|
|
|
|
6
|
$json->{state} eq 'created' or die $json->{state}; |
336
|
|
|
|
|
|
|
|
337
|
1
|
|
|
|
|
4
|
for my $link (@{ $json->{links} }) { |
|
1
|
|
|
|
|
5
|
|
338
|
3
|
|
|
|
|
206
|
my $key = "$link->{rel}_url"; |
339
|
3
|
|
|
|
|
15
|
$key =~ s!_url_url$!_url!; |
340
|
3
|
|
|
|
|
12
|
$res->param($key => $link->{href}); |
341
|
|
|
|
|
|
|
} |
342
|
|
|
|
|
|
|
|
343
|
1
|
|
|
|
|
45
|
$token = Mojo::URL->new($res->param('approval_url'))->query->param('token'); |
344
|
|
|
|
|
|
|
|
345
|
1
|
|
|
|
|
282
|
$res->param(state => $json->{state}); |
346
|
1
|
|
|
|
|
49
|
$res->param(transaction_id => $json->{id}); |
347
|
1
|
|
|
|
|
48
|
$res->headers->location($res->param('approval_url')); |
348
|
1
|
|
|
|
|
53
|
$res->code(302); |
349
|
1
|
|
|
|
|
9
|
$delay->pass($res); |
350
|
1
|
|
|
|
|
29
|
$self->transaction_id_mapper->($self, $token => $json->{id}, $delay->begin); |
351
|
1
|
|
|
|
|
19
|
1; |
352
|
1
|
50
|
|
|
|
3
|
} or do { |
353
|
0
|
|
|
|
|
0
|
warn "[MOJO_PAYPAL] ! $@" if DEBUG; |
354
|
0
|
|
|
|
|
0
|
$delay->pass($self->_extract_error($res, $@)); |
355
|
|
|
|
|
|
|
}; |
356
|
|
|
|
|
|
|
}, |
357
|
|
|
|
|
|
|
sub { |
358
|
1
|
|
|
1
|
|
157
|
my ($delay, $res, $err, $id) = @_; |
359
|
|
|
|
|
|
|
|
360
|
1
|
50
|
|
|
|
8
|
return $self->$cb($self->_error($err)) if $err; |
361
|
1
|
|
|
|
|
4
|
return $self->$cb($res); |
362
|
|
|
|
|
|
|
}, |
363
|
1
|
|
|
|
|
228
|
); |
364
|
|
|
|
|
|
|
|
365
|
1
|
|
|
|
|
105
|
$self; |
366
|
|
|
|
|
|
|
} |
367
|
|
|
|
|
|
|
|
368
|
|
|
|
|
|
|
=head2 register |
369
|
|
|
|
|
|
|
|
370
|
|
|
|
|
|
|
$app->plugin(PayPal => \%config); |
371
|
|
|
|
|
|
|
|
372
|
|
|
|
|
|
|
Called when registering this plugin in the main L application. |
373
|
|
|
|
|
|
|
|
374
|
|
|
|
|
|
|
=cut |
375
|
|
|
|
|
|
|
|
376
|
|
|
|
|
|
|
sub register { |
377
|
2
|
|
|
2
|
1
|
82
|
my ($self, $app, $config) = @_; |
378
|
|
|
|
|
|
|
|
379
|
|
|
|
|
|
|
# self contained |
380
|
2
|
100
|
|
|
|
14
|
if (ref $config->{secret}) { |
|
|
50
|
|
|
|
|
|
381
|
1
|
|
|
|
|
4
|
$self->_add_routes($app); |
382
|
1
|
|
|
|
|
20
|
$self->_ua->server->app($app); |
383
|
1
|
|
|
|
|
62
|
$config->{secret} = ${ $config->{secret} }; |
|
1
|
|
|
|
|
3
|
|
384
|
|
|
|
|
|
|
} |
385
|
|
|
|
|
|
|
elsif ($app->mode eq 'production') { |
386
|
1
|
|
50
|
|
|
15
|
$config->{base_url} ||= 'https://api.paypal.com'; |
387
|
|
|
|
|
|
|
} |
388
|
|
|
|
|
|
|
|
389
|
|
|
|
|
|
|
# copy config to this object |
390
|
2
|
|
|
|
|
7
|
for (grep { $self->$_ } keys %$config) { |
|
2
|
|
|
|
|
9
|
|
391
|
2
|
|
|
|
|
23
|
$self->{$_} = $config->{$_}; |
392
|
|
|
|
|
|
|
} |
393
|
|
|
|
|
|
|
|
394
|
2
|
50
|
|
|
|
9
|
unless ($self->transaction_id_mapper) { |
395
|
|
|
|
|
|
|
$self->transaction_id_mapper(sub { |
396
|
2
|
|
|
2
|
|
31
|
my ($self, $token, $transaction_id, $cb) = @_; |
397
|
2
|
|
|
|
|
9
|
$app->log->warn("You need to set 'transaction_id_mapper' in Mojolicious::Plugin::PayPal"); |
398
|
2
|
|
66
|
|
|
72
|
$self->$cb('', $self->{transaction_id_map}{$token} //= $transaction_id); |
399
|
2
|
|
|
|
|
22
|
}); |
400
|
|
|
|
|
|
|
} |
401
|
|
|
|
|
|
|
|
402
|
|
|
|
|
|
|
$app->helper( |
403
|
|
|
|
|
|
|
paypal => sub { |
404
|
5
|
|
|
5
|
|
52524
|
my $c = shift; |
405
|
5
|
100
|
|
|
|
27
|
return $self unless @_; |
406
|
3
|
|
|
|
|
15
|
my $method = sprintf '%s_payment', shift; |
407
|
3
|
|
|
|
|
17
|
$self->$method($c, @_); |
408
|
3
|
|
|
|
|
34
|
return $c; |
409
|
|
|
|
|
|
|
} |
410
|
2
|
|
|
|
|
26
|
); |
411
|
|
|
|
|
|
|
} |
412
|
|
|
|
|
|
|
|
413
|
|
|
|
|
|
|
sub _add_routes { |
414
|
1
|
|
|
1
|
|
2
|
my ($self, $app) = @_; |
415
|
1
|
|
|
|
|
7
|
my $r = $app->routes; |
416
|
1
|
|
50
|
|
|
18
|
my $payments = $self->{payments} ||= {}; # just here for debug purposes, may change without warning |
417
|
|
|
|
|
|
|
|
418
|
1
|
|
|
|
|
4
|
$self->base_url('/paypal'); |
419
|
|
|
|
|
|
|
|
420
|
1
|
|
|
|
|
21
|
$r->post('/paypal/v1/oauth2/token' => { template => 'paypal/v1/oauth2/token', format => 'json' }); |
421
|
|
|
|
|
|
|
|
422
|
|
|
|
|
|
|
$r->post('/paypal/v1/payments/payment' => sub { |
423
|
1
|
|
|
1
|
|
4620
|
my $self = shift; |
424
|
1
|
|
|
|
|
5
|
my $token = 'EC-60U79048BN7719609'; |
425
|
1
|
|
|
|
|
8
|
$payments->{$token} = $self->req->json; |
426
|
1
|
|
|
|
|
511
|
$self->render('paypal/v1/payments/payment', token => $token, format => 'json'); |
427
|
1
|
|
|
|
|
473
|
}); |
428
|
|
|
|
|
|
|
|
429
|
|
|
|
|
|
|
$r->get('/paypal/webscr')->to(cb => sub { |
430
|
1
|
|
|
1
|
|
13444
|
my $self = shift; |
431
|
1
|
|
50
|
|
|
6
|
my $token = $self->param('token') || 'missing'; |
432
|
1
|
|
|
|
|
376
|
$payments->{CR87QHB7JTRSC} = $payments->{$token}; # payer_id = CR87QHB7JTRSC |
433
|
1
|
|
|
|
|
7
|
$self->render('paypal/webscr', format => 'html', payment => $payments->{$token}); |
434
|
1
|
|
|
|
|
433
|
}); |
435
|
|
|
|
|
|
|
|
436
|
|
|
|
|
|
|
$r->post('/paypal/v1/payments/payment/:transaction_id/execute')->to(cb => sub { |
437
|
1
|
|
|
1
|
|
4283
|
my $self = shift; |
438
|
1
|
|
50
|
|
|
6
|
my $payer_id = $self->req->json->{payer_id} || 'missing'; |
439
|
1
|
|
|
|
|
159
|
$self->render('paypal/v1/payments/payment/execute', payment => $payments->{$payer_id}, format => 'json'); |
440
|
1
|
|
|
|
|
292
|
}); |
441
|
|
|
|
|
|
|
|
442
|
1
|
|
|
|
|
520
|
push @{ $app->renderer->classes }, __PACKAGE__; |
|
1
|
|
|
|
|
7
|
|
443
|
|
|
|
|
|
|
} |
444
|
|
|
|
|
|
|
|
445
|
|
|
|
|
|
|
sub _error { |
446
|
1
|
|
|
1
|
|
29
|
my ($self, $err) = @_; |
447
|
1
|
|
|
|
|
15
|
my $res = Mojolicious::Plugin::NetsPayment::Res->new; |
448
|
1
|
|
|
|
|
11
|
$res->code(400); |
449
|
1
|
|
|
|
|
12
|
$res->param(message => $err); |
450
|
1
|
|
|
|
|
147
|
$res->param(source => __PACKAGE__); |
451
|
1
|
|
|
|
|
46
|
$res; |
452
|
|
|
|
|
|
|
} |
453
|
|
|
|
|
|
|
|
454
|
|
|
|
|
|
|
sub _extract_error { |
455
|
0
|
|
|
0
|
|
0
|
my ($self, $res, $e) = @_; |
456
|
0
|
|
|
|
|
0
|
my $err = ''; # TODO |
457
|
|
|
|
|
|
|
|
458
|
0
|
|
|
|
|
0
|
$res->code(500); |
459
|
0
|
|
0
|
|
|
0
|
$res->param(message => $err // $e); |
460
|
0
|
0
|
|
|
|
0
|
$res->param(source => $err ? $self->base_url : __PACKAGE__); |
461
|
0
|
|
|
|
|
0
|
$res; |
462
|
|
|
|
|
|
|
} |
463
|
|
|
|
|
|
|
|
464
|
|
|
|
|
|
|
sub _get_access_token { |
465
|
1
|
|
|
1
|
|
11
|
my ($self, $cb) = @_; |
466
|
1
|
|
|
|
|
4
|
my $token_url = $self->_url('/v1/oauth2/token'); |
467
|
1
|
|
|
|
|
5
|
my %headers = ( 'Accept' => 'application/json', 'Accept-Language' => 'en_US' ); |
468
|
|
|
|
|
|
|
|
469
|
1
|
|
|
|
|
5
|
$token_url->userinfo(join ':', $self->client_id, $self->secret); |
470
|
1
|
|
|
|
|
17
|
warn "[MOJO_PAYPAL] Token URL $token_url\n" if DEBUG == 2; |
471
|
|
|
|
|
|
|
|
472
|
|
|
|
|
|
|
Mojo::IOLoop->delay( |
473
|
|
|
|
|
|
|
sub { |
474
|
1
|
|
|
1
|
|
98
|
my ($delay) = @_; |
475
|
1
|
|
|
|
|
4
|
$self->_ua->post($token_url, \%headers, form => { grant_type => 'client_credentials' }, $delay->begin); |
476
|
|
|
|
|
|
|
}, |
477
|
|
|
|
|
|
|
sub { |
478
|
1
|
|
|
1
|
|
18190
|
my ($delay, $tx) = @_; |
479
|
1
|
|
50
|
|
|
3
|
my $json = eval { $tx->res->json } || {}; |
480
|
|
|
|
|
|
|
|
481
|
1
|
|
50
|
|
|
352
|
$json->{access_token} //= ''; |
482
|
1
|
|
|
|
|
10
|
$self->$cb($self->{access_token} = $json->{access_token}, $tx); |
483
|
|
|
|
|
|
|
}, |
484
|
1
|
|
|
|
|
9
|
); |
485
|
|
|
|
|
|
|
} |
486
|
|
|
|
|
|
|
|
487
|
|
|
|
|
|
|
# https://developer.paypal.com/webapps/developer/docs/integration/direct/make-your-first-call/ |
488
|
|
|
|
|
|
|
sub _make_request_with_token { |
489
|
2
|
|
|
2
|
|
1222
|
my ($self, $method, $url, $body, $cb) = @_; |
490
|
2
|
|
|
|
|
8
|
my %headers = ( 'Content-Type' => 'application/json' ); |
491
|
|
|
|
|
|
|
|
492
|
|
|
|
|
|
|
Mojo::IOLoop->delay( |
493
|
|
|
|
|
|
|
sub { # get token unless we have it |
494
|
2
|
|
|
2
|
|
250
|
my ($delay) = @_; |
495
|
2
|
100
|
|
|
|
11
|
return $delay->pass($self->{access_token}, undef) if $self->{access_token}; |
496
|
1
|
|
|
|
|
5
|
return $self->_get_access_token($delay->begin); |
497
|
|
|
|
|
|
|
}, |
498
|
|
|
|
|
|
|
sub { # abort or make request with token |
499
|
2
|
|
|
2
|
|
119
|
my ($delay, $token, $tx) = @_; |
500
|
2
|
50
|
|
|
|
8
|
return $self->$cb($tx) unless $token; |
501
|
2
|
|
|
|
|
9
|
$headers{Authorization} = "Bearer $token"; |
502
|
2
|
|
|
|
|
5
|
warn "[MOJO_PAYPAL] Authorization: Bearer $token\n" if DEBUG; |
503
|
2
|
|
|
|
|
11
|
return $self->_ua->$method($url, \%headers, $body, $delay->begin); |
504
|
|
|
|
|
|
|
}, |
505
|
|
|
|
|
|
|
sub { # get token if it has expired |
506
|
2
|
|
|
2
|
|
13808
|
my ($delay, $tx) = @_; |
507
|
2
|
50
|
|
|
|
8
|
return $self->_get_access_token($delay->begin) if $tx->res->code == 401; |
508
|
2
|
|
|
|
|
30
|
return $delay->pass(undef, $tx); # success |
509
|
|
|
|
|
|
|
}, |
510
|
|
|
|
|
|
|
sub { # return or retry request with new token |
511
|
2
|
|
|
2
|
|
422
|
my ($delay, $token, $tx) = @_; |
512
|
2
|
50
|
|
|
|
13
|
return $self->$cb($tx) unless $token; # return success or error $tx |
513
|
0
|
|
|
|
|
0
|
$headers{Authorization} = "Bearer $token"; |
514
|
0
|
|
|
|
|
0
|
warn "[MOJO_PAYPAL] Authorization: Bearer $token\n" if DEBUG; |
515
|
0
|
|
|
|
|
0
|
return $self->_ua->$method($url, \%headers, $body, $cb); |
516
|
|
|
|
|
|
|
}, |
517
|
2
|
|
|
|
|
51
|
); |
518
|
|
|
|
|
|
|
} |
519
|
|
|
|
|
|
|
|
520
|
|
|
|
|
|
|
sub _url { |
521
|
4
|
|
|
4
|
|
22
|
my $url = Mojo::URL->new($_[0]->base_url .$_[1]); |
522
|
4
|
|
|
|
|
226
|
warn "[MOJO_PAYPAL] URL $url\n" if DEBUG; |
523
|
4
|
|
|
|
|
9
|
$url; |
524
|
|
|
|
|
|
|
} |
525
|
|
|
|
|
|
|
|
526
|
|
|
|
|
|
|
package |
527
|
|
|
|
|
|
|
Mojolicious::Plugin::NetsPayment::Res; |
528
|
2
|
|
|
2
|
|
14
|
use Mojo::Base 'Mojo::Message::Response'; |
|
2
|
|
|
|
|
2
|
|
|
2
|
|
|
|
|
10
|
|
529
|
27
|
|
|
27
|
|
553
|
sub param { shift->body_params->param(@_) } |
530
|
|
|
|
|
|
|
|
531
|
|
|
|
|
|
|
=head1 COPYRIGHT AND LICENSE |
532
|
|
|
|
|
|
|
|
533
|
|
|
|
|
|
|
Copyright (C) 2014, Jan Henning Thorsen |
534
|
|
|
|
|
|
|
|
535
|
|
|
|
|
|
|
This program is free software, you can redistribute it and/or modify it under |
536
|
|
|
|
|
|
|
the terms of the Artistic License version 2.0. |
537
|
|
|
|
|
|
|
|
538
|
|
|
|
|
|
|
=head1 AUTHOR |
539
|
|
|
|
|
|
|
|
540
|
|
|
|
|
|
|
Jan Henning Thorsen - C |
541
|
|
|
|
|
|
|
|
542
|
|
|
|
|
|
|
=head1 CONTRIBUTORS |
543
|
|
|
|
|
|
|
|
544
|
|
|
|
|
|
|
Yu Pan - C |
545
|
|
|
|
|
|
|
|
546
|
|
|
|
|
|
|
=cut |
547
|
|
|
|
|
|
|
|
548
|
|
|
|
|
|
|
1; |
549
|
|
|
|
|
|
|
|
550
|
|
|
|
|
|
|
package Mojolicious::Plugin::PayPal; |
551
|
|
|
|
|
|
|
__DATA__ |