import re import stripe import stripe.error import logging from config import etcd_client as client, config as config stripe.api_key = config.get('stripe', 'private_key') def handle_stripe_error(f): def handle_problems(*args, **kwargs): response = { 'paid': False, 'response_object': None, 'error': None } common_message = "Currently it's not possible to make payments." try: response_object = f(*args, **kwargs) response = { 'response_object': response_object, 'error': None } return response except stripe.error.CardError as e: # Since it's a decline, stripe.error.CardError will be caught body = e.json_body err = body['error'] response.update({'error': err['message']}) logging.error(str(e)) return response except stripe.error.RateLimitError: response.update( {'error': "Too many requests made to the API too quickly"}) return response except stripe.error.InvalidRequestError as e: logging.error(str(e)) response.update({'error': "Invalid parameters"}) return response except stripe.error.AuthenticationError as e: # Authentication with Stripe's API failed # (maybe you changed API keys recently) logging.error(str(e)) response.update({'error': common_message}) return response except stripe.error.APIConnectionError as e: logging.error(str(e)) response.update({'error': common_message}) return response except stripe.error.StripeError as e: # maybe send email logging.error(str(e)) response.update({'error': common_message}) return response except Exception as e: # maybe send email logging.error(str(e)) response.update({'error': common_message}) return response return handle_problems class StripeUtils(object): CURRENCY = 'chf' INTERVAL = 'month' SUCCEEDED_STATUS = 'succeeded' STRIPE_PLAN_ALREADY_EXISTS = 'Plan already exists' STRIPE_NO_SUCH_PLAN = 'No such plan' PLAN_EXISTS_ERROR_MSG = 'Plan {} exists already.\nCreating a local StripePlan now.' PLAN_DOES_NOT_EXIST_ERROR_MSG = 'Plan {} does not exist.' def __init__(self, private_key): self.stripe = stripe stripe.api_key = private_key @handle_stripe_error def card_exists(self, customer, cc_number, exp_month, exp_year, cvc): token_obj = stripe.Token.create( card={ 'number': cc_number, 'exp_month': exp_month, 'exp_year': exp_year, 'cvc': cvc, }, ) cards = stripe.Customer.list_sources( customer, limit=20, object='card' ) for card in cards.data: if (card.fingerprint == token_obj.card.fingerprint and int(card.exp_month) == int(exp_month) and int(card.exp_year) == int(exp_year)): return True return False @staticmethod def get_stripe_customer_from_email(email): customer = stripe.Customer.list(limit=1, email=email) return customer.data[0] if len(customer.data) == 1 else None @staticmethod def update_customer_token(customer, token): customer.source = token customer.save() @handle_stripe_error def get_token_from_card(self, cc_number, cvc, expiry_month, expiry_year): token_obj = stripe.Token.create( card={ 'number': cc_number, 'exp_month': expiry_month, 'exp_year': expiry_year, 'cvc': cvc, }, ) return token_obj @handle_stripe_error def associate_customer_card(self, stripe_customer_id, token, set_as_default=False): customer = stripe.Customer.retrieve(stripe_customer_id) card = customer.sources.create(source=token) if set_as_default: customer.default_source = card.id customer.save() return True @handle_stripe_error def dissociate_customer_card(self, stripe_customer_id, card_id): customer = stripe.Customer.retrieve(stripe_customer_id) card = customer.sources.retrieve(card_id) card.delete() @handle_stripe_error def update_customer_card(self, customer_id, token): customer = stripe.Customer.retrieve(customer_id) current_card_token = customer.default_source customer.sources.retrieve(current_card_token).delete() customer.source = token customer.save() credit_card_raw_data = customer.sources.data.pop() new_card_data = { 'last4': credit_card_raw_data.last4, 'brand': credit_card_raw_data.brand } return new_card_data @handle_stripe_error def get_card_details(self, customer_id): customer = stripe.Customer.retrieve(customer_id) credit_card_raw_data = customer.sources.data.pop() card_details = { 'last4': credit_card_raw_data.last4, 'brand': credit_card_raw_data.brand, 'exp_month': credit_card_raw_data.exp_month, 'exp_year': credit_card_raw_data.exp_year, 'fingerprint': credit_card_raw_data.fingerprint, 'card_id': credit_card_raw_data.id } return card_details @handle_stripe_error def get_all_invoices(self, customer_id, created_gt): return_list = [] has_more_invoices = True starting_after = False while has_more_invoices: if starting_after: invoices = stripe.Invoice.list( limit=10, customer=customer_id, created={'gt': created_gt}, starting_after=starting_after ) else: invoices = stripe.Invoice.list( limit=10, customer=customer_id, created={'gt': created_gt} ) has_more_invoices = invoices.has_more for invoice in invoices.data: sub_ids = [] for line in invoice.lines.data: if line.type == 'subscription': sub_ids.append(line.id) elif line.type == 'invoiceitem': sub_ids.append(line.subscription) else: sub_ids.append('') invoice_details = { 'created': invoice.created, 'receipt_number': invoice.receipt_number, 'invoice_number': invoice.number, 'paid_at': invoice.status_transitions.paid_at if invoice.paid else 0, 'period_start': invoice.period_start, 'period_end': invoice.period_end, 'billing_reason': invoice.billing_reason, 'discount': invoice.discount.coupon.amount_off if invoice.discount else 0, 'total': invoice.total, # to see how many line items we have in this invoice and # then later check if we have more than 1 'lines_data_count': len(invoice.lines.data) if invoice.lines.data is not None else 0, 'invoice_id': invoice.id, 'lines_meta_data_csv': ','.join( [line.metadata.VM_ID if hasattr(line.metadata, 'VM_ID') else '' for line in invoice.lines.data] ), 'subscription_ids_csv': ','.join(sub_ids), 'line_items': invoice.lines.data } starting_after = invoice.id return_list.append(invoice_details) return return_list @handle_stripe_error def get_cards_details_from_token(self, token): stripe_token = stripe.Token.retrieve(token) card_details = { 'last4': stripe_token.card.last4, 'brand': stripe_token.card.brand, 'exp_month': stripe_token.card.exp_month, 'exp_year': stripe_token.card.exp_year, 'fingerprint': stripe_token.card.fingerprint, 'card_id': stripe_token.card.id } return card_details def check_customer(self, stripe_cus_api_id, user, token): try: customer = stripe.Customer.retrieve(stripe_cus_api_id) except stripe.error.InvalidRequestError: customer = self.create_customer(token, user.email, user.name) user.stripecustomer.stripe_id = customer.get( 'response_object').get('id') user.stripecustomer.save() if type(customer) is dict: customer = customer['response_object'] return customer @handle_stripe_error def get_customer(self, stripe_api_cus_id): customer = stripe.Customer.retrieve(stripe_api_cus_id) # data = customer.get('response_object') return customer @handle_stripe_error def create_customer(self, token, email, name=None, address=None): if name is None or name.strip() == "": name = email customer = self.stripe.Customer.create( source=token, description=name, email=email, address=address ) return customer @handle_stripe_error def make_charge(self, amount=None, customer=None): _amount = float(amount) amount = int(_amount * 100) # stripe amount unit, in cents charge = self.stripe.Charge.create( amount=amount, # in cents currency=self.CURRENCY, customer=customer ) return charge @staticmethod def _get_all_stripe_plans(): all_stripe_plans = client.get("/v1/stripe_plans") all_stripe_plans_set = set() if all_stripe_plans: all_stripe_plans_obj = all_stripe_plans.value if all_stripe_plans_obj and len(all_stripe_plans_obj['plans']) > 0: all_stripe_plans_set = set(all_stripe_plans_obj["plans"]) return all_stripe_plans_set @staticmethod def _save_all_stripe_plans(stripe_plans): client.put("/v1/stripe_plans", {"plans": list(stripe_plans)}) @handle_stripe_error def get_or_create_stripe_plan(self, product_name, amount, stripe_plan_id, interval=INTERVAL): """ This function checks if a StripePlan with the given stripe_plan_id already exists. If it exists then the function returns this object otherwise it creates a new StripePlan and returns the new object. :param amount: The amount in CHF cents :param product_name: The name of the Stripe plan (product) to be created. :param stripe_plan_id: The id of the Stripe plan to be created. Use get_stripe_plan_id_string function to obtain the name of the plan to be created :param interval: The interval for subscription {month, year}. Defaults to month if not provided :return: The StripePlan object if it exists else creates a Plan object in Stripe and a local StripePlan and returns it. Returns None in case of Stripe error """ _amount = float(amount) amount = int(_amount * 100) # stripe amount unit, in cents all_stripe_plans = self._get_all_stripe_plans() if stripe_plan_id in all_stripe_plans: logging.debug("{} plan exists in db.".format(stripe_plan_id)) else: logging.debug(("{} plan DOES NOT exist in db. " "Creating").format(stripe_plan_id)) try: plan_obj = self.stripe.Plan.retrieve(id=stripe_plan_id) logging.debug("{} plan exists in Stripe".format(stripe_plan_id)) all_stripe_plans.add(stripe_plan_id) except stripe.error.InvalidRequestError as e: if "No such plan" in str(e): logging.debug("Plan {} does not exist in Stripe, Creating") plan_obj = self.stripe.Plan.create( amount=amount, product={'name': product_name}, interval=interval, currency=self.CURRENCY, id=stripe_plan_id) logging.debug(self.PLAN_EXISTS_ERROR_MSG.format(stripe_plan_id)) all_stripe_plans.add(stripe_plan_id) self._save_all_stripe_plans(all_stripe_plans) return stripe_plan_id @handle_stripe_error def delete_stripe_plan(self, stripe_plan_id): """ Deletes the Plan in Stripe and also deletes the local db copy of the plan if it exists :param stripe_plan_id: The stripe plan id that needs to be deleted :return: True if the plan was deleted successfully from Stripe, False otherwise. """ return_value = False try: plan = self.stripe.Plan.retrieve(stripe_plan_id) plan.delete() return_value = True all_stripe_plans = self._get_all_stripe_plans() all_stripe_plans.remove(stripe_plan_id) self._save_all_stripe_plans(all_stripe_plans) except stripe.error.InvalidRequestError as e: if self.STRIPE_NO_SUCH_PLAN in str(e): logging.debug( self.PLAN_DOES_NOT_EXIST_ERROR_MSG.format(stripe_plan_id)) return return_value @handle_stripe_error def subscribe_customer_to_plan(self, customer, plans, trial_end=None): """ Subscribes the given customer to the list of given plans :param customer: The stripe customer identifier :param plans: A list of stripe plans. :param trial_end: An integer representing when the Stripe subscription is supposed to end Ref: https://stripe.com/docs/api/python#create_subscription-items e.g. plans = [ { "plan": "dcl-v1-cpu-2-ram-5gb-ssd-10gb", }, ] :return: The subscription StripeObject """ subscription_result = self.stripe.Subscription.create( customer=customer, items=plans, trial_end=trial_end ) return subscription_result @handle_stripe_error def set_subscription_metadata(self, subscription_id, metadata): subscription = stripe.Subscription.retrieve(subscription_id) subscription.metadata = metadata subscription.save() @handle_stripe_error def unsubscribe_customer(self, subscription_id): """ Cancels a given subscription :param subscription_id: The Stripe subscription id string :return: """ sub = stripe.Subscription.retrieve(subscription_id) return sub.delete() @handle_stripe_error def make_payment(self, customer, amount, token): charge = self.stripe.Charge.create( amount=amount, # in cents currency=self.CURRENCY, customer=customer ) return charge @staticmethod def get_stripe_plan_id(cpu, ram, ssd, version, app='dcl', hdd=None, price=None): """ Returns the Stripe plan id string of the form `dcl-v1-cpu-2-ram-5gb-ssd-10gb` based on the input parameters :param cpu: The number of cores :param ram: The size of the RAM in GB :param ssd: The size of ssd storage in GB :param hdd: The size of hdd storage in GB :param version: The version of the Stripe plans :param app: The application to which the stripe plan belongs to. By default it is 'dcl' :param price: The price for this plan :return: A string of the form `dcl-v1-cpu-2-ram-5gb-ssd-10gb` """ dcl_plan_string = 'cpu-{cpu}-ram-{ram}gb-ssd-{ssd}gb'.format(cpu=cpu, ram=ram, ssd=ssd) if hdd is not None: dcl_plan_string = '{dcl_plan_string}-hdd-{hdd}gb'.format( dcl_plan_string=dcl_plan_string, hdd=hdd) stripe_plan_id_string = '{app}-v{version}-{plan}'.format( app=app, version=version, plan=dcl_plan_string ) if price is not None: stripe_plan_id_string_with_price = '{}-{}chf'.format( stripe_plan_id_string, round(price, 2) ) return stripe_plan_id_string_with_price else: return stripe_plan_id_string @staticmethod def get_vm_config_from_stripe_id(stripe_id): """ Given a string like "dcl-v1-cpu-2-ram-5gb-ssd-10gb" return different configuration params as a dict :param stripe_id|str :return: dict """ pattern = re.compile(r'^dcl-v(\d+)-cpu-(\d+)-ram-(\d+\.?\d*)gb-ssd-(\d+)gb-?(\d*\.?\d*)(chf)?$') match_res = pattern.match(stripe_id) if match_res is not None: price = None try: price = match_res.group(5) except IndexError: logging.debug("Did not find price in {}".format(stripe_id)) return { 'version': match_res.group(1), 'cores': match_res.group(2), 'ram': match_res.group(3), 'ssd': match_res.group(4), 'price': price } @staticmethod def get_stripe_plan_name(cpu, memory, disk_size, price): """ Returns the Stripe plan name :return: """ return "{cpu} Cores, {memory} GB RAM, {disk_size} GB SSD, " \ "{price} CHF".format( cpu=cpu, memory=memory, disk_size=disk_size, price=round(price, 2) ) @handle_stripe_error def set_subscription_meta_data(self, subscription_id, meta_data): """ Adds VM metadata to a subscription :param subscription_id: Stripe identifier for the subscription :param meta_data: A dict of meta data to be added :return: """ subscription = stripe.Subscription.retrieve(subscription_id) subscription.metadata = meta_data subscription.save()