InAppPurchaseController.py 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438
  1. # @Author : Rocky
  2. # @File : InAppPurchaseController.py
  3. # @Time : 2024/6/21 9:10
  4. import logging
  5. import time
  6. import json
  7. import requests
  8. from appstoreserverlibrary.api_client import AppStoreServerAPIClient, GetTransactionHistoryVersion
  9. from appstoreserverlibrary.models.Environment import Environment
  10. from appstoreserverlibrary.receipt_utility import ReceiptUtility
  11. from appstoreserverlibrary.models.HistoryResponse import HistoryResponse
  12. from appstoreserverlibrary.models.TransactionHistoryRequest import TransactionHistoryRequest, ProductType, Order
  13. from appstoreserverlibrary.signed_data_verifier import SignedDataVerifier
  14. from cryptography.hazmat.backends import default_backend
  15. from cryptography.hazmat.primitives.serialization import load_pem_private_key
  16. from django.db.models import Q
  17. from django.views import View
  18. from django.http import HttpResponse
  19. from Ansjer.config import LOGGER, CONFIG_INFO, CONFIG_TEST, PAY_TYPE_IN_APP_PURCHASE, BASE_DIR, CONFIG_US
  20. from Controller.CheckUserData import DataValid
  21. from Model.models import Order_Model, Store_Meal, Device_Info, UID_Bucket, Unused_Uid_Meal, AiService, Device_User, \
  22. SysMsgModel, InAppPurchasePackage
  23. from Object.AWS.S3Email import S3Email
  24. from Object.AliSmsObject import AliSmsObject
  25. from Object.AppleInAppPurchaseSubscriptionObject import InAppPurchase
  26. from Object.RedisObject import RedisObject
  27. from Service.CommonService import CommonService
  28. ENV = Environment.SANDBOX if CONFIG_INFO == CONFIG_TEST else Environment.PRODUCTION
  29. class InAppPurchaseView(View):
  30. def get(self, request, *args, **kwargs):
  31. request.encoding = 'utf-8'
  32. operation = kwargs.get('operation')
  33. return self.validation(request.GET, request, operation)
  34. def post(self, request, *args, **kwargs):
  35. request.encoding = 'utf-8'
  36. operation = kwargs.get('operation')
  37. return self.validation(request.POST, request, operation)
  38. def validation(self, request_dict, request, operation):
  39. if operation == 'AppStoreServerNotifications': # App Store服务器通知
  40. return self.app_store_server_notifications(request)
  41. elif operation == 'VseesNotifications':
  42. return self.vsees_notifications(request)
  43. token_code, user_id, response = CommonService.verify_token_get_user_id(request_dict, request)
  44. if token_code != 0:
  45. return response.json(token_code)
  46. if operation == 'verifyTransaction': # 认证交易
  47. return self.verify_transaction(user_id, request_dict, response)
  48. @classmethod
  49. def verify_transaction(cls, user_id, request_dict, response):
  50. """
  51. 认证交易
  52. @param user_id: 用户id
  53. @param request_dict: 请求参数
  54. @request_dict receipt: 收据
  55. @param response: 响应对象
  56. @return: response
  57. """
  58. receipt = request_dict.get('receipt', None)
  59. order_id = request_dict.get('orderID', None)
  60. uid = request_dict.get('uid', None)
  61. lang = request_dict.get('lang', 'en')
  62. channel = request_dict.get('channel', None)
  63. app_type = request_dict.get('appType', 1)
  64. logger = logging.getLogger('apple_pay')
  65. logger.info(f"receipt: {receipt}, 订单orderId: {order_id}, uid: {uid}")
  66. if not all([receipt, uid, channel, order_id]):
  67. return response.json(444)
  68. # redis加锁,防止订单重复
  69. redis_obj = RedisObject()
  70. redis_key = order_id + 'in_app_purchase'
  71. is_lock = redis_obj.CONN.setnx(redis_key, 1)
  72. redis_obj.CONN.expire(redis_key, 60)
  73. if not is_lock:
  74. return response.json(5)
  75. try:
  76. # 检查商品id是否正确
  77. app_type = int(app_type)
  78. if app_type == 1:
  79. bundle_id = "com.ansjer.zccloud"
  80. elif app_type == 2:
  81. bundle_id = "com.ansjer.zccloud"
  82. else:
  83. return response.json(444, "app_type不存在")
  84. # 实例化订阅类
  85. in_app_purchase = InAppPurchase(bundle_id=bundle_id)
  86. # ReceiptUtility 用于解析收据为transaction_id
  87. receipt_util = in_app_purchase.receipt_util
  88. # AppStoreServerAPIClient 用于查询交易信息
  89. client = in_app_purchase.client
  90. # SignedDataVerifier 用于解析查询到的交易信息
  91. signed_data_verifier = in_app_purchase.verifier
  92. # 解析收据(循环扣款时不需要这一步, 直接获取transaction_id)
  93. transaction_id = receipt_util.extract_transaction_id_from_app_receipt(receipt)
  94. if transaction_id is None:
  95. pay_result_url = CommonService.get_payment_status_url(lang, 'fail')
  96. return response.json(0, {'url': pay_result_url})
  97. logger.info(f"订单orderId:{order_id}, transaction_id:{transaction_id}")
  98. # 查询交易信息
  99. transaction_info = client.get_transaction_info(transaction_id)
  100. signed_transaction_info = transaction_info.signedTransactionInfo
  101. # 解析交易信息
  102. payload = signed_data_verifier.verify_and_decode_signed_transaction(signed_transaction_info)
  103. # 获取交易的商品id
  104. product_id = payload.productId if payload and payload.productId else None
  105. if not product_id:
  106. pay_result_url = CommonService.get_payment_status_url(lang, 'fail')
  107. return response.json(0, {'url': pay_result_url})
  108. in_app_purchase_package_qs = InAppPurchasePackage.objects.filter(product_id=product_id, app_type=app_type)
  109. if not in_app_purchase_package_qs.exists():
  110. return response.json(173, "内购商品id不存在")
  111. # 验证订单是否存在
  112. order_qs = Order_Model.objects.filter(orderID=order_id, UID=uid, app_type=app_type).values("rank_id")
  113. if not order_qs.exists():
  114. return response.json(173, "订单不存在")
  115. # 验证套餐是否存在
  116. store_qs = Store_Meal.objects.filter(id=order_qs[0]['rank_id']).values(
  117. 'id', 'currency', 'price', 'lang__content', 'day', 'commodity_type', 'lang__title', 'expire',
  118. 'commodity_code', 'discount_price', 'bucket_id', 'bucket__mold', 'cycle_config_id', 'is_ai')
  119. if not store_qs.exists():
  120. return response.json(173, "云存套餐不存在")
  121. # 设备开通云存
  122. now_time = int(time.time())
  123. uid_bucket_id = cls.enable_cloud(channel, now_time, order_id, store_qs, uid)
  124. # 修改订单信息
  125. order_qs.update(status=1, uid_bucket_id=uid_bucket_id, transaction_id=transaction_id, create_vod=1)
  126. # 构建云存套餐消息
  127. sys_msg_text_list = cls.cloud_storage_message(uid)
  128. cls.do_vod_msg_notice(uid, user_id, lang, sys_msg_text_list)
  129. # 删除缓存
  130. redis_obj.del_data(redis_key)
  131. pay_result_url = CommonService.get_payment_status_url(lang, 'success')
  132. return response.json(0, {'url': pay_result_url})
  133. except Exception as e:
  134. redis_obj.del_data(redis_key)
  135. LOGGER.info('苹果内购认证交易接口异常:{}'.
  136. format('error_line:{}, error_msg:{}'.format(e.__traceback__.tb_lineno, repr(e))))
  137. pay_result_url = CommonService.get_payment_status_url(lang, 'fail')
  138. return response.json(0, {'url': pay_result_url})
  139. @classmethod
  140. def cloud_storage_message(cls, uid):
  141. # 发送云存开通信息
  142. date_time = time.strftime("%Y-%m-%d", time.localtime())
  143. # 如果存在序列号,消息提示用序列号
  144. device_info_qs = Device_Info.objects.filter(UID=uid).values('serial_number', 'Type')
  145. serial_number = device_info_qs[0]['serial_number']
  146. device_type = device_info_qs[0]['Type']
  147. if serial_number:
  148. device_name = CommonService.get_full_serial_number(uid, serial_number, device_type)
  149. else:
  150. device_name = uid
  151. sys_msg_text_list = [
  152. '温馨提示:尊敬的客户,您的{}设备在{}已成功购买云存套餐'.format(device_name, date_time),
  153. 'Dear customer,you already subscribed the cloud storage package successfully for device {} on '.
  154. format(device_name, time.strftime('%b %dth,%Y', time.localtime()))]
  155. return sys_msg_text_list
  156. @classmethod
  157. def enable_cloud(cls, channel, now_time, order_id, store_qs, uid):
  158. bucket_id = store_qs[0]['bucket_id']
  159. is_ai = store_qs[0]['is_ai']
  160. expire = store_qs[0]['expire']
  161. end_time = CommonService.calcMonthLater(expire)
  162. # 查询设备是否已开过云存
  163. use_flag = True
  164. uid_bucket_qs = UID_Bucket.objects.filter(uid=uid). \
  165. values('id', 'bucket_id', 'bucket__region', 'endTime', 'use_status')
  166. if uid_bucket_qs.exists():
  167. uid_bucket = uid_bucket_qs.first()
  168. uid_bucket_id = uid_bucket['id']
  169. # 有正在使用的套餐,创建为未使用套餐
  170. if uid_bucket['use_status'] == 1 and uid_bucket['endTime'] > now_time:
  171. Unused_Uid_Meal.objects.create(
  172. uid=uid, channel=channel, addTime=now_time, order_id=order_id, expire=expire, is_ai=is_ai,
  173. bucket_id=bucket_id)
  174. UID_Bucket.objects.filter(id=uid_bucket_id).update(has_unused=1)
  175. use_flag = False
  176. # 无正在使用套餐,直接使用套餐
  177. else:
  178. UID_Bucket.objects.filter(id=uid_bucket_id).update(
  179. channel=channel, bucket_id=bucket_id, endTime=end_time, updateTime=now_time, use_status=1,
  180. orderId=order_id)
  181. else:
  182. uid_bucket = UID_Bucket.objects.create(
  183. uid=uid, channel=channel, bucket_id=bucket_id, endTime=end_time, use_status=1, orderId=order_id,
  184. addTime=now_time, updateTime=now_time)
  185. uid_bucket_id = uid_bucket.id
  186. # 开通AI服务
  187. if is_ai and use_flag:
  188. ai_service = AiService.objects.filter(uid=uid, channel=channel)
  189. # 有正在使用的套餐,叠加套餐时间,否则创建
  190. if ai_service.exists():
  191. ai_service.update(updTime=now_time, use_status=1, orders_id=order_id, endTime=end_time)
  192. else:
  193. AiService.objects.create(
  194. uid=uid, channel=channel, detect_status=1, use_status=1, orders_id=order_id,
  195. addTime=now_time, updTime=now_time, endTime=end_time)
  196. return uid_bucket_id
  197. @classmethod
  198. def do_vod_msg_notice(cls, uid, user_id, lang, sys_msg_text_list):
  199. """
  200. 发送云存开通信息
  201. @param uid: uid
  202. @param user_id: 用户id
  203. @param lang: 语言
  204. @param sys_msg_text_list: 消息列表
  205. @return: response
  206. """
  207. if lang == 'cn':
  208. sys_msg_text = sys_msg_text_list[0]
  209. else:
  210. sys_msg_text = sys_msg_text_list[1]
  211. now_time = int(time.time())
  212. create_data = {
  213. 'userID_id': user_id,
  214. 'msg': sys_msg_text,
  215. 'addTime': now_time,
  216. 'updTime': now_time,
  217. 'uid': uid,
  218. 'eventType': 0
  219. }
  220. SysMsgModel.objects.create(**create_data)
  221. # 不接收邮件用户
  222. if user_id == '167015836969813800138000':
  223. return
  224. user_qs = Device_User.objects.filter(userID=user_id)
  225. if user_qs.exists():
  226. user = user_qs.first()
  227. username = user.username
  228. data_valid = DataValid()
  229. if data_valid.email_validate(username):
  230. S3Email().faEmail(sys_msg_text, username)
  231. elif data_valid.mobile_validate(username):
  232. # 如果存在序列号,消息提示用序列号
  233. device_info_qs = Device_Info.objects.filter(UID=uid).values('serial_number', 'Type')
  234. if device_info_qs.exists():
  235. serial_number = device_info_qs[0]['serial_number']
  236. device_type = device_info_qs[0]['Type']
  237. if serial_number:
  238. device_name = CommonService.get_full_serial_number(uid, serial_number, device_type)
  239. else:
  240. device_name = uid
  241. params = '{"devname":"%s","submittime":"%s"}' % (
  242. device_name, time.strftime("%Y-%m-%d", time.localtime()))
  243. cls.send_message(username, params, 'SMS_219738485')
  244. @staticmethod
  245. def send_message(phone, params, temp_msg):
  246. """
  247. 发送手机消息
  248. @param phone: 用户名
  249. @param params: 消息参数
  250. @param temp_msg: sms码
  251. """
  252. sign_ms = '周视'
  253. ali_sms = AliSmsObject()
  254. ali_sms.send_code_sms_cloud(phone=phone, params=params, sign_name=sign_ms, temp_msg=temp_msg)
  255. @classmethod
  256. def app_store_server_notifications(cls, request):
  257. logger = logging.getLogger('apple_pay')
  258. logger.info('App Store服务器通知请求类型:{}'.format(request.method))
  259. logger.info('App Store服务器通知参数:{}'.format(request.POST))
  260. logger.info('App Store服务器通知请求body:{}'.format(request.body))
  261. payload = json.loads(request.body.decode('utf-8'))
  262. logger.info('App Store服务器通知payload:{}'.format(payload))
  263. # 获取 signedPayload
  264. signed_payload = payload.get('signedPayload')
  265. if not signed_payload:
  266. return HttpResponse(status=400)
  267. bundle_id = 'com.ansjer.zccloud'
  268. environment = ENV
  269. root_certificates = []
  270. for cert_name in [
  271. 'AppleIncRootCertificate.cer', 'AppleComputerRootCertificate.cer',
  272. 'AppleRootCA-G2.cer', 'AppleRootCA-G3.cer'
  273. ]:
  274. cert_path = '{}/Ansjer/file/in_app_purchase/{}'.format(BASE_DIR, cert_name)
  275. with open(cert_path, 'rb') as file:
  276. # 读取文件内容
  277. root_certificates.append(file.read())
  278. enable_online_checks = True
  279. app_apple_id = 1355964934 # 生产环境必需
  280. # 验证签名并解码 payload
  281. verifier = SignedDataVerifier(
  282. root_certificates, enable_online_checks, environment, bundle_id, app_apple_id)
  283. decoded_payload = verifier.verify_and_decode_notification(signed_payload)
  284. logger.info('App Store服务器通知decoded_payload: {}'.format(decoded_payload))
  285. status_code = 200
  286. if str(decoded_payload.rawNotificationType) == "REFUND":
  287. # 一种通知类型,表示 App Store 成功退还了消耗性应用内购买、非消耗性应用内购买、自动续订或不可续订的交易。
  288. # revocationDate 包含退款交易的时间戳。originalTransactionId 和 productId 用于标识原始交易和产品。revocationReason 包含原因。
  289. # 要请求客户所有退款交易的列表,请参阅 App Store 服务器 API 中的获取退款历史记录。
  290. # 1. 找套餐 使用 transaction_id 找orders
  291. decoded_transaction_information = verifier.verify_and_decode_signed_transaction(
  292. decoded_payload.data.signedTransactionInfo)
  293. transaction_id = decoded_transaction_information.transactionId
  294. logger.info('App Store服务器通知退款, transaction_id:{}'.format(transaction_id))
  295. orders_qs = Order_Model.objects.filter(transaction_id=transaction_id)
  296. # 2. 查找云存套餐使用表 和 云存套餐
  297. if orders_qs.exists():
  298. orders_qs.update(status=11)
  299. orderID = orders_qs[0].orderID
  300. uid = orders_qs[0].UID
  301. user_id = orders_qs[0].userID
  302. # 3. 未使用则删除未使用套餐表,已使用过则删除设备正在使用套餐,并关闭设备云存
  303. uid_bucket_qs = UID_Bucket.objects.filter(uid=uid, orderId=orderID, use_status=1,
  304. endTime__gt=int(time.time()))
  305. unused_uid_meal_qs = Unused_Uid_Meal.objects.filter(order_id=orderID)
  306. ai_service_qs = AiService.objects.filter(uid=uid, orderId=orderID, use_status=1,
  307. endTime__gt=int(time.time()))
  308. if unused_uid_meal_qs.exists():
  309. unused_uid_meal_qs.delete()
  310. if uid_bucket_qs.exists():
  311. uid_bucket_qs.update(status=0, use_status=2, endTime=int(time.time()), updateTime=int(time.time()))
  312. if ai_service_qs.exists():
  313. ai_service_qs.update(detect_status=0, use_status=2, endTime=int(time.time()),
  314. updTime=int(time.time()))
  315. # 关闭ai
  316. msg = {'commandType': 'AIDisable'}
  317. thing_name = CommonService.query_serial_with_uid(uid) # 存在序列号则为使用序列号作为物品名
  318. topic_name = 'ansjer/generic/{}'.format(thing_name)
  319. req_success = CommonService.req_publish_mqtt_msg(thing_name, topic_name, msg)
  320. LOGGER.info(f'App Store服务器通知用户退款, 关闭AI:{req_success}')
  321. # 4.发送邮件告知用户退款
  322. email_content = f'{CONFIG_INFO}用户{user_id}, 订单:{orderID}, 设备{uid}退款'
  323. S3Email().faEmail(email_content, 'servers@ansjer.com')
  324. else:
  325. if CONFIG_INFO == CONFIG_US:
  326. url = "https://api.zositeche.com/inAppPurchase/AppStoreServerNotifications"
  327. eur_response = requests.post(url=url, json=json.loads(request.body))
  328. status_code = eur_response.status_code
  329. return HttpResponse(status=status_code)
  330. @classmethod
  331. def vsees_notifications(cls, request):
  332. logger = logging.getLogger('apple_pay')
  333. logger.info('Vsees: App Store服务器通知请求类型:{}'.format(request.method))
  334. logger.info('Vsees: App Store服务器通知参数:{}'.format(request.POST))
  335. logger.info('Vsees: App Store服务器通知请求body:{}'.format(request.body))
  336. payload = json.loads(request.body.decode('utf-8'))
  337. logger.info('Vsees: App Store服务器通知payload:{}'.format(payload))
  338. # 获取 signedPayload
  339. signed_payload = payload.get('signedPayload')
  340. if not signed_payload:
  341. return HttpResponse(status=400)
  342. in_app_purchase = InAppPurchase(bundle_id="com.cloudlife.commissionf")
  343. # SignedDataVerifier 用于解析查询到的交易信息
  344. verifier = in_app_purchase.verifier
  345. decoded_payload = verifier.verify_and_decode_notification(signed_payload)
  346. logger.info('Vsees: App Store服务器通知decoded_payload: {}'.format(decoded_payload))
  347. status_code = 200
  348. if str(decoded_payload.rawNotificationType) == "REFUND":
  349. # 一种通知类型,表示 App Store 成功退还了消耗性应用内购买、非消耗性应用内购买、自动续订或不可续订的交易。
  350. # revocationDate 包含退款交易的时间戳。originalTransactionId 和 productId 用于标识原始交易和产品。revocationReason 包含原因。
  351. # 要请求客户所有退款交易的列表,请参阅 App Store 服务器 API 中的获取退款历史记录。
  352. # 1. 找套餐 使用 transaction_id 找orders
  353. decoded_transaction_information = verifier.verify_and_decode_signed_transaction(
  354. decoded_payload.data.signedTransactionInfo)
  355. transaction_id = decoded_transaction_information.transactionId
  356. logger.info('Vsees: App Store服务器通知退款, transaction_id:{}'.format(transaction_id))
  357. orders_qs = Order_Model.objects.filter(transaction_id=transaction_id)
  358. # 2. 查找云存套餐使用表 和 云存套餐
  359. if orders_qs.exists():
  360. orders_qs.update(status=11)
  361. orderID = orders_qs[0].orderID
  362. uid = orders_qs[0].UID
  363. user_id = orders_qs[0].userID
  364. # 3. 未使用则删除未使用套餐表,已使用过则删除设备正在使用套餐,并关闭设备云存
  365. uid_bucket_qs = UID_Bucket.objects.filter(uid=uid, orderId=orderID, use_status=1,
  366. endTime__gt=int(time.time()))
  367. unused_uid_meal_qs = Unused_Uid_Meal.objects.filter(order_id=orderID)
  368. ai_service_qs = AiService.objects.filter(uid=uid, orderId=orderID, use_status=1,
  369. endTime__gt=int(time.time()))
  370. if unused_uid_meal_qs.exists():
  371. unused_uid_meal_qs.delete()
  372. if uid_bucket_qs.exists():
  373. uid_bucket_qs.update(status=0, use_status=2, endTime=int(time.time()), updateTime=int(time.time()))
  374. if ai_service_qs.exists():
  375. ai_service_qs.update(detect_status=0, use_status=2, endTime=int(time.time()),
  376. updTime=int(time.time()))
  377. # 关闭ai
  378. msg = {'commandType': 'AIDisable'}
  379. thing_name = CommonService.query_serial_with_uid(uid) # 存在序列号则为使用序列号作为物品名
  380. topic_name = 'ansjer/generic/{}'.format(thing_name)
  381. req_success = CommonService.req_publish_mqtt_msg(thing_name, topic_name, msg)
  382. LOGGER.info(f'App Store服务器通知用户退款, 关闭AI:{req_success}')
  383. # 4.发送邮件告知用户退款
  384. email_content = f'{CONFIG_INFO}用户{user_id}, 订单:{orderID}, 设备{uid}退款'
  385. S3Email().faEmail(email_content, 'servers@ansjer.com')
  386. else:
  387. if CONFIG_INFO == CONFIG_US:
  388. url = "https://api.zositeche.com/inAppPurchase/AppStoreServerNotifications"
  389. eur_response = requests.post(url=url, json=json.loads(request.body))
  390. status_code = eur_response.status_code
  391. return HttpResponse(status=status_code)