InAppPurchaseController.py 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246
  1. # @Author : Rocky
  2. # @File : InAppPurchaseController.py
  3. # @Time : 2024/6/21 9:10
  4. import logging
  5. import time
  6. import itunesiap
  7. from django.db.models import Q
  8. from django.http import HttpResponseRedirect
  9. from django.views import View
  10. from Ansjer.config import LOGGER, CONFIG_INFO, CONFIG_TEST, PAY_TYPE_IN_APP_PURCHASE
  11. from Controller.CheckUserData import DataValid
  12. from Model.models import Order_Model, Store_Meal, Device_Info, UID_Bucket, Unused_Uid_Meal, AiService, Device_User, \
  13. SysMsgModel
  14. from Object.AWS.S3Email import S3Email
  15. from Object.AliSmsObject import AliSmsObject
  16. from Object.RedisObject import RedisObject
  17. from Service.CommonService import CommonService
  18. ENV = itunesiap.env.sandbox if CONFIG_INFO == CONFIG_TEST else itunesiap.env.production
  19. class InAppPurchaseView(View):
  20. def get(self, request, *args, **kwargs):
  21. request.encoding = 'utf-8'
  22. operation = kwargs.get('operation')
  23. return self.validation(request.GET, request, operation)
  24. def post(self, request, *args, **kwargs):
  25. request.encoding = 'utf-8'
  26. operation = kwargs.get('operation')
  27. return self.validation(request.POST, request, operation)
  28. def validation(self, request_dict, request, operation):
  29. if operation == 'AppStoreServerNotifications': # App Store服务器通知
  30. return self.app_store_server_notifications(request_dict)
  31. token_code, user_id, response = CommonService.verify_token_get_user_id(request_dict, request)
  32. if token_code != 0:
  33. return response.json(token_code)
  34. if operation == 'verifyTransaction': # 认证交易
  35. return self.verify_transaction(user_id, request_dict, response)
  36. @classmethod
  37. def verify_transaction(cls, user_id, request_dict, response):
  38. """
  39. 认证交易
  40. @param user_id: 用户id
  41. @param request_dict: 请求参数
  42. @request_dict receipt: 收据
  43. @param response: 响应对象
  44. @return: response
  45. """
  46. receipt = request_dict.get('receipt', None)
  47. uid = request_dict.get('uid', None)
  48. lang = request_dict.get('lang', 'en')
  49. channel = request_dict.get('channel', None)
  50. if not all([receipt, uid, channel]):
  51. return response.json(444)
  52. # redis加锁,防止订单重复
  53. redis_obj = RedisObject()
  54. redis_key = uid + 'in_app_purchase'
  55. is_lock = redis_obj.CONN.setnx(redis_key, 1)
  56. redis_obj.CONN.expire(redis_key, 60)
  57. if not is_lock:
  58. return response.json(5)
  59. try:
  60. device_info_qs = Device_Info.objects.filter(userID_id=user_id, UID=uid, isShare=False, isExist=1).values(
  61. 'vodPrimaryUserID',
  62. 'vodPrimaryMaster')
  63. if not device_info_qs.exists():
  64. return response.json(12)
  65. device_info_qs = Device_Info.objects.filter(Q(UID=uid), ~Q(vodPrimaryUserID='')).values('vodPrimaryUserID')
  66. if device_info_qs.exists():
  67. if device_info_qs[0]['vodPrimaryUserID'] != user_id:
  68. return response.json(10033)
  69. order_id = CommonService.createOrderID()
  70. verify_response = itunesiap.verify(receipt, env=ENV)
  71. LOGGER.info('苹果内购收据认证响应:{}'.format(verify_response))
  72. product_id = verify_response.receipt.in_app[0]['product_id']
  73. pay_type = PAY_TYPE_IN_APP_PURCHASE
  74. now_time = int(time.time())
  75. store_qs = Store_Meal.objects.filter(
  76. product_id=product_id, lang__lang=lang, is_show=0).\
  77. values(
  78. 'id', 'currency', 'price', 'lang__content', 'day', 'commodity_type', 'lang__title', 'expire',
  79. 'commodity_code', 'discount_price', 'bucket_id', 'bucket__mold', 'cycle_config_id', 'is_ai')
  80. if not store_qs.exists():
  81. return response.json(173)
  82. rank_id = store_qs[0]['id']
  83. bucket_id = store_qs[0]['bucket_id']
  84. currency = store_qs[0]['currency']
  85. price = store_qs[0]['price']
  86. is_ai = store_qs[0]['is_ai']
  87. expire = store_qs[0]['expire']
  88. end_time = CommonService.calcMonthLater(expire)
  89. content = store_qs[0]['lang__content']
  90. commodity_code = store_qs[0]['commodity_code']
  91. commodity_type = store_qs[0]['commodity_type']
  92. order_type = 1 if is_ai else 0
  93. store_meal_qs = Store_Meal.objects.filter(id=rank_id, lang__lang='cn', is_show=0).\
  94. values('lang__title', 'lang__content')
  95. if store_meal_qs.exists():
  96. store_meal_name = store_meal_qs[0]['lang__title'] + '-' + store_meal_qs[0]['lang__content']
  97. else:
  98. store_meal_name = '未知套餐'
  99. # 查询设备是否已开过云存
  100. use_flag = True
  101. uid_bucket_qs = UID_Bucket.objects.filter(uid=uid).\
  102. values('id', 'bucket_id', 'bucket__region', 'endTime', 'use_status')
  103. if uid_bucket_qs.exists():
  104. uid_bucket = uid_bucket_qs.first()
  105. uid_bucket_id = uid_bucket['id']
  106. # 叠加相同套餐的过期时间
  107. if uid_bucket['use_status'] == 1 and uid_bucket['endTime'] > now_time:
  108. Unused_Uid_Meal.objects.create(
  109. uid=uid, channel=channel, addTime=now_time, order_id=order_id, expire=expire, is_ai=is_ai,
  110. bucket_id=bucket_id)
  111. UID_Bucket.objects.filter(id=uid_bucket_id).update(has_unused=1)
  112. use_flag = False
  113. # 更新套餐的过期时间
  114. else:
  115. UID_Bucket.objects.filter(id=uid_bucket_id).update(
  116. channel=channel, bucket_id=bucket_id, endTime=end_time, updateTime=now_time, use_status=1,
  117. orderId=order_id)
  118. else:
  119. uid_bucket = UID_Bucket.objects.create(
  120. uid=uid, channel=channel, bucket_id=bucket_id, endTime=end_time, use_status=1, orderId=order_id,
  121. addTime=now_time, updateTime=now_time)
  122. uid_bucket_id = uid_bucket.id
  123. # 开通AI服务
  124. if is_ai and use_flag:
  125. ai_service = AiService.objects.filter(uid=uid, channel=channel)
  126. # 有正在使用的套餐,叠加套餐时间,否则创建
  127. if ai_service.exists():
  128. ai_service.update(updTime=now_time, use_status=1, orders_id=order_id, endTime=end_time)
  129. else:
  130. AiService.objects.create(
  131. uid=uid, channel=channel, detect_status=1, use_status=1, orders_id=order_id,
  132. addTime=now_time, updTime=now_time, endTime=end_time)
  133. Order_Model.objects.create(
  134. orderID=order_id, UID=uid, channel=channel, userID_id=user_id, desc=content, payType=pay_type,
  135. payTime=now_time, price=price, currency=currency, addTime=now_time, updTime=now_time,
  136. order_type=order_type, commodity_code=commodity_code, commodity_type=commodity_type, rank_id=rank_id,
  137. ai_rank_id=1, status=1, create_vod=1, store_meal_name=store_meal_name, uid_bucket_id=uid_bucket_id)
  138. # 发送云存开通信息
  139. date_time = time.strftime("%Y-%m-%d", time.localtime())
  140. # 如果存在序列号,消息提示用序列号
  141. device_info_qs = Device_Info.objects.filter(UID=uid).values('serial_number', 'Type')
  142. serial_number = device_info_qs[0]['serial_number']
  143. device_type = device_info_qs[0]['Type']
  144. if serial_number:
  145. device_name = CommonService.get_full_serial_number(uid, serial_number, device_type)
  146. else:
  147. device_name = uid
  148. sys_msg_text_list = [
  149. '温馨提示:尊敬的客户,您的{}设备在{}已成功购买云存套餐'.format(device_name, date_time),
  150. 'Dear customer,you already subscribed the cloud storage package successfully for device {} on '.
  151. format(device_name, time.strftime('%b %dth,%Y', time.localtime()))]
  152. cls.do_vod_msg_notice(uid, user_id, lang, sys_msg_text_list)
  153. redis_obj.del_data(redis_key)
  154. pay_success_url = CommonService.get_payment_status_url(lang, 'success')
  155. return HttpResponseRedirect(pay_success_url)
  156. except Exception as e:
  157. redis_obj.del_data(redis_key)
  158. pay_failed_url = CommonService.get_payment_status_url(lang, 'fail')
  159. return HttpResponseRedirect(pay_failed_url)
  160. @classmethod
  161. def do_vod_msg_notice(cls, uid, user_id, lang, sys_msg_text_list):
  162. """
  163. 发送云存开通信息
  164. @param uid: uid
  165. @param user_id: 用户id
  166. @param lang: 语言
  167. @param sys_msg_text_list: 消息列表
  168. @return: response
  169. """
  170. if lang == 'cn':
  171. sys_msg_text = sys_msg_text_list[0]
  172. else:
  173. sys_msg_text = sys_msg_text_list[1]
  174. now_time = int(time.time())
  175. create_data = {
  176. 'userID_id': user_id,
  177. 'msg': sys_msg_text,
  178. 'addTime': now_time,
  179. 'updTime': now_time,
  180. 'uid': uid,
  181. 'eventType': 0
  182. }
  183. SysMsgModel.objects.create(**create_data)
  184. # 不接收邮件用户
  185. if user_id == '167015836969813800138000':
  186. return
  187. user_qs = Device_User.objects.filter(userID=user_id)
  188. if user_qs.exists():
  189. user = user_qs.first()
  190. username = user.username
  191. data_valid = DataValid()
  192. if data_valid.email_validate(username):
  193. S3Email().faEmail(sys_msg_text, username)
  194. elif data_valid.mobile_validate(username):
  195. # 如果存在序列号,消息提示用序列号
  196. device_info_qs = Device_Info.objects.filter(UID=uid).values('serial_number', 'Type')
  197. if device_info_qs.exists():
  198. serial_number = device_info_qs[0]['serial_number']
  199. device_type = device_info_qs[0]['Type']
  200. if serial_number:
  201. device_name = CommonService.get_full_serial_number(uid, serial_number, device_type)
  202. else:
  203. device_name = uid
  204. params = '{"devname":"%s","submittime":"%s"}' % (
  205. device_name, time.strftime("%Y-%m-%d", time.localtime()))
  206. cls.send_message(username, params, 'SMS_219738485')
  207. @staticmethod
  208. def send_message(phone, params, temp_msg):
  209. """
  210. 发送手机消息
  211. @param phone: 用户名
  212. @param params: 消息参数
  213. @param temp_msg: sms码
  214. """
  215. sign_ms = '周视'
  216. ali_sms = AliSmsObject()
  217. ali_sms.send_code_sms_cloud(phone=phone, params=params, sign_name=sign_ms, temp_msg=temp_msg)
  218. @staticmethod
  219. def app_store_server_notifications(request_dict):
  220. logger = logging.getLogger('apple_pay')
  221. logger.info('App Store服务器通知参数:{}'.format(request_dict))