'01', 'Feb' => '02', 'Mar' => '03', 'Apr' => '04', 'May' => '05', 'Jun' => '06', 'Jul' => '07', 'Aug' => '08', 'Sep' => '09', 'Oct' => '10', 'Nov' => '11', 'Dec' => '12'); $hour = substr($paypalDateTime, 0, 2);$minute = substr($paypalDateTime, 3, 2);$second = substr($paypalDateTime, 6, 2); $month = $months[substr($paypalDateTime, 9, 3)]; $day = (strlen($day = preg_replace("/,/" , '' , substr($paypalDateTime, 13, 2))) < 2) ? '0'.$day: $day; $year = substr($paypalDateTime, -8, 4); if (strlen($day)<2) $day = '0'.$day; return ($year . "-" . $month . "-" . $day . " " . $hour . ":" . $minute . ":" . $second); } function ipn_debug_email($message, $email_address = '', $always_send = false, $subjecttext = 'IPN DEBUG message') { static $paypal_error_counter; static $paypal_instance_id; $logfile = ''; if ($email_address == '') $email_address = (defined('MODULE_PAYMENT_PAYPAL_DEBUG_EMAIL_ADDRESS') ? MODULE_PAYMENT_PAYPAL_DEBUG_EMAIL_ADDRESS : STORE_OWNER_EMAIL_ADDRESS); if(!isset($paypal_error_counter)) $paypal_error_counter = 0; if(!isset($paypal_instance_id)) $paypal_instance_id = time() . '_' . zen_create_random_value(4); if ((defined('MODULE_PAYMENT_PAYPALWPP_DEBUGGING') && MODULE_PAYMENT_PAYPALWPP_DEBUGGING == 'Log and Email') || (defined('MODULE_PAYMENT_PAYPAL_IPN_DEBUG') && MODULE_PAYMENT_PAYPAL_IPN_DEBUG == 'Log and Email') || $always_send) { $paypal_error_counter ++; zen_mail(STORE_OWNER, $email_address, $subjecttext . ' (' . $paypal_instance_id . ') #' . $paypal_error_counter, $message, STORE_OWNER, STORE_OWNER_EMAIL_ADDRESS, array('EMAIL_MESSAGE_HTML'=>$message), 'debug'); } if ((defined('MODULE_PAYMENT_PAYPAL_IPN_DEBUG') && (MODULE_PAYMENT_PAYPAL_IPN_DEBUG == 'Log and Email' || MODULE_PAYMENT_PAYPAL_IPN_DEBUG == 'Log File' || MODULE_PAYMENT_PAYPAL_IPN_DEBUG == 'Yes')) || (defined('MODULE_PAYMENT_PAYPALWPP_DEBUGGING') && (MODULE_PAYMENT_PAYPALWPP_DEBUGGING == 'Log File' || MODULE_PAYMENT_PAYPALWPP_DEBUGGING == 'Log and Email'))) $logfile = ipn_add_error_log($message, $paypal_instance_id); return $logfile; } function ipn_get_stored_session($session_stuff) { global $db; if (!is_array($session_stuff)) { ipn_debug_email('IPN FATAL ERROR :: Could not find Zen Cart custom variable in POST, cannot validate or re-create session as a transaction initiated from this store. Might be from another source such as eBay or another PayPal store using this PayPal account.'); return false; } $sql = "SELECT * FROM " . TABLE_PAYPAL_SESSION . " WHERE session_id = :sessionID"; $sql = $db->bindVars($sql, ':sessionID', $session_stuff[1], 'string'); $stored_session = $db->Execute($sql); if ($stored_session->recordCount() < 1) { global $isECtransaction, $isDPtransaction; if ($_POST['payment_type'] == 'instant' && $isDPtransaction && ((isset($_POST['auth_status']) && $_POST['auth_status'] == 'Completed') || $_POST['payment_status'] == 'Completed')) { $session_stuff[1] = '(EC/DP transaction)'; } ipn_debug_email('IPN ERROR :: Could not find stored session {' . $session_stuff[1] . '} in DB; thus cannot validate or re-create session as a transaction awaiting PayPal Website Payments Standard confirmation initiated by this store. Might be an Express Checkout or eBay transaction or some other action that triggers PayPal IPN notifications.'); return false; } $_SESSION = unserialize(base64_decode($stored_session->fields['saved_session'])); return true; } /** * look up parent/original transaction record data and return matching order info if found, along with txn_type */ function ipn_lookup_transaction($postArray) { global $db; // find Zen Cart order number from the transactionID in the IPN $ordersID = 0; $paypalipnID = 0; $transType = 'unknown'; $sql = "SELECT order_id, paypal_ipn_id, payment_status, txn_type, pending_reason FROM " . TABLE_PAYPAL . " WHERE txn_id = :transactionID: OR invoice = :transactionID: ORDER BY order_id DESC LIMIT 1 "; $sqlParent = $db->bindVars($sql, ':transactionID:', $postArray['parent_txn_id'], 'string'); $sqlTxn = $db->bindVars($sql, ':transactionID:', $postArray['txn_id'], 'string'); if (isset($postArray['parent_txn_id']) && trim($postArray['parent_txn_id']) != '') { $ipn_id = $db->Execute($sqlParent); if($ipn_id->RecordCount() > 0) { ipn_debug_email('IPN NOTICE :: This transaction HAS a parent record. Thus this is an update of some sort.'); $transType = ($ipn_id->fields['pending_reason'] == 'paymentreview') ? 'reviewed' : 'parent'; $ordersID = $ipn_id->fields['order_id']; $paypalipnID = $ipn_id->fields['paypal_ipn_id']; } } else { $ipn_id = $db->Execute($sqlTxn); if ($ipn_id->RecordCount() <= 0) { ipn_debug_email('IPN NOTICE :: Could not find matched txn_id record in DB. Therefore is new to us. '); $transType = 'unique'; } else { while(!$ipn_id->EOF) { switch ($ipn_id->fields['pending_reason']) { case 'address': ipn_debug_email('IPN NOTICE :: Found pending-address record in database'); if ($postArray['payment_status'] == 'Completed') $transType = 'cleared-address'; if ($postArray['payment_status'] == 'Denied') $transType = 'denied-address'; if ($postArray['payment_status'] == 'Pending') $transType = 'pending-address'; break; case 'multi_currency': ipn_debug_email('IPN NOTICE :: Found pending-multicurrency record in database'); if ($postArray['payment_status'] == 'Completed') $transType = 'cleared-multicurrency'; if ($postArray['payment_status'] == 'Denied') $transType = 'denied-multicurrency'; if ($postArray['payment_status'] == 'Pending') $transType = 'pending-multicurrency'; break; case 'echeck': ipn_debug_email('IPN NOTICE :: Found pending-echeck record in database'); if ($postArray['payment_status'] == 'Completed') $transType = 'cleared-echeck'; if ($postArray['payment_status'] == 'Completed' && $postArray['txn_type'] == 'web_accept') $transType = 'cleared-echeck'; if ($postArray['payment_status'] == 'Denied') $transType = 'denied-echeck'; if ($postArray['payment_status'] == 'Failed') $transType = 'failed-echeck'; if ($postArray['payment_status'] == 'Pending') $transType = 'pending-echeck'; break; case 'authorization': ipn_debug_email('IPN NOTICE :: Found pending-authorization record in database'); $transType = 'cleared-authorization'; if ($postArray['payment_status'] == 'Voided') $transType = 'voided'; if ($postArray['payment_status'] == 'Pending') $transType = 'pending-authorization'; if ($postArray['payment_status'] == 'Captured') $transType = 'captured'; if ($postArray['payment_status'] == 'Completed') $transType = 'cleared-authorization'; if ($postArray['auth_status'] == 'In_Progress') $transType = 'partial-authorization'; break; case 'verify': ipn_debug_email('IPN NOTICE :: Found pending-verify record in database'); $transType = 'cleared-verify'; break; case 'paymentreview': ipn_debug_email('IPN NOTICE :: Found pending-review record in database'); $transType = 'pending-paymentreview'; if ($postArray['payment_status'] == 'Completed') $transType = 'cleared-review'; break; case 'intl': ipn_debug_email('IPN NOTICE :: Found pending-intl record in database'); if ($postArray['payment_status'] == 'Completed') $transType = 'cleared-intl'; if ($postArray['payment_status'] == 'Denied') $transType = 'denied-intl'; if ($postArray['payment_status'] == 'Pending') $transType = 'pending-intl'; break; case 'unilateral': ipn_debug_email('IPN NOTICE :: Found record in database.' . "\n" . '*** NOTE: TRANSACTION IS IN *unilateral* STATUS pending creation of a PayPal account for this receiver_email address.' . "\n" . 'Please create the account, or make sure the account is *Verified*.'); $transType = 'pending-unilateral'; break; } if ($transType != 'unknown') { $ordersID = $ipn_id->fields['order_id']; $paypalipnID = $ipn_id->fields['paypal_ipn_id']; } $ipn_id->MoveNext(); } } } return array('order_id' => $ordersID, 'paypal_ipn_id' => $paypalipnID, 'txn_type' => $transType); } /** * IPN Validation * - match email addresses * - ensure that "VERIFIED" has been returned (otherwise somebody is trying to spoof) */ function ipn_validate_transaction($info, $postArray, $mode='IPN') { if ($mode == 'IPN' && !preg_match("/VERIFIED/i", $info) && !preg_match("/SUCCESS/i", $info)) { ipn_debug_email('IPN WARNING :: Transaction was NOT marked as VERIFIED. Keep this report for potential use in fraud investigations.' . "\n" . 'IPN Info: ' . "\n" . $info); return false; } elseif ($mode == 'PDT' && (!preg_match("/SUCCESS/i", $info) || preg_match("/FAIL/i", $info))) { ipn_debug_email('IPN WARNING :: PDT Transaction was NOT marked as SUCCESS. Keep this report for potential use in fraud investigations.' . "\n" . 'IPN Info: ' . "\n" . $info); return false; } $ppBusEmail = false; $ppRecEmail = false; if (defined('MODULE_PAYMENT_PAYPAL_BUSINESS_ID')) { if (strtolower(trim($postArray['business'])) == strtolower(trim(MODULE_PAYMENT_PAYPAL_BUSINESS_ID))) $ppBusEmail = true; if (strtolower(trim($postArray['receiver_email'])) == strtolower(trim(MODULE_PAYMENT_PAYPAL_BUSINESS_ID))) $ppRecEmail = true; if (!$ppBusEmail && !$ppRecEmail) { ipn_debug_email('IPN WARNING :: Transaction email address NOT matched.' . "\n" . 'From IPN = ' . $postArray['business'] . ' | ' . $postArray['receiver_email'] . "\n" . 'From CONFIG = ' . MODULE_PAYMENT_PAYPAL_BUSINESS_ID); return false; } ipn_debug_email('IPN INFO :: Transaction email details.' . "\n" . 'From IPN = ' . $postArray['business'] . ' | ' . $postArray['receiver_email'] . "\n" . 'From CONFIG = ' . MODULE_PAYMENT_PAYPAL_BUSINESS_ID); } return true; } // determine acceptable currencies function select_pp_currency() { if (MODULE_PAYMENT_PAYPAL_CURRENCY == 'Selected Currency') { $my_currency = $_SESSION['currency']; } else { $my_currency = substr(MODULE_PAYMENT_PAYPAL_CURRENCY, 5); } $pp_currencies = array('CAD', 'EUR', 'GBP', 'JPY', 'USD', 'AUD', 'CHF', 'CZK', 'DKK', 'HKD', 'HUF', 'NOK', 'NZD', 'PLN', 'SEK', 'SGD', 'THB', 'MXN', 'ILS', 'PHP', 'TWD', 'BRL', 'MYR'); if (!in_array($my_currency, $pp_currencies)) { $my_currency = 'USD'; } return $my_currency; } function valid_payment($amount, $currency, $mode = 'IPN') { global $currencies; $my_currency = select_pp_currency(); $exchanged_amount = ($mode == 'IPN' ? ($amount * $currencies->get_value($my_currency)) : $amount); $transaction_amount = preg_replace('/[^0-9.]/', '', number_format($exchanged_amount, $currencies->get_decimal_places($my_currency), '.', '')); if ( ($_POST['mc_currency'] != $my_currency) || ($_POST['mc_gross'] != $transaction_amount && $_POST['mc_gross'] != -0.01) && MODULE_PAYMENT_PAYPAL_TESTING != 'Test' ) { ipn_debug_email('IPN WARNING :: Currency/Amount Mismatch. Details: ' . "\n" . 'PayPal email address = ' . $_POST['business'] . "\n" . ' | mc_currency = ' . $_POST['mc_currency'] . "\n" . ' | submitted_currency = ' . $my_currency . "\n" . ' | order_currency = ' . $currency . "\n" . ' | mc_gross = ' . $_POST['mc_gross'] . "\n" . ' | converted_amount = ' . $transaction_amount . "\n" . ' | order_amount = ' . $amount ); return false; } ipn_debug_email('IPN INFO :: Currency/Amount Details: ' . "\n" . 'PayPal email address = ' . $_POST['business'] . "\n" . ' | mc_currency = ' . $_POST['mc_currency'] . "\n" . ' | submitted_currency = ' . $my_currency . "\n" . ' | order_currency = ' . $currency . "\n" . ' | mc_gross = ' . $_POST['mc_gross'] . "\n" . ' | converted_amount = ' . $transaction_amount . "\n" . ' | order_amount = ' . $amount ); return true; } /** * is this an existing transaction? * (1) we find a matching record in the "paypal" table * (2) we check for valid txn_types or payment_status such as Denied, Refunded, Partially-Refunded, Reversed, Voided, Expired */ function ipn_determine_txn_type($postArray, $txn_type = 'unknown') { global $db, $parentLookup; if (substr($txn_type,0,8) == 'cleared-') return $txn_type; if ($postArray['txn_type'] == 'send_money') return $postArray['txn_type']; if ($postArray['txn_type'] == 'express_checkout' || $postArray['txn_type'] == 'cart') $txn_type = $postArray['txn_type']; // if it's not unique or linked to a parent, then: // 1. could be an e-check denied / cleared // 2. could be an express-checkout "pending" transaction which has been Accepted in the merchant's PayPal console and needs activation in Zen Cart if ($postArray['payment_status']=='Completed' && txn_type=='express_checkout' && $postArray['payment_type']=='echeck') { $txn_type = 'express-checkout-cleared'; return $txn_type; } if ($postArray['payment_status']=='Completed' && $postArray['payment_type']=='echeck') { $txn_type = 'echeck-cleared'; return $txn_type; } if (($postArray['payment_status']=='Denied' || $postArray['payment_status']=='Failed') && $postArray['payment_type']=='echeck') { $txn_type = 'echeck-denied'; return $txn_type; } if ($postArray['payment_status']=='Denied') { $txn_type = 'denied'; return $txn_type; } if (($postArray['payment_status']=='Pending') && $postArray['pending_reason']=='echeck') { $txn_type = 'pending-echeck'; return $txn_type; } if (($postArray['payment_status']=='Pending') && $postArray['pending_reason']=='address') { $txn_type = 'pending-address'; return $txn_type; } if (($postArray['payment_status']=='Pending') && $postArray['pending_reason']=='intl') { $txn_type = 'pending-intl'; return $txn_type; } if (($postArray['payment_status']=='Pending') && $postArray['pending_reason']=='multi_currency') { $txn_type = 'pending-multicurrency'; return $txn_type; } if (($postArray['payment_status']=='Pending') && $postArray['pending_reason']=='paymentreview') { $txn_type = 'pending-paymentreview'; return $txn_type; } if (($postArray['payment_status']=='Pending') && $postArray['pending_reason']=='verify') { $txn_type = 'pending-verify'; return $txn_type; } if ($parentLookup == 'parent' && $postArray['payment_status']=='Completed' && $postArray['payment_type']=='instant') { $txn_type = 'cleared-authorization'; return $txn_type; } if (($postArray['payment_status']=='Voided') && $postArray['payment_type']=='instant') { $txn_type = 'voided'; return $txn_type; } return $txn_type; } /** * Create order record from IPN data */ function ipn_create_order_array($new_order_id, $txn_type) { $sql_data_array = array('order_id' => $new_order_id, 'txn_type' => $txn_type, 'module_name' => 'paypal (ipn-handler)', 'module_mode' => 'IPN', 'reason_code' => $_POST['reason_code'], 'payment_type' => $_POST['payment_type'], 'payment_status' => $_POST['payment_status'], 'pending_reason' => $_POST['pending_reason'], 'invoice' => $_POST['invoice'], 'mc_currency' => $_POST['mc_currency'], 'first_name' => $_POST['first_name'], 'last_name' => $_POST['last_name'], 'payer_business_name' => $_POST['payer_business_name'], 'address_name' => $_POST['address_name'], 'address_street' => $_POST['address_street'], 'address_city' => $_POST['address_city'], 'address_state' => $_POST['address_state'], 'address_zip' => $_POST['address_zip'], 'address_country' => $_POST['address_country'], 'address_status' => $_POST['address_status'], 'payer_email' => $_POST['payer_email'], 'payer_id' => $_POST['payer_id'], 'payer_status' => $_POST['payer_status'], 'payment_date' => datetime_to_sql_format($_POST['payment_date']), 'business' => $_POST['business'], 'receiver_email' => $_POST['receiver_email'], 'receiver_id' => $_POST['receiver_id'], 'txn_id' => $_POST['txn_id'], 'parent_txn_id' => $_POST['parent_txn_id'], 'num_cart_items' => (int)$_POST['num_cart_items'], 'mc_gross' => $_POST['mc_gross'], 'mc_fee' => $_POST['mc_fee'], 'settle_amount' => (isset($_POST['settle_amount']) && $_POST['settle_amount'] != '' ? $_POST['settle_amount'] : 0), 'settle_currency' => $_POST['settle_currency'], 'exchange_rate' => (isset($_POST['exchange_rate']) && $_POST['exchange_rate'] != '' ? $_POST['exchange_rate'] : 1), 'notify_version' => $_POST['notify_version'], 'verify_sign' => $_POST['verify_sign'], 'date_added' => 'now()', 'memo' => '{Record generated by IPN}' ); if (isset($_POST['protection_eligibility']) && $_POST['protection_eligibility'] != '') $sql_data_array['memo'] .= ' [ProtectionEligibility:' . $_POST['protection_eligibility'] .']'; if (isset($_POST['memo']) && $_POST['memo'] != '') $sql_data_array['memo'] .= ' [Customer Comments:' . $_POST['memo'] .']'; return $sql_data_array; } /** * Create order-history record from IPN data */ function ipn_create_order_history_array($insert_id) { $sql_data_array = array ('paypal_ipn_id' => $insert_id, 'txn_id' => $_POST['txn_id'], 'parent_txn_id' => $_POST['parent_txn_id'], 'payment_status' => $_POST['payment_status'], 'pending_reason' => $_POST['pending_reason'], 'date_added' => 'now()' ); return $sql_data_array; } /** * Create order-update from IPN data */ function ipn_create_order_update_array($txn_type) { $sql_data_array = array('payment_type' => $_POST['payment_type'], 'txn_type' => $txn_type, 'parent_txn_id' => $_POST['parent_txn_id'], 'payment_status' => $_POST['payment_status'], 'pending_reason' => $_POST['pending_reason'], 'payer_email' => $_POST['payer_email'], 'payer_id' => $_POST['payer_id'], 'business' => $_POST['business'], 'receiver_email' => $_POST['receiver_email'], 'receiver_id' => $_POST['receiver_id'], 'notify_version' => $_POST['notify_version'], 'verify_sign' => $_POST['verify_sign'], 'last_modified' => 'now()' ); if (isset($_POST['address_street']) && $_POST['address_street'] != '') $sql_data_array = array_merge($sql_data_array, array('address_name' => $_POST['address_name'], 'address_street' => $_POST['address_street'], 'address_city' => $_POST['address_city'], 'address_state' => $_POST['address_state'], 'address_zip' => $_POST['address_zip'], 'address_country' => $_POST['address_country'])); if (isset($_POST['payer_business_name']) && $_POST['payer_business_name'] != '') $sql_data_array['payer_business_name'] = $_POST['payer_business_name']; if (isset($_POST['reason_code']) && $_POST['reason_code'] != '') $sql_data_array['reason_code'] = $_POST['reason_code']; if (isset($_POST['invoice']) && $_POST['invoice'] != '') $sql_data_array['invoice'] = $_POST['invoice']; if (isset($_POST['mc_gross']) && $_POST['mc_gross'] > 0) $sql_data_array['mc_gross'] = $_POST['mc_gross']; if (isset($_POST['mc_fee']) && $_POST['mc_fee'] > 0) $sql_data_array['mc_fee'] = $_POST['mc_fee']; if (isset($_POST['settle_amount']) && $_POST['settle_amount'] > 0) $sql_data_array['settle_amount'] = $_POST['settle_amount']; if (isset($_POST['first_name']) && $_POST['first_name'] != '') $sql_data_array['first_name'] = $_POST['first_name']; if (isset($_POST['last_name']) && $_POST['last_name'] != '') $sql_data_array['last_name'] = $_POST['last_name']; if (isset($_POST['mc_currency']) && $_POST['mc_currency'] != '') $sql_data_array['mc_currency'] = $_POST['mc_currency']; if (isset($_POST['settle_currency']) && $_POST['settle_currency'] != '') $sql_data_array['settle_currency'] = $_POST['settle_currency']; if (isset($_POST['num_cart_items']) && $_POST['num_cart_items'] > 0) $sql_data_array['num_cart_items'] = $_POST['num_cart_items']; if (isset($_POST['exchange_rate']) && $_POST['exchange_rate'] > 0) $sql_data_array['exchange_rate'] = $_POST['exchange_rate']; $sql_data_array['memo'] = '{Record generated by IPN}'; if (isset($_POST['protection_eligibility']) && $_POST['protection_eligibility'] != '') $sql_data_array['memo'] .= ' [ProtectionEligibility:' . $_POST['protection_eligibility'] .']'; if (isset($_POST['memo']) && $_POST['memo'] != '') $sql_data_array['memo'] .= ' [Customer Comments:' . $_POST['memo'] .']'; return $sql_data_array; } /** * Debug to file */ function ipn_fopen($filename) { $response = ''; $fp = @fopen($filename,'rb'); if ($fp) { $response = getRequestBodyContents($fp); fclose($fp); } return $response; } function getRequestBodyContents(&$handle) { if ($handle) { while(!feof($handle)) { $line .= @fgets($handle, 1024); } return $line; } return false; } /** * Verify IPN by sending it back to PayPal for confirmation */ function ipn_postback($mode = 'IPN', $pdtTX = '') { $postdata = ''; $postback = ''; $postback_array = array(); //build postback string if ($mode == 'PDT') { if ($pdtTX == '') return FALSE; // TX value not supplied, therefore PDT is disabled on merchant's PayPal profile. ipn_debug_email('PDT PROCESSING INITIATED.' . "\n" . 'Preparing to verify transaction via PDT.' . "\n\n" . 'The TX token for verification is: ' . print_r($_GET, TRUE)); $postback .= "cmd=_notify-synch"; $postback .= "&tx=" . $_GET['tx']; $postback .= "&at=" . trim(MODULE_PAYMENT_PAYPAL_PDTTOKEN); $postback .= "&"; $postback_array['cmd'] = "_notify-sync"; $postback_array['tx'] = $_GET['tx']; $postback_array['at'] = substr(MODULE_PAYMENT_PAYPAL_PDTTOKEN, 0, 5) . '**********' . substr(MODULE_PAYMENT_PAYPAL_PDTTOKEN,-5); } elseif ($mode == 'IPN') { $postback .= "cmd=_notify-validate"; $postback .= "&"; $postback_array['cmd'] = "_notify-validate"; } foreach($_POST as $key=>$value) { $postdata .= $key . "=" . urlencode(stripslashes($value)) . "&"; $postback .= $key . "=" . urlencode(stripslashes($value)) . "&"; $postback_array[$key] = $value; } if (substr($postdata, -2) == '=&') { ipn_debug_email('IPN NOTICE :: No POST data to process -- Bad IPN data'); return $postdata; } $postback = rtrim($postback, '&'); $postdata = rtrim($postdata, '&'); $postdata_array = $_POST; ksort($postdata_array); if ($mode == 'IPN') { ipn_debug_email('IPN INFO - POST VARS received (sorted):' . "\n" . stripslashes(urldecode(print_r($postdata_array, true)))); if (sizeof($postdata_array) == 0) die('Nothing to process. Please return to home page.'); } // send received data back to PayPal for validation $scheme = 'http://'; //Parse url $web = parse_url($scheme . (defined('MODULE_PAYMENT_PAYPAL_HANDLER') ? MODULE_PAYMENT_PAYPAL_HANDLER : 'www.paypal.com/cgi-bin/webscr')); if (isset($_POST['test_ipn']) && $_POST['test_ipn'] == 1) { $web = parse_url($scheme . 'www.sandbox.paypal.com/cgi-bin/webscr'); } //Set the port number if($web['scheme'] == "https") { $web['port']="443"; $ssl = "ssl://"; } else { $web['port']="80"; $ssl = ""; } $result = ''; if (function_exists('curl_init')) { $result = doPayPalIPNCurlPostback($web, $postback, $postback_array, $mode); } if ($mode == 'PDT') { $info = $result['info']; $result = $result['status']; } //DEBUG ONLY: ipn_debug_email('After CURL: $result='.$result); if (!in_array(trim($result), array('VERIFIED', 'SUCCESS', 'INVALID', 'FAIL'))) { ipn_debug_email('IPN NOTICE: Could not get usable response via CURL. Trying fsockopen() as fallback.' . ($result != '' ? ' ['.$result.']' : '')); $result = doPayPalIPNFsockopenPostback($web, $postback, $postback_array, $ssl, $mode); if ($mode == 'PDT') { $info = $result['info']; $result = $result['status']; } } return ($mode == 'PDT') ? array('status' => $result, 'info' => $info) : trim($result); } function doPayPalIPNFsockopenPostback($web, $postback, $postback_array, $ssl, $mode = 'IPN') { $header = "POST " . $web['path'] . " HTTP/1.1\r\n"; $header .= "Host: " . $web['host'] . "\r\n"; $header .= "Content-type: application/x-www-form-urlencoded\r\n"; $header .= "Content-length: " . strlen($postback) . "\r\n"; $header .= "Connection: close\r\n\r\n"; $errnum = 0; $errstr = ''; ipn_debug_email('IPN INFO - POST VARS to be sent back (unsorted) for validation (using fsockopen): ' . "\n" . 'To: ' . $ssl . $web['host'] . ':' . $web['port'] . "\n" . $header . stripslashes(print_r($postback_array, true))); //Create paypal connection if (MODULE_PAYMENT_PAYPAL_IPN_DEBUG == 'Yes') { $fp=fsockopen($ssl . $web['host'], $web['port'], $errnum, $errstr, 30); } else { $fp=@fsockopen($ssl . $web['host'], $web['port'], $errnum, $errstr, 30); } if(!$fp && $ssl == 'ssl://') { ipn_debug_email('IPN ERROR :: Could not establish fsockopen: ' . "\n" . 'Host Details = ' . $ssl . $web['host'] . ':' . $web['port'] . ' (' . $errnum . ') ' . $errstr . "\n Trying again with HTTPS over 443 ..."); $ssl = 'https://'; $web['port'] = '443'; $fp=@fsockopen($ssl . $web['host'], $web['port'], $errnum, $errstr, 30); } if(!$fp && $ssl == 'https://') { ipn_debug_email('IPN ERROR :: Could not establish fsockopen: ' . "\n" . 'Host Details = ' . $ssl . $web['host'] . ':' . $web['port'] . ' (' . $errnum . ') ' . $errstr . "\n Trying again directly over 443 ..."); $ssl = ''; $web['port'] = '443'; $fp=@fsockopen($ssl . $web['host'], $web['port'], $errnum, $errstr, 30); } if(!$fp) { ipn_debug_email('IPN ERROR :: Could not establish fsockopen: ' . "\n" . 'Host Details = ' . $ssl . $web['host'] . ':' . $web['port'] . ' (' . $errnum . ') ' . $errstr . "\n Trying again with HTTP over port 80 ..."); $ssl = 'http://'; $web['port'] = '80'; $fp=@fsockopen($ssl . $web['host'], $web['port'], $errnum, $errstr, 30); } if(!$fp) { ipn_debug_email('IPN ERROR :: Could not establish fsockopen: ' . "\n" . 'Host Details = ' . $ssl . $web['host'] . ':' . $web['port'] . ' (' . $errnum . ') ' . $errstr . "\n Trying again without any specified protocol, using port 80 ..."); $ssl = ''; $web['port'] = '80'; $fp=@fsockopen($ssl . $web['host'], $web['port'], $errnum, $errstr, 30); } if(!$fp) { ipn_debug_email('IPN FATAL ERROR :: Could not establish fsockopen. ' . "\n" . 'Host Details = ' . $ssl . $web['host'] . ':' . $web['port'] . ' (' . $errnum . ') ' . $errstr . "\nABORTED."); die(); } $info = array(); fputs($fp, $header . $postback . "\r\n\r\n"); $header_data = ''; $headerdone = false; //loop through the response from the server while(!feof($fp)) { $line = @fgets($fp, 1024); if (strcmp($line, "\r\n") == 0) { // this is a header row $headerdone = true; $header_data .= $line; } else if ($headerdone) { // header has been read. now read the contents $info[] = $line; } } //close $fp - we are done with it fclose($fp); //break up results into a string $info = implode("", $info); $firstline = trim(substr($info, 0, 20)); $status = ''; if ($status == '' && substr($firstline, 0, 8) == 'VERIFIED') $status = 'VERIFIED'; if ($status == '' && substr($firstline, 0, 7) == 'SUCCESS') $status = 'SUCCESS'; if ($status == '' && substr($firstline, 0, 4) == 'FAIL') $status = 'FAIL'; if ($status == '' && substr($firstline, 0, 7) == 'INVALID') $status = 'INVALID'; if ($status == '' && substr($firstline, 0, 12) == 'UNDETERMINED') $status = 'UNDETERMINED'; ipn_debug_email('IPN INFO (fs) - Confirmation/Validation response ' . ($status != '' ? $status : $header_data . $info)); return ($mode == 'PDT') ? array('status' => $status, 'info' => $info) : $status; } function doPayPalIPNCurlPostback($url, $vars, $varsArray, $mode = 'IPN') { ipn_debug_email('IPN INFO - POST VARS to be sent back (unsorted) for validation (using CURL): ' . "\n" . 'To: ' . $url['host'] . ':' . $url['port'] . "\n" . stripslashes(print_r($varsArray, true))); $curlOpts = array(CURLOPT_URL => 'https://' . $url['host'] . $url['path'], CURLOPT_POST => TRUE, CURLOPT_POSTFIELDS => $vars, CURLOPT_TIMEOUT => 45, CURLOPT_CONNECTTIMEOUT => 30, CURLOPT_VERBOSE => FALSE, CURLOPT_HEADER => FALSE, CURLOPT_FOLLOWLOCATION => FALSE, CURLOPT_RETURNTRANSFER => TRUE, CURLOPT_SSL_VERIFYPEER => FALSE, CURLOPT_SSL_VERIFYHOST => 2, CURLOPT_FORBID_REUSE => TRUE, CURLOPT_FRESH_CONNECT => TRUE, CURLOPT_HTTP_VERSION => CURL_HTTP_VERSION_1_1, CURLOPT_USERAGENT => 'Zen Cart(R) - IPN Postback', ); if (CURL_PROXY_REQUIRED == 'True') { $proxy_tunnel_flag = (defined('CURL_PROXY_TUNNEL_FLAG') && strtoupper(CURL_PROXY_TUNNEL_FLAG) == 'FALSE') ? false : true; $curlOpts[CURLOPT_HTTPPROXYTUNNEL] = $proxy_tunnel_flag; $curlOpts[CURLOPT_PROXYTYPE] = CURLPROXY_HTTP; $curlOpts[CURLOPT_PROXY] = CURL_PROXY_SERVER_DETAILS; } $ch = curl_init(); curl_setopt_array($ch, $curlOpts); $response = curl_exec($ch); $commError = curl_error($ch); $commErrNo = curl_errno($ch); $commInfo = @curl_getinfo($ch); curl_close($ch); $errors = ($commErrNo != 0 ? "CURL communication ERROR: (" . $commErrNo . ') ' . $commError : ''); $response .= ($commErrNo != 0 ? '&CURL_ERRORS=' . urlencode('(' . $commErrNo . ') ' . $commError) : '') ; // $response .= ($commErrNo != 0 ? '&CURL_INFO=' . urlencode($commInfo) : ''); ipn_debug_email('CURL OPTS: ' . print_r($curlOpts, true)); ipn_debug_email('CURL response: ' . $response); if ($errors != '') { ipn_debug_email('CURL errors: ' . $errors, print_r($commInfo, true)); } //echo 'INFO:
'; print_r($commInfo); echo '