InAppPurchaseController.py 27 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517
  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
  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)
  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. uid = request_dict.get('uid', None)
  56. lang = request_dict.get('lang', 'en')
  57. channel = request_dict.get('channel', None)
  58. logger = logging.getLogger('apple_pay')
  59. logger.info(f"认证交易receipt:{receipt}")
  60. if not all([receipt, uid, channel]):
  61. return response.json(444)
  62. # redis加锁,防止订单重复
  63. redis_obj = RedisObject()
  64. redis_key = uid + 'in_app_purchase'
  65. is_lock = redis_obj.CONN.setnx(redis_key, 1)
  66. redis_obj.CONN.expire(redis_key, 60)
  67. # if not is_lock:
  68. # return response.json(5)
  69. try:
  70. device_info_qs = Device_Info.objects.filter(userID_id=user_id, UID=uid, isShare=False, isExist=1).values(
  71. 'vodPrimaryUserID',
  72. 'vodPrimaryMaster')
  73. if not device_info_qs.exists():
  74. return response.json(12)
  75. device_info_qs = Device_Info.objects.filter(Q(UID=uid), ~Q(vodPrimaryUserID='')).values('vodPrimaryUserID')
  76. if device_info_qs.exists():
  77. if device_info_qs[0]['vodPrimaryUserID'] != user_id:
  78. return response.json(10033)
  79. # 从交易信息中获取product_id
  80. key_path = '{}/Ansjer/file/in_app_purchase/SubscriptionKey_N42WMFCV6A.p8'.format(BASE_DIR)
  81. with open(key_path, 'rb') as file:
  82. # 读取文件内容
  83. private_key = file.read()
  84. key_id = 'N42WMFCV6A'
  85. issuer_id = '69a6de8c-789b-47e3-e053-5b8c7c11a4d1'
  86. bundle_id = 'com.ansjer.zccloud'
  87. environment = ENV
  88. client = AppStoreServerAPIClient(private_key, key_id, issuer_id, bundle_id, environment)
  89. receipt_util = ReceiptUtility()
  90. transaction_id = receipt_util.extract_transaction_id_from_app_receipt(receipt)
  91. if transaction_id is None:
  92. pay_result_url = CommonService.get_payment_status_url(lang, 'fail')
  93. return response.json(0, {'url': pay_result_url})
  94. transaction_info = client.get_transaction_info(transaction_id)
  95. signed_transaction_info = transaction_info.signedTransactionInfo
  96. root_certificates = []
  97. for cert_name in [
  98. 'AppleIncRootCertificate.cer', 'AppleComputerRootCertificate.cer',
  99. 'AppleRootCA-G2.cer', 'AppleRootCA-G3.cer'
  100. ]:
  101. cert_path = '{}/Ansjer/file/in_app_purchase/{}'.format(BASE_DIR, cert_name)
  102. with open(cert_path, 'rb') as file:
  103. # 读取文件内容
  104. root_certificates.append(file.read())
  105. enable_online_checks = True
  106. app_apple_id = None # 生产环境必需
  107. signed_data_verifier = SignedDataVerifier(
  108. root_certificates, enable_online_checks, environment, bundle_id, app_apple_id)
  109. payload = signed_data_verifier.verify_and_decode_signed_transaction(signed_transaction_info)
  110. product_id = None
  111. originalTransaction_id = None
  112. if payload and payload.productId and payload.originalTransactionId:
  113. product_id = payload.productId
  114. originalTransaction_id = payload.originalTransactionId
  115. if not product_id or not originalTransaction_id:
  116. pay_result_url = CommonService.get_payment_status_url(lang, 'fail')
  117. return response.json(0, {'url': pay_result_url})
  118. # 之前订阅过任意套餐返回不可再次订阅
  119. if Store_Meal.objects.filter(product_id=product_id, cycle_config_id__isnull=False).exists():
  120. transaction_subscription = client.get_all_subscription_statuses(transaction_id)
  121. if transaction_subscription.data:
  122. for subscription in transaction_subscription.data[0].lastTransactions:
  123. if str(subscription.status) == "Status.ACTIVE":
  124. key = str(originalTransaction_id) + "_SUBSCRIBED"
  125. redis_originalTransaction_id = redis_obj.get_data(key)
  126. if redis_originalTransaction_id != str(originalTransaction_id):
  127. return response.json(10048, f"订阅状态为 {subscription.status} ")
  128. pay_type = PAY_TYPE_IN_APP_PURCHASE
  129. now_time = int(time.time())
  130. store_qs = Store_Meal.objects.filter(
  131. product_id=product_id, lang__lang=lang, is_show=0). \
  132. values(
  133. 'id', 'currency', 'price', 'lang__content', 'day', 'commodity_type', 'lang__title', 'expire',
  134. 'commodity_code', 'discount_price', 'bucket_id', 'bucket__mold', 'cycle_config_id', 'is_ai')
  135. if not store_qs.exists():
  136. return response.json(173)
  137. order_id = CommonService.createOrderID()
  138. rank_id = store_qs[0]['id']
  139. bucket_id = store_qs[0]['bucket_id']
  140. currency = store_qs[0]['currency']
  141. price = store_qs[0]['price']
  142. is_ai = store_qs[0]['is_ai']
  143. expire = store_qs[0]['expire']
  144. end_time = CommonService.calcMonthLater(expire)
  145. content = store_qs[0]['lang__content']
  146. commodity_code = store_qs[0]['commodity_code']
  147. commodity_type = store_qs[0]['commodity_type']
  148. order_type = 1 if is_ai else 0
  149. store_meal_qs = Store_Meal.objects.filter(id=rank_id, lang__lang='cn', is_show=0). \
  150. values('lang__title', 'lang__content')
  151. if store_meal_qs.exists():
  152. store_meal_name = store_meal_qs[0]['lang__title'] + '-' + store_meal_qs[0]['lang__content']
  153. else:
  154. store_meal_name = '未知套餐'
  155. # 查询设备是否已开过云存
  156. use_flag = True
  157. uid_bucket_qs = UID_Bucket.objects.filter(uid=uid). \
  158. values('id', 'bucket_id', 'bucket__region', 'endTime', 'use_status')
  159. if uid_bucket_qs.exists():
  160. uid_bucket = uid_bucket_qs.first()
  161. uid_bucket_id = uid_bucket['id']
  162. # 叠加相同套餐的过期时间
  163. if uid_bucket['use_status'] == 1 and uid_bucket['endTime'] > now_time:
  164. Unused_Uid_Meal.objects.create(
  165. uid=uid, channel=channel, addTime=now_time, order_id=order_id, expire=expire, is_ai=is_ai,
  166. bucket_id=bucket_id)
  167. UID_Bucket.objects.filter(id=uid_bucket_id).update(has_unused=1)
  168. use_flag = False
  169. # 更新套餐的过期时间
  170. else:
  171. UID_Bucket.objects.filter(id=uid_bucket_id).update(
  172. channel=channel, bucket_id=bucket_id, endTime=end_time, updateTime=now_time, use_status=1,
  173. orderId=order_id)
  174. else:
  175. uid_bucket = UID_Bucket.objects.create(
  176. uid=uid, channel=channel, bucket_id=bucket_id, endTime=end_time, use_status=1, orderId=order_id,
  177. addTime=now_time, updateTime=now_time)
  178. uid_bucket_id = uid_bucket.id
  179. # 开通AI服务
  180. if is_ai and use_flag:
  181. ai_service = AiService.objects.filter(uid=uid, channel=channel)
  182. # 有正在使用的套餐,叠加套餐时间,否则创建
  183. if ai_service.exists():
  184. ai_service.update(updTime=now_time, use_status=1, orders_id=order_id, endTime=end_time)
  185. else:
  186. AiService.objects.create(
  187. uid=uid, channel=channel, detect_status=1, use_status=1, orders_id=order_id,
  188. addTime=now_time, updTime=now_time, endTime=end_time)
  189. Order_Model.objects.create(
  190. orderID=order_id, UID=uid, channel=channel, userID_id=user_id, desc=content, payType=pay_type,
  191. payTime=now_time, price=price, currency=currency, addTime=now_time, updTime=now_time,
  192. order_type=order_type, commodity_code=commodity_code, commodity_type=commodity_type, rank_id=rank_id,
  193. ai_rank_id=1, status=1, create_vod=1, store_meal_name=store_meal_name, uid_bucket_id=uid_bucket_id)
  194. # 发送云存开通信息
  195. date_time = time.strftime("%Y-%m-%d", time.localtime())
  196. # 如果存在序列号,消息提示用序列号
  197. device_info_qs = Device_Info.objects.filter(UID=uid).values('serial_number', 'Type')
  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. sys_msg_text_list = [
  205. '温馨提示:尊敬的客户,您的{}设备在{}已成功购买云存套餐'.format(device_name, date_time),
  206. 'Dear customer,you already subscribed the cloud storage package successfully for device {} on '.
  207. format(device_name, time.strftime('%b %dth,%Y', time.localtime()))]
  208. cls.do_vod_msg_notice(uid, user_id, lang, sys_msg_text_list)
  209. redis_obj.del_data(redis_key)
  210. pay_result_url = CommonService.get_payment_status_url(lang, 'success')
  211. return response.json(0, {'url': pay_result_url})
  212. except Exception as e:
  213. redis_obj.del_data(redis_key)
  214. LOGGER.info('苹果内购认证交易接口异常:{}'.
  215. format('error_line:{}, error_msg:{}'.format(e.__traceback__.tb_lineno, repr(e))))
  216. pay_result_url = CommonService.get_payment_status_url(lang, 'fail')
  217. return response.json(0, {'url': pay_result_url})
  218. @classmethod
  219. def do_vod_msg_notice(cls, uid, user_id, lang, sys_msg_text_list):
  220. """
  221. 发送云存开通信息
  222. @param uid: uid
  223. @param user_id: 用户id
  224. @param lang: 语言
  225. @param sys_msg_text_list: 消息列表
  226. @return: response
  227. """
  228. if lang == 'cn':
  229. sys_msg_text = sys_msg_text_list[0]
  230. else:
  231. sys_msg_text = sys_msg_text_list[1]
  232. now_time = int(time.time())
  233. create_data = {
  234. 'userID_id': user_id,
  235. 'msg': sys_msg_text,
  236. 'addTime': now_time,
  237. 'updTime': now_time,
  238. 'uid': uid,
  239. 'eventType': 0
  240. }
  241. SysMsgModel.objects.create(**create_data)
  242. # 不接收邮件用户
  243. if user_id == '167015836969813800138000':
  244. return
  245. user_qs = Device_User.objects.filter(userID=user_id)
  246. if user_qs.exists():
  247. user = user_qs.first()
  248. username = user.username
  249. data_valid = DataValid()
  250. if data_valid.email_validate(username):
  251. S3Email().faEmail(sys_msg_text, username)
  252. elif data_valid.mobile_validate(username):
  253. # 如果存在序列号,消息提示用序列号
  254. device_info_qs = Device_Info.objects.filter(UID=uid).values('serial_number', 'Type')
  255. if device_info_qs.exists():
  256. serial_number = device_info_qs[0]['serial_number']
  257. device_type = device_info_qs[0]['Type']
  258. if serial_number:
  259. device_name = CommonService.get_full_serial_number(uid, serial_number, device_type)
  260. else:
  261. device_name = uid
  262. params = '{"devname":"%s","submittime":"%s"}' % (
  263. device_name, time.strftime("%Y-%m-%d", time.localtime()))
  264. cls.send_message(username, params, 'SMS_219738485')
  265. @staticmethod
  266. def send_message(phone, params, temp_msg):
  267. """
  268. 发送手机消息
  269. @param phone: 用户名
  270. @param params: 消息参数
  271. @param temp_msg: sms码
  272. """
  273. sign_ms = '周视'
  274. ali_sms = AliSmsObject()
  275. ali_sms.send_code_sms_cloud(phone=phone, params=params, sign_name=sign_ms, temp_msg=temp_msg)
  276. @classmethod
  277. def app_store_server_notifications(cls, request):
  278. logger = logging.getLogger('apple_pay')
  279. try:
  280. logger.info('App Store服务器通知请求类型:{}'.format(request.method))
  281. logger.info('App Store服务器通知参数:{}'.format(request.POST))
  282. logger.info('App Store服务器通知请求body:{}'.format(request.body))
  283. if request.method == 'POST':
  284. payload = json.loads(request.body.decode('utf-8'))
  285. logger.info('App Store服务器通知payload:{}'.format(payload))
  286. # 获取 signedPayload
  287. signed_payload = payload.get('signedPayload')
  288. if not signed_payload:
  289. return HttpResponse(status=400)
  290. bundle_id = 'com.ansjer.zccloud'
  291. environment = ENV
  292. root_certificates = []
  293. for cert_name in [
  294. 'AppleIncRootCertificate.cer', 'AppleComputerRootCertificate.cer',
  295. 'AppleRootCA-G2.cer', 'AppleRootCA-G3.cer'
  296. ]:
  297. cert_path = '{}/Ansjer/file/in_app_purchase/{}'.format(BASE_DIR, cert_name)
  298. with open(cert_path, 'rb') as file:
  299. # 读取文件内容
  300. root_certificates.append(file.read())
  301. enable_online_checks = True
  302. app_apple_id = None # 生产环境必需
  303. # 验证签名并解码 payload
  304. verifier = SignedDataVerifier(
  305. root_certificates, enable_online_checks, environment, bundle_id, app_apple_id)
  306. decoded_payload = verifier.verify_and_decode_notification(signed_payload)
  307. logger.info(f"App Store服务器通知解码后decoded_payload:{decoded_payload}")
  308. logger.info(
  309. f"App Store服务器通知decoded_payload.rawNotificationType{str(decoded_payload.rawNotificationType)}")
  310. if str(decoded_payload.rawNotificationType) == "DID_RENEW":
  311. decoded_transaction_information = verifier.verify_and_decode_signed_transaction(
  312. decoded_payload.data.signedTransactionInfo)
  313. # originalTransactionId 原始购买的交易标识符
  314. originalTransaction_id = decoded_transaction_information.originalTransactionId
  315. logger.info(f"App Store服务器通知originalTransactionId原始购买的交易标识符{originalTransaction_id}")
  316. if not originalTransaction_id:
  317. logger.info(f"App Store服务器通知originalTransactionId原始购买的交易标识符为空, 返回状态 400")
  318. return HttpResponse(status=400)
  319. else:
  320. ord_order = Order_Model.objects.filter(agreement_id=originalTransaction_id).order_by(
  321. 'addTime').first()
  322. channel = ord_order.channel
  323. uid = ord_order.uid
  324. pay_type = ord_order.payType
  325. user_id = ord_order.userID_id
  326. store_qs = Store_Meal.objects.filter(id=ord_order.rank). \
  327. values(
  328. 'id', 'currency', 'price', 'lang__content', 'day', 'commodity_type', 'lang__title',
  329. 'expire', 'lang__lang',
  330. 'commodity_code', 'discount_price', 'bucket_id', 'bucket__mold', 'cycle_config_id', 'is_ai')
  331. if not store_qs.exists():
  332. logger.info(f"App Store服务器通知云存套餐不存在, 返回状态 400")
  333. return HttpResponse(status=400)
  334. order_id = CommonService.createOrderID()
  335. rank_id = store_qs[0]['id']
  336. bucket_id = store_qs[0]['bucket_id']
  337. currency = store_qs[0]['currency']
  338. price = store_qs[0]['price']
  339. is_ai = store_qs[0]['is_ai']
  340. expire = store_qs[0]['expire']
  341. end_time = CommonService.calcMonthLater(expire)
  342. content = store_qs[0]['lang__content']
  343. commodity_code = store_qs[0]['commodity_code']
  344. commodity_type = store_qs[0]['commodity_type']
  345. lang = store_qs[0]['lang__lang']
  346. order_type = 1 if is_ai else 0
  347. store_meal_qs = Store_Meal.objects.filter(id=rank_id, lang__lang='cn', is_show=0). \
  348. values('lang__title', 'lang__content')
  349. if store_meal_qs.exists():
  350. store_meal_name = store_meal_qs[0]['lang__title'] + '-' + store_meal_qs[0]['lang__content']
  351. else:
  352. store_meal_name = '未知套餐'
  353. # 查询设备是否已开过云存
  354. use_flag = True
  355. uid_bucket_qs = UID_Bucket.objects.filter(uid=uid). \
  356. values('id', 'bucket_id', 'bucket__region', 'endTime', 'use_status')
  357. now_time = int(time.time())
  358. if uid_bucket_qs.exists():
  359. uid_bucket = uid_bucket_qs.first()
  360. uid_bucket_id = uid_bucket['id']
  361. # 叠加相同套餐的过期时间
  362. if uid_bucket['use_status'] == 1 and uid_bucket['endTime'] > now_time:
  363. Unused_Uid_Meal.objects.create(
  364. uid=uid, channel=channel, addTime=now_time, order_id=order_id, expire=expire,
  365. is_ai=is_ai,
  366. bucket_id=bucket_id)
  367. UID_Bucket.objects.filter(id=uid_bucket_id).update(has_unused=1)
  368. use_flag = False
  369. # 更新套餐的过期时间
  370. else:
  371. UID_Bucket.objects.filter(id=uid_bucket_id).update(
  372. channel=channel, bucket_id=bucket_id, endTime=end_time, updateTime=now_time,
  373. use_status=1,
  374. orderId=order_id)
  375. else:
  376. uid_bucket = UID_Bucket.objects.create(
  377. uid=uid, channel=channel, bucket_id=bucket_id, endTime=end_time, use_status=1,
  378. orderId=order_id,
  379. addTime=now_time, updateTime=now_time)
  380. uid_bucket_id = uid_bucket.id
  381. # 开通AI服务
  382. if is_ai and use_flag:
  383. ai_service = AiService.objects.filter(uid=uid, channel=channel)
  384. # 有正在使用的套餐,叠加套餐时间,否则创建
  385. if ai_service.exists():
  386. ai_service.update(updTime=now_time, use_status=1, orders_id=order_id,
  387. endTime=end_time)
  388. else:
  389. AiService.objects.create(
  390. uid=uid, channel=channel, detect_status=1, use_status=1, orders_id=order_id,
  391. addTime=now_time, updTime=now_time, endTime=end_time)
  392. Order_Model.objects.create(
  393. orderID=order_id, UID=uid, channel=channel, userID_id=user_id, desc=content,
  394. payType=pay_type,
  395. payTime=now_time, price=price, currency=currency, addTime=now_time, updTime=now_time,
  396. order_type=order_type, commodity_code=commodity_code, commodity_type=commodity_type,
  397. rank_id=rank_id,
  398. ai_rank_id=1, status=1, create_vod=1, store_meal_name=store_meal_name,
  399. uid_bucket_id=uid_bucket_id, agreement_id=originalTransaction_id
  400. )
  401. # 发送云存开通信息
  402. date_time = time.strftime("%Y-%m-%d", time.localtime())
  403. # 如果存在序列号,消息提示用序列号
  404. device_info_qs = Device_Info.objects.filter(UID=uid).values('serial_number', 'Type')
  405. serial_number = device_info_qs[0]['serial_number']
  406. device_type = device_info_qs[0]['Type']
  407. if serial_number:
  408. device_name = CommonService.get_full_serial_number(uid, serial_number, device_type)
  409. else:
  410. device_name = uid
  411. sys_msg_text_list = [
  412. '温馨提示:尊敬的客户,您的{}设备在{}已成功续订云存套餐'.format(device_name, date_time),
  413. 'Dear customer,you already subscribed the cloud storage package successfully for device {} on '.
  414. format(device_name, time.strftime('%b %dth,%Y', time.localtime()))]
  415. cls.do_vod_msg_notice(uid, user_id, lang, sys_msg_text_list)
  416. elif str(decoded_payload.rawNotificationType) == "SUBSCRIBED":
  417. # 处理订阅
  418. # 一种通知类型,与其子类型一起表示客户订阅了自动续订的套餐。如果子类型为
  419. # INITIAL_BUY(首次购买),则表示用户首次购买或通过家庭共享访问该订阅。如果子类型为
  420. decoded_transaction_information = verifier.verify_and_decode_signed_transaction(
  421. decoded_payload.data.signedTransactionInfo)
  422. originalTransaction_id = decoded_transaction_information.originalTransactionId
  423. # redis上锁,保证验证时是同一个订单
  424. redis_obj = RedisObject()
  425. redis_key = str(originalTransaction_id) + "_SUBSCRIBED"
  426. redis_obj.set_data(redis_key, originalTransaction_id, 3600)
  427. # 之前订阅过不可重复订阅
  428. key_id = 'N42WMFCV6A'
  429. issuer_id = '69a6de8c-789b-47e3-e053-5b8c7c11a4d1'
  430. bundle_id = 'com.ansjer.zccloud'
  431. environment = ENV
  432. key_path = '{}/Ansjer/file/in_app_purchase/SubscriptionKey_N42WMFCV6A.p8'.format(BASE_DIR)
  433. with open(key_path, 'rb') as file:
  434. # 读取文件内容
  435. private_key = file.read()
  436. client = AppStoreServerAPIClient(private_key, key_id, issuer_id, bundle_id, environment)
  437. transaction_id = decoded_transaction_information.transactionId
  438. transaction_subscription = client.get_all_subscription_statuses(transaction_id)
  439. if transaction_subscription.data:
  440. for subscription in transaction_subscription.data[0].lastTransactions:
  441. if str(subscription.status) == "Status.ACTIVE":
  442. return HttpResponse(status=400)
  443. elif str(decoded_payload.rawNotificationType) == "EXPIRED":
  444. # 一种通知类型,与其子类型一起表示订阅已过期。如果subtype为
  445. # VOLUNTARY(自愿),则表示订阅在用户禁用订阅续订后过期。如果subtype是
  446. # BILLING_RETRY(计费重试),则表示订阅过期,因为计费重试期结束时没有成功的计费交易。如果subtype为
  447. # PRICE_INCREASE,则表示订阅已过期,因为客户不同意需要客户同意的价格上涨。如果subtype为
  448. # PRODUCT_NOT_FOR_SALE,则表示订阅已过期,因为在订阅尝试续订时,产品已不可购买。
  449. # 没有子类型的通知表示订阅因其他原因过期。
  450. pass
  451. else:
  452. logger.info(f"App Store服务器通知decoded_payload.rawNotificationType 未处理")
  453. return HttpResponse(status=500)
  454. return HttpResponse(status=200)
  455. else:
  456. logger.info('App Store服务器通知不是post请求')
  457. return HttpResponse(status=400)
  458. except Exception as e:
  459. logger.info('App Store服务器通知异常:{}'.
  460. format('error_line:{}, error_msg:{}'.format(e.__traceback__.tb_lineno, repr(e))))
  461. return HttpResponse(status=500)