| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710 | 
							- # @Author    : Rocky
 
- # @File      : InAppPurchaseController.py
 
- # @Time      : 2024/6/21 9:10
 
- import logging
 
- import time
 
- import json
 
- from appstoreserverlibrary.api_client import AppStoreServerAPIClient, GetTransactionHistoryVersion
 
- from appstoreserverlibrary.models.Environment import Environment
 
- from appstoreserverlibrary.receipt_utility import ReceiptUtility
 
- from appstoreserverlibrary.models.HistoryResponse import HistoryResponse
 
- from appstoreserverlibrary.models.TransactionHistoryRequest import TransactionHistoryRequest, ProductType, Order
 
- from appstoreserverlibrary.signed_data_verifier import SignedDataVerifier
 
- from cryptography.hazmat.backends import default_backend
 
- from cryptography.hazmat.primitives.serialization import load_pem_private_key
 
- from django.db.models import Q
 
- from django.views import View
 
- from django.http import HttpResponse
 
- from Ansjer.config import LOGGER, CONFIG_INFO, CONFIG_TEST, PAY_TYPE_IN_APP_PURCHASE, BASE_DIR
 
- from Controller.CheckUserData import DataValid
 
- from Model.models import Order_Model, Store_Meal, Device_Info, UID_Bucket, Unused_Uid_Meal, AiService, Device_User, \
 
-     SysMsgModel, DeviceApplePackage, InAppPurchasePackage
 
- from Object.AWS.S3Email import S3Email
 
- from Object.AliSmsObject import AliSmsObject
 
- from Object.RedisObject import RedisObject
 
- from Service.CommonService import CommonService
 
- ENV = Environment.SANDBOX if CONFIG_INFO == CONFIG_TEST else Environment.PRODUCTION
 
- class InAppPurchaseView(View):
 
-     def get(self, request, *args, **kwargs):
 
-         request.encoding = 'utf-8'
 
-         operation = kwargs.get('operation')
 
-         return self.validation(request.GET, request, operation)
 
-     def post(self, request, *args, **kwargs):
 
-         request.encoding = 'utf-8'
 
-         operation = kwargs.get('operation')
 
-         return self.validation(request.POST, request, operation)
 
-     def validation(self, request_dict, request, operation):
 
-         if operation == 'AppStoreServerNotifications':  # App Store服务器通知
 
-             return self.app_store_server_notifications(request, request_dict)
 
-         token_code, user_id, response = CommonService.verify_token_get_user_id(request_dict, request)
 
-         if token_code != 0:
 
-             return response.json(token_code)
 
-         if operation == 'verifyTransaction':  # 认证交易
 
-             return self.verify_transaction(user_id, request_dict, response)
 
-     @classmethod
 
-     def verify_transaction(cls, user_id, request_dict, response):
 
-         """
 
-         认证交易
 
-         @param user_id: 用户id
 
-         @param request_dict: 请求参数
 
-         @request_dict receipt: 收据
 
-         @param response: 响应对象
 
-         @return: response
 
-         """
 
-         receipt = request_dict.get('receipt', None)
 
-         transaction_identifier = request_dict.get('transactionIdentifier', "")
 
-         original_transaction_identifier = request_dict.get('originalTransactionIdentifier', "")
 
-         order_id = request_dict.get('orderID', None)
 
-         uid = request_dict.get('uid', None)
 
-         lang = request_dict.get('lang', 'en')
 
-         channel = request_dict.get('channel', None)
 
-         logger = logging.getLogger('apple_pay')
 
-         logger.info(
 
-             f"transactionIdentifier:{transaction_identifier}, original_transaction_identifier:{original_transaction_identifier} ,订单orderId:{order_id}")
 
-         if not all([receipt, uid, channel, order_id]):
 
-             return response.json(444)
 
-         # redis加锁,防止订单重复
 
-         redis_obj = RedisObject()
 
-         redis_key = order_id + 'in_app_purchase'
 
-         is_lock = redis_obj.CONN.setnx(redis_key, 1)
 
-         redis_obj.CONN.expire(redis_key, 60)
 
-         # if not is_lock:
 
-         #     return response.json(5)
 
-         try:
 
-             # 从交易信息中获取product_id
 
-             key_path = '{}/Ansjer/file/in_app_purchase/SubscriptionKey_N42WMFCV6A.p8'.format(BASE_DIR)
 
-             with open(key_path, 'rb') as file:
 
-                 # 读取文件内容
 
-                 private_key = file.read()
 
-             key_id = 'N42WMFCV6A'
 
-             issuer_id = '69a6de8c-789b-47e3-e053-5b8c7c11a4d1'
 
-             bundle_id = 'com.ansjer.zccloud'
 
-             environment = ENV
 
-             client = AppStoreServerAPIClient(private_key, key_id, issuer_id, bundle_id, environment)
 
-             transaction_id = transaction_identifier
 
-             if transaction_identifier == "":
 
-                 receipt_util = ReceiptUtility()
 
-                 transaction_id = receipt_util.extract_transaction_id_from_app_receipt(receipt)
 
-             if transaction_id is None:
 
