From 702b7e49b19971cf1e43bc379ca482d6c7ff88a2 Mon Sep 17 00:00:00 2001 From: Florent Torregrosa Date: Wed, 8 May 2019 22:33:02 +0200 Subject: [PATCH] Update commerce_paypal to 7.x-2.7 --- .../commerce_paypal/commerce_paypal.info | 6 +- .../commerce_paypal/commerce_paypal.module | 7 +- .../checkout/commerce_paypal_checkout.api.php | 32 + .../checkout/commerce_paypal_checkout.info | 18 + .../checkout/commerce_paypal_checkout.install | 13 + .../checkout/commerce_paypal_checkout.module | 1233 +++++++++++++++++ .../checkout/css/commerce_paypal_checkout.css | 9 + .../commerce_paypal_checkout.admin.inc | 297 ++++ ...commerce_paypal_checkout.checkout_pane.inc | 206 +++ .../checkout/js/commerce_paypal_checkout.js | 66 + .../checkout/lib/PayPalCheckoutClient.php | 393 ++++++ .../checkout/lib/PayPalCheckoutExceptions.php | 40 + .../modules/ec/commerce_paypal_ec.info | 6 +- .../modules/ec/commerce_paypal_ec.module | 29 + .../modules/payflow/commerce_payflow.info | 6 +- .../modules/wpp/commerce_paypal_wpp.info | 6 +- .../modules/wps/commerce_paypal_wps.info | 6 +- .../modules/wps/commerce_paypal_wps.module | 189 ++- 18 files changed, 2542 insertions(+), 20 deletions(-) create mode 100644 www7/sites/all/modules/contrib/commerce_paypal/modules/checkout/commerce_paypal_checkout.api.php create mode 100644 www7/sites/all/modules/contrib/commerce_paypal/modules/checkout/commerce_paypal_checkout.info create mode 100644 www7/sites/all/modules/contrib/commerce_paypal/modules/checkout/commerce_paypal_checkout.install create mode 100644 www7/sites/all/modules/contrib/commerce_paypal/modules/checkout/commerce_paypal_checkout.module create mode 100644 www7/sites/all/modules/contrib/commerce_paypal/modules/checkout/css/commerce_paypal_checkout.css create mode 100644 www7/sites/all/modules/contrib/commerce_paypal/modules/checkout/includes/commerce_paypal_checkout.admin.inc create mode 100644 www7/sites/all/modules/contrib/commerce_paypal/modules/checkout/includes/commerce_paypal_checkout.checkout_pane.inc create mode 100644 www7/sites/all/modules/contrib/commerce_paypal/modules/checkout/js/commerce_paypal_checkout.js create mode 100644 www7/sites/all/modules/contrib/commerce_paypal/modules/checkout/lib/PayPalCheckoutClient.php create mode 100644 www7/sites/all/modules/contrib/commerce_paypal/modules/checkout/lib/PayPalCheckoutExceptions.php diff --git a/www7/sites/all/modules/contrib/commerce_paypal/commerce_paypal.info b/www7/sites/all/modules/contrib/commerce_paypal/commerce_paypal.info index 7a8793257..6f50d5a0b 100644 --- a/www7/sites/all/modules/contrib/commerce_paypal/commerce_paypal.info +++ b/www7/sites/all/modules/contrib/commerce_paypal/commerce_paypal.info @@ -10,8 +10,8 @@ core = 7.x ; Simple tests ; files[] = tests/commerce_paypal.test -; Information added by Drupal.org packaging script on 2018-10-22 -version = "7.x-2.6" +; Information added by Drupal.org packaging script on 2019-05-06 +version = "7.x-2.7" core = "7.x" project = "commerce_paypal" -datestamp = "1540244884" +datestamp = "1557130694" diff --git a/www7/sites/all/modules/contrib/commerce_paypal/commerce_paypal.module b/www7/sites/all/modules/contrib/commerce_paypal/commerce_paypal.module index dad78213f..2a617538f 100644 --- a/www7/sites/all/modules/contrib/commerce_paypal/commerce_paypal.module +++ b/www7/sites/all/modules/contrib/commerce_paypal/commerce_paypal.module @@ -488,13 +488,12 @@ function commerce_paypal_reverse_payment_action($payment_action) { */ function commerce_paypal_currencies($method_id) { switch ($method_id) { - case 'paypal_wpp': - case 'paypal_ec': - case 'payflow_link': - return drupal_map_assoc(array('AUD', 'BRL', 'CAD', 'CHF', 'CZK', 'DKK', 'EUR', 'GBP', 'HKD', 'HUF', 'JPY', 'NOK', 'NZD', 'PLN', 'SEK', 'SGD', 'USD')); case 'paypal_ppa': return drupal_map_assoc(array('AUD', 'CAD', 'EUR', 'GBP', 'JPY', 'USD')); case 'paypal_wps': + case 'paypal_wpp': + case 'paypal_ec': + case 'payflow_link': return drupal_map_assoc(array('AUD', 'BRL', 'CAD', 'CHF', 'CZK', 'DKK', 'EUR', 'GBP', 'HKD', 'HUF', 'ILS', 'INR', 'JPY', 'MXN', 'MYR', 'NOK', 'NZD', 'PHP', 'PLN', 'RUB', 'SEK', 'SGD', 'THB', 'TRY', 'TWD', 'USD')); } } diff --git a/www7/sites/all/modules/contrib/commerce_paypal/modules/checkout/commerce_paypal_checkout.api.php b/www7/sites/all/modules/contrib/commerce_paypal/modules/checkout/commerce_paypal_checkout.api.php new file mode 100644 index 000000000..c1ad7eb89 --- /dev/null +++ b/www7/sites/all/modules/contrib/commerce_paypal/modules/checkout/commerce_paypal_checkout.api.php @@ -0,0 +1,32 @@ + array( + 'title' => 'Capture', + 'page callback' => 'drupal_get_form', + 'page arguments' => array('commerce_paypal_checkout_capture_form', 3, 5), + 'access callback' => 'commerce_paypal_checkout_capture_void_access', + 'access arguments' => array(3, 5), + 'type' => MENU_DEFAULT_LOCAL_TASK, + 'context' => MENU_CONTEXT_INLINE, + 'weight' => 2, + 'file' => 'includes/commerce_paypal_checkout.admin.inc', + ), + // Add a menu item for voiding authorizations. + 'admin/commerce/orders/%commerce_order/payment/%commerce_payment_transaction/paypal-checkout-void' => array( + 'title' => 'Void', + 'page callback' => 'drupal_get_form', + 'page arguments' => array('commerce_paypal_checkout_void_form', 3, 5), + 'access callback' => 'commerce_paypal_checkout_capture_void_access', + 'access arguments' => array(3, 5), + 'type' => MENU_DEFAULT_LOCAL_TASK, + 'context' => MENU_CONTEXT_INLINE, + 'weight' => 4, + 'file' => 'includes/commerce_paypal_checkout.admin.inc', + ), + // Add a menu item for refunding settled transactions. + 'admin/commerce/orders/%commerce_order/payment/%commerce_payment_transaction/paypal-checkout-refund' => array( + 'title' => 'Refund', + 'page callback' => 'drupal_get_form', + 'page arguments' => array('commerce_paypal_checkout_refund_form', 3, 5), + 'access callback' => 'commerce_paypal_checkout_refund_access', + 'access arguments' => array(3, 5), + 'type' => MENU_DEFAULT_LOCAL_TASK, + 'context' => MENU_CONTEXT_INLINE, + 'weight' => 4, + 'file' => 'includes/commerce_paypal_checkout.admin.inc', + ), + 'commerce-paypal-checkout/create-order/%commerce_order/%commerce_paypal_checkout_method_instance' => array( + 'page callback' => 'commerce_paypal_checkout_create_order', + 'page arguments' => array(2, 3), + 'access callback' => 'commerce_checkout_access', + 'access arguments' => array(2), + 'type' => MENU_CALLBACK, + ), + 'commerce-paypal-checkout/approve-order/%commerce_order/%commerce_paypal_checkout_method_instance/%' => array( + 'page callback' => 'commerce_paypal_checkout_approve_order', + 'page arguments' => array(2, 3, 4), + 'access callback' => 'commerce_checkout_access', + 'access arguments' => array(2), + 'type' => MENU_CALLBACK, + ), + ); +} + +/** + * Determines access to the prior authorization capture form or void form for + * Paypal Checkout transactions. + * + * @param $order + * The order the transaction is on. + * @param $transaction + * The payment transaction object to be captured. + * + * @return + * TRUE or FALSE indicating access. + */ +function commerce_paypal_checkout_capture_void_access($order, $transaction) { + // Return FALSE if the transaction isn't for Paypal Checkout or isn't + // awaiting capture. + if ($transaction->payment_method != 'paypal_checkout' || strtolower($transaction->remote_status) != 'created') { + return FALSE; + } + + // Return FALSE if the transaction is not pending. + if ($transaction->status != COMMERCE_PAYMENT_STATUS_PENDING) { + return FALSE; + } + + // Return FALSE if it is more than 29 days past the original authorization. + if (REQUEST_TIME - $transaction->created > 86400 * 29) { + return FALSE; + } + + // Allow access if the user can update payments on this transaction. + return commerce_payment_transaction_access('update', $transaction); +} + +/** + * Determines access to the refund form for Paypal Checkout transactions. + * + * @param $order + * The order the transaction is on. + * @param $transaction + * The payment transaction object to be captured. + * + * @return + * TRUE or FALSE indicating access. + */ +function commerce_paypal_checkout_refund_access($order, $transaction) { + // Return FALSE if the transaction isn't Completed. + if ($transaction->payment_method != 'paypal_checkout' || strtolower($transaction->remote_status) != 'completed') { + return FALSE; + } + + // Return FALSE if the transaction was not a success. + if ($transaction->status != COMMERCE_PAYMENT_STATUS_SUCCESS) { + return FALSE; + } + + // Return FALSE if it is more than 60 days since the original transaction. + if (REQUEST_TIME - $transaction->created > 86400 * 60) { + return FALSE; + } + + // Allow access if the user can update payments on this transaction. + return commerce_payment_transaction_access('update', $transaction); +} + +/** + * Load the given PayPal checkout payment method instance. + * + * @param $rule_name + * The enabling's rule name. + * + * @return + * The payment method instance object which is identical to the payment method + * object with the addition of the settings array. + */ +function commerce_paypal_checkout_method_instance_load($rule_name) { + return commerce_payment_method_instance_load("paypal_checkout|$rule_name"); +} + +/** + * Implements hook_commerce_checkout_page_info(). + */ +function commerce_paypal_checkout_commerce_checkout_page_info() { + $checkout_pages = array(); + + $checkout_pages['paypal_checkout'] = array( + 'title' => t('Confirm order'), + 'help' => t('Confirm your order information and use the button at the bottom of the page to finalize your payment.'), + 'status_cart' => FALSE, + 'locked' => TRUE, + 'buttons' => FALSE, + 'weight' => 30, + ); + + return $checkout_pages; +} + +/** + * Implements hook_commerce_checkout_pane_info(). + */ +function commerce_paypal_checkout_commerce_checkout_pane_info() { + $checkout_panes = array(); + + $checkout_panes['paypal_checkout_review'] = array( + 'title' => t('Review and confirm your order'), + 'name' => t('PayPal Checkout review and confirm (only to be used on the confirm order page)'), + 'file' => 'includes/commerce_paypal_checkout.checkout_pane.inc', + 'base' => 'commerce_paypal_checkout_review_pane', + 'page' => 'paypal_checkout', + 'fieldset' => FALSE, + ); + + return $checkout_panes; +} + +/** + * Implements hook_commerce_checkout_router(). + */ +function commerce_paypal_checkout_commerce_checkout_router($order, $checkout_page) { + // If the current page is the PayPal Checkout page but the current order did + // not use the shortcut Checkout flow... + if ($checkout_page['page_id'] == 'paypal_checkout' && + (empty($order->data['commerce_paypal_checkout']['flow']) || $order->data['commerce_paypal_checkout']['flow'] != 'shortcut')) { + // Update the order status to the next checkout page. + $next_page = $checkout_page['next_page']; + $order = commerce_order_status_update($order, 'checkout_' . $next_page, FALSE, FALSE); + + // Inform modules of checkout completion if the next page is Completed. + if ($next_page == 'complete') { + commerce_checkout_complete($order); + } + + // Redirect to the URL for the new checkout page. + $target_uri = commerce_checkout_order_uri($order); + return drupal_goto($target_uri); + } +} + +/** + * Returns the checkout pane IDs of checkout panes that should be embedded in + * the PayPal Checkout review and confirm page. + */ +function commerce_paypal_checkout_embedded_checkout_panes() { + return array_filter(variable_get('commerce_paypal_checkout_review_embedded_panes', array())); +} + +/** + * Implements hook_commerce_payment_method_info(). + */ +function commerce_paypal_checkout_commerce_payment_method_info() { + $payment_methods = array(); + + $payment_methods['paypal_checkout'] = array( + 'base' => 'commerce_paypal_checkout', + 'title' => t('PayPal Checkout'), + 'display_title' => t('PayPal'), + 'short_title' => t('PayPal Checkout'), + 'description' => t('PayPal Checkout'), + 'terminal' => FALSE, + 'offsite' => TRUE, + 'offsite_autoredirect' => FALSE, + ); + + return $payment_methods; +} + +/** + * Returns the default settings for the PayPal Checkout payment method. + */ +function commerce_paypal_checkout_default_settings() { + $default_settings = array( + 'client_id' => '', + 'secret' => '', + 'server' => 'sandbox', + 'intent' => 'capture', + 'disable_funding' => array(), + 'disable_card' => array(), + 'shipping_preference' => 'get_from_file', + 'update_billing_profiles' => TRUE, + 'style' => array(), + 'enable_on_cart' => TRUE, + ); + + if (module_exists('commerce_shipping')) { + $default_settings['update_shipping_profiles'] = TRUE; + } + + return $default_settings; +} + +/** + * Payment method callback: settings form. + */ +function commerce_paypal_checkout_settings_form($settings = array()) { + $form = array(); + + // Merge default settings into the stored settings array. + $settings = (array) $settings + commerce_paypal_checkout_default_settings(); + $shipping_module_enabled = module_exists('commerce_shipping'); + + $form['client_id'] = array( + '#type' => 'textfield', + '#title' => t('Client ID'), + '#maxlength' => 255, + '#default_value' => $settings['client_id'], + '#required' => TRUE, + ); + $form['secret'] = array( + '#type' => 'textfield', + '#title' => t('Secret'), + '#maxlength' => 255, + '#default_value' => $settings['secret'], + '#required' => TRUE, + ); + $form['server'] = array( + '#type' => 'radios', + '#title' => t('PayPal server'), + '#options' => array( + 'sandbox' => ('Sandbox - use for testing, requires a PayPal Sandbox account'), + 'live' => ('Live - use for processing real transactions'), + ), + '#default_value' => $settings['server'], + ); + $form['intent'] = array( + '#type' => 'radios', + '#title' => t('Transaction type'), + '#options' => array( + 'capture' => t('Capture'), + 'authorize' => t('Authorize'), + ), + '#default_value' => $settings['intent'], + ); + $form['disable_funding'] = array( + '#type' => 'checkboxes', + '#title' => t('Disable funding sources'), + '#description' => t('The disabled funding sources for the transaction. Any funding sources passed are not displayed in the Smart payment buttons. By default, funding source eligibility is smartly decided based on a variety of factors.'), + '#options' => array( + 'card' => t('Credit or Debit Cards'), + 'credit' => t('PayPal Credit'), + 'sepa' => t('SEPA-Lastschrift'), + ), + '#default_value' => $settings['disable_funding'], + '#element_validate' => array('commerce_paypal_checkout_disable_funding_validate'), + ); + $form['disable_card'] = array( + '#title' => t('Disable card types'), + '#description' => t('The disabled cards for the transaction. Any cards passed do not display in the Smart payment buttons. By default, card eligibility is smartly decided based on a variety of factors.'), + '#type' => 'checkboxes', + '#options' => array( + 'visa' => t('Visa'), + 'mastercard' => t('Mastercard'), + 'amex' => t('American Express'), + 'discover' => t('Discover'), + 'jcb' => t('JCB'), + 'elo' => t('Elo'), + 'hiper' => t('Hiper'), + ), + '#default_value' => $settings['disable_card'], + '#element_validate' => array('commerce_paypal_checkout_disable_card_validate'), + ); + $form['shipping_preference'] = array( + '#type' => 'radios', + '#title' => t('Shipping address collection'), + '#description' => t('PayPal Checkout will only request a shipping address if the Shipping module is enabled to store the address in the order.'), + '#options' => array( + 'no_shipping' => t('Do not ask for a shipping address at PayPal.'), + ), + '#default_value' => 'no_shipping', + ); + + if ($shipping_module_enabled) { + $form['shipping_preference']['#options'] += array( + 'get_from_file' => t('Ask for a shipping address at PayPal even if the order already has one.'), + 'set_provided_address' => t('Ask for a shipping address at PayPal if the order does not have one yet.'), + ); + $form['shipping_preference']['#default_value'] = $settings['shipping_preference']; + } + + $form['update_billing_profiles'] = array( + '#type' => 'checkbox', + '#title' => t('Update billing customer profiles with address information the customer enters at PayPal.'), + '#default_value' => $settings['update_billing_profiles'], + ); + + if ($shipping_module_enabled) { + $form['update_shipping_profiles'] = array( + '#type' => 'checkbox', + '#title' => t('Update shipping customer profiles with address information the customer enters at PayPal.'), + '#default_value' => $settings['update_shipping_profiles'], + ); + } + + $form['customize_buttons'] = array( + '#type' => 'checkbox', + '#title' => t('Smart button style'), + '#default_value' => !empty($settings['style']), + '#title_display' => 'before', + '#field_suffix' => t('Customize'), + '#description_display' => 'before', + '#element_validate' => array('commerce_paypal_checkout_customize_buttons_validate'), + ); + $form['style'] = array( + '#type' => 'fieldset', + '#title' => t('Settings'), + '#description' => t('For more information, please visit customize the PayPal buttons.', array('@url' => 'https://developer.paypal.com/docs/checkout/integration-features/customize-button/#layout')), + '#states' => array( + 'visible' => array( + ':input[name="parameter[payment_method][settings][payment_method][settings][customize_buttons]"]' => array('checked' => TRUE), + ), + ), + ); + // Define some default values for the style configuration. + $settings['style'] += [ + 'layout' => 'vertical', + 'color' => 'gold', + 'shape' => 'rect', + 'label' => 'paypal', + 'tagline' => FALSE, + ]; + $form['style']['layout'] = array( + '#type' => 'select', + '#title' => t('Layout'), + '#default_value' => $settings['style']['layout'], + '#options' => array( + 'vertical' => t('Vertical (Recommended)'), + 'horizontal' => t('Horizontal'), + ), + ); + $form['style']['color'] = array( + '#type' => 'select', + '#title' => t('Color'), + '#options' => array( + 'gold' => t('Gold (Recommended)'), + 'blue' => t('Blue'), + 'silver' => t('Silver'), + ), + '#default_value' => $settings['style']['color'], + ); + $form['style']['shape'] = array( + '#type' => 'select', + '#title' => t('Shape'), + '#options' => array( + 'rect' => t('Rect (Default)'), + 'pill' => t('Pill'), + ), + '#default_value' => $settings['style']['shape'], + ); + $form['style']['label'] = array( + '#type' => 'select', + '#title' => t('Label'), + '#options' => array( + 'paypal' => t('Displays the PayPal logo (Default)'), + 'checkout' => t('Displays the PayPal Checkout button'), + 'pay' => t('Displays the Pay With PayPal button and initializes the checkout flow'), + ), + '#default_value' => $settings['style']['label'], + ); + $form['style']['tagline'] = array( + '#type' => 'checkbox', + '#title' => t('Display tagline'), + '#default_value' => $settings['style']['tagline'], + '#states' => array( + 'visible' => array( + ':input[name="parameter[payment_method][settings][payment_method][settings][style][layout]"]' => array('value' => 'horizontal'), + ), + ), + ); + + $form['enable_on_cart'] = array( + '#type' => 'checkbox', + '#title' => t('Show the Smart payment buttons on the cart form.'), + '#default_value' => $settings['enable_on_cart'], + ); + + return $form; +} + +/** + * Element validate callback for the customize_buttons payment method setting. + */ +function commerce_paypal_checkout_customize_buttons_validate($element, &$form_state, $form) { + // Make sure no style is saved when the the "customize_buttons" checkbox + // is unchecked. + $parents = $element['#parents']; + array_pop($parents); + $parents[] = 'style'; + if (empty($element['#value'])) { + drupal_array_set_nested_value($form_state['values'], $parents, array(), TRUE); + } + else { + $style = drupal_array_get_nested_value($form_state['values'], $parents); + // The tagline is not allowed for the vertical layout. + if ($style['layout'] == 'vertical') { + unset($style['tagline']); + drupal_array_set_nested_value($form_state['values'], $parents, $style, TRUE); + } + } +} + +/** + * Element validate callback for the disable_funding payment method setting. + */ +function commerce_paypal_checkout_disable_funding_validate($element, &$form_state, $form) { + form_set_value($element, array_filter($element['#value']), $form_state); +} + +/** + * Element validate callback for the disable_card payment method setting. + */ +function commerce_paypal_checkout_disable_card_validate($element, &$form_state, $form) { + $parents = $element['#parents']; + array_pop($parents); + $parents[] = 'disable_funding'; + $disable_funding = drupal_array_get_nested_value($form_state['values'], $parents); + + // When the "card" funding source is disabled, the "disable_card" setting + // cannot be specified. + if (isset($disable_funding['card'])) { + $element['#value'] = array(); + } + form_set_value($element, array_filter($element['#value']), $form_state); +} + +/** + * Page callback: Provide the createOrder() callback expected by the SDK. + */ +function commerce_paypal_checkout_create_order($order, $payment_method) { + $settings = $payment_method['settings']; + $api_client = commerce_paypal_checkout_api_client($settings); + + if (!$api_client) { + drupal_json_output(array()); + drupal_exit(); + } + try { + $request_body = commerce_paypal_checkout_prepare_order_request($order, $payment_method['settings']); + drupal_alter('commerce_paypal_checkout_create_order_request', $request_body, $order); + $json = $api_client->createOrder($request_body); + drupal_json_output(array('id' => $json['id'])); + drupal_exit(); + } + catch (\Exception $exception) { + watchdog_exception('commerce_paypal_checkout', $exception); + } + drupal_json_output(array()); + drupal_exit(); +} + +/** + * Page callback: Provide the onApprove() callback expected by the SDK. + */ +function commerce_paypal_checkout_approve_order($order, $payment_method, $flow) { + $data = drupal_json_decode(file_get_contents('php://input')); + + if (!in_array($flow, array('shortcut', 'mark')) || !isset($data['id'])) { + drupal_json_output(array()); + drupal_exit(); + } + // Store the PayPal order ID, and the "flow" used ("shortcut"|"mark"). + // Note that we don't perform any validation here, that happens inside + // commerce_paypal_checkout_redirect_form_validate(). + $order->data['commerce_paypal_checkout'] = array( + 'flow' => $flow, + 'remote_id' => $data['id'], + ); + + // The payment_redirect key is required in the payment return url. + if (empty($order->data['payment_redirect_key'])) { + $order->data['payment_redirect_key'] = drupal_hash_base64(time()); + } + // We have to manually set the payment method if empty, it's also required + // by the payment redirect form validate callback. + if (empty($order->data['payment_method'])) { + $order->data['payment_method'] = $payment_method['instance_id']; + } + + // Update the order status to the payment page for the shortcut flow. + if ($flow == 'shortcut') { + commerce_order_status_update($order, 'checkout_payment', FALSE, NULL, t('Customer clicked the Smart payment buttons on the cart page.')); + } + else { + commerce_order_save($order); + } + + $return_url = url('checkout/' . $order->order_id . '/payment/return/' . $order->data['payment_redirect_key']); + drupal_json_output(array('redirectUri' => $return_url)); + drupal_exit(); +} + +/** + * Prepare the request parameters for the create/update order request. + * + * @param $order + * The order to prepare the request for. + * @param array $settings + * The payment method settings. + */ +function commerce_paypal_checkout_prepare_order_request($order, $settings) { + $order_wrapper = entity_metadata_wrapper('commerce_order', $order); + $product_line_item_types = commerce_product_line_item_types(); + $item_total = 0; + $discount_total = 0; + $shipping_total = 0; + $items = array(); + $order_total = $order_wrapper->commerce_order_total->value(); + $currency_code = $order_total['currency_code']; + foreach ($order_wrapper->commerce_line_items as $line_item_wrapper) { + if (!$line_item_wrapper->value()) { + continue; + } + $unit_price = $line_item_wrapper->commerce_unit_price->value(); + if ($line_item_wrapper->getBundle() == 'shipping') { + $shipping_total += $unit_price['amount']; + } + elseif ($line_item_wrapper->getBundle() == 'commerce_discount') { + $discount_total += -$unit_price['amount']; + } + // PayPal Checkout doesn't support passing negative items. + elseif ($unit_price['amount'] >= 0) { + $item_total += $line_item_wrapper->commerce_total->amount->value(); + $item = array( + 'name' => mb_substr(commerce_line_item_title($line_item_wrapper->value()), 0, 127), + 'unit_amount' => array( + 'currency_code' => $currency_code, + 'value' => commerce_paypal_checkout_price_amount($unit_price['amount'], $unit_price['currency_code']), + ), + 'quantity' => intval($line_item_wrapper->quantity->value()), + ); + // Pass the "SKU" for product line items. + if (in_array($line_item_wrapper->getBundle(), $product_line_item_types)) { + $item['sku'] = $line_item_wrapper->commerce_product->sku->value(); + } + $items[] = $item; + } + } + + // @todo: Support passing discount in the breakdown when + // https://github.com/paypal/paypal-checkout-components/issues/1016 is fixed. + $breakdown = array( + 'item_total' => array( + 'currency_code' => $currency_code, + 'value' => commerce_paypal_checkout_price_amount($item_total, $currency_code), + ), + ); + + if ($shipping_total) { + $breakdown['shipping'] = array( + 'currency_code' => $currency_code, + 'value' => commerce_paypal_checkout_price_amount($shipping_total, $currency_code), + ); + } + + if (module_exists('commerce_tax')) { + $tax_total = commerce_round(COMMERCE_ROUND_HALF_UP, commerce_tax_total_amount($order_total['data']['components'], FALSE, $currency_code)); + if ($tax_total) { + $breakdown['tax_total'] = array( + 'currency_code' => $currency_code, + 'value' => commerce_paypal_checkout_price_amount($tax_total, $currency_code), + ); + } + } + + if ($discount_total) { + $breakdown['discount'] = array( + 'currency_code' => $currency_code, + 'value' => commerce_paypal_checkout_price_amount($discount_total, $currency_code), + ); + } + + $payer = array(); + if (!empty($order->mail)) { + $payer['email_address'] = $order->mail; + } + + // If we have a billing address, pass it to PayPal. + if (isset($order_wrapper->commerce_customer_billing) && !empty($order_wrapper->commerce_customer_billing->commerce_customer_address)) { + $address = $order_wrapper->commerce_customer_billing->commerce_customer_address->value(); + $payer += commerce_paypal_checkout_format_address($address, 'billing'); + } + + $request_body = array( + 'intent' => strtoupper($settings['intent']), + 'purchase_units' => array( + array( + 'reference_id' => 'default', + 'custom_id' => $order->order_id, + 'invoice_id' => $order->order_id . '-' . time(), + 'amount' => array( + 'currency_code' => $currency_code, + 'value' => commerce_paypal_checkout_price_amount($order_total['amount'], $order_total['currency_code']), + 'breakdown' => $breakdown, + ), + 'items' => $items, + ), + ), + 'application_context' => array( + 'brand_name' => mb_substr(variable_get('site_name', ''), 0, 127), + ), + ); + + // Send the payer if not empty. + if ($payer) { + $request_body['payer'] = $payer; + } + + $shipping_exists = module_exists('commerce_shipping'); + $shipping_address = FALSE; + // If the shipping module is enabled... + if ($shipping_exists) { + // If we have a shipping address, pass it to PayPal. + if (isset($order_wrapper->commerce_customer_shipping) && !empty($order_wrapper->commerce_customer_shipping->commerce_customer_address)) { + $address = $order_wrapper->commerce_customer_shipping->commerce_customer_address->value(); + $shipping_address = commerce_paypal_checkout_format_address($address, 'shipping'); + } + } + $shipping_preference = $settings['shipping_preference']; + + // The shipping module isn't enabled, override the shipping preference + // configured. + if (!$shipping_exists) { + $shipping_preference = 'no_shipping'; + } + else { + // If no shipping address was already collected, override the shipping + // preference to "GET_FROM_FILE" so that the shipping address is collected + // on the PayPal site. + if ($shipping_preference == 'set_provided_address' && !$shipping_address) { + $shipping_preference = 'get_from_file'; + } + } + + // No need to pass a shipping_address if the shipping address collection + // is configured to "no_shipping". + if ($shipping_address && $shipping_preference !== 'no_shipping') { + $request_body['purchase_units'][0]['shipping'] = $shipping_address; + } + $request_body['application_context']['shipping_preference'] = strtoupper($shipping_preference); + return $request_body; +} + +/** + * Formats the given address into a format expected by PayPal. + * + * @param array $address + * The address to format. + * @param $profile_type + * The profile type ("billing"|"shipping"). + * + * @return array + * The formatted address. + */ +function commerce_paypal_checkout_format_address($address, $profile_type) { + $return = array( + 'address' => array( + 'address_line_1' => $address['thoroughfare'], + 'address_line_2' => $address['premise'], + 'admin_area_2' => mb_substr($address['locality'], 0, 120), + 'admin_area_1' => $address['administrative_area'], + 'postal_code' => mb_substr($address['postal_code'], 0, 60), + 'country_code' => $address['country'], + ), + ); + if ($profile_type == 'billing') { + $return['name'] = array( + 'given_name' => $address['first_name'], + 'surname' => $address['last_name'], + ); + } + elseif ($profile_type == 'shipping') { + $return['name'] = array( + 'full_name' => $address['name_line'], + ); + } + return $return; +} + +/** + * Implements hook_theme(). + */ +function commerce_paypal_checkout_theme($existing, $type, $theme, $path) { + return array( + 'commerce_paypal_checkout_smart_payment_buttons' => array( + 'variables' => array( + 'payment_method' => array(), + 'commit' => FALSE, + 'order' => NULL, + 'flow' => NULL, + ), + ), + ); +} + +/** + * Returns HTML for the Smart payment buttons. + * + * @param $variables + * An associative array containing: + * - payment_method: The payment method instance. + * - commit: A boolean indicating whether to commit the transaction. + * - order: The order. + * - flow: The flow ("shortcut"|"mark"). + * + * @ingroup themeable + */ +function theme_commerce_paypal_checkout_smart_payment_buttons($variables) { + $payment_method = $variables['payment_method']; + $settings = $payment_method['settings'] + commerce_paypal_checkout_default_settings(); + if (empty($settings['client_id']) || empty($variables['order']) || empty($variables['flow'])) { + return; + } + $order = $variables['order']; + $order_total = field_get_items('commerce_order', $order, 'commerce_order_total', LANGUAGE_NONE); + if (!isset($order_total[0]['currency_code'])) { + return; + } + $flow = $variables['flow']; + list(, $rule_name) = explode('|', $payment_method['instance_id']); + $options = array( + 'external' => TRUE, + 'query' => array( + 'client-id' => $settings['client_id'], + 'commit' => $variables['commit'] ? 'true' : 'false', + 'intent' => $settings['intent'], + 'currency' => $order_total[0]['currency_code'], + ), + ); + if (!empty($settings['disable_funding'])) { + $options['query']['disable-funding'] = implode(',', $settings['disable_funding']); + } + if (!empty($settings['disable_card'])) { + $options['query']['disable-card'] = implode(',', $settings['disable_card']); + } + $path = drupal_get_path('module', 'commerce_paypal_checkout'); + $js_settings = array( + 'paypalCheckout' => array( + 'src' => url('https://www.paypal.com/sdk/js', $options), + 'createOrderUri' => url("commerce-paypal-checkout/create-order/$order->order_id/$rule_name"), + 'onApproveUri' => url("commerce-paypal-checkout/approve-order/$order->order_id/$rule_name/$flow"), + 'style' => $settings['style'], + ), + ); + drupal_add_css($path . '/css/commerce_paypal_checkout.css'); + drupal_add_js($js_settings, 'setting'); + drupal_add_js($path . '/js/commerce_paypal_checkout.js'); + $id = drupal_html_id('paypal-buttons-container'); + return '
'; +} + +/** + * Returns the first configured PayPal checkout payment method instance for the + * given order, if any. + * + * @param $order + * The order that needs to be checked. + * + * @return array|bool + * The first payment method instance found for given order, FALSE otherwise. + */ +function commerce_paypal_checkout_get_payment_method_instance($order) { + if (empty($order->payment_methods)) { + $order->payment_methods = array(); + rules_invoke_all('commerce_payment_methods', $order); + + // Sort the payment methods array by the enabling Rules' weight values. + uasort($order->payment_methods, 'drupal_sort_weight'); + } + + foreach (array_keys($order->payment_methods) as $instance_id) { + // Explode the method key into its component parts. + list($method_id) = explode('|', $instance_id); + if ($method_id != 'paypal_checkout') { + continue; + } + return commerce_payment_method_instance_load($instance_id); + } + + return FALSE; +} + +/** + * Implements hook_form_alter(). + */ +function commerce_paypal_checkout_form_alter(&$form, &$form_state, $form_id) { + if (!is_string($form_id)) { + return; + } + + // If we're altering a shopping cart form. + if (strpos($form_id, 'views_form_commerce_cart_form_') === 0) { + // If the cart form View shows line items... + if (!empty($form_state['build_info']['args'][0]->result)) { + $order = $form_state['order']; + $payment_method = commerce_paypal_checkout_get_payment_method_instance($order); + + // If no PayPal checkout payment method is configured, or if the buttons + // are explicitly not shown on the cart page, stop here. + if (!$payment_method || empty($payment_method['settings']['enable_on_cart'])) { + return; + } + + $form['smart_payment_buttons'] = array( + '#theme' => 'commerce_paypal_checkout_smart_payment_buttons', + '#payment_method' => $payment_method, + '#commit' => FALSE, + '#flow' => 'shortcut', + '#order' => $order, + '#weight' => 100, + ); + } + } +} + +/** + * Returns an instantiated PayPal Checkout API client for the given settings. + * + * @param array $config + * An associative array containing at least the following keys: + * - client_id: The client ID. + * - secret: The client secret. + * - server: The API server ("sandbox or "live"). + * + * @return PayPalCheckoutClient|NULL. + * An instantiated PayPalCheckout client, NULL if no client_id/secret were + * specified. + */ +function commerce_paypal_checkout_api_client($config) { + if (!isset($config['client_id']) || !isset($config['secret'])) { + return NULL; + } + $instances = &drupal_static(__FUNCTION__, array()); + + if (isset($instances[$config['client_id']])) { + return $instances[$config['client_id']]; + } + + $instances[$config['client_id']] = new PayPalCheckoutClient($config); + return $instances[$config['client_id']]; +} + +/** + * Payment method callback: redirect form. + */ +function commerce_paypal_checkout_redirect_form($form, &$form_state, $order, $payment_method) { + $form['smart_payment_buttons'] = array( + '#theme' => 'commerce_paypal_checkout_smart_payment_buttons', + '#payment_method' => $payment_method, + '#commit' => TRUE, + '#order' => $order, + '#flow' => 'mark', + ); + return $form; +} + +/** + * Payment method callback: redirect form return validation. + */ +function commerce_paypal_checkout_redirect_form_validate($order, $payment_method) { + $payment_method['settings'] += commerce_paypal_checkout_default_settings(); + + // Check if the PayPal order ID is known, as well as the "flow". + if (empty($order->data['commerce_paypal_checkout']['remote_id'] || + !isset($order->data['commerce_paypal_checkout']['flow']))) { + return FALSE; + } + $flow = $order->data['commerce_paypal_checkout']['flow']; + $api_client = commerce_paypal_checkout_api_client($payment_method['settings']); + if (!$api_client) { + return FALSE; + } + $remote_id = $order->data['commerce_paypal_checkout']['remote_id']; + try { + $paypal_order = $api_client->getOrder($remote_id); + } + catch (\Exception $exception) { + watchdog_exception('commerce_paypal_checkout', $exception); + return FALSE; + } + $order_total = field_get_items('commerce_order', $order, 'commerce_order_total', LANGUAGE_NONE); + $paypal_amount = $paypal_order['purchase_units'][0]['amount']; + $paypal_total = commerce_currency_decimal_to_amount($paypal_amount['value'], $paypal_amount['currency_code']); + // Check the remote status, and that the PayPal total matches the order total. + if (!in_array($paypal_order['status'], ['APPROVED', 'SAVED']) || + $paypal_total != $order_total[0]['amount'] || + $paypal_amount['currency_code'] != $order_total[0]['currency_code']) { + return FALSE; + } + // Store the intent for later reuse, it can't be updated, so no risk in + // being out of sync. + $order->data['commerce_paypal_checkout']['intent'] = strtolower($paypal_order['intent']); + + $payer = $paypal_order['payer']; + // If the user is anonymous, add their PayPal e-mail to the order. + if (empty($order->mail)) { + $order->mail = $payer['email_address']; + } + // Create a billing information profile for the order with the available info. + if (!empty($payment_method['settings']['update_billing_profiles'])) { + commerce_paypal_checkout_customer_profile($order, 'billing', $paypal_order); + } + // If the shipping module exists on the site, create a shipping information + // profile for the order with the available info. + if (module_exists('commerce_shipping') && !empty($payment_method['settings']['update_shipping_profiles'])) { + commerce_paypal_checkout_customer_profile($order, 'shipping', $paypal_order); + } + + // Recalculate the price of products on the order in case taxes have + // changed or prices have otherwise been affected. + if ($flow == 'shortcut') { + commerce_cart_order_refresh($order); + } + // Save the changes to the order. + commerce_order_save($order); + + if ($flow == 'mark') { + return commerce_paypal_checkout_do_payment($order, $payment_method); + } +} + +/** + * Capture/authorize a PayPal order and create a payment transaction. + * + * @param $order + * The order the payment is for. + * @param $payment_method + * The PayPal Checkout payment method instance whose settings should + * be used to submit the request. + * + * @return + * Boolean indicating the success or failure of the payment request. + */ +function commerce_paypal_checkout_do_payment($order, $payment_method) { + if (empty($order->data['commerce_paypal_checkout']['remote_id'])) { + return FALSE; + } + $paypal_checkout_data = $order->data['commerce_paypal_checkout']; + $intent = isset($paypal_checkout_data['intent']) ? $paypal_checkout_data['intent'] : $payment_method['settings']['intent']; + $api_client = commerce_paypal_checkout_api_client($payment_method['settings']); + try { + if ($intent == 'capture') { + $response = $api_client->captureOrder($paypal_checkout_data['remote_id']); + $remote_payment = $response['purchase_units'][0]['payments']['captures'][0]; + } + else { + $response = $api_client->authorizeOrder($paypal_checkout_data['remote_id']); + $remote_payment = $response['purchase_units'][0]['payments']['authorizations'][0]; + } + } + catch (\Exception $exception) { + watchdog_exception('commerce_paypal_checkout', $exception); + // Display an error message and remain on the same page. + drupal_set_message(t('We could not complete your payment with PayPal. Please try again or contact us if the problem persists.'), 'error'); + watchdog('commerce_paypal_checkout', 'PayPal Checkout transaction failed for order @order_number.', array('@order_number' => $order->order_number), WATCHDOG_ERROR); + return FALSE; + } + $remote_status = strtolower($remote_payment['status']); + + // Prepare a transaction object to log the API response. + $transaction = commerce_payment_transaction_new('paypal_checkout', $order->order_id); + $transaction->instance_id = $payment_method['instance_id']; + $transaction->amount = commerce_currency_decimal_to_amount($remote_payment['amount']['value'], $remote_payment['amount']['currency_code']); + $transaction->currency_code = $remote_payment['amount']['currency_code']; + $transaction->payload[REQUEST_TIME] = $response; + $transaction->remote_id = $remote_payment['id']; + $transaction->remote_status = $remote_payment['status']; + + // Store the transaction ID as the parent transaction ID in case subsequent + // API operations alter this transaction's remote ID. + if (!empty($transaction->remote_id)) { + $transaction->data['commerce_paypal_checkout']['original_remote_id'] = $transaction->remote_id; + } + + if (in_array($remote_status, ['denied', 'expired', 'declined'])) { + $transaction->status = COMMERCE_PAYMENT_STATUS_FAILURE; + commerce_payment_transaction_save($transaction); + // Display an error message and remain on the same page. + drupal_set_message(t('We could not complete your payment with PayPal. Please try again or contact us if the problem persists.'), 'error'); + watchdog('commerce_paypal_checkout', 'PayPal Checkout transaction failed for order @order_number.', array('@order_number' => $order->order_number), WATCHDOG_ERROR); + return FALSE; + } + + // Map the remote status to a Drupal commerce payment status. + $status_mapping = array( + 'created' => COMMERCE_PAYMENT_STATUS_PENDING, + 'pending' => COMMERCE_PAYMENT_STATUS_PENDING, + 'completed' => COMMERCE_PAYMENT_STATUS_SUCCESS, + ); + + // If we do not know how to handle this remote payment status, set the payment + // status to failure and stop here. + if (!isset($status_mapping[$remote_status])) { + $transaction->status = COMMERCE_PAYMENT_STATUS_FAILURE; + commerce_payment_transaction_save($transaction); + // Display an error message and remain on the same page. + drupal_set_message(t('We could not complete your payment with PayPal. Please try again or contact us if the problem persists.'), 'error'); + watchdog('commerce_paypal_checkout', 'PayPal Checkout transaction failed for order @order_number.', array('@order_number' => $order->order_number), WATCHDOG_ERROR); + return FALSE; + } + $transaction->status = $status_mapping[$remote_status]; + commerce_payment_transaction_save($transaction); + return TRUE; +} + +/** + * Creates or updates a customer profile for an order based on information + * obtained from PayPal after a payment via PayPal Checkout. + * + * @param $order + * The order that was paid via PayPal Checkout. + * @param $profile_type + * The type of the customer profile that should be created or updated. + * @param array $paypal_order + * The PayPal order retrieved via the getOrder() API call. + * @param $skip_save + * Boolean indicating whether or not this function should skip saving the + * order after setting it to reference the newly created customer profile; + * defaults to TRUE, requiring the caller to save the order. + */ +function commerce_paypal_checkout_customer_profile($order, $profile_type, $paypal_order, $skip_save = TRUE) { + // First check if the order already references a customer profile of the + // specified type. + $order_wrapper = entity_metadata_wrapper('commerce_order', $order); + $field_name = variable_get('commerce_customer_profile_' . $profile_type . '_field', ''); + + // If the associated order field has been set and the order currently + // references a customer profile through it... + if (!empty($field_name) && !empty($order_wrapper->{$field_name})) { + // Update the existing customer profile. + $profile = $order_wrapper->{$field_name}->value(); + } + elseif (!empty($order->data['profiles']['customer_profile_' . $profile_type])) { + // Otherwise look for an association stored in the order's data array. + $profile = commerce_customer_profile_load($order->data['profiles']['customer_profile_' . $profile_type]); + } + + // Create a new profile if we could not find an existing one. + if (empty($profile)) { + $profile = commerce_customer_profile_new($profile_type, $order->uid); + } + + // Add the order context to the profile to ensure it can be updated without + // resulting in customer profile duplication. + $profile->entity_context = array( + 'entity_type' => 'commerce_order', + 'entity_id' => $order->order_id, + ); + + // Prepare an addressfield array to set to the customer profile. + $field = field_info_field('commerce_customer_address'); + $instance = field_info_instance('commerce_customer_profile', 'commerce_customer_address', $profile_type); + $address = addressfield_default_values($field, $instance); + + $paypal_address = array(); + // Use the first name and last name if the profile is a billing profile. + if ($profile_type == 'billing') { + $address['first_name'] = $paypal_order['payer']['name']['given_name']; + $address['last_name'] = $paypal_order['payer']['name']['surname']; + $address['name_line'] = $address['first_name'] . ' ' . $address['last_name']; + if (isset($paypal_order['payer']['address'])) { + $paypal_address = $paypal_order['payer']['address']; + } + } + // Otherwise if it's a shipping profile, populate the address with all of + // the shipping information returned from PayPal. + elseif ($profile_type == 'shipping') { + // If no shipping info was returned by PayPal. + if (empty($paypal_order['purchase_units'][0]['shipping'])) { + return; + } + $shipping_info = $paypal_order['purchase_units'][0]['shipping']; + $names = explode(' ', $shipping_info['name']['full_name']); + $first_name = array_shift($names); + $last_name = implode(' ', $names); + $address['first_name'] = $first_name; + $address['last_name'] = $last_name; + $address['name_line'] = $first_name . ' ' . $last_name; + if (isset($shipping_info['address'])) { + $paypal_address = $shipping_info['address']; + } + } + + if ($paypal_address) { + // Map PayPal address keys to Address field keys. + $mapping = array( + 'address_line_1' => 'thoroughfare', + 'address_line_2' => 'premise', + 'admin_area_1' => 'administrative_area', + 'admin_area_2' => 'locality', + 'postal_code' => 'postal_code', + 'country_code' => 'country', + ); + foreach ($paypal_address as $key => $value) { + if (!isset($mapping[$key])) { + continue; + } + // PayPal address fields have a higher maximum length than ours. + $value = $key == 'country_code' ? $value : mb_substr($value, 0, 255); + $address[$mapping[$key]] = $value; + } + } + + // Add the addressfield value to the customer profile. + $profile_wrapper = entity_metadata_wrapper('commerce_customer_profile', $profile); + $profile_wrapper->commerce_customer_address = $address; + + // Save the customer profile and update the order to reference it. + $profile_wrapper->save(); + $order_wrapper->{'commerce_customer_' . $profile_type} = $profile_wrapper; + + // Save the order if specified. + if (!$skip_save) { + $order_wrapper->save(); + } +} + +/** + * Update the PayPal order. + * + * @param $order + * The order. + * @param $payment_method + * The payment method instance. + * + * @return bool + * TRUE if the update was successful, FALSE otherwise. + * + * @see commerce_paypal_checkout_review_pane_checkout_form_submit() + */ +function commerce_paypal_checkout_update_order($order, $payment_method) { + if (!isset($order->data['commerce_paypal_checkout']['remote_id'])) { + return FALSE; + } + $remote_id = $order->data['commerce_paypal_checkout']['remote_id']; + $request_body = commerce_paypal_checkout_prepare_order_request($order, $payment_method['settings']); + $update_params = array( + array( + 'op' => 'replace', + 'path' => "/purchase_units/@reference_id=='default'", + 'value' => $request_body['purchase_units'][0], + ), + ); + drupal_alter('commerce_paypal_checkout_update_order_request', $update_params, $order); + $api_client = commerce_paypal_checkout_api_client($payment_method['settings']); + try { + $api_client->updateOrder($remote_id, $update_params); + // Assume the update worked if we ended up here, if the update failed, + // an exception was thrown. + return TRUE; + } + catch (\Exception $exception) { + watchdog_exception('commerce_paypal_checkout', $exception); + return FALSE; + } +} + +/** + * Formats a price amount into a decimal value as expected by PayPal. + * + * @param $amount + * An integer price amount. + * @param $currency_code + * The currency code of the price. + * + * @return + * The decimal price amount as expected by PayPal API servers. + */ +function commerce_paypal_checkout_price_amount($amount, $currency_code) { + $rounded_amount = commerce_currency_round($amount, commerce_currency_load($currency_code)); + return number_format(commerce_currency_amount_to_decimal($rounded_amount, $currency_code), 2, '.', ''); +} diff --git a/www7/sites/all/modules/contrib/commerce_paypal/modules/checkout/css/commerce_paypal_checkout.css b/www7/sites/all/modules/contrib/commerce_paypal/modules/checkout/css/commerce_paypal_checkout.css new file mode 100644 index 000000000..deeda620b --- /dev/null +++ b/www7/sites/all/modules/contrib/commerce_paypal/modules/checkout/css/commerce_paypal_checkout.css @@ -0,0 +1,9 @@ +@media all and (min-width: 61em) { + .page-cart .paypal-buttons-container { + float: right; + width: 50%; + } +} +.page-checkout-payment .checkout-help { + display: none; +} diff --git a/www7/sites/all/modules/contrib/commerce_paypal/modules/checkout/includes/commerce_paypal_checkout.admin.inc b/www7/sites/all/modules/contrib/commerce_paypal/modules/checkout/includes/commerce_paypal_checkout.admin.inc new file mode 100644 index 000000000..668a337ed --- /dev/null +++ b/www7/sites/all/modules/contrib/commerce_paypal/modules/checkout/includes/commerce_paypal_checkout.admin.inc @@ -0,0 +1,297 @@ +instance_id); + $form_state['payment_method'] = $payment_method; + $balance = commerce_payment_order_balance($order); + + + if ($balance['amount'] > 0 && $balance['amount'] < $transaction->amount) { + $default_amount = $balance['amount']; + } + else { + $default_amount = $transaction->amount; + } + + // Convert the price amount to a user friendly decimal value. + $default_amount = number_format(commerce_currency_amount_to_decimal($default_amount, $transaction->currency_code), 2, '.', ''); + + $description = implode('
', array( + t('Authorization: @amount', array('@amount' => commerce_currency_format($transaction->amount, $transaction->currency_code))), + t('Order balance: @balance', array('@balance' => commerce_currency_format($balance['amount'], $balance['currency_code']))), + )); + + $form['amount'] = array( + '#type' => 'textfield', + '#title' => t('Capture amount'), + '#description' => $description, + '#default_value' => $default_amount, + '#field_suffix' => check_plain($transaction->currency_code), + '#size' => 16, + ); + + $form = confirm_form($form, + t('What amount do you want to capture?'), + 'admin/commerce/orders/' . $order->order_id . '/payment', + '', + t('Capture'), + t('Cancel'), + 'confirm' + ); + + return $form; +} + +/** + * Validate handler: ensure a valid amount is given. + */ +function commerce_paypal_checkout_capture_form_validate($form, &$form_state) { + $transaction = $form_state['transaction']; + $amount = $form_state['values']['amount']; + + // Ensure a positive numeric amount has been entered for capture. + if (!is_numeric($amount) || $amount <= 0) { + form_set_error('amount', t('You must specify a positive numeric amount to capture.')); + } + + // Ensure the amount is within the allowed limit for PayPal authorizations. + $authorization_amount = commerce_currency_amount_to_decimal($transaction->amount, $transaction->currency_code); + $authorization_amount_upper = commerce_currency_amount_to_decimal($transaction->amount, $transaction->currency_code) + 75; + + if ($amount > $authorization_amount * 1.15 || $amount > $authorization_amount_upper) { + form_set_error('amount', t('You cannot capture an amount $75 or 115% greater than the authorization amount in PayPal Checkout.')); + } + + // If the authorization has expired, display an error message and redirect. + if (REQUEST_TIME - $transaction->created > 86400 * 29) { + drupal_set_message(t('This authorization has passed its 29 day limit and cannot be captured.'), 'error'); + drupal_goto('admin/commerce/orders/' . $form_state['order']->order_id . '/payment'); + } +} + +/** + * Submit handler: process a prior authorization capture via PayPal Checkout. + */ +function commerce_paypal_checkout_capture_form_submit($form, &$form_state) { + $transaction = $form_state['transaction']; + $order = $form_state['order']; + $order_wrapper = entity_metadata_wrapper('commerce_order', $order); + $order_total = $order_wrapper->commerce_order_total->value(); + $amount = commerce_currency_decimal_to_amount($form_state['values']['amount'], $transaction->currency_code); + $api_client = commerce_paypal_checkout_api_client($form_state['payment_method']['settings']); + + $params = array( + 'final_capture' => $order_total['amount'] == $amount, + 'amount' => array( + 'value' => commerce_paypal_checkout_price_amount($amount, $transaction->currency_code), + 'currency_code' => $transaction->currency_code, + ), + ); + + try { + // If the transaction was authorized more than 3 days ago, we need to + // re-authorize the payment. + if (time() >= ($transaction->created + (86400 * 3))) { + $api_client->reAuthorizePayment($transaction->remote_id, array('amount' => $params['amount'])); + } + + $response = $api_client->capturePayment($transaction->remote_id, $params); + if (strtolower($response['status']) == 'completed') { + $transaction->status = COMMERCE_PAYMENT_STATUS_SUCCESS; + } + else { + $transaction->status = COMMERCE_PAYMENT_STATUS_FAILURE; + } + $transaction->remote_id = $response['id']; + $transaction->amount = $amount; + $transaction->remote_status = $response['status']; + // Note the capture in the transaction message. + $transaction->message .= '
' . t('Captured: @date', array('@date' => format_date(REQUEST_TIME, 'short'))); + $transaction->payload[REQUEST_TIME . '-capture'] = $response; + // Save the updated original transaction. + commerce_payment_transaction_save($transaction); + } + catch (PayPalCheckoutHttpException $exception) { + watchdog_exception('commerce_paypal_checkout', $exception); + // Display an error message but leave the transaction pending. + drupal_set_message(t('Prior authorization capture failed, so the transaction will remain in a pending status.'), 'error'); + } + + // Redirect back to the current order payment page. + $form_state['redirect'] = 'admin/commerce/orders/' . $form_state['order']->order_id . '/payment'; +} + +/** + * Form callback: allows the user to void a transaction. + */ +function commerce_paypal_checkout_void_form($form, &$form_state, $order, $transaction) { + $form_state['order'] = $order; + $form_state['transaction'] = $transaction; + + // Load and store the payment method instance for this transaction. + $payment_method = commerce_payment_method_instance_load($transaction->instance_id); + $form_state['payment_method'] = $payment_method; + + $form['markup'] = array( + '#markup' => t('Are you sure that you want to void this transaction?'), + ); + + $form = confirm_form($form, + t('Are you sure that you want to void this transaction?'), + 'admin/commerce/orders/' . $order->order_id . '/payment', + '', + t('Void'), + t('Cancel'), + 'confirm' + ); + + return $form; +} + +/** + * Submit handler: process the void request. + */ +function commerce_paypal_checkout_void_form_submit($form, &$form_state) { + $transaction = $form_state['transaction']; + $api_client = commerce_paypal_checkout_api_client($form_state['payment_method']['settings']); + + try { + // This API call doesn't return content, so nothing to check, if no + // exception is thrown, this means the void operation succeeded. + $api_client->voidPayment($transaction->remote_id); + drupal_set_message(t('Transaction successfully voided.')); + + // Set the remote and local status accordingly. + $transaction->status = COMMERCE_PAYMENT_STATUS_FAILURE; + $transaction->remote_status = COMMERCE_CREDIT_VOID; + + // Update the transaction message to show that it has been voided. + $transaction->message .= '
' . t('Voided: @date', array('@date' => format_date(REQUEST_TIME, 'short'))); + commerce_payment_transaction_save($transaction); + } + catch (PayPalCheckoutHttpException $exception) { + drupal_set_message(t('The void operation failed, so the transaction will remain in a pending status.'), 'error'); + } + $form_state['redirect'] = 'admin/commerce/orders/' . $form_state['order']->order_id . '/payment'; +} + +/** + * Form callback: allows the user to issue a credit on a prior transaction. + */ +function commerce_paypal_checkout_refund_form($form, &$form_state, $order, $transaction) { + $form_state['order'] = $order; + $form_state['transaction'] = $transaction; + + // Load and store the payment method instance for this transaction. + $payment_method = commerce_payment_method_instance_load($transaction->instance_id); + $form_state['payment_method'] = $payment_method; + + $default_amount = number_format(commerce_currency_amount_to_decimal($transaction->amount, $transaction->currency_code), 2, '.', ''); + + $form['amount'] = array( + '#type' => 'textfield', + '#title' => t('Refund amount'), + '#description' => t('Enter the amount to be refunded back to the PayPal account.'), + '#default_value' => $default_amount, + '#field_suffix' => check_plain($transaction->currency_code), + '#size' => 16, + ); + + $form = confirm_form($form, + t('What amount do you want to refund?'), + 'admin/commerce/orders/' . $order->order_id . '/payment', + '', + t('Refund'), + t('Cancel'), + 'confirm' + ); + + return $form; +} + +/** + * Validate handler: check the credit amount before attempting a refund request. + */ +function commerce_paypal_checkout_refund_form_validate($form, &$form_state) { + $transaction = $form_state['transaction']; + $amount = $form_state['values']['amount']; + + // Ensure a positive numeric amount has been entered for refund. + if (!is_numeric($amount) || $amount <= 0) { + form_set_error('amount', t('You must specify a positive numeric amount to refund.')); + } + + // Ensure the amount is less than or equal to the captured amount. + if ($amount > commerce_currency_amount_to_decimal($transaction->amount, $transaction->currency_code)) { + form_set_error('amount', t('You cannot refund more than you captured.')); + } + + // If the transaction is older than 60 days, display an error message and redirect. + if (time() - $transaction->created > 86400 * 60) { + drupal_set_message(t('This transaction has passed its 60 day limit for issuing refunds.'), 'error'); + drupal_goto('admin/commerce/orders/' . $form_state['order']->order_id . '/payment'); + } +} + +/** + * Submit handler: process a refund request. + */ +function commerce_paypal_checkout_refund_form_submit($form, &$form_state) { + $transaction = $form_state['transaction']; + $payment_method = $form_state['payment_method']; + + $order = $form_state['order']; + $order_wrapper = entity_metadata_wrapper('commerce_order', $order); + $order_total = $order_wrapper->commerce_order_total->value(); + $api_client = commerce_paypal_checkout_api_client($form_state['payment_method']['settings']); + $amount = commerce_currency_decimal_to_amount($form_state['values']['amount'], $order_total['currency_code']); + + $params = array( + 'amount' => array( + 'value' => commerce_paypal_checkout_price_amount($amount, $transaction->currency_code), + 'currency_code' => $transaction->currency_code, + ), + ); + try { + $response = $api_client->refundPayment($transaction->remote_id, $params); + // Not sure this can happen... + if (strtolower($response['status']) != 'completed') { + drupal_set_message(t('Refund failed')); + } + else { + drupal_set_message(t('Refund for @amount issued successfully.', array('@amount' => commerce_currency_format($amount, $transaction->currency_code)))); + + // Create a new transaction to record the credit. + $credit_transaction = commerce_payment_transaction_new($payment_method['method_id'], $order->order_id); + $credit_transaction->instance_id = $payment_method['instance_id']; + $credit_transaction->remote_id = $response['id']; + $credit_transaction->amount = $amount * -1; + $credit_transaction->currency_code = $transaction->currency_code; + $credit_transaction->payload[REQUEST_TIME] = $response; + $credit_transaction->status = COMMERCE_PAYMENT_STATUS_SUCCESS; + $credit_transaction->remote_status = COMMERCE_CREDIT_CREDIT; + $credit_transaction->message = t('Refunded to @remote_id.', array('@remote_id' => $transaction->remote_id)); + + // Save the credit transaction. + commerce_payment_transaction_save($credit_transaction); + } + } + catch (PayPalCheckoutHttpException $exception) { + watchdog_exception('commerce_paypal_checkout', $exception); + drupal_set_message(t('Refund failed')); + } + + $form_state['redirect'] = 'admin/commerce/orders/' . $order->order_id . '/payment'; +} diff --git a/www7/sites/all/modules/contrib/commerce_paypal/modules/checkout/includes/commerce_paypal_checkout.checkout_pane.inc b/www7/sites/all/modules/contrib/commerce_paypal/modules/checkout/includes/commerce_paypal_checkout.checkout_pane.inc new file mode 100644 index 000000000..fa8d1e530 --- /dev/null +++ b/www7/sites/all/modules/contrib/commerce_paypal/modules/checkout/includes/commerce_paypal_checkout.checkout_pane.inc @@ -0,0 +1,206 @@ + $checkout_pane) { + if (!in_array($pane_id, $excluded_panes) && $checkout_pane['module'] != 'commerce_customer' ) { + $options[$pane_id] = check_plain($checkout_pane['name']); + } + } + + // If we have available checkout panes for this page... + if (!empty($options)) { + // Allow the administrator to choose which panes the customer should see + // upon returning from PayPal Checkout. + $form['commerce_paypal_checkout_review_embedded_panes'] = array( + '#type' => 'checkboxes', + '#title' => t('Checkout panes to include on the PayPal Checkout review and confirm page'), + '#options' => $options, + '#default_value' => variable_get('commerce_paypal_checkout_review_embedded_panes', array()), + ); + } + + return $form; +} + +/** + * Checkout pane callback: returns a pane allowing the customer to review the + * final details of the order and provide any final information required. + */ +function commerce_paypal_checkout_review_pane_checkout_form(&$form, &$form_state, $checkout_pane, $order) { + $form_state['build_info']['files']['pane'] = drupal_get_path('module', 'commerce_paypal_checkout') . '/includes/commerce_paypal_checkout.checkout_pane.inc'; + $form_state['order'] = $order; + + // Adjust the weights of the help text and pane form to appear before the + // embedded checkout panes. + $form['help']['#weight'] = -10; + + $pane_form = array( + '#weight' => -5, + ); + + // Duplicate the review checkout pane's contents. + $pane_form['review'] = array( + '#theme' => 'commerce_checkout_review', + '#data' => array(), + ); + + // Loop through all the pages before the review page... + foreach (commerce_checkout_pages() as $page_id => $checkout_page) { + // Exit the loop once the review page is reached. + if ($page_id == 'review') { + break; + } + + // Loop through all the panes on the current page specifying review... + foreach (commerce_checkout_panes(array('page' => $page_id, 'enabled' => TRUE, 'review' => TRUE)) as $pane_id => $checkout_pane_local) { + // If the pane has a valid review callback... + if ($callback = commerce_checkout_pane_callback($checkout_pane_local, 'review')) { + // Get the review data for this pane. + $pane_data = $callback($form, $form_state, $checkout_pane_local, $order); + + // Only display the pane if there is data in the pane. + if (!empty($pane_data)) { + // Add a row for it in the review data. + $pane_form['review']['#data'][$pane_id] = array( + 'title' => $checkout_pane_local['title'], + 'data' => $pane_data, + ); + } + } + } + } + + // Embed other specified checkout panes in this checkout pane. + foreach (commerce_paypal_checkout_embedded_checkout_panes() as $embedded_pane_id) { + $embedded_pane = commerce_checkout_pane_load($embedded_pane_id); + + // If the checkout pane defines a checkout form callback... + if ($embedded_pane && $callback = commerce_checkout_pane_callback($embedded_pane, 'checkout_form')) { + // Get the form for the embedded checkout pane. + $embedded_pane_form = $callback($form, $form_state, $embedded_pane, $order); + + // Embed it on this checkout pane if the form returned data. + if (!empty($embedded_pane_form)) { + $form[$embedded_pane_id] = array( + '#type' => $embedded_pane['fieldset'] ? 'fieldset' : 'container', + '#title' => check_plain($embedded_pane['title']), + '#collapsible' => $embedded_pane['collapsible'], + '#collapsed' => $embedded_pane['collapsed'], + '#attributes' => array('class' => array($embedded_pane_id)), + '#tree' => TRUE, + ) + $embedded_pane_form; + } + } + } + + $form['pay_now'] = array( + '#type' => 'submit', + '#value' => t('Pay Now'), + '#validate' => array('commerce_paypal_checkout_review_pane_checkout_form_validate'), + '#submit' => array('commerce_paypal_checkout_review_pane_checkout_form_submit'), + ); + + return $pane_form; +} + +/** + * Validate handler for the PayPal Checkout review and confirm page. + */ +function commerce_paypal_checkout_review_pane_checkout_form_validate($form, &$form_state) { + $order = $form_state['order']; + $form_validate = TRUE; + + // Loop through the enabled checkout panes included in this page to validate + // and submit them. + foreach (commerce_paypal_checkout_embedded_checkout_panes() as $embedded_pane_id) { + $embedded_pane_validate = TRUE; + + // Load the checkout pane to find its checkout pane validate callback. + $embedded_checkout_pane = commerce_checkout_pane_load($embedded_pane_id); + + // If it has a validate callback. + if ($callback = commerce_checkout_pane_callback($embedded_checkout_pane, 'checkout_form_validate')) { + // Give it a chance to process the submitted data. + $embedded_pane_validate &= $callback($form, $form_state, $embedded_checkout_pane, $order); + } + + // Submit the pane if it validated. + if ($embedded_pane_validate && $callback = commerce_checkout_pane_callback($embedded_checkout_pane, 'checkout_form_submit')) { + $callback($form, $form_state, $embedded_checkout_pane, $order); + } + + // A failed pane makes the form fail. + $form_validate &= $embedded_pane_validate; + } + + // Save the updated order object. + commerce_order_save($order); + + // If a pane failed validation or the form state has otherwise been altered to + // initiate a rebuild, return without moving to the next checkout page. + if (!$form_validate || $form_state['rebuild']) { + $form_state['rebuild'] = TRUE; + } +} + +/** + * Submit handler for the PayPal Checkout review and confirm page. + */ +function commerce_paypal_checkout_review_pane_checkout_form_submit($form, &$form_state) { + $order = $form_state['order']; + $payment_method = commerce_payment_method_instance_load($order->data['payment_method']); + // Update the order in PayPal to make sure it reflects the current state of + // the order in Drupal. + // If the order could not be updated in PayPal, stop here. + if (!commerce_paypal_checkout_update_order($order, $payment_method)) { + drupal_set_message(t('We could not complete your payment with PayPal. Please try again or contact us if the problem persists.'), 'error'); + watchdog('commerce_paypal_checkout', 'Could not update the PayPal order for order @order_number.', array('@order_number' => $order->order_number), WATCHDOG_ERROR); + return; + } + + // Attempt to process the payment. + if (commerce_paypal_checkout_do_payment($order, $payment_method)) { + // Proceed to the next page if it succeeded. + $order_status = commerce_order_status_load($order->status); + $checkout_page = commerce_checkout_page_load($order_status['checkout_page']); + $next_page = $checkout_page['next_page']; + + // Update the order status to the next checkout page. + $order = commerce_order_status_update($order, 'checkout_' . $next_page, FALSE, FALSE); + + // Inform modules of checkout completion if the next page is completed. + if ($next_page == 'complete') { + commerce_checkout_complete($order); + } + + // Redirect to the URL for the new checkout page. + $form_state['redirect'] = commerce_checkout_order_uri($order); + } +} diff --git a/www7/sites/all/modules/contrib/commerce_paypal/modules/checkout/js/commerce_paypal_checkout.js b/www7/sites/all/modules/contrib/commerce_paypal/modules/checkout/js/commerce_paypal_checkout.js new file mode 100644 index 000000000..308429faf --- /dev/null +++ b/www7/sites/all/modules/contrib/commerce_paypal/modules/checkout/js/commerce_paypal_checkout.js @@ -0,0 +1,66 @@ +/** + * @file + * Renders the PayPal Smart payment buttons. + */ + +(function($) { + + Drupal.paypalCheckout = { + renderButtons: function(settings) { + $('.paypal-buttons-container').once('rendered').each(function() { + paypal.Buttons({ + createOrder: function() { + return fetch(settings.createOrderUri) + .then(function(res) { + return res.json(); + }).then(function(data) { + return data.id ? data.id : ''; + }); + }, + onApprove: function (data) { + return fetch(settings.onApproveUri, { + method: 'post', + body: JSON.stringify({ + id: data.orderID + }) + }).then(function(res) { + return res.json(); + }).then(function(data) { + if (data.hasOwnProperty('redirectUri')) { + window.location.href = data.redirectUri; + } + }); + }, + style: settings['style'] + }).render('#' + $(this).attr('id')); + }); + }, + initialize: function (context, settings) { + if (context === document) { + var script = document.createElement('script'); + script.src = settings.src; + script.type = 'text/javascript'; + script.setAttribute('data-partner-attribution-id', 'CommerceGuys_Cart_SPB'); + document.getElementsByTagName('head')[0].appendChild(script); + } + var waitForSdk = function(settings) { + if (typeof paypal !== 'undefined') { + Drupal.paypalCheckout.renderButtons(settings); + } + else { + setTimeout(function() { + waitForSdk(settings) + }, 100); + } + }; + waitForSdk(settings); + } + }; + + Drupal.behaviors.commercePaypalCheckout = { + attach: function(context, settings) { + Drupal.paypalCheckout.initialize(context, settings.paypalCheckout); + } + }; + +}(jQuery)); diff --git a/www7/sites/all/modules/contrib/commerce_paypal/modules/checkout/lib/PayPalCheckoutClient.php b/www7/sites/all/modules/contrib/commerce_paypal/modules/checkout/lib/PayPalCheckoutClient.php new file mode 100644 index 000000000..335960d7e --- /dev/null +++ b/www7/sites/all/modules/contrib/commerce_paypal/modules/checkout/lib/PayPalCheckoutClient.php @@ -0,0 +1,393 @@ +config = $config; + $this->logger = $logger; + $this->retryCount = 0; + $this->headers = array( + 'Accept' => 'application/json', + // Defaults the "Content-Type" header to application/json. + 'Content-Type' => 'application/json', + 'PayPal-Partner-Attribution-Id' => 'CommerceGuys_Cart_SPB', + ); + } + + /** + * Sets the default cURL options. + */ + public static function setDefaultCurlOptions($ch) { + curl_setopt($ch, CURLOPT_HEADER, FALSE); + curl_setopt($ch, CURLINFO_HEADER_OUT, TRUE); + curl_setopt($ch, CURLOPT_RETURNTRANSFER, TRUE); + curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, TRUE); + curl_setopt($ch, CURLOPT_VERBOSE, FALSE); + curl_setopt($ch, CURLOPT_CONNECTTIMEOUT, 30); + curl_setopt($ch, CURLOPT_TIMEOUT, 180); + } + + /** + * Send a message to the logger. + * + * @param string $message + * The message to log. + */ + public function logMessage($message) { + if (is_callable($this->logger)) { + call_user_func($this->logger, $message); + } + } + + /** + * Returns the base URL for the API, based on the specified mode. + * + * @return string + * The base URL for the API that query parameters will be appended to when + * submitting API requests. + */ + public function baseUrl() { + switch ($this->config['server']) { + case 'live': + return 'https://api.paypal.com'; + + case 'sandbox': + default: + return 'https://api.sandbox.paypal.com'; + } + } + + /** + * Acquire an access token from PayPal. + * + * @return string[] + * The API response JSON converted to an associative array. + * + * @throws PayPalCheckoutAuthenticationException + */ + protected function acquireAccessToken() { + $ch = curl_init(); + static::setDefaultCurlOptions($ch); + $headers = array( + 'Authorization' => 'Basic ' . base64_encode($this->config['client_id'] . ':' . $this->config['secret']), + 'Content-Type' => 'application/x-www-form-urlencoded', + ) + $this->headers; + $url = $this->baseUrl() . '/v1/oauth2/token'; + curl_setopt($ch, CURLOPT_HTTPHEADER, static::formatHeaders($headers)); + curl_setopt($ch, CURLOPT_CUSTOMREQUEST, 'POST'); + curl_setopt($ch, CURLOPT_URL, $url); + curl_setopt($ch, CURLOPT_POSTFIELDS, 'grant_type=client_credentials'); + + // Submit the request to the PayPal server. + $response = curl_exec($ch); + + // Ensure we got a successful response by inspecting the HTTP status code. + $response_info = curl_getinfo($ch); + + if (is_resource($ch)) { + curl_close($ch); + } + + $json = drupal_json_decode($response); + + if ($response_info['http_code'] == 401) { + // Throw an exception indicating authentication failed. + $message = 'Authentication failed.'; + if (isset($json['error_description'])) { + $message = sprintf('Error description: %s.', $json['error_description']); + } + throw new PayPalCheckoutAuthenticationException($message); + } + + return $json; + } + + /** + * Returns the access token stored locally if any, or get one from PayPal. + * + * @return string + * The Oauth2 access token. + * + * @throws PayPalCheckoutAuthenticationException + */ + public function getAccessToken() { + $access_token = variable_get('commerce_paypal_checkout_access_token', array()); + if (!empty($access_token['token']) && $access_token['expires'] > time()) { + return $access_token['token']; + } + $response = $this->acquireAccessToken(); + variable_set('commerce_paypal_checkout_access_token', array( + 'token' => $response['access_token'], + 'expires' => time() + $response['expires_in'], + )); + return $response['access_token']; + } + + /** + * Creates an order in PayPal. + * + * @param array $parameters + * An array of parameters to include with the request. + * + * @return string[] + * The API response JSON converted to an associative array. + */ + public function createOrder($parameters) { + return $this->submitRequest('POST', 'v2/checkout/orders', $parameters); + } + + /** + * Get an existing order from PayPal. + * + * @param $remote_id + * The PayPal order ID. + * + * @return string[] + * The API response JSON converted to an associative array. + */ + public function getOrder($remote_id) { + return $this->submitRequest('GET', sprintf('v2/checkout/orders/%s', $remote_id)); + } + + /** + * Updates an existing PayPal order. + * + * @param $remote_id + * The PayPal order ID. + * @param array $parameters + * An array of parameters to include with the request. + * + * @return string[] + * The API response JSON converted to an associative array. + */ + public function updateOrder($remote_id, $parameters) { + return $this->submitRequest('PATCH', sprintf('v2/checkout/orders/%s', $remote_id), $parameters); + } + + /** + * Authorize payment for order. + * + * @param $remote_id + * The PayPal order ID. + * + * @return string[] + * The API response JSON converted to an associative array. + */ + public function authorizeOrder($remote_id) { + return $this->submitRequest('POST', sprintf('v2/checkout/orders/%s/authorize', $remote_id)); + + } + + /** + * Capture payment for order. + * + * @param $remote_id + * The PayPal order ID. + * + * @return string[] + * The API response JSON converted to an associative array. + */ + public function captureOrder($remote_id) { + return $this->submitRequest('POST', sprintf('v2/checkout/orders/%s/capture', $remote_id)); + } + + /** + * Captures an authorized payment, by ID. + * + * @param $authorization_id + * The PayPal-generated ID for the authorized payment to capture. + * @param array $parameters + * (optional An array of parameters to pass as the request body. + * + * @return string[] + * The API response JSON converted to an associative array. + */ + public function capturePayment($authorization_id, array $parameters = array()) { + return $this->submitRequest('POST', sprintf('v2/payments/authorizations/%s/capture', $authorization_id), $parameters); + } + + /** + * Reauthorizes an authorized PayPal account payment, by ID. + * + * @param $authorization_id + * The PayPal-generated ID of the authorized payment to reauthorize. + * @param array $parameters + * (optional An array of parameters to pass as the request body. + * + * @return string[] + * The API response JSON converted to an associative array. + */ + public function reAuthorizePayment($authorization_id, array $parameters = array()) { + return $this->submitRequest('POST', sprintf('v2/payments/authorizations/%s/reauthorize', $authorization_id), $parameters); + } + + /** + * Refunds a captured payment, by ID. + * + * @param $capture_id + * The PayPal-generated ID for the captured payment to refund. + * @param array $parameters + * (optional An array of parameters to pass as the request body. + * + * @return string[] + * The API response JSON converted to an associative array. + */ + public function refundPayment($capture_id, array $parameters = array()) { + return $this->submitRequest('POST', sprintf('v2/payments/captures/%s/refund', $capture_id), $parameters); + } + + /** + * Voids, or cancels, an authorized payment, by ID. + * + * @param $authorization_id + * The PayPal-generated ID of the authorized payment to void. + */ + public function voidPayment($authorization_id) { + return $this->submitRequest('POST', sprintf('v2/payments/authorizations/%s/void', $authorization_id)); + } + + /** + * Submits an API request to the PayPal server. + * + * @param string $method + * The HTTP method to use. One of: 'GET', 'POST', 'PATCH', 'PUT', DELETE'. + * @param string $path + * The remote path. The base URL will be automatically appended. + * @param array $parameters + * An array of parameters to include with the request. Optional. + * + * @throws PayPalCheckoutAuthenticationException if the request fails authentication. + * @throws PayPalCheckoutHttpServerErrorException if the response status code is 5xx. + * @throws PayPalCheckoutHttpClientErrorException if the response status code is 4xx. + * @throws PayPalCheckoutHttpRedirectionException if the response status code is 3xx. + * @throws PayPalCheckoutInvalidResponseJsonException if the response is not valid JSON. + * + * @return string[] + * The API response JSON converted to an associative array. + */ + public function submitRequest($method, $path, $parameters = array()) { + $this->headers['Authorization'] = 'Bearer ' . $this->getAccessToken(); + $url = $this->baseUrl() . '/' . $path; + $ch = curl_init(); + static::setDefaultCurlOptions($ch); + curl_setopt($ch, CURLOPT_CUSTOMREQUEST, $method); + if (!empty($parameters)) { + if ($this->headers['Content-Type'] == 'application/json') { + // JSON encode the fields and set them to the request body. + curl_setopt($ch, CURLOPT_POSTFIELDS, drupal_json_encode($parameters)); + } + else { + curl_setopt($ch, CURLOPT_POSTFIELDS, http_build_query($parameters, '', '&')); + } + } + curl_setopt($ch, CURLOPT_HTTPHEADER, static::formatHeaders($this->headers)); + + // Submit the request to the PayPal server. + $this->logMessage(sprintf('Request URL: %s', $url)); + curl_setopt($ch, CURLOPT_URL, $url); + $response = curl_exec($ch); + + // Ensure we got a successful response by inspecting the HTTP status code. + $response_info = curl_getinfo($ch); + + if (is_resource($ch)) { + curl_close($ch); + } + + if ($response_info['http_code'] == 401) { + // Attempt to get a new access token if the authentication failed. + // This might happen if we're sending an expired access token to PayPal. + if ($this->retryCount < static::RETRY_LIMIT) { + $this->retryCount++; + // Ensure we get a fresh access token next time. + variable_del('commerce_paypal_checkout_access_token'); + return $this->submitRequest($method, $path, $parameters); + } + + $json = drupal_json_decode($response); + // Throw an exception indicating authentication failed. + $message = 'Authentication failed.'; + if (isset($json['error_description'])) { + $message = sprintf('Error description: %s.', $json['error_description']); + } + throw new PayPalCheckoutAuthenticationException($message); + } + elseif ($response_info['http_code'] >= 500) { + // Throw an exception indicating a server error. + throw new PayPalCheckoutHttpServerErrorException('', $response_info['http_code']); + } + elseif ($response_info['http_code'] >= 400) { + // Throw an exception indicating a client error. + throw new PayPalCheckoutHttpClientErrorException('', $response_info['http_code']); + } + elseif ($response_info['http_code'] >= 300) { + // Throw an exception indicating a redirection that this library is not + // going to automatically follow. + throw new PayPalCheckoutHttpRedirectionException('', $response_info['http_code']); + } + + // Attempt to convert the response body to an associative array. + try { + $json = drupal_json_decode($response); + } + catch (\Exception $e) { + throw new PayPalCheckoutInvalidResponseJsonException('The API response string could not be parsed as JSON.'); + } + + return $json; + } + + protected static function formatHeaders(array $headers) { + $http_headers = array(); + foreach ($headers as $key => $value) { + $http_headers[] = "$key: $value"; + } + return $http_headers; + } + +} diff --git a/www7/sites/all/modules/contrib/commerce_paypal/modules/checkout/lib/PayPalCheckoutExceptions.php b/www7/sites/all/modules/contrib/commerce_paypal/modules/checkout/lib/PayPalCheckoutExceptions.php new file mode 100644 index 000000000..5006d9d69 --- /dev/null +++ b/www7/sites/all/modules/contrib/commerce_paypal/modules/checkout/lib/PayPalCheckoutExceptions.php @@ -0,0 +1,40 @@ +data['commerce_paypal_ec']['parenttransactionid'] = $transaction->remote_id; } + // Check if there is an error code in the response. + if (!empty($response['L_ERRORCODE0'])) { + // Log the error in a payment transaction. + $transaction->status = COMMERCE_PAYMENT_STATUS_FAILURE; + $transaction->remote_status = ''; + $message = array(); + $message[] = '' . t('Payment failed') . ''; + $message[] = t('Error @code: @message', array('@code' => $response['L_ERRORCODE0'], '@message' => $response['L_SHORTMESSAGE0'])); + $transaction->message = implode('
', $message); + commerce_payment_transaction_save($transaction); + + // If the response error indicates a funding failure, redirect the customer + // back to PayPal for another attempt. + // @see: https://developer.paypal.com/docs/classic/express-checkout/ht_ec_fundingfailure10486/ + if ($response['L_ERRORCODE0'] === '10486') { + // Log the error in watchdog. + watchdog('commerce_paypal_ec', 'PayPal Express Checkout transaction funding failed for order @order_number. Redirecting the user back to PayPal.', array('@order_number' => $order->order_number), WATCHDOG_NOTICE); + // Update order. + commerce_order_status_update($order, 'checkout_payment', FALSE, NULL, t('Customer payment transaction funding failed. Redirecting customer back to the PayPal Express Checkout page.')); + // Redirect the user back to PayPal. + drupal_goto(commerce_paypal_ec_checkout_url($payment_method['settings']['server'], $order->data['commerce_paypal_ec']['token'])); + } + + // Log the error in watchdog. + watchdog('commerce_paypal_ec', 'PayPal Express Checkout transaction failed for order @order_number.', array('@order_number' => $order->order_number), WATCHDOG_ERROR); + + return FALSE; + } + // If we received an unknown response status... if (!isset($response['PAYMENTINFO_0_PAYMENTSTATUS']) || !in_array($response['PAYMENTINFO_0_PAYMENTSTATUS'], array('Failed', 'Voided', 'Pending', 'Completed', 'Refunded'))) { // Display an error message and remain on the same page. diff --git a/www7/sites/all/modules/contrib/commerce_paypal/modules/payflow/commerce_payflow.info b/www7/sites/all/modules/contrib/commerce_paypal/modules/payflow/commerce_payflow.info index 9b1ce3718..954e25c0b 100644 --- a/www7/sites/all/modules/contrib/commerce_paypal/modules/payflow/commerce_payflow.info +++ b/www7/sites/all/modules/contrib/commerce_paypal/modules/payflow/commerce_payflow.info @@ -11,8 +11,8 @@ core = 7.x ; Simple tests ; files[] = tests/commerce_paypal_wps.test -; Information added by Drupal.org packaging script on 2018-10-22 -version = "7.x-2.6" +; Information added by Drupal.org packaging script on 2019-05-06 +version = "7.x-2.7" core = "7.x" project = "commerce_paypal" -datestamp = "1540244884" +datestamp = "1557130694" diff --git a/www7/sites/all/modules/contrib/commerce_paypal/modules/wpp/commerce_paypal_wpp.info b/www7/sites/all/modules/contrib/commerce_paypal/modules/wpp/commerce_paypal_wpp.info index 5651b4993..ee8da4976 100644 --- a/www7/sites/all/modules/contrib/commerce_paypal/modules/wpp/commerce_paypal_wpp.info +++ b/www7/sites/all/modules/contrib/commerce_paypal/modules/wpp/commerce_paypal_wpp.info @@ -11,8 +11,8 @@ core = 7.x ; Simple tests ; files[] = tests/commerce_paypal_wps.test -; Information added by Drupal.org packaging script on 2018-10-22 -version = "7.x-2.6" +; Information added by Drupal.org packaging script on 2019-05-06 +version = "7.x-2.7" core = "7.x" project = "commerce_paypal" -datestamp = "1540244884" +datestamp = "1557130694" diff --git a/www7/sites/all/modules/contrib/commerce_paypal/modules/wps/commerce_paypal_wps.info b/www7/sites/all/modules/contrib/commerce_paypal/modules/wps/commerce_paypal_wps.info index e79cb283b..7711fd3e4 100644 --- a/www7/sites/all/modules/contrib/commerce_paypal/modules/wps/commerce_paypal_wps.info +++ b/www7/sites/all/modules/contrib/commerce_paypal/modules/wps/commerce_paypal_wps.info @@ -11,8 +11,8 @@ core = 7.x ; Simple tests ; files[] = tests/commerce_paypal_wps.test -; Information added by Drupal.org packaging script on 2018-10-22 -version = "7.x-2.6" +; Information added by Drupal.org packaging script on 2019-05-06 +version = "7.x-2.7" core = "7.x" project = "commerce_paypal" -datestamp = "1540244884" +datestamp = "1557130694" diff --git a/www7/sites/all/modules/contrib/commerce_paypal/modules/wps/commerce_paypal_wps.module b/www7/sites/all/modules/contrib/commerce_paypal/modules/wps/commerce_paypal_wps.module index 818e42893..021358ef2 100644 --- a/www7/sites/all/modules/contrib/commerce_paypal/modules/wps/commerce_paypal_wps.module +++ b/www7/sites/all/modules/contrib/commerce_paypal/modules/wps/commerce_paypal_wps.module @@ -545,26 +545,213 @@ function commerce_paypal_wps_server_url($server) { function commerce_paypal_wps_languages() { return array( t('By country') => array( + 'AL' => t('Albania'), + 'DZ' => t('Algeria'), + 'AD' => t('Andorra'), + 'AO' => t('Angola'), + 'AI' => t('Anguilla'), + 'AG' => t('Antigua and Barbuda'), + 'AR' => t('Argentina'), + 'AM' => t('Armenia'), + 'AW' => t('Aruba'), 'AU' => t('Australia'), 'AT' => t('Austria'), + 'AZ' => t('Azerbaijan Republic'), + 'BS' => t('Bahamas'), + 'BH' => t('Bahrain'), + 'BB' => t('Barbados'), + 'BY' => t('Belarus'), 'BE' => t('Belgium'), + 'BZ' => t('Belize'), + 'BJ' => t('Benin'), + 'BM' => t('Bermuda'), + 'BT' => t('Bhutan'), + 'BO' => t('Bolivia'), + 'BA' => t('Bosnia and Herzegovina'), + 'BW' => t('Botswana'), 'BR' => t('Brazil'), + 'BN' => t('Brunei'), + 'BG' => t('Bulgaria'), + 'BF' => t('Burkina Faso'), + 'BI' => t('Burundi'), + 'KH' => t('Cambodia'), + 'CM' => t('Cameroon'), 'CA' => t('Canada'), - 'CN' => t('China'), + 'CV' => t('Cape Verde'), + 'KY' => t('Cayman Islands'), + 'TD' => t('Chad'), + 'CL' => t('Chile'), + 'C2' => t('China'), + 'CO' => t('Colombia'), + 'KM' => t('Comoros'), + 'CK' => t('Cook Islands'), + 'CR' => t('Costa Rica'), + 'CI' => t('Cote D\'Ivoire'), + 'HR' => t('Croatia'), + 'CY' => t('Cyprus'), + 'CZ' => t('Czech Republic'), + 'CD' => t('Democratic Republic of the Congo'), + 'DK' => t('Denmark'), + 'DJ' => t('Djibouti'), + 'DM' => t('Dominica'), + 'DO' => t('Dominican Republic'), + 'EC' => t('Ecuador'), + 'EG' => t('Egypt'), + 'SV' => t('El Salvador'), + 'ER' => t('Eritrea'), + 'EE' => t('Estonia'), + 'ET' => t('Ethiopia'), + 'FK' => t('Falkland Islands'), + 'FO' => t('Faroe Islands'), + 'FJ' => t('Fiji'), + 'FI' => t('Finland'), 'FR' => t('France'), + 'GF' => t('French Guiana'), + 'PF' => t('French Polynesia'), + 'GA' => t('Gabon Republic'), + 'GM' => t('Gambia'), + 'GE' => t('Georgia'), 'DE' => t('Germany'), + 'GI' => t('Gibraltar'), + 'GR' => t('Greece'), + 'GL' => t('Greenland'), + 'GD' => t('Grenada'), + 'GP' => t('Guadeloupe'), + 'GT' => t('Guatemala'), + 'GN' => t('Guinea'), + 'GW' => t('Guinea Bissau'), + 'GY' => t('Guyana'), + 'HN' => t('Honduras'), + 'HK' => t('Hong Kong'), + 'HU' => t('Hungary'), + 'IS' => t('Iceland'), + 'IN' => t('India'), + 'ID' => t('Indonesia'), + 'IE' => t('Ireland'), + 'IL' => t('Israel'), 'IT' => t('Italy'), + 'JM' => t('Jamaica'), + 'JP' => t('Japan'), + 'JO' => t('Jordan'), + 'KZ' => t('Kazakhstan'), + 'KE' => t('Kenya'), + 'KI' => t('Kiribati'), + 'KW' => t('Kuwait'), + 'KG' => t('Kyrgyzstan'), + 'LA' => t('Laos'), + 'LV' => t('Latvia'), + 'LS' => t('Lesotho'), + 'LI' => t('Liechtenstein'), + 'LT' => t('Lithuania'), + 'LU' => t('Luxembourg'), + 'MK' => t('Macedonia'), + 'MG' => t('Madagascar'), + 'MW' => t('Malawi'), + 'MY' => t('Malaysia'), + 'MV' => t('Maldives'), + 'ML' => t('Mali'), + 'MT' => t('Malta'), + 'MH' => t('Marshall Islands'), + 'MQ' => t('Martinique'), + 'MR' => t('Mauritania'), + 'MU' => t('Mauritius'), + 'YT' => t('Mayotte'), + 'MX' => t('Mexico'), + 'FM' => t('Micronesia'), + 'MD' => t('Moldova'), + 'MC' => t('Monaco'), + 'MN' => t('Mongolia'), + 'ME' => t('Montenegro'), + 'MS' => t('Montserrat'), + 'MA' => t('Morocco'), + 'MZ' => t('Mozambique'), + 'NA' => t('Namibia'), + 'NR' => t('Nauru'), + 'NP' => t('Nepal'), 'NL' => t('Netherlands'), + 'AN' => t('Netherlands Antilles'), + 'NC' => t('New Caledonia'), + 'NZ' => t('New Zealand'), + 'NI' => t('Nicaragua'), + 'NE' => t('Niger'), + 'NG' => t('Nigeria'), + 'NU' => t('Niue'), + 'NF' => t('Norfolk Island'), + 'NO' => t('Norway'), + 'OM' => t('Oman'), + 'PW' => t('Palau'), + 'PA' => t('Panama'), + 'PG' => t('Papua New Guinea'), + 'PY' => t('Paraguay'), + 'PE' => t('Peru'), + 'PH' => t('Philippines'), + 'PN' => t('Pitcairn Islands'), 'PL' => t('Poland'), 'PT' => t('Portugal'), + 'QA' => t('Qatar'), + 'CG' => t('Republic of the Congo'), + 'RE' => t('Reunion'), + 'RO' => t('Romania'), 'RU' => t('Russia'), + 'RW' => t('Rwanda'), + 'KN' => t('Saint Kitts and Nevis Anguilla'), + 'PM' => t('Saint Pierre and Miquelon'), + 'VC' => t('Saint Vincent and Grenadines'), + 'WS' => t('Samoa'), + 'SM' => t('San Marino'), + 'ST' => t('São Tomé and Príncipe'), + 'SA' => t('Saudi Arabia'), + 'SN' => t('Senegal'), + 'RS' => t('Serbia'), + 'SC' => t('Seychelles'), + 'SL' => t('Sierra Leone'), + 'SG' => t('Singapore'), + 'SK' => t('Slovakia'), + 'SI' => t('Slovenia'), + 'SB' => t('Solomon Islands'), + 'SO' => t('Somalia'), + 'ZA' => t('South Africa'), + 'KR' => t('South Korea'), 'ES' => t('Spain'), + 'LK' => t('Sri Lanka'), + 'SH' => t('St. Helena'), + 'LC' => t('St. Lucia'), + 'SR' => t('Suriname'), + 'SJ' => t('Svalbard and Jan Mayen Islands'), + 'SZ' => t('Swaziland'), + 'SE' => t('Sweden'), 'CH' => t('Switzerland'), + 'TW' => t('Taiwan'), + 'TJ' => t('Tajikistan'), + 'TZ' => t('Tanzania'), + 'TH' => t('Thailand'), + 'TG' => t('Togo'), + 'TO' => t('Tonga'), + 'TT' => t('Trinidad and Tobago'), + 'TN' => t('Tunisia'), + 'TR' => t('Turkey'), + 'TM' => t('Turkmenistan'), + 'TC' => t('Turks and Caicos Islands'), + 'TV' => t('Tuvalu'), + 'UG' => t('Uganda'), + 'UA' => t('Ukraine'), + 'AE' => t('United Arab Emirates'), 'GB' => t('United Kingdom'), 'US' => t('United States'), + 'UY' => t('Uruguay'), + 'VU' => t('Vanuatu'), + 'VA' => t('Vatican City State'), + 'VE' => t('Venezuela'), + 'VN' => t('Vietnam'), + 'VG' => t('Virgin Islands (British)'), + 'WF' => t('Wallis and Futuna Islands'), + 'YE' => t('Yemen'), + 'ZM' => t('Zambia'), + 'ZW' => t('Zimbabwe'), ), t('By language') => array( 'da_DK' => t('Danish (for Denmark only)'), + 'fr_CA' => t('French (for Canada only)'), 'he_IL' => t('Hebrew (for all)'), 'id_ID' => t('Indonesian (for Indonesia only)'), 'jp_JP' => t('Japanese (for Japan only)'),