AppleInAppPurchaseSubscriptionObject.py 5.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151
  1. # @Author : linhaohong
  2. # @File : AppleInAppPurchaseSubscriptionObject.py
  3. # @Time : 2024/9/4 11:58
  4. from appstoreserverlibrary.receipt_utility import ReceiptUtility
  5. from Ansjer.config import LOGGER, CONFIG_INFO, CONFIG_TEST, BASE_DIR, IN_APP_CONFIG
  6. from appstoreserverlibrary.api_client import AppStoreServerAPIClient
  7. from appstoreserverlibrary.models.Environment import Environment
  8. from appstoreserverlibrary.signed_data_verifier import SignedDataVerifier
  9. from Model.models import Order_Model, DeviceApplePackage
  10. ENV = Environment.SANDBOX if CONFIG_INFO == CONFIG_TEST else Environment.PRODUCTION
  11. class InAppConfig:
  12. def __init__(self, bundle_id: str, user_id=""):
  13. """
  14. 初始化 AppConfig,加载与指定 bundle_id 对应的配置。
  15. :param bundle_id: 应用的 bundle ID,用于检索相关配置。
  16. """
  17. config = IN_APP_CONFIG.get(bundle_id)
  18. if not config:
  19. raise ValueError(f"没有找到与 bundle_id '{bundle_id}' 对应的配置")
  20. self.key_id = config['key_id']
  21. self.issuer_id = config['issuer_id']
  22. self.key_filename = config['key_filename']
  23. self.cert_names = config['cert_names']
  24. self.app_apple_id = config.get("app_apple_id")
  25. if user_id == "167083887153913800138000":
  26. self.environment = Environment.SANDBOX
  27. else:
  28. self.environment = ENV
  29. class InAppPurchase:
  30. def __init__(self, bundle_id="com.ansjer.zccloud", user_id=""):
  31. """
  32. 初始化 InAppSubscription,加载私钥并初始化客户端和解码器。
  33. :param app_config: 包含内购相关配置的 AppConfig 实例。
  34. """
  35. app_config = InAppConfig(bundle_id, user_id)
  36. self.bundle_id = bundle_id
  37. self.config = app_config
  38. self.private_key = self._load_private_key(self.config.key_filename)
  39. self.client = self._initialize_client()
  40. self.verifier = self._initialize_verifier()
  41. self.receipt_util = self._initialize_receipt_util()
  42. def _load_private_key(self, key_filename: str) -> bytes:
  43. """
  44. 加载私钥文件并返回其内容。
  45. :param key_filename: 私钥文件的名称。
  46. :return: 返回私钥文件的内容,类型为 bytes。
  47. """
  48. key_path = f'{BASE_DIR}/Ansjer/file/in_app_purchase/{key_filename}'
  49. try:
  50. with open(key_path, 'rb') as file:
  51. return file.read()
  52. except FileNotFoundError:
  53. raise ValueError(f"Private key file not found: {key_path}")
  54. def _initialize_client(self) -> AppStoreServerAPIClient:
  55. """
  56. 初始化 AppStoreServerAPIClient 实例,用于与 App Store 进行通信。
  57. :return: 返回初始化后的 AppStoreServerAPIClient 实例。
  58. """
  59. return AppStoreServerAPIClient(
  60. self.private_key,
  61. self.config.key_id,
  62. self.config.issuer_id,
  63. self.bundle_id,
  64. self.config.environment
  65. )
  66. def _initialize_verifier(self) -> SignedDataVerifier:
  67. """
  68. 初始化 SignedDataVerifier 实例,用于内购验证签名数据。
  69. :return: 返回初始化后的 SignedDataVerifier 实例。
  70. """
  71. root_certificates = [
  72. self._load_certificate(cert_name)
  73. for cert_name in self.config.cert_names
  74. ]
  75. enable_online_checks = True # 根据需要设定在线检查是否启用
  76. app_apple_id = None if CONFIG_INFO == CONFIG_TEST else self.config.app_apple_id
  77. return SignedDataVerifier(
  78. root_certificates,
  79. enable_online_checks,
  80. self.config.environment,
  81. self.bundle_id,
  82. app_apple_id
  83. )
  84. def _load_certificate(self, cert_name: str) -> bytes:
  85. """
  86. 加载指定的根证书文件并返回其内容。
  87. :param cert_name: 根证书文件的名称。
  88. :return: 返回根证书文件的内容,类型为 bytes。
  89. """
  90. cert_path = f'{BASE_DIR}/Ansjer/file/in_app_purchase/{cert_name}'
  91. try:
  92. with open(cert_path, 'rb') as file:
  93. return file.read()
  94. except FileNotFoundError:
  95. raise ValueError(f"Certificate file not found: {cert_path}")
  96. def _initialize_receipt_util(self):
  97. receipt_util = ReceiptUtility()
  98. return receipt_util
  99. def check_subscriptions(self, uid: str, subscription_group_id: str) -> bool:
  100. try:
  101. has_order = self._get_last_order(uid)
  102. if not has_order:
  103. return False
  104. subscription_status = self._get_subscription_status(has_order, subscription_group_id)
  105. return subscription_status
  106. except Exception as e:
  107. LOGGER.error(f"Error checking subscriptions: {str(e)}")
  108. return False
  109. def _get_last_order(self, uid: str) -> dict:
  110. return Order_Model.objects.filter(UID=uid, payType=5).exclude(
  111. original_transaction_id=""
  112. ).values(
  113. 'transaction_id', 'original_transaction_id', 'orderID', 'addTime'
  114. ).order_by('-addTime').first()
  115. def _get_subscription_status(self, order: dict, subscription_group_id: str) -> bool:
  116. device_apple_package = DeviceApplePackage.objects.filter(
  117. original_transaction_id=order['original_transaction_id']
  118. ).values("subscription_status").first()
  119. if not device_apple_package or device_apple_package['subscription_status'] != 1:
  120. return False
  121. subscription_statuses = self.client.get_all_subscription_statuses(order['transaction_id'])
  122. if not subscription_statuses.data:
  123. return False
  124. for subscription_status in subscription_statuses.data:
  125. if str(subscription_status.subscriptionGroupIdentifier) == subscription_group_id:
  126. if any(str(item.status) == "Status.ACTIVE" for item in subscription_status.lastTransactions):
  127. return True
  128. return False