You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

562 lines
21 KiB
Python

"""
This file is a wrapper for stripe and store handling operations :
@TODO:
- Add logging using transport
"""
import stripe
from utils.params import PARAMS as SYS_ARGS
from utils import void
import json
class User :
def __init__(self,email,plans):
self.plans = plans
self.me = {"plan":[],"customer":{},"payments":[],"plan2sub":{}}
self.init(email)
self.format() #-- creating properties
pass
def info(self):
return self.me['customer']
def _add_card(self,card):
try:
stripe.token.Token.create(customer=self.me["customer"]['id'],card=card)
self.init(self.me["customer"]['email'])
return 1
except Exception as error :
print (error)
return 0
def charge(self,**args) :
"""
This function will charge a user for a service (given a plan id)
:plan_id plan identifier for the current product
:card_index provide a token (otherwise will use a valid card)
"""
plan_id = args['plan_id']
if len(self.me["payments"]) == 1 :
card_index = 0
else:
card_index = args['card_index']
plan = [item[plan_id] for item in self.plans if item['id'] == plan_id]
card = {} if plan['amount'] == 0 else self.me["payments"][card_index]
params = {}
params['customer'] = self.me["customer"]['id']
params ['amount'] = plan['amount']
params ['currency'] = plan['currency']
params ['description'] = " ".join([self.product['statement_descriptor'],plan['nickname']])
if 'source' in card :
params ['source'] = card['source']
stripe.charge.Charge.create(params)
del params['amount'], params['currency']
stripe.invoice.Invoice.create(**params)
pass
def format(self):
self.plan = void()
self.plan.info = lambda: self.me["plan"]
self.plan.sub_id = lambda: self.me['plan2sub'].get(self.plan.info()['id'],{})
# self.user.plan.cancel = lambda id: stripe.subscription.Subscription.delete(self.customer['id'],prorate=True,invoice_now=True)
self.card = void ()
self.card.exists = lambda: len(self.me["payments"]) > 0
self.card.add = self._add_card
self.card.charge = self.charge
self.card.html = lambda: [{} for card in self.me["payments"]]
self.card.delete = lambda index: stripe.customer.Customer.delete_source(self.me["customer"]['id'],self.me["payment"][index])
def init(self,email):
#
# get customer information
# email = args['email']
# self.user_plan = []
# self.customer = {}
customer = stripe.customer.Customer.list(email=email).to_dict_recursive()['data']
free_plan = [item for item in self.plans if item['amount'] == 0]
if customer :
customer = customer[0]
else:
#
# The user doesn't exist, we must create the user (without payment initially)
# We can probably assign the user to a free plan if available
#
customer = stripe.customer.Customer.create(email=email)
# if free_plan :
# free_plan = free_plan[0]
# self.plan.subscribe(free_plan['id'],email)
# self.customer = {"email":email,"id":customers['id']}
self.me["customer"] = {"email":email,"id":customer['id']}
lsub = customer.subscriptions.data
for sub in lsub :
id = sub.plan.id
self.me['plan2sub'][id] = sub.id
#
# extract payments available,
if 'sources' in customer and 'data' in customer['sources'] :
if customer['sources']['data'] :
# self.payment = [ card.to_dict_recursive() for card in customers['sources']['data'] ]
self.me['payments'] = [ card.to_dict_recursive() for card in customer['sources']['data'] ]
# if not self._has_plan(customer) and free_plan:
# free_plan = free_plan[0]
# self.subscribe(free_plan['id'],email)
# info = customers #self.me['customer']
# subscriptions = info['subscriptions']['data']
# ids = [plan['id'] for plan in self.plans ]
# _found = None
# for sub in subscriptions :
# aplan = sub['plan']
# if set([aplan['id']]) & set(ids):
# _found = sub.to_dict_recursive()
# break
self.me["plan"] = self._has_plan(customer) #_found if _found else []
# self.user_plan = _found if _found else None
def _has_plan(self,customer):
subscriptions = customer['subscriptions']['data']
ids = [plan['id'] for plan in self.plans ]
_found = None
for sub in subscriptions :
aplan = sub['plan']
if set([aplan['id']]) & set(ids):
_found = sub.to_dict_recursive()['plan']
if 'features' in _found['metadata'] :
_found ['metadata']['features'] = json.loads(_found['metadata']['features'])
break
return _found if _found else []
class Store :
def __init__(self,**args):
"""
Creating an instance of a store. A store is a proxy to stripe (https://stripe.com)
:name name of the product
:secret stripe api key
"""
self.products = []
self.plans = []
self.product = {}
product_name = args['name']
stripe.api_key = args['secret'] if 'secret' in args else stripe.api_key
resp = stripe.product.Product.list().to_dict_recursive()['data']
if resp :
self.products = [item.to_dict_recursive() for item in resp]
if len(self.products) > 1 :
self.product = [item for item in self.products if item['name'] == product_name]
self.product = self.product[0] if self.product else []
else:
self.product = self.products[0]
#
# if we have a product we should get the list of plans associated
#
def init(self,email):
"""
This function pulls a user's information, think of it like a customer walks into a store ...
:email
"""
pass
class Product(Store):
"""
This class implements a wrapper around products, services or just simple donations
we are inheriting product, plans
"""
def __init__(self,**args):
Store.__init__(self,**args)
def set(self,key,value):
if key == 'plan' :
#
# do we have this plan
found = [item for item in self.plans if item['id'] == value['id']]
if not found :
plan = value
else:
plan = dict(found[0],**value)
if 'metadata' in value :
plan['metadata'] = self.meta(plan,value)
self.save('plan',plan)
pass
else :
if key in ['type','name','id','statement_descriptor'] :
if value :
self.product[key] = value
elif key in self.product :
del self.product[key]
elif key == 'metadata' :
self.product['metadata'] = self.meta(self.product,value)
self.save('product',self.product)
def meta(self,object,value) :
for k in value :
if value[k] :
object['metadata'][k] = json.dumps(value[k]) if type(value[k]) == dict else value[k]
else:
del object['metadata'][k]
return object
def save(self,id,object):
pointer = None
if id == 'product' :
p = 'about' in object['metadata']
q = 'uri' in object['metadata']
if p and q :
pointer = stripe.product.Product
return 1
else :
return 0
elif self.product['id']:
pointer = stripe.plan.Plan
object = dict(object,**{"product":{"name":self.product['id']}} )
if pointer :
pointer.create(**object)
return 1
return 0
def get(self):
pass
class Plans(Store) :
def __init__(self,**args):
"""
This function will retrieve plan information associated with the given product id
"""
Store.__init__(self,**args)
self.plans = stripe.plan.Plan.list(product=self.product['id']).to_dict_recursive()['data']
for item in self.plans :
index = self.plans.index(item)
self.plans[index] = item.to_dict_recursive()
if 'features' in self.plans[index]['metadata'] :
self.plans[index]['metadata']['features'] = json.loads(self.plans[index]['metadata']['features'])
self.plans.sort(key=lambda item: item['amount'])
if 'email' in args :
self.user = User(args['email'],self.plans)
else:
self.user = None
# self.plans = [item.to_dict_recursive() for item in self.plans]
self.format()
def format(self):
self.get = void()
self.get.plans = lambda:self.plans
self.get.products = lambda:self.products
self.plan = void()
self.plan.upgrade = self.upgrade
self.plan.cancel = self.cancel
self.plan.subscribe = self.subscribe
def cancel(self,id) :
"""
This function will cancel a subscription given a user's email/or a subscription for the given product/user
We are assuming one subscription one plan, no complex billing is involved (not good for startups)
:email,subscription
"""
stripe.subscription.Subscription.delete(id,prorate=True,invoice_now=True)
pass
def upgrade(self,id,email):
"""
A change of plan suggests cancelling the current plan and assigning a new one.
If it's a paid plan, the calling code must insure the user has an active token
"""
if not self.user and email:
self.user = User(email,self.plans)
# sub_id = self.user.plan.info()['id']
sub_id = self.user.plan.sub_id()
try:
self.cancel(sub_id)
r = self.subscribe(id=id,email=email)
#
# We need to reload everything
self.user.init(email)
except Exception as error :
print (error) ;
return 0
return 1
pass
# def subscribe(self,id,email):
def subscribe(self,**args):
"""
Subscribe a user to a given plan assuming the user has already been initialized
The responsibility falls upon the calling code to perform the cancellatioin of the existing plan and perform an upgrade
:email user's email
:id plan id (optional) will default to free plan
"""
free_plan = [item['id'] for item in self.plans if item['amount'] == 0]
if free_plan :
free_plan = free_plan[0]
# id = args['id'] if 'id' in args else free_plan
email = args['email']
if not self.user :
self.user = User(email,self.plans)
plan = {"plan": args['id'] if 'id' in args else free_plan}
amount = sum([item['amount'] for item in self.plans if item['id'] == 0])
if 'email' in self.user.info() and email == self.user.info()['email'] :
email = self.user.info()['email']
else:
self.user.init(email)
#
# We must insure the user is one of our customers i.e perhaps
if amount == 0 or self.payment :
r = stripe.subscription.Subscription.create(
customer = self.user.info()['id'],
items = [plan]
)
# stripe.subscription.Subscription.create(
# customer=self.user.info()['id'], #if not email else email,
# items=[plan]
# )
self.user.init(email)
return 1
else:
return 0
class factory:
@staticmethod
def instance (**args) :
p = Plans(**args)
# p.user.subscribe = p.subscribe
return p
# if 'email' in args :
# store = Plans(**args)
# else:
# store = Plans(**args)
# return store
# class _Store :
# # stripe_keys = {
# # "secret_key":SYS_ARGS['stripe']['secret'].strip(),
# # "publishable_key":SYS_ARGS['stripe']['pub'].strip()
# # }
# # stripe.api_key = self.stripe_keys['secret_key']
# def __init__(saelf,**args):
# self.payment = []
# self.user_plan = {}
# self.customer = {}
# self.get = void()
# # self.get.user = void()
# # self.get.user.info = self._get_user
# # self.get.user.plan = lambda: self.user_plan if self.user_plan else None
# # self.add = void()
# # self.add.user = self.init
# # self.add.card = lambda card: stripe.token.Token.create(customer=customer['id'],card=card)
# self.user = void()
# self.user.init = self.init
# self.user.info = self._get_user
# self.user.plan = void()
# self.user.plan.info = lambda: self.user_plan if self.user_plan else None
# # self.user.plan.cancel = lambda id: stripe.subscription.Subscription.delete(self.customer['id'],prorate=True,invoice_now=True)
# self.user.card = void ()
# self.user.card.exists = lambda: len(self.payment) > 0
# self.user.card.add = self._add_card
# self.user.card.charge = self._apply_charge
# self.user.card.html = lambda: [{} for card in self.payment]
# self.user.card.delete = lambda index: stripe.customer.Customer.delete_source(self.customer['id'],self.payment[index])
# self.get.plans = self._get_plans
# self.has = void ()
# product_name = args['product']
# stripe.api_key = args['secret'] if 'secret' in args else stripe.api_key
# if stripe.api_key :
# self.__init_product(product_name)
# else:
# print (['secret' in args, stripe.api_key is None])
# self.plan = void()
# self.plan.cancel = self.__cancel_plan
# self.plan.subscribe = self.__subscribe_plan
# self.plan.upgrade = self.__upgrade_plan
# def _apply_charge(self,**args) :
# """
# This function will charge a user for a service (given a plan id)
# :plan_id plan identifier for the current product
# :card_index provide a token (otherwise will use a valid card)
# """
# plan_id = args['plan_id']
# if len(self.payment) == 1 :
# card_index = 0
# else:
# card_index = args['card_index']
# plan = [item[plan_id] for item in self.plans if item['id'] == plan_id]
# card = {} if plan['amount'] == 0 else self.payment[card_index]
# params = {}
# params['customer'] = self.customer['id']
# params ['amount'] = plan['amount']
# params ['currency'] = plan['currency']
# params ['description'] = " ".join([self.product['statement_descriptor'],plan['nickname']])
# if 'source' in card :
# params ['source'] = card['source']
# stripe.charge.Charge.create(params)
# del params['amount'], params['currency']
# stripe.invoice.Invoice.create(**params )
# pass
# def __init_product(self,product_name):
# PRODUCT_INDEX = 2
# DATA_TYPE_INDEX = 1
# STATUS_INDEX = 0
# resp = stripe.product.Product.list().to_dict_recursive()['data']
# self.product = []
# self.plan = []
# if resp :
# if resp :
# self.product = [p.to_dict_recursive() for p in resp if p['name'] == product_name]
# self.product = self.product[0] if self.product else []
# else:
# self.product = resp[0]
# if 'id' in self.product :
# self.__init_plan(self.product['id'])
# def __init_plan(self,id):
# """
# This function will retrieve plan information associated with the given product id
# """
# self.plans = stripe.plan.Plan.list(product=id).to_dict_recursive()['data']
# for item in self.plans :
# index = self.plans.index(item)
# self.plans[index] = item.to_dict_recursive()
# if 'features' in self.plans[index]['metadata'] :
# self.plans[index]['metadata']['features'] = json.loads(self.plans[index]['metadata']['features'])
# #
# # sort items by amounts
# #
# self.plans.sort(key=lambda item: item['amount'])
# # self.plans = [item.to_dict_recursive() for item in self.plans]
# def init(self,email):
# #
# # get customer information
# # email = args['email']
# self.user_plan = []
# self.customer = {}
# customers = stripe.customer.Customer.list(email=email).to_dict_recursive()['data']
# if customers :
# customers = customers[0]
# else:
# #
# # The user doesn't exist, we must create the user (without payment initially)
# # We can probably assign the user to a free plan if available
# #
# customers = stripe.customer.Customer.Create(email=email)
# free_plan = [item for item in self.plans if item['amount'] == 0]
# if free_plan :
# free_plan = free_plan[0]
# self.plan.subscribe(free_plan['id'],email)
# self.customer = {"email":email,"id":customers['id']}
# #
# # extract payments available,
# if 'sources' in customers and 'data' in customers['sources'] :
# if customers['sources']['data'] :
# self.payment = [ card.to_dict_recursive() for card in customers['sources']['data'] ]
# #
# # user plans
# info = customers
# subscriptions = info['subscriptions']['data']
# ids = [plan['id'] for plan in self.plans ]
# _found = None
# for sub in subscriptions :
# aplan = sub['plan']
# if set([aplan['id']]) & set(ids):
# _found = sub.to_dict_recursive()
# break
# self.user_plan = _found if _found else None
# def _get_user(self,**args):
# """
# This function will return a customer information and associated plan (hopefully)
# :email
# """
# return self.customer
# def __cancel_plan(self,id) :
# """
# This function will cancel a subscription given a user's email/or a subscription for the given product/user
# We are assuming one subscription one plan, no complex billing is involved (not good for startups)
# :email,subscription
# """
# stripe.subscription.Subscription.delete(id,prorate=True,invoice_now=True)
# pass
# def __upgrade_plan(self,id,email=None):
# """
# A change of plan suggests cancelling the current plan and assigning a new one.
# If it's a paid plan, the calling code must insure the user has an active token
# """
# email = self.customer['email'] if email is None else email
# sub_id = self.user.plan.info()['id']
# try:
# self.plan.cancel(sub_id)
# self.plan.subscribe(id,email)
# #
# # We need to reload everything
# self.user.init(email)
# except Exception as error :
# print (error) ;
# return 0
# return 1
# pass
# def __subscribe_plan(self,id,email):
# """
# Subscribe a user to a given plan assuming the user has already been initialized
# The responsibility falls upon the calling code to perform the cancellatioin of the existing plan and perform an upgrade
# """
# plan = {"plan": id}
# amount = sum([item['amount'] for item in self.plans if item['id'] == 0])
# if 'email' in self.customer and email == self.customer['email'] :
# email = self.customer['email']
# else:
# self.init(email)
# #
# # We must insure the user is one of our customers i.e perhaps
# if amount == 0 or self.payment :
# stripe.subscription.Subscription.create(
# customer=self.customer['id'], #if not email else email,
# items=[plan]
# )
# return 1
# else:
# return 0
# def _add_card(self,card):
# try:
# stripe.token.Token.create(customer=customer['id'],card=card)
# self.user.init(self.customer['email'])
# return 1
# except Exception as error :
# print (error)
# return 0
# def _get_plans(self,**args) :
# """
# This function will provide an plan for a given user given an email or the list of plans
# :email user email
# """
# return self.user_plan if 'email' in args else self.plans