dynamicweb/utils/stripe_utils.py

431 lines
16 KiB
Python
Raw Permalink Normal View History

import logging
import re
import stripe
from django.conf import settings
from datacenterlight.models import StripePlan
2016-05-01 12:13:12 +00:00
stripe.api_key = settings.STRIPE_API_PRIVATE_KEY
logger = logging.getLogger(__name__)
def handleStripeError(f):
2016-05-04 15:36:54 +00:00
def handleProblems(*args, **kwargs):
response = {
'paid': False,
'response_object': None,
'error': None
}
2017-06-04 22:05:48 +00:00
2017-08-07 06:56:57 +00:00
common_message = "Currently it's not possible to make payments."
2016-05-04 15:36:54 +00:00
try:
response_object = f(*args, **kwargs)
response = {
2016-05-04 15:36:54 +00:00
'response_object': response_object,
'error': None
}
2016-05-04 15:36:54 +00:00
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))
2016-05-04 15:36:54 +00:00
return response
except stripe.error.RateLimitError as e:
2017-08-24 05:01:57 +00:00
response.update(
{'error': "Too many requests made to the API too quickly"})
2016-05-04 15:36:54 +00:00
return response
except stripe.error.InvalidRequestError as e:
logger.error(str(e))
2016-05-04 15:36:54 +00:00
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))
2016-05-04 15:36:54 +00:00
response.update({'error': common_message})
return response
except stripe.error.APIConnectionError as e:
logger.error(str(e))
2016-05-04 15:36:54 +00:00
response.update({'error': common_message})
return response
except stripe.error.StripeError as e:
# maybe send email
logger.error(str(e))
2016-05-04 15:36:54 +00:00
response.update({'error': common_message})
return response
except Exception as e:
# maybe send email
logger.error(str(e))
2016-05-04 15:36:54 +00:00
response.update({'error': common_message})
return response
2016-05-04 15:36:54 +00:00
return handleProblems
2016-05-01 12:13:12 +00:00
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
2016-05-01 12:13:12 +00:00
2016-12-19 14:33:15 +00:00
def update_customer_token(self, customer, token):
2017-01-20 16:33:25 +00:00
customer.source = token
customer.save()
2016-12-19 14:33:15 +00:00
@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
2017-10-21 18:45:00 +00:00
def dissociate_customer_card(self, stripe_customer_id, card_id):
customer = stripe.Customer.retrieve(stripe_customer_id)
2017-10-21 18:45:00 +00:00
card = customer.sources.retrieve(card_id)
card.delete()
2017-01-20 16:33:25 +00:00
@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()
2016-12-19 14:33:15 +00:00
customer.source = token
customer.save()
2017-01-20 16:33:25 +00:00
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
2016-12-19 14:33:15 +00:00
@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
2019-04-02 07:18:15 +00:00
@handleStripeError
def get_all_invoices(self, customer_id, created_gt):
2019-04-02 07:18:15 +00:00
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:
2019-04-20 06:57:13 +00:00
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(
2019-04-03 05:32:18 +00:00
[line.metadata.VM_ID if hasattr(line.metadata, 'VM_ID') else '' for line in invoice.lines.data]
),
2019-04-20 06:57:13 +00:00
'subscription_ids_csv': ','.join(sub_ids),
2019-04-13 11:42:04 +00:00
'line_items': invoice.lines.data
}
starting_after = invoice.id
return_list.append(invoice_details)
2019-04-02 07:18:15 +00:00
return return_list
@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)
2017-10-21 18:45:00 +00:00
user.stripecustomer.stripe_id = customer.get(
'response_object').get('id')
user.stripecustomer.save()
if type(customer) is dict:
customer = customer['response_object']
2016-05-01 12:13:12 +00:00
return customer
2017-01-20 16:33:25 +00:00
@handleStripeError
2017-10-26 22:41:49 +00:00
def get_customer(self, stripe_api_cus_id):
customer = stripe.Customer.retrieve(stripe_api_cus_id)
2017-01-20 16:33:25 +00:00
# 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(
2016-05-01 12:13:12 +00:00
source=token,
description=name,
2016-05-01 12:13:12 +00:00
email=email
)
return customer
@handleStripeError
def make_charge(self, amount=None, customer=None):
2017-05-25 18:04:29 +00:00
_amount = float(amount)
amount = int(_amount * 100) # stripe amount unit, in cents
charge = self.stripe.Charge.create(
2016-05-04 15:36:54 +00:00
amount=amount, # in cents
currency=self.CURRENCY,
customer=customer
)
return charge
@handleStripeError
def get_or_create_stripe_plan(self, amount, name, stripe_plan_id):
"""
2017-08-24 05:01:57 +00:00
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.
2017-08-24 05:01:57 +00:00
: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
2017-08-17 09:04:22 +00:00
stripe_plan_db_obj = None
try:
2017-08-24 05:01:57 +00:00
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)
2017-08-24 05:01:57 +00:00
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):
2017-08-24 05:01:57 +00:00
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):
"""
2017-08-24 05:01:57 +00:00
Deletes the Plan in Stripe and also deletes the local db copy
of the plan if it exists
2017-08-24 05:01:57 +00:00
: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
2017-08-24 05:01:57 +00:00
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):
2017-08-24 05:01:57 +00:00
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
2017-08-24 05:01:57 +00:00
:param plans: A list of stripe plans.
:param trial_end: An integer representing when the Stripe subscription
is supposed to end
2017-08-24 05:01:57 +00:00
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()
2017-10-03 11:06:26 +00:00
@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()
2016-05-04 15:36:54 +00:00
@handleStripeError
def make_payment(self, customer, amount, token):
2016-05-04 15:36:54 +00:00
charge = self.stripe.Charge.create(
amount=amount, # in cents
currency=self.CURRENCY,
customer=customer
)
return charge
@staticmethod
2018-09-05 21:26:51 +00:00
def get_stripe_plan_id(cpu, ram, ssd, version, app='dcl', hdd=None,
price=None):
"""
Returns the Stripe plan id string of the form
2017-08-24 05:01:57 +00:00
`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
2017-08-24 05:01:57 +00:00
:param app: The application to which the stripe plan belongs
to. By default it is 'dcl'
2018-09-05 21:26:51 +00:00
:param price: The price for this plan
:return: A string of the form `dcl-v1-cpu-2-ram-5gb-ssd-10gb`
"""
2017-08-24 05:01:57 +00:00
dcl_plan_string = 'cpu-{cpu}-ram-{ram}gb-ssd-{ssd}gb'.format(cpu=cpu,
ram=ram,
ssd=ssd)
if hdd is not None:
2017-08-24 05:01:57 +00:00
dcl_plan_string = '{dcl_plan_string}-hdd-{hdd}gb'.format(
dcl_plan_string=dcl_plan_string, hdd=hdd)
2017-09-22 21:55:48 +00:00
stripe_plan_id_string = '{app}-v{version}-{plan}'.format(
app=app,
version=version,
2018-09-05 21:26:51 +00:00
plan=dcl_plan_string
)
if price is not None:
stripe_plan_id_string_with_price = '{}-{}chf'.format(
stripe_plan_id_string,
2018-09-05 22:31:17 +00:00
round(price, 2)
2018-09-05 21:26:51 +00:00
)
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 as ie:
logger.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
2018-09-05 21:26:51 +00:00
def get_stripe_plan_name(cpu, memory, disk_size, price):
"""
Returns the Stripe plan name
:return:
"""
2018-09-05 22:52:55 +00:00
return "{cpu} Cores, {memory} GB RAM, {disk_size} GB SSD, " \
"{price} CHF".format(
2018-09-05 21:26:51 +00:00
cpu=cpu,
memory=memory,
disk_size=disk_size,
2018-09-05 22:31:17 +00:00
price=round(price, 2)
2018-09-05 21:26:51 +00:00
)
@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
2017-10-28 20:44:58 +00:00
subscription.save()