Completed
Branch BUG/paypal-pro-can-handle-disc... (28c801)
by
unknown
09:09 queued 35s
created
caffeinated/payment_methods/Paypal_Pro/EEG_Paypal_Pro.gateway.php 1 patch
Indentation   +596 added lines, -596 removed lines patch added patch discarded remove patch
@@ -11,600 +11,600 @@
 block discarded – undo
11 11
 class EEG_Paypal_Pro extends EE_Onsite_Gateway
12 12
 {
13 13
 
14
-    /**
15
-     * @var $_paypal_api_username string
16
-     */
17
-    protected $_username = null;
18
-
19
-    /**
20
-     * @var $_password string
21
-     */
22
-    protected $_password = null;
23
-
24
-    /**
25
-     * @var $_signature string
26
-     */
27
-    protected $_signature = null;
28
-
29
-    /**
30
-     * @var $_credit_card_types array with the keys for credit card types accepted on this account
31
-     */
32
-    protected $_credit_card_types    = null;
33
-
34
-    protected $_currencies_supported = array(
35
-        'USD',
36
-        'GBP',
37
-        'CAD',
38
-        'AUD',
39
-        'BRL',
40
-        'CHF',
41
-        'CZK',
42
-        'DKK',
43
-        'EUR',
44
-        'HKD',
45
-        'HUF',
46
-        'ILS',
47
-        'JPY',
48
-        'MXN',
49
-        'MYR',
50
-        'NOK',
51
-        'NZD',
52
-        'PHP',
53
-        'PLN',
54
-        'SEK',
55
-        'SGD',
56
-        'THB',
57
-        'TRY',
58
-        'TWD',
59
-        'RUB',
60
-        'INR',
61
-    );
62
-
63
-
64
-
65
-    /**
66
-     * @param EEI_Payment $payment
67
-     * @param array       $billing_info {
68
-     * @type string $credit_card
69
-     * @type string $credit_card_type
70
-     * @type string $exp_month always 2 characters
71
-     * @type string $exp_year always 4 characters
72
-     * @type string $cvv
73
-     * }
74
-     * @see      parent::do_direct_payment for more info
75
-     * @return EE_Payment|EEI_Payment
76
-     * @throws EE_Error
77
-     */
78
-    public function do_direct_payment($payment, $billing_info = null)
79
-    {
80
-        $transaction = $payment->transaction();
81
-        if (! $transaction instanceof EEI_Transaction) {
82
-            throw new EE_Error(
83
-                esc_html__('No transaction for payment while paying with PayPal Pro.', 'event_espresso')
84
-            );
85
-        }
86
-        $primary_registrant = $transaction->primary_registration();
87
-        if (! $primary_registrant instanceof EEI_Registration) {
88
-            throw new EE_Error(
89
-                esc_html__(
90
-                    'No primary registration on transaction while paying with PayPal Pro.',
91
-                    'event_espresso'
92
-                )
93
-            );
94
-        }
95
-        $attendee = $primary_registrant->attendee();
96
-        if (! $attendee instanceof EEI_Attendee) {
97
-            throw new EE_Error(
98
-                esc_html__(
99
-                    'No attendee on primary registration while paying with PayPal Pro.',
100
-                    'event_espresso'
101
-                )
102
-            );
103
-        }
104
-        $gateway_formatter = $this->_get_gateway_formatter();
105
-        $order_description = substr($gateway_formatter->formatOrderDescription($payment), 0, 127);
106
-        // charge for the full amount. Show itemized list
107
-        if ($this->_money->compare_floats($payment->amount(), $transaction->total(), '==')) {
108
-            $item_num = 1;
109
-            $total_line_item = $transaction->total_line_item();
110
-            $order_items = array();
111
-            foreach ($total_line_item->get_items() as $line_item) {
112
-                // ignore line items with a quantity of 0
113
-                if ($line_item->quantity() == 0) {
114
-                    continue;
115
-                }
116
-                // For percent items, whose unit_price is 0, use the total instead.
117
-                if ($line_item->is_percent()) {
118
-                    $unit_price = $line_item->total();
119
-                    $line_item_quantity = 1;
120
-                } else {
121
-                    $unit_price = $line_item->unit_price();
122
-                    $line_item_quantity = $line_item->quantity();
123
-                }
124
-                $item = array(
125
-                    // Item Name.  127 char max.
126
-                    'l_name'                 => substr(
127
-                        $gateway_formatter->formatLineItemName($line_item, $payment),
128
-                        0,
129
-                        127
130
-                    ),
131
-                    // Item description.  127 char max.
132
-                    'l_desc'                 => substr(
133
-                        $gateway_formatter->formatLineItemDesc($line_item, $payment),
134
-                        0,
135
-                        127
136
-                    ),
137
-                    // Cost of individual item.
138
-                    'l_amt'                  => $unit_price,
139
-                    // Item Number.  127 char max.
140
-                    'l_number'               => $item_num++,
141
-                    // Item quantity.  Must be any positive integer.
142
-                    'l_qty'                  => $line_item_quantity,
143
-                    // Item's sales tax amount.
144
-                    'l_taxamt'               => '',
145
-                    // eBay auction number of item.
146
-                    'l_ebayitemnumber'       => '',
147
-                    // eBay transaction ID of purchased item.
148
-                    'l_ebayitemauctiontxnid' => '',
149
-                    // eBay order ID for the item.
150
-                    'l_ebayitemorderid'      => '',
151
-                );
152
-                // add to array of all items
153
-                array_push($order_items, $item);
154
-            }
155
-            $item_amount = $total_line_item->get_items_total();
156
-            $tax_amount = $total_line_item->get_total_tax();
157
-        } else {
158
-            $order_items = array();
159
-            $item_amount = $payment->amount();
160
-            $tax_amount = 0;
161
-            array_push($order_items, array(
162
-                // Item Name.  127 char max.
163
-                'l_name'   => substr(
164
-                    $gateway_formatter->formatPartialPaymentLineItemName($payment),
165
-                    0,
166
-                    127
167
-                ),
168
-                // Item description.  127 char max.
169
-                'l_desc'   => substr(
170
-                    $gateway_formatter->formatPartialPaymentLineItemDesc($payment),
171
-                    0,
172
-                    127
173
-                ),
174
-                // Cost of individual item.
175
-                'l_amt'    => $payment->amount(),
176
-                // Item Number.  127 char max.
177
-                'l_number' => 1,
178
-                // Item quantity.  Must be any positive integer.
179
-                'l_qty'    => 1,
180
-            ));
181
-        }
182
-        // Populate data arrays with order data.
183
-        $DPFields = array(
184
-            // How you want to obtain payment ?
185
-            // Authorization indicates the payment is a basic auth subject to settlement with Auth & Capture.
186
-            // Sale indicates that this is a final sale for which you are requesting payment.  Default is Sale.
187
-            'paymentaction'    => 'Sale',
188
-            // Required.  IP address of the payer's browser.
189
-            'ipaddress'        => $_SERVER['REMOTE_ADDR'],
190
-            // Flag to determine whether you want the results returned by FMF.  1 or 0.  Default is 0.
191
-            'returnfmfdetails' => '1',
192
-        );
193
-        $CCDetails = array(
194
-            // Required. Type of credit card.  Visa, MasterCard, Discover, Amex, Maestro, Solo.
195
-            // If Maestro or Solo, the currency code must be GBP.
196
-            //  In addition, either start date or issue number must be specified.
197
-            'creditcardtype' => $billing_info['credit_card_type'],
198
-            // Required.  Credit card number.  No spaces or punctuation.
199
-            'acct'           => $billing_info['credit_card'],
200
-            // Required.  Credit card expiration date.  Format is MMYYYY
201
-            'expdate'        => $billing_info['exp_month'] . $billing_info['exp_year'],
202
-            // Requirements determined by your PayPal account settings.  Security digits for credit card.
203
-            'cvv2'           => $billing_info['cvv'],
204
-        );
205
-        $PayerInfo = array(
206
-            // Email address of payer.
207
-            'email'       => $billing_info['email'],
208
-            // Unique PayPal customer ID for payer.
209
-            'payerid'     => '',
210
-            // Status of payer.  Values are verified or unverified
211
-            'payerstatus' => '',
212
-            // Payer's business name.
213
-            'business'    => '',
214
-        );
215
-        $PayerName = array(
216
-            // Payer's salutation.  20 char max.
217
-            'salutation' => '',
218
-            // Payer's first name.  25 char max.
219
-            'firstname'  => substr($billing_info['first_name'], 0, 25),
220
-            // Payer's middle name.  25 char max.
221
-            'middlename' => '',
222
-            // Payer's last name.  25 char max.
223
-            'lastname'   => substr($billing_info['last_name'], 0, 25),
224
-            // Payer's suffix.  12 char max.
225
-            'suffix'     => '',
226
-        );
227
-        $BillingAddress = array(
228
-            // Required.  First street address.
229
-            'street'      => $billing_info['address'],
230
-            // Second street address.
231
-            'street2'     => $billing_info['address2'],
232
-            // Required.  Name of City.
233
-            'city'        => $billing_info['city'],
234
-            // Required. Name of State or Province.
235
-            'state'       => substr($billing_info['state'], 0, 40),
236
-            // Required.  Country code.
237
-            'countrycode' => $billing_info['country'],
238
-            // Required.  Postal code of payer.
239
-            'zip'         => $billing_info['zip'],
240
-        );
241
-        // check if the registration info contains the needed fields for paypal pro
242
-        // (see https://developer.paypal.com/docs/classic/api/merchant/DoDirectPayment_API_Operation_NVP/)
243
-        if ($attendee->address() && $attendee->city() && $attendee->country_ID()) {
244
-            $use_registration_address_info = true;
245
-        } else {
246
-            $use_registration_address_info = false;
247
-        }
248
-        // so if the attendee has enough data to fill out PayPal Pro's shipping info, use it.
249
-        // If not, use the billing info again
250
-        $ShippingAddress = array(
251
-            'shiptoname'     => substr($use_registration_address_info
252
-                ? $attendee->full_name()
253
-                : $billing_info['first_name'] . ' ' . $billing_info['last_name'], 0, 32),
254
-            'shiptostreet'   => substr($use_registration_address_info
255
-                ? $attendee->address()
256
-                : $billing_info['address'], 0, 100),
257
-            'shiptostreet2'  => substr($use_registration_address_info
258
-                ? $attendee->address2() : $billing_info['address2'], 0, 100),
259
-            'shiptocity'     => substr($use_registration_address_info
260
-                ? $attendee->city()
261
-                : $billing_info['city'], 0, 40),
262
-            'state'          => substr($use_registration_address_info
263
-                ? $attendee->state_name()
264
-                : $billing_info['state'], 0, 40),
265
-            'shiptocountry'  => $use_registration_address_info
266
-                ? $attendee->country_ID()
267
-                : $billing_info['country'],
268
-            'shiptozip'      => substr($use_registration_address_info
269
-                ? $attendee->zip()
270
-                : $billing_info['zip'], 0, 20),
271
-            'shiptophonenum' => substr($use_registration_address_info
272
-                ? $attendee->phone()
273
-                : $billing_info['phone'], 0, 20),
274
-        );
275
-        $PaymentDetails = array(
276
-            // Required.  Total amount of order, including shipping, handling, and tax.
277
-            'amt'          => $gateway_formatter->formatCurrency($payment->amount()),
278
-            // Required.  Three-letter currency code.  Default is USD.
279
-            'currencycode' => $payment->currency_code(),
280
-            // Required if you include itemized cart details. (L_AMTn, etc.)
281
-            // Subtotal of items not including S&H, or tax.
282
-            'itemamt'      => $gateway_formatter->formatCurrency($item_amount),//
283
-            // Total shipping costs for the order.  If you specify shippingamt, you must also specify itemamt.
284
-            'shippingamt'  => '',
285
-            // Total handling costs for the order.  If you specify handlingamt, you must also specify itemamt.
286
-            'handlingamt'  => '',
287
-            // Required if you specify itemized cart tax details.
288
-            // Sum of tax for all items on the order.  Total sales tax.
289
-            'taxamt'       => $gateway_formatter->formatCurrency($tax_amount),
290
-            // Description of the order the customer is purchasing.  127 char max.
291
-            'desc'         => $order_description,
292
-            // Free-form field for your own use.  256 char max.
293
-            'custom'       => $primary_registrant ? $primary_registrant->ID() : '',
294
-            // Your own invoice or tracking number
295
-            'invnum'       => wp_generate_password(12, false),// $transaction->ID(),
296
-            // URL for receiving Instant Payment Notifications.  This overrides what your profile is set to use.
297
-            'notifyurl'    => '',
298
-            'buttonsource' => 'EventEspresso_SP',// EE will blow up if you change this
299
-        );
300
-        // Wrap all data arrays into a single, "master" array which will be passed into the class function.
301
-        $PayPalRequestData = array(
302
-            'DPFields'        => $DPFields,
303
-            'CCDetails'       => $CCDetails,
304
-            'PayerInfo'       => $PayerInfo,
305
-            'PayerName'       => $PayerName,
306
-            'BillingAddress'  => $BillingAddress,
307
-            'ShippingAddress' => $ShippingAddress,
308
-            'PaymentDetails'  => $PaymentDetails,
309
-            'OrderItems'      => $order_items,
310
-        );
311
-        $this->_log_clean_request($PayPalRequestData, $payment);
312
-        try {
313
-            $PayPalResult = $this->prep_and_curl_request($PayPalRequestData);
314
-            // remove PCI-sensitive data so it doesn't get stored
315
-            $PayPalResult = $this->_log_clean_response($PayPalResult, $payment);
316
-            $message = isset($PayPalResult['L_LONGMESSAGE0']) ? $PayPalResult['L_LONGMESSAGE0'] : $PayPalResult['ACK'];
317
-            if (empty($PayPalResult['RAWRESPONSE'])) {
318
-                $payment->set_status($this->_pay_model->failed_status());
319
-                $payment->set_gateway_response(__('No response received from Paypal Pro', 'event_espresso'));
320
-                $payment->set_details($PayPalResult);
321
-            } else {
322
-                if ($this->_APICallSuccessful($PayPalResult)) {
323
-                    $payment->set_status($this->_pay_model->approved_status());
324
-                } else {
325
-                    $payment->set_status($this->_pay_model->declined_status());
326
-                }
327
-                // make sure we interpret the AMT as a float, not an international string
328
-                // (where periods are thousand separators)
329
-                $payment->set_amount(isset($PayPalResult['AMT']) ? floatval($PayPalResult['AMT']) : 0);
330
-                $payment->set_gateway_response($message);
331
-                $payment->set_txn_id_chq_nmbr(isset($PayPalResult['TRANSACTIONID'])
332
-                    ? $PayPalResult['TRANSACTIONID']
333
-                    : null);
334
-                $primary_registration_code = $primary_registrant instanceof EE_Registration
335
-                    ? $primary_registrant->reg_code()
336
-                    : '';
337
-                $payment->set_extra_accntng($primary_registration_code);
338
-                $payment->set_details($PayPalResult);
339
-            }
340
-        } catch (Exception $e) {
341
-            $payment->set_status($this->_pay_model->failed_status());
342
-            $payment->set_gateway_response($e->getMessage());
343
-        }
344
-        // $payment->set_status( $this->_pay_model->declined_status() );
345
-        // $payment->set_gateway_response( '' );
346
-        return $payment;
347
-    }
348
-
349
-
350
-
351
-    /**
352
-     * CLeans out sensitive CC data and then logs it, and returns the cleaned request
353
-     *
354
-     * @param array       $request
355
-     * @param EEI_Payment $payment
356
-     * @return void
357
-     */
358
-    private function _log_clean_request($request, $payment)
359
-    {
360
-        $cleaned_request_data = $request;
361
-        unset($cleaned_request_data['CCDetails']['acct']);
362
-        unset($cleaned_request_data['CCDetails']['cvv2']);
363
-        unset($cleaned_request_data['CCDetails']['expdate']);
364
-        $this->log(array('Paypal Request' => $cleaned_request_data), $payment);
365
-    }
366
-
367
-
368
-
369
-    /**
370
-     * Cleans the response, logs it, and returns it
371
-     *
372
-     * @param array       $response
373
-     * @param EEI_Payment $payment
374
-     * @return array cleaned
375
-     */
376
-    private function _log_clean_response($response, $payment)
377
-    {
378
-        unset($response['REQUESTDATA']['CREDITCARDTYPE']);
379
-        unset($response['REQUESTDATA']['ACCT']);
380
-        unset($response['REQUESTDATA']['EXPDATE']);
381
-        unset($response['REQUESTDATA']['CVV2']);
382
-        unset($response['RAWREQUEST']);
383
-        $this->log(array('Paypal Response' => $response), $payment);
384
-        return $response;
385
-    }
386
-
387
-
388
-
389
-    /**
390
-     * @param $DataArray
391
-     * @return array
392
-     */
393
-    private function prep_and_curl_request($DataArray)
394
-    {
395
-        // Create empty holders for each portion of the NVP string
396
-        $DPFieldsNVP = '&METHOD=DoDirectPayment&BUTTONSOURCE=AngellEYE_PHP_Class_DDP';
397
-        $CCDetailsNVP = '';
398
-        $PayerInfoNVP = '';
399
-        $PayerNameNVP = '';
400
-        $BillingAddressNVP = '';
401
-        $ShippingAddressNVP = '';
402
-        $PaymentDetailsNVP = '';
403
-        $OrderItemsNVP = '';
404
-        $Secure3DNVP = '';
405
-        // DP Fields
406
-        $DPFields = isset($DataArray['DPFields']) ? $DataArray['DPFields'] : array();
407
-        foreach ($DPFields as $DPFieldsVar => $DPFieldsVal) {
408
-            $DPFieldsNVP .= '&' . strtoupper($DPFieldsVar) . '=' . urlencode($DPFieldsVal);
409
-        }
410
-        // CC Details Fields
411
-        $CCDetails = isset($DataArray['CCDetails']) ? $DataArray['CCDetails'] : array();
412
-        foreach ($CCDetails as $CCDetailsVar => $CCDetailsVal) {
413
-            $CCDetailsNVP .= '&' . strtoupper($CCDetailsVar) . '=' . urlencode($CCDetailsVal);
414
-        }
415
-        // PayerInfo Type Fields
416
-        $PayerInfo = isset($DataArray['PayerInfo']) ? $DataArray['PayerInfo'] : array();
417
-        foreach ($PayerInfo as $PayerInfoVar => $PayerInfoVal) {
418
-            $PayerInfoNVP .= '&' . strtoupper($PayerInfoVar) . '=' . urlencode($PayerInfoVal);
419
-        }
420
-        // Payer Name Fields
421
-        $PayerName = isset($DataArray['PayerName']) ? $DataArray['PayerName'] : array();
422
-        foreach ($PayerName as $PayerNameVar => $PayerNameVal) {
423
-            $PayerNameNVP .= '&' . strtoupper($PayerNameVar) . '=' . urlencode($PayerNameVal);
424
-        }
425
-        // Address Fields (Billing)
426
-        $BillingAddress = isset($DataArray['BillingAddress']) ? $DataArray['BillingAddress'] : array();
427
-        foreach ($BillingAddress as $BillingAddressVar => $BillingAddressVal) {
428
-            $BillingAddressNVP .= '&' . strtoupper($BillingAddressVar) . '=' . urlencode($BillingAddressVal);
429
-        }
430
-        // Payment Details Type Fields
431
-        $PaymentDetails = isset($DataArray['PaymentDetails']) ? $DataArray['PaymentDetails'] : array();
432
-        foreach ($PaymentDetails as $PaymentDetailsVar => $PaymentDetailsVal) {
433
-            $PaymentDetailsNVP .= '&' . strtoupper($PaymentDetailsVar) . '=' . urlencode($PaymentDetailsVal);
434
-        }
435
-        // Payment Details Item Type Fields
436
-        $OrderItems = isset($DataArray['OrderItems']) ? $DataArray['OrderItems'] : array();
437
-        $n = 0;
438
-        foreach ($OrderItems as $OrderItemsVar => $OrderItemsVal) {
439
-            $CurrentItem = $OrderItems[ $OrderItemsVar ];
440
-            foreach ($CurrentItem as $CurrentItemVar => $CurrentItemVal) {
441
-                $OrderItemsNVP .= '&' . strtoupper($CurrentItemVar) . $n . '=' . urlencode($CurrentItemVal);
442
-            }
443
-            $n++;
444
-        }
445
-        // Ship To Address Fields
446
-        $ShippingAddress = isset($DataArray['ShippingAddress']) ? $DataArray['ShippingAddress'] : array();
447
-        foreach ($ShippingAddress as $ShippingAddressVar => $ShippingAddressVal) {
448
-            $ShippingAddressNVP .= '&' . strtoupper($ShippingAddressVar) . '=' . urlencode($ShippingAddressVal);
449
-        }
450
-        // 3D Secure Fields
451
-        $Secure3D = isset($DataArray['Secure3D']) ? $DataArray['Secure3D'] : array();
452
-        foreach ($Secure3D as $Secure3DVar => $Secure3DVal) {
453
-            $Secure3DNVP .= '&' . strtoupper($Secure3DVar) . '=' . urlencode($Secure3DVal);
454
-        }
455
-        // Now that we have each chunk we need to go ahead and append them all together for our entire NVP string
456
-        $NVPRequest = 'USER='
457
-                      . $this->_username
458
-                      . '&PWD='
459
-                      . $this->_password
460
-                      . '&VERSION=64.0'
461
-                      . '&SIGNATURE='
462
-                      . $this->_signature
463
-                      . $DPFieldsNVP
464
-                      . $CCDetailsNVP
465
-                      . $PayerInfoNVP
466
-                      . $PayerNameNVP
467
-                      . $BillingAddressNVP
468
-                      . $PaymentDetailsNVP
469
-                      . $OrderItemsNVP
470
-                      . $ShippingAddressNVP
471
-                      . $Secure3DNVP;
472
-        $NVPResponse = $this->_CURLRequest($NVPRequest);
473
-        $NVPRequestArray = $this->_NVPToArray($NVPRequest);
474
-        $NVPResponseArray = $this->_NVPToArray($NVPResponse);
475
-        $Errors = $this->_GetErrors($NVPResponseArray);
476
-        $NVPResponseArray['ERRORS'] = $Errors;
477
-        $NVPResponseArray['REQUESTDATA'] = $NVPRequestArray;
478
-        $NVPResponseArray['RAWREQUEST'] = $NVPRequest;
479
-        $NVPResponseArray['RAWRESPONSE'] = $NVPResponse;
480
-        return $NVPResponseArray;
481
-    }
482
-
483
-
484
-
485
-    /**
486
-     * @param $Request
487
-     * @return mixed
488
-     */
489
-    private function _CURLRequest($Request)
490
-    {
491
-        $EndPointURL = $this->_debug_mode ? 'https://api-3t.sandbox.paypal.com/nvp' : 'https://api-3t.paypal.com/nvp';
492
-        $curl = curl_init();
493
-        curl_setopt($curl, CURLOPT_VERBOSE, apply_filters('FHEE__EEG_Paypal_Pro__CurlRequest__CURLOPT_VERBOSE', true));
494
-        curl_setopt($curl, CURLOPT_SSL_VERIFYPEER, false);
495
-        curl_setopt($curl, CURLOPT_TIMEOUT, 60);
496
-        curl_setopt($curl, CURLOPT_URL, $EndPointURL);
497
-        curl_setopt($curl, CURLOPT_RETURNTRANSFER, 1);
498
-        curl_setopt($curl, CURLOPT_POSTFIELDS, $Request);
499
-        curl_setopt($curl, CURLOPT_HTTP_VERSION, CURL_HTTP_VERSION_1_1);
500
-        // execute the curl POST
501
-        $Response = curl_exec($curl);
502
-        curl_close($curl);
503
-        return $Response;
504
-    }
505
-
506
-
507
-
508
-    /**
509
-     * @param $NVPString
510
-     * @return array
511
-     */
512
-    private function _NVPToArray($NVPString)
513
-    {
514
-        // prepare responses into array
515
-        $proArray = array();
516
-        while (strlen($NVPString)) {
517
-            // name
518
-            $keypos = strpos($NVPString, '=');
519
-            $keyval = substr($NVPString, 0, $keypos);
520
-            // value
521
-            $valuepos = strpos($NVPString, '&') ? strpos($NVPString, '&') : strlen($NVPString);
522
-            $valval = substr($NVPString, $keypos + 1, $valuepos - $keypos - 1);
523
-            // decoding the response
524
-            $proArray[ $keyval ] = urldecode($valval);
525
-            $NVPString = substr($NVPString, $valuepos + 1, strlen($NVPString));
526
-        }
527
-        return $proArray;
528
-    }
529
-
530
-
531
-
532
-    /**
533
-     * @param array $PayPalResult
534
-     * @return bool
535
-     */
536
-    private function _APICallSuccessful($PayPalResult)
537
-    {
538
-        $approved = false;
539
-        // check main response message from PayPal
540
-        if (isset($PayPalResult['ACK']) && ! empty($PayPalResult['ACK'])) {
541
-            $ack = strtoupper($PayPalResult['ACK']);
542
-            $approved = ($ack == 'SUCCESS' || $ack == 'SUCCESSWITHWARNING' || $ack == 'PARTIALSUCCESS') ? true : false;
543
-        }
544
-        return $approved;
545
-    }
546
-
547
-
548
-
549
-    /**
550
-     * @param $DataArray
551
-     * @return array
552
-     */
553
-    private function _GetErrors($DataArray)
554
-    {
555
-        $Errors = array();
556
-        $n = 0;
557
-        while (isset($DataArray[ 'L_ERRORCODE' . $n . '' ])) {
558
-            $LErrorCode = isset($DataArray[ 'L_ERRORCODE' . $n . '' ]) ? $DataArray[ 'L_ERRORCODE' . $n . '' ] : '';
559
-            $LShortMessage = isset($DataArray[ 'L_SHORTMESSAGE' . $n . '' ])
560
-                ? $DataArray[ 'L_SHORTMESSAGE' . $n . '' ]
561
-                : '';
562
-            $LLongMessage = isset($DataArray[ 'L_LONGMESSAGE' . $n . '' ])
563
-                ? $DataArray[ 'L_LONGMESSAGE' . $n . '' ]
564
-                : '';
565
-            $LSeverityCode = isset($DataArray[ 'L_SEVERITYCODE' . $n . '' ])
566
-                ? $DataArray[ 'L_SEVERITYCODE' . $n . '' ]
567
-                : '';
568
-            $CurrentItem = array(
569
-                'L_ERRORCODE'    => $LErrorCode,
570
-                'L_SHORTMESSAGE' => $LShortMessage,
571
-                'L_LONGMESSAGE'  => $LLongMessage,
572
-                'L_SEVERITYCODE' => $LSeverityCode,
573
-            );
574
-            array_push($Errors, $CurrentItem);
575
-            $n++;
576
-        }
577
-        return $Errors;
578
-    }
579
-
580
-
581
-
582
-    /**
583
-     *        nothing to see here...  move along....
584
-     *
585
-     * @access protected
586
-     * @param $Errors
587
-     * @return string
588
-     */
589
-    private function _DisplayErrors($Errors)
590
-    {
591
-        $error = '';
592
-        foreach ($Errors as $ErrorVar => $ErrorVal) {
593
-            $CurrentError = $Errors[ $ErrorVar ];
594
-            foreach ($CurrentError as $CurrentErrorVar => $CurrentErrorVal) {
595
-                $CurrentVarName = '';
596
-                if ($CurrentErrorVar == 'L_ERRORCODE') {
597
-                    $CurrentVarName = 'Error Code';
598
-                } elseif ($CurrentErrorVar == 'L_SHORTMESSAGE') {
599
-                    $CurrentVarName = 'Short Message';
600
-                } elseif ($CurrentErrorVar == 'L_LONGMESSAGE') {
601
-                    $CurrentVarName = 'Long Message';
602
-                } elseif ($CurrentErrorVar == 'L_SEVERITYCODE') {
603
-                    $CurrentVarName = 'Severity Code';
604
-                }
605
-                $error .= '<br />' . $CurrentVarName . ': ' . $CurrentErrorVal;
606
-            }
607
-        }
608
-        return $error;
609
-    }
14
+	/**
15
+	 * @var $_paypal_api_username string
16
+	 */
17
+	protected $_username = null;
18
+
19
+	/**
20
+	 * @var $_password string
21
+	 */
22
+	protected $_password = null;
23
+
24
+	/**
25
+	 * @var $_signature string
26
+	 */
27
+	protected $_signature = null;
28
+
29
+	/**
30
+	 * @var $_credit_card_types array with the keys for credit card types accepted on this account
31
+	 */
32
+	protected $_credit_card_types    = null;
33
+
34
+	protected $_currencies_supported = array(
35
+		'USD',
36
+		'GBP',
37
+		'CAD',
38
+		'AUD',
39
+		'BRL',
40
+		'CHF',
41
+		'CZK',
42
+		'DKK',
43
+		'EUR',
44
+		'HKD',
45
+		'HUF',
46
+		'ILS',
47
+		'JPY',
48
+		'MXN',
49
+		'MYR',
50
+		'NOK',
51
+		'NZD',
52
+		'PHP',
53
+		'PLN',
54
+		'SEK',
55
+		'SGD',
56
+		'THB',
57
+		'TRY',
58
+		'TWD',
59
+		'RUB',
60
+		'INR',
61
+	);
62
+
63
+
64
+
65
+	/**
66
+	 * @param EEI_Payment $payment
67
+	 * @param array       $billing_info {
68
+	 * @type string $credit_card
69
+	 * @type string $credit_card_type
70
+	 * @type string $exp_month always 2 characters
71
+	 * @type string $exp_year always 4 characters
72
+	 * @type string $cvv
73
+	 * }
74
+	 * @see      parent::do_direct_payment for more info
75
+	 * @return EE_Payment|EEI_Payment
76
+	 * @throws EE_Error
77
+	 */
78
+	public function do_direct_payment($payment, $billing_info = null)
79
+	{
80
+		$transaction = $payment->transaction();
81
+		if (! $transaction instanceof EEI_Transaction) {
82
+			throw new EE_Error(
83
+				esc_html__('No transaction for payment while paying with PayPal Pro.', 'event_espresso')
84
+			);
85
+		}
86
+		$primary_registrant = $transaction->primary_registration();
87
+		if (! $primary_registrant instanceof EEI_Registration) {
88
+			throw new EE_Error(
89
+				esc_html__(
90
+					'No primary registration on transaction while paying with PayPal Pro.',
91
+					'event_espresso'
92
+				)
93
+			);
94
+		}
95
+		$attendee = $primary_registrant->attendee();
96
+		if (! $attendee instanceof EEI_Attendee) {
97
+			throw new EE_Error(
98
+				esc_html__(
99
+					'No attendee on primary registration while paying with PayPal Pro.',
100
+					'event_espresso'
101
+				)
102
+			);
103
+		}
104
+		$gateway_formatter = $this->_get_gateway_formatter();
105
+		$order_description = substr($gateway_formatter->formatOrderDescription($payment), 0, 127);
106
+		// charge for the full amount. Show itemized list
107
+		if ($this->_money->compare_floats($payment->amount(), $transaction->total(), '==')) {
108
+			$item_num = 1;
109
+			$total_line_item = $transaction->total_line_item();
110
+			$order_items = array();
111
+			foreach ($total_line_item->get_items() as $line_item) {
112
+				// ignore line items with a quantity of 0
113
+				if ($line_item->quantity() == 0) {
114
+					continue;
115
+				}
116
+				// For percent items, whose unit_price is 0, use the total instead.
117
+				if ($line_item->is_percent()) {
118
+					$unit_price = $line_item->total();
119
+					$line_item_quantity = 1;
120
+				} else {
121
+					$unit_price = $line_item->unit_price();
122
+					$line_item_quantity = $line_item->quantity();
123
+				}
124
+				$item = array(
125
+					// Item Name.  127 char max.
126
+					'l_name'                 => substr(
127
+						$gateway_formatter->formatLineItemName($line_item, $payment),
128
+						0,
129
+						127
130
+					),
131
+					// Item description.  127 char max.
132
+					'l_desc'                 => substr(
133
+						$gateway_formatter->formatLineItemDesc($line_item, $payment),
134
+						0,
135
+						127
136
+					),
137
+					// Cost of individual item.
138
+					'l_amt'                  => $unit_price,
139
+					// Item Number.  127 char max.
140
+					'l_number'               => $item_num++,
141
+					// Item quantity.  Must be any positive integer.
142
+					'l_qty'                  => $line_item_quantity,
143
+					// Item's sales tax amount.
144
+					'l_taxamt'               => '',
145
+					// eBay auction number of item.
146
+					'l_ebayitemnumber'       => '',
147
+					// eBay transaction ID of purchased item.
148
+					'l_ebayitemauctiontxnid' => '',
149
+					// eBay order ID for the item.
150
+					'l_ebayitemorderid'      => '',
151
+				);
152
+				// add to array of all items
153
+				array_push($order_items, $item);
154
+			}
155
+			$item_amount = $total_line_item->get_items_total();
156
+			$tax_amount = $total_line_item->get_total_tax();
157
+		} else {
158
+			$order_items = array();
159
+			$item_amount = $payment->amount();
160
+			$tax_amount = 0;
161
+			array_push($order_items, array(
162
+				// Item Name.  127 char max.
163
+				'l_name'   => substr(
164
+					$gateway_formatter->formatPartialPaymentLineItemName($payment),
165
+					0,
166
+					127
167
+				),
168
+				// Item description.  127 char max.
169
+				'l_desc'   => substr(
170
+					$gateway_formatter->formatPartialPaymentLineItemDesc($payment),
171
+					0,
172
+					127
173
+				),
174
+				// Cost of individual item.
175
+				'l_amt'    => $payment->amount(),
176
+				// Item Number.  127 char max.
177
+				'l_number' => 1,
178
+				// Item quantity.  Must be any positive integer.
179
+				'l_qty'    => 1,
180
+			));
181
+		}
182
+		// Populate data arrays with order data.
183
+		$DPFields = array(
184
+			// How you want to obtain payment ?
185
+			// Authorization indicates the payment is a basic auth subject to settlement with Auth & Capture.
186
+			// Sale indicates that this is a final sale for which you are requesting payment.  Default is Sale.
187
+			'paymentaction'    => 'Sale',
188
+			// Required.  IP address of the payer's browser.
189
+			'ipaddress'        => $_SERVER['REMOTE_ADDR'],
190
+			// Flag to determine whether you want the results returned by FMF.  1 or 0.  Default is 0.
191
+			'returnfmfdetails' => '1',
192
+		);
193
+		$CCDetails = array(
194
+			// Required. Type of credit card.  Visa, MasterCard, Discover, Amex, Maestro, Solo.
195
+			// If Maestro or Solo, the currency code must be GBP.
196
+			//  In addition, either start date or issue number must be specified.
197
+			'creditcardtype' => $billing_info['credit_card_type'],
198
+			// Required.  Credit card number.  No spaces or punctuation.
199
+			'acct'           => $billing_info['credit_card'],
200
+			// Required.  Credit card expiration date.  Format is MMYYYY
201
+			'expdate'        => $billing_info['exp_month'] . $billing_info['exp_year'],
202
+			// Requirements determined by your PayPal account settings.  Security digits for credit card.
203
+			'cvv2'           => $billing_info['cvv'],
204
+		);
205
+		$PayerInfo = array(
206
+			// Email address of payer.
207
+			'email'       => $billing_info['email'],
208
+			// Unique PayPal customer ID for payer.
209
+			'payerid'     => '',
210
+			// Status of payer.  Values are verified or unverified
211
+			'payerstatus' => '',
212
+			// Payer's business name.
213
+			'business'    => '',
214
+		);
215
+		$PayerName = array(
216
+			// Payer's salutation.  20 char max.
217
+			'salutation' => '',
218
+			// Payer's first name.  25 char max.
219
+			'firstname'  => substr($billing_info['first_name'], 0, 25),
220
+			// Payer's middle name.  25 char max.
221
+			'middlename' => '',
222
+			// Payer's last name.  25 char max.
223
+			'lastname'   => substr($billing_info['last_name'], 0, 25),
224
+			// Payer's suffix.  12 char max.
225
+			'suffix'     => '',
226
+		);
227
+		$BillingAddress = array(
228
+			// Required.  First street address.
229
+			'street'      => $billing_info['address'],
230
+			// Second street address.
231
+			'street2'     => $billing_info['address2'],
232
+			// Required.  Name of City.
233
+			'city'        => $billing_info['city'],
234
+			// Required. Name of State or Province.
235
+			'state'       => substr($billing_info['state'], 0, 40),
236
+			// Required.  Country code.
237
+			'countrycode' => $billing_info['country'],
238
+			// Required.  Postal code of payer.
239
+			'zip'         => $billing_info['zip'],
240
+		);
241
+		// check if the registration info contains the needed fields for paypal pro
242
+		// (see https://developer.paypal.com/docs/classic/api/merchant/DoDirectPayment_API_Operation_NVP/)
243
+		if ($attendee->address() && $attendee->city() && $attendee->country_ID()) {
244
+			$use_registration_address_info = true;
245
+		} else {
246
+			$use_registration_address_info = false;
247
+		}
248
+		// so if the attendee has enough data to fill out PayPal Pro's shipping info, use it.
249
+		// If not, use the billing info again
250
+		$ShippingAddress = array(
251
+			'shiptoname'     => substr($use_registration_address_info
252
+				? $attendee->full_name()
253
+				: $billing_info['first_name'] . ' ' . $billing_info['last_name'], 0, 32),
254
+			'shiptostreet'   => substr($use_registration_address_info
255
+				? $attendee->address()
256
+				: $billing_info['address'], 0, 100),
257
+			'shiptostreet2'  => substr($use_registration_address_info
258
+				? $attendee->address2() : $billing_info['address2'], 0, 100),
259
+			'shiptocity'     => substr($use_registration_address_info
260
+				? $attendee->city()
261
+				: $billing_info['city'], 0, 40),
262
+			'state'          => substr($use_registration_address_info
263
+				? $attendee->state_name()
264
+				: $billing_info['state'], 0, 40),
265
+			'shiptocountry'  => $use_registration_address_info
266
+				? $attendee->country_ID()
267
+				: $billing_info['country'],
268
+			'shiptozip'      => substr($use_registration_address_info
269
+				? $attendee->zip()
270
+				: $billing_info['zip'], 0, 20),
271
+			'shiptophonenum' => substr($use_registration_address_info
272
+				? $attendee->phone()
273
+				: $billing_info['phone'], 0, 20),
274
+		);
275
+		$PaymentDetails = array(
276
+			// Required.  Total amount of order, including shipping, handling, and tax.
277
+			'amt'          => $gateway_formatter->formatCurrency($payment->amount()),
278
+			// Required.  Three-letter currency code.  Default is USD.
279
+			'currencycode' => $payment->currency_code(),
280
+			// Required if you include itemized cart details. (L_AMTn, etc.)
281
+			// Subtotal of items not including S&H, or tax.
282
+			'itemamt'      => $gateway_formatter->formatCurrency($item_amount),//
283
+			// Total shipping costs for the order.  If you specify shippingamt, you must also specify itemamt.
284
+			'shippingamt'  => '',
285
+			// Total handling costs for the order.  If you specify handlingamt, you must also specify itemamt.
286
+			'handlingamt'  => '',
287
+			// Required if you specify itemized cart tax details.
288
+			// Sum of tax for all items on the order.  Total sales tax.
289
+			'taxamt'       => $gateway_formatter->formatCurrency($tax_amount),
290
+			// Description of the order the customer is purchasing.  127 char max.
291
+			'desc'         => $order_description,
292
+			// Free-form field for your own use.  256 char max.
293
+			'custom'       => $primary_registrant ? $primary_registrant->ID() : '',
294
+			// Your own invoice or tracking number
295
+			'invnum'       => wp_generate_password(12, false),// $transaction->ID(),
296
+			// URL for receiving Instant Payment Notifications.  This overrides what your profile is set to use.
297
+			'notifyurl'    => '',
298
+			'buttonsource' => 'EventEspresso_SP',// EE will blow up if you change this
299
+		);
300
+		// Wrap all data arrays into a single, "master" array which will be passed into the class function.
301
+		$PayPalRequestData = array(
302
+			'DPFields'        => $DPFields,
303
+			'CCDetails'       => $CCDetails,
304
+			'PayerInfo'       => $PayerInfo,
305
+			'PayerName'       => $PayerName,
306
+			'BillingAddress'  => $BillingAddress,
307
+			'ShippingAddress' => $ShippingAddress,
308
+			'PaymentDetails'  => $PaymentDetails,
309
+			'OrderItems'      => $order_items,
310
+		);
311
+		$this->_log_clean_request($PayPalRequestData, $payment);
312
+		try {
313
+			$PayPalResult = $this->prep_and_curl_request($PayPalRequestData);
314
+			// remove PCI-sensitive data so it doesn't get stored
315
+			$PayPalResult = $this->_log_clean_response($PayPalResult, $payment);
316
+			$message = isset($PayPalResult['L_LONGMESSAGE0']) ? $PayPalResult['L_LONGMESSAGE0'] : $PayPalResult['ACK'];
317
+			if (empty($PayPalResult['RAWRESPONSE'])) {
318
+				$payment->set_status($this->_pay_model->failed_status());
319
+				$payment->set_gateway_response(__('No response received from Paypal Pro', 'event_espresso'));
320
+				$payment->set_details($PayPalResult);
321
+			} else {
322
+				if ($this->_APICallSuccessful($PayPalResult)) {
323
+					$payment->set_status($this->_pay_model->approved_status());
324
+				} else {
325
+					$payment->set_status($this->_pay_model->declined_status());
326
+				}
327
+				// make sure we interpret the AMT as a float, not an international string
328
+				// (where periods are thousand separators)
329
+				$payment->set_amount(isset($PayPalResult['AMT']) ? floatval($PayPalResult['AMT']) : 0);
330
+				$payment->set_gateway_response($message);
331
+				$payment->set_txn_id_chq_nmbr(isset($PayPalResult['TRANSACTIONID'])
332
+					? $PayPalResult['TRANSACTIONID']
333
+					: null);
334
+				$primary_registration_code = $primary_registrant instanceof EE_Registration
335
+					? $primary_registrant->reg_code()
336
+					: '';
337
+				$payment->set_extra_accntng($primary_registration_code);
338
+				$payment->set_details($PayPalResult);
339
+			}
340
+		} catch (Exception $e) {
341
+			$payment->set_status($this->_pay_model->failed_status());
342
+			$payment->set_gateway_response($e->getMessage());
343
+		}
344
+		// $payment->set_status( $this->_pay_model->declined_status() );
345
+		// $payment->set_gateway_response( '' );
346
+		return $payment;
347
+	}
348
+
349
+
350
+
351
+	/**
352
+	 * CLeans out sensitive CC data and then logs it, and returns the cleaned request
353
+	 *
354
+	 * @param array       $request
355
+	 * @param EEI_Payment $payment
356
+	 * @return void
357
+	 */
358
+	private function _log_clean_request($request, $payment)
359
+	{
360
+		$cleaned_request_data = $request;
361
+		unset($cleaned_request_data['CCDetails']['acct']);
362
+		unset($cleaned_request_data['CCDetails']['cvv2']);
363
+		unset($cleaned_request_data['CCDetails']['expdate']);
364
+		$this->log(array('Paypal Request' => $cleaned_request_data), $payment);
365
+	}
366
+
367
+
368
+
369
+	/**
370
+	 * Cleans the response, logs it, and returns it
371
+	 *
372
+	 * @param array       $response
373
+	 * @param EEI_Payment $payment
374
+	 * @return array cleaned
375
+	 */
376
+	private function _log_clean_response($response, $payment)
377
+	{
378
+		unset($response['REQUESTDATA']['CREDITCARDTYPE']);
379
+		unset($response['REQUESTDATA']['ACCT']);
380
+		unset($response['REQUESTDATA']['EXPDATE']);
381
+		unset($response['REQUESTDATA']['CVV2']);
382
+		unset($response['RAWREQUEST']);
383
+		$this->log(array('Paypal Response' => $response), $payment);
384
+		return $response;
385
+	}
386
+
387
+
388
+
389
+	/**
390
+	 * @param $DataArray
391
+	 * @return array
392
+	 */
393
+	private function prep_and_curl_request($DataArray)
394
+	{
395
+		// Create empty holders for each portion of the NVP string
396
+		$DPFieldsNVP = '&METHOD=DoDirectPayment&BUTTONSOURCE=AngellEYE_PHP_Class_DDP';
397
+		$CCDetailsNVP = '';
398
+		$PayerInfoNVP = '';
399
+		$PayerNameNVP = '';
400
+		$BillingAddressNVP = '';
401
+		$ShippingAddressNVP = '';
402
+		$PaymentDetailsNVP = '';
403
+		$OrderItemsNVP = '';
404
+		$Secure3DNVP = '';
405
+		// DP Fields
406
+		$DPFields = isset($DataArray['DPFields']) ? $DataArray['DPFields'] : array();
407
+		foreach ($DPFields as $DPFieldsVar => $DPFieldsVal) {
408
+			$DPFieldsNVP .= '&' . strtoupper($DPFieldsVar) . '=' . urlencode($DPFieldsVal);
409
+		}
410
+		// CC Details Fields
411
+		$CCDetails = isset($DataArray['CCDetails']) ? $DataArray['CCDetails'] : array();
412
+		foreach ($CCDetails as $CCDetailsVar => $CCDetailsVal) {
413
+			$CCDetailsNVP .= '&' . strtoupper($CCDetailsVar) . '=' . urlencode($CCDetailsVal);
414
+		}
415
+		// PayerInfo Type Fields
416
+		$PayerInfo = isset($DataArray['PayerInfo']) ? $DataArray['PayerInfo'] : array();
417
+		foreach ($PayerInfo as $PayerInfoVar => $PayerInfoVal) {
418
+			$PayerInfoNVP .= '&' . strtoupper($PayerInfoVar) . '=' . urlencode($PayerInfoVal);
419
+		}
420
+		// Payer Name Fields
421
+		$PayerName = isset($DataArray['PayerName']) ? $DataArray['PayerName'] : array();
422
+		foreach ($PayerName as $PayerNameVar => $PayerNameVal) {
423
+			$PayerNameNVP .= '&' . strtoupper($PayerNameVar) . '=' . urlencode($PayerNameVal);
424
+		}
425
+		// Address Fields (Billing)
426
+		$BillingAddress = isset($DataArray['BillingAddress']) ? $DataArray['BillingAddress'] : array();
427
+		foreach ($BillingAddress as $BillingAddressVar => $BillingAddressVal) {
428
+			$BillingAddressNVP .= '&' . strtoupper($BillingAddressVar) . '=' . urlencode($BillingAddressVal);
429
+		}
430
+		// Payment Details Type Fields
431
+		$PaymentDetails = isset($DataArray['PaymentDetails']) ? $DataArray['PaymentDetails'] : array();
432
+		foreach ($PaymentDetails as $PaymentDetailsVar => $PaymentDetailsVal) {
433
+			$PaymentDetailsNVP .= '&' . strtoupper($PaymentDetailsVar) . '=' . urlencode($PaymentDetailsVal);
434
+		}
435
+		// Payment Details Item Type Fields
436
+		$OrderItems = isset($DataArray['OrderItems']) ? $DataArray['OrderItems'] : array();
437
+		$n = 0;
438
+		foreach ($OrderItems as $OrderItemsVar => $OrderItemsVal) {
439
+			$CurrentItem = $OrderItems[ $OrderItemsVar ];
440
+			foreach ($CurrentItem as $CurrentItemVar => $CurrentItemVal) {
441
+				$OrderItemsNVP .= '&' . strtoupper($CurrentItemVar) . $n . '=' . urlencode($CurrentItemVal);
442
+			}
443
+			$n++;
444
+		}
445
+		// Ship To Address Fields
446
+		$ShippingAddress = isset($DataArray['ShippingAddress']) ? $DataArray['ShippingAddress'] : array();
447
+		foreach ($ShippingAddress as $ShippingAddressVar => $ShippingAddressVal) {
448
+			$ShippingAddressNVP .= '&' . strtoupper($ShippingAddressVar) . '=' . urlencode($ShippingAddressVal);
449
+		}
450
+		// 3D Secure Fields
451
+		$Secure3D = isset($DataArray['Secure3D']) ? $DataArray['Secure3D'] : array();
452
+		foreach ($Secure3D as $Secure3DVar => $Secure3DVal) {
453
+			$Secure3DNVP .= '&' . strtoupper($Secure3DVar) . '=' . urlencode($Secure3DVal);
454
+		}
455
+		// Now that we have each chunk we need to go ahead and append them all together for our entire NVP string
456
+		$NVPRequest = 'USER='
457
+					  . $this->_username
458
+					  . '&PWD='
459
+					  . $this->_password
460
+					  . '&VERSION=64.0'
461
+					  . '&SIGNATURE='
462
+					  . $this->_signature
463
+					  . $DPFieldsNVP
464
+					  . $CCDetailsNVP
465
+					  . $PayerInfoNVP
466
+					  . $PayerNameNVP
467
+					  . $BillingAddressNVP
468
+					  . $PaymentDetailsNVP
469
+					  . $OrderItemsNVP
470
+					  . $ShippingAddressNVP
471
+					  . $Secure3DNVP;
472
+		$NVPResponse = $this->_CURLRequest($NVPRequest);
473
+		$NVPRequestArray = $this->_NVPToArray($NVPRequest);
474
+		$NVPResponseArray = $this->_NVPToArray($NVPResponse);
475
+		$Errors = $this->_GetErrors($NVPResponseArray);
476
+		$NVPResponseArray['ERRORS'] = $Errors;
477
+		$NVPResponseArray['REQUESTDATA'] = $NVPRequestArray;
478
+		$NVPResponseArray['RAWREQUEST'] = $NVPRequest;
479
+		$NVPResponseArray['RAWRESPONSE'] = $NVPResponse;
480
+		return $NVPResponseArray;
481
+	}
482
+
483
+
484
+
485
+	/**
486
+	 * @param $Request
487
+	 * @return mixed
488
+	 */
489
+	private function _CURLRequest($Request)
490
+	{
491
+		$EndPointURL = $this->_debug_mode ? 'https://api-3t.sandbox.paypal.com/nvp' : 'https://api-3t.paypal.com/nvp';
492
+		$curl = curl_init();
493
+		curl_setopt($curl, CURLOPT_VERBOSE, apply_filters('FHEE__EEG_Paypal_Pro__CurlRequest__CURLOPT_VERBOSE', true));
494
+		curl_setopt($curl, CURLOPT_SSL_VERIFYPEER, false);
495
+		curl_setopt($curl, CURLOPT_TIMEOUT, 60);
496
+		curl_setopt($curl, CURLOPT_URL, $EndPointURL);
497
+		curl_setopt($curl, CURLOPT_RETURNTRANSFER, 1);
498
+		curl_setopt($curl, CURLOPT_POSTFIELDS, $Request);
499
+		curl_setopt($curl, CURLOPT_HTTP_VERSION, CURL_HTTP_VERSION_1_1);
500
+		// execute the curl POST
501
+		$Response = curl_exec($curl);
502
+		curl_close($curl);
503
+		return $Response;
504
+	}
505
+
506
+
507
+
508
+	/**
509
+	 * @param $NVPString
510
+	 * @return array
511
+	 */
512
+	private function _NVPToArray($NVPString)
513
+	{
514
+		// prepare responses into array
515
+		$proArray = array();
516
+		while (strlen($NVPString)) {
517
+			// name
518
+			$keypos = strpos($NVPString, '=');
519
+			$keyval = substr($NVPString, 0, $keypos);
520
+			// value
521
+			$valuepos = strpos($NVPString, '&') ? strpos($NVPString, '&') : strlen($NVPString);
522
+			$valval = substr($NVPString, $keypos + 1, $valuepos - $keypos - 1);
523
+			// decoding the response
524
+			$proArray[ $keyval ] = urldecode($valval);
525
+			$NVPString = substr($NVPString, $valuepos + 1, strlen($NVPString));
526
+		}
527
+		return $proArray;
528
+	}
529
+
530
+
531
+
532
+	/**
533
+	 * @param array $PayPalResult
534
+	 * @return bool
535
+	 */
536
+	private function _APICallSuccessful($PayPalResult)
537
+	{
538
+		$approved = false;
539
+		// check main response message from PayPal
540
+		if (isset($PayPalResult['ACK']) && ! empty($PayPalResult['ACK'])) {
541
+			$ack = strtoupper($PayPalResult['ACK']);
542
+			$approved = ($ack == 'SUCCESS' || $ack == 'SUCCESSWITHWARNING' || $ack == 'PARTIALSUCCESS') ? true : false;
543
+		}
544
+		return $approved;
545
+	}
546
+
547
+
548
+
549
+	/**
550
+	 * @param $DataArray
551
+	 * @return array
552
+	 */
553
+	private function _GetErrors($DataArray)
554
+	{
555
+		$Errors = array();
556
+		$n = 0;
557
+		while (isset($DataArray[ 'L_ERRORCODE' . $n . '' ])) {
558
+			$LErrorCode = isset($DataArray[ 'L_ERRORCODE' . $n . '' ]) ? $DataArray[ 'L_ERRORCODE' . $n . '' ] : '';
559
+			$LShortMessage = isset($DataArray[ 'L_SHORTMESSAGE' . $n . '' ])
560
+				? $DataArray[ 'L_SHORTMESSAGE' . $n . '' ]
561
+				: '';
562
+			$LLongMessage = isset($DataArray[ 'L_LONGMESSAGE' . $n . '' ])
563
+				? $DataArray[ 'L_LONGMESSAGE' . $n . '' ]
564
+				: '';
565
+			$LSeverityCode = isset($DataArray[ 'L_SEVERITYCODE' . $n . '' ])
566
+				? $DataArray[ 'L_SEVERITYCODE' . $n . '' ]
567
+				: '';
568
+			$CurrentItem = array(
569
+				'L_ERRORCODE'    => $LErrorCode,
570
+				'L_SHORTMESSAGE' => $LShortMessage,
571
+				'L_LONGMESSAGE'  => $LLongMessage,
572
+				'L_SEVERITYCODE' => $LSeverityCode,
573
+			);
574
+			array_push($Errors, $CurrentItem);
575
+			$n++;
576
+		}
577
+		return $Errors;
578
+	}
579
+
580
+
581
+
582
+	/**
583
+	 *        nothing to see here...  move along....
584
+	 *
585
+	 * @access protected
586
+	 * @param $Errors
587
+	 * @return string
588
+	 */
589
+	private function _DisplayErrors($Errors)
590
+	{
591
+		$error = '';
592
+		foreach ($Errors as $ErrorVar => $ErrorVal) {
593
+			$CurrentError = $Errors[ $ErrorVar ];
594
+			foreach ($CurrentError as $CurrentErrorVar => $CurrentErrorVal) {
595
+				$CurrentVarName = '';
596
+				if ($CurrentErrorVar == 'L_ERRORCODE') {
597
+					$CurrentVarName = 'Error Code';
598
+				} elseif ($CurrentErrorVar == 'L_SHORTMESSAGE') {
599
+					$CurrentVarName = 'Short Message';
600
+				} elseif ($CurrentErrorVar == 'L_LONGMESSAGE') {
601
+					$CurrentVarName = 'Long Message';
602
+				} elseif ($CurrentErrorVar == 'L_SEVERITYCODE') {
603
+					$CurrentVarName = 'Severity Code';
604
+				}
605
+				$error .= '<br />' . $CurrentVarName . ': ' . $CurrentErrorVal;
606
+			}
607
+		}
608
+		return $error;
609
+	}
610 610
 }
Please login to merge, or discard this patch.