InAppPurchaseController.py 38 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670
  1. # @Author : Rocky
  2. # @File : InAppPurchaseController.py
  3. # @Time : 2024/6/21 9:10
  4. import logging
  5. import time
  6. import json
  7. from appstoreserverlibrary.api_client import AppStoreServerAPIClient, GetTransactionHistoryVersion
  8. from appstoreserverlibrary.models.Environment import Environment
  9. from appstoreserverlibrary.receipt_utility import ReceiptUtility
  10. from appstoreserverlibrary.models.HistoryResponse import HistoryResponse
  11. from appstoreserverlibrary.models.TransactionHistoryRequest import TransactionHistoryRequest, ProductType, Order
  12. from appstoreserverlibrary.signed_data_verifier import SignedDataVerifier
  13. from cryptography.hazmat.backends import default_backend
  14. from cryptography.hazmat.primitives.serialization import load_pem_private_key
  15. from django.db.models import Q
  16. from django.views import View
  17. from django.http import HttpResponse
  18. from Ansjer.config import LOGGER, CONFIG_INFO, CONFIG_TEST, PAY_TYPE_IN_APP_PURCHASE, BASE_DIR
  19. from Controller.CheckUserData import DataValid
  20. from Model.models import Order_Model, Store_Meal, Device_Info, UID_Bucket, Unused_Uid_Meal, AiService, Device_User, \
  21. SysMsgModel, DeviceApplePackage, InAppPurchasePackage
  22. from Object.AWS.S3Email import S3Email
  23. from Object.AliSmsObject import AliSmsObject
  24. from Object.RedisObject import RedisObject
  25. from Service.CommonService import CommonService
  26. ENV = Environment.SANDBOX if CONFIG_INFO == CONFIG_TEST else Environment.PRODUCTION
  27. class InAppPurchaseView(View):
  28. def get(self, request, *args, **kwargs):
  29. request.encoding = 'utf-8'
  30. operation = kwargs.get('operation')
  31. return self.validation(request.GET, request, operation)
  32. def post(self, request, *args, **kwargs):
  33. request.encoding = 'utf-8'
  34. operation = kwargs.get('operation')
  35. return self.validation(request.POST, request, operation)
  36. def validation(self, request_dict, request, operation):
  37. if operation == 'AppStoreServerNotifications': # App Store服务器通知
  38. return self.app_store_server_notifications(request, request_dict)
  39. token_code, user_id, response = CommonService.verify_token_get_user_id(request_dict, request)
  40. if token_code != 0:
  41. return response.json(token_code)
  42. if operation == 'verifyTransaction': # 认证交易
  43. return self.verify_transaction(user_id, request_dict, response)
  44. @classmethod
  45. def verify_transaction(cls, user_id, request_dict, response):
  46. """
  47. 认证交易
  48. @param user_id: 用户id
  49. @param request_dict: 请求参数
  50. @request_dict receipt: 收据
  51. @param response: 响应对象
  52. @return: response
  53. """
  54. receipt = request_dict.get('receipt', None)
  55. transaction_identifier = request_dict.get('transactionIdentifier', None)
  56. original_transaction_identifier = request_dict.get('originalTransactionIdentifier', None)
  57. order_id = request_dict.get('orderID', None)
  58. uid = request_dict.get('uid', None)
  59. lang = request_dict.get('lang', 'en')
  60. channel = request_dict.get('channel', None)
  61. logger = logging.getLogger('apple_pay')
  62. logger.info(
  63. f"transactionIdentifier:{transaction_identifier}, original_transaction_identifier:{original_transaction_identifier} ,订单orderId:{order_id}")
  64. if not all([transaction_identifier, uid, channel, order_id]):
  65. return response.json(444)
  66. # redis加锁,防止订单重复
  67. redis_obj = RedisObject()
  68. redis_key = order_id + 'in_app_purchase'
  69. is_lock = redis_obj.CONN.setnx(redis_key, 1)
  70. redis_obj.CONN.expire(redis_key, 60)
  71. # if not is_lock:
  72. # return response.json(5)
  73. try:
  74. # 从交易信息中获取product_id
  75. key_path = '{}/Ansjer/file/in_app_purchase/SubscriptionKey_N42WMFCV6A.p8'.format(BASE_DIR)
  76. with open(key_path, 'rb') as file:
  77. # 读取文件内容
  78. private_key = file.read()
  79. key_id = 'N42WMFCV6A'
  80. issuer_id = '69a6de8c-789b-47e3-e053-5b8c7c11a4d1'
  81. bundle_id = 'com.ansjer.zccloud'
  82. environment = ENV
  83. client = AppStoreServerAPIClient(private_key, key_id, issuer_id, bundle_id, environment)
  84. transaction_id = transaction_identifier
  85. if transaction_id is None:
  86. pay_result_url = CommonService.get_payment_status_url(lang, 'fail')
  87. return response.json(0, {'url': pay_result_url})
  88. transaction_info = client.get_transaction_info(transaction_id)
  89. signed_transaction_info = transaction_info.signedTransactionInfo
  90. if original_transaction_identifier != "":
  91. device_apple_package_qs = DeviceApplePackage.objects.filter(
  92. original_transaction_id=original_transaction_identifier)
  93. if device_apple_package_qs.exists():
  94. # 第一种情况: 套餐已过期再次订阅
  95. if device_apple_package_qs[0].uid == uid and device_apple_package_qs[0].subscription_status == 2:
  96. # 使用App Store服务器通知接口订阅
  97. pay_result_url = CommonService.get_payment_status_url(lang, 'success')
  98. return response.json(0, {'url': pay_result_url})
  99. # 第二种情况: 套餐未过期已取消再次订阅
  100. elif device_apple_package_qs[0].uid == uid and device_apple_package_qs[0].subscription_status == 3:
  101. # 使用App Store服务器通知接口修改订阅状态
  102. pay_result_url = CommonService.get_payment_status_url(lang, 'fail')
  103. return response.json(0, {'url': pay_result_url})
  104. # 第三种情况: 首次订阅
  105. elif device_apple_package_qs[0].uid == uid and device_apple_package_qs[0].subscription_status == 0:
  106. logger.info(f"首次订阅直接充值,orderID:{order_id}")
  107. else:
  108. logger.info(f"错误调用此借口,orderID:{order_id}, uid:{uid}, 订阅状态:{device_apple_package_qs[0].subscription_status}")
  109. pay_result_url = CommonService.get_payment_status_url(lang, 'fail')
  110. return response.json(0, {'url': pay_result_url})
  111. # transaction_id相同的情况 ---- 本次订阅未过期,用户在苹果设置中将订阅重新打开时会传上次订阅相同的 transaction_id。
  112. if Order_Model.objects.filter(payType=5, transaction_id=transaction_id).exists():
  113. logger.info(f"该transaction_id已订阅过:{transaction_id}")
  114. return response.json(10048)
  115. root_certificates = []
  116. for cert_name in [
  117. 'AppleIncRootCertificate.cer', 'AppleComputerRootCertificate.cer',
  118. 'AppleRootCA-G2.cer', 'AppleRootCA-G3.cer'
  119. ]:
  120. cert_path = '{}/Ansjer/file/in_app_purchase/{}'.format(BASE_DIR, cert_name)
  121. with open(cert_path, 'rb') as file:
  122. # 读取文件内容
  123. root_certificates.append(file.read())
  124. enable_online_checks = True
  125. app_apple_id = None # 生产环境必需
  126. signed_data_verifier = SignedDataVerifier(
  127. root_certificates, enable_online_checks, environment, bundle_id, app_apple_id)
  128. payload = signed_data_verifier.verify_and_decode_signed_transaction(signed_transaction_info)
  129. product_id = None
  130. original_transaction_id = ""
  131. if payload and payload.productId:
  132. product_id = payload.productId
  133. if not product_id:
  134. pay_result_url = CommonService.get_payment_status_url(lang, 'fail')
  135. return response.json(0, {'url': pay_result_url})
  136. if payload.rawType == "Auto-Renewable Subscription":
  137. original_transaction_id = payload.originalTransactionId
  138. now_time = int(time.time())
  139. order_qs = Order_Model.objects.filter(orderID=order_id, UID=uid).values("rank_id")
  140. if not order_qs.exists():
  141. return response.json(173, "订单不存在")
  142. store_qs = Store_Meal.objects.filter(id=order_qs[0]['rank_id']).values(
  143. 'id', 'currency', 'price', 'lang__content', 'day', 'commodity_type', 'lang__title', 'expire',
  144. 'commodity_code', 'discount_price', 'bucket_id', 'bucket__mold', 'cycle_config_id', 'is_ai')
  145. if not store_qs.exists():
  146. return response.json(173, "套餐不存在")
  147. bucket_id = store_qs[0]['bucket_id']
  148. is_ai = store_qs[0]['is_ai']
  149. expire = store_qs[0]['expire']
  150. end_time = CommonService.calcMonthLater(expire)
  151. # 查询设备是否已开过云存
  152. use_flag = True
  153. uid_bucket_qs = UID_Bucket.objects.filter(uid=uid). \
  154. values('id', 'bucket_id', 'bucket__region', 'endTime', 'use_status')
  155. if uid_bucket_qs.exists():
  156. uid_bucket = uid_bucket_qs.first()
  157. uid_bucket_id = uid_bucket['id']
  158. # 叠加相同套餐的过期时间
  159. if uid_bucket['use_status'] == 1 and uid_bucket['endTime'] > now_time:
  160. Unused_Uid_Meal.objects.create(
  161. uid=uid, channel=channel, addTime=now_time, order_id=order_id, expire=expire, is_ai=is_ai,
  162. bucket_id=bucket_id)
  163. UID_Bucket.objects.filter(id=uid_bucket_id).update(has_unused=1)
  164. use_flag = False
  165. # 更新套餐的过期时间
  166. else:
  167. UID_Bucket.objects.filter(id=uid_bucket_id).update(
  168. channel=channel, bucket_id=bucket_id, endTime=end_time, updateTime=now_time, use_status=1,
  169. orderId=order_id)
  170. else:
  171. uid_bucket = UID_Bucket.objects.create(
  172. uid=uid, channel=channel, bucket_id=bucket_id, endTime=end_time, use_status=1, orderId=order_id,
  173. addTime=now_time, updateTime=now_time)
  174. uid_bucket_id = uid_bucket.id
  175. # 开通AI服务
  176. if is_ai and use_flag:
  177. ai_service = AiService.objects.filter(uid=uid, channel=channel)
  178. # 有正在使用的套餐,叠加套餐时间,否则创建
  179. if ai_service.exists():
  180. ai_service.update(updTime=now_time, use_status=1, orders_id=order_id, endTime=end_time)
  181. else:
  182. AiService.objects.create(
  183. uid=uid, channel=channel, detect_status=1, use_status=1, orders_id=order_id,
  184. addTime=now_time, updTime=now_time, endTime=end_time)
  185. # 修改订阅状态
  186. if payload.rawType == "Auto-Renewable Subscription":
  187. in_app_purchase_package_qs = InAppPurchasePackage.objects.filter(product_id=product_id)
  188. if not in_app_purchase_package_qs.exists():
  189. return response.json(173, "内购套餐不存在")
  190. in_app_purchase_package = in_app_purchase_package_qs.values('id').first()
  191. package_id = in_app_purchase_package['id']
  192. DeviceApplePackage.objects.filter(userID=user_id, uid=uid, package_id=package_id).update(
  193. subscription_status=1,
  194. original_transaction_id=original_transaction_id,
  195. update_time=int(time.time())
  196. )
  197. order_qs.update(status=1, uid_bucket_id=uid_bucket_id,
  198. transaction_id=transaction_id, create_vod=1,
  199. original_transaction_id=original_transaction_id)
  200. # 发送云存开通信息
  201. date_time = time.strftime("%Y-%m-%d", time.localtime())
  202. # 如果存在序列号,消息提示用序列号
  203. device_info_qs = Device_Info.objects.filter(UID=uid).values('serial_number', 'Type')
  204. serial_number = device_info_qs[0]['serial_number']
  205. device_type = device_info_qs[0]['Type']
  206. if serial_number:
  207. device_name = CommonService.get_full_serial_number(uid, serial_number, device_type)
  208. else:
  209. device_name = uid
  210. sys_msg_text_list = [
  211. '温馨提示:尊敬的客户,您的{}设备在{}已成功购买云存套餐'.format(device_name, date_time),
  212. 'Dear customer,you already subscribed the cloud storage package successfully for device {} on '.
  213. format(device_name, time.strftime('%b %dth,%Y', time.localtime()))]
  214. cls.do_vod_msg_notice(uid, user_id, lang, sys_msg_text_list)
  215. redis_obj.del_data(redis_key)
  216. pay_result_url = CommonService.get_payment_status_url(lang, 'success')
  217. return response.json(0, {'url': pay_result_url})
  218. except Exception as e:
  219. redis_obj.del_data(redis_key)
  220. LOGGER.info('苹果内购认证交易接口异常:{}'.
  221. format('error_line:{}, error_msg:{}'.format(e.__traceback__.tb_lineno, repr(e))))
  222. pay_result_url = CommonService.get_payment_status_url(lang, 'fail')
  223. return response.json(0, {'url': pay_result_url})
  224. @classmethod
  225. def do_vod_msg_notice(cls, uid, user_id, lang, sys_msg_text_list):
  226. """
  227. 发送云存开通信息
  228. @param uid: uid
  229. @param user_id: 用户id
  230. @param lang: 语言
  231. @param sys_msg_text_list: 消息列表
  232. @return: response
  233. """
  234. if lang == 'cn':
  235. sys_msg_text = sys_msg_text_list[0]
  236. else:
  237. sys_msg_text = sys_msg_text_list[1]
  238. now_time = int(time.time())
  239. create_data = {
  240. 'userID_id': user_id,
  241. 'msg': sys_msg_text,
  242. 'addTime': now_time,
  243. 'updTime': now_time,
  244. 'uid': uid,
  245. 'eventType': 0
  246. }
  247. SysMsgModel.objects.create(**create_data)
  248. # 不接收邮件用户
  249. if user_id == '167015836969813800138000':
  250. return
  251. user_qs = Device_User.objects.filter(userID=user_id)
  252. if user_qs.exists():
  253. user = user_qs.first()
  254. username = user.username
  255. data_valid = DataValid()
  256. if data_valid.email_validate(username):
  257. S3Email().faEmail(sys_msg_text, username)
  258. elif data_valid.mobile_validate(username):
  259. # 如果存在序列号,消息提示用序列号
  260. device_info_qs = Device_Info.objects.filter(UID=uid).values('serial_number', 'Type')
  261. if device_info_qs.exists():
  262. serial_number = device_info_qs[0]['serial_number']
  263. device_type = device_info_qs[0]['Type']
  264. if serial_number:
  265. device_name = CommonService.get_full_serial_number(uid, serial_number, device_type)
  266. else:
  267. device_name = uid
  268. params = '{"devname":"%s","submittime":"%s"}' % (
  269. device_name, time.strftime("%Y-%m-%d", time.localtime()))
  270. cls.send_message(username, params, 'SMS_219738485')
  271. @staticmethod
  272. def send_message(phone, params, temp_msg):
  273. """
  274. 发送手机消息
  275. @param phone: 用户名
  276. @param params: 消息参数
  277. @param temp_msg: sms码
  278. """
  279. sign_ms = '周视'
  280. ali_sms = AliSmsObject()
  281. ali_sms.send_code_sms_cloud(phone=phone, params=params, sign_name=sign_ms, temp_msg=temp_msg)
  282. @classmethod
  283. def app_store_server_notifications(cls, request, request_dict):
  284. logger = logging.getLogger('apple_pay')
  285. try:
  286. logger.info('App Store服务器通知请求类型:{}'.format(request.method))
  287. logger.info('App Store服务器通知参数:{}'.format(request.POST))
  288. logger.info('App Store服务器通知请求body:{}'.format(request.body))
  289. if request.method != 'POST':
  290. logger.info(f'App Store服务器通知不是post请求, 参数{request_dict}')
  291. return HttpResponse(status=400)
  292. payload = json.loads(request.body.decode('utf-8'))
  293. logger.info('App Store服务器通知payload:{}'.format(payload))
  294. # 获取 signedPayload
  295. signed_payload = payload.get('signedPayload')
  296. if not signed_payload:
  297. return HttpResponse(status=400)
  298. bundle_id = 'com.ansjer.zccloud'
  299. environment = ENV
  300. root_certificates = []
  301. for cert_name in [
  302. 'AppleIncRootCertificate.cer', 'AppleComputerRootCertificate.cer',
  303. 'AppleRootCA-G2.cer', 'AppleRootCA-G3.cer'
  304. ]:
  305. cert_path = '{}/Ansjer/file/in_app_purchase/{}'.format(BASE_DIR, cert_name)
  306. with open(cert_path, 'rb') as file:
  307. # 读取文件内容
  308. root_certificates.append(file.read())
  309. enable_online_checks = True
  310. app_apple_id = None # 生产环境必需
  311. # 验证签名并解码 payload
  312. verifier = SignedDataVerifier(
  313. root_certificates, enable_online_checks, environment, bundle_id, app_apple_id)
  314. decoded_payload = verifier.verify_and_decode_notification(signed_payload)
  315. logger.info(f"App Store服务器通知解码后decoded_payload:{decoded_payload}")
  316. logger.info(
  317. f"App Store服务器通知decoded_payload.rawNotificationType{str(decoded_payload.rawNotificationType)}")
  318. if str(decoded_payload.rawNotificationType) == "DID_RENEW":
  319. decoded_transaction_information = verifier.verify_and_decode_signed_transaction(
  320. decoded_payload.data.signedTransactionInfo)
  321. # originalTransactionId 原始购买的交易标识符
  322. original_transaction_id = decoded_transaction_information.originalTransactionId
  323. transaction_id = decoded_transaction_information.transactionId
  324. logger.info(
  325. f"App Store服务器通知originalTransactionId原始购买的交易标识符{original_transaction_id}")
  326. if not original_transaction_id:
  327. logger.info(f"App Store服务器通知originalTransactionId原始购买的交易标识符为空, 返回状态 400")
  328. return HttpResponse(status=400)
  329. else:
  330. ord_order = Order_Model.objects.filter(original_transaction_id=original_transaction_id).order_by(
  331. '-addTime').values("channel", "UID", "payType", "userID_id", "rank_id")
  332. if not ord_order.exists():
  333. logger.info(f"App Store服务器通知未查询到旧订单信息, 返回状态 400")
  334. return HttpResponse(status=400)
  335. channel = ord_order[0]["channel"]
  336. uid = ord_order[0]["UID"]
  337. pay_type = ord_order[0]["payType"]
  338. user_id = ord_order[0]["userID_id"]
  339. store_qs = Store_Meal.objects.filter(id=ord_order[0]["rank_id"]). \
  340. values(
  341. 'id', 'currency', 'price', 'lang__content', 'day', 'commodity_type', 'lang__title',
  342. 'expire', 'lang__lang',
  343. 'commodity_code', 'discount_price', 'bucket_id', 'bucket__mold', 'cycle_config_id', 'is_ai')
  344. if not store_qs.exists():
  345. logger.info(f"App Store服务器通知云存套餐不存在, 返回状态 400")
  346. return HttpResponse(status=400)
  347. order_id = CommonService.createOrderID()
  348. rank_id = store_qs[0]['id']
  349. bucket_id = store_qs[0]['bucket_id']
  350. currency = store_qs[0]['currency']
  351. price = store_qs[0]['price']
  352. is_ai = store_qs[0]['is_ai']
  353. expire = store_qs[0]['expire']
  354. end_time = CommonService.calcMonthLater(expire)
  355. content = store_qs[0]['lang__content']
  356. commodity_code = store_qs[0]['commodity_code']
  357. commodity_type = store_qs[0]['commodity_type']
  358. lang = store_qs[0]['lang__lang']
  359. order_type = 1 if is_ai else 0
  360. store_meal_qs = Store_Meal.objects.filter(id=rank_id, lang__lang='cn', is_show=0). \
  361. values('lang__title', 'lang__content')
  362. if store_meal_qs.exists():
  363. store_meal_name = store_meal_qs[0]['lang__title'] + '-' + store_meal_qs[0]['lang__content']
  364. else:
  365. store_meal_name = '未知套餐'
  366. # 查询设备是否已开过云存
  367. use_flag = True
  368. uid_bucket_qs = UID_Bucket.objects.filter(uid=uid). \
  369. values('id', 'bucket_id', 'bucket__region', 'endTime', 'use_status')
  370. now_time = int(time.time())
  371. if uid_bucket_qs.exists():
  372. uid_bucket = uid_bucket_qs.first()
  373. uid_bucket_id = uid_bucket['id']
  374. # 叠加相同套餐的过期时间
  375. if uid_bucket['use_status'] == 1 and uid_bucket['endTime'] > now_time:
  376. Unused_Uid_Meal.objects.create(
  377. uid=uid, channel=channel, addTime=now_time, order_id=order_id, expire=expire,
  378. is_ai=is_ai,
  379. bucket_id=bucket_id)
  380. UID_Bucket.objects.filter(id=uid_bucket_id).update(has_unused=1)
  381. use_flag = False
  382. # 更新套餐的过期时间
  383. else:
  384. UID_Bucket.objects.filter(id=uid_bucket_id).update(
  385. channel=channel, bucket_id=bucket_id, endTime=end_time, updateTime=now_time,
  386. use_status=1,
  387. orderId=order_id)
  388. else:
  389. uid_bucket = UID_Bucket.objects.create(
  390. uid=uid, channel=channel, bucket_id=bucket_id, endTime=end_time, use_status=1,
  391. orderId=order_id,
  392. addTime=now_time, updateTime=now_time)
  393. uid_bucket_id = uid_bucket.id
  394. # 开通AI服务
  395. if is_ai and use_flag:
  396. ai_service = AiService.objects.filter(uid=uid, channel=channel)
  397. # 有正在使用的套餐,叠加套餐时间,否则创建
  398. if ai_service.exists():
  399. ai_service.update(updTime=now_time, use_status=1, orders_id=order_id,
  400. endTime=end_time)
  401. else:
  402. AiService.objects.create(
  403. uid=uid, channel=channel, detect_status=1, use_status=1, orders_id=order_id,
  404. addTime=now_time, updTime=now_time, endTime=end_time)
  405. Order_Model.objects.create(
  406. orderID=order_id, UID=uid, channel=channel, userID_id=user_id, desc=content,
  407. payType=pay_type,
  408. payTime=now_time, price=price, currency=currency, addTime=now_time, updTime=now_time,
  409. order_type=order_type, commodity_code=commodity_code, commodity_type=commodity_type,
  410. rank_id=rank_id,
  411. ai_rank_id=1, status=1, create_vod=1, store_meal_name=store_meal_name,
  412. uid_bucket_id=uid_bucket_id, transaction_id=transaction_id,
  413. original_transaction_id=original_transaction_id
  414. )
  415. # 发送云存开通信息
  416. date_time = time.strftime("%Y-%m-%d", time.localtime())
  417. # 如果存在序列号,消息提示用序列号
  418. device_info_qs = Device_Info.objects.filter(UID=uid).values('serial_number', 'Type')
  419. serial_number = device_info_qs[0]['serial_number']
  420. device_type = device_info_qs[0]['Type']
  421. if serial_number:
  422. device_name = CommonService.get_full_serial_number(uid, serial_number, device_type)
  423. else:
  424. device_name = uid
  425. sys_msg_text_list = [
  426. '温馨提示:尊敬的客户,您的{}设备在{}已成功续订云存套餐'.format(device_name, date_time),
  427. 'Dear customer,you already subscribed the cloud storage package successfully for device {} on '.
  428. format(device_name, time.strftime('%b %dth,%Y', time.localtime()))]
  429. cls.do_vod_msg_notice(uid, user_id, lang, sys_msg_text_list)
  430. elif str(decoded_payload.rawNotificationType) == "SUBSCRIBED":
  431. # 处理订阅 ---> 首次充值逻辑写在了认证交易
  432. if decoded_payload.rawSubtype == "RESUBSCRIBE":
  433. decoded_transaction_information = verifier.verify_and_decode_signed_transaction(
  434. decoded_payload.data.signedTransactionInfo)
  435. # originalTransactionId 原始购买的交易标识符
  436. original_transaction_id = decoded_transaction_information.originalTransactionId
  437. transaction_id = decoded_transaction_information.transactionId
  438. logger.info(
  439. f"App Store服务器通知originalTransactionId原始购买的交易标识符{original_transaction_id}")
  440. if not original_transaction_id:
  441. logger.info(f"App Store服务器通知originalTransactionId原始购买的交易标识符为空, 返回状态 400")
  442. return HttpResponse(status=400)
  443. else:
  444. ord_order = Order_Model.objects.filter(
  445. original_transaction_id=original_transaction_id).order_by('-addTime').values("channel",
  446. "UID",
  447. "payType",
  448. "userID_id")
  449. channel = ord_order[0]["channel"]
  450. uid = ord_order[0]["UID"]
  451. pay_type = ord_order[0]["payType"]
  452. user_id = ord_order[0]["userID_id"]
  453. # 用产品id找到使用的套餐
  454. product_id = decoded_transaction_information.productId
  455. rank_id = InAppPurchasePackage.objects.filter(product_id=product_id).values("rank")[0]["rank"]
  456. store_qs = Store_Meal.objects.filter(id=rank_id). \
  457. values(
  458. 'id', 'currency', 'price', 'lang__content', 'day', 'commodity_type', 'lang__title',
  459. 'expire', 'lang__lang',
  460. 'commodity_code', 'discount_price', 'bucket_id', 'bucket__mold', 'cycle_config_id', 'is_ai')
  461. if not store_qs.exists():
  462. logger.info(f"App Store服务器通知云存套餐不存在, 返回状态 400")
  463. return HttpResponse(status=400)
  464. order_id = CommonService.createOrderID()
  465. rank_id = store_qs[0]['id']
  466. bucket_id = store_qs[0]['bucket_id']
  467. currency = store_qs[0]['currency']
  468. price = store_qs[0]['price']
  469. is_ai = store_qs[0]['is_ai']
  470. expire = store_qs[0]['expire']
  471. end_time = CommonService.calcMonthLater(expire)
  472. content = store_qs[0]['lang__content']
  473. commodity_code = store_qs[0]['commodity_code']
  474. commodity_type = store_qs[0]['commodity_type']
  475. lang = store_qs[0]['lang__lang']
  476. order_type = 1 if is_ai else 0
  477. store_meal_qs = Store_Meal.objects.filter(id=rank_id, lang__lang='cn', is_show=0). \
  478. values('lang__title', 'lang__content')
  479. if store_meal_qs.exists():
  480. store_meal_name = store_meal_qs[0]['lang__title'] + '-' + store_meal_qs[0]['lang__content']
  481. else:
  482. store_meal_name = '未知套餐'
  483. # 查询设备是否已开过云存
  484. use_flag = True
  485. uid_bucket_qs = UID_Bucket.objects.filter(uid=uid). \
  486. values('id', 'bucket_id', 'bucket__region', 'endTime', 'use_status')
  487. now_time = int(time.time())
  488. if uid_bucket_qs.exists():
  489. uid_bucket = uid_bucket_qs.first()
  490. uid_bucket_id = uid_bucket['id']
  491. # 叠加相同套餐的过期时间
  492. if uid_bucket['use_status'] == 1 and uid_bucket['endTime'] > now_time:
  493. Unused_Uid_Meal.objects.create(
  494. uid=uid, channel=channel, addTime=now_time, order_id=order_id, expire=expire,
  495. is_ai=is_ai,
  496. bucket_id=bucket_id)
  497. UID_Bucket.objects.filter(id=uid_bucket_id).update(has_unused=1)
  498. use_flag = False
  499. # 更新套餐的过期时间
  500. else:
  501. UID_Bucket.objects.filter(id=uid_bucket_id).update(
  502. channel=channel, bucket_id=bucket_id, endTime=end_time, updateTime=now_time,
  503. use_status=1,
  504. orderId=order_id)
  505. else:
  506. uid_bucket = UID_Bucket.objects.create(
  507. uid=uid, channel=channel, bucket_id=bucket_id, endTime=end_time, use_status=1,
  508. orderId=order_id,
  509. addTime=now_time, updateTime=now_time)
  510. uid_bucket_id = uid_bucket.id
  511. # 开通AI服务
  512. if is_ai and use_flag:
  513. ai_service = AiService.objects.filter(uid=uid, channel=channel)
  514. # 有正在使用的套餐,叠加套餐时间,否则创建
  515. if ai_service.exists():
  516. ai_service.update(updTime=now_time, use_status=1, orders_id=order_id,
  517. endTime=end_time)
  518. else:
  519. AiService.objects.create(
  520. uid=uid, channel=channel, detect_status=1, use_status=1, orders_id=order_id,
  521. addTime=now_time, updTime=now_time, endTime=end_time)
  522. DeviceApplePackage.objects.filter(userID=user_id, uid=uid).update(subscription_status=1,
  523. update_time=int(time.time()))
  524. Order_Model.objects.create(
  525. orderID=order_id, UID=uid, channel=channel, userID_id=user_id, desc=content,
  526. payType=pay_type,
  527. payTime=now_time, price=price, currency=currency, addTime=now_time, updTime=now_time,
  528. order_type=order_type, commodity_code=commodity_code, commodity_type=commodity_type,
  529. rank_id=rank_id,
  530. ai_rank_id=1, status=1, create_vod=1, store_meal_name=store_meal_name,
  531. uid_bucket_id=uid_bucket_id, transaction_id=transaction_id,
  532. original_transaction_id=original_transaction_id
  533. )
  534. # 发送云存开通信息
  535. date_time = time.strftime("%Y-%m-%d", time.localtime())
  536. # 如果存在序列号,消息提示用序列号
  537. device_info_qs = Device_Info.objects.filter(UID=uid).values('serial_number', 'Type')
  538. serial_number = device_info_qs[0]['serial_number']
  539. device_type = device_info_qs[0]['Type']
  540. if serial_number:
  541. device_name = CommonService.get_full_serial_number(uid, serial_number, device_type)
  542. else:
  543. device_name = uid
  544. sys_msg_text_list = [
  545. '温馨提示:尊敬的客户,您的{}设备在{}已成功续订云存套餐'.format(device_name, date_time),
  546. 'Dear customer,you already subscribed the cloud storage package successfully for device {} on '.
  547. format(device_name, time.strftime('%b %dth,%Y', time.localtime()))]
  548. cls.do_vod_msg_notice(uid, user_id, lang, sys_msg_text_list)
  549. elif str(decoded_payload.rawNotificationType) == "EXPIRED":
  550. # 一种通知类型,与其子类型一起表示订阅已过期。如果subtype为
  551. # VOLUNTARY(自愿),则表示订阅在用户禁用订阅续订后过期。如果subtype是
  552. # BILLING_RETRY(计费重试),则表示订阅过期,因为计费重试期结束时没有成功的计费交易。如果subtype为
  553. # PRICE_INCREASE,则表示订阅已过期,因为客户不同意需要客户同意的价格上涨。如果subtype为
  554. # PRODUCT_NOT_FOR_SALE,则表示订阅已过期,因为在订阅尝试续订时,产品已不可购买。
  555. # 没有子类型的通知表示订阅因其他原因过期。
  556. decoded_transaction_information = verifier.verify_and_decode_signed_transaction(
  557. decoded_payload.data.signedTransactionInfo)
  558. # originalTransactionId 原始购买的交易标识符
  559. original_transaction_id = decoded_transaction_information.originalTransactionId
  560. if original_transaction_id:
  561. DeviceApplePackage.objects.filter(original_transaction_id=original_transaction_id).update(
  562. subscription_status=2, update_time=int(time.time()))
  563. elif str(decoded_payload.rawNotificationType) == "CONSUMPTION_REQUEST":
  564. # 一种通知类型,指示客户发起了消费型 App 内购买项目或自动续期订阅的退款请求,并且 App Store 要求您提供消费数据。有关详细信息,请参阅发送消耗信息。
  565. pass
  566. elif str(decoded_payload.rawNotificationType) == "DID_CHANGE_RENEWAL_STATUS":
  567. if decoded_payload.rawSubtype == "AUTO_RENEW_DISABLED":
  568. # 自动续订被禁用
  569. decoded_transaction_information = verifier.verify_and_decode_signed_transaction(
  570. decoded_payload.data.signedTransactionInfo)
  571. original_transaction_id = decoded_transaction_information.originalTransactionId
  572. if original_transaction_id:
  573. DeviceApplePackage.objects.filter(original_transaction_id=original_transaction_id).update(
  574. subscription_status=3, update_time=int(time.time()))
  575. elif decoded_payload.rawSubtype == "AUTO_RENEW_ENABLED":
  576. # 自动续订被开启
  577. decoded_transaction_information = verifier.verify_and_decode_signed_transaction(
  578. decoded_payload.data.signedTransactionInfo)
  579. original_transaction_id = decoded_transaction_information.originalTransactionId
  580. if original_transaction_id:
  581. DeviceApplePackage.objects.filter(original_transaction_id=original_transaction_id).update(
  582. subscription_status=1, update_time=int(time.time()))
  583. elif str(decoded_payload.rawNotificationType) == "REFUND":
  584. # 一种通知类型,表示 App Store 成功退还了消耗性应用内购买、非消耗性应用内购买、自动续订或不可续订的交易。
  585. # revocationDate 包含退款交易的时间戳。originalTransactionId 和 productId 用于标识原始交易和产品。revocationReason 包含原因。
  586. # 要请求客户所有退款交易的列表,请参阅 App Store 服务器 API 中的获取退款历史记录。
  587. pass
  588. elif str(decoded_payload.rawNotificationType) == "REFUND_DECLINED":
  589. # 一种通知类型,表示 App Store 由于客户提出的争议而撤销了先前批准的退款。如果您的应用程序因相关退款而撤销了内容或服务,则需要恢复这些内容或服务。
  590. # 此通知类型可适用于任何应用程序内购买类型:消耗品、非消耗品、不可续订订阅和自动续订订阅。对于自动续订,当 App Store 撤销退款时,续订日期保持不变。
  591. pass
  592. else:
  593. logger.info(f"App Store服务器通知decoded_payload.rawNotificationType 未处理")
  594. return HttpResponse(status=500)
  595. return HttpResponse(status=200)
  596. except Exception as e:
  597. logger.info('App Store服务器通知异常:{}'.
  598. format('error_line:{}, error_msg:{}'.format(e.__traceback__.tb_lineno, repr(e))))
  599. return HttpResponse(status=500)