'paypal_payer_id', self::PAYER_EMAIL => 'paypal_payer_email', self::PAYER_STATUS => 'paypal_payer_status', self::ADDRESS_ID => 'paypal_address_id', self::ADDRESS_STATUS => 'paypal_address_status', self::PROTECTION_EL => 'paypal_protection_eligibility', self::FRAUD_FILTERS => 'paypal_fraud_filters', self::CORRELATION_ID => 'paypal_correlation_id', self::AVS_CODE => 'paypal_avs_code', self::CVV2_MATCH => 'paypal_cvv2_match', self::CENTINEL_VPAS => self::CENTINEL_VPAS, self::CENTINEL_ECI => self::CENTINEL_ECI, self::BUYER_TAX_ID => self::BUYER_TAX_ID, self::BUYER_TAX_ID_TYPE => self::BUYER_TAX_ID_TYPE, ); /** * System information map * * @var array */ protected $_systemMap = array( self::PAYMENT_STATUS => self::PAYMENT_STATUS_GLOBAL, self::PENDING_REASON => self::PENDING_REASON_GLOBAL, self::IS_FRAUD => self::IS_FRAUD_GLOBAL, ); /** * PayPal payment status possible values * * @var string */ const PAYMENTSTATUS_NONE = 'none'; const PAYMENTSTATUS_COMPLETED = 'completed'; const PAYMENTSTATUS_DENIED = 'denied'; const PAYMENTSTATUS_EXPIRED = 'expired'; const PAYMENTSTATUS_FAILED = 'failed'; const PAYMENTSTATUS_INPROGRESS = 'in_progress'; const PAYMENTSTATUS_PENDING = 'pending'; const PAYMENTSTATUS_REFUNDED = 'refunded'; const PAYMENTSTATUS_REFUNDEDPART = 'partially_refunded'; const PAYMENTSTATUS_REVERSED = 'reversed'; const PAYMENTSTATUS_UNREVERSED = 'canceled_reversal'; const PAYMENTSTATUS_PROCESSED = 'processed'; const PAYMENTSTATUS_VOIDED = 'voided'; /** * Map of payment information available to customer * * @var array */ protected $_paymentPublicMap = array( 'paypal_payer_email', self::BUYER_TAX_ID, self::BUYER_TAX_ID_TYPE ); /** * Rendered payment map cache * * @var array */ protected $_paymentMapFull = array(); /** * All available payment info getter * * @param Mage_Payment_Model_Info $payment * @param bool $labelValuesOnly * @return array */ public function getPaymentInfo(Mage_Payment_Model_Info $payment, $labelValuesOnly = false) { // collect paypal-specific info $result = $this->_getFullInfo(array_values($this->_paymentMap), $payment, $labelValuesOnly); // add last_trans_id $label = Mage::helper('paypal')->__('Last Transaction ID'); $value = $payment->getLastTransId(); if ($labelValuesOnly) { $result[$label] = $value; } else { $result['last_trans_id'] = array('label' => $label, 'value' => $value); } return $result; } /** * Public payment info getter * * @param Mage_Payment_Model_Info $payment * @param bool $labelValuesOnly * @return array */ public function getPublicPaymentInfo(Mage_Payment_Model_Info $payment, $labelValuesOnly = false) { return $this->_getFullInfo($this->_paymentPublicMap, $payment, $labelValuesOnly); } /** * Grab data from source and map it into payment * * @param array|Varien_Object|callback $from * @param Mage_Payment_Model_Info $payment */ public function importToPayment($from, Mage_Payment_Model_Info $payment) { $fullMap = array_merge($this->_paymentMap, $this->_systemMap); if (is_object($from)) { $from = array($from, 'getDataUsingMethod'); } Varien_Object_Mapper::accumulateByMap($from, array($payment, 'setAdditionalInformation'), $fullMap); } /** * Grab data from payment and map it into target * * @param Mage_Payment_Model_Info $payment * @param array|Varien_Object|callback $to * @param array $map * @return array|Varien_Object */ public function &exportFromPayment(Mage_Payment_Model_Info $payment, $to, array $map = null) { $fullMap = array_merge($this->_paymentMap, $this->_systemMap); Varien_Object_Mapper::accumulateByMap(array($payment, 'getAdditionalInformation'), $to, $map ? $map : array_flip($fullMap) ); return $to; } /** * Check whether the payment is in review state * * @param Mage_Payment_Model_Info $payment * @return bool */ public static function isPaymentReviewRequired(Mage_Payment_Model_Info $payment) { $paymentStatus = $payment->getAdditionalInformation(self::PAYMENT_STATUS_GLOBAL); if (self::PAYMENTSTATUS_PENDING === $paymentStatus) { $pendingReason = $payment->getAdditionalInformation(self::PENDING_REASON_GLOBAL); return !in_array($pendingReason, array('authorization', 'order')); } return false; } /** * Check whether fraud order review detected and can be reviewed * * @param Mage_Payment_Model_Info $payment * @return bool */ public static function isFraudReviewAllowed(Mage_Payment_Model_Info $payment) { return self::isPaymentReviewRequired($payment) && 1 == $payment->getAdditionalInformation(self::IS_FRAUD_GLOBAL); } /** * Check whether the payment is completed * * @param Mage_Payment_Model_Info $payment * @return bool */ public static function isPaymentCompleted(Mage_Payment_Model_Info $payment) { $paymentStatus = $payment->getAdditionalInformation(self::PAYMENT_STATUS_GLOBAL); return self::PAYMENTSTATUS_COMPLETED === $paymentStatus; } /** * Check whether the payment was processed successfully * * @param Mage_Payment_Model_Info $payment * @return bool */ public static function isPaymentSuccessful(Mage_Payment_Model_Info $payment) { $paymentStatus = $payment->getAdditionalInformation(self::PAYMENT_STATUS_GLOBAL); if (in_array($paymentStatus, array( self::PAYMENTSTATUS_COMPLETED, self::PAYMENTSTATUS_INPROGRESS, self::PAYMENTSTATUS_REFUNDED, self::PAYMENTSTATUS_REFUNDEDPART, self::PAYMENTSTATUS_UNREVERSED, self::PAYMENTSTATUS_PROCESSED, ))) { return true; } $pendingReason = $payment->getAdditionalInformation(self::PENDING_REASON_GLOBAL); return self::PAYMENTSTATUS_PENDING === $paymentStatus && in_array($pendingReason, array('authorization', 'order')); } /** * Check whether the payment was processed unsuccessfully or failed * * @param Mage_Payment_Model_Info $payment * @return bool */ public static function isPaymentFailed(Mage_Payment_Model_Info $payment) { $paymentStatus = $payment->getAdditionalInformation(self::PAYMENT_STATUS_GLOBAL); return in_array($paymentStatus, array( self::PAYMENTSTATUS_DENIED, self::PAYMENTSTATUS_EXPIRED, self::PAYMENTSTATUS_FAILED, self::PAYMENTSTATUS_REVERSED, self::PAYMENTSTATUS_VOIDED, )); } /** * Explain pending payment reason code * * @param string $code * @return string * @link https://cms.paypal.com/us/cgi-bin/?&cmd=_render-content&content_ID=developer/e_howto_html_IPNandPDTVariables * @link https://cms.paypal.com/us/cgi-bin/?&cmd=_render-content&content_ID=developer/e_howto_api_nvp_r_GetTransactionDetails */ public static function explainPendingReason($code) { switch ($code) { case 'address': return Mage::helper('paypal')->__('Customer did not include a confirmed address.'); case 'authorization': case 'order': return Mage::helper('paypal')->__('The payment is authorized but not settled.'); case 'echeck': return Mage::helper('paypal')->__('The payment eCheck is not yet cleared.'); case 'intl': return Mage::helper('paypal')->__('Merchant holds a non-U.S. account and does not have a withdrawal mechanism.'); case 'multi-currency': // break is intentionally omitted case 'multi_currency': // break is intentionally omitted case 'multicurrency': return Mage::helper('paypal')->__('The payment curency does not match any of the merchant\'s balances currency.'); case 'paymentreview': return Mage::helper('paypal')->__('The payment is pending while it is being reviewed by PayPal for risk.'); case 'unilateral': return Mage::helper('paypal')->__('The payment is pending because it was made to an email address that is not yet registered or confirmed.'); case 'verify': return Mage::helper('paypal')->__('The merchant account is not yet verified.'); case 'upgrade': return Mage::helper('paypal')->__('The payment was made via credit card. In order to receive funds merchant must upgrade account to Business or Premier status.'); case 'none': // break is intentionally omitted case 'other': // break is intentionally omitted default: return Mage::helper('paypal')->__('Unknown reason. Please contact PayPal customer service.'); } } /** * Explain the refund or chargeback reason code * * @param $code * @return string * @link https://cms.paypal.com/us/cgi-bin/?&cmd=_render-content&content_ID=developer/e_howto_html_IPNandPDTVariables * @link https://cms.paypal.com/us/cgi-bin/?&cmd=_render-content&content_ID=developer/e_howto_api_nvp_r_GetTransactionDetails */ public static function explainReasonCode($code) { switch ($code) { case 'chargeback': return Mage::helper('paypal')->__('Chargeback by customer.'); case 'guarantee': return Mage::helper('paypal')->__('Customer triggered a money-back guarantee.'); case 'buyer-complaint': return Mage::helper('paypal')->__('Customer complaint.'); case 'refund': return Mage::helper('paypal')->__('Refund issued by merchant.'); case 'adjustment_reversal': return Mage::helper('paypal')->__('Reversal of an adjustment.'); case 'chargeback_reimbursement': return Mage::helper('paypal')->__('Reimbursement for a chargeback.'); case 'chargeback_settlement': return Mage::helper('paypal')->__('Settlement of a chargeback.'); case 'none': // break is intentionally omitted case 'other': default: return Mage::helper('paypal')->__('Unknown reason. Please contact PayPal customer service.'); } } /** * Whether a reversal/refund can be disputed with PayPal * * @param string $code * @return bool; */ public static function isReversalDisputable($code) { switch ($code) { case 'none': case 'other': case 'chargeback': case 'buyer-complaint': case 'adjustment_reversal': return true; case 'guarantee': case 'refund': case 'chargeback_reimbursement': case 'chargeback_settlement': default: return false; } } /** * Render info item * * @param array $keys * @param Mage_Payment_Model_Info $payment * @param bool $labelValuesOnly */ protected function _getFullInfo(array $keys, Mage_Payment_Model_Info $payment, $labelValuesOnly) { $result = array(); foreach ($keys as $key) { if (!isset($this->_paymentMapFull[$key])) { $this->_paymentMapFull[$key] = array(); } if (!isset($this->_paymentMapFull[$key]['label'])) { if (!$payment->hasAdditionalInformation($key)) { $this->_paymentMapFull[$key]['label'] = false; $this->_paymentMapFull[$key]['value'] = false; } else { $value = $payment->getAdditionalInformation($key); $this->_paymentMapFull[$key]['label'] = $this->_getLabel($key); $this->_paymentMapFull[$key]['value'] = $this->_getValue($value, $key); } } if (!empty($this->_paymentMapFull[$key]['value'])) { if ($labelValuesOnly) { $result[$this->_paymentMapFull[$key]['label']] = $this->_paymentMapFull[$key]['value']; } else { $result[$key] = $this->_paymentMapFull[$key]; } } } return $result; } /** * Render info item labels * * @param string $key */ protected function _getLabel($key) { switch ($key) { case 'paypal_payer_id': return Mage::helper('paypal')->__('Payer ID'); case 'paypal_payer_email': return Mage::helper('paypal')->__('Payer Email'); case 'paypal_payer_status': return Mage::helper('paypal')->__('Payer Status'); case 'paypal_address_id': return Mage::helper('paypal')->__('Payer Address ID'); case 'paypal_address_status': return Mage::helper('paypal')->__('Payer Address Status'); case 'paypal_protection_eligibility': return Mage::helper('paypal')->__('Merchant Protection Eligibility'); case 'paypal_fraud_filters': return Mage::helper('paypal')->__('Triggered Fraud Filters'); case 'paypal_correlation_id': return Mage::helper('paypal')->__('Last Correlation ID'); case 'paypal_avs_code': return Mage::helper('paypal')->__('Address Verification System Response'); case 'paypal_cvv2_match': return Mage::helper('paypal')->__('CVV2 Check Result by PayPal'); case self::BUYER_TAX_ID : return Mage::helper('paypal')->__('Buyer\'s Tax ID'); case self::BUYER_TAX_ID_TYPE : return Mage::helper('paypal')->__('Buyer\'s Tax ID Type'); case self::CENTINEL_VPAS: return Mage::helper('paypal')->__('PayPal/Centinel Visa Payer Authentication Service Result'); case self::CENTINEL_ECI: return Mage::helper('paypal')->__('PayPal/Centinel Electronic Commerce Indicator'); } return ''; } /** * Apply a filter upon value getting * * @param string $value * @param string $key * @return string */ protected function _getValue($value, $key) { $label = ''; switch ($key) { case 'paypal_avs_code': $label = $this->_getAvsLabel($value); break; case 'paypal_cvv2_match': $label = $this->_getCvv2Label($value); break; case self::CENTINEL_VPAS: $label = $this->_getCentinelVpasLabel($value); break; case self::CENTINEL_ECI: $label = $this->_getCentinelEciLabel($value); break; case self::BUYER_TAX_ID_TYPE : $value = $this->_getBuyerIdTypeValue($value); default: return $value; } return sprintf('#%s%s', $value, $value == $label ? '' : ': ' . $label); } /** * Attempt to convert AVS check result code into label * * @link https://cms.paypal.com/us/cgi-bin/?&cmd=_render-content&content_ID=developer/e_howto_api_nvp_AVSResponseCodes * @param string $value * @return string */ protected function _getAvsLabel($value) { switch ($value) { // Visa, MasterCard, Discover and American Express case 'A': case 'YN': return Mage::helper('paypal')->__('Matched Address only (no ZIP)'); case 'B': // international "A" return Mage::helper('paypal')->__('Matched Address only (no ZIP). International'); case 'N': return Mage::helper('paypal')->__('No Details matched'); case 'C': // international "N" return Mage::helper('paypal')->__('No Details matched. International'); case 'X': return Mage::helper('paypal')->__('Exact Match. Address and nine-digit ZIP code'); case 'D': // international "X" return Mage::helper('paypal')->__('Exact Match. Address and Postal Code. International'); case 'F': // UK-specific "X" return Mage::helper('paypal')->__('Exact Match. Address and Postal Code. UK-specific'); case 'E': return Mage::helper('paypal')->__('N/A. Not allowed for MOTO (Internet/Phone) transactions'); case 'G': return Mage::helper('paypal')->__('N/A. Global Unavailable'); case 'I': return Mage::helper('paypal')->__('N/A. International Unavailable'); case 'Z': case 'NY': return Mage::helper('paypal')->__('Matched five-digit ZIP only (no Address)'); case 'P': // international "Z" case 'NY': return Mage::helper('paypal')->__('Matched Postal Code only (no Address)'); case 'R': return Mage::helper('paypal')->__('N/A. Retry'); case 'S': return Mage::helper('paypal')->__('N/A. Service not Supported'); case 'U': return Mage::helper('paypal')->__('N/A. Unavailable'); case 'W': return Mage::helper('paypal')->__('Matched whole nine-didgit ZIP (no Address)'); case 'Y': return Mage::helper('paypal')->__('Yes. Matched Address and five-didgit ZIP'); // Maestro and Solo case '0': return Mage::helper('paypal')->__('All the address information matched'); case '1': return Mage::helper('paypal')->__('None of the address information matched'); case '2': return Mage::helper('paypal')->__('Part of the address information matched'); case '3': return Mage::helper('paypal')->__('N/A. The merchant did not provide AVS information'); case '4': return Mage::helper('paypal')->__('N/A. Address not checked, or acquirer had no response. Service not available'); default: return $value; } } /** * Attempt to convert CVV2 check result code into label * * @link https://cms.paypal.com/us/cgi-bin/?&cmd=_render-content&content_ID=developer/e_howto_api_nvp_AVSResponseCodes * @param string $value * @return string */ protected function _getCvv2Label($value) { switch ($value) { // Visa, MasterCard, Discover and American Express case 'M': return Mage::helper('paypal')->__('Matched (CVV2CSC)'); case 'N': return Mage::helper('paypal')->__('No match'); case 'P': return Mage::helper('paypal')->__('N/A. Not processed'); case 'S': return Mage::helper('paypal')->__('N/A. Service not supported'); case 'U': return Mage::helper('paypal')->__('N/A. Service not available'); case 'X': return Mage::helper('paypal')->__('N/A. No response'); // Maestro and Solo case '0': return Mage::helper('paypal')->__('Matched (CVV2)'); case '1': return Mage::helper('paypal')->__('No match'); case '2': return Mage::helper('paypal')->__('N/A. The merchant has not implemented CVV2 code handling'); case '3': return Mage::helper('paypal')->__('N/A. Merchant has indicated that CVV2 is not present on card'); case '4': return Mage::helper('paypal')->__('N/A. Service not available'); default: return $value; } } /** * Attempt to convert centinel VPAS result into label * * @link https://cms.paypal.com/us/cgi-bin/?&cmd=_render-content&content_ID=developer/e_howto_api_nvp_r_DoDirectPayment * @param string $value * @return string */ private function _getCentinelVpasLabel($value) { switch ($value) { case '2': case 'D': return Mage::helper('paypal')->__('Authenticated, Good Result'); case '1': return Mage::helper('paypal')->__('Authenticated, Bad Result'); case '3': case '6': case '8': case 'A': case 'C': return Mage::helper('paypal')->__('Attempted Authentication, Good Result'); case '4': case '7': case '9': return Mage::helper('paypal')->__('Attempted Authentication, Bad Result'); case '': case '0': case 'B': return Mage::helper('paypal')->__('No Liability Shift'); default: return $value; } } /** * Attempt to convert centinel ECI result into label * * @link https://cms.paypal.com/us/cgi-bin/?&cmd=_render-content&content_ID=developer/e_howto_api_nvp_r_DoDirectPayment * @param string $value * @return string */ private function _getCentinelEciLabel($value) { switch ($value) { case '01': case '07': return Mage::helper('paypal')->__('Merchant Liability'); case '02': case '05': case '06': return Mage::helper('paypal')->__('Issuer Liability'); default: return $value; } } /** * Retrieve buyer id type value based on code received from PayPal (Brazil only) * * @param string $code * @return string */ protected function _getBuyerIdTypeValue($code) { $value = ''; switch ($code) { case self::BUYER_TAX_ID_TYPE_CNPJ : $value = Mage::helper('paypal')->__('CNPJ'); break; case self::BUYER_TAX_ID_TYPE_CPF : $value = Mage::helper('paypal')->__('CPF'); break; } return $value; } }