-                 pay_result_url = CommonService.get_payment_status_url(lang, 'fail')
 
-                 return response.json(0, {'url': pay_result_url})
 
-             transaction_info = client.get_transaction_info(transaction_id)
 
-             signed_transaction_info = transaction_info.signedTransactionInfo
 
-             if original_transaction_identifier != "":
 
-                 device_apple_package_qs = DeviceApplePackage.objects.filter(
 
-                     original_transaction_id=original_transaction_identifier)
 
-                 if device_apple_package_qs.exists():
 
-                     # 第一种情况: 套餐已过期再次订阅
 
-                     if device_apple_package_qs[0].uid == uid and device_apple_package_qs[0].subscription_status == 2:
 
-                         # 使用App Store服务器通知接口订阅
 
-                         pay_result_url = CommonService.get_payment_status_url(lang, 'success')
 
-                         return response.json(0, {'url': pay_result_url})
 
-                     # 第二种情况: 套餐未过期已取消再次订阅
 
-                     elif device_apple_package_qs[0].uid == uid and device_apple_package_qs[0].subscription_status == 3:
 
-                         # 使用App Store服务器通知接口修改订阅状态
 
-                         pay_result_url = CommonService.get_payment_status_url(lang, 'fail')
 
-                         return response.json(0, {'url': pay_result_url})
 
-                     # 第三种情况: 首次订阅
 
-                     elif device_apple_package_qs[0].uid == uid and device_apple_package_qs[0].subscription_status == 0:
 
-                         logger.info(f"首次订阅直接充值,orderID:{order_id}")
 
-                     else:
 
-                         logger.info(f"错误调用此借口,orderID:{order_id}, uid:{uid}, 订阅状态:{device_apple_package_qs[0].subscription_status}")
 
-                         pay_result_url = CommonService.get_payment_status_url(lang, 'fail')
 
-                         return response.json(0, {'url': pay_result_url})
 
-             # transaction_id相同的情况 ---- 本次订阅未过期,用户在苹果设置中将订阅重新打开时会传上次订阅相同的 transaction_id。
 
-             if Order_Model.objects.filter(payType=5, transaction_id=transaction_id).exists():
 
-                 logger.info(f"该transaction_id已订阅过:{transaction_id}")
 
-                 return response.json(10048)
 
-             root_certificates = []
 
-             for cert_name in [
 
-                 'AppleIncRootCertificate.cer', 'AppleComputerRootCertificate.cer',
 
-                 'AppleRootCA-G2.cer', 'AppleRootCA-G3.cer'
 
-             ]:
 
-                 cert_path = '{}/Ansjer/file/in_app_purchase/{}'.format(BASE_DIR, cert_name)
 
-                 with open(cert_path, 'rb') as file:
 
-                     # 读取文件内容
 
-                     root_certificates.append(file.read())
 
-             enable_online_checks = True
 
-             app_apple_id = None  # 生产环境必需
 
-             signed_data_verifier = SignedDataVerifier(
 
-                 root_certificates, enable_online_checks, environment, bundle_id, app_apple_id)
 
-             payload = signed_data_verifier.verify_and_decode_signed_transaction(signed_transaction_info)
 
-             product_id = None
 
-             original_transaction_id = ""
 
-             if payload and payload.productId:
 
-                 product_id = payload.productId
 
-             if not product_id:
 
-                 pay_result_url = CommonService.get_payment_status_url(lang, 'fail')
 
-                 return response.json(0, {'url': pay_result_url})
 
-             if payload.rawType == "Auto-Renewable Subscription":
 
-                 original_transaction_id = payload.originalTransactionId
 
-             now_time = int(time.time())
 
-             order_qs = Order_Model.objects.filter(orderID=order_id, UID=uid).values("rank_id")
 
-             if not order_qs.exists():
 
-                 return response.json(173, "订单不存在")
 
-             store_qs = Store_Meal.objects.filter(id=order_qs[0]['rank_id']).values(
 
-                 'id', 'currency', 'price', 'lang__content', 'day', 'commodity_type', 'lang__title', 'expire',
 
-                 'commodity_code', 'discount_price', 'bucket_id', 'bucket__mold', 'cycle_config_id', 'is_ai')
 
-             if not store_qs.exists():
 
-                 return response.json(173, "套餐不存在")
 
-             bucket_id = store_qs[0]['bucket_id']
 
-             is_ai = store_qs[0]['is_ai']
 
-             expire = store_qs[0]['expire']
 
-             end_time = CommonService.calcMonthLater(expire)
 
-             # 查询设备是否已开过云存
 
-             use_flag = True
 
-             uid_bucket_qs = UID_Bucket.objects.filter(uid=uid). \
 
-                 values('id', 'bucket_id', 'bucket__region', 'endTime', 'use_status')
 
-             if uid_bucket_qs.exists():
 
-                 uid_bucket = uid_bucket_qs.first()
 
-                 uid_bucket_id = uid_bucket['id']
 
-                 # 叠加相同套餐的过期时间
 
-                 if uid_bucket['use_status'] == 1 and uid_bucket['endTime'] > now_time:
 
