-
Notifications
You must be signed in to change notification settings - Fork 0
/
referralapi.py
301 lines (240 loc) · 9.76 KB
/
referralapi.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
"""
Getupcloud referral program api
To development run:
$ export FLASK_APP=referralapi
$ export FLASK_DEBUG=1
$ flask run
"""
from functools import reduce
from decimal import Decimal
import logging
from utils import *
import configuration as config
from peewee import IntegrityError
from flask import request, url_for, abort
from flask.ext.api import (
FlaskAPI,
status,
exceptions
)
from database import configure_database
import requests
app = FlaskAPI(__name__)
app.config.from_object('configuration')
database = configure_database(app)
from models import (
ReferralProgram, User,
IndicatedEmail, TransactionLog
)
handler = logging.StreamHandler()
handler.setLevel(logging.DEBUG)
formatter = logging.Formatter( "[%(asctime)s] {%(pathname)s:%(lineno)d} | %(funcName)s | %(levelname)s - %(message)s")
handler.setFormatter(formatter)
app.logger.addHandler(handler)
#=========== Operation Types - For TransactionLog
O_INDICATION = "indication"
O_SIGNUP = "signup"
O_SIGNUP_CREDIT = "signup_credit" # -> Credito que o usuário INDICADO recebe quando faz signup
O_POTENCIAL_CREDIT = "potencial_credit" # -> Credito que o usuário INDICADOR tem a receber (gerado no signup do seu indicado)
O_ACQUIRED_CREDIT = "acquired_credit" # -> Credito que o usuário INDICADOR recebeu e já está no saldo efetivo.
#===========
@app.route("/", methods=("GET",))
def index():
return {
'version':config.VERSION,
'endpoints': list_routes(app)
}
@app.route("/health", methods=("GET",))
def health():
return {"status":"ok"}
@app.route("/program", methods=("GET","POST"))
@app.route("/program/<program_id>", methods=("GET","PUT","DELETE"))
def program(program_id=None):
if request.method == "POST":
rp = ReferralProgram()
rp.name = request.data.get('name', '')
rp.credit_value = request.data.get("credit_value",0)
rp.target_value = request.data.get("target_value",0)
rp.save()
return to_dict(rp), status.HTTP_201_CREATED
if request.method == "GET":
# import pdb; pdb.set_trace()
if not program_id:
programs = ReferralProgram.select()
return to_list_dict(programs)
else:
try:
program = ReferralProgram.select().where(ReferralProgram.id==program_id).first()
if program:
return to_dict(program)
else:
return ({'error': 'ID not found'},
status.HTTP_404_NOT_FOUND)
except Exception as e:
return {'error': str(e)}, status.HTTP_500_INTERNAL_SERVER_ERROR
if request.method == "DELETE":
try:
rp = ReferralProgram.select().where(ReferralProgram.id==program_id).first()
if rp:
rp.delete_instance()
return '', status.HTTP_204_NO_CONTENT
else:
return {'error': 'ID not found'}, status.HTTP_404_NOT_FOUND
except ValidationError as e:
return {'error': 'Invalid ID'}, status.HTTP_404_NOT_FOUND
if request.method == "PUT":
rp = ReferralProgram.select().where(ReferralProgram.id==program_id).first()
if request.data.get('name', ''):
rp.name = request.data.get('name', '')
if request.data.get('credit_value', 0):
rp.credit_value = request.data.get("credit_value",0)
if request.data.get('target_value', 0):
rp.target_value = request.data.get("target_value",0)
rp.save()
return to_dict(rp)
@app.route("/register", methods=("POST",))
def register():
user_indicator_hash = request.data.get("user_indicator",'')
user_indicated_hash = request.data.get("user_indicated",'')
if not user_indicated_hash or not user_indicator_hash:
return ({'error': 'Needed user_indicator and user_indicated hashes'},
status.HTTP_412_PRECONDITION_FAILED)
the_indicator = User.select().where(User.hash==user_indicator_hash).first()
if not the_indicator:
return {'error': 'User indicator hash not found'}, status.HTTP_404_NOT_FOUND
#====================================================================
# OBS IMPORTANTE: É necessário receber o identificador do programa de
# referral que será adicionado o usuário.
#====================================================================
referral_program_id = request.data.get("referral_program_id",'')
if not referral_program_id:
referral_program = ReferralProgram.select().first()
else:
referral_program = ReferralProgram.select().where(
ReferralProgram.id==referral_program_id).first()
if not referral_program:
return {'error': 'Referral program id not found'}, status.HTTP_404_NOT_FOUND
#Saves the new user indicated
try:
new_user = User()
new_user.hash = user_indicated_hash
new_user.referral_program = referral_program
new_user.user_indicator = the_indicator
new_user.save()
except IntegrityError as e:
return {'error': str(e)}, status.HTTP_500_INTERNAL_SERVER_ERROR
#Register the Transaction log
log1 = TransactionLog()
log1.user_indicated = new_user
log1.user_indicator = the_indicator
log1.user_indicated_referral_program = referral_program
log1.operation = O_SIGNUP
log1.save()
try:
#Vai chamar o endpoint de criação de crédito para o usuário indicado (/django/billing/credit/users/<userhash_indicado>/?
data = {
'amount': referral_program.credit_value,
'program_name': referral_program.name,
'user_indicator_hash': user_indicator_hash
}
r = requests.post(config.ENDPOINT_CALLBACK_CREDIT_ON_SIGNUP % user_indicated_hash , data)
except Exception as e:
app.logger.error("Error on calling billing endpoint: %s", str(e))
app.logger.error(str(data))
# app.logger.error(e)
#Register the Transaction log
log2 = TransactionLog()
log2.user_indicated = new_user
log2.user_indicated_referral_program = referral_program
log2.amount = referral_program.credit_value
log2.operation = O_SIGNUP_CREDIT
log2.save()
#Register the Transaction log
log3 = TransactionLog()
log3.user_indicator = the_indicator
log3.user_indicated_referral_program = referral_program
log3.amount = referral_program.target_value
log3.operation = O_POTENCIAL_CREDIT
log3.save()
return to_dict(new_user), status.HTTP_201_CREATED
@app.route("/invoice/user/<user_hash>", methods=("PUT",))
def invoice_user(user_hash):
total_amount_paid = request.data.get("total_amount_paid",0)
user = User.objects(hash=user_hash).first()
referral_program = user.referral_program
if Decimal(total_amount_paid) >= referral_program.target_value:
log = TransactionLog()
log.user_indicated = user
log.amount = referral_program.target_value
log.user_indicated_referral_program = referral_program
log.operation = O_ACQUIRED_CREDIT
log.save()
try:
#Chama o endpoint de criação de crédito
data = {
'amount': referral_program.target_value,
'program_name': referral_program.name,
'user_indicator_hash': user_hash
}
r = requests.post(config.ENDPOINT_CALLBACK_CREDIT_ON_SIGNUP % user_hash , data)
except Exception as e:
app.logger.error("Error on calling billing endpoint")
app.logger.error(data)
app.logger.error(e)
return {'acquired_credit':float(referral_program.target_value)}
return {'acquired_credit':0}
@app.route("/statement/<user_hash>", methods=("GET",))
def statement(user_hash):
user = User.objects(hash=user_hash).first()
credits = TransactionLog.objects(
user_indicator=user, operation=O_POTENCIAL_CREDIT)
total_potencial_credit = reduce(lambda x,y: x + y.amount, credits)
user_statement = {
'potencial_credit': total_potencial_credit,
'referral_program': to_dict(user.referral_program)
}
return user_statement
@app.route("/emailindication", methods=("POST",))
def emailindication():
user_indicator = request.data.get("user_indicator")
if not user_indicator:
return ({'error': 'Need user_indicator parameter'},
status.HTTP_412_PRECONDITION_FAILED)
indicator = User.objects(hash=user_indicator).first()
if not indicator:
return ({'error': 'User indicator not found'},
status.HTTP_412_PRECONDITION_FAILED)
emails = request.data.get("emails")
saved_emails = []
for email in emails.split(","):
indic = IndicatedEmail()
indic.email = email
indic.user_indicator = indicator
try:
indic.save()
saved_emails.append(email)
except Exception as e:
app.logger.warning("Email '%s' already indicated by other user", email)
return {'saved_emails': saved_emails}, status.HTTP_201_CREATED
@app.route("/emailverification", methods=("POST",))
def emailverification():
emails = request.data.get("emails")
verification = {}
for email in emails.split(","):
ver = IndicatedEmail.objects(email=email).first()
verification[email] = ver.user_indicator.hash if bool(ver) else False
return {'emails':verification}
def import_users():
file = open('../users.txt','r')
rp = ReferralProgram.select().first()
if not rp:
print("Have no one Referral Program")
exit()
for line in file.readlines():
dados = [x.strip() for x in line.split("|") if x.strip()]
print("Saving {0} {1}".format(*dados))
u = User()
u.hash = dados[0]
u.referral_program = rp
# u.user_indicator = None
u.save()