InAppPurchaseController.py 52 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936
  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 threading
  8. import requests
  9. from appstoreserverlibrary.models.Environment import Environment
  10. from appstoreserverlibrary.api_client import AppStoreServerAPIClient, GetTransactionHistoryVersion, APIException
  11. from appstoreserverlibrary.models.AccountTenure import AccountTenure
  12. from appstoreserverlibrary.models.ConsumptionRequest import ConsumptionRequest
  13. from appstoreserverlibrary.models.ConsumptionStatus import ConsumptionStatus
  14. from appstoreserverlibrary.models.DeliveryStatus import DeliveryStatus
  15. from appstoreserverlibrary.models.Environment import Environment
  16. from appstoreserverlibrary.models.LifetimeDollarsPurchased import LifetimeDollarsPurchased
  17. from appstoreserverlibrary.models.LifetimeDollarsRefunded import LifetimeDollarsRefunded
  18. from appstoreserverlibrary.models.Platform import Platform
  19. from appstoreserverlibrary.models.PlayTime import PlayTime
  20. from appstoreserverlibrary.models.RefundPreference import RefundPreference
  21. from appstoreserverlibrary.models.UserStatus import UserStatus
  22. from appstoreserverlibrary.receipt_utility import ReceiptUtility
  23. from django.views import View
  24. from django.http import HttpResponse
  25. from Ansjer.config import LOGGER, CONFIG_INFO, CONFIG_TEST, PAY_TYPE_IN_APP_PURCHASE, BASE_DIR, CONFIG_US
  26. from Controller.CheckUserData import DataValid
  27. from Model.models import Order_Model, Store_Meal, Device_Info, UID_Bucket, Unused_Uid_Meal, AiService, Device_User, \
  28. SysMsgModel, DeviceApplePackage, InAppPurchasePackage, InAppRefund, OrderPayLog, CouponModel
  29. from Object.AWS.S3Email import S3Email
  30. from Object.AliSmsObject import AliSmsObject
  31. from Object.AppleInAppPurchaseSubscriptionObject import InAppPurchase
  32. from Object.Enums.RedisKeyConstant import RedisKeyConstant
  33. from Object.RedisObject import RedisObject
  34. from Service.CommonService import CommonService
  35. ENV = Environment.SANDBOX if CONFIG_INFO == CONFIG_TEST else Environment.PRODUCTION
  36. logger = logging.getLogger('apple_pay')
  37. class InAppPurchaseView(View):
  38. def get(self, request, *args, **kwargs):
  39. request.encoding = 'utf-8'
  40. operation = kwargs.get('operation')
  41. return self.validation(request.GET, request, operation)
  42. def post(self, request, *args, **kwargs):
  43. request.encoding = 'utf-8'
  44. operation = kwargs.get('operation')
  45. return self.validation(request.POST, request, operation)
  46. def validation(self, request_dict, request, operation):
  47. if operation == 'AppStoreServerNotifications': # App Store服务器通知(用于转发通知)
  48. return self.app_store_server_notifications(request, request_dict)
  49. elif operation == 'AppStoreServerNotificationsVsees': # App Store服务器通知(用于转发通知)
  50. return self.app_store_server_notifications_vsees(request, request_dict)
  51. elif operation == 'vseesNotifications':
  52. return self.vsees_notifications(request)
  53. elif operation == 'serverNotifications': # App Store服务器通知
  54. return self.server_notifications(request)
  55. elif operation == 'putRefundOrder': # App Store服务器通知
  56. return self.put_refund_order()
  57. token_code, user_id, response = CommonService.verify_token_get_user_id(request_dict, request)
  58. if token_code != 0:
  59. return response.json(token_code)
  60. if operation == 'verifyTransaction': # 认证交易
  61. return self.verify_transaction(user_id, request_dict, response)
  62. @classmethod
  63. def verify_transaction(cls, user_id, request_dict, response):
  64. """
  65. 认证交易
  66. @param user_id: 用户id
  67. @param request_dict: 请求参数
  68. @request_dict receipt: 收据
  69. @param response: 响应对象
  70. @return: response
  71. """
  72. receipt = request_dict.get('receipt', None)
  73. transaction_identifier = request_dict.get('transactionIdentifier', "")
  74. original_transaction_identifier = request_dict.get('originalTransactionIdentifier', "")
  75. order_id = request_dict.get('orderID', None)
  76. uid = request_dict.get('uid', None)
  77. lang = request_dict.get('lang', 'en')
  78. channel = request_dict.get('channel', None)
  79. app_type = request_dict.get('app_type', 1)
  80. logger.info(
  81. f"苹果内购认证交易订单orderID:{order_id},"
  82. f"transaction_id: {transaction_identifier},"
  83. f"original_transaction_id: {original_transaction_identifier},"
  84. f"receipt: {receipt}, app_type: {app_type}"
  85. )
  86. if not all([uid, channel, order_id]):
  87. return response.json(444)
  88. # redis加锁,防止订单重复
  89. redis_obj = RedisObject()
  90. redis_key = order_id + 'in_app_purchase'
  91. is_lock = redis_obj.CONN.setnx(redis_key, 1)
  92. if not is_lock:
  93. return response.json(5)
  94. redis_obj.CONN.expire(redis_key, 60)
  95. try:
  96. # 检查商品id是否正确
  97. app_type = int(app_type)
  98. if app_type == 1:
  99. bundle_id = "com.ansjer.zccloud"
  100. elif app_type == 2:
  101. bundle_id = "com.cloudlife.commissionf"
  102. else:
  103. return response.json(444, "app_type不存在")
  104. # 验证订单是否存在
  105. order_qs = Order_Model.objects.filter(orderID=order_id, UID=uid, app_type=app_type).values("rank_id",
  106. "transaction_id",
  107. "coupon_id")
  108. if not order_qs.exists():
  109. return response.json(173)
  110. if order_qs[0]["transaction_id"]:
  111. return response.json(0)
  112. if UID_Bucket.objects.filter(orderId=order_id).exists():
  113. return response.json(0)
  114. if Unused_Uid_Meal.objects.filter(order_id=order_id).exists():
  115. return response.json(0)
  116. # 实例化订阅类
  117. in_app_purchase = InAppPurchase(bundle_id=bundle_id)
  118. # ReceiptUtility 用于解析收据为transaction_id
  119. receipt_util = in_app_purchase.receipt_util
  120. # AppStoreServerAPIClient 用于查询交易信息
  121. client = in_app_purchase.client
  122. # SignedDataVerifier 用于解析查询到的交易信息
  123. signed_data_verifier = in_app_purchase.verifier
  124. # 解析收据(循环扣款时不需要这一步, 直接获取transaction_id)
  125. transaction_id = transaction_identifier
  126. if transaction_identifier == "":
  127. transaction_id = receipt_util.extract_transaction_id_from_app_receipt(receipt)
  128. if not transaction_id:
  129. logger.info(f"苹果内购认证交易订单orderID:{order_id}, 没有transaction_id")
  130. pay_result_url = CommonService.get_payment_status_url(lang, 'fail')
  131. return response.json(0, {'url': pay_result_url})
  132. logger.info(f"苹果内购认证交易订单orderID:{order_id}, transaction_id:{transaction_id}, 时间戳: {int(time.time())}")
  133. OrderPayLog.objects.create(order_id=order_id, order_no=transaction_id,
  134. business_name=f"内购验单",
  135. created_time=int(time.time()), updated_time=int(time.time()),
  136. access_result="SUCCESS")
  137. # 查询交易信息
  138. try:
  139. transaction_info = client.get_transaction_info(transaction_id)
  140. except APIException:
  141. in_app_purchase = InAppPurchase(bundle_id=bundle_id, user_id=user_id)
  142. # AppStoreServerAPIClient 用于查询交易信息
  143. client = in_app_purchase.client
  144. # SignedDataVerifier 用于解析查询到的交易信息
  145. signed_data_verifier = in_app_purchase.verifier
  146. transaction_info = client.get_transaction_info(transaction_id)
  147. signed_transaction_info = transaction_info.signedTransactionInfo
  148. # 解析交易信息
  149. payload = signed_data_verifier.verify_and_decode_signed_transaction(signed_transaction_info)
  150. # 获取交易的商品id
  151. product_id = payload.productId if payload and payload.productId else None
  152. if not product_id:
  153. logger.info(f"苹果内购认证交易订单orderID:{order_id}, product_id获取失败")
  154. pay_result_url = CommonService.get_payment_status_url(lang, 'fail')
  155. return response.json(0, {'url': pay_result_url})
  156. in_app_purchase_package_qs = InAppPurchasePackage.objects.filter(product_id=product_id)
  157. if not in_app_purchase_package_qs.exists():
  158. logger.info(f"苹果内购认证交易订单orderID:{order_id}, InAppPurchasePackage表未查询到product_id")
  159. return response.json(173, "内购商品id不存在")
  160. # 验证套餐是否存在
  161. store_qs = Store_Meal.objects.filter(id=order_qs[0]['rank_id']).values(
  162. 'id', 'currency', 'price', 'lang__content', 'day', 'commodity_type', 'lang__title', 'expire',
  163. 'commodity_code', 'discount_price', 'bucket_id', 'bucket__mold', 'cycle_config_id', 'is_ai')
  164. if not store_qs.exists():
  165. return response.json(173, "云存套餐不存在")
  166. # 验证内购套餐是否存在
  167. in_app_purchase_package_qs = InAppPurchasePackage.objects.filter(product_id=product_id)
  168. if not in_app_purchase_package_qs.exists():
  169. return response.json(173, "内购套餐不存在")
  170. # 循环扣款
  171. if original_transaction_identifier != "" and in_app_purchase_package_qs[0].package_type == 1:
  172. device_apple_package_qs = DeviceApplePackage.objects.filter(
  173. original_transaction_id=original_transaction_identifier)
  174. if device_apple_package_qs.exists():
  175. # 第一种情况: 套餐已过期再次订阅
  176. if device_apple_package_qs[0].uid == uid and device_apple_package_qs[0].subscription_status == 2:
  177. # 使用App Store服务器通知接口订阅
  178. Order_Model.objects.filter(orderID=order_id).delete()
  179. pay_result_url = CommonService.get_payment_status_url(lang, 'success')
  180. return response.json(0, {'url': pay_result_url})
  181. # 第二种情况: 套餐未过期已取消再次订阅
  182. elif device_apple_package_qs[0].uid == uid and device_apple_package_qs[0].subscription_status == 3:
  183. # 使用App Store服务器通知接口修改订阅状态
  184. Order_Model.objects.filter(orderID=order_id).delete()
  185. pay_result_url = CommonService.get_payment_status_url(lang, 'success')
  186. return response.json(0, {'url': pay_result_url})
  187. # 第三种情况: 首次订阅
  188. elif device_apple_package_qs[0].uid == uid and device_apple_package_qs[0].subscription_status == 0:
  189. logger.info(f"苹果内购认证交易订单orderID:{order_id}, 用户首次订阅")
  190. else:
  191. logger.info(
  192. f"错误调用此借口,orderID:{order_id}, uid:{uid}, 订阅状态:{device_apple_package_qs[0].subscription_status}")
  193. pay_result_url = CommonService.get_payment_status_url(lang, 'fail')
  194. return response.json(0, {'url': pay_result_url})
  195. else:
  196. return response.json(173, "内购套餐未分配")
  197. # 设备开通云存
  198. now_time = int(time.time())
  199. uid_bucket_id = cls.enable_cloud(channel, now_time, order_id, store_qs, uid)
  200. if order_qs[0]["coupon_id"]:
  201. c_id = order_qs[0]["coupon_id"]
  202. key_coupon = RedisKeyConstant.COUPON_ID_LOCK.value + c_id
  203. redis_obj.del_data(key_coupon)
  204. CouponModel.objects.filter(id=c_id).update(use_status=2, update_time=now_time)
  205. # 修改订阅状态
  206. if payload.rawType == "Auto-Renewable Subscription":
  207. original_transaction_id = payload.originalTransactionId
  208. in_app_purchase_package = in_app_purchase_package_qs.values('id').first()
  209. package_id = in_app_purchase_package['id']
  210. DeviceApplePackage.objects.filter(userID=user_id, uid=uid, package_id=package_id).update(
  211. subscription_status=1, original_transaction_id=original_transaction_id,
  212. )
  213. order_qs.update(status=1, uid_bucket_id=uid_bucket_id,
  214. transaction_id=transaction_id, create_vod=1,
  215. payTime=now_time, updTime=now_time,
  216. original_transaction_id=original_transaction_id)
  217. else:
  218. order_qs.update(status=1, uid_bucket_id=uid_bucket_id,
  219. transaction_id=transaction_id, create_vod=1,
  220. payTime=now_time, updTime=now_time)
  221. # 构建云存套餐消息
  222. sys_msg_text_list = cls.cloud_storage_message(uid)
  223. # 发送云存套餐购买消息
  224. asy = threading.Thread(target=cls.do_vod_msg_notice,
  225. args=(uid, user_id, lang, sys_msg_text_list))
  226. asy.start()
  227. redis_obj.del_data(redis_key)
  228. pay_result_url = CommonService.get_payment_status_url(lang, 'success')
  229. OrderPayLog.objects.create(order_id=order_id, order_no=transaction_id,
  230. business_name=f"内购充值成功",
  231. created_time=int(time.time()), updated_time=int(time.time()),
  232. access_result="SUCCESS")
  233. return response.json(0, {'url': pay_result_url})
  234. except Exception as e:
  235. redis_obj.del_data(redis_key)
  236. logger.info('苹果内购认证交易接口异常:{}'.
  237. format('error_line:{}, error_msg:{}'.format(e.__traceback__.tb_lineno, repr(e))))
  238. OrderPayLog.objects.create(order_id=order_id, business_name=f"内购验单异常",
  239. created_time=int(time.time()), updated_time=int(time.time()), access_result="ERROR")
  240. pay_result_url = CommonService.get_payment_status_url(lang, 'fail')
  241. return response.json(0, {'url': pay_result_url})
  242. @classmethod
  243. def cloud_storage_message(cls, uid):
  244. # 发送云存开通信息
  245. date_time = time.strftime("%Y-%m-%d", time.localtime())
  246. # 如果存在序列号,消息提示用序列号
  247. device_info_qs = Device_Info.objects.filter(UID=uid).values('serial_number', 'Type')
  248. serial_number = device_info_qs[0]['serial_number']
  249. device_type = device_info_qs[0]['Type']
  250. if serial_number:
  251. device_name = CommonService.get_full_serial_number(uid, serial_number, device_type)
  252. else:
  253. device_name = uid
  254. sys_msg_text_list = [
  255. '温馨提示:尊敬的客户,您的{}设备在{}已成功购买云存套餐'.format(device_name, date_time),
  256. 'Dear customer,you already subscribed the cloud storage package successfully for device {} on '.
  257. format(device_name, time.strftime('%b %dth,%Y', time.localtime()))]
  258. return sys_msg_text_list
  259. @classmethod
  260. def enable_cloud(cls, channel, now_time, order_id, store_qs, uid):
  261. bucket_id = store_qs[0]['bucket_id']
  262. is_ai = store_qs[0]['is_ai']
  263. expire = store_qs[0]['expire']
  264. end_time = CommonService.calcMonthLater(expire)
  265. # 查询设备是否已开过云存
  266. use_flag = True
  267. uid_bucket_qs = UID_Bucket.objects.filter(uid=uid). \
  268. values('id', 'bucket_id', 'bucket__region', 'endTime', 'use_status')
  269. if uid_bucket_qs.exists():
  270. uid_bucket = uid_bucket_qs.first()
  271. uid_bucket_id = uid_bucket['id']
  272. # 有正在使用的套餐,创建为未使用套餐
  273. if uid_bucket['use_status'] == 1 and uid_bucket['endTime'] > now_time:
  274. Unused_Uid_Meal.objects.create(
  275. uid=uid, channel=channel, addTime=now_time, order_id=order_id, expire=expire, is_ai=is_ai,
  276. bucket_id=bucket_id)
  277. UID_Bucket.objects.filter(id=uid_bucket_id).update(has_unused=1)
  278. use_flag = False
  279. # 无正在使用套餐,直接使用套餐
  280. else:
  281. UID_Bucket.objects.filter(id=uid_bucket_id).update(
  282. channel=channel, bucket_id=bucket_id, endTime=end_time, updateTime=now_time, use_status=1,
  283. orderId=order_id)
  284. else:
  285. uid_bucket = UID_Bucket.objects.create(
  286. uid=uid, channel=channel, bucket_id=bucket_id, endTime=end_time, use_status=1, orderId=order_id,
  287. addTime=now_time, updateTime=now_time)
  288. uid_bucket_id = uid_bucket.id
  289. # 开通AI服务
  290. if is_ai and use_flag:
  291. ai_service = AiService.objects.filter(uid=uid, channel=channel)
  292. # 有正在使用的套餐,叠加套餐时间,否则创建
  293. if ai_service.exists():
  294. ai_service.update(updTime=now_time, use_status=1, orders_id=order_id, endTime=end_time)
  295. else:
  296. AiService.objects.create(
  297. uid=uid, channel=channel, detect_status=1, use_status=1, orders_id=order_id,
  298. addTime=now_time, updTime=now_time, endTime=end_time)
  299. return uid_bucket_id
  300. @classmethod
  301. def do_vod_msg_notice(cls, uid, user_id, lang, sys_msg_text_list):
  302. """
  303. 发送云存开通信息
  304. @param uid: uid
  305. @param user_id: 用户id
  306. @param lang: 语言
  307. @param sys_msg_text_list: 消息列表
  308. @return: response
  309. """
  310. if lang == 'cn':
  311. sys_msg_text = sys_msg_text_list[0]
  312. else:
  313. sys_msg_text = sys_msg_text_list[1]
  314. now_time = int(time.time())
  315. create_data = {
  316. 'userID_id': user_id,
  317. 'msg': sys_msg_text,
  318. 'addTime': now_time,
  319. 'updTime': now_time,
  320. 'uid': uid,
  321. 'eventType': 0
  322. }
  323. SysMsgModel.objects.create(**create_data)
  324. # 不接收邮件用户
  325. if user_id == '167015836969813800138000':
  326. return
  327. user_qs = Device_User.objects.filter(userID=user_id)
  328. if user_qs.exists():
  329. user = user_qs.first()
  330. username = user.username
  331. data_valid = DataValid()
  332. if data_valid.email_validate(username):
  333. S3Email().faEmail(sys_msg_text, username)
  334. elif data_valid.mobile_validate(username):
  335. # 如果存在序列号,消息提示用序列号
  336. device_info_qs = Device_Info.objects.filter(UID=uid).values('serial_number', 'Type')
  337. if device_info_qs.exists():
  338. serial_number = device_info_qs[0]['serial_number']
  339. device_type = device_info_qs[0]['Type']
  340. if serial_number:
  341. device_name = CommonService.get_full_serial_number(uid, serial_number, device_type)
  342. else:
  343. device_name = uid
  344. params = '{"devname":"%s","submittime":"%s"}' % (
  345. device_name, time.strftime("%Y-%m-%d", time.localtime()))
  346. cls.send_message(username, params, 'SMS_219738485')
  347. @staticmethod
  348. def send_message(phone, params, temp_msg):
  349. """
  350. 发送手机消息
  351. @param phone: 用户名
  352. @param params: 消息参数
  353. @param temp_msg: sms码
  354. """
  355. sign_ms = '周视'
  356. ali_sms = AliSmsObject()
  357. ali_sms.send_code_sms_cloud(phone=phone, params=params, sign_name=sign_ms, temp_msg=temp_msg)
  358. @classmethod
  359. def app_store_server_notifications(cls, request, request_dict):
  360. logger = logging.getLogger('apple_pay')
  361. if request.method != 'POST':
  362. logger.info(f'App Store服务器通知不是post请求, 参数{request_dict}')
  363. return HttpResponse(status=400)
  364. try:
  365. request_data = json.loads(request.body)
  366. except json.JSONDecodeError:
  367. logger.info('无法解析请求体为JSON')
  368. return HttpResponse(status=400)
  369. request_data['bundleId'] = 'com.ansjer.zccloud'
  370. updated_request_body = json.dumps(request_data)
  371. if CONFIG_INFO == CONFIG_TEST:
  372. logger.info('测试环境, App Store服务器通知发送到测试服')
  373. response_test = requests.post(url="https://test.zositechc.cn/inAppPurchase/serverNotifications",
  374. json=updated_request_body)
  375. return HttpResponse(status=response_test.status_code)
  376. response_us = requests.post(url="https://www.dvema.com/inAppPurchase/serverNotifications",
  377. json=updated_request_body)
  378. status_code = response_us.status_code
  379. if status_code != 200:
  380. response_eu = requests.post(url="https://api.zositeche.com/inAppPurchase/serverNotifications",
  381. json=updated_request_body)
  382. status_code = response_eu.status_code
  383. if status_code == 200:
  384. return HttpResponse(status=200)
  385. return HttpResponse(status=400)
  386. @classmethod
  387. def server_notifications(cls, request):
  388. try:
  389. logger.info('App Store服务器通知请求类型:{}'.format(request.method))
  390. logger.info('App Store服务器通知参数:{}'.format(request.POST))
  391. logger.info('App Store服务器通知请求body:{}'.format(request.body))
  392. payload = json.loads(json.loads(request.body))
  393. logger.info('App Store服务器通知payload:{}'.format(payload))
  394. # 获取 signedPayload
  395. signed_payload = payload.get('signedPayload')
  396. bundle_id = payload.get('bundleId')
  397. if not signed_payload:
  398. return HttpResponse(status=400)
  399. if bundle_id == "com.ansjer.zccloud":
  400. app_type = 1
  401. else:
  402. app_type = 2
  403. in_app_purchase_obj = InAppPurchase(bundle_id=bundle_id)
  404. # AppStoreServerAPIClient 用于查询交易信息
  405. client = in_app_purchase_obj.client
  406. # SignedDataVerifier 用于解析查询到的交易信息
  407. signed_data_verifier = in_app_purchase_obj.verifier
  408. # 验证签名并解码 payload
  409. decoded_payload = signed_data_verifier.verify_and_decode_notification(signed_payload)
  410. logger.info(f"App Store服务器通知解码后decoded_payload:{decoded_payload}")
  411. raw_notification_type = str(decoded_payload.rawNotificationType)
  412. raw_subtype = str(decoded_payload.rawSubtype)
  413. logger.info(f"App Store服务器通知, 大类型{raw_notification_type}, 小类型{raw_subtype}")
  414. if str(decoded_payload.rawNotificationType) == "DID_RENEW":
  415. # 续订
  416. decoded_transaction_information = signed_data_verifier.verify_and_decode_signed_transaction(
  417. decoded_payload.data.signedTransactionInfo)
  418. # originalTransactionId 原始购买的交易标识符
  419. original_transaction_id = decoded_transaction_information.originalTransactionId
  420. transaction_id = decoded_transaction_information.transactionId
  421. logger.info(f"App Store服务器通知, 续订originalTransactionId{original_transaction_id}")
  422. if not original_transaction_id:
  423. logger.info(f"App Store服务器通知 originalTransactionId原始购买的交易标识符为空, 返回状态 400")
  424. return HttpResponse(status=400)
  425. ord_order = Order_Model.objects.filter(original_transaction_id=original_transaction_id).order_by(
  426. '-addTime').values("channel", "UID", "payType", "userID_id", "rank_id")
  427. if not ord_order.exists():
  428. logger.info(
  429. f"App Store服务器通知, 未查询到旧订单信息, originalTransactionId:{original_transaction_id}, 返回状态 400")
  430. return HttpResponse(status=400)
  431. channel = ord_order[0]["channel"]
  432. uid = ord_order[0]["UID"]
  433. pay_type = ord_order[0]["payType"]
  434. user_id = ord_order[0]["userID_id"]
  435. store_qs = Store_Meal.objects.filter(id=ord_order[0]["rank_id"]). \
  436. values(
  437. 'id', 'currency', 'price', 'lang__content', 'day', 'commodity_type', 'lang__title',
  438. 'expire', 'lang__lang',
  439. 'commodity_code', 'discount_price', 'bucket_id', 'bucket__mold', 'cycle_config_id', 'is_ai')
  440. if not store_qs.exists():
  441. logger.info(f"App Store服务器通知云存套餐不存在, 返回状态 400")
  442. return HttpResponse(status=400)
  443. # 解决云存充值成功, 由于一些原因返回500 导致苹果未扣款的问题
  444. if Order_Model.objects.filter(transaction_id=transaction_id, status=1).exists():
  445. logger.info(f"App Store服务器通知云存续订订单已存在, transactionId:{transaction_id} 返回状态 200")
  446. return HttpResponse(status=200)
  447. order_id = CommonService.createOrderID()
  448. rank_id = store_qs[0]['id']
  449. currency = store_qs[0]['currency']
  450. price = store_qs[0]['price']
  451. is_ai = store_qs[0]['is_ai']
  452. content = store_qs[0]['lang__content']
  453. commodity_code = store_qs[0]['commodity_code']
  454. commodity_type = store_qs[0]['commodity_type']
  455. lang = store_qs[0]['lang__lang']
  456. order_type = 1 if is_ai else 0
  457. now_time = int(time.time())
  458. store_meal_qs = Store_Meal.objects.filter(id=rank_id, lang__lang='cn', is_show=0). \
  459. values('lang__title', 'lang__content')
  460. if store_meal_qs.exists():
  461. store_meal_name = store_meal_qs[0]['lang__title'] + '-' + store_meal_qs[0]['lang__content']
  462. else:
  463. store_meal_name = '未知套餐'
  464. # 创建订单
  465. order = Order_Model.objects.create(
  466. orderID=order_id, UID=uid, channel=channel, userID_id=user_id, desc=content,
  467. payType=pay_type, payTime=now_time, price=price, currency=currency, addTime=now_time,
  468. updTime=now_time, order_type=order_type, commodity_code=commodity_code,
  469. commodity_type=commodity_type, rank_id=rank_id, ai_rank_id=1, status=1,
  470. store_meal_name=store_meal_name, app_type=app_type
  471. )
  472. # 充值云存套餐
  473. uid_bucket_id = cls.enable_cloud(channel, now_time, order_id, store_qs, uid)
  474. # 修改订单信息
  475. order.uid_bucket_id = uid_bucket_id
  476. order.transaction_id = transaction_id
  477. order.original_transaction_id = original_transaction_id
  478. order.save()
  479. # 构建云存套餐消息
  480. sys_msg_text_list = cls.cloud_storage_message(uid)
  481. cls.do_vod_msg_notice(uid, user_id, lang, sys_msg_text_list)
  482. return HttpResponse(status=200)
  483. elif str(decoded_payload.rawNotificationType) == "SUBSCRIBED":
  484. # 处理订阅 ---> 首次充值逻辑写在了认证交易
  485. if decoded_payload.rawSubtype == "RESUBSCRIBE":
  486. decoded_transaction_information = signed_data_verifier.verify_and_decode_signed_transaction(
  487. decoded_payload.data.signedTransactionInfo)
  488. # originalTransactionId 原始购买的交易标识符
  489. original_transaction_id = decoded_transaction_information.originalTransactionId
  490. transaction_id = decoded_transaction_information.transactionId
  491. app_account_token = decoded_transaction_information.appAccountToken
  492. app_account_token = json.loads(app_account_token)
  493. logger.info(
  494. f"App Store服务器通知,再次订阅originalTransactionId原始购买的交易标识符{original_transaction_id}")
  495. if not original_transaction_id:
  496. logger.info(f"App Store服务器通知 originalTransactionId原始购买的交易标识符为空, 返回状态 400")
  497. return HttpResponse(status=400)
  498. # 查旧订单消息
  499. ord_order_qs = Order_Model.objects.filter(original_transaction_id=original_transaction_id)
  500. if not ord_order_qs.exists():
  501. logger.info(f"App Store服务器通知未查询到旧订单信息, originalTransactionId:{original_transaction_id}, 返回状态 400")
  502. return HttpResponse(status=400)
  503. # 解决云存充值成功, 由于一些原因返回500 导致苹果未扣款的问题
  504. if Order_Model.objects.filter(transaction_id=transaction_id, status=1).exists():
  505. logger.info(
  506. f"App Store服务器通知云存续订订单已存在, transactionId:{transaction_id} 返回状态 200")
  507. return HttpResponse(status=200)
  508. ord_order = ord_order_qs.order_by('-addTime').values("channel", "UID", "payType", "userID_id")
  509. channel = ord_order[0]["channel"]
  510. uid = ord_order[0]["UID"]
  511. pay_type = ord_order[0]["payType"]
  512. user_id = ord_order[0]["userID_id"]
  513. new_user_id = app_account_token["user_id"]
  514. if new_user_id != user_id:
  515. uid = app_account_token["UID"]
  516. user_id = new_user_id
  517. # 用产品id找到使用的套餐
  518. product_id = decoded_transaction_information.productId
  519. rank_id = InAppPurchasePackage.objects.filter(product_id=product_id).values("rank")[0]["rank"]
  520. store_qs = Store_Meal.objects.filter(id=rank_id). \
  521. values(
  522. 'id', 'currency', 'price', 'lang__content', 'day', 'commodity_type', 'lang__title',
  523. 'expire', 'lang__lang',
  524. 'commodity_code', 'discount_price', 'bucket_id', 'bucket__mold', 'cycle_config_id', 'is_ai')
  525. if not store_qs.exists():
  526. logger.info(f"App Store服务器通知云存套餐不存在, 返回状态 400")
  527. return HttpResponse(status=400)
  528. order_id = CommonService.createOrderID()
  529. rank_id = store_qs[0]['id']
  530. currency = store_qs[0]['currency']
  531. price = store_qs[0]['price']
  532. is_ai = store_qs[0]['is_ai']
  533. content = store_qs[0]['lang__content']
  534. commodity_code = store_qs[0]['commodity_code']
  535. commodity_type = store_qs[0]['commodity_type']
  536. lang = store_qs[0]['lang__lang']
  537. order_type = 1 if is_ai else 0
  538. now_time = int(time.time())
  539. store_meal_qs = Store_Meal.objects.filter(id=rank_id, lang__lang='cn', is_show=0). \
  540. values('lang__title', 'lang__content')
  541. if store_meal_qs.exists():
  542. store_meal_name = store_meal_qs[0]['lang__title'] + '-' + store_meal_qs[0]['lang__content']
  543. else:
  544. store_meal_name = '未知套餐'
  545. order = Order_Model.objects.create(
  546. orderID=order_id, UID=uid, channel=channel, userID_id=user_id, desc=content,
  547. payType=pay_type, payTime=now_time, price=price, currency=currency, addTime=now_time,
  548. updTime=now_time, order_type=order_type, commodity_code=commodity_code,
  549. commodity_type=commodity_type, rank_id=rank_id, ai_rank_id=1, status=1,
  550. store_meal_name=store_meal_name, app_type=app_type
  551. )
  552. # 充值云存套餐
  553. uid_bucket_id = cls.enable_cloud(channel, now_time, order_id, store_qs, uid)
  554. # 修改订单信息
  555. order.uid_bucket_id = uid_bucket_id
  556. order.transaction_id = transaction_id
  557. order.original_transaction_id = original_transaction_id
  558. order.save()
  559. DeviceApplePackage.objects.filter(userID=user_id, uid=uid).update(subscription_status=1,
  560. update_time=int(time.time()))
  561. # 构建云存套餐消息
  562. sys_msg_text_list = cls.cloud_storage_message(uid)
  563. cls.do_vod_msg_notice(uid, user_id, lang, sys_msg_text_list)
  564. return HttpResponse(status=200)
  565. elif str(decoded_payload.rawNotificationType) == "EXPIRED":
  566. # 一种通知类型,与其子类型一起表示订阅已过期。如果subtype为
  567. # VOLUNTARY(自愿),则表示订阅在用户禁用订阅续订后过期。如果subtype是
  568. # BILLING_RETRY(计费重试),则表示订阅过期,因为计费重试期结束时没有成功的计费交易。如果subtype为
  569. # PRICE_INCREASE,则表示订阅已过期,因为客户不同意需要客户同意的价格上涨。如果subtype为
  570. # PRODUCT_NOT_FOR_SALE,则表示订阅已过期,因为在订阅尝试续订时,产品已不可购买。
  571. # 没有子类型的通知表示订阅因其他原因过期。
  572. decoded_transaction_information = signed_data_verifier.verify_and_decode_signed_transaction(
  573. decoded_payload.data.signedTransactionInfo)
  574. # originalTransactionId 原始购买的交易标识符
  575. original_transaction_id = decoded_transaction_information.originalTransactionId
  576. if not original_transaction_id:
  577. logger.info(f"App Store服务器通知 originalTransactionId原始购买的交易标识符为空, 返回状态 400")
  578. return HttpResponse(status=400)
  579. device_apple_package_qs = DeviceApplePackage.objects.filter(
  580. original_transaction_id=original_transaction_id)
  581. if not device_apple_package_qs.exists():
  582. return HttpResponse(status=400)
  583. device_apple_package_qs.update(subscription_status=2, update_time=int(time.time()))
  584. elif str(decoded_payload.rawNotificationType) == "CONSUMPTION_REQUEST":
  585. # 一种通知类型,指示客户发起了消费型 App 内购买项目或自动续期订阅的退款请求,并且 App Store 要求您提供消费数据。有关详细信息,请参阅发送消耗信息。
  586. decoded_transaction_information = signed_data_verifier.verify_and_decode_signed_transaction(
  587. decoded_payload.data.signedTransactionInfo)
  588. transaction_id = decoded_transaction_information.transactionId
  589. app_account_token = decoded_transaction_information.appAccountToken
  590. if not app_account_token:
  591. app_account_token = ""
  592. orders_qs = Order_Model.objects.filter(transaction_id=transaction_id)
  593. if not orders_qs.exists():
  594. return HttpResponse(status=400)
  595. orderID = orders_qs[0].orderID
  596. uid = orders_qs[0].UID
  597. now_time = int(time.time())
  598. put_time = now_time + 11.5 * 60 * 60
  599. in_app_refund_qs = InAppRefund.objects.filter(transaction_id=transaction_id)
  600. if in_app_refund_qs.exists():
  601. in_app_refund_qs.update(refund_progress=0, updated_time=now_time,
  602. put_time=put_time, app_account_token=app_account_token)
  603. InAppRefund.objects.create(transaction_id=transaction_id, orderID=orderID,
  604. uid=uid, app_type=app_type, created_time=now_time,
  605. updated_time=now_time, put_time=put_time,
  606. app_account_token=app_account_token)
  607. return HttpResponse(status=200)
  608. elif str(decoded_payload.rawNotificationType) == "DID_CHANGE_RENEWAL_STATUS":
  609. decoded_transaction_information = signed_data_verifier.verify_and_decode_signed_transaction(
  610. decoded_payload.data.signedTransactionInfo)
  611. original_transaction_id = decoded_transaction_information.originalTransactionId
  612. if not original_transaction_id:
  613. logger.info(f"App Store服务器通知 originalTransactionId原始购买的交易标识符为空, 返回状态 400")
  614. return HttpResponse(status=400)
  615. if decoded_payload.rawSubtype == "AUTO_RENEW_ENABLED":
  616. # 自动续订被开启
  617. device_apple_package_qs = DeviceApplePackage.objects.filter(
  618. original_transaction_id=original_transaction_id)
  619. if not device_apple_package_qs.exists():
  620. return HttpResponse(status=400)
  621. device_apple_package_qs.update(subscription_status=1, update_time=int(time.time()))
  622. else:
  623. # 自动续订被禁用
  624. device_apple_package_qs = DeviceApplePackage.objects.filter(
  625. original_transaction_id=original_transaction_id)
  626. if not device_apple_package_qs.exists():
  627. return HttpResponse(status=400)
  628. device_apple_package_qs.update(subscription_status=3, update_time=int(time.time()))
  629. elif str(decoded_payload.rawNotificationType) == "REFUND":
  630. # 一种通知类型,表示 App Store 成功退还了消耗性应用内购买、非消耗性应用内购买、自动续订或不可续订的交易。
  631. # revocationDate 包含退款交易的时间戳。originalTransactionId 和 productId 用于标识原始交易和产品。revocationReason 包含原因。
  632. # 要请求客户所有退款交易的列表,请参阅 App Store 服务器 API 中的获取退款历史记录。
  633. decoded_transaction_information = signed_data_verifier.verify_and_decode_signed_transaction(
  634. decoded_payload.data.signedTransactionInfo)
  635. transaction_id = decoded_transaction_information.transactionId
  636. logger.info('App Store服务器通知退款, transaction_id:{}'.format(transaction_id))
  637. orders_qs = Order_Model.objects.filter(transaction_id=transaction_id)
  638. if not orders_qs.exists():
  639. return HttpResponse(status=400)
  640. orders_qs.update(status=5, updTime=int(time.time()))
  641. orderID = orders_qs[0].orderID
  642. uid = orders_qs[0].UID
  643. uid_bucket_qs = UID_Bucket.objects.filter(uid=uid, orderId=orderID, use_status=1,
  644. endTime__gt=int(time.time()))
  645. unused_uid_meal_qs = Unused_Uid_Meal.objects.filter(order_id=orderID)
  646. ai_service_qs = AiService.objects.filter(uid=uid, orders=orderID, use_status=1,
  647. endTime__gt=int(time.time()))
  648. if unused_uid_meal_qs.exists():
  649. unused_uid_meal_qs.delete()
  650. if uid_bucket_qs.exists():
  651. uid_bucket_qs.update(status=0, use_status=2, endTime=int(time.time()),
  652. updateTime=int(time.time()))
  653. if ai_service_qs.exists():
  654. ai_service_qs.update(detect_status=0, use_status=2, endTime=int(time.time()),
  655. updTime=int(time.time()))
  656. # 关闭ai
  657. msg = {'commandType': 'AIDisable'}
  658. thing_name = CommonService.query_serial_with_uid(uid) # 存在序列号则为使用序列号作为物品名
  659. topic_name = 'ansjer/generic/{}'.format(thing_name)
  660. req_success = CommonService.req_publish_mqtt_msg(thing_name, topic_name, msg)
  661. logger.info(f'App Store服务器通知用户退款, 关闭AI:{req_success}')
  662. InAppRefund.objects.filter(transaction_id=transaction_id).update(updated_time=int(time.time()),
  663. refund_progress=2)
  664. return HttpResponse(status=200)
  665. elif str(decoded_payload.rawNotificationType) == "REFUND_REVERSED":
  666. # 一种通知类型,表示 App Store 由于客户提出的争议而撤销了先前批准的退款。如果您的应用程序因相关退款而撤销了内容或服务,则需要恢复这些内容或服务。
  667. # 此通知类型可适用于任何应用程序内购买类型:消耗品、非消耗品、不可续订订阅和自动续订订阅。对于自动续订,当 App Store 撤销退款时,续订日期保持不变。
  668. decoded_transaction_information = signed_data_verifier.verify_and_decode_signed_transaction(
  669. decoded_payload.data.signedTransactionInfo)
  670. transaction_id = decoded_transaction_information.transactionId
  671. logger.info(
  672. 'App Store服务器通知,撤销了批准的退款,不恢复套餐,手动处理 transaction_id:{}'.format(transaction_id))
  673. return HttpResponse(status=200)
  674. elif str(decoded_payload.rawNotificationType) == "REFUND_DECLINED":
  675. decoded_transaction_information = signed_data_verifier.verify_and_decode_signed_transaction(
  676. decoded_payload.data.signedTransactionInfo)
  677. transaction_id = decoded_transaction_information.transactionId
  678. orders_qs = Order_Model.objects.filter(transaction_id=transaction_id)
  679. if not orders_qs.exists():
  680. return HttpResponse(status=400)
  681. logger.info(
  682. 'App Store服务器通知, 不批准退款, transaction_id:{}'.format(transaction_id))
  683. InAppRefund.objects.filter(transaction_id=transaction_id).update(refund_progress=3)
  684. else:
  685. logger.info(f"App Store服务器通知decoded_payload.rawNotificationType 未处理")
  686. return HttpResponse(status=200)
  687. except Exception as e:
  688. logger.info('App Store服务器通知异常:{}'.
  689. format('error_line:{}, error_msg:{}'.format(e.__traceback__.tb_lineno, repr(e))))
  690. return HttpResponse(status=500)
  691. @classmethod
  692. def app_store_server_notifications_vsees(cls, request, request_dict):
  693. logger = logging.getLogger('apple_pay')
  694. if request.method != 'POST':
  695. logger.info(f'App Store服务器通知不是post请求, 参数{request_dict}')
  696. return HttpResponse(status=400)
  697. try:
  698. request_data = json.loads(request.body)
  699. except json.JSONDecodeError:
  700. logger.error('无法解析请求体为JSON')
  701. return HttpResponse(status=400)
  702. request_data['bundleId'] = 'com.ansjer.zccloud'
  703. updated_request_body = json.dumps(request_data)
  704. if CONFIG_INFO == CONFIG_TEST:
  705. logger.info('测试环境, App Store服务器通知发送到测试服')
  706. response_test = requests.post(url="https://test.zositechc.cn/inAppPurchase/serverNotifications",
  707. json=updated_request_body)
  708. return HttpResponse(status=response_test.status_code)
  709. response_us = requests.post(url="https://www.dvema.com/inAppPurchase/serverNotifications",
  710. json=updated_request_body)
  711. status_code = response_us.status_code
  712. if status_code != 200:
  713. response_eu = requests.post(url="https://api.zositeche.com/inAppPurchase/serverNotifications",
  714. json=updated_request_body)
  715. status_code = response_eu.status_code
  716. if status_code == 200:
  717. return HttpResponse(status=200)
  718. return HttpResponse(status=400)
  719. @classmethod
  720. def vsees_notifications(cls, request):
  721. logger = logging.getLogger('apple_pay')
  722. logger.info('Vsees: App Store服务器通知请求类型:{}'.format(request.method))
  723. logger.info('Vsees: App Store服务器通知参数:{}'.format(request.POST))
  724. logger.info('Vsees: App Store服务器通知请求body:{}'.format(request.body))
  725. payload = json.loads(request.body.decode('utf-8'))
  726. logger.info('Vsees: App Store服务器通知payload:{}'.format(payload))
  727. # 获取 signedPayload
  728. signed_payload = payload.get('signedPayload')
  729. if not signed_payload:
  730. return HttpResponse(status=400)
  731. in_app_purchase = InAppPurchase(bundle_id="com.cloudlife.commissionf")
  732. # SignedDataVerifier 用于解析查询到的交易信息
  733. verifier = in_app_purchase.verifier
  734. decoded_payload = verifier.verify_and_decode_notification(signed_payload)
  735. logger.info('Vsees: App Store服务器通知decoded_payload: {}'.format(decoded_payload))
  736. status_code = 200
  737. if str(decoded_payload.rawNotificationType) == "REFUND":
  738. # 一种通知类型,表示 App Store 成功退还了消耗性应用内购买、非消耗性应用内购买、自动续订或不可续订的交易。
  739. # revocationDate 包含退款交易的时间戳。originalTransactionId 和 productId 用于标识原始交易和产品。revocationReason 包含原因。
  740. # 要请求客户所有退款交易的列表,请参阅 App Store 服务器 API 中的获取退款历史记录。
  741. # 1. 找套餐 使用 transaction_id 找orders
  742. decoded_transaction_information = verifier.verify_and_decode_signed_transaction(
  743. decoded_payload.data.signedTransactionInfo)
  744. transaction_id = decoded_transaction_information.transactionId
  745. logger.info('Vsees: App Store服务器通知退款, transaction_id:{}'.format(transaction_id))
  746. orders_qs = Order_Model.objects.filter(transaction_id=transaction_id)
  747. # 2. 查找云存套餐使用表 和 云存套餐
  748. if orders_qs.exists():
  749. orders_qs.update(status=11)
  750. orderID = orders_qs[0].orderID
  751. uid = orders_qs[0].UID
  752. user_id = orders_qs[0].userID
  753. # 3. 未使用则删除未使用套餐表,已使用过则删除设备正在使用套餐,并关闭设备云存
  754. uid_bucket_qs = UID_Bucket.objects.filter(uid=uid, orderId=orderID, use_status=1,
  755. endTime__gt=int(time.time()))
  756. unused_uid_meal_qs = Unused_Uid_Meal.objects.filter(order_id=orderID)
  757. ai_service_qs = AiService.objects.filter(uid=uid, orderId=orderID, use_status=1,
  758. endTime__gt=int(time.time()))
  759. if unused_uid_meal_qs.exists():
  760. unused_uid_meal_qs.delete()
  761. if uid_bucket_qs.exists():
  762. uid_bucket_qs.update(status=0, use_status=2, endTime=int(time.time()), updateTime=int(time.time()))
  763. if ai_service_qs.exists():
  764. ai_service_qs.update(detect_status=0, use_status=2, endTime=int(time.time()),
  765. updTime=int(time.time()))
  766. # 关闭ai
  767. msg = {'commandType': 'AIDisable'}
  768. thing_name = CommonService.query_serial_with_uid(uid) # 存在序列号则为使用序列号作为物品名
  769. topic_name = 'ansjer/generic/{}'.format(thing_name)
  770. req_success = CommonService.req_publish_mqtt_msg(thing_name, topic_name, msg)
  771. LOGGER.info(f'App Store服务器通知用户退款, 关闭AI:{req_success}')
  772. # 4.发送邮件告知用户退款
  773. email_content = f'{CONFIG_INFO}用户{user_id}, 订单:{orderID}, 设备{uid}退款'
  774. S3Email().faEmail(email_content, 'servers@ansjer.com')
  775. else:
  776. if CONFIG_INFO == CONFIG_US:
  777. url = "https://api.zositeche.com/inAppPurchase/AppStoreServerNotifications"
  778. eur_response = requests.post(url=url, json=json.loads(request.body))
  779. status_code = eur_response.status_code
  780. return HttpResponse(status=status_code)
  781. @staticmethod
  782. def put_refund_order():
  783. put_time = int(time.time())
  784. in_app_refund_qs = InAppRefund.objects.filter(refund_progress=0, put_time__lt=put_time)
  785. for in_app_refund in in_app_refund_qs:
  786. transaction_id = in_app_refund.transaction_id
  787. app_type = in_app_refund.app_type
  788. if app_type == 1:
  789. bundle_id = "com.ansjer.zccloud"
  790. elif app_type == 2:
  791. bundle_id = "com.cloudlife.commissionf"
  792. else:
  793. return HttpResponse(status=200)
  794. in_app_purchase_obj = InAppPurchase(bundle_id=bundle_id)
  795. # AppStoreServerAPIClient 用于查询交易信息
  796. client = in_app_purchase_obj.client
  797. orderID = in_app_refund.orderID
  798. app_account_token = in_app_refund.app_account_token
  799. unused_uid_meal_qs = Unused_Uid_Meal.objects.filter(order_id=orderID)
  800. uid_bucket_qs = UID_Bucket.objects.filter(orderId=orderID, endTime__gt=int(time.time()))
  801. if unused_uid_meal_qs.exists():
  802. consumptionStatus = ConsumptionStatus.NOT_CONSUMED
  803. deliveryStatus = DeliveryStatus.DELIVERED_AND_WORKING_PROPERLY
  804. elif uid_bucket_qs.exists():
  805. consumptionStatus = ConsumptionStatus.PARTIALLY_CONSUMED
  806. deliveryStatus = DeliveryStatus.DELIVERED_AND_WORKING_PROPERLY
  807. elif UID_Bucket.objects.filter(orderId=orderID, endTime__lt=int(time.time())):
  808. consumptionStatus = ConsumptionStatus.FULLY_CONSUMED
  809. deliveryStatus = DeliveryStatus.DELIVERED_AND_WORKING_PROPERLY
  810. else:
  811. consumptionStatus = ConsumptionStatus.UNDECLARED
  812. deliveryStatus = DeliveryStatus.DID_NOT_DELIVER_FOR_OTHER_REASON
  813. if in_app_refund.refund_preference == 1:
  814. refundPreference = RefundPreference.PREFER_GRANT
  815. else:
  816. refundPreference = RefundPreference.PREFER_DECLINE
  817. consumption_request = ConsumptionRequest(
  818. customerConsented=True,
  819. consumptionStatus=consumptionStatus,
  820. platform=Platform.UNDECLARED,
  821. sampleContentProvided=True,
  822. deliveryStatus=deliveryStatus,
  823. appAccountToken=app_account_token,
  824. accountTenure=AccountTenure.UNDECLARED,
  825. playTime=PlayTime.UNDECLARED,
  826. lifetimeDollarsRefunded=LifetimeDollarsRefunded.UNDECLARED,
  827. lifetimeDollarsPurchased=LifetimeDollarsPurchased.UNDECLARED,
  828. userStatus=UserStatus.ACTIVE,
  829. refundPreference=refundPreference,
  830. )
  831. client.send_consumption_data(transaction_id, consumption_request)
  832. logger.info(f'内购退款消费数据提交, 订单orderID:{orderID}, transaction_id:{transaction_id}')
  833. in_app_refund.refund_progress = 1
  834. in_app_refund.updated_time = int(time.time())
  835. in_app_refund.put_time = int(time.time())
  836. in_app_refund.save()
  837. return HttpResponse(status=200)