کدهای پاسخ BillingResult را مدیریت کنید

وقتی تماس کتابخانه صورت‌حساب Play اقدامی را راه‌اندازی می‌کند، کتابخانه پاسخ BillingResult را برای اطلاع توسعه‌دهندگان از نتیجه برمی‌گرداند. به عنوان مثال، اگر از queryProductDetailsAsync برای دریافت پیشنهادات موجود برای کاربر استفاده می‌کنید، کد پاسخ یا حاوی یک کد OK است و شیء ProductDetails مناسب را ارائه می‌کند، یا حاوی پاسخ متفاوتی است که دلیل عدم ارائه شی ProductDetails را نشان می‌دهد. .

همه کدهای پاسخ خطا نیستند. صفحه مرجع BillingResponseCode شرح مفصلی از هر یک از پاسخ های مورد بحث در این راهنما ارائه می دهد. چند نمونه از کدهای پاسخ که خطا را نشان نمی دهند عبارتند از:

  • BillingClient.BillingResponseCode.OK : اقدامی که توسط تماس ایجاد شد با موفقیت انجام شد.
  • BillingClient.BillingResponseCode.USER_CANCELED : برای اقداماتی که جریان های رابط کاربری Play Store را به کاربر نمایش می دهند، این پاسخ نشان می دهد که کاربر بدون تکمیل فرآیند از آن جریان های رابط کاربری دور شده است.

وقتی کد پاسخ یک خطا را نشان می دهد، گاهی اوقات علت آن به دلیل شرایط گذرا است و بنابراین بازیابی امکان پذیر است. وقتی تماس با روش کتابخانه صورت‌حساب Play یک مقدار BillingResponseCode را برمی‌گرداند که نشان‌دهنده وضعیت قابل بازیابی است، باید تماس را دوباره امتحان کنید. در موارد دیگر، شرایط گذرا در نظر گرفته نمی شوند و بنابراین سعی مجدد توصیه نمی شود.

خطاهای گذرا بسته به عواملی مانند این که آیا خطا زمانی که کاربران در جلسه هستند رخ می دهد یا خیر - برای مثال زمانی که کاربر در حال گذراندن یک جریان خرید است - یا خطا در پس زمینه رخ می دهد - به عنوان مثال، زمانی که شما در حال پرس و جو هستید، به استراتژی های متفاوتی نیاز دارند. خریدهای موجود کاربر در طول onResume . بخش استراتژی‌های امتحان مجدد در زیر نمونه‌هایی از این استراتژی‌های مختلف را ارائه می‌دهد و بخش پاسخ‌های BillingResult قابل بازگشت پیشنهاد می‌کند که کدام استراتژی برای هر کد پاسخ بهتر عمل می‌کند.

علاوه بر کد پاسخ، برخی از پاسخ های خطا شامل پیام هایی برای اشکال زدایی و ورود به سیستم هستند.

استراتژی ها را دوباره امتحان کنید

سعی مجدد ساده

در شرایطی که کاربر در جلسه است، بهتر است یک استراتژی ساده مجدد را اجرا کنید تا خطا تا حد امکان تجربه کاربر را مختل کند. در آن صورت، ما یک استراتژی ساده سعی مجدد با حداکثر تعداد تلاش را به عنوان شرط خروج توصیه می کنیم.

مثال زیر یک استراتژی ساده برای تلاش مجدد برای رسیدگی به یک خطا در هنگام ایجاد اتصال BillingClient را نشان می دهد:

class BillingClientWrapper(context: Context) : PurchasesUpdatedListener {
  // Initialize the BillingClient.
  private val billingClient = BillingClient.newBuilder(context)
    .setListener(this)
    .enablePendingPurchases()
    .build()

  // Establish a connection to Google Play.
  fun startBillingConnection() {
    billingClient.startConnection(object : BillingClientStateListener {
      override fun onBillingSetupFinished(billingResult: BillingResult) {
        if (billingResult.responseCode == BillingClient.BillingResponseCode.OK) {
          Log.d(TAG, "Billing response OK")
          // The BillingClient is ready. You can now query Products Purchases.
        } else {
          Log.e(TAG, billingResult.debugMessage)
          retryBillingServiceConnection()
        }
      }

      override fun onBillingServiceDisconnected() {
        Log.e(TAG, "GBPL Service disconnected")
        retryBillingServiceConnection()
      }
    })
  }