-                     Unused_Uid_Meal.objects.create(
 
-                         uid=uid, channel=channel, addTime=now_time, order_id=order_id, expire=expire, is_ai=is_ai,
 
-                         bucket_id=bucket_id)
 
-                     UID_Bucket.objects.filter(id=uid_bucket_id).update(has_unused=1)
 
-                     use_flag = False
 
-                 # 更新套餐的过期时间
 
-                 else:
 
-                     UID_Bucket.objects.filter(id=uid_bucket_id).update(
 
-                         channel=channel, bucket_id=bucket_id, endTime=end_time, updateTime=now_time, use_status=1,
 
-                         orderId=order_id)
 
-             else:
 
-                 uid_bucket = UID_Bucket.objects.create(
 
-                     uid=uid, channel=channel, bucket_id=bucket_id, endTime=end_time, use_status=1, orderId=order_id,
 
-                     addTime=now_time, updateTime=now_time)
 
-                 uid_bucket_id = uid_bucket.id
 
-             # 开通AI服务
 
-             if is_ai and use_flag:
 
-                 ai_service = AiService.objects.filter(uid=uid, channel=channel)
 
-                 # 有正在使用的套餐,叠加套餐时间,否则创建
 
-                 if ai_service.exists():
 
-                     ai_service.update(updTime=now_time, use_status=1, orders_id=order_id, endTime=end_time)
 
-                 else:
 
-                     AiService.objects.create(
 
-                         uid=uid, channel=channel, detect_status=1, use_status=1, orders_id=order_id,
 
-                         addTime=now_time, updTime=now_time, endTime=end_time)
 
-             # 修改订阅状态
 
-             if payload.rawType == "Auto-Renewable Subscription":
 
-                 in_app_purchase_package_qs = InAppPurchasePackage.objects.filter(product_id=product_id)
 
-                 if not in_app_purchase_package_qs.exists():
 
-                     return response.json(173, "内购套餐不存在")
 
-                 in_app_purchase_package = in_app_purchase_package_qs.values('id').first()
 
-                 package_id = in_app_purchase_package['id']
 
-                 DeviceApplePackage.objects.filter(userID=user_id, uid=uid, package_id=package_id).update(
 
-                     subscription_status=1,
 
-                     original_transaction_id=original_transaction_id,
 
-                     update_time=int(time.time())
 
-                 )
 
-             order_qs.update(status=1, uid_bucket_id=uid_bucket_id,
 
-                             transaction_id=transaction_id, create_vod=1,
 
-                             original_transaction_id=original_transaction_id)
 
-             # 发送云存开通信息
 
-             date_time = time.strftime("%Y-%m-%d", time.localtime())
 
-             # 如果存在序列号,消息提示用序列号
 
-             device_info_qs = Device_Info.objects.filter(UID=uid).values('serial_number', 'Type')
 
-             serial_number = device_info_qs[0]['serial_number']
 
-             device_type = device_info_qs[0]['Type']
 
-             if serial_number:
 
-                 device_name = CommonService.get_full_serial_number(uid, serial_number, device_type)
 
-             else:
 
-                 device_name = uid
 
-             sys_msg_text_list = [
 
-                 '温馨提示:尊敬的客户,您的{}设备在{}已成功购买云存套餐'.format(device_name, date_time),
 
-                 'Dear customer,you already subscribed the cloud storage package successfully for device {} on '.
 
-                 format(device_name, time.strftime('%b %dth,%Y', time.localtime()))]
 
-             cls.do_vod_msg_notice(uid, user_id, lang, sys_msg_text_list)
 
-             redis_obj.del_data(redis_key)
 
-             pay_result_url = CommonService.get_payment_status_url(lang, 'success')
 
-             return response.json(0, {'url': pay_result_url})
 
-         except Exception as e:
 
-             redis_obj.del_data(redis_key)
 
-             LOGGER.info('苹果内购认证交易接口异常:{}'.
 
-                         format('error_line:{}, error_msg:{}'.format(e.__traceback__.tb_lineno, repr(e))))
 
-             pay_result_url = CommonService.get_payment_status_url(lang, 'fail')
 
-             return response.json(0, {'url': pay_result_url})
 
-     @classmethod
 
-     def do_vod_msg_notice(cls, uid, user_id, lang, sys_msg_text_list):
 
-         """
 
-         发送云存开通信息
 
-         @param uid: uid
 
-         @param user_id: 用户id
 
-         @param lang: 语言
 
-         @param sys_msg_text_list: 消息列表
 
-         @return: response
 
-         """
 
-         if lang == 'cn':
 
-             sys_msg_text = sys_msg_text_list[0]
 
-         else:
 
-             sys_msg_text = sys_msg_text_list[1]
 
-         now_time = int(time.time())
 
-         create_data = {
 
-             'userID_id': user_id,
 
-             'msg': sys_msg_text,
 
-             'addTime': now_time,
 
-             'updTime': now_time,
 
-             'uid': uid,
 
-             'eventType': 0
 
-         }
 
-         SysMsgModel.objects.create(**create_data)
 
-         # 不接收邮件用户
 
-         if user_id == '167015836969813800138000':
 
-             return
 
