import logging import stripe from django.conf import settings from datacenterlight.models import StripePlan stripe.api_key = settings.STRIPE_API_PRIVATE_KEY logger = logging.getLogger(__name__) def handleStripeError(f): def handleProblems(*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']}) logger.error(str(e)) return response except stripe.error.RateLimitError as e: response.update( {'error': "Too many requests made to the API too quickly"}) return response except stripe.error.InvalidRequestError as e: logger.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) logger.error(str(e)) response.update({'error': common_message}) return response except stripe.error.APIConnectionError as e: logger.error(str(e)) response.update({'error': common_message}) return response except stripe.error.StripeError as e: # maybe send email logger.error(str(e)) response.update({'error': common_message}) return response except Exception as e: # maybe send email logger.error(str(e)) response.update({'error': common_message}) return response return handleProblems 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): self.stripe = stripe def update_customer_token(self, customer, token): customer.source = token customer.save() @handleStripeError 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 @handleStripeError 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() @handleStripeError 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 @handleStripeError 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 @handleStripeError 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.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 @handleStripeError def get_customer(self, stripe_api_cus_id): customer = stripe.Customer.retrieve(stripe_api_cus_id) # data = customer.get('response_object') return customer @handleStripeError def create_customer(self, token, email, name=None): if name is None or name.strip() == "": name = email customer = self.stripe.Customer.create( source=token, description=name, email=email ) return customer @handleStripeError 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 @handleStripeError def get_or_create_stripe_plan(self, amount, name, stripe_plan_id): """ 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 :param name: The name of the Stripe plan 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 :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 stripe_plan_db_obj = None try: stripe_plan_db_obj = StripePlan.objects.get( stripe_plan_id=stripe_plan_id) except StripePlan.DoesNotExist: try: self.stripe.Plan.create( amount=amount, interval=self.INTERVAL, name=name, currency=self.CURRENCY, id=stripe_plan_id) stripe_plan_db_obj = StripePlan.objects.create( stripe_plan_id=stripe_plan_id) except stripe.error.InvalidRequestError as e: if self.STRIPE_PLAN_ALREADY_EXISTS in str(e): logger.debug( self.PLAN_EXISTS_ERROR_MSG.format(stripe_plan_id)) stripe_plan_db_obj = StripePlan.objects.create( stripe_plan_id=stripe_plan_id) return stripe_plan_db_obj @handleStripeError 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 StripePlan.objects.filter( stripe_plan_id=stripe_plan_id).all().delete() except stripe.error.InvalidRequestError as e: if self.STRIPE_NO_SUCH_PLAN in str(e): logger.debug( self.PLAN_DOES_NOT_EXIST_ERROR_MSG.format(stripe_plan_id)) return return_value @handleStripeError 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 @handleStripeError def set_subscription_metadata(self, subscription_id, metadata): subscription = stripe.Subscription.retrieve(subscription_id) subscription.metadata = metadata subscription.save() @handleStripeError 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() @handleStripeError 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_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) ) @handleStripeError 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()