  // Billing connection retry logic. This is a simple max retry pattern
  private fun retryBillingServiceConnection() {
    val maxTries = 3
    var tries = 1
    var isConnectionEstablished = false
    do {
      try {
        billingClient.startConnection(object : BillingClientStateListener {
          override fun onBillingSetupFinished(billingResult: BillingResult) {
            if (billingResult.responseCode == BillingClient.BillingResponseCode.OK) {
              isConnectionEstablished = true
              Log.d(TAG, "Billing connection retry succeeded.")
            } else {
              Log.e(
                TAG,
                "Billing connection retry failed: ${billingResult.debugMessage}"
              )
            }
          }
        })
      } catch (e: Exception) {
        e.message?.let { Log.e(TAG, it) }
      } finally {
        tries++
      }
    } while (tries <= maxTries && !isConnectionEstablished)
  }
  ...
}

تلاش مجدد نمایی عقبگرد

توصیه می‌کنیم برای عملیات کتابخانه صورت‌حساب Play که در پس‌زمینه انجام می‌شوند و در زمانی که کاربر در جلسه است، روی تجربه کاربر تأثیری نمی‌گذارند، از عقب‌نشینی نمایی استفاده کنید.

به عنوان مثال، اجرای این مورد هنگام تأیید خریدهای جدید مناسب است زیرا این عملیات می‌تواند در پس‌زمینه اتفاق بیفتد، و اگر خطایی رخ دهد، نیازی نیست که تأیید در زمان واقعی اتفاق بیفتد.

private fun acknowledge(purchaseToken: String): BillingResult {
  val params = AcknowledgePurchaseParams.newBuilder()
    .setPurchaseToken(purchaseToken)
    .build()
  var ackResult = BillingResult()
  billingClient.acknowledgePurchase(params) { billingResult ->
    ackResult = billingResult
  }
  return ackResult
}

suspend fun acknowledgePurchase(purchaseToken: String) {

  val retryDelayMs = 2000L
  val retryFactor = 2
  val maxTries = 3

  withContext(Dispatchers.IO) {
    acknowledge(purchaseToken)
  }

  AcknowledgePurchaseResponseListener { acknowledgePurchaseResult ->
    val playBillingResponseCode =
    PlayBillingResponseCode(acknowledgePurchaseResult.responseCode)
    when (playBillingResponseCode) {
      BillingClient.BillingResponseCode.OK -> {
        Log.i(TAG, "Acknowledgement was successful")
      }
      BillingClient.BillingResponseCode.ITEM_NOT_OWNED -> {
        // This is possibly related to a stale Play cache.
        // Querying purchases again.
        Log.d(TAG, "Acknowledgement failed with ITEM_NOT_OWNED")
        billingClient.queryPurchasesAsync(
          QueryPurchasesParams.newBuilder()
            .setProductType(BillingClient.ProductType.SUBS)
            .build()
        )
        { billingResult, purchaseList ->
          when (billingResult.responseCode) {
            BillingClient.BillingResponseCode.OK -> {
              purchaseList.forEach { purchase ->
                acknowledge(purchase.purchaseToken)
              }
            }
          }
        }
      }
      in setOf(
         BillingClient.BillingResponseCode.ERROR,
         BillingClient.BillingResponseCode.SERVICE_DISCONNECTED,
         BillingClient.BillingResponseCode.SERVICE_UNAVAILABLE,
       ) -> {
        Log.d(
          TAG,
          "Acknowledgement failed, but can be retried --
          Response Code: ${acknowledgePurchaseResult.responseCode} --
          Debug Message: ${acknowledgePurchaseResult.debugMessage}"
        )
        runBlocking {
          exponentialRetry(
            maxTries = maxTries,
            initialDelay = retryDelayMs,
            retryFactor = retryFactor
          ) { acknowledge(purchaseToken) }
        }
      }
      in setOf(
         BillingClient.BillingResponseCode.BILLING_UNAVAILABLE,
         BillingClient.BillingResponseCode.DEVELOPER_ERROR,
         BillingClient.BillingResponseCode.FEATURE_NOT_SUPPORTED,
       ) -> {
        Log.e(
          TAG,
          "Acknowledgement failed and cannot be retried --
          Response Code: ${acknowledgePurchaseResult.responseCode} --
          Debug Message: ${acknowledgePurchaseResult.debugMessage}"
        )
        throw Exception("Failed to acknowledge the purchase!")
      }
    }
  }
}

