diff --git a/CRM/Core/Payment/Stripe.php b/CRM/Core/Payment/Stripe.php
index 8d23a316b4d63f076e9680f5e12a8ddf727e879b..ff133c2d31992555a10a7427fa76e4b758f9956c 100644
--- a/CRM/Core/Payment/Stripe.php
+++ b/CRM/Core/Payment/Stripe.php
@@ -21,6 +21,13 @@ class CRM_Core_Payment_Stripe extends CRM_Core_Payment {
    */
   protected $_mode = NULL;
 
+  /**
+   * TRUE if we are dealing with a live transaction
+   *
+   * @var boolean
+   */
+  private $_islive = FALSE;
+
   /**
    * Constructor
    *
@@ -61,6 +68,22 @@ class CRM_Core_Payment_Stripe extends CRM_Core_Payment {
     }
   }
 
+  /**
+   * Get the currency for the transaction.
+   *
+   * Handle any inconsistency about how it is passed in here.
+   *
+   * @param $params
+   *
+   * @return string
+   */
+  public function getAmount($params) {
+    // Stripe amount required in cents.
+    $amount = number_format($params['amount'], 2, '.', '');
+    $amount = (int) preg_replace('/[^\d]/', '', strval($amount));
+    return $amount;
+  }
+
   /**
    * Helper log function.
    *
@@ -70,25 +93,47 @@ class CRM_Core_Payment_Stripe extends CRM_Core_Payment {
    *   The error!
    */
   public function logStripeException($op, $exception) {
-    $body = print_r($exception->getJsonBody(), TRUE);
-    Civi::log()->debug("Stripe_Error {$op}:  <pre> {$body} </pre>");
+    Civi::log()->debug("Stripe_Error {$op}: " . print_r($exception->getJsonBody(), TRUE));
   }
 
   /**
    * Check if return from stripeCatchErrors was an error object
    * that should be passed back to original api caller.
    *
-   * @param  $stripeReturn
+   * @param array $err
    *   The return from a call to stripeCatchErrors
    *
    * @return bool
-   *
    */
-  public function isErrorReturn($stripeReturn) {
-    if (is_object($stripeReturn) && get_class($stripeReturn) == 'CRM_Core_Error') {
+  public function isErrorReturn($err) {
+    if (!empty($err['is_error'])) {
       return TRUE;
     }
-    return FALSE;
+    else {
+      return FALSE;
+    }
+  }
+
+  /**
+   * Handle an error from Stripe API and notify the user
+   *
+   * @param array $err
+   * @param string $bounceURL
+   *
+   * @return string errorMessage (or statusbounce if URL is specified)
+   */
+  public function handleErrorNotification($err, $bounceURL = NULL) {
+    $errorMessage = 'Payment Response: <br />' .
+      'Type: ' . $err['type'] . '<br />' .
+      'Code: ' . $err['code'] . '<br />' .
+      'Message: ' . $err['message'] . '<br />';
+
+    Civi::log()->debug('Stripe Payment Error: ' . $errorMessage);
+
+    if ($bounceURL) {
+      CRM_Core_Error::statusBounce($errorMessage, $bounceURL, 'Payment Error');
+    }
+    return $errorMessage;
   }
 
   /**
@@ -107,7 +152,6 @@ class CRM_Core_Payment_Stripe extends CRM_Core_Payment {
    * @throws \CiviCRM_API3_Exception
    */
   public function stripeCatchErrors($op = 'create_customer', $stripe_params, $params, $ignores = array()) {
-    $error_url = $params['stripe_error_url'];
     $return = FALSE;
     // Check for errors before trying to submit.
     try {
@@ -159,23 +203,17 @@ class CRM_Core_Payment_Stripe extends CRM_Core_Payment {
       }
 
       $this->logStripeException($op, $e);
-      $error_message = '';
       // Since it's a decline, Stripe_CardError will be caught
       $body = $e->getJsonBody();
       $err = $body['error'];
       if (!isset($err['code'])) {
-        $err['code'] = null;
+        // A "fake" error code
+        $err['code'] = 9000;
       }
-      //$error_message .= 'Status is: ' . $e->getHttpStatus() . "<br />";
-      ////$error_message .= 'Param is: ' . $err['param'] . "<br />";
-      $error_message .= 'Type: ' . $err['type'] . '<br />';
-      $error_message .= 'Code: ' . $err['code'] . '<br />';
-      $error_message .= 'Message: ' . $err['message'] . '<br />';
 
       if (is_a($e, 'Stripe_CardError')) {
-        $newnote = civicrm_api3('Note', 'create', array(
-          'sequential' => 1,
-          'entity_id' => $params['contactID'],
+        civicrm_api3('Note', 'create', array(
+          'entity_id' => self::getContactId($params),
           'contact_id' => $params['contributionID'],
           'subject' => $err['type'],
           'note' => $err['code'],
@@ -183,23 +221,9 @@ class CRM_Core_Payment_Stripe extends CRM_Core_Payment {
         ));
       }
 
-      if (isset($error_url)) {
-      // Redirect to first page of form and present error.
-      CRM_Core_Error::statusBounce("Oops!  Looks like there was an error.  Payment Response:
-        <br /> {$error_message}", $error_url);
-      }
-      else {
-        // Don't have return url - return error object to api
-        $core_err = CRM_Core_Error::singleton();
-        $message = 'Oops!  Looks like there was an error.  Payment Response: <br />' . $error_message;
-        if ($err['code']) {
-          $core_err->push($err['code'], 0, NULL, $message);
-        }
-        else {
-          $core_err->push(9000, 0, NULL, 'Unknown Error: ' . $message);
-        }
-        return $core_err;
-      }
+      // Flag to detect error return
+      $err['is_error'] = TRUE;
+      return $err;
     }
 
     return $return;
@@ -399,13 +423,13 @@ class CRM_Core_Payment_Stripe extends CRM_Core_Payment {
     if (!(array_key_exists('qfKey', $params))) {
       // Probably not called from a civicrm form (e.g. webform) -
       // will return error object to original api caller.
-      $params['stripe_error_url'] = $error_url = null;
+      $params['stripe_error_url'] = NULL;
     }
     else {
       $qfKey = $params['qfKey'];
       $parsed_url = parse_url($params['entryURL']);
       $url_path = substr($parsed_url['path'], 1);
-      $params['stripe_error_url'] = $error_url = CRM_Utils_System::url($url_path,
+      $params['stripe_error_url'] = CRM_Utils_System::url($url_path,
       $parsed_url['query'] . "&_qf_Main_display=1&qfKey={$qfKey}", FALSE, NULL, FALSE);
     }
 
@@ -413,9 +437,7 @@ class CRM_Core_Payment_Stripe extends CRM_Core_Payment {
     \Stripe\Stripe::setAppInfo('CiviCRM', CRM_Utils_System::version(), CRM_Utils_System::baseURL());
     \Stripe\Stripe::setApiKey($this->_paymentProcessor['user_name']);
 
-    // Stripe amount required in cents.
-    $amount = number_format($params['amount'], 2, '.', '');
-    $amount = (int) preg_replace('/[^\d]/', '', strval($amount));
+    $amount = self::getAmount($params);
 
     // Use Stripe.js instead of raw card details.
     if (!empty($params['stripe_token'])) {
@@ -429,186 +451,64 @@ class CRM_Core_Payment_Stripe extends CRM_Core_Payment {
       Civi::log()->debug('Stripe.js token was not passed!  Report this message to the site administrator. $params: ' . print_r($params, TRUE));
     }
 
-    // Check for existing customer, create new otherwise.
-    // Possible email fields.
-    $email_fields = array(
-      'email',
-      'email-5',
-      'email-Primary',
-    );
-
-    // Possible contact ID fields.
-    $contact_id_fields = array(
-      'contact_id',
-      'contactID',
-    );
-
-    // Find out which email field has our yummy value.
-    foreach ($email_fields as $email_field) {
-      if (!empty($params[$email_field])) {
-        $email = $params[$email_field];
-        break;
-      }
-    }
-
-    // We didn't find an email, but never fear - this might be a backend contrib.
-    // We can look for a contact ID field and get the email address.
-    if (empty($email)) {
-      foreach ($contact_id_fields as $cid_field) {
-        if (!empty($params[$cid_field])) {
-          $email = civicrm_api3('Contact', 'getvalue', array(
-            'id' => $params[$cid_field],
-            'return' => 'email',
-          ));
-          break;
-        }
-      }
-    }
+    $contactId = self::getContactId($params);
+    $email = self::getBillingEmail($params, $contactId);
 
-    // We still didn't get an email address?!  /ragemode on
-    if (empty($email)) {
-      CRM_Core_Error::fatal(ts('No email address found.  Please report this issue.'));
-    }
-
-    // Prepare escaped query params.
-    $query_params = array(
-      1 => array($email, 'String'),
-      2 => array($this->_paymentProcessor['id'], 'Integer'),
-    );
-
-    $customer_query = CRM_Core_DAO::singleValueQuery("SELECT id
-      FROM civicrm_stripe_customers
-      WHERE email = %1 AND is_live = '{$this->_islive}' AND processor_id = %2", $query_params);
-
-    /****
-     * If for some reason you cannot use Stripe.js and you are aware of PCI Compliance issues,
-     * here is the alternative to Stripe.js:
-     ****/
-
-    /*
-      // Get Cardholder's full name.
-      $cc_name = $params['first_name'] . " ";
-      if (strlen($params['middle_name']) > 0) {
-        $cc_name .= $params['middle_name'] . " ";
-      }
-      $cc_name .= $params['last_name'];
-
-      // Prepare Card details in advance to use for new Stripe Customer object if we need.
-      $card_details = array(
-        'number' => $params['credit_card_number'],
-        'exp_month' => $params['month'],
-        'exp_year' => $params['year'],
-        'cvc' => $params['cvv2'],
-        'name' => $cc_name,
-        'address_line1' => $params['street_address'],
-        'address_state' => $params['state_province'],
-        'address_zip' => $params['postal_code'],
-      );
-    */
-
-    // drastik - Uncomment this for Drupal debugging to dblog.
-    /*
-     $zz = print_r(get_defined_vars(), TRUE);
-     $debug_code = '<pre>' . $zz . '</pre>';
-     watchdog('Stripe', $debug_code);
-    */
-
-    // Customer not in civicrm_stripe database.  Create a new Customer in Stripe.
-    if (!isset($customer_query)) {
-      $sc_create_params = array(
-        'description' => 'Donor from CiviCRM',
-        'card' => $card_token,
-        'email' => $email,
-      );
+    // See if we already have a stripe customer
+    $customerParams = [
+      'contact_id' => $contactId,
+      'card_token' => $card_token,
+      'is_live' => $this->_islive,
+      'processor_id' => $this->_paymentProcessor['id'],
+      'email' => $email,
+    ];
 
-      $stripe_customer = $this->stripeCatchErrors('create_customer', $sc_create_params, $params);
+    $stripeCustomerId = CRM_Stripe_Customer::find($customerParams);
 
-      // Store the relationship between CiviCRM's email address for the Contact & Stripe's Customer ID.
-      if (isset($stripe_customer)) {
-        if ($this->isErrorReturn($stripe_customer)) {
-          return $stripe_customer;
-        }
-        // Prepare escaped query params.
-        $query_params = array(
-          1 => array($email, 'String'),
-          2 => array($stripe_customer->id, 'String'),
-          3 => array($this->_paymentProcessor['id'], 'Integer'),
-        );
-
-        CRM_Core_DAO::executeQuery("INSERT INTO civicrm_stripe_customers
-          (email, id, is_live, processor_id) VALUES (%1, %2, '{$this->_islive}', %3)", $query_params);
-      }
-      else {
-        CRM_Core_Error::fatal(ts('There was an error saving new customer within Stripe.  Is Stripe down?'));
-      }
+    // Customer not in civicrm database.  Create a new Customer in Stripe.
+    if (!isset($stripeCustomerId)) {
+      $stripeCustomer = CRM_Stripe_Customer::create($customerParams, $this);
     }
     else {
-      // Customer was found in civicrm_stripe database, fetch from Stripe.
-      $stripe_customer = $this->stripeCatchErrors('retrieve_customer', $customer_query, $params);
-      if (!empty($stripe_customer)) {
-        if ($this->isErrorReturn($stripe_customer)) {
-          return $stripe_customer;
-        }
-        // Avoid the 'use same token twice' issue while still using latest card.
-        if (!empty($params['is_secondary_financial_transaction'])) {
-          // This is a Contribution page with "Separate Membership Payment".
-          // Charge is coming through for the 2nd time.
-          // Don't update customer again or we will get "token_already_used" error from Stripe.
+      // Customer was found in civicrm database, fetch from Stripe.
+      $stripeCustomer = $this->stripeCatchErrors('retrieve_customer', $stripeCustomerId, $params);
+      if (!empty($stripeCustomer)) {
+        if ($this->isErrorReturn($stripeCustomer)) {
+          if (($stripeCustomer['type'] == 'invalid_request_error') && ($stripeCustomer['code'] == 'resource_missing')) {
+            // Customer doesn't exist, create a new one
+            CRM_Stripe_Customer::delete($customerParams);
+            $stripeCustomer = CRM_Stripe_Customer::create($customerParams, $this);
+          }
+          if ($this->isErrorReturn($stripeCustomer)) {
+            // We still failed to create a customer
+            self::handleErrorNotification($stripeCustomer, $params['stripe_error_url']);
+            return $stripeCustomer;
+          }
         }
-        else {
-          $stripe_customer->card = $card_token;
-          $response = $this->stripeCatchErrors('save', $stripe_customer, $params);
-            if (isset($response) && $this->isErrorReturn($response)) {
-              return $response;
-            }
+
+        $stripeCustomer->card = $card_token;
+        $updatedStripeCustomer = $this->stripeCatchErrors('save', $stripeCustomer, $params);
+        if ($this->isErrorReturn($updatedStripeCustomer)) {
+          if (($updatedStripeCustomer['type'] == 'invalid_request_error') && ($updatedStripeCustomer['code'] == 'token_already_used')) {
+            // This error is ok, we've already used the token during create_customer
+          }
+          else {
+            self::handleErrorNotification($updatedStripeCustomer, $params['stripe_error_url']);
+            return $updatedStripeCustomer;
+          }
         }
       }
       else {
-        // Customer was found in civicrm_stripe database, but unable to be
-        // retrieved from Stripe.  Was he deleted?
-        $sc_create_params = array(
-          'description' => 'Donor from CiviCRM',
-          'card' => $card_token,
-          'email' => $email,
-        );
-
-        $stripe_customer = $this->stripeCatchErrors('create_customer', $sc_create_params, $params);
-
-        // Somehow a customer ID saved in the system no longer pairs
-        // with a Customer within Stripe.  (Perhaps deleted using Stripe interface?).
-        // Store the relationship between CiviCRM's email address for the Contact & Stripe's Customer ID.
-        if (isset($stripe_customer)) {
-          /*if ($this->isErrorReturn($stripe_customer)) {
-            return $stripe_customer;
-          }*/
-          // Delete whatever we have for this customer.
-          $query_params = array(
-            1 => array($email, 'String'),
-            2 => array($this->_paymentProcessor['id'], 'Integer'),
-          );
-          CRM_Core_DAO::executeQuery("DELETE FROM civicrm_stripe_customers
-            WHERE email = %1 AND is_live = '{$this->_islive}' AND processor_id = %2", $query_params);
-
-          // Create new record for this customer.
-          $query_params = array(
-            1 => array($email, 'String'),
-            2 => array($stripe_customer->id, 'String'),
-            3 => array($this->_paymentProcessor['id'], 'Integer'),
-          );
-          CRM_Core_DAO::executeQuery("INSERT INTO civicrm_stripe_customers (email, id, is_live, processor_id)
-            VALUES (%1, %2, '{$this->_islive}, %3')", $query_params);
-        }
-        else {
-          // Customer was found in civicrm_stripe database, but unable to be
-          // retrieved from Stripe, and unable to be created in Stripe.  What luck :(
-          CRM_Core_Error::fatal(ts('There was an error saving new customer within Stripe.  Is Stripe down?'));
-        }
+        // Customer was found in civicrm_stripe database, but not in Stripe.
+        // Delete existing customer record from CiviCRM and create a new customer
+        CRM_Stripe_Customer::delete($customerParams);
+        $stripeCustomer = CRM_Stripe_Customer::create($customerParams, $this);
       }
     }
 
     // Prepare the charge array, minus Customer/Card details.
     if (empty($params['description'])) {
-      $params['description'] = ts('Backend contribution');
+      $params['description'] = ts('Backend Stripe contribution');
     }
 
     // Stripe charge.
@@ -619,8 +519,8 @@ class CRM_Core_Payment_Stripe extends CRM_Core_Payment {
     );
 
     // Use Stripe Customer if we have a valid one.  Otherwise just use the card.
-    if (!empty($stripe_customer->id)) {
-      $stripe_charge['customer'] = $stripe_customer->id;
+    if (!empty($stripeCustomer->id)) {
+      $stripe_charge['customer'] = $stripeCustomer->id;
     }
     else {
       $stripe_charge['card'] = $card_token;
@@ -628,37 +528,39 @@ class CRM_Core_Payment_Stripe extends CRM_Core_Payment {
 
     // Handle recurring payments in doRecurPayment().
     if (CRM_Utils_Array::value('is_recur', $params) && $params['contributionRecurID']) {
-      return $this->doRecurPayment($params, $amount, $stripe_customer);
+      return $this->doRecurPayment($params, $amount, $stripeCustomer);
     }
 
     // Fire away!  Check for errors before trying to submit.
-    $stripe_response = $this->stripeCatchErrors('charge', $stripe_charge, $params);
-    if (!empty($stripe_response)) {
-      if ($this->isErrorReturn($stripe_response)) {
-        return $stripe_response;
+    $stripeCharge = $this->stripeCatchErrors('charge', $stripe_charge, $params);
+    if (!empty($stripeCharge)) {
+      if ($this->isErrorReturn($stripeCharge)) {
+        self::handleErrorNotification($stripeCharge, $params['stripe_error_url']);
+        return $stripeCharge;
       }
       // Success!  Return some values for CiviCRM.
-      $params['trxn_id'] = $stripe_response->id;
+      $params['trxn_id'] = $stripeCharge->id;
       // Return fees & net amount for Civi reporting.
       // Uses new Balance Trasaction object.
-      $balance_transaction = $this->stripeCatchErrors('retrieve_balance_transaction', $stripe_response->balance_transaction, $params);
-      if (!empty($balance_transaction)) {
-        if ($this->isErrorReturn($balance_transaction)) {
-          return $balance_transaction;
+      $balanceTransaction = $this->stripeCatchErrors('retrieve_balance_transaction', $stripeCharge->balance_transaction, $params);
+      if (!empty($balanceTransaction)) {
+        if ($this->isErrorReturn($balanceTransaction)) {
+          self::handleErrorNotification($balanceTransaction, $params['stripe_error_url']);
+          return $balanceTransaction;
         }
-        $params['fee_amount'] = $balance_transaction->fee / 100;
-        $params['net_amount'] = $balance_transaction->net / 100;
+        $params['fee_amount'] = $balanceTransaction->fee / 100;
+        $params['net_amount'] = $balanceTransaction->net / 100;
       }
     }
     else {
       // There was no response from Stripe on the create charge command.
-      if (isset($error_url)) {
-        CRM_Core_Error::statusBounce('Stripe transaction response not recieved!  Check the Logs section of your stripe.com account.', $error_url);
+      if (isset($params['stripe_error_url'])) {
+        CRM_Core_Error::statusBounce('Stripe transaction response not received!  Check the Logs section of your stripe.com account.', $params['stripe_error_url']);
       }
       else {
         // Don't have return url - return error object to api
         $core_err = CRM_Core_Error::singleton();
-        $core_err->push(9000, 0, NULL, 'Stripe transaction response not recieved!  Check the Logs section of your stripe.com account.');
+        $core_err->push(9000, 0, NULL, 'Stripe transaction response not received!  Check the Logs section of your stripe.com account.');
         return $core_err;
       }
     }
@@ -674,7 +576,7 @@ class CRM_Core_Payment_Stripe extends CRM_Core_Payment {
    *   Assoc array of input parameters for this transaction.
    * @param int $amount
    *   Transaction amount in USD cents.
-   * @param object $stripe_customer
+   * @param object $stripeCustomer
    *   Stripe customer object generated by Stripe API.
    *
    * @return array
@@ -682,7 +584,7 @@ class CRM_Core_Payment_Stripe extends CRM_Core_Payment {
    *
    * @throws \CiviCRM_API3_Exception
    */
-  public function doRecurPayment(&$params, $amount, $stripe_customer) {
+  public function doRecurPayment(&$params, $amount, $stripeCustomer) {
     // Get recurring contrib properties.
     $frequency = $params['frequency_unit'];
     $frequency_interval = (empty($params['frequency_interval']) ? 1 : $params['frequency_interval']);
@@ -744,8 +646,8 @@ class CRM_Core_Payment_Stripe extends CRM_Core_Payment {
         }
      }
         // Apply the disount through a negative balance.
-       $stripe_customer->account_balance = -$discount_in_cents;
-       $stripe_customer->save();
+       $stripeCustomer->account_balance = -$discount_in_cents;
+       $stripeCustomer->save();
      }
 
     // Tying a plan to a membership (or priceset->membership) makes it possible
@@ -776,12 +678,6 @@ class CRM_Core_Payment_Stripe extends CRM_Core_Payment {
     }
     $plan_id = "{$membership_type_tag}every-{$frequency_interval}-{$frequency}-{$amount}-{$currency}{$mode_tag}";
 
-    // Prepare escaped query params.
-    $query_params = array(
-      1 => array($plan_id, 'String'),
-    );
-
-
     // Prepare escaped query params.
     $query_params = array(
       1 => array($plan_id, 'String'),
@@ -841,14 +737,14 @@ class CRM_Core_Payment_Stripe extends CRM_Core_Payment {
       'prorate' => FALSE,
       'plan' => $plan_id,
     );
-    $stripe_response = $stripe_customer->subscriptions->create($cust_sub_params);
-    $subscription_id = $stripe_response->id;
+    $stripeSubscription = $stripeCustomer->subscriptions->create($cust_sub_params);
+    $subscription_id = $stripeSubscription->id;
     $recuring_contribution_id = $params['contributionRecurID'];
 
     // Prepare escaped query params.
     $query_params = array(
       1 => array($subscription_id, 'String'),
-      2 => array($stripe_customer->id, 'String'),
+      2 => array($stripeCustomer->id, 'String'),
       3 => array($recuring_contribution_id, 'String'),
       4 => array($this->_paymentProcessor['id'], 'Integer'),
     );
@@ -887,8 +783,7 @@ class CRM_Core_Payment_Stripe extends CRM_Core_Payment {
    * @param array $params
    *   Name value pair of contribution data.
    *
-   * @return void
-   *
+   * @throws \CiviCRM_API3_Exception
    */
   public function doTransferCheckout(&$params, $component) {
     self::doDirectPayment($params);
@@ -924,5 +819,57 @@ class CRM_Core_Payment_Stripe extends CRM_Core_Payment {
     $ipnClass->main();
   }
 
+
+  /*******************************************************************
+   * THE FOLLOWING FUNCTIONS SHOULD BE REMOVED ONCE THEY ARE IN CORE
+   * getBillingEmail
+   * getContactId
+   ******************************************************************/
+
+  /**
+   * Get the billing email address
+   *
+   * @param array $params
+   * @param int $contactId
+   *
+   * @return string|NULL
+   */
+  protected static function getBillingEmail($params, $contactId) {
+    $billingLocationId = CRM_Core_BAO_LocationType::getBilling();
+
+    $emailAddress = CRM_Utils_Array::value("email-{$billingLocationId}", $params,
+      CRM_Utils_Array::value('email-Primary', $params,
+        CRM_Utils_Array::value('email', $params, NULL)));
+
+    if (empty($emailAddress) && !empty($contactId)) {
+      // Try and retrieve an email address from Contact ID
+      try {
+        $emailAddress = civicrm_api3('Email', 'getvalue', array(
+          'contact_id' => $contactId,
+          'return' => ['email'],
+        ));
+      }
+      catch (CiviCRM_API3_Exception $e) {
+        return NULL;
+      }
+    }
+    return $emailAddress;
+  }
+
+  /**
+   * Get the contact id
+   *
+   * @param array $params
+   *
+   * @return int ContactID
+   */
+  protected static function getContactId($params) {
+    return CRM_Utils_Array::value('contactID', $params,
+      CRM_Utils_Array::value('contact_id', $params,
+        CRM_Utils_Array::value('cms_contactID', $params,
+          CRM_Utils_Array::value('cid', $params, NULL
+          ))));
+  }
+
 }
 
diff --git a/CRM/Core/Payment/StripeIPN.php b/CRM/Core/Payment/StripeIPN.php
index 4fee96fe24e84ec1d1f783950e2a934fd32efac1..c974da52edd9df70b6c99435cfd6b553aee132e1 100644
--- a/CRM/Core/Payment/StripeIPN.php
+++ b/CRM/Core/Payment/StripeIPN.php
@@ -4,53 +4,62 @@
  * Handle Stripe Webhooks for recurring payments.
  */
 
-require_once 'CRM/Core/Page.php';
-
 class CRM_Core_Payment_StripeIPN extends CRM_Core_Payment_BaseIPN {
-  var $ppid = NULL;
-  var $secret_key = NULL;
-  var $is_email_receipt = 1;
+
+  // TODO: These vars should probably be protected, not public - but need to check them all first
+  public $ppid = NULL;
+  public $secret_key = NULL;
+  public $is_email_receipt = 1;
+
   // By default, always retrieve the event from stripe to ensure we are
   // not being fed garbage. However, allow an override so when we are 
   // testing, we can properly test a failed recurring contribution.
-  var $verify_event = TRUE;
+  public $verify_event = TRUE;
 
   // Properties of the event.
-  var $test_mode;
-  var $event_type = NULL;
-  var $subscription_id = NULL;
-  var $charge_id = NULL;
-  var $previous_plan_id = NULL;
-  var $plan_id = NULL;
-  var $plan_amount = NULL;
-  var $frequency_interval = NULL;
-  var $frequency_unit = NULL;
-  var $plan_name = NULL;
-  var $plan_start = NULL;
+  public $test_mode;
+  public $event_type = NULL;
+  public $subscription_id = NULL;
+  public $charge_id = NULL;
+  public $previous_plan_id = NULL;
+  public $plan_id = NULL;
+  public $plan_amount = NULL;
+  public $frequency_interval = NULL;
+  public $frequency_unit = NULL;
+  public $plan_name = NULL;
+  public $plan_start = NULL;
        
   // Derived properties.
-  var $contact_id = NULL;
-  var $contribution_recur_id = NULL;
-  var $membership_id = NULL;
-  var $event_id = NULL;
-  var $invoice_id = NULL;
-  var $receive_date = NULL;
-  var $amount = NULL;
-  var $fee = NULL;
-  var $net_amount = NULL;
-  var $previous_contribution_id = NULL;
-  var $previous_contribution_status_id = NULL;
-  var $previous_contribution_total_amount = NULL;
-  var $previous_completed_contribution_id = NULL;
+  public $contact_id = NULL;
+  public $contribution_recur_id = NULL;
+  public $membership_id = NULL;
+  public $event_id = NULL;
+  public $invoice_id = NULL;
+  public $receive_date = NULL;
+  public $amount = NULL;
+  public $fee = NULL;
+  public $net_amount = NULL;
+  public $previous_contribution_id = NULL;
+  public $previous_contribution_status_id = NULL;
+  public $previous_contribution_total_amount = NULL;
+  public $previous_completed_contribution_id = NULL;
 
+  /**
+   * CRM_Core_Payment_StripeIPN constructor.
+   *
+   * @param $inputData
+   * @param bool $verify
+   *
+   * @throws \CRM_Core_Exception
+   */
   public function __construct($inputData, $verify = TRUE) {
     $this->verify_event = $verify;
     $this->setInputParameters($inputData);
     parent::__construct();
   }
+
   /**
    * Store input array on the class.
-   *
    * We override base because our input parameter is an object
    *
    * @param array $parameters
@@ -58,8 +67,8 @@ class CRM_Core_Payment_StripeIPN extends CRM_Core_Payment_BaseIPN {
    * @throws CRM_Core_Exception
   */
   public function setInputParameters($parameters) {
-    if (empty($parameters) || !is_object($parameters) || empty($parameters->id)) {
-      throw new CRM_Core_Exception('Invalid input parameters. Should be an object with an id parameter.');
+    if (!is_object($parameters)) {
+      $this->exception('Invalid input parameters');
     }
 
     // Determine the proper Stripe Processor ID so we can get the secret key
@@ -67,7 +76,7 @@ class CRM_Core_Payment_StripeIPN extends CRM_Core_Payment_BaseIPN {
     
     // The $_GET['processor_id'] value is set by CRM_Core_Payment::handlePaymentMethod.
     if (!array_key_exists('processor_id', $_GET) || empty($_GET['processor_id'])) {
-      throw new CRM_Core_Exception('Cannot determine processor id.');
+      $this->exception('Cannot determine processor id');
     }
     $this->ppid = $_GET['processor_id'];
 
@@ -77,7 +86,7 @@ class CRM_Core_Payment_StripeIPN extends CRM_Core_Payment_BaseIPN {
       $this->secret_key = civicrm_api3('PaymentProcessor', 'getvalue', $params);
     }
     catch(Exception $e) {
-      throw new CRM_Core_Exception('Failed to get Stripe secret key.');
+      $this->exception('Failed to get Stripe secret key');
     }
 
     // Now re-retrieve the data from Stripe to ensure it's legit.
@@ -91,17 +100,18 @@ class CRM_Core_Payment_StripeIPN extends CRM_Core_Payment_BaseIPN {
       $this->_inputParameters = $parameters;
     }
   }
+
   /**
-    * 
-    * Get a parameter given to us by Stripe.
-    * @param string $name
-    * @param $type
-    * @param bool $abort
-    *
-    * @return mixed
+   * Get a parameter given to us by Stripe.
+   *
+   * @param string $name
+   * @param $type
+   * @param bool $abort
+   *
+   * @return false|int|null|string
+   * @throws \CRM_Core_Exception
    */
   public function retrieve($name, $type, $abort = TRUE) {
-
     $class_name = get_class($this->_inputParameters->data->object);
     $value = NULL;
     switch ($name) {
@@ -175,27 +185,35 @@ class CRM_Core_Payment_StripeIPN extends CRM_Core_Payment_BaseIPN {
     }
     $value = CRM_Utils_Type::validate($value, $type, FALSE);
     if ($abort && $value === NULL) {
-      throw new CRM_Core_Exception("Could not find an entry for '$name'.");
+      echo "Failure: Missing Parameter<p>" . CRM_Utils_Type::escape($name, 'String');
+      $this->exception("Could not find an entry for $name");
     }
     return $value;
   }
 
-  function main() {
+  /**
+   * @return bool
+   * @throws \CRM_Core_Exception
+   * @throws \CiviCRM_API3_Exception
+   */
+  public function main() {
     // Collect and determine all data about this event.
-    $this->setInfo();
+    $this->event_type = $this->retrieve('event_type', 'String');
+
+    $pendingStatusId = CRM_Core_PseudoConstant::getKey('CRM_Contribute_BAO_Contribution', 'contribution_status_id', 'Pending');
 
     switch($this->event_type) {
       // Successful recurring payment.
       case 'invoice.payment_succeeded':
+        $this->setInfo();
         // Lets do a check to make sure this payment has the amount same as that of first contribution.
         // If it's not a match, something is wrong (since when we update a plan, we generate a whole
         // new recurring contribution).
         if ($this->previous_contribution_total_amount != $this->amount) {
-          throw new CRM_Core_Exception("Subscription amount mismatch. I have " . $this->amount . " and I expect " . $this->previous_contribution_total_amount . ".");
-          return FALSE;
+          $this->exception("Subscription amount mismatch. I have " . $this->amount . " and I expect " . $this->previous_contribution_total_amount);
         }
 
-        if ($this->previous_contribution_status_id == 2) {
+        if ($this->previous_contribution_status_id == $pendingStatusId) {
           // Update the contribution to include the fee.
           civicrm_api3('Contribution', 'create', array(
             'id' => $this->previous_contribution_id,
@@ -204,8 +222,7 @@ class CRM_Core_Payment_StripeIPN extends CRM_Core_Payment_BaseIPN {
             'net_amount' => $this->net_amount,
           ));
           // The last one was not completed, so complete it.
-          $result = civicrm_api3('Contribution', 'completetransaction', array(
-            'sequential' => 1,
+          civicrm_api3('Contribution', 'completetransaction', array(
             'id' => $this->previous_contribution_id,
             'trxn_date' => $this->receive_date,
             'trxn_id' => $this->charge_id,
@@ -215,13 +232,13 @@ class CRM_Core_Payment_StripeIPN extends CRM_Core_Payment_BaseIPN {
             'payment_processor_id' => $this->ppid,
             'is_email_receipt' => $this->is_email_receipt,
            ));
-        } else {
+        }
+        else {
           // The first contribution was completed, so create a new one.
           
           // api contribution repeattransaction repeats the appropriate contribution if it is given
-          // simply the recurring contribution id. It also updates the membership for us. 
-
-          $result = civicrm_api3('Contribution', 'repeattransaction', array(
+          // simply the recurring contribution id. It also updates the membership for us.
+          civicrm_api3('Contribution', 'repeattransaction', array(
             // Actually, don't use contribution_recur_id until CRM-19945 patches make it in to 4.6/4.7
             // and we have a way to require a minimum minor CiviCRM version.
             //'contribution_recur_id' => $this->recurring_info->id,
@@ -247,22 +264,22 @@ class CRM_Core_Payment_StripeIPN extends CRM_Core_Payment_BaseIPN {
         }
 
         // Successful charge & more to come. 
-        $result = civicrm_api3('ContributionRecur', 'create', array(
-          'sequential' => 1,
+        civicrm_api3('ContributionRecur', 'create', array(
           'id' => $this->contribution_recur_id,
           'failure_count' => 0,
           'contribution_status_id' => "In Progress"
         ));
 
-        return TRUE;
+        return;
 
       // Failed recurring payment.
       case 'invoice.payment_failed':
+        $this->setInfo();
         $fail_date = date("Y-m-d H:i:s");
 
-        if ($this->previous_contribution_status_id == 2) {
+        if ($this->previous_contribution_status_id == $pendingStatusId) {
           // If this contribution is Pending, set it to Failed.
-          $result = civicrm_api3('Contribution', 'create', array(
+          civicrm_api3('Contribution', 'create', array(
             'id' => $this->previous_contribution_id,
             'contribution_status_id' => "Failed",
             'receive_date' => $fail_date,
@@ -289,64 +306,64 @@ class CRM_Core_Payment_StripeIPN extends CRM_Core_Payment_BaseIPN {
         ));
         $failure_count++;
 
-        //  Change the status of the Recurring and update failed attempts.
-        $result = civicrm_api3('ContributionRecur', 'create', array(
-          'sequential' => 1,
+        // Change the status of the Recurring and update failed attempts.
+        civicrm_api3('ContributionRecur', 'create', array(
           'id' => $this->contribution_recur_id,
           'contribution_status_id' => "Failed",
           'failure_count' => $failure_count,
           'modified_date' => $fail_date,
         ));
 
-        return TRUE;
+        return;
 
-      //Subscription is cancelled
+      // Subscription is cancelled
       case 'customer.subscription.deleted':
-        //Cancel the recurring contribution
-        $result = civicrm_api3('ContributionRecur', 'cancel', array(
+        $this->setInfo();
+        // Cancel the recurring contribution
+        civicrm_api3('ContributionRecur', 'cancel', array(
           'id' => $this->contribution_recur_id,
         ));
 
-        //Delete the record from Stripe's subscriptions table
+        // Delete the record from Stripe's subscriptions table
         $query_params = array(
           1 => array($this->subscription_id, 'String'),
         );
         CRM_Core_DAO::executeQuery("DELETE FROM civicrm_stripe_subscriptions
           WHERE subscription_id = %1", $query_params);
 
-        return TRUE;
+        return;
 
       // One-time donation and per invoice payment.
       case 'charge.succeeded':
+        //$this->setInfo();
+        // TODO: Implement this so we can mark payments as failed?
         // Not implemented.
-        return TRUE;
+        return;
 
      // Subscription is updated. Delete existing recurring contribution and start a fresh one.
      // This tells a story to site admins over editing a recurring contribution record.
      case 'customer.subscription.updated':
+       $this->setInfo();
        if (empty($this->previous_plan_id)) {
          // Not a plan change...don't care.
-         return TRUE;
+         return;
        }
        
        $new_civi_invoice = md5(uniqid(rand(), TRUE));
 
-       if ($this->previous_contribution_status_id == 2) {
+       if ($this->previous_contribution_status_id == $pendingStatusId) {
          // Cancel the pending contribution.
-         $result = civicrm_api3('Contribution', 'delete', array(
-           'sequential' => 1,
+         civicrm_api3('Contribution', 'delete', array(
            'id' => $this->previous_contribution_id,
          ));
        }
 
        // Cancel the old recurring contribution.
-       $result = civicrm_api3('ContributionRecur', 'cancel', array(
-         'sequential' => 1,
+       civicrm_api3('ContributionRecur', 'cancel', array(
          'id' => $this->contribution_recur_id
        ));
 
        $new_contribution_recur = civicrm_api3('ContributionRecur', 'create', array(
-          'sequential' => 1,
           'contact_id' => $this->contact_id,
           'invoice_id' => $new_civi_invoice,
           'amount' => $this->plan_amount,
@@ -387,6 +404,7 @@ class CRM_Core_Payment_StripeIPN extends CRM_Core_Payment_BaseIPN {
           $query_params
         );
 
+        // FIXME: MJW Do we need this custom handling for memberships here? Core should do all we need
         if ($this->membership_id) { 
           $plan_elements = explode("-", $this->plan_id);
           $plan_name_elements = explode("-", $this->plan_name);
@@ -399,8 +417,7 @@ class CRM_Core_Payment_StripeIPN extends CRM_Core_Payment_BaseIPN {
 
           // Adjust to the new membership level.
           if (!empty($new_membership_type_id)) {
-            $result = civicrm_api3('Membership', 'create', array(
-              'sequential' => 1,
+            civicrm_api3('Membership', 'create', array(
               'id' => $this->membership_id,
               'membership_type_id' => $new_membership_type_id,
               'contribution_recur_id' => $new_contribution_recur_id,
@@ -408,17 +425,17 @@ class CRM_Core_Payment_StripeIPN extends CRM_Core_Payment_BaseIPN {
             ));
 
             // Create a new membership payment record.
-            $result = civicrm_api3('MembershipPayment', 'create', array(
-              'sequential' => 1,
+            civicrm_api3('MembershipPayment', 'create', array(
               'membership_id' => $this->membership_id,
               'contribution_id' => $new_contribution_id,
             ));
           }
         }
-        return TRUE;
+        return;
 
       // Keep plans table in sync with Stripe when a plan is deleted.
      case 'plan.deleted':
+       $this->setInfo();
        $is_live = $this->test_mode == 1 ? 0 : 1;
        $query_params = array(
          1 => array($this->plan_id, 'String'),
@@ -428,10 +445,10 @@ class CRM_Core_Payment_StripeIPN extends CRM_Core_Payment_BaseIPN {
        CRM_Core_DAO::executeQuery("DELETE FROM civicrm_stripe_plans WHERE
          plan_id = %1 AND  processor_id = %2 and is_live = %3", $query_params);
 
-       return TRUE;
+       return;
     }
     // Unhandled event type.
-    return TRUE;
+    return;
   }
 
   /**
@@ -440,10 +457,11 @@ class CRM_Core_Payment_StripeIPN extends CRM_Core_Payment_BaseIPN {
    * Given the data passed to us via the Stripe Event, try to determine
    * as much as we can about this event and set that information as 
    * properties to be used later.
+   *
+   * @throws \CRM_Core_Exception
+   * @throws \CiviCRM_API3_Exception
    */
-  function setInfo() {
-
-    $this->event_type = $this->retrieve('event_type', 'String');
+  public function setInfo() {
     $this->test_mode = $this->retrieve('test_mode', 'Integer');
 
     $abort = FALSE;
@@ -478,7 +496,7 @@ class CRM_Core_Payment_StripeIPN extends CRM_Core_Payment_BaseIPN {
         }
       }
       catch(Exception $e) {
-        throw new CRM_Core_Exception('Cannot get contribution amounts from Stripe.');
+        $this->exception('Cannot get contribution amounts');
       }
     } else {
       // The customer had a credit on their subscription from a downgrade or gift card.
@@ -528,7 +546,7 @@ class CRM_Core_Payment_StripeIPN extends CRM_Core_Payment_BaseIPN {
           // This is an unrecoverable error - without a contribution_recur record
           // there is nothing we can do with an invoice.payment_succeeded
           // event.
-          throw new CRM_Core_Exception('I cannot find contribution_recur record for subscription: ' . $this->subscription_id);
+          $this->exception('I cannot find contribution_recur record for subscription: ' . $this->subscription_id);
         }
       }
 
@@ -564,14 +582,21 @@ class CRM_Core_Payment_StripeIPN extends CRM_Core_Payment_BaseIPN {
         }
 
         // Check for membership id.
+        // FIXME: MJW Not sure why we assign membership_id here
         $membership = civicrm_api3('Membership', 'get', array(
           'contribution_recur_id' => $this->contribution_recur_id,
         ));
         if ($membership['count'] == 1) {
           $this->membership_id = $membership['id'];
         }
-
       }
     }
   }
+
+  public function exception($message) {
+    $errorMessage = 'StripeIPN Exception: Event: ' . $this->event_type . ' Error: ' . $message;
+    Civi::log()->debug($errorMessage);
+    //throw new CRM_Core_Exception($errorMessage);
+    exit();
+  }
 }
diff --git a/CRM/Stripe/Customer.php b/CRM/Stripe/Customer.php
new file mode 100644
index 0000000000000000000000000000000000000000..5bfcfe4ea7b0cd388bf9c9f74f62a7b930c30482
--- /dev/null
+++ b/CRM/Stripe/Customer.php
@@ -0,0 +1,127 @@
+<?php
+
+class CRM_Stripe_Customer {
+
+  /**
+   * Find an existing Stripe customer in the CiviCRM database
+   *
+   * @param $params
+   *
+   * @return null|string
+   * @throws \CRM_Core_Exception
+   */
+  public static function find($params) {
+    $requiredParams = ['is_live', 'processor_id'];
+    foreach ($requiredParams as $required) {
+      if (empty($required)) {
+        throw new CRM_Core_Exception('Stripe Customer (find): Missing required parameter: ' . $required);
+      }
+    }
+    if (empty($params['contact_id'])) {
+      throw new CRM_Core_Exception('Stripe Customer (find): contact_id is required');
+    }
+    $queryParams = [
+      1 => [$params['contact_id'], 'String'],
+      2 => [$params['is_live'], 'Boolean'],
+      3 => [$params['processor_id'], 'Positive'],
+    ];
+
+    return CRM_Core_DAO::singleValueQuery("SELECT id
+      FROM civicrm_stripe_customers
+      WHERE contact_id = %1 AND is_live = %2 AND processor_id = %3", $queryParams);
+  }
+
+  /**
+   * Add a new Stripe customer to the CiviCRM database
+   *
+   * @param $params
+   *
+   * @throws \CRM_Core_Exception
+   */
+  public static function add($params) {
+    $requiredParams = ['contact_id', 'customer_id', 'is_live', 'processor_id'];
+    foreach ($requiredParams as $required) {
+      if (empty($required)) {
+        throw new CRM_Core_Exception('Stripe Customer (add): Missing required parameter: ' . $required);
+      }
+    }
+
+    $queryParams = [
+      1 => [$params['contact_id'], 'String'],
+      2 => [$params['customer_id'], 'String'],
+      3 => [$params['is_live'], 'Boolean'],
+      4 => [$params['processor_id'], 'Integer'],
+    ];
+    CRM_Core_DAO::executeQuery("INSERT INTO civicrm_stripe_customers
+          (contact_id, id, is_live, processor_id) VALUES (%1, %2, %3, %4)", $queryParams);
+  }
+
+  public static function create($params, $paymentProcessor) {
+    $requiredParams = ['contact_id', 'card_token', 'is_live', 'processor_id'];
+    // $optionalParams = ['email'];
+    foreach ($requiredParams as $required) {
+      if (empty($required)) {
+        throw new CRM_Core_Exception('Stripe Customer (create): Missing required parameter: ' . $required);
+      }
+    }
+
+    $contactDisplayName = civicrm_api3('Contact', 'getvalue', [
+      'return' => 'display_name',
+      'id' => $params['contact_id'],
+    ]);
+
+    $sc_create_params = [
+      'description' => $contactDisplayName . ' (CiviCRM)',
+      'card' => $params['card_token'],
+      'email' => CRM_Utils_Array::value('email', $params),
+      'metadata' => ['civicrm_contact_id' => $params['contact_id']],
+    ];
+
+    $stripeCustomer = $paymentProcessor->stripeCatchErrors('create_customer', $sc_create_params, $params);
+
+    // Store the relationship between CiviCRM's email address for the Contact & Stripe's Customer ID.
+    if (isset($stripeCustomer)) {
+      if ($paymentProcessor->isErrorReturn($stripeCustomer)) {
+        return $stripeCustomer;
+      }
+
+      $params = [
+        'contact_id' => $params['contact_id'],
+        'customer_id' => $stripeCustomer->id,
+        'is_live' => $params['is_live'],
+        'processor_id' => $params['processor_id'],
+      ];
+      self::add($params);
+    }
+    else {
+      Throw new CRM_Core_Exception(ts('There was an error saving new customer within Stripe.'));
+    }
+    return $stripeCustomer;
+  }
+
+  /**
+   * Delete a Stripe customer from the CiviCRM database
+   *
+   * @param array $params
+   *
+   * @throws \CRM_Core_Exception
+   */
+  public static function delete($params) {
+    $requiredParams = ['contact_id', 'is_live', 'processor_id'];
+    foreach ($requiredParams as $required) {
+      if (empty($required)) {
+        throw new CRM_Core_Exception('Stripe Customer (delete): Missing required parameter: ' . $required);
+      }
+    }
+
+    $queryParams = [
+      1 => [$params['contact_id'], 'String'],
+      2 => [$params['is_live'], 'Boolean'],
+      3 => [$params['processor_id'], 'Integer'],
+    ];
+    $sql = "DELETE FROM civicrm_stripe_customers
+            WHERE contact_id = %1 AND is_live = %2 AND processor_id = %3";
+    CRM_Core_DAO::executeQuery($sql, $queryParams);
+  }
+
+}
diff --git a/CRM/Stripe/Page/Webhook.php b/CRM/Stripe/Page/Webhook.php
deleted file mode 100644
index ed454eb7aeb0ddf0e8a59859c811ccad1cadcff3..0000000000000000000000000000000000000000
--- a/CRM/Stripe/Page/Webhook.php
+++ /dev/null
@@ -1,494 +0,0 @@
-<?php
-/*
- * @file
- * Handle Stripe Webhooks for recurring payments.
- */
-
-require_once 'CRM/Core/Page.php';
-
-class CRM_Stripe_Page_Webhook extends CRM_Core_Page {
-  function run() {
-    function getRecurInfo($subscription_id,$test_mode) {
-
-        $query_params = array(
-          1 => array($subscription_id, 'String'),
-        );
-        $sub_info_query = CRM_Core_DAO::executeQuery("SELECT contribution_recur_id
-          FROM civicrm_stripe_subscriptions
-          WHERE subscription_id = %1",
-          $query_params);
-
-        if (!empty($sub_info_query)) {
-          $sub_info_query->fetch();
-
-          if(!empty($sub_info_query->contribution_recur_id)) {
-          $recurring_info = new StdClass;
-          $recurring_info->id = $sub_info_query->contribution_recur_id;
-          }
-          else {
-            header('HTTP/1.1 400 Bad Request');
-            CRM_Core_Error::Fatal("Error relating this subscription id ($subscription_id) to the one in civicrm_stripe_subscriptions");
-            CRM_Utils_System::civiExit();
-          }
-        }
-        // Same approach as api repeattransaction. Find last contribution ascociated
-        // with our recurring contribution.
-        $recurring_info->previous_contribution_id = civicrm_api3('contribution', 'getvalue', array(
-         'return' => 'id',
-         'contribution_recur_id' => $recurring_info->id,
-         'options' => array('limit' => 1, 'sort' => 'id DESC'),
-         'contribution_test' => $test_mode,
-        ));
-        // Workaround for CRM-19945.
-        try {
-          $recurring_info->previous_completed_contribution_id = civicrm_api3('contribution', 'getvalue', array(
-           'return' => 'id',
-           'contribution_recur_id' => $recurring_info->id,
-           'contribution_status_id' => array('IN' => array('Completed')),
-           'options' => array('limit' => 1, 'sort' => 'id DESC'),
-           'contribution_test' => $test_mode,
-          ));
-        } catch (Exception $e) {
-         // This is fine....could only be a pending in the db.
-        }
-        if (!empty($recurring_info->previous_contribution_id)) {
-         //$previous_contribution_query->fetch();
-         }
-        else {
-          header('HTTP/1.1 400 Bad Request');
-          CRM_Core_Error::Fatal("ERROR: Stripe could not find contribution ($recurring_info->previous_contribution_id)  in civicrm_contribution: " . $stripe_event_data);
-          CRM_Utils_System::civiExit();
-        }
-        $current_recurring_contribution = civicrm_api3('ContributionRecur', 'get', array(
-          'sequential' => 1,
-          'return' => "payment_processor_id, financial_type_id, payment_instrument_id",
-          'id' => $recurring_info->id,
-         ));
-        $recurring_info->payment_processor_id = $current_recurring_contribution['values'][0]['payment_processor_id'];
-        $recurring_info->financial_type_id = $current_recurring_contribution['values'][0]['financial_type_id'];
-        $recurring_info->payment_instrument_id = $current_recurring_contribution['values'][0]['payment_instrument_id'];
-        $recurring_info->contact_id = civicrm_api3('Contribution', 'getvalue', array(
-         'sequential' => 1,
-         'return' => "contact_id",
-         'id' => $recurring_info->previous_contribution_id,
-        ));
-
-        return $recurring_info;
-    }
-    // Get the data from Stripe.
-    $data_raw = file_get_contents("php://input");
-    $data = json_decode($data_raw);
-    if (!$data) {
-      header('HTTP/1.1 406 Not acceptable');
-      CRM_Core_Error::Fatal("Stripe Callback: cannot json_decode data, exiting. <br /> $data");
-      CRM_Utils_System::civiExit();
-    }
-
-    // Test mode is the opposite of live mode.
-    $test_mode = (int)!$data->livemode;
-
-    $processorId = CRM_Utils_Request::retrieve('ppid', 'Integer');
-    try {
-      if (empty($processorId)) {
-        $stripe_key = civicrm_api3('PaymentProcessor', 'getvalue', array(
-          'return' => 'user_name',
-          'payment_processor_type_id' => 'Stripe',
-          'is_test' => $test_mode,
-          'is_active' => 1,
-          'options' => array('limit' => 1),
-        ));
-      }
-      else {
-        $stripe_key = civicrm_api3('PaymentProcessor', 'getvalue', array(
-          'return' => 'user_name',
-          'id' => $processorId,
-        ));
-      }
-    }
-    catch (CiviCRM_API3_Exception $e) {
-      header('HTTP/1.1 400 Bad Request');
-      CRM_Core_Error::fatal('Cannot find Stripe API key: ' . $e->getMessage());
-      CRM_Utils_System::civiExit();
-    }
-
-    \Stripe\Stripe::setAppInfo('CiviCRM', CRM_Utils_System::version(), CRM_Utils_System::baseURL());
-    \Stripe\Stripe::setApiKey($stripe_key);
-
-    // Retrieve Event from Stripe using ID even though we already have the values now.
-    // This is for extra security precautions mentioned here: https://stripe.com/docs/webhooks
-    $stripe_event_data = \Stripe\Event::retrieve($data->id);
-    // Not all event objects have a customer property. Check first.
-    if (isset($stripe_event_data->data->object->customer)) {
-      $customer_id = $stripe_event_data->data->object->customer;
-    }  
-    
-    switch($stripe_event_data->type) {
-      // Successful recurring payment.
-      case 'invoice.payment_succeeded':
-        $subscription_id = $stripe_event_data->data->object->subscription;
-        $new_invoice_id = $stripe_event_data->data->object->id;
-        $receive_date = date("Y-m-d H:i:s", $stripe_event_data->data->object->date);
-        $charge_id = $stripe_event_data->data->object->charge;
-
-        // Get the Stripe charge object if one exists. Null charge still needs processing.
-        if ( $charge_id !== null ) {
-          try {
-            $charge = \Stripe\Charge::retrieve($charge_id);
-            $balance_transaction_id = $charge->balance_transaction;
-            $balance_transaction = \Stripe\BalanceTransaction::retrieve($balance_transaction_id);
-	    $amount = $charge->amount / 100;
-	    $fee = $balance_transaction->fee / 100;
-          }
-          catch(Exception $e) {
-            header('HTTP/1.1 400 Bad Request');
-            CRM_Core_Error::Fatal("Failed to retrieve Stripe charge.  Message: " . $e->getMessage());
-            CRM_Utils_System::civiExit();
-          }
-        } else {
-        // The customer had a credit on their subscription from a downgrade or gift card.
-        $amount = 0;
-        $fee = 0;
-        }
-
-        // First, get the recurring contribution id and previous contribution id.
-        $recurring_info = getRecurInfo($subscription_id,$test_mode);
-
-        // Fetch the previous contribution's status.
-        $previous_contribution = civicrm_api3('Contribution', 'get', array(
-          'sequential' => 1,
-          'return' => "contribution_status_id,invoice_id",
-          'id' => $recurring_info->previous_contribution_id,
-	  'contribution_test' => $test_mode,
-         ));
-        $previous_contribution_status = $previous_contribution['values'][0]['contribution_status_id'];
-
-        // Check if the previous contribution's status is pending and update it
-        // using create and then complete it, else repeat it if not pending.
-        // When a member upgrades/downgrades mid-term, (or recurring contributor
-        // changes levels), we are in a unique situation not knowing ahead of time
-        // what the contribution amount really is. completetransaction can't modify
-        // our amounts (except for fee). We'll need to update the contribution amounts
-        // to the actual values from Stripe for accounting.
-
-        if ($previous_contribution_status == "2") {
-          // Note: using create contribution to edit won't recalculate the net_amount.
-          // We need to calculate and explicitly change it.
-          $net_amount = $amount - $fee;
-          $pending_contribution = civicrm_api3('Contribution', 'create', array(
-           'id' => $recurring_info->previous_contribution_id,
- 	    'total_amount' => $amount,
-            'fee_amount' => $fee,
-            'net_amount' => $net_amount,
-            'receive_date' => $receive_date,
-           ));
-          // Leave some indication that this is legitimately supposed to be a $0 contribution,
-          // by not leaving trxn_id empty.
-          if ( $amount == 0 ) {
-            $charge_id = $previous_contribution['values'][0]['invoice_id'];
-          }
-          // Now complete it.
-          $result = civicrm_api3('Contribution', 'completetransaction', array(
-            'sequential' => 1,
-            'id' => $recurring_info->previous_contribution_id,
-            'trxn_date' => $receive_date,
-            'trxn_id' => $charge_id,
-            'total_amount' => $amount,
-            'fee_amount' => $fee,
-           ));
-
-          return;
-
-	 } else {
-
-	 // api contribution repeattransaction repeats the appropriate contribution if it is given
-	 // simply the recurring contribution id. It also updates the membership for us. However,
-         // we add the amount and fee regardless of the expected amounts because we may have
-         // upgraded or downgraded the membership, or recurring contribution level.  This means
-         // prorated invoices.
-
-         $result = civicrm_api3('Contribution', 'repeattransaction', array(
-           // Actually, don't use contribution_recur_id until CRM-19945 patches make it in to 4.6/4.7
-           // and we have a way to require a minimum minor CiviCRM version.
-	   //'contribution_recur_id' => $recurring_info->id,
-	   'original_contribution_id' => $recurring_info->previous_completed_contribution_id,
-           'contribution_status_id' => "Completed",
-           'receive_date' => $receive_date,
-           'trxn_id' => $charge_id,
- 	   'total_amount' => $amount,
-	   'fee_amount' => $fee,
-	  //'invoice_id' => $new_invoice_id - contribution.repeattransaction doesn't support it currently
-	   'is_email_receipt' => 1,
-         ));
-
-        // Update invoice_id manually. repeattransaction doesn't return the new contrib id either, so we update the db.
-        $query_params = array(
-          1 => array($new_invoice_id, 'String'),
-          2 => array($charge_id, 'String'),
-         );
-        CRM_Core_DAO::executeQuery("UPDATE civicrm_contribution
-          SET invoice_id = %1
-          WHERE trxn_id = %2",
-         $query_params);
-
-
-        // Successful charge & more to come
-        $result = civicrm_api3('ContributionRecur', 'create', array(
-          'sequential' => 1,
-          'id' => $recurring_info->id,
-          'failure_count' => 0,
-          'contribution_status_id' => "In Progress"
-         ));
-
-        CRM_Utils_System::civiExit();
-        }
-        break;
-
-      // Failed recurring payment.
-      case 'invoice.payment_failed':
-        // Get the Stripe charge object.
-        try {
-          $charge = \Stripe\Charge::retrieve($stripe_event_data->data->object->charge);
-        }
-        catch(Exception $e) {
-          header('HTTP/1.1 400 Bad Request');
-          CRM_Core_Error::Fatal("Failed to retrieve Stripe charge.  Message: " . $e->getMessage());
-          CRM_Utils_System::civiExit();
-        }
-
-        // Build some params.
-        $subscription_id = $stripe_event_data->data->object->subscription;
-        $new_invoice_id = $stripe_event_data->data->object->id;
-        $charge_id = $stripe_event_data->data->object->charge;
-        $attempt_count = $stripe_event_data->data->object->attempt_count;
-        $fail_date = date("Y-m-d H:i:s");
-        $amount = $charge->amount / 100;
-        $fee_amount = isset($charge->fee) ? ($charge->fee / 100) : 0;
-        $transaction_id = $charge->id;
-
-        // First, get the recurring contribution id and previous contribution id.
-        $recurring_info = getRecurInfo($subscription_id,$test_mode);
-
-        // Fetch the previous contribution's status.
-        $previous_contribution_status = civicrm_api3('Contribution', 'getvalue', array(
-          'sequential' => 1,
-          'return' => "contribution_status_id",
-          'id' => $recurring_info->previous_contribution_id,
-	  'contribution_test' => $test_mode,
-         ));
-
-          if ($previous_contribution_status == 2) {
-          // If this contribution is Pending, set it to Failed.
-          $result = civicrm_api3('Contribution', 'create', array(
-            'id' => $recurring_info->previous_contribution_id,
-	    'contribution_recur_id' => $recurring_info->id,
-            'contribution_status_id' => "Failed",
-            'contact_id' => $recurring_info->contact_id,
-            'financial_type_id' => $recurring_info->financial_type_id,
-            'receive_date' => $fail_date,
-            'total_amount' => $amount,
-	    'is_email_receipt' => 1,
-	    'is_test' => $test_mode,
-          ));
-
-          }
-          else {
-          // Record a Failed contribution. Use repeattransaction for this when CRM-19984
-          // patch makes it in 4.6/4.7.
-          $result = civicrm_api3('Contribution', 'create', array(
-	    'contribution_recur_id' => $recurring_info->id,
-            'contribution_status_id' => "Failed",
-            'contact_id' => $recurring_info->contact_id,
-            'financial_type_id' => $recurring_info->financial_type_id,
-            'receive_date' => $fail_date,
-            'total_amount' => $amount,
-	    'is_email_receipt' => 1,
-	    'is_test' => $test_mode,
-          ));
-         }
-
-          $failure_count = civicrm_api3('ContributionRecur', 'getvalue', array(
-            'sequential' => 1,
-            'id' => $recurring_info->id,
-            'return' => 'failure_count',
-            ));
-          $failure_count++;
-          //  Change the status of the Recurring and update failed attempts.
-          $result = civicrm_api3('ContributionRecur', 'create', array(
-            'sequential' => 1,
-            'id' => $recurring_info->id,
-            'contribution_status_id' => "Failed",
-            'failure_count' => $failure_count,
-            'modified_date' => $fail_date,
-            'is_test' => $test_mode,
-            ));
-
-          return;
-        break;
-
-
-      //Subscription is cancelled
-      case 'customer.subscription.deleted':
-        $subscription_id = $stripe_event_data->data->object->id;
-
-        // First, get the recurring contribution id and previous contribution id.
-        $recurring_info = getRecurInfo($subscription_id,$test_mode);
-
-        //Cancel the recurring contribution
-        $result = civicrm_api3('ContributionRecur', 'cancel', array(
-            'sequential' => 1,
-            'id' => $recurring_info->id,
-        ));
-
-        //Delete the record from Stripe's subscriptions table
-        $query_params = array(
-            1 => array($subscription_id, 'String'),
-        );
-        CRM_Core_DAO::executeQuery("DELETE FROM civicrm_stripe_subscriptions
-              WHERE subscription_id = %1", $query_params);
-
-        break;
-
-      // One-time donation and per invoice payment.
-      case 'charge.succeeded':
-        // Not implemented.
-        CRM_Utils_System::civiExit();
-        break;
-
-
-      // Subscription is updated. Delete existing recurring contribution and start a fresh one.
-      // This tells a story to site admins over editing a recurring contribution record.
-     case 'customer.subscription.updated':
-         if (empty($stripe_event_data->data->previous_attributes->plan->id)) {
-           // Not a plan change...don't care.
-           CRM_Utils_System::civiExit();
-         }
-         $subscription_id = $stripe_event_data->data->object->id;
-         $new_amount = $stripe_event_data->data->object->plan->amount / 100;
-         $new_frequency_interval = $stripe_event_data->data->object->plan->interval_count;
-         $new_frequency_unit = $stripe_event_data->data->object->plan->interval;
-         $plan_id = $stripe_event_data->data->object->plan->id;
-         $plan_name = $stripe_event_data->data->object->plan->name;
-         $plan_elements = explode("-", $plan_id);
-         $plan_name_elements = explode("-", $plan_name);
-         $created_date = date("Y-m-d H:i:s", $stripe_event_data->data->object->start);
-         $new_civi_invoice = md5(uniqid(rand(), TRUE));
-
-         // First, get the recurring contribution id and previous contribution id.
-         $recurring_info = getRecurInfo($subscription_id,$test_mode);
-
-         // Is there a pending charge due to a subcription change?  Make up your mind!!
-         $previous_contribution = civicrm_api3('Contribution', 'get', array(
-          'sequential' => 1,
-          'return' => "contribution_status_id,invoice_id",
-          'id' => $recurring_info->previous_contribution_id,
-	  'contribution_test' => $test_mode,
-         ));
-         if ($previous_contribution['values'][0]['contribution_status_id'] == "2") {
-           // Cancel the pending contribution.
-           $result = civicrm_api3('Contribution', 'delete', array(
-            'sequential' => 1,
-            'id' => $recurring_info->previous_contribution_id,
-           ));
-         }
-
-         // Cancel the old recurring contribution.
-         $result = civicrm_api3('ContributionRecur', 'cancel', array(
-          'sequential' => 1,
-          'id' => $recurring_info->id
-         ));
-
-         $new_recurring_contribution = civicrm_api3('ContributionRecur', 'create', array(
-          'sequential' => 1,
-          'contact_id' => $recurring_info->contact_id,
-          'invoice_id' => $new_civi_invoice,
-          'amount' => $new_amount,
-          'auto_renew' => 1,
-          'created_date' => $created_date,
-          'frequency_unit' => $new_frequency_unit,
-          'frequency_interval' => $new_frequency_interval,
-          'contribution_status_id' => "In Progress",
-          'payment_processor_id' =>  $recurring_info->payment_processor_id,
-          'financial_type_id' => $recurring_info->financial_type_id,
-          'payment_instrument_id' => $recurring_info->payment_instrument_id,
-          'is_test' => $test_mode,
-          ));
-         $new_recurring_contribution_id = $new_recurring_contribution['values'][0]['id'];
-         $new_contribution = civicrm_api3('Contribution', 'create', array(
-          'sequential' => 1,
-          'contact_id' => $recurring_info->contact_id,
-          'invoice_id' => $new_civi_invoice,
-          'total_amount' => $new_amount,
-          'contribution_recur_id' => $new_recurring_contribution_id,
-          'contribution_status_id' => "Pending",
-          'financial_type_id' => $recurring_info->financial_type_id,
-          'payment_instrument_id' => $recurring_info->payment_instrument_id,
-          'note' => "Created by Stripe webhook.",
-          'is_test' => $test_mode,
-          ));
-
-          // Prepare escaped query params.
-      $query_params = array(
-        1 => array($new_recurring_contribution_id, 'Integer'),
-        2 => array($subscription_id, 'String'),
-      );
-      CRM_Core_DAO::executeQuery("UPDATE civicrm_stripe_subscriptions
-        SET contribution_recur_id  = %1 where subscription_id = %2", $query_params);
-
-       // Find out if the plan is ascociated with a membership and if so
-       // adjust it to the new level.
-
-          $membership_result = civicrm_api3('Membership', 'get', array(
-           'sequential' => 1,
-           'return' => "membership_type_id,id",
-           'contribution_recur_id' => $recurring_info->id,
-          ));
-
-          if ("membertype_" == substr($plan_elements[0],0,11)) {
-            $new_membership_type_id = substr($plan_elements[0],strrpos($plan_elements[0],'_') + 1);
-          } else if  ("membertype_" == substr($plan_name_elements[0],0,11)) {
-             $new_membership_type_id = substr($plan_name_elements[0],strrpos($plan_name_elements[0],'_') + 1);
-          }
-
-          // Adjust to the new membership level.
-          if (!empty($new_membership_type_id)) {
-            $membership_id = $membership_result['values'][0]['id'];
-            $result = civicrm_api3('Membership', 'create', array(
-             'sequential' => 1,
-             'id' => $membership_id,
-             'membership_type_id' => $new_membership_type_id,
-             'contact_id' => $recurring_info->contact_id,
-             'contribution_recur_id' => $new_recurring_contribution_id,
-             'num_terms' => 0,
-            ));
-
-          // Create a new membership payment record.
-          $result = civicrm_api3('MembershipPayment', 'create', array(
-           'sequential' => 1,
-           'membership_id' => $membership_id,
-           'contribution_id' => $new_contribution['values'][0]['id'],
-          ));
-           }
-
-          break;
-
-      // Keep plans table in sync with Stripe when a plan is deleted.
-     case 'plan.deleted':
-       $plan_id = $stripe_event_data->data->object->id;
-       // Prepare escaped query params.
-        $query_params = array(
-          1 => array($plan_id, 'String'),
-          2 => array($processorId, 'Integer'),
-        );
-       CRM_Core_DAO::executeQuery("DELETE FROM civicrm_stripe_plans WHERE
-         plan_id = %1 AND  processor_id = %2", $query_params);
-
-       break;
-
-       return;
-
-    }
-
-    parent::run();
-  }
-
-}
diff --git a/CRM/Stripe/Upgrader.php b/CRM/Stripe/Upgrader.php
index 30b7caa39d74550b85644082baadeee93e473c8b..656ffd0247d22cd91a5f08dbfcf0dd4e1591ee5f 100644
--- a/CRM/Stripe/Upgrader.php
+++ b/CRM/Stripe/Upgrader.php
@@ -10,18 +10,6 @@ class CRM_Stripe_Upgrader extends CRM_Stripe_Upgrader_Base {
   // By convention, functions that look like "function upgrade_NNNN()" are
   // upgrade tasks. They are executed in order (like Drupal's hook_update_N).
 
-  /**
-   * Standard: run an install sql script
-   */
-  public function install() {
-  }
-
-  /**
-   * Standard: run an uninstall script
-   */
-  public function uninstall() {
-  }
-
   /**
    * Add is_live column to civicrm_stripe_plans and civicrm_stripe_customers tables.
    *
@@ -379,4 +367,20 @@ class CRM_Stripe_Upgrader extends CRM_Stripe_Upgrader_Base {
     }
     return TRUE;
   }
+
+  public function upgrade_5010() {
+    $this->ctx->log->info('Applying Stripe update 5010.  Adding contact_id to civicrm_stripe_customers.');
+    if (!CRM_Core_BAO_SchemaHandler::checkIfFieldExists('civicrm_stripe_customers', 'contact_id', FALSE)) {
+      CRM_Core_DAO::executeQuery('ALTER TABLE `civicrm_stripe_customers` 
+       ADD COLUMN `contact_id` int(10) UNSIGNED DEFAULT NULL COMMENT "FK ID from civicrm_contact"');
+      CRM_Core_DAO::executeQuery('ALTER TABLE `civicrm_stripe_customers` 
+       ADD CONSTRAINT `FK_civicrm_stripe_customers_contact_id` FOREIGN KEY (`contact_id`) REFERENCES `civicrm_contact` (`id`) ON DELETE CASCADE;');
+    }
+
+    $this->ctx->log->info('Applying Stripe update 5010. Getting Contact IDs for civicrm_stripe_customers.');
+    civicrm_api3('StripeCustomer', 'updatecontactids', []);
+
+    return TRUE;
+  }
+
 }
diff --git a/README.md b/README.md
index 66229b1d8142b261bcdb569e6b033c30215e5209..adf1dd3da161fb1fabbe7c42677742f1c0a2eab8 100644
--- a/README.md
+++ b/README.md
@@ -71,7 +71,7 @@ Therefore the call back address for your site will be:
 See below for the full address to add to the endpoint (replace NN with your actual ID number):
 
 * For Drupal:  https://example.com/civicrm/payment/ipn/NN
-* For Joomla:  https://example.com/index.php/component/civicrm/?task=civicrm/payment/ipn/NN
+* For Joomla:  https://example.com/?option=com_civicrm&task=civicrm/payment/ipn/NN
 * For Wordpress:  https://example.com/?page=CiviCRM&q=civicrm/payment/ipn/NN
 
 Typically, you only need to configure the end point to send live transactions and you want it to send all events.
diff --git a/api/v3/StripeCustomer.php b/api/v3/StripeCustomer.php
new file mode 100644
index 0000000000000000000000000000000000000000..cf01932aadc389892c38e8d8456fed9640a8a717
--- /dev/null
+++ b/api/v3/StripeCustomer.php
@@ -0,0 +1,160 @@
+<?php
+
+/**
+ * Stripe Customer API
+ *
+ */
+
+/**
+ * StripeCustomer.Get API specification
+ *
+ * @param array $spec description of fields supported by this API call
+ * @return void
+ * @see http://wiki.civicrm.org/confluence/display/CRMDOC/API+Architecture+Standards
+ */
+function _civicrm_api3_stripe_customer_get_spec(&$spec) {
+  $spec['id']['title'] = ts("Stripe Customer ID");
+  $spec['id']['type'] = CRM_Utils_Type::T_STRING;
+  $spec['contact_id']['title'] = ts("CiviCRM Contact ID");
+  $spec['contact_id']['type'] = CRM_Utils_Type::T_INT;
+  $spec['is_live']['title'] = ts("Is live processor");
+  $spec['is_live']['type'] = CRM_Utils_Type::T_BOOLEAN;
+  $spec['processor_id']['title'] = ts("Payment Processor ID");
+  $spec['processor_id']['type'] = CRM_Utils_Type::T_INT;
+}
+
+/**
+ * StripeCustomer.Get API
+ *  This api will update the civicrm_stripe_customers table and add contact IDs for all known email addresses
+ *
+ * @param array $params
+ * @see civicrm_api3_create_success
+ *
+ * @return array
+ */
+function civicrm_api3_stripe_customer_get($params) {
+  foreach ($params as $key => $value) {
+    $index = 1;
+    switch ($key) {
+      case 'id':
+        $where[$index] = "{$key}=%{$index}";
+        $whereParam[$index] = [$value, 'String'];
+        $index++;
+        break;
+
+      case 'contact_id':
+        $where[$index] = "{$key}=%{$index}";
+        $whereParam[$index] = [$value, 'Integer'];
+        $index++;
+        break;
+
+      case 'is_live':
+        $where[$index] = "{$key}=%{$index}";
+        $whereParam[$index] = [$value, 'Boolean'];
+        $index++;
+        break;
+
+      case 'processor_id':
+        $where[$index] = "{$key}=%{$index}";
+        $whereParam[$index] = [$value, 'Integer'];
+        $index++;
+        break;
+    }
+  }
+
+  $query = "SELECT * FROM civicrm_stripe_customers ";
+  if (count($where)) {
+    $whereClause = implode(' AND ', $where);
+    $query .= "WHERE {$whereClause}";
+  }
+  $dao = CRM_Core_DAO::executeQuery($query, $whereParam);
+
+  while ($dao->fetch()) {
+    $result = [
+      'id' => $dao->id,
+      'contact_id' => $dao->contact_id,
+      'is_live' => $dao->is_live,
+      'processor_id' => $dao->processor_id,
+    ];
+    if ($dao->email) {
+      $result['email'] = $dao->email;
+    }
+    $results[] = $result;
+  }
+  return civicrm_api3_create_success($results);
+}
+
+/**
+ * Stripe.Customer.Updatecontactids API
+ *  This api will update the civicrm_stripe_customers table and add contact IDs for all known email addresses
+ *
+ * @param array $params
+ * @see civicrm_api3_create_success
+ *
+ * @return array
+ */
+function civicrm_api3_stripe_customer_updatecontactids($params) {
+  $dao = CRM_Core_DAO::executeQuery('SELECT email, id FROM civicrm_stripe_customers WHERE contact_id IS NULL');
+  $counts = [
+    'updated' => 0,
+    'failed' => 0,
+  ];
+  while ($dao->fetch()) {
+    $contactId = NULL;
+    try {
+      $contactId = civicrm_api3('Contact', 'getvalue', [
+        'return' => "id",
+        'email' => $dao->email,
+      ]);
+    }
+    catch (Exception $e) {
+      // Most common problem is duplicates.
+      if(preg_match("/Expected one Contact but found/", $e->getMessage())) {
+        // If we find more than one, first try to find it via a related subscription record
+        // using the customer id.
+        $sql = "SELECT c.id
+          FROM civicrm_contribution_recur rc
+            JOIN civicrm_stripe_subscriptions sc ON
+              rc.id = sc.contribution_recur_id
+            JOIN civicrm_contact c ON c.id = rc.contact_id
+          WHERE c.is_deleted = 0 AND customer_id = %0
+          ORDER BY start_date DESC LIMIT 1";
+        $dao_contribution = CRM_Core_DAO::executeQuery($sql, [0 => [$dao->id, 'String']]);
+        $dao_contribution->fetch();
+        if ($dao_contribution->id) {
+          $contactId = $dao_contribution->id;
+        }
+        if (empty($contactId)) {
+          // Still no luck. Now get desperate.
+          $sql = "SELECT c.id
+            FROM civicrm_contact c JOIN civicrm_email e ON c.id = e.contact_id
+            JOIN civicrm_contribution cc ON c.id = cc.contact_id
+            WHERE e.email = %0 AND c.is_deleted = 0 AND is_test = 0 AND
+              trxn_id LIKE 'ch_%' AND contribution_status_id = 1
+            ORDER BY receive_date DESC LIMIT 1";
+          $dao_contribution = CRM_Core_DAO::executeQuery($sql, [0 => [$dao->email, 'String']]);
+          $dao_contribution->fetch();
+          if ($dao_contribution->id) {
+            $contactId = $dao_contribution->id;
+          }
+        }
+      }
+      if (empty($contactId)) {
+        // Still no luck. Log it and move on.
+        Civi::log()->debug('Stripe Upgrader: No contact ID found for stripe customer with email: ' . $dao->email);
+        $counts['failed']++;
+        continue;
+      }
+    }
+
+    $sqlParams = [
+      1 => [$contactId, 'Integer'],
+      2 => [$dao->email, 'String'],
+    ];
+    $sql = 'UPDATE civicrm_stripe_customers SET contact_id=%1 WHERE email=%2';
+    CRM_Core_DAO::executeQuery($sql, $sqlParams);
+    $counts['updated']++;
+  }
+
+  return civicrm_api3_create_success($counts);
+}
diff --git a/info.xml b/info.xml
index 1eb8f63006280c51a580e6ad7a7ff1bd5a9b6dc6..b145ab5078bdcd828b710cb4b2b12b611578a113 100644
--- a/info.xml
+++ b/info.xml
@@ -12,11 +12,11 @@
     <author>Matthew Wire (MJW Consulting)</author>
     <email>mjw@mjwconsult.co.uk</email>
   </maintainer>
-  <releaseDate>2018-09-19</releaseDate>
-  <version>5.0</version>
-  <develStage>stable</develStage>
+  <releaseDate>2018-10-18</releaseDate>
+  <version>5.1</version>
+  <develStage>beta</develStage>
   <compatibility>
-    <ver>5.0</ver>
+    <ver>5.3</ver>
   </compatibility>
   <comments>Original Author: Joshua Walker (drastik) - Drastik by Design.
     Jamie Mcclelland (ProgressiveTech) did a lot of the 5.x compatibility work.
diff --git a/sql/auto_install.sql b/sql/auto_install.sql
new file mode 100644
index 0000000000000000000000000000000000000000..6c7a9db03dd55c7b2d9a8c32cb435297a1903699
--- /dev/null
+++ b/sql/auto_install.sql
@@ -0,0 +1,29 @@
+/* Create required tables for Stripe */
+  CREATE TABLE IF NOT EXISTS `civicrm_stripe_customers` (
+    `id` varchar(255) COLLATE utf8_unicode_ci DEFAULT NULL,
+    `contact_id` int(10) UNSIGNED DEFAULT NULL COMMENT 'FK ID from civicrm_contact',
+    `is_live` tinyint(4) NOT NULL COMMENT 'Whether this is a live or test transaction',
+    `processor_id` int(10) DEFAULT NULL COMMENT 'ID from civicrm_payment_processor',
+    UNIQUE KEY `id` (`id`),
+    CONSTRAINT `FK_civicrm_stripe_customers_contact_id` FOREIGN KEY (`contact_id`)
+      REFERENCES `civicrm_contact` (`id`) ON DELETE CASCADE
+    ) ENGINE=InnoDB DEFAULT CHARSET=utf8 COLLATE=utf8_unicode_ci;
+
+  CREATE TABLE IF NOT EXISTS `civicrm_stripe_plans` (
+    `plan_id` varchar(255) COLLATE utf8_unicode_ci NOT NULL,
+    `is_live` tinyint(4) NOT NULL COMMENT 'Whether this is a live or test transaction',
+    `processor_id` int(10) DEFAULT NULL COMMENT 'ID from civicrm_payment_processor',
+    UNIQUE KEY `plan_id` (`plan_id`)
+    ) ENGINE=InnoDB DEFAULT CHARSET=utf8 COLLATE=utf8_unicode_ci;
+
+  CREATE TABLE IF NOT EXISTS `civicrm_stripe_subscriptions` (
+    `subscription_id` varchar(255) COLLATE utf8_unicode_ci NOT NULL,
+    `customer_id` varchar(255) COLLATE utf8_unicode_ci NOT NULL,
+    `contribution_recur_id` INT(10) UNSIGNED NULL DEFAULT NULL,
+    `end_time` int(11) NOT NULL DEFAULT '0',
+    `is_live` tinyint(4) NOT NULL COMMENT 'Whether this is a live or test transaction',
+    `processor_id` int(10) DEFAULT NULL COMMENT 'ID from civicrm_payment_processor',
+    KEY `end_time` (`end_time`), PRIMARY KEY `subscription_id` (`subscription_id`),
+    CONSTRAINT `FK_civicrm_stripe_contribution_recur_id` FOREIGN KEY (`contribution_recur_id`)
+    REFERENCES `civicrm_contribution_recur`(`id`) ON DELETE SET NULL ON UPDATE RESTRICT
+    ) ENGINE=InnoDB DEFAULT CHARSET=utf8 COLLATE=utf8_unicode_ci;
diff --git a/sql/auto_uninstall.sql b/sql/auto_uninstall.sql
new file mode 100644
index 0000000000000000000000000000000000000000..e687830c59cc9896cc67e07de02e99c2c110ef33
--- /dev/null
+++ b/sql/auto_uninstall.sql
@@ -0,0 +1,4 @@
+/* Remove Stripe tables on uninstall. */
+DROP TABLE civicrm_stripe_customers;
+DROP TABLE civicrm_stripe_plans;
+DROP TABLE civicrm_stripe_subscriptions;
\ No newline at end of file
diff --git a/stripe.php b/stripe.php
index bb1d5e2826ad79dfbb9a781ac529589541023db1..8fda3cd6a8d09db52c90234f66c83bb2a72550c3 100644
--- a/stripe.php
+++ b/stripe.php
@@ -25,41 +25,6 @@ function stripe_civicrm_xmlMenu(&$files) {
  * Implementation of hook_civicrm_install().
  */
 function stripe_civicrm_install() {
-  // Create required tables for Stripe.
-  require_once "CRM/Core/DAO.php";
-  CRM_Core_DAO::executeQuery("
-  CREATE TABLE IF NOT EXISTS `civicrm_stripe_customers` (
-    `email` varchar(64) COLLATE utf8_unicode_ci DEFAULT NULL,
-    `id` varchar(255) COLLATE utf8_unicode_ci DEFAULT NULL,
-    `is_live` tinyint(4) NOT NULL COMMENT 'Whether this is a live or test transaction',
-    `processor_id` int(10) DEFAULT NULL COMMENT 'ID from civicrm_payment_processor',
-    UNIQUE KEY `id` (`id`)
-    ) ENGINE=InnoDB DEFAULT CHARSET=utf8 COLLATE=utf8_unicode_ci;
-  ");
-
-  CRM_Core_DAO::executeQuery("
-  CREATE TABLE IF NOT EXISTS `civicrm_stripe_plans` (
-    `plan_id` varchar(255) COLLATE utf8_unicode_ci NOT NULL,
-    `is_live` tinyint(4) NOT NULL COMMENT 'Whether this is a live or test transaction',
-    `processor_id` int(10) DEFAULT NULL COMMENT 'ID from civicrm_payment_processor',
-    UNIQUE KEY `plan_id` (`plan_id`)
-    ) ENGINE=InnoDB DEFAULT CHARSET=utf8 COLLATE=utf8_unicode_ci;
-  ");
-
-  CRM_Core_DAO::executeQuery("
-  CREATE TABLE IF NOT EXISTS `civicrm_stripe_subscriptions` (
-    `subscription_id` varchar(255) COLLATE utf8_unicode_ci NOT NULL,
-    `customer_id` varchar(255) COLLATE utf8_unicode_ci NOT NULL,
-    `contribution_recur_id` INT(10) UNSIGNED NULL DEFAULT NULL,
-    `end_time` int(11) NOT NULL DEFAULT '0',
-    `is_live` tinyint(4) NOT NULL COMMENT 'Whether this is a live or test transaction',
-    `processor_id` int(10) DEFAULT NULL COMMENT 'ID from civicrm_payment_processor',
-    KEY `end_time` (`end_time`), PRIMARY KEY `subscription_id` (`subscription_id`),
-    CONSTRAINT `FK_civicrm_stripe_contribution_recur_id` FOREIGN KEY (`contribution_recur_id`) 
-    REFERENCES `civicrm_contribution_recur`(`id`) ON DELETE SET NULL ON UPDATE RESTRICT 
-    ) ENGINE=InnoDB DEFAULT CHARSET=utf8 COLLATE=utf8_unicode_ci;
-  ");
-
   _stripe_civix_civicrm_install();
 }
 
@@ -67,12 +32,6 @@ function stripe_civicrm_install() {
  * Implementation of hook_civicrm_uninstall().
  */
 function stripe_civicrm_uninstall() {
-  // Remove Stripe tables on uninstall.
-  require_once "CRM/Core/DAO.php";
-  CRM_Core_DAO::executeQuery("DROP TABLE civicrm_stripe_customers");
-  CRM_Core_DAO::executeQuery("DROP TABLE civicrm_stripe_plans");
-  CRM_Core_DAO::executeQuery("DROP TABLE civicrm_stripe_subscriptions");
-
   _stripe_civix_civicrm_uninstall();
 }
 
@@ -80,22 +39,12 @@ function stripe_civicrm_uninstall() {
  * Implementation of hook_civicrm_enable().
  */
 function stripe_civicrm_enable() {
-  $UF_webhook_paths = array(
-    "Drupal"    => "/civicrm/payment/ipn/NN",
-    "Joomla"    => "/index.php/component/civicrm/?task=civicrm/payment/ipn/NN",
-    "WordPress" => "/?page=CiviCRM&q=civicrm/payment/ipn/NN"
-  );
-
-  // Use Drupal path as default if the UF isn't in the map above
-  $webookhook_path = (array_key_exists(CIVICRM_UF, $UF_webhook_paths)) ?
-    CIVICRM_UF_BASEURL . $UF_webhook_paths[CIVICRM_UF] :
-    CIVICRM_UF_BASEURL . $UF_webhook_paths['Drupal'];
-
+  $UFWebhookPath = stripe_get_webhook_path(TRUE);
   CRM_Core_Session::setStatus(
     "
     <br />Don't forget to set up Webhooks in Stripe so that recurring contributions are ended!
     <br />Webhook path to enter in Stripe:
-    <br/><em>$webookhook_path</em>
+    <br/><em>$UFWebhookPath</em>
     <br />Replace NN with the actual payment processor ID configured on your site.
     <br />
     ",
@@ -251,21 +200,37 @@ function stripe_civicrm_buildForm($formName, &$form) {
   }
 }
 
+/**
+ * Get the path of the webhook depending on the UF (eg Drupal, Joomla, Wordpress)
+ *
+ * @param bool $includeBaseUrl
+ *
+ * @return string
+ */
+function stripe_get_webhook_path($includeBaseUrl = TRUE) {
+  $UFWebhookPaths = [
+    "Drupal"    => "civicrm/payment/ipn/NN",
+    "Joomla"    => "?option=com_civicrm&task=civicrm/payment/ipn/NN",
+    "WordPress" => "?page=CiviCRM&q=civicrm/payment/ipn/NN"
+  ];
+
+
+  // Use Drupal path as default if the UF isn't in the map above
+  $UFWebhookPath = (array_key_exists(CIVICRM_UF, $UFWebhookPaths)) ?
+    $UFWebhookPaths[CIVICRM_UF] :
+    $UFWebhookPaths['Drupal'];
+  if ($includeBaseUrl) {
+    return CIVICRM_UF_BASEURL . '/' . $UFWebhookPath;
+  }
+  return $UFWebhookPath;
+}
+
 /*
  * Implementation of hook_idsException.
  *
  * Ensure webhooks don't get caught in the IDS check.
  */
 function stripe_civicrm_idsException(&$skip) {
-  // Handle old method.
-  $skip[] = 'civicrm/stripe/webhook';
-  $result = civicrm_api3('PaymentProcessor', 'get', array(
-    'sequential' => 1,
-    'return' => "id",
-    'class_name' => "Payment_stripe",
-    'is_active' => 1,
-  ));
-  foreach($result['values'] as $value) {
-    $skip[] = 'civicrm/payment/ipn/' . $value['id'];
-  }
+  // Path is always set to civicrm/payment/ipn (checked on Drupal/Joomla)
+  $skip[] = 'civicrm/payment/ipn';
 }
diff --git a/templates/CRM/Stripe/Page/Webhook.tpl b/templates/CRM/Stripe/Page/Webhook.tpl
deleted file mode 100644
index 35e65227d2880ed1914437418cc34026ed05f17f..0000000000000000000000000000000000000000
--- a/templates/CRM/Stripe/Page/Webhook.tpl
+++ /dev/null
@@ -1,3 +0,0 @@
-<h3>This page is generated by CRM/Stripe/Page/Webhook.php</h3>
-
-<p>This is the path to use with your Webhook settings in your Stripe account.</p>
diff --git a/tests/katalon/civicrm-stripe-test-suite.html b/tests/katalon/civicrm-stripe-test-suite.html
index 597c470d4ddeceedc623bcc420fbcc9d77797bd9..23dd702bb2d62877360ba98f1d7be47027c64063 100644
--- a/tests/katalon/civicrm-stripe-test-suite.html
+++ b/tests/katalon/civicrm-stripe-test-suite.html
@@ -3,7 +3,7 @@
 <html xmlns="http://www.w3.org/1999/xhtml" xml:lang="en" lang="en">
 <head>
 	<meta content="text/html; charset=UTF-8" http-equiv="content-type" />
-	<title>civicrm-stripe-test-suite</title>
+	<title>new-test-suit</title>
 </head>
 <body>
 <table cellpadding="1" cellspacing="1" border="1">
@@ -101,7 +101,7 @@
 <tbody>
 <tr><td>open</td><td>http://localhost:8001/civicrm/contact/view?reset=1&amp;cid=109&amp;selectedChild=contribute<datalist><option>http://localhost:8001/civicrm/admin/paymentProcessor?reset=1</option><option>http://localhost:8001/civicrm/contact/view?reset=1&amp;cid=109&amp;selectedChild=contribute</option></datalist></td><td></td>
 </tr>
-<tr><td>click</td><td>//form[@id='Search']/div[2]/div[2]/a[2]/span<datalist><option>//form[@id='Search']/div[2]/div[2]/a[2]/span</option><option>//a[2]/span</option></datalist></td><td></td>
+<tr><td>click</td><td>xpath=(.//*[normalize-space(text()) and normalize-space(.)='Record Contribution (Check, Cash, EFT ...)'])[1]/following::span[1]<datalist><option>xpath=(.//*[normalize-space(text()) and normalize-space(.)='Record Contribution (Check, Cash, EFT ...)'])[1]/following::span[1]</option><option>xpath=(.//*[normalize-space(text()) and normalize-space(.)='Submit Credit Card Contribution'])[1]/following::span[2]</option><option>//div[@id='contributions-subtab']/div[2]/a[2]/span</option><option>//a[2]/span</option></datalist></td><td></td>
 </tr>
 <tr><td>click</td><td>id=financial_type_id<datalist><option>id=financial_type_id</option><option>name=financial_type_id</option><option>//select[@id='financial_type_id']</option><option>//form[@id='Contribution']/div[3]/table/tbody/tr[2]/td[2]/select</option><option>//select</option><option>css=#financial_type_id</option></datalist></td><td></td>
 </tr>
@@ -143,7 +143,7 @@
 <tbody>
 <tr><td>open</td><td>http://localhost:8001/civicrm/contact/view?reset=1&amp;cid=109&amp;selectedChild=contribute<datalist><option>http://localhost:8001/civicrm/contact/view?reset=1&amp;cid=109&amp;selectedChild=contribute</option><option>http://localhost:8001/civicrm/admin/paymentProcessor?reset=1</option></datalist></td><td></td>
 </tr>
-<tr><td>click</td><td>//form[@id='Search']/div[2]/div[2]/a[2]/span<datalist><option>//form[@id='Search']/div[2]/div[2]/a[2]/span</option><option>//a[2]/span</option></datalist></td><td></td>
+<tr><td>click</td><td>xpath=(.//*[normalize-space(text()) and normalize-space(.)='Record Contribution (Check, Cash, EFT ...)'])[1]/following::span[1]<datalist><option>//form[@id='Search']/div[2]/div[2]/a[2]/span</option><option>//a[2]/span</option><option>xpath=(.//*[normalize-space(text()) and normalize-space(.)='Record Contribution (Check, Cash, EFT ...)'])[1]/following::span[1]</option></datalist></td><td></td>
 </tr>
 <tr><td>click</td><td>id=financial_type_id<datalist><option>id=financial_type_id</option><option>name=financial_type_id</option><option>//select[@id='financial_type_id']</option><option>//form[@id='Contribution']/div[3]/table/tbody/tr[2]/td[2]/select</option><option>//select</option><option>css=#financial_type_id</option></datalist></td><td></td>
 </tr>
@@ -191,7 +191,7 @@
 <tbody>
 <tr><td>open</td><td>http://localhost:8001/civicrm/contact/view?reset=1&amp;cid=109&amp;selectedChild=member<datalist><option>http://localhost:8001/civicrm/contact/view?reset=1&amp;cid=109&amp;selectedChild=contribute</option><option>http://localhost:8001/civicrm/admin/paymentProcessor?reset=1</option><option>http://localhost:8001/civicrm/contact/view?reset=1&amp;cid=109&amp;selectedChild=member</option></datalist></td><td></td>
 </tr>
-<tr><td>click</td><td>//div[@id='ui-id-31']/div/div[2]/a[2]<datalist><option>//div[@id='ui-id-31']/div/div[2]/a[2]</option><option>xpath=(//a[contains(@href, '/civicrm/contact/view/membership?reset=1&amp;action=add&amp;cid=109&amp;context=membership&amp;mode=live')])[2]</option><option>//div[4]/div/div[2]/a[2]</option></datalist></td><td></td>
+<tr><td>click</td><td>xpath=(.//*[normalize-space(text()) and normalize-space(.)='Add Membership'])[3]/following::span[1]<datalist><option>xpath=(.//*[normalize-space(text()) and normalize-space(.)='Add Membership'])[3]/following::span[1]</option><option>xpath=(.//*[normalize-space(text()) and normalize-space(.)='Submit Credit Card Membership'])[1]/following::span[2]</option><option>xpath=(.//*[normalize-space(text()) and normalize-space(.)='New Activity'])[1]/preceding::span[1]</option><option>//div[@id='ui-id-27']/div/div[2]/a[2]/span</option><option>//div[4]/div/div[2]/a[2]/span</option></datalist></td><td></td>
 </tr>
 <tr><td>click</td><td>id=membership_type_id_1<datalist><option>id=membership_type_id_1</option><option>name=membership_type_id[1]</option><option>//select[@id='membership_type_id_1']</option><option>//span[@id='mem_type_id']/select[2]</option><option>//select[2]</option><option>css=#membership_type_id_1</option></datalist></td><td></td>
 </tr>
@@ -237,7 +237,7 @@
 <tbody>
 <tr><td>open</td><td>http://localhost:8001/civicrm/contact/view?reset=1&amp;cid=109&amp;selectedChild=member<datalist><option>http://localhost:8001/civicrm/contact/view?reset=1&amp;cid=109&amp;selectedChild=member</option><option>http://localhost:8001/civicrm/contact/view?reset=1&amp;cid=109&amp;selectedChild=contribute</option><option>http://localhost:8001/civicrm/admin/paymentProcessor?reset=1</option></datalist></td><td></td>
 </tr>
-<tr><td>click</td><td>//div[@id='ui-id-31']/div/div[2]/a[2]<datalist><option>//div[@id='ui-id-31']/div/div[2]/a[2]</option><option>xpath=(//a[contains(@href, '/civicrm/contact/view/membership?reset=1&amp;action=add&amp;cid=109&amp;context=membership&amp;mode=live')])[2]</option><option>//div[4]/div/div[2]/a[2]</option></datalist></td><td></td>
+<tr><td>click</td><td>xpath=(.//*[normalize-space(text()) and normalize-space(.)='Add Membership'])[3]/following::span[1]<datalist><option>xpath=(.//*[normalize-space(text()) and normalize-space(.)='Add Membership'])[3]/following::span[1]</option><option>xpath=(.//*[normalize-space(text()) and normalize-space(.)='Submit Credit Card Membership'])[1]/following::span[2]</option><option>xpath=(.//*[normalize-space(text()) and normalize-space(.)='Active Memberships'])[1]/preceding::span[1]</option><option>xpath=(.//*[normalize-space(text()) and normalize-space(.)='Membership'])[2]/preceding::span[1]</option><option>//div[@id='ui-id-27']/div/div[2]/a[2]/span</option><option>//div[4]/div/div[2]/a[2]/span</option></datalist></td><td></td>
 </tr>
 <tr><td>click</td><td>id=membership_type_id_1<datalist><option>id=membership_type_id_1</option><option>name=membership_type_id[1]</option><option>//select[@id='membership_type_id_1']</option><option>//span[@id='mem_type_id']/select[2]</option><option>//select[2]</option><option>css=#membership_type_id_1</option></datalist></td><td></td>
 </tr>
@@ -289,7 +289,7 @@
 <tbody>
 <tr><td>open</td><td>http://localhost:8001/civicrm/contact/view?reset=1&amp;cid=109&amp;selectedChild=participant<datalist><option>http://localhost:8001/civicrm/contact/view?reset=1&amp;cid=109&amp;selectedChild=participant</option><option>http://localhost:8001/civicrm/contact/view?reset=1&amp;cid=109&amp;selectedChild=member</option><option>http://localhost:8001/civicrm/contact/view?reset=1&amp;cid=109&amp;selectedChild=contribute</option><option>http://localhost:8001/civicrm/admin/paymentProcessor?reset=1</option></datalist></td><td></td>
 </tr>
-<tr><td>click</td><td>//form[@id='Search']/div[3]/a[2]<datalist><option>//form[@id='Search']/div[3]/a[2]</option><option>xpath=(//a[contains(@href, '/civicrm/contact/view/participant?reset=1&amp;action=add&amp;cid=109&amp;context=participant&amp;mode=live')])[2]</option><option>//div[3]/a[2]</option></datalist></td><td></td>
+<tr><td>click</td><td>xpath=(.//*[normalize-space(text()) and normalize-space(.)='Add Event Registration'])[2]/following::span[1]<datalist><option>xpath=(.//*[normalize-space(text()) and normalize-space(.)='Add Event Registration'])[2]/following::span[1]</option><option>xpath=(.//*[normalize-space(text()) and normalize-space(.)='Submit Credit Card Event Registration'])[1]/following::span[2]</option><option>//form[@id='Search']/div[3]/a[2]/span</option><option>//div[3]/a[2]/span</option></datalist></td><td></td>
 </tr>
 <tr><td>mouseDown</td><td>css=#s2id_event_id a.select2-choice<datalist><option>css=#s2id_event_id a.select2-choice</option><option>id=membership_type_id_1</option><option>name=membership_type_id[1]</option><option>//select[@id='membership_type_id_1']</option><option>//span[@id='mem_type_id']/select[2]</option><option>//select[2]</option><option>css=#membership_type_id_1</option></datalist></td><td></td>
 </tr>
@@ -327,7 +327,7 @@
 <tbody>
 <tr><td>open</td><td>http://localhost:8001/civicrm/contact/view?reset=1&amp;cid=109&amp;selectedChild=participant<datalist><option>http://localhost:8001/civicrm/contact/view?reset=1&amp;cid=109&amp;selectedChild=member</option><option>http://localhost:8001/civicrm/contact/view?reset=1&amp;cid=109&amp;selectedChild=contribute</option><option>http://localhost:8001/civicrm/admin/paymentProcessor?reset=1</option><option>http://localhost:8001/civicrm/contact/view?reset=1&amp;cid=109&amp;selectedChild=participant</option></datalist></td><td></td>
 </tr>
-<tr><td>click</td><td>//form[@id='Search']/div[3]/a[2]<datalist><option>//form[@id='Search']/div[3]/a[2]</option><option>xpath=(//a[contains(@href, '/civicrm/contact/view/participant?reset=1&amp;action=add&amp;cid=109&amp;context=participant&amp;mode=live')])[2]</option><option>//div[3]/a[2]</option></datalist></td><td></td>
+<tr><td>click</td><td>xpath=(.//*[normalize-space(text()) and normalize-space(.)='Add Event Registration'])[2]/following::span[1]<datalist><option>xpath=(.//*[normalize-space(text()) and normalize-space(.)='Add Event Registration'])[2]/following::span[1]</option><option>xpath=(.//*[normalize-space(text()) and normalize-space(.)='Submit Credit Card Event Registration'])[1]/following::span[2]</option><option>//form[@id='Search']/div[3]/a[2]/span</option><option>//div[3]/a[2]/span</option></datalist></td><td></td>
 </tr>
 <tr><td>mouseDown</td><td>css=#s2id_event_id a.select2-choice<datalist><option>css=#s2id_event_id a.select2-choice</option><option>id=membership_type_id_1</option><option>name=membership_type_id[1]</option><option>//select[@id='membership_type_id_1']</option><option>//span[@id='mem_type_id']/select[2]</option><option>//select[2]</option><option>css=#membership_type_id_1</option></datalist></td><td></td>
 </tr>
@@ -371,8 +371,6 @@
 <tbody>
 <tr><td>open</td><td>http://localhost:8001/civicrm/admin/contribute/amount?reset=1&amp;action=update&amp;id=1<datalist><option>http://localhost:8001/civicrm/admin/contribute?reset=1</option><option>http://localhost:8001/civicrm/admin/contribute/amount?reset=1&amp;action=update&amp;id=1</option></datalist></td><td></td>
 </tr>
-<tr><td>check</td><td>id=payment_processor_5<datalist><option>id=payment_processor_5</option><option>name=payment_processor[5]</option><option>//input[@id='payment_processor_5']</option><option>//form[@id='Amount']/div[2]/table/tbody/tr[3]/td/input</option><option>//tr[3]/td/input</option><option>css=#payment_processor_5</option></datalist></td><td></td>
-</tr>
 <tr><td>uncheck</td><td>id=payment_processor_1<datalist><option>id=payment_processor_1</option><option>name=payment_processor[1]</option><option>//input[@id='payment_processor_1']</option><option>//form[@id='Amount']/div[2]/table/tbody/tr[3]/td/input[3]</option><option>//td/input[3]</option><option>css=#payment_processor_1</option></datalist></td><td></td>
 </tr>
 <tr><td>uncheck</td><td>id=payment_processor_3<datalist><option>id=payment_processor_1</option><option>name=payment_processor[1]</option><option>//input[@id='payment_processor_1']</option><option>//form[@id='Amount']/div[2]/table/tbody/tr[3]/td/input[3]</option><option>//td/input[3]</option><option>css=#payment_processor_1</option><option>id=payment_processor_3</option></datalist></td><td></td>
@@ -389,8 +387,6 @@
 <tbody>
 <tr><td>open</td><td>http://localhost:8001/civicrm/contribute/transact?reset=1&amp;id=1<datalist><option>http://localhost:8001/civicrm/admin/contribute?reset=1</option><option>http://localhost:8001/civicrm/contribute/transact?reset=1&amp;id=1</option></datalist></td><td></td>
 </tr>
-<tr><td>selectWindow</td><td>win_ser_1<datalist><option>win_ser_1</option></datalist></td><td></td>
-</tr>
 <tr><td>click</td><td>id=credit_card_number<datalist><option>id=credit_card_number</option><option>name=credit_card_number</option><option>//input[@id='credit_card_number']</option><option>//div[@id='payment_information']/fieldset/div/div[2]/div[2]/input</option><option>//fieldset/div/div[2]/div[2]/input</option><option>css=#credit_card_number</option></datalist></td><td></td>
 </tr>
 <tr><td>type</td><td>id=credit_card_number<datalist><option>id=credit_card_number</option><option>name=credit_card_number</option><option>//input[@id='credit_card_number']</option><option>//div[@id='payment_information']/fieldset/div/div[2]/div[2]/input</option><option>//fieldset/div/div[2]/div[2]/input</option><option>css=#credit_card_number</option></datalist></td><td>4242424242424242</td>
@@ -515,71 +511,11 @@
 </tr>
 <tr><td>check</td><td>id=is_pay_later<datalist><option>id=is_pay_later</option><option>id=payment_processor_1</option><option>name=payment_processor[1]</option><option>//input[@id='payment_processor_1']</option><option>//form[@id='Amount']/div[2]/table/tbody/tr[3]/td/input[3]</option><option>//td/input[3]</option><option>css=#payment_processor_1</option></datalist></td><td></td>
 </tr>
-<tr><td>selectFrame</td><td>index=0<datalist><option>index=0</option></datalist></td><td></td>
-</tr>
-<tr><td>editContent</td><td>//body<datalist><option>//body</option><option>css=body.cke_editable.cke_editable_themed.cke_contents_ltr.cke_show_borders</option></datalist></td><td>&lt;p&gt;Do something&lt;/p&gt;</td>
-</tr>
-<tr><td>selectFrame</td><td>relative=parent<datalist><option>relative=parent</option></datalist></td><td></td>
-</tr>
 <tr><td>click</td><td>id=_qf_Amount_next-top<datalist><option>id=_qf_Amount_next-top</option><option>name=_qf_Amount_next</option><option>//input[@id='_qf_Amount_next-top']</option><option>//form[@id='Amount']/div[2]/div[2]/span/input</option><option>//span/input</option><option>css=#_qf_Amount_next-top</option></datalist></td><td></td>
 </tr>
 </tbody></table>
 <table cellpadding="1" cellspacing="1" border="1">
 <thead>
-<tr><td rowspan="1" colspan="3">Online Contribution Page Stripe Default, Pay Later</td></tr>
-</thead>
-<tbody>
-<tr><td>open</td><td>http://localhost:8001/civicrm/contribute/transact?reset=1&amp;id=1<datalist><option>http://localhost:8001/civicrm/contribute/transact?reset=1&amp;id=1</option><option>http://localhost:8001/civicrm/admin/contribute?reset=1</option></datalist></td><td></td>
-</tr>
-<tr><td>pause</td><td>2000<datalist><option>2000</option><option>10000</option></datalist></td><td></td>
-</tr>
-<tr><td>click</td><td>//form[@id='Main']/div[2]/fieldset/div/div[2]/label[2]<datalist><option>//form[@id='Main']/div[2]/fieldset/div/div[2]/label[2]</option><option>//fieldset/div/div[2]/label[2]</option></datalist></td><td></td>
-</tr>
-<tr><td>click</td><td>//form[@id='Main']/div[2]/fieldset/div/div[2]/label<datalist><option>//form[@id='Main']/div[2]/fieldset/div/div[2]/label</option><option>//fieldset/div/div[2]/label</option><option>css=div.crm-public-form-item.crm-section.payment_processor-section &gt; div.content &gt; label</option></datalist></td><td></td>
-</tr>
-<tr><td>pause</td><td>10000<datalist><option>10000</option></datalist></td><td></td>
-</tr>
-<tr><td>click</td><td>id=credit_card_number<datalist><option>id=credit_card_number</option><option>name=credit_card_number</option><option>//input[@id='credit_card_number']</option><option>//div[@id='payment_information']/fieldset/div/div[2]/div[2]/input</option><option>//fieldset/div/div[2]/div[2]/input</option><option>css=#credit_card_number</option></datalist></td><td></td>
-</tr>
-<tr><td>type</td><td>id=credit_card_number<datalist><option>id=credit_card_number</option><option>name=credit_card_number</option><option>//input[@id='credit_card_number']</option><option>//div[@id='payment_information']/fieldset/div/div[2]/div[2]/input</option><option>//fieldset/div/div[2]/div[2]/input</option><option>css=#credit_card_number</option></datalist></td><td>4242424242424242</td>
-</tr>
-<tr><td>click</td><td>id=cvv2<datalist><option>id=cvv2</option><option>name=cvv2</option><option>//input[@id='cvv2']</option><option>//div[@id='payment_information']/fieldset/div/div[3]/div[2]/input</option><option>//div/div[3]/div[2]/input</option><option>css=#cvv2</option></datalist></td><td></td>
-</tr>
-<tr><td>type</td><td>id=cvv2<datalist><option>id=cvv2</option><option>name=cvv2</option><option>//input[@id='cvv2']</option><option>//div[@id='payment_information']/fieldset/div/div[3]/div[2]/input</option><option>//div/div[3]/div[2]/input</option><option>css=#cvv2</option></datalist></td><td>123</td>
-</tr>
-<tr><td>click</td><td>id=credit_card_exp_date_M<datalist><option>id=credit_card_exp_date_M</option><option>name=credit_card_exp_date[M]</option><option>//select[@id='credit_card_exp_date_M']</option><option>//div[@id='payment_information']/fieldset/div/div[4]/div[2]/select</option><option>//fieldset/div/div[4]/div[2]/select</option><option>css=#credit_card_exp_date_M</option></datalist></td><td></td>
-</tr>
-<tr><td>select</td><td>id=credit_card_exp_date_M<datalist><option>id=credit_card_exp_date_M</option><option>name=credit_card_exp_date[M]</option><option>//select[@id='credit_card_exp_date_M']</option><option>//div[@id='payment_information']/fieldset/div/div[4]/div[2]/select</option><option>//fieldset/div/div[4]/div[2]/select</option><option>css=#credit_card_exp_date_M</option></datalist></td><td>label=Dec</td>
-</tr>
-<tr><td>click</td><td>//option[@value='12']<datalist><option>//option[@value='12']</option><option>//select[@id='credit_card_exp_date_M']/option[13]</option><option>//option[13]</option><option>css=option[value="12"]</option></datalist></td><td></td>
-</tr>
-<tr><td>click</td><td>id=credit_card_exp_date_Y<datalist><option>id=credit_card_exp_date_Y</option><option>name=credit_card_exp_date[Y]</option><option>//select[@id='credit_card_exp_date_Y']</option><option>//div[@id='payment_information']/fieldset/div/div[4]/div[2]/select[2]</option><option>//select[2]</option><option>css=#credit_card_exp_date_Y</option></datalist></td><td></td>
-</tr>
-<tr><td>select</td><td>id=credit_card_exp_date_Y<datalist><option>id=credit_card_exp_date_Y</option><option>name=credit_card_exp_date[Y]</option><option>//select[@id='credit_card_exp_date_Y']</option><option>//div[@id='payment_information']/fieldset/div/div[4]/div[2]/select[2]</option><option>//select[2]</option><option>css=#credit_card_exp_date_Y</option></datalist></td><td>label=2028</td>
-</tr>
-<tr><td>click</td><td>//option[@value='2028']<datalist><option>//option[@value='2028']</option><option>//select[@id='credit_card_exp_date_Y']/option[12]</option><option>//select[2]/option[12]</option><option>css=option[value="2028"]</option></datalist></td><td></td>
-</tr>
-<tr><td>click</td><td>id=billing_first_name<datalist><option>id=billing_first_name</option><option>name=billing_first_name</option><option>//input[@id='billing_first_name']</option><option>//div[@id='payment_information']/fieldset[2]/div/div/div[2]/input</option><option>//fieldset[2]/div/div/div[2]/input</option><option>css=#billing_first_name</option></datalist></td><td></td>
-</tr>
-<tr><td>type</td><td>id=billing_first_name<datalist><option>id=billing_first_name</option><option>name=billing_first_name</option><option>//input[@id='billing_first_name']</option><option>//div[@id='payment_information']/fieldset[2]/div/div/div[2]/input</option><option>//fieldset[2]/div/div/div[2]/input</option><option>css=#billing_first_name</option></datalist></td><td>Joe</td>
-</tr>
-<tr><td>type</td><td>id=billing_last_name<datalist><option>id=billing_last_name</option><option>name=billing_last_name</option><option>//input[@id='billing_last_name']</option><option>//div[@id='payment_information']/fieldset[2]/div/div[3]/div[2]/input</option><option>//fieldset[2]/div/div[3]/div[2]/input</option><option>css=#billing_last_name</option></datalist></td><td>Joe</td>
-</tr>
-<tr><td>type</td><td>id=billing_street_address-5<datalist><option>id=billing_street_address-5</option><option>name=billing_street_address-5</option><option>//input[@id='billing_street_address-5']</option><option>//div[@id='payment_information']/fieldset[2]/div/div[4]/div[2]/input</option><option>//fieldset[2]/div/div[4]/div[2]/input</option><option>css=#billing_street_address-5</option></datalist></td><td>123</td>
-</tr>
-<tr><td>type</td><td>id=billing_city-5<datalist><option>id=billing_city-5</option><option>name=billing_city-5</option><option>//input[@id='billing_city-5']</option><option>//div[@id='payment_information']/fieldset[2]/div/div[5]/div[2]/input</option><option>//fieldset[2]/div/div[5]/div[2]/input</option><option>css=#billing_city-5</option></datalist></td><td>Joe</td>
-</tr>
-<tr><td>click</td><td>id=billing_postal_code-5<datalist><option>id=billing_postal_code-5</option><option>name=billing_postal_code-5</option><option>//input[@id='billing_postal_code-5']</option><option>//div[@id='payment_information']/fieldset[2]/div/div[8]/div[2]/input</option><option>//div[8]/div[2]/input</option><option>css=#billing_postal_code-5</option></datalist></td><td></td>
-</tr>
-<tr><td>type</td><td>id=billing_postal_code-5<datalist><option>id=billing_postal_code-5</option><option>name=billing_postal_code-5</option><option>//input[@id='billing_postal_code-5']</option><option>//div[@id='payment_information']/fieldset[2]/div/div[8]/div[2]/input</option><option>//div[8]/div[2]/input</option><option>css=#billing_postal_code-5</option></datalist></td><td>12345</td>
-</tr>
-<tr><td>clickAndWait</td><td>id=_qf_Main_upload-bottom<datalist><option>id=_qf_Main_upload-bottom</option><option>name=_qf_Main_upload</option><option>//input[@id='_qf_Main_upload-bottom']</option><option>//div[@id='crm-submit-buttons']/span/input</option><option>//div[12]/span/input</option><option>css=#_qf_Main_upload-bottom</option></datalist></td><td></td>
-</tr>
-<tr><td>click</td><td>id=_qf_Confirm_next-top<datalist><option>id=_qf_Confirm_next-top</option><option>name=_qf_Confirm_next</option><option>//input[@id='_qf_Confirm_next-top']</option><option>//div[@id='crm-submit-buttons']/span/input</option><option>//span/input</option><option>css=#_qf_Confirm_next-top</option></datalist></td><td></td>
-</tr>
-</tbody></table>
-<table cellpadding="1" cellspacing="1" border="1">
-<thead>
 <tr><td rowspan="1" colspan="3">Setup Contribution Page Stripe Default, no confirmation</td></tr>
 </thead>
 <tbody>
@@ -597,9 +533,11 @@
 </tr>
 <tr><td>click</td><td>id=ui-id-25<datalist><option>id=ui-id-25</option><option>link=Title</option><option>//a[contains(text(),'Title')]</option><option>//a[@id='ui-id-25']</option><option>//li[@id='tab_settings']/a</option><option>//a[contains(@href, '/civicrm/admin/contribute/settings?reset=1&amp;action=update&amp;id=1')]</option><option>//div[3]/div[2]/div/ul/li/a</option><option>css=#ui-id-25</option></datalist></td><td></td>
 </tr>
-<tr><td>uncheck</td><td>id=is_confirm_enabled<datalist><option>id=is_confirm_enabled</option><option>name=is_confirm_enabled</option><option>//input[@id='is_confirm_enabled']</option><option>//form[@id='Settings']/div[3]/table[3]/tbody/tr/td[2]/input</option><option>//table[3]/tbody/tr/td[2]/input</option><option>css=#is_confirm_enabled</option></datalist></td><td></td>
+<tr><td>open</td><td>http://localhost:8001/civicrm/admin/contribute/settings?reset=1&amp;action=update&amp;id=1<datalist><option>http://localhost:8001/civicrm/admin/contribute/settings?reset=1&amp;action=update&amp;id=1</option></datalist></td><td></td>
 </tr>
-<tr><td>click</td><td>id=_qf_Settings_next-bottom<datalist><option>id=_qf_Settings_next-bottom</option><option>//input[@id='_qf_Settings_next-bottom']</option><option>//form[@id='Settings']/div[3]/div[2]/span/input</option><option>//div[2]/span/input</option><option>css=#_qf_Settings_next-bottom</option></datalist></td><td></td>
+<tr><td>uncheck</td><td>id=is_confirm_enabled<datalist><option>id=is_confirm_enabled</option><option>name=is_confirm_enabled</option><option>xpath=(.//*[normalize-space(text()) and normalize-space(.)='Profile to be included in the honoree section'])[1]/following::input[1]</option><option>xpath=(.//*[normalize-space(text()) and normalize-space(.)='Email Address'])[1]/following::input[2]</option><option>xpath=(.//*[normalize-space(text()) and normalize-space(.)='Use a confirmation page?'])[1]/preceding::input[1]</option><option>xpath=(.//*[normalize-space(text()) and normalize-space(.)='Allow sharing through social media?'])[1]/preceding::input[2]</option><option>//input[@id='is_confirm_enabled']</option><option>//form[@id='Settings']/div[3]/table[3]/tbody/tr/td[2]/input</option><option>//table[3]/tbody/tr/td[2]/input</option><option>css=#is_confirm_enabled</option></datalist></td><td></td>
+</tr>
+<tr><td>click</td><td>id=_qf_Settings_next-top<datalist><option>id=_qf_Settings_next-top</option><option>name=_qf_Settings_next</option><option>xpath=(.//*[normalize-space(text()) and normalize-space(.)='Personal Campaigns'])[1]/following::input[6]</option><option>xpath=(.//*[normalize-space(text()) and normalize-space(.)='Widgets'])[1]/following::input[6]</option><option>xpath=(.//*[normalize-space(text()) and normalize-space(.)='*'])[1]/preceding::input[4]</option><option>//input[@id='_qf_Settings_next-top']</option><option>//form[@id='Settings']/div[3]/div/span/input</option><option>//span/input</option><option>css=#_qf_Settings_next-top</option></datalist></td><td></td>
 </tr>
 </tbody></table>
 <table cellpadding="1" cellspacing="1" border="1">
@@ -678,6 +616,56 @@
 </tbody></table>
 <table cellpadding="1" cellspacing="1" border="1">
 <thead>
+<tr><td rowspan="1" colspan="3">Online Contribution Page Stripe Default, Pay Later</td></tr>
+</thead>
+<tbody>
+<tr><td>open</td><td>http://localhost:8001/civicrm/contribute/transact?reset=1&amp;id=1<datalist><option>http://localhost:8001/civicrm/contribute/transact?reset=1&amp;id=1</option><option>http://localhost:8001/civicrm/admin/contribute?reset=1</option></datalist></td><td></td>
+</tr>
+<tr><td>pause</td><td>2000<datalist><option>2000</option><option>10000</option></datalist></td><td></td>
+</tr>
+<tr><td>pause</td><td>10000<datalist><option>10000</option></datalist></td><td></td>
+</tr>
+<tr><td>click</td><td>id=credit_card_number<datalist><option>id=credit_card_number</option><option>name=credit_card_number</option><option>//input[@id='credit_card_number']</option><option>//div[@id='payment_information']/fieldset/div/div[2]/div[2]/input</option><option>//fieldset/div/div[2]/div[2]/input</option><option>css=#credit_card_number</option></datalist></td><td></td>
+</tr>
+<tr><td>type</td><td>id=credit_card_number<datalist><option>id=credit_card_number</option><option>name=credit_card_number</option><option>//input[@id='credit_card_number']</option><option>//div[@id='payment_information']/fieldset/div/div[2]/div[2]/input</option><option>//fieldset/div/div[2]/div[2]/input</option><option>css=#credit_card_number</option></datalist></td><td>4242424242424242</td>
+</tr>
+<tr><td>click</td><td>id=cvv2<datalist><option>id=cvv2</option><option>name=cvv2</option><option>//input[@id='cvv2']</option><option>//div[@id='payment_information']/fieldset/div/div[3]/div[2]/input</option><option>//div/div[3]/div[2]/input</option><option>css=#cvv2</option></datalist></td><td></td>
+</tr>
+<tr><td>type</td><td>id=cvv2<datalist><option>id=cvv2</option><option>name=cvv2</option><option>//input[@id='cvv2']</option><option>//div[@id='payment_information']/fieldset/div/div[3]/div[2]/input</option><option>//div/div[3]/div[2]/input</option><option>css=#cvv2</option></datalist></td><td>123</td>
+</tr>
+<tr><td>click</td><td>id=credit_card_exp_date_M<datalist><option>id=credit_card_exp_date_M</option><option>name=credit_card_exp_date[M]</option><option>//select[@id='credit_card_exp_date_M']</option><option>//div[@id='payment_information']/fieldset/div/div[4]/div[2]/select</option><option>//fieldset/div/div[4]/div[2]/select</option><option>css=#credit_card_exp_date_M</option></datalist></td><td></td>
+</tr>
+<tr><td>select</td><td>id=credit_card_exp_date_M<datalist><option>id=credit_card_exp_date_M</option><option>name=credit_card_exp_date[M]</option><option>//select[@id='credit_card_exp_date_M']</option><option>//div[@id='payment_information']/fieldset/div/div[4]/div[2]/select</option><option>//fieldset/div/div[4]/div[2]/select</option><option>css=#credit_card_exp_date_M</option></datalist></td><td>label=Dec</td>
+</tr>
+<tr><td>click</td><td>//option[@value='12']<datalist><option>//option[@value='12']</option><option>//select[@id='credit_card_exp_date_M']/option[13]</option><option>//option[13]</option><option>css=option[value="12"]</option></datalist></td><td></td>
+</tr>
+<tr><td>click</td><td>id=credit_card_exp_date_Y<datalist><option>id=credit_card_exp_date_Y</option><option>name=credit_card_exp_date[Y]</option><option>//select[@id='credit_card_exp_date_Y']</option><option>//div[@id='payment_information']/fieldset/div/div[4]/div[2]/select[2]</option><option>//select[2]</option><option>css=#credit_card_exp_date_Y</option></datalist></td><td></td>
+</tr>
+<tr><td>select</td><td>id=credit_card_exp_date_Y<datalist><option>id=credit_card_exp_date_Y</option><option>name=credit_card_exp_date[Y]</option><option>//select[@id='credit_card_exp_date_Y']</option><option>//div[@id='payment_information']/fieldset/div/div[4]/div[2]/select[2]</option><option>//select[2]</option><option>css=#credit_card_exp_date_Y</option></datalist></td><td>label=2028</td>
+</tr>
+<tr><td>click</td><td>//option[@value='2028']<datalist><option>//option[@value='2028']</option><option>//select[@id='credit_card_exp_date_Y']/option[12]</option><option>//select[2]/option[12]</option><option>css=option[value="2028"]</option></datalist></td><td></td>
+</tr>
+<tr><td>click</td><td>id=billing_first_name<datalist><option>id=billing_first_name</option><option>name=billing_first_name</option><option>//input[@id='billing_first_name']</option><option>//div[@id='payment_information']/fieldset[2]/div/div/div[2]/input</option><option>//fieldset[2]/div/div/div[2]/input</option><option>css=#billing_first_name</option></datalist></td><td></td>
+</tr>
+<tr><td>type</td><td>id=billing_first_name<datalist><option>id=billing_first_name</option><option>name=billing_first_name</option><option>//input[@id='billing_first_name']</option><option>//div[@id='payment_information']/fieldset[2]/div/div/div[2]/input</option><option>//fieldset[2]/div/div/div[2]/input</option><option>css=#billing_first_name</option></datalist></td><td>Joe</td>
+</tr>
+<tr><td>type</td><td>id=billing_last_name<datalist><option>id=billing_last_name</option><option>name=billing_last_name</option><option>//input[@id='billing_last_name']</option><option>//div[@id='payment_information']/fieldset[2]/div/div[3]/div[2]/input</option><option>//fieldset[2]/div/div[3]/div[2]/input</option><option>css=#billing_last_name</option></datalist></td><td>Joe</td>
+</tr>
+<tr><td>type</td><td>id=billing_street_address-5<datalist><option>id=billing_street_address-5</option><option>name=billing_street_address-5</option><option>//input[@id='billing_street_address-5']</option><option>//div[@id='payment_information']/fieldset[2]/div/div[4]/div[2]/input</option><option>//fieldset[2]/div/div[4]/div[2]/input</option><option>css=#billing_street_address-5</option></datalist></td><td>123</td>
+</tr>
+<tr><td>type</td><td>id=billing_city-5<datalist><option>id=billing_city-5</option><option>name=billing_city-5</option><option>//input[@id='billing_city-5']</option><option>//div[@id='payment_information']/fieldset[2]/div/div[5]/div[2]/input</option><option>//fieldset[2]/div/div[5]/div[2]/input</option><option>css=#billing_city-5</option></datalist></td><td>Joe</td>
+</tr>
+<tr><td>click</td><td>id=billing_postal_code-5<datalist><option>id=billing_postal_code-5</option><option>name=billing_postal_code-5</option><option>//input[@id='billing_postal_code-5']</option><option>//div[@id='payment_information']/fieldset[2]/div/div[8]/div[2]/input</option><option>//div[8]/div[2]/input</option><option>css=#billing_postal_code-5</option></datalist></td><td></td>
+</tr>
+<tr><td>type</td><td>id=billing_postal_code-5<datalist><option>id=billing_postal_code-5</option><option>name=billing_postal_code-5</option><option>//input[@id='billing_postal_code-5']</option><option>//div[@id='payment_information']/fieldset[2]/div/div[8]/div[2]/input</option><option>//div[8]/div[2]/input</option><option>css=#billing_postal_code-5</option></datalist></td><td>12345</td>
+</tr>
+<tr><td>clickAndWait</td><td>id=_qf_Main_upload-bottom<datalist><option>id=_qf_Main_upload-bottom</option><option>name=_qf_Main_upload</option><option>//input[@id='_qf_Main_upload-bottom']</option><option>//div[@id='crm-submit-buttons']/span/input</option><option>//div[12]/span/input</option><option>css=#_qf_Main_upload-bottom</option></datalist></td><td></td>
+</tr>
+<tr><td>click</td><td>id=_qf_Confirm_next-top<datalist><option>id=_qf_Confirm_next-top</option><option>name=_qf_Confirm_next</option><option>//input[@id='_qf_Confirm_next-top']</option><option>//div[@id='crm-submit-buttons']/span/input</option><option>//span/input</option><option>css=#_qf_Confirm_next-top</option></datalist></td><td></td>
+</tr>
+</tbody></table>
+<table cellpadding="1" cellspacing="1" border="1">
+<thead>
 <tr><td rowspan="1" colspan="3">Test Webform</td></tr>
 </thead>
 <tbody>
@@ -717,4 +705,4 @@
 </tr>
 </tbody></table>
 </body>
-</html>
\ No newline at end of file
+</html>
diff --git a/tests/phpunit/CRM/Stripe/BaseTest.php b/tests/phpunit/CRM/Stripe/BaseTest.php
index 47878b0ac0b04366033b7769bccdc584ea2920f1..e26ebb93bd9f5e0b893b96f11453b8c3b202a202 100644
--- a/tests/phpunit/CRM/Stripe/BaseTest.php
+++ b/tests/phpunit/CRM/Stripe/BaseTest.php
@@ -157,6 +157,7 @@ class CRM_Stripe_BaseTest extends \PHPUnit_Framework_TestCase implements Headles
         'address_zip' => '12345',
       ),
       'email' => $this->contact->email,
+      'contactID' => $this->contact->id,
       'description' => 'Test from Stripe Test Code',
       'currencyID' => 'USD',
       'invoiceID' => $this->_invoiceID,
diff --git a/tests/phpunit/CRM/Stripe/IpnTest.php b/tests/phpunit/CRM/Stripe/IpnTest.php
index a4306ee4a01c71ad7c5fb8735561b52810196ed3..884ed05af8cdd26808015f99590d572f103a71e3 100644
--- a/tests/phpunit/CRM/Stripe/IpnTest.php
+++ b/tests/phpunit/CRM/Stripe/IpnTest.php
@@ -90,11 +90,16 @@ class CRM_Stripe_IpnTest extends CRM_Stripe_BaseTest {
     }
     catch (Stripe\Error\InvalidRequest $e) {
       // The plan has not been created yet, so create it.
+      $product = \Stripe\Product::create(array(
+        "name" => "CiviCRM testing product",
+        "type" => "service"
+      ));
+
       $plan_details = array(
         'id' => $plan_id,
         'amount' => '40000',
         'interval' => 'month',
-        'name' => "Test Updated Plan",
+        'product' => $product,
         'currency' => 'usd',
         'interval_count' => 2
       );
@@ -249,16 +254,11 @@ class CRM_Stripe_IpnTest extends CRM_Stripe_BaseTest {
    *
    */
   public function ipn($data, $verify = TRUE) {
-    if (!class_exists('CRM_Core_Payment_StripeIPN')) {
-      // The $_GET['processor_id'] value is normally set by 
-      // CRM_Core_Payment::handlePaymentMethod
-      $_GET['processor_id'] = $this->_paymentProcessorID;
-      $ipnClass = new CRM_Core_Payment_StripeIPN($data, $verify);
-      $ipnClass->main();
-    }
-    else {
-      trigger_error("Test suite depends on CRM_Core_Payment_StripeIPN");
-    }
+    // The $_GET['processor_id'] value is normally set by 
+    // CRM_Core_Payment::handlePaymentMethod
+    $_GET['processor_id'] = $this->_paymentProcessorID;
+    $ipnClass = new CRM_Core_Payment_StripeIPN($data, $verify);
+    $ipnClass->main();
   }
 
   /**
diff --git a/xml/Menu/stripe.xml b/xml/Menu/stripe.xml
deleted file mode 100644
index 1214117eb956c85a0dc14cab52b92af0db2f5715..0000000000000000000000000000000000000000
--- a/xml/Menu/stripe.xml
+++ /dev/null
@@ -1,9 +0,0 @@
-<?xml version="1.0"?>
-<menu>
-  <item>
-    <path>civicrm/stripe/webhook</path>
-    <page_callback>CRM_Stripe_Page_Webhook</page_callback>
-    <title>Webhook</title>
-    <access_arguments>make online contributions</access_arguments>
-  </item>
-</menu>