123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940 |
- # @Author : Rocky
- # @File : InAppPurchaseController.py
- # @Time : 2024/6/21 9:10
- import logging
- import time
- import json
- import threading
- import requests
- from appstoreserverlibrary.models.Environment import Environment
- from appstoreserverlibrary.api_client import AppStoreServerAPIClient, GetTransactionHistoryVersion
- from appstoreserverlibrary.models.AccountTenure import AccountTenure
- from appstoreserverlibrary.models.ConsumptionRequest import ConsumptionRequest
- from appstoreserverlibrary.models.ConsumptionStatus import ConsumptionStatus
- from appstoreserverlibrary.models.DeliveryStatus import DeliveryStatus
- from appstoreserverlibrary.models.Environment import Environment
- from appstoreserverlibrary.models.LifetimeDollarsPurchased import LifetimeDollarsPurchased
- from appstoreserverlibrary.models.LifetimeDollarsRefunded import LifetimeDollarsRefunded
- from appstoreserverlibrary.models.Platform import Platform
- from appstoreserverlibrary.models.PlayTime import PlayTime
- from appstoreserverlibrary.models.RefundPreference import RefundPreference
- from appstoreserverlibrary.models.UserStatus import UserStatus
- from appstoreserverlibrary.receipt_utility import ReceiptUtility
- 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, CONFIG_US
- 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, InAppRefund, OrderPayLog, CouponModel
- from Object.AWS.S3Email import S3Email
- from Object.AliSmsObject import AliSmsObject
- from Object.AppleInAppPurchaseSubscriptionObject import InAppPurchase
- from Object.Enums.RedisKeyConstant import RedisKeyConstant
- from Object.RedisObject import RedisObject
- from Service.CommonService import CommonService
- ENV = Environment.SANDBOX if CONFIG_INFO == CONFIG_TEST else Environment.PRODUCTION
- logger = logging.getLogger('apple_pay')
- 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)
- elif operation == 'AppStoreServerNotificationsVsees': # App Store服务器通知(用于转发通知)
- return self.app_store_server_notifications_vsees(request, request_dict)
- elif operation == 'vseesNotifications':
- return self.vsees_notifications(request)
- elif operation == 'serverNotifications': # App Store服务器通知
- return self.server_notifications(request)
- elif operation == 'putRefundOrder': # App Store服务器通知
- return self.put_refund_order()
- 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)
- app_type = request_dict.get('app_type', 1)
- logger.info(
- f"苹果内购认证交易订单orderID:{order_id},"
- f"transaction_id: {transaction_identifier},"
- f"original_transaction_id: {original_transaction_identifier},"
- f"receipt: {receipt}, app_type: {app_type}"
- )
- if not all([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)
- if not is_lock:
- return response.json(5)
- redis_obj.CONN.expire(redis_key, 60)
- try:
- # 检查商品id是否正确
- app_type = int(app_type)
- if app_type == 1:
- bundle_id = "com.ansjer.zccloud"
- elif app_type == 2:
- bundle_id = "com.cloudlife.commissionf"
- else:
- return response.json(444, "app_type不存在")
- # 验证订单是否存在
- order_qs = Order_Model.objects.filter(orderID=order_id, UID=uid, app_type=app_type).values("rank_id",
- "transaction_id",
- "coupon_id")
- if not order_qs.exists():
- return response.json(173)
- if order_qs[0]["transaction_id"]:
- return response.json(0)
- if UID_Bucket.objects.filter(orderId=order_id).exists():
- return response.json(0)
- if Unused_Uid_Meal.objects.filter(order_id=order_id).exists():
- return response.json(0)
- # 实例化订阅类
- in_app_purchase = InAppPurchase(bundle_id=bundle_id)
- # ReceiptUtility 用于解析收据为transaction_id
- receipt_util = in_app_purchase.receipt_util
- # AppStoreServerAPIClient 用于查询交易信息
- client = in_app_purchase.client
- # SignedDataVerifier 用于解析查询到的交易信息
- signed_data_verifier = in_app_purchase.verifier
- # 解析收据(循环扣款时不需要这一步, 直接获取transaction_id)
- transaction_id = transaction_identifier
- if transaction_identifier == "":
- transaction_id = receipt_util.extract_transaction_id_from_app_receipt(receipt)
- if not transaction_id:
- logger.info(f"苹果内购认证交易订单orderID:{order_id}, 没有transaction_id")
- pay_result_url = CommonService.get_payment_status_url(lang, 'fail')
- return response.json(0, {'url': pay_result_url})
- logger.info(f"苹果内购认证交易订单orderID:{order_id}, transaction_id:{transaction_id}, 时间戳: {int(time.time())}")
- OrderPayLog.objects.create(order_id=order_id, order_no=transaction_id,
- business_name=f"内购验单",
- created_time=int(time.time()), updated_time=int(time.time()),
- access_result="SUCCESS")
- # 查询交易信息
- transaction_info = ""
- attempts = 0
- while attempts < 6:
- try:
- transaction_info = client.get_transaction_info(transaction_id)
- break
- except ConnectionError as err:
- attempts += 1
- if attempts == 5:
- OrderPayLog.objects.create(order_id=order_id, order_no=transaction_id,
- business_name=f"{order_id}获取transactionInfo超时",
- created_time=int(time.time()), updated_time=int(time.time()),
- access_result="ERROR")
- return response.json(5)
- logger.info(
- f"订单orderId:{order_id}, transaction_id:{transaction_id}, 第{attempts}次获取支付信息超时")
- signed_transaction_info = transaction_info.signedTransactionInfo
- # 解析交易信息
- payload = signed_data_verifier.verify_and_decode_signed_transaction(signed_transaction_info)
- # 获取交易的商品id
- product_id = payload.productId if payload and payload.productId else None
- if not product_id:
- logger.info(f"苹果内购认证交易订单orderID:{order_id}, product_id获取失败")
- pay_result_url = CommonService.get_payment_status_url(lang, 'fail')
- return response.json(0, {'url': pay_result_url})
- in_app_purchase_package_qs = InAppPurchasePackage.objects.filter(product_id=product_id)
- if not in_app_purchase_package_qs.exists():
- logger.info(f"苹果内购认证交易订单orderID:{order_id}, InAppPurchasePackage表未查询到product_id")
- return response.json(173, "内购商品id不存在")
- # 验证套餐是否存在
- 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, "云存套餐不存在")
- # 验证内购套餐是否存在
- in_app_purchase_package_qs = InAppPurchasePackage.objects.filter(product_id=product_id)
- if not in_app_purchase_package_qs.exists():
- return response.json(173, "内购套餐不存在")
- # 循环扣款
- if original_transaction_identifier != "" and in_app_purchase_package_qs[0].package_type == 1:
- 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服务器通知接口订阅
- Order_Model.objects.filter(orderID=order_id).delete()
- 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服务器通知接口修改订阅状态
- Order_Model.objects.filter(orderID=order_id).delete()
- 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 == 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})
- else:
- return response.json(173, "内购套餐未分配")
- # 设备开通云存
- now_time = int(time.time())
- uid_bucket_id = cls.enable_cloud(channel, now_time, order_id, store_qs, uid)
- if order_qs[0]["coupon_id"]:
- c_id = order_qs[0]["coupon_id"]
- key_coupon = RedisKeyConstant.COUPON_ID_LOCK.value + c_id
- redis_obj.del_data(key_coupon)
- CouponModel.objects.filter(id=c_id).update(use_status=2, update_time=now_time)
- # 修改订阅状态
- if payload.rawType == "Auto-Renewable Subscription":
- original_transaction_id = payload.originalTransactionId
- 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,
- )
- order_qs.update(status=1, uid_bucket_id=uid_bucket_id,
- transaction_id=transaction_id, create_vod=1,
- payTime=now_time, updTime=now_time,
- original_transaction_id=original_transaction_id)
- else:
- order_qs.update(status=1, uid_bucket_id=uid_bucket_id,
- transaction_id=transaction_id, create_vod=1,
- payTime=now_time, updTime=now_time)
- # 构建云存套餐消息
- sys_msg_text_list = cls.cloud_storage_message(uid)
- # 发送云存套餐购买消息
- asy = threading.Thread(target=cls.do_vod_msg_notice,
- args=(uid, user_id, lang, sys_msg_text_list))
- asy.start()
- redis_obj.del_data(redis_key)
- pay_result_url = CommonService.get_payment_status_url(lang, 'success')
- OrderPayLog.objects.create(order_id=order_id, order_no=transaction_id,
- business_name=f"内购充值成功",
- created_time=int(time.time()), updated_time=int(time.time()),
- access_result="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))))
- OrderPayLog.objects.create(order_id=order_id, business_name=f"内购验单异常",
- created_time=int(time.time()), updated_time=int(time.time()), access_result="ERROR")
- pay_result_url = CommonService.get_payment_status_url(lang, 'fail')
- return response.json(0, {'url': pay_result_url})
- @classmethod
- def cloud_storage_message(cls, uid):
- # 发送云存开通信息
- 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()))]
- return sys_msg_text_list
- @classmethod
- def enable_cloud(cls, channel, now_time, order_id, store_qs, uid):
- 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)
- return uid_bucket_id
- @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')
- if request.method != 'POST':
- logger.info(f'App Store服务器通知不是post请求, 参数{request_dict}')
- return HttpResponse(status=400)
- try:
- request_data = json.loads(request.body)
- except json.JSONDecodeError:
- logger.info('无法解析请求体为JSON')
- return HttpResponse(status=400)
- request_data['bundleId'] = 'com.ansjer.zccloud'
- updated_request_body = json.dumps(request_data)
- if CONFIG_INFO == CONFIG_TEST:
- logger.info('测试环境, App Store服务器通知发送到测试服')
- response_test = requests.post(url="https://test.zositechc.cn/inAppPurchase/serverNotifications",
- json=updated_request_body)
- return HttpResponse(status=response_test.status_code)
- response_us = requests.post(url="https://www.dvema.com/inAppPurchase/serverNotifications",
- json=updated_request_body)
- status_code = response_us.status_code
- if status_code != 200:
- response_eu = requests.post(url="https://api.zositeche.com/inAppPurchase/serverNotifications",
- json=updated_request_body)
- status_code = response_eu.status_code
- if status_code == 200:
- return HttpResponse(status=200)
- return HttpResponse(status=400)
- @classmethod
- def server_notifications(cls, request):
- try:
- logger.info('App Store服务器通知请求类型:{}'.format(request.method))
- logger.info('App Store服务器通知参数:{}'.format(request.POST))
- logger.info('App Store服务器通知请求body:{}'.format(request.body))
- payload = json.loads(json.loads(request.body))
- logger.info('App Store服务器通知payload:{}'.format(payload))
- # 获取 signedPayload
- signed_payload = payload.get('signedPayload')
- bundle_id = payload.get('bundleId')
- if not signed_payload:
- return HttpResponse(status=400)
- if bundle_id == "com.ansjer.zccloud":
- app_type = 1
- else:
- app_type = 2
- in_app_purchase_obj = InAppPurchase(bundle_id=bundle_id)
- # AppStoreServerAPIClient 用于查询交易信息
- client = in_app_purchase_obj.client
- # SignedDataVerifier 用于解析查询到的交易信息
- signed_data_verifier = in_app_purchase_obj.verifier
- # 验证签名并解码 payload
- decoded_payload = signed_data_verifier.verify_and_decode_notification(signed_payload)
- logger.info(f"App Store服务器通知解码后decoded_payload:{decoded_payload}")
- raw_notification_type = str(decoded_payload.rawNotificationType)
- raw_subtype = str(decoded_payload.rawSubtype)
- logger.info(f"App Store服务器通知, 大类型{raw_notification_type}, 小类型{raw_subtype}")
- if str(decoded_payload.rawNotificationType) == "DID_RENEW":
- # 续订
- decoded_transaction_information = signed_data_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)
- 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服务器通知, 未查询到旧订单信息, originalTransactionId:{original_transaction_id}, 返回状态 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)
- # 解决云存充值成功, 由于一些原因返回500 导致苹果未扣款的问题
- if Order_Model.objects.filter(transaction_id=transaction_id, status=1).exists():
- logger.info(f"App Store服务器通知云存续订订单已存在, transactionId:{transaction_id} 返回状态 200")
- return HttpResponse(status=200)
- order_id = CommonService.createOrderID()
- rank_id = store_qs[0]['id']
- currency = store_qs[0]['currency']
- price = store_qs[0]['price']
- is_ai = store_qs[0]['is_ai']
- 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
- now_time = int(time.time())
- 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 = '未知套餐'
- # 创建订单
- order = 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,
- store_meal_name=store_meal_name, app_type=app_type
- )
- # 充值云存套餐
- uid_bucket_id = cls.enable_cloud(channel, now_time, order_id, store_qs, uid)
- # 修改订单信息
- order.uid_bucket_id = uid_bucket_id
- order.transaction_id = transaction_id
- order.original_transaction_id = original_transaction_id
- order.save()
- # 构建云存套餐消息
- sys_msg_text_list = cls.cloud_storage_message(uid)
- cls.do_vod_msg_notice(uid, user_id, lang, sys_msg_text_list)
- return HttpResponse(status=200)
- elif str(decoded_payload.rawNotificationType) == "SUBSCRIBED":
- # 处理订阅 ---> 首次充值逻辑写在了认证交易
- if decoded_payload.rawSubtype == "RESUBSCRIBE":
- decoded_transaction_information = signed_data_verifier.verify_and_decode_signed_transaction(
- decoded_payload.data.signedTransactionInfo)
- # originalTransactionId 原始购买的交易标识符
- original_transaction_id = decoded_transaction_information.originalTransactionId
- transaction_id = decoded_transaction_information.transactionId
- app_account_token = decoded_transaction_information.appAccountToken
- app_account_token = json.loads(app_account_token)
- 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)
- # 查旧订单消息
- ord_order_qs = Order_Model.objects.filter(original_transaction_id=original_transaction_id)
- if not ord_order_qs.exists():
- logger.info(f"App Store服务器通知未查询到旧订单信息, originalTransactionId:{original_transaction_id}, 返回状态 400")
- return HttpResponse(status=400)
- # 解决云存充值成功, 由于一些原因返回500 导致苹果未扣款的问题
- if Order_Model.objects.filter(transaction_id=transaction_id, status=1).exists():
- logger.info(
- f"App Store服务器通知云存续订订单已存在, transactionId:{transaction_id} 返回状态 200")
- return HttpResponse(status=200)
- ord_order = ord_order_qs.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"]
- new_user_id = app_account_token["user_id"]
- if new_user_id != user_id:
- uid = app_account_token["UID"]
- user_id = new_user_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']
- currency = store_qs[0]['currency']
- price = store_qs[0]['price']
- is_ai = store_qs[0]['is_ai']
- 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
- now_time = int(time.time())
- 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 = '未知套餐'
- order = 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,
- store_meal_name=store_meal_name, app_type=app_type
- )
- # 充值云存套餐
- uid_bucket_id = cls.enable_cloud(channel, now_time, order_id, store_qs, uid)
- # 修改订单信息
- order.uid_bucket_id = uid_bucket_id
- order.transaction_id = transaction_id
- order.original_transaction_id = original_transaction_id
- order.save()
- DeviceApplePackage.objects.filter(userID=user_id, uid=uid).update(subscription_status=1,
- update_time=int(time.time()))
- # 构建云存套餐消息
- sys_msg_text_list = cls.cloud_storage_message(uid)
- cls.do_vod_msg_notice(uid, user_id, lang, sys_msg_text_list)
- return HttpResponse(status=200)
- elif str(decoded_payload.rawNotificationType) == "EXPIRED":
- # 一种通知类型,与其子类型一起表示订阅已过期。如果subtype为
- # VOLUNTARY(自愿),则表示订阅在用户禁用订阅续订后过期。如果subtype是
- # BILLING_RETRY(计费重试),则表示订阅过期,因为计费重试期结束时没有成功的计费交易。如果subtype为
- # PRICE_INCREASE,则表示订阅已过期,因为客户不同意需要客户同意的价格上涨。如果subtype为
- # PRODUCT_NOT_FOR_SALE,则表示订阅已过期,因为在订阅尝试续订时,产品已不可购买。
- # 没有子类型的通知表示订阅因其他原因过期。
- decoded_transaction_information = signed_data_verifier.verify_and_decode_signed_transaction(
- decoded_payload.data.signedTransactionInfo)
- # originalTransactionId 原始购买的交易标识符
- original_transaction_id = decoded_transaction_information.originalTransactionId
- if not original_transaction_id:
- logger.info(f"App Store服务器通知 originalTransactionId原始购买的交易标识符为空, 返回状态 400")
- return HttpResponse(status=400)
- device_apple_package_qs = DeviceApplePackage.objects.filter(
- original_transaction_id=original_transaction_id)
- if not device_apple_package_qs.exists():
- return HttpResponse(status=400)
- device_apple_package_qs.update(subscription_status=2, update_time=int(time.time()))
- elif str(decoded_payload.rawNotificationType) == "CONSUMPTION_REQUEST":
- # 一种通知类型,指示客户发起了消费型 App 内购买项目或自动续期订阅的退款请求,并且 App Store 要求您提供消费数据。有关详细信息,请参阅发送消耗信息。
- decoded_transaction_information = signed_data_verifier.verify_and_decode_signed_transaction(
- decoded_payload.data.signedTransactionInfo)
- transaction_id = decoded_transaction_information.transactionId
- app_account_token = decoded_transaction_information.appAccountToken
- if not app_account_token:
- app_account_token = ""
- orders_qs = Order_Model.objects.filter(transaction_id=transaction_id)
- if not orders_qs.exists():
- return HttpResponse(status=400)
- orderID = orders_qs[0].orderID
- uid = orders_qs[0].UID
- now_time = int(time.time())
- put_time = now_time + 11.5 * 60 * 60
- in_app_refund_qs = InAppRefund.objects.filter(transaction_id=transaction_id)
- if in_app_refund_qs.exists():
- in_app_refund_qs.update(refund_progress=0, updated_time=now_time,
- put_time=put_time, app_account_token=app_account_token)
- InAppRefund.objects.create(transaction_id=transaction_id, orderID=orderID,
- uid=uid, app_type=app_type, created_time=now_time,
- updated_time=now_time, put_time=put_time,
- app_account_token=app_account_token)
- return HttpResponse(status=200)
- elif str(decoded_payload.rawNotificationType) == "DID_CHANGE_RENEWAL_STATUS":
- decoded_transaction_information = signed_data_verifier.verify_and_decode_signed_transaction(
- decoded_payload.data.signedTransactionInfo)
- original_transaction_id = decoded_transaction_information.originalTransactionId
- if not original_transaction_id:
- logger.info(f"App Store服务器通知 originalTransactionId原始购买的交易标识符为空, 返回状态 400")
- return HttpResponse(status=400)
- if decoded_payload.rawSubtype == "AUTO_RENEW_ENABLED":
- # 自动续订被开启
- device_apple_package_qs = DeviceApplePackage.objects.filter(
- original_transaction_id=original_transaction_id)
- if not device_apple_package_qs.exists():
- return HttpResponse(status=400)
- device_apple_package_qs.update(subscription_status=1, update_time=int(time.time()))
- else:
- # 自动续订被禁用
- device_apple_package_qs = DeviceApplePackage.objects.filter(
- original_transaction_id=original_transaction_id)
- if not device_apple_package_qs.exists():
- return HttpResponse(status=400)
- device_apple_package_qs.update(subscription_status=3, update_time=int(time.time()))
- elif str(decoded_payload.rawNotificationType) == "REFUND":
- # 一种通知类型,表示 App Store 成功退还了消耗性应用内购买、非消耗性应用内购买、自动续订或不可续订的交易。
- # revocationDate 包含退款交易的时间戳。originalTransactionId 和 productId 用于标识原始交易和产品。revocationReason 包含原因。
- # 要请求客户所有退款交易的列表,请参阅 App Store 服务器 API 中的获取退款历史记录。
- decoded_transaction_information = signed_data_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)
- if not orders_qs.exists():
- return HttpResponse(status=400)
- orders_qs.update(status=5, updTime=int(time.time()))
- orderID = orders_qs[0].orderID
- uid = orders_qs[0].UID
- 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, orders=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}')
- InAppRefund.objects.filter(transaction_id=transaction_id).update(updated_time=int(time.time()),
- refund_progress=2)
- return HttpResponse(status=200)
- elif str(decoded_payload.rawNotificationType) == "REFUND_DECLINED":
- # 一种通知类型,表示 App Store 由于客户提出的争议而撤销了先前批准的退款。如果您的应用程序因相关退款而撤销了内容或服务,则需要恢复这些内容或服务。
- # 此通知类型可适用于任何应用程序内购买类型:消耗品、非消耗品、不可续订订阅和自动续订订阅。对于自动续订,当 App Store 撤销退款时,续订日期保持不变。
- decoded_transaction_information = signed_data_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))
- elif str(decoded_payload.rawNotificationType) == "REFUND_DECLINED":
- decoded_transaction_information = signed_data_verifier.verify_and_decode_signed_transaction(
- decoded_payload.data.signedTransactionInfo)
- transaction_id = decoded_transaction_information.transactionId
- orders_qs = Order_Model.objects.filter(transaction_id=transaction_id)
- if not orders_qs.exists():
- return HttpResponse(status=400)
- InAppRefund.objects.filter(transaction_id=transaction_id).update(refund_progress=3)
- else:
- logger.info(f"App Store服务器通知decoded_payload.rawNotificationType 未处理")
- 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)
- @classmethod
- def app_store_server_notifications_vsees(cls, request, request_dict):
- logger = logging.getLogger('apple_pay')
- if request.method != 'POST':
- logger.info(f'App Store服务器通知不是post请求, 参数{request_dict}')
- return HttpResponse(status=400)
- try:
- request_data = json.loads(request.body)
- except json.JSONDecodeError:
- logger.error('无法解析请求体为JSON')
- return HttpResponse(status=400)
- request_data['bundleId'] = 'com.ansjer.zccloud'
- updated_request_body = json.dumps(request_data)
- if CONFIG_INFO == CONFIG_TEST:
- logger.info('测试环境, App Store服务器通知发送到测试服')
- response_test = requests.post(url="https://test.zositechc.cn/inAppPurchase/serverNotifications",
- json=updated_request_body)
- return HttpResponse(status=response_test.status_code)
- response_us = requests.post(url="https://www.dvema.com/inAppPurchase/serverNotifications",
- json=updated_request_body)
- status_code = response_us.status_code
- if status_code != 200:
- response_eu = requests.post(url="https://api.zositeche.com/inAppPurchase/serverNotifications",
- json=updated_request_body)
- status_code = response_eu.status_code
- if status_code == 200:
- return HttpResponse(status=200)
- return HttpResponse(status=400)
- @classmethod
- def vsees_notifications(cls, request):
- logger = logging.getLogger('apple_pay')
- logger.info('Vsees: App Store服务器通知请求类型:{}'.format(request.method))
- logger.info('Vsees: App Store服务器通知参数:{}'.format(request.POST))
- logger.info('Vsees: App Store服务器通知请求body:{}'.format(request.body))
- payload = json.loads(request.body.decode('utf-8'))
- logger.info('Vsees: App Store服务器通知payload:{}'.format(payload))
- # 获取 signedPayload
- signed_payload = payload.get('signedPayload')
- if not signed_payload:
- return HttpResponse(status=400)
- in_app_purchase = InAppPurchase(bundle_id="com.cloudlife.commissionf")
- # SignedDataVerifier 用于解析查询到的交易信息
- verifier = in_app_purchase.verifier
- decoded_payload = verifier.verify_and_decode_notification(signed_payload)
- logger.info('Vsees: App Store服务器通知decoded_payload: {}'.format(decoded_payload))
- status_code = 200
- if 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('Vsees: 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'{CONFIG_INFO}用户{user_id}, 订单:{orderID}, 设备{uid}退款'
- S3Email().faEmail(email_content, 'servers@ansjer.com')
- else:
- if CONFIG_INFO == CONFIG_US:
- url = "https://api.zositeche.com/inAppPurchase/AppStoreServerNotifications"
- eur_response = requests.post(url=url, json=json.loads(request.body))
- status_code = eur_response.status_code
- return HttpResponse(status=status_code)
- @staticmethod
- def put_refund_order():
- put_time = int(time.time())
- in_app_refund_qs = InAppRefund.objects.filter(refund_progress=0, put_time__lt=put_time)
- for in_app_refund in in_app_refund_qs:
- transaction_id = in_app_refund.transaction_id
- app_type = in_app_refund.app_type
- if app_type == 1:
- bundle_id = "com.ansjer.zccloud"
- elif app_type == 2:
- bundle_id = "com.cloudlife.commissionf"
- else:
- return HttpResponse(status=200)
- in_app_purchase_obj = InAppPurchase(bundle_id=bundle_id)
- # AppStoreServerAPIClient 用于查询交易信息
- client = in_app_purchase_obj.client
- orderID = in_app_refund.orderID
- app_account_token = in_app_refund.app_account_token
- unused_uid_meal_qs = Unused_Uid_Meal.objects.filter(order_id=orderID)
- uid_bucket_qs = UID_Bucket.objects.filter(orderId=orderID, endTime__gt=int(time.time()))
- if unused_uid_meal_qs.exists():
- consumptionStatus = ConsumptionStatus.NOT_CONSUMED
- deliveryStatus = DeliveryStatus.DELIVERED_AND_WORKING_PROPERLY
- elif uid_bucket_qs.exists():
- consumptionStatus = ConsumptionStatus.PARTIALLY_CONSUMED
- deliveryStatus = DeliveryStatus.DELIVERED_AND_WORKING_PROPERLY
- elif UID_Bucket.objects.filter(orderId=orderID, endTime__lt=int(time.time())):
- consumptionStatus = ConsumptionStatus.FULLY_CONSUMED
- deliveryStatus = DeliveryStatus.DELIVERED_AND_WORKING_PROPERLY
- else:
- consumptionStatus = ConsumptionStatus.UNDECLARED
- deliveryStatus = DeliveryStatus.DID_NOT_DELIVER_FOR_OTHER_REASON
- if in_app_refund.refund_preference == 1:
- refundPreference = RefundPreference.PREFER_GRANT
- else:
- refundPreference = RefundPreference.PREFER_DECLINE
- consumption_request = ConsumptionRequest(
- customerConsented=True,
- consumptionStatus=consumptionStatus,
- platform=Platform.UNDECLARED,
- sampleContentProvided=True,
- deliveryStatus=deliveryStatus,
- appAccountToken=app_account_token,
- accountTenure=AccountTenure.UNDECLARED,
- playTime=PlayTime.UNDECLARED,
- lifetimeDollarsRefunded=LifetimeDollarsRefunded.UNDECLARED,
- lifetimeDollarsPurchased=LifetimeDollarsPurchased.UNDECLARED,
- userStatus=UserStatus.ACTIVE,
- refundPreference=refundPreference,
- )
- client.send_consumption_data(transaction_id, consumption_request)
- logger.info(f'内购退款消费数据提交, 订单orderID:{orderID}, transaction_id:{transaction_id}')
- in_app_refund.refund_progress = 1
- in_app_refund.updated_time = int(time.time())
- in_app_refund.put_time = int(time.time())
- in_app_refund.save()
- return HttpResponse(status=200)
|