private suspend fun <T> exponentialRetry(
  maxTries: Int = Int.MAX_VALUE,
  initialDelay: Long = Long.MAX_VALUE,
  retryFactor: Int = Int.MAX_VALUE,
  block: suspend () -> T
): T? {
  var currentDelay = initialDelay
  var retryAttempt = 1
  do {
    runCatching {
      delay(currentDelay)
      block()
    }
      .onSuccess {
        Log.d(TAG, "Retry succeeded")
        return@onSuccess;
      }
      .onFailure { throwable ->
        Log.e(
          TAG,
          "Retry Failed -- Cause: ${throwable.cause} -- Message: ${throwable.message}"
        )
      }
    currentDelay *= retryFactor
    retryAttempt++
  } while (retryAttempt < maxTries)

  return block() // last attempt
}

پاسخ های Retriable BillingResult

NETWORK_ERROR (کد خطا 12)

مشکل

این خطا نشان می دهد که مشکلی در اتصال شبکه بین دستگاه و سیستم های Play وجود دارد.

وضوح ممکن

برای بازیابی، بسته به اینکه کدام عمل باعث ایجاد خطا شده است، از تلاش های مجدد ساده یا عقب نشینی نمایی استفاده کنید.

SERVICE_TIMEOUT (کد خطا -3)

مشکل

این خطا نشان می‌دهد که درخواست قبل از اینکه Google Play بتواند پاسخ دهد به حداکثر زمان پایان رسیده است. به عنوان مثال، این ممکن است به دلیل تاخیر در اجرای عملکرد درخواست شده توسط تماس کتابخانه صورت‌حساب Play باشد.

وضوح ممکن

این معمولا یک مسئله گذرا است. بسته به اینکه کدام عمل خطا را برمی گرداند، درخواست را با استفاده از یک استراتژی عقب نشینی ساده یا نمایی دوباره امتحان کنید.

برخلاف SERVICE_DISCONNECTED در زیر، اتصال به سرویس صورت‌حساب Google Play قطع نمی‌شود، و فقط باید هر عملیاتی را که کتابخانه صورت‌حساب Play انجام شد، دوباره امتحان کنید.

SERVICE_DISCONNECTED (کد خطا -1)

مشکل

این خطای مهلک نشان می دهد که اتصال برنامه مشتری به سرویس فروشگاه Google Play از طریق BillingClient قطع شده است.

وضوح ممکن

برای جلوگیری از این خطا تا حد امکان، همیشه قبل از برقراری تماس با کتابخانه صورت‌حساب Play، با تماس با BillingClient.isReady() اتصال به خدمات Google Play را بررسی کنید.

برای تلاش برای بازیابی از SERVICE_DISCONNECTED ، برنامه مشتری شما باید سعی کند با استفاده از BillingClient.startConnection اتصال را دوباره برقرار کند.

دقیقاً مانند SERVICE_TIMEOUT ، بسته به اینکه کدام عمل باعث خطا شده است، از تلاش‌های مجدد ساده یا عقب‌نشینی نمایی استفاده کنید.

SERVICE_UNAVAILABLE (کد خطا 2)

نکته مهم:

از کتابخانه صورت‌حساب Google Play نسخه 6.0.0، SERVICE_UNAVAILABLE دیگر برای مشکلات شبکه بازگردانده نمی‌شود. زمانی که سرویس صورت‌حساب در دسترس نباشد و سناریوهای موردی SERVICE_TIMEOUT منسوخ شده باشد، بازگردانده می‌شود.

مشکل

این خطای گذرا نشان می‌دهد که سرویس صورت‌حساب Google Play در حال حاضر در دسترس نیست. در بیشتر موارد، این بدان معناست که مشکل اتصال شبکه در هر نقطه بین دستگاه مشتری و خدمات صورت‌حساب Google Play وجود دارد.

وضوح ممکن

این معمولا یک مسئله گذرا است. بسته به اینکه کدام عمل خطا را برمی گرداند، درخواست را با استفاده از یک استراتژی عقب نشینی ساده یا نمایی دوباره امتحان کنید.