-         user_qs = Device_User.objects.filter(userID=user_id)
 
-         if user_qs.exists():
 
-             user = user_qs.first()
 
-             username = user.username
 
-             data_valid = DataValid()
 
-             if data_valid.email_validate(username):
 
-                 S3Email().faEmail(sys_msg_text, username)
 
-             elif data_valid.mobile_validate(username):
 
-                 # 如果存在序列号,消息提示用序列号
 
-                 device_info_qs = Device_Info.objects.filter(UID=uid).values('serial_number', 'Type')
 
-                 if device_info_qs.exists():
 
-                     serial_number = device_info_qs[0]['serial_number']
 
-                     device_type = device_info_qs[0]['Type']
 
-                     if serial_number:
 
-                         device_name = CommonService.get_full_serial_number(uid, serial_number, device_type)
 
-                     else:
 
-                         device_name = uid
 
-                     params = '{"devname":"%s","submittime":"%s"}' % (
 
-                         device_name, time.strftime("%Y-%m-%d", time.localtime()))
 
-                     cls.send_message(username, params, 'SMS_219738485')
 
-     @staticmethod
 
-     def send_message(phone, params, temp_msg):
 
-         """
 
-         发送手机消息
 
-         @param phone: 用户名
 
-         @param params: 消息参数
 
-         @param temp_msg: sms码
 
-         """
 
-         sign_ms = '周视'
 
-         ali_sms = AliSmsObject()
 
-         ali_sms.send_code_sms_cloud(phone=phone, params=params, sign_name=sign_ms, temp_msg=temp_msg)
 
-     @classmethod
 
-     def app_store_server_notifications(cls, request, request_dict):
 
-         logger = logging.getLogger('apple_pay')
 
-         try:
 
-             logger.info('App Store服务器通知请求类型:{}'.format(request.method))
 
-             logger.info('App Store服务器通知参数:{}'.format(request.POST))
 
-             logger.info('App Store服务器通知请求body:{}'.format(request.body))
 
-             if request.method != 'POST':
 
-                 logger.info(f'App Store服务器通知不是post请求, 参数{request_dict}')
 
-                 return HttpResponse(status=400)
 
-             payload = json.loads(request.body.decode('utf-8'))
 
-             logger.info('App Store服务器通知payload:{}'.format(payload))
 
-             # 获取 signedPayload
 
-             signed_payload = payload.get('signedPayload')
 
-             if not signed_payload:
 
-                 return HttpResponse(status=400)
 
-             bundle_id = 'com.ansjer.zccloud'
 
-             environment = ENV
 
-             root_certificates = []
 
-             for cert_name in [
 
-                 'AppleIncRootCertificate.cer', 'AppleComputerRootCertificate.cer',
 
-                 'AppleRootCA-G2.cer', 'AppleRootCA-G3.cer'
 
-             ]:
 
-                 cert_path = '{}/Ansjer/file/in_app_purchase/{}'.format(BASE_DIR, cert_name)
 
-                 with open(cert_path, 'rb') as file:
 
-                     # 读取文件内容
 
-                     root_certificates.append(file.read())
 
-             enable_online_checks = True
 
-             app_apple_id = None  # 生产环境必需
 
-             # 验证签名并解码 payload
 
-             verifier = SignedDataVerifier(
 
-                 root_certificates, enable_online_checks, environment, bundle_id, app_apple_id)
 
-             decoded_payload = verifier.verify_and_decode_notification(signed_payload)
 
-             logger.info(f"App Store服务器通知解码后decoded_payload:{decoded_payload}")
 
-             logger.info(
 
-                 f"App Store服务器通知decoded_payload.rawNotificationType{str(decoded_payload.rawNotificationType)}")
 
-             if str(decoded_payload.rawNotificationType) == "DID_RENEW":
 
-                 decoded_transaction_information = verifier.verify_and_decode_signed_transaction(
 
-                     decoded_payload.data.signedTransactionInfo)
 
-                 # originalTransactionId 原始购买的交易标识符
 
-                 original_transaction_id = decoded_transaction_information.originalTransactionId
 
-                 transaction_id = decoded_transaction_information.transactionId
 
-                 logger.info(
 
-                     f"App Store服务器通知originalTransactionId原始购买的交易标识符{original_transaction_id}")
 
-                 if not original_transaction_id:
 
-                     logger.info(f"App Store服务器通知originalTransactionId原始购买的交易标识符为空, 返回状态 400")
 
-                     return HttpResponse(status=400)
 
-                 else:
 
-                     ord_order = Order_Model.objects.filter(original_transaction_id=original_transaction_id).order_by(
 
-                         '-addTime').values("channel", "UID", "payType", "userID_id", "rank_id")
 
-                     if not ord_order.exists():
 
-                         logger.info(f"App Store服务器通知未查询到旧订单信息, 返回状态 400")
 
-                         return HttpResponse(status=400)
 
-                     channel = ord_order[0]["channel"]
 
-                     uid = ord_order[0]["UID"]
 
-                     pay_type = ord_order[0]["payType"]
 
-                     user_id = ord_order[0]["userID_id"]
 
-                     store_qs = Store_Meal.objects.filter(id=ord_order[0]["rank_id"]). \
 
-                         values(
 
-                         'id', 'currency', 'price', 'lang__content', 'day', 'commodity_type', 'lang__title',
 
-                         'expire', 'lang__lang',
 
-                         'commodity_code', 'discount_price', 'bucket_id', 'bucket__mold', 'cycle_config_id', 'is_ai')
 
-                     if not store_qs.exists():
 
-                         logger.info(f"App Store服务器通知云存套餐不存在, 返回状态 400")
 
-                         return HttpResponse(status=400)
 
-                     order_id = CommonService.createOrderID()
 
-                     rank_id = store_qs[0]['id']
 
-                     bucket_id = store_qs[0]['bucket_id']
 
-                     currency = store_qs[0]['currency']
 
-                     price = store_qs[0]['price']
 
-                     is_ai = store_qs[0]['is_ai']
 
-                     expire = store_qs[0]['expire']
 
-                     end_time = CommonService.calcMonthLater(expire)
 
-                     content = store_qs[0]['lang__content']
 
-                     commodity_code = store_qs[0]['commodity_code']
 
-                     commodity_type = store_qs[0]['commodity_type']
 
-                     lang = store_qs[0]['lang__lang']
 
-                     order_type = 1 if is_ai else 0
 
-                     store_meal_qs = Store_Meal.objects.filter(id=rank_id, lang__lang='cn', is_show=0). \
 
-                         values('lang__title', 'lang__content')
 
-                     if store_meal_qs.exists():
 
-                         store_meal_name = store_meal_qs[0]['lang__title'] + '-' + store_meal_qs[0]['lang__content']
 
-                     else:
 
-                         store_meal_name = '未知套餐'
 
-                     # 查询设备是否已开过云存
 
-                     use_flag = True
 
-                     uid_bucket_qs = UID_Bucket.objects.filter(uid=uid). \
 
-                         values('id', 'bucket_id', 'bucket__region', 'endTime', 'use_status')
 
-                     now_time = int(time.time())
 
-                     if uid_bucket_qs.exists():
 
-                         uid_bucket = uid_bucket_qs.first()
 
-                         uid_bucket_id = uid_bucket['id']
 
-                         # 叠加相同套餐的过期时间
 
-                         if uid_bucket['use_status'] == 1 and uid_bucket['endTime'] > now_time:
 
-                             Unused_Uid_Meal.objects.create(
 
-                                 uid=uid, channel=channel, addTime=now_time, order_id=order_id, expire=expire,
 
-                                 is_ai=is_ai,
 
-                                 bucket_id=bucket_id)
 
-                             UID_Bucket.objects.filter(id=uid_bucket_id).update(has_unused=1)
 
-                             use_flag = False
 
-                         # 更新套餐的过期时间
 
-                         else:
 
-                             UID_Bucket.objects.filter(id=uid_bucket_id).update(
 
-                                 channel=channel, bucket_id=bucket_id, endTime=end_time, updateTime=now_time,
 
-                                 use_status=1,
 
-                                 orderId=order_id)
 
-                     else:
 
-                         uid_bucket = UID_Bucket.objects.create(
 
-                             uid=uid, channel=channel, bucket_id=bucket_id, endTime=end_time, use_status=1,
 
-                             orderId=order_id,
 
-                             addTime=now_time, updateTime=now_time)
 
-                         uid_bucket_id = uid_bucket.id
 
-                     # 开通AI服务
 
-                     if is_ai and use_flag:
 
-                         ai_service = AiService.objects.filter(uid=uid, channel=channel)
 
-                         # 有正在使用的套餐,叠加套餐时间,否则创建
 
-                         if ai_service.exists():
 
-                             ai_service.update(updTime=now_time, use_status=1, orders_id=order_id,
 
-                                               endTime=end_time)
 
-                         else:
 
-                             AiService.objects.create(
 
-                                 uid=uid, channel=channel, detect_status=1, use_status=1, orders_id=order_id,
 
-                                 addTime=now_time, updTime=now_time, endTime=end_time)
 
-                     Order_Model.objects.create(
 
-                         orderID=order_id, UID=uid, channel=channel, userID_id=user_id, desc=content,
 
-                         payType=pay_type,
 
-                         payTime=now_time, price=price, currency=currency, addTime=now_time, updTime=now_time,
 
-                         order_type=order_type, commodity_code=commodity_code, commodity_type=commodity_type,
 
-                         rank_id=rank_id,
 
-                         ai_rank_id=1, status=1, create_vod=1, store_meal_name=store_meal_name,
 
-                         uid_bucket_id=uid_bucket_id, transaction_id=transaction_id,
 
-                         original_transaction_id=original_transaction_id
 
-                     )
 
-                     # 发送云存开通信息
 
-                     date_time = time.strftime("%Y-%m-%d", time.localtime())
 
-                     # 如果存在序列号,消息提示用序列号
 
-                     device_info_qs = Device_Info.objects.filter(UID=uid).values('serial_number', 'Type')
 
-                     serial_number = device_info_qs[0]['serial_number']
 
-                     device_type = device_info_qs[0]['Type']
 
-                     if serial_number:
 
-                         device_name = CommonService.get_full_serial_number(uid, serial_number, device_type)
 
-                     else:
 
-                         device_name = uid
 
-                     sys_msg_text_list = [
 
-                         '温馨提示:尊敬的客户,您的{}设备在{}已成功续订云存套餐'.format(device_name, date_time),
 
-                         'Dear customer,you already subscribed the cloud storage package successfully for device {} on '.
 
-                         format(device_name, time.strftime('%b %dth,%Y', time.localtime()))]
 
-                     cls.do_vod_msg_notice(uid, user_id, lang, sys_msg_text_list)
 
-             elif str(decoded_payload.rawNotificationType) == "SUBSCRIBED":
 
-                 # 处理订阅 ---> 首次充值逻辑写在了认证交易
 
-                 if decoded_payload.rawSubtype == "RESUBSCRIBE":
 
-                     decoded_transaction_information = verifier.verify_and_decode_signed_transaction(
 
-                         decoded_payload.data.signedTransactionInfo)
 
-                     # originalTransactionId 原始购买的交易标识符
 
-                     original_transaction_id = decoded_transaction_information.originalTransactionId
 
-                     transaction_id = decoded_transaction_information.transactionId
 
-                     logger.info(
 
-                         f"App Store服务器通知originalTransactionId原始购买的交易标识符{original_transaction_id}")
 
-                     if not original_transaction_id:
 
-                         logger.info(f"App Store服务器通知originalTransactionId原始购买的交易标识符为空, 返回状态 400")
 
-                         return HttpResponse(status=400)
 
-                     else:
 
-                         ord_order = Order_Model.objects.filter(
 
-                             original_transaction_id=original_transaction_id).order_by('-addTime').values("channel",
 
-                                                                                                          "UID",
 
-                                                                                                          "payType",
 
-                                                                                                          "userID_id")
 
-                         channel = ord_order[0]["channel"]
 
-                         uid = ord_order[0]["UID"]
 
-                         pay_type = ord_order[0]["payType"]
 
-                         user_id = ord_order[0]["userID_id"]
 
-                         # 用产品id找到使用的套餐
 
-                         product_id = decoded_transaction_information.productId
 
-                         rank_id = InAppPurchasePackage.objects.filter(product_id=product_id).values("rank")[0]["rank"]
 
-                         store_qs = Store_Meal.objects.filter(id=rank_id). \
 
-                             values(
 
-                             'id', 'currency', 'price', 'lang__content', 'day', 'commodity_type', 'lang__title',
 
-                             'expire', 'lang__lang',
 
-                             'commodity_code', 'discount_price', 'bucket_id', 'bucket__mold', 'cycle_config_id', 'is_ai')
 
-                         if not store_qs.exists():
 
-                             logger.info(f"App Store服务器通知云存套餐不存在, 返回状态 400")
 
-                             return HttpResponse(status=400)
 
-                         order_id = CommonService.createOrderID()
 
-                         rank_id = store_qs[0]['id']
 
-                         bucket_id = store_qs[0]['bucket_id']
 
-                         currency = store_qs[0]['currency']
 
-                         price = store_qs[0]['price']
 
-                         is_ai = store_qs[0]['is_ai']
 
-                         expire = store_qs[0]['expire']
 
-                         end_time = CommonService.calcMonthLater(expire)
 
-                         content = store_qs[0]['lang__content']
 
-                         commodity_code = store_qs[0]['commodity_code']
 
-                         commodity_type = store_qs[0]['commodity_type']
 
-                         lang = store_qs[0]['lang__lang']
 
-                         order_type = 1 if is_ai else 0
 
-                         store_meal_qs = Store_Meal.objects.filter(id=rank_id, lang__lang='cn', is_show=0). \
 
-                             values('lang__title', 'lang__content')
 
-                         if store_meal_qs.exists():
 
-                             store_meal_name = store_meal_qs[0]['lang__title'] + '-' + store_meal_qs[0]['lang__content']
 
-                         else:
 
-                             store_meal_name = '未知套餐'
 
-                         # 查询设备是否已开过云存
 
-                         use_flag = True
 
-                         uid_bucket_qs = UID_Bucket.objects.filter(uid=uid). \
 
-                             values('id', 'bucket_id', 'bucket__region', 'endTime', 'use_status')
 
-                         now_time = int(time.time())
 
-                         if uid_bucket_qs.exists():
 
-                             uid_bucket = uid_bucket_qs.first()
 
-                             uid_bucket_id = uid_bucket['id']
 
-                             # 叠加相同套餐的过期时间
 
-                             if uid_bucket['use_status'] == 1 and uid_bucket['endTime'] > now_time:
 
-                                 Unused_Uid_Meal.objects.create(
 
-                                     uid=uid, channel=channel, addTime=now_time, order_id=order_id, expire=expire,
 
-                                     is_ai=is_ai,
 
-                                     bucket_id=bucket_id)
 
-                                 UID_Bucket.objects.filter(id=uid_bucket_id).update(has_unused=1)
 
-                                 use_flag = False
 
-                             # 更新套餐的过期时间
 
-                             else:
 
-                                 UID_Bucket.objects.filter(id=uid_bucket_id).update(
 
-                                     channel=channel, bucket_id=bucket_id, endTime=end_time, updateTime=now_time,
 
-                                     use_status=1,
 
-                                     orderId=order_id)
 
-                         else:
 
-                             uid_bucket = UID_Bucket.objects.create(
 
-                                 uid=uid, channel=channel, bucket_id=bucket_id, endTime=end_time, use_status=1,
 
-                                 orderId=order_id,
 
-                                 addTime=now_time, updateTime=now_time)
 
-                             uid_bucket_id = uid_bucket.id
 
-                         # 开通AI服务
 
-                         if is_ai and use_flag:
 
-                             ai_service = AiService.objects.filter(uid=uid, channel=channel)
 
-                             # 有正在使用的套餐,叠加套餐时间,否则创建
 
-                             if ai_service.exists():
 
-                                 ai_service.update(updTime=now_time, use_status=1, orders_id=order_id,
 
-                                                   endTime=end_time)
 
-                             else:
 
-                                 AiService.objects.create(
 
-                                     uid=uid, channel=channel, detect_status=1, use_status=1, orders_id=order_id,
 
-                                     addTime=now_time, updTime=now_time, endTime=end_time)
 
-                         DeviceApplePackage.objects.filter(userID=user_id, uid=uid).update(subscription_status=1,
 
-                                                                                           update_time=int(time.time()))
 
-                         Order_Model.objects.create(
 
-                             orderID=order_id, UID=uid, channel=channel, userID_id=user_id, desc=content,
 
-                             payType=pay_type,
 
-                             payTime=now_time, price=price, currency=currency, addTime=now_time, updTime=now_time,
 
-                             order_type=order_type, commodity_code=commodity_code, commodity_type=commodity_type,
 
-                             rank_id=rank_id,
 
-                             ai_rank_id=1, status=1, create_vod=1, store_meal_name=store_meal_name,
 
-                             uid_bucket_id=uid_bucket_id, transaction_id=transaction_id,
 
-                             original_transaction_id=original_transaction_id
 
-                         )
 
-                         # 发送云存开通信息
 
-                         date_time = time.strftime("%Y-%m-%d", time.localtime())
 
-                         # 如果存在序列号,消息提示用序列号
 
-                         device_info_qs = Device_Info.objects.filter(UID=uid).values('serial_number', 'Type')
 
-                         serial_number = device_info_qs[0]['serial_number']
 
-                         device_type = device_info_qs[0]['Type']
 
-                         if serial_number:
 
-                             device_name = CommonService.get_full_serial_number(uid, serial_number, device_type)
 
-                         else:
 
-                             device_name = uid
 
-                         sys_msg_text_list = [
 
-                             '温馨提示:尊敬的客户,您的{}设备在{}已成功续订云存套餐'.format(device_name, date_time),
 
-                             'Dear customer,you already subscribed the cloud storage package successfully for device {} on '.
 
-                             format(device_name, time.strftime('%b %dth,%Y', time.localtime()))]
 
-                         cls.do_vod_msg_notice(uid, user_id, lang, sys_msg_text_list)
 
-             elif str(decoded_payload.rawNotificationType) == "EXPIRED":
 
-                 # 一种通知类型,与其子类型一起表示订阅已过期。如果subtype为
 
-                 # VOLUNTARY(自愿),则表示订阅在用户禁用订阅续订后过期。如果subtype是
 
-                 # BILLING_RETRY(计费重试),则表示订阅过期,因为计费重试期结束时没有成功的计费交易。如果subtype为
 
-                 # PRICE_INCREASE,则表示订阅已过期,因为客户不同意需要客户同意的价格上涨。如果subtype为
 
-                 # PRODUCT_NOT_FOR_SALE,则表示订阅已过期,因为在订阅尝试续订时,产品已不可购买。
 
-                 # 没有子类型的通知表示订阅因其他原因过期。
 
-                 decoded_transaction_information = verifier.verify_and_decode_signed_transaction(
 
-                     decoded_payload.data.signedTransactionInfo)
 
-                 # originalTransactionId 原始购买的交易标识符
 
-                 original_transaction_id = decoded_transaction_information.originalTransactionId
 
-                 if original_transaction_id:
 
-                     DeviceApplePackage.objects.filter(original_transaction_id=original_transaction_id).update(
 
-                         subscription_status=2, update_time=int(time.time()))
 
-             elif str(decoded_payload.rawNotificationType) == "CONSUMPTION_REQUEST":
 
-                 # 一种通知类型,指示客户发起了消费型 App 内购买项目或自动续期订阅的退款请求,并且 App Store 要求您提供消费数据。有关详细信息,请参阅发送消耗信息。
 
-                 pass
 
-             elif str(decoded_payload.rawNotificationType) == "DID_CHANGE_RENEWAL_STATUS":
 
-                 if decoded_payload.rawSubtype == "AUTO_RENEW_DISABLED":
 
-                     # 自动续订被禁用
 
-                     decoded_transaction_information = verifier.verify_and_decode_signed_transaction(
 
-                         decoded_payload.data.signedTransactionInfo)
 
-                     original_transaction_id = decoded_transaction_information.originalTransactionId
 
-                     if original_transaction_id:
 
-                         DeviceApplePackage.objects.filter(original_transaction_id=original_transaction_id).update(
 
-                             subscription_status=3, update_time=int(time.time()))
 
-                 elif decoded_payload.rawSubtype == "AUTO_RENEW_ENABLED":
 
-                     # 自动续订被开启
 
-                     decoded_transaction_information = verifier.verify_and_decode_signed_transaction(
 
-                         decoded_payload.data.signedTransactionInfo)
 
-                     original_transaction_id = decoded_transaction_information.originalTransactionId
 
-                     if original_transaction_id:
 
-                         DeviceApplePackage.objects.filter(original_transaction_id=original_transaction_id).update(
 
-                             subscription_status=1, update_time=int(time.time()))
 
-             elif str(decoded_payload.rawNotificationType) == "REFUND":
 
-                 # 一种通知类型,表示 App Store 成功退还了消耗性应用内购买、非消耗性应用内购买、自动续订或不可续订的交易。
 
-                 # revocationDate 包含退款交易的时间戳。originalTransactionId 和 productId 用于标识原始交易和产品。revocationReason 包含原因。
 
-                 # 要请求客户所有退款交易的列表,请参阅 App Store 服务器 API 中的获取退款历史记录。
 
-                 # 1. 找套餐 使用 transaction_id 找orders
 
-                 decoded_transaction_information = verifier.verify_and_decode_signed_transaction(
 
-                     decoded_payload.data.signedTransactionInfo)
 
-                 transaction_id = decoded_transaction_information.transactionId
 
-                 logger.info('App Store服务器通知退款, transaction_id:{}'.format(transaction_id))
 
-                 orders_qs = Order_Model.objects.filter(transaction_id=transaction_id)
 
-                 # 2. 查找云存套餐使用表 和 云存套餐
 
-                 if orders_qs.exists():
 
-                     orders_qs.update(status=11)
 
-                     orderID = orders_qs[0].orderID
 
-                     uid = orders_qs[0].UID
 
-                     user_id = orders_qs[0].userID
 
-                     # 3. 未使用则删除未使用套餐表,已使用过则删除设备正在使用套餐,并关闭设备云存
 
-                     uid_bucket_qs = UID_Bucket.objects.filter(uid=uid, orderId=orderID, use_status=1,
 
-                                                               endTime__gt=int(time.time()))
 
-                     unused_uid_meal_qs = Unused_Uid_Meal.objects.filter(order_id=orderID)
 
-                     ai_service_qs = AiService.objects.filter(uid=uid, orderId=orderID, use_status=1,
 
-                                                              endTime__gt=int(time.time()))
 
-                     if unused_uid_meal_qs.exists():
 
-                         unused_uid_meal_qs.delete()
 
-                     if uid_bucket_qs.exists():
 
-                         uid_bucket_qs.update(status=0, use_status=2, endTime=int(time.time()),
 
-                                              updateTime=int(time.time()))
 
-                     if ai_service_qs.exists():
 
-                         ai_service_qs.update(detect_status=0, use_status=2, endTime=int(time.time()),
 
-                                              updTime=int(time.time()))
 
-                         # 关闭ai
 
-                         msg = {'commandType': 'AIDisable'}
 
-                         thing_name = CommonService.query_serial_with_uid(uid)  # 存在序列号则为使用序列号作为物品名
 
-                         topic_name = 'ansjer/generic/{}'.format(thing_name)
 
-                         req_success = CommonService.req_publish_mqtt_msg(thing_name, topic_name, msg)
 
-                         LOGGER.info(f'App Store服务器通知用户退款, 关闭AI:{req_success}')
 
-                     # 4.发送邮件告知用户退款
 
-                     email_content = f'用户{user_id}, 订单:{orderID}, 设备{uid}退款'
 
-                 else:
 
-                     email_content = f'transaction_id:{transaction_id}退款, 未查询到订单'
 
-                 S3Email().faEmail(email_content, 'servers@ansjer.com')
 
-             elif str(decoded_payload.rawNotificationType) == "REFUND_DECLINED":
 
-                 # 一种通知类型,表示 App Store 由于客户提出的争议而撤销了先前批准的退款。如果您的应用程序因相关退款而撤销了内容或服务,则需要恢复这些内容或服务。
 
-                 # 此通知类型可适用于任何应用程序内购买类型:消耗品、非消耗品、不可续订订阅和自动续订订阅。对于自动续订,当 App Store 撤销退款时,续订日期保持不变。
 
-                 pass
 
-             else:
 
-                 logger.info(f"App Store服务器通知decoded_payload.rawNotificationType 未处理")
 
-                 return HttpResponse(status=500)
 
-             return HttpResponse(status=200)
 
-         except Exception as e:
 
-             logger.info('App Store服务器通知异常:{}'.
 
-                         format('error_line:{}, error_msg:{}'.format(e.__traceback__.tb_lineno, repr(e))))
 
-             return HttpResponse(status=500)
 
 
  |