برخلاف SERVICE_DISCONNECTED ، اتصال به سرویس صورت‌حساب Google Play قطع نمی‌شود، و باید هر عملیاتی را که در حال انجام است، دوباره امتحان کنید.

BILLING_UNAVAILABLE (کد خطا 3)

مشکل

این خطا نشان می دهد که یک خطای صورتحساب کاربر در طول فرآیند خرید رخ داده است. نمونه هایی از زمانی که ممکن است این اتفاق بیفتد عبارتند از:

  • برنامه Play Store در دستگاه کاربر قدیمی است.
  • کاربر در یک کشور پشتیبانی نشده است.
  • کاربر یک کاربر سازمانی است و سرپرست سازمانی آنها کاربران را از خرید غیرفعال کرده است.
  • Google Play قادر به کسر هزینه از روش پرداخت کاربر نیست. به عنوان مثال، کارت اعتباری کاربر ممکن است منقضی شده باشد.

وضوح ممکن

بعید است که تکرار خودکار در این مورد کمک کند. با این حال، اگر کاربر شرایطی را که باعث ایجاد مشکل شده است، برطرف کند، یک امتحان مجدد دستی می‌تواند کمک کند. به عنوان مثال، اگر کاربر نسخه Play Store خود را به نسخه پشتیبانی شده به‌روزرسانی کند، یک امتحان مجدد دستی از عملیات اولیه می‌تواند کارساز باشد.

اگر این خطا زمانی رخ دهد که کاربر در جلسه نیست، ممکن است تلاش مجدد منطقی نباشد. هنگامی که یک خطای BILLING_UNAVAILABLE را در نتیجه جریان خرید دریافت می‌کنید، به احتمال زیاد کاربر در طول فرآیند خرید بازخوردی از Google Play دریافت کرده و ممکن است از اشتباه خود آگاه باشد. در این مورد، می‌توانید یک پیام خطایی نشان دهید که مشخص می‌کند مشکلی پیش آمده است و یک دکمه «دوباره امتحان کنید» را ارائه دهید تا پس از رفع مشکل، گزینه امتحان مجدد دستی را در اختیار کاربر قرار دهید.

ERROR (کد خطا 6)

مشکل

این یک خطای مرگبار است که نشان دهنده یک مشکل داخلی در خود گوگل پلی است.

وضوح ممکن

گاهی اوقات مشکلات داخلی Google Play که منجر به ERROR می‌شوند، گذرا هستند و برای کاهش می‌توان یک تلاش مجدد با عقب‌نشینی نمایی انجام داد. هنگامی که کاربران در جلسه هستند، یک امتحان مجدد ساده ترجیح داده می شود.

ITEM_ALREADY_OWNED

مشکل

این پاسخ نشان می‌دهد که کاربر Google Play از قبل مالک اشتراک یا محصول یک‌بار خریدی است که می‌خواهد بخرد. در بیشتر موارد، این یک خطای گذرا نیست، مگر زمانی که ناشی از حافظه پنهان Google Play قدیمی باشد.

وضوح ممکن

برای جلوگیری از وقوع این خطا زمانی که علت آن مشکل حافظه پنهان نیست، زمانی که کاربر قبلاً محصولی را در اختیار دارد، آن را برای خرید پیشنهاد نکنید. اطمینان حاصل کنید که هنگام نمایش محصولات موجود برای خرید، حقوق کاربر را بررسی کرده و آنچه را که کاربر می‌تواند خریداری کند، فیلتر کنید. هنگامی که برنامه مشتری این خطا را به دلیل مشکل حافظه پنهان دریافت می کند، این خطا باعث می شود که حافظه پنهان Google Play با آخرین داده های باطن Play به روز شود. تلاش مجدد پس از خطا باید این نمونه گذرا خاص را در این مورد برطرف کند. پس از دریافت ITEM_ALREADY_OWNED با BillingClient.queryPurchasesAsync() تماس بگیرید تا بررسی کنید آیا کاربر محصول را خریداری کرده است یا خیر، و اگر اینطور نیست، یک منطق ساده امتحان مجدد را برای تلاش مجدد برای خرید اجرا کنید.

ITEM_NOT_OWNED

مشکل

این پاسخ خرید نشان می‌دهد که کاربر Google Play مالک اشتراک یا محصول خرید یک‌باره‌ای نیست که کاربر سعی در جایگزینی، تأیید یا مصرف آن دارد. این یک خطای گذرا در بیشتر موارد نیست، مگر زمانی که به دلیل قرار گرفتن حافظه پنهان Google Play در حالت بیات ایجاد شود.

وضوح ممکن

هنگامی که خطا به دلیل مشکل حافظه پنهان دریافت می‌شود، این خطا باعث می‌شود که حافظه پنهان Google Play با آخرین داده‌های باطن Play به‌روزرسانی شود. تلاش مجدد با یک استراتژی ساده مجدد پس از خطا باید این نمونه گذرا خاص را حل کند. پس از دریافت ITEM_NOT_OWNED با BillingClient.queryPurchasesAsync() تماس بگیرید تا بررسی کنید که آیا کاربر محصول را خریداری کرده است یا خیر. اگر این کار را نکرده‌اند، از منطق ساده تلاش مجدد برای خرید مجدد استفاده کنید.

پاسخ های غیر قابل برگشت BillingResult

با استفاده از منطق سعی مجدد نمی توانید از این خطاها بازیابی کنید.

FEATURE_NOT_SUPPORTED

مشکل

این خطای غیر قابل برگشت نشان می دهد که ویژگی صورتحساب Google Play در دستگاه کاربر پشتیبانی نمی شود، احتمالاً به دلیل نسخه قدیمی Play Store.

به عنوان مثال، شاید برخی از دستگاه های کاربران شما از پیام رسانی درون برنامه ای پشتیبانی نکنند.

کاهش احتمالی

از BillingClient.isFeatureSupported() برای بررسی پشتیبانی از ویژگی ها قبل از برقراری تماس با کتابخانه صورتحساب Play استفاده کنید.

when {
  billingClient.isReady -> {
    if (billingClient.isFeatureSupported(BillingClient.FeatureType.IN_APP_MESSAGING)) {
       // use feature
    }
  }
}

USER_CANCELED

مشکل

کاربر از رابط کاربری جریان صورت‌حساب کلیک کرده است.

وضوح ممکن

این فقط اطلاعاتی است و ممکن است به طرز دلپذیری شکست بخورد.

ITEM_UNAVAILABLE

مشکل

اشتراک صورت‌حساب Google Play یا محصول یک‌بار خرید برای این کاربر در دسترس نیست.

کاهش احتمالی

اطمینان حاصل کنید که برنامه شما جزئیات محصول را از طریق queryProductDetailsAsync همانطور که توصیه می شود بازخوانی می کند. به تعداد دفعات تغییر کاتالوگ محصولتان در پیکربندی کنسول Play توجه کنید تا در صورت نیاز، به‌روزرسانی‌های اضافی را اعمال کنید. فقط سعی کنید محصولاتی را در صورت‌حساب Google Play بفروشید که اطلاعات درست را از طریق queryProductDetailsAsync برمی‌گردانند. پیکربندی واجد شرایط بودن محصول را برای هرگونه ناهماهنگی بررسی کنید. به عنوان مثال، ممکن است در حال جستجو برای محصولی باشید که فقط برای منطقه ای غیر از منطقه ای که کاربر سعی در خرید آن دارد در دسترس است. برای اینکه یک محصول برای خرید در دسترس باشد، باید فعال باشد، برنامه آن باید منتشر شود، و برنامه آن باید در کشور کاربر موجود باشد.

گاهی اوقات، به ویژه در هنگام آزمایش، همه چیز در پیکربندی محصول درست است، اما کاربران همچنان این خطا را مشاهده می کنند. این ممکن است به دلیل تأخیر انتشار جزئیات محصول در سرورهای Google باشد. بعداً دوباره امتحان کنید.

DEVELOPER_ERROR

مشکل

این یک خطای مهلک است که نشان می دهد شما به درستی از یک API استفاده نمی کنید. به عنوان مثال، ارائه پارامترهای نادرست به BillingClient.launchBillingFlow می تواند باعث این خطا شود.

وضوح ممکن

مطمئن شوید که به درستی از تماس‌های مختلف کتابخانه صورت‌حساب Play استفاده می‌کنید. همچنین، برای اطلاعات بیشتر در مورد خطا، پیام اشکال زدایی را بررسی کنید.