Google Play Billing は、Android におけるアプリ内購入やサブスクリプションを処理するための包括的な API を提供しています。ほとんどの開発者は、標準的な購入フローには慣れているでしょう。すなわち、課金フローを起動し、結果を受け取り、購入を承認し、エンタイトルメントを付与するという流れです。しかし、本番環境の課金システムでは、チュートリアルやサンプルコードでは十分に扱われない、より幅広いシナリオに対応する必要があります。保留中の購入(Pending purchases)、複数量の消耗型アイテム、日割り計算を伴うサブスクリプションのダウングレード、そして ITEM_ALREADY_OWNED レスポンスは、いずれも実際の環境で発生するケースであり、これらを誤って処理すると、収益の損失、ユーザーの混乱、購入失敗につながる可能性があります。

本記事では、Google Play Billing における代表的なエッジケースを取り上げ、それぞれが発生する理由を理解し、Play Billing Library を使って正しく処理する方法を確認します。また、RevenueCat がこれらのシナリオをどのように簡素化し、課金インフラではなくプロダクト開発に集中できるようにするのかも見ていきます。

根本的な問題:ハッピーパスだけでは不十分

多くの課金実装は、Android の公式ドキュメントにあるサンプルコードから始まります。

1// The happy path
2billingClient.launchBillingFlow(activity, params)
3
4// In PurchasesUpdatedListener
5override fun onPurchasesUpdated(
6    billingResult: BillingResult,
7    purchases: List<Purchase>?
8) {
9    if (billingResult.responseCode == BillingClient.BillingResponseCode.OK) {
10        purchases?.forEach { purchase ->
11            if (purchase.purchaseState == Purchase.PurchaseState.PURCHASED) {
12                acknowledgePurchase(purchase)
13                grantEntitlement(purchase)
14            }
15        }
16    }
17}

これは、成功した即時購入を処理するケースを想定しています。しかし、ユーザーがコンビニで支払いを行い、支払いが48時間遅れる場合はどうでしょうか? 以前の承認処理が静かに失敗しており、ユーザーがすでにその商品を所有している場合はどうでしょうか? サブスクリプションのダウングレードが即時ではなく、次回の更新時に反映される場合はどうでしょうか? これらの各シナリオには個別の対応が必要であり、無視するとサポート問い合わせの増加、返金リクエスト、そしてサブスクライバーの離脱につながります。

保留中の購入(Pending purchases):支払いが即時ではない場合

すべての購入が即座に完了するわけではありません。コンビニでの現金支払い、銀行振込、一部のキャリア決済など、特定の支払い方法では非同期処理が必要になります。ユーザーがこれらの方法で購入を開始すると、Google Play は PURCHASED 状態ではなく、PENDING  状態の購入として返します。

なぜ保留中の購入が発生するのか

保留中の購入は、クレジットカードの普及率が低い市場で一般的です。

支払い方法一般的な地域通常の処理時間
現金支払い(コンビニ)日本、メキシコ、インドネシア24〜48時間
銀行振込ドイツ、オランダ、ブラジル1〜3営業日
キャリア決済(一部のキャリア)各国数分〜数時間

アプリをグローバルに展開している場合、保留中の購入に遭遇するのは避けられません。この状態を無視すると、これらの地域のユーザーはそもそも商品を購入できなくなるか、あるいは購入が「消えた」ように見える混乱した挙動に直面することになります。

保留状態の検出と処理

PurchasesUpdatedListener は、完了した購入と同様に、保留中の購入も受け取ります。重要な違いは、 purchaseState フィールドにあります。

1override fun onPurchasesUpdated(
2    billingResult: BillingResult,
3    purchases: List<Purchase>?
4) {
5    if (billingResult.responseCode == BillingClient.BillingResponseCode.OK) {
6        purchases?.forEach { purchase ->
7            when (purchase.purchaseState) {
8                Purchase.PurchaseState.PURCHASED -> {
9                    // Payment complete, grant access
10                    processPurchase(purchase)
11                }
12                Purchase.PurchaseState.PENDING -> {
13                    // Payment not yet complete
14                    handlePendingPurchase(purchase)
15                }
16                Purchase.PurchaseState.UNSPECIFIED_STATE -> {
17                    // Unknown state, query backend for clarification
18                    queryBackendForState(purchase)
19                }
20            }
21        }
22    }
23}

重要なルールは次のとおりです。保留中の購入に対してエンタイトルメントを付与してはいけません。ユーザーはまだ支払いを完了していません。代わりに、保留中の購入を記録し、そのステータスを明確にユーザーへ伝える必要があります。

1fun handlePendingPurchase(purchase: Purchase) {
2    // Store the pending purchase token for later verification
3    purchaseRepository.savePendingPurchase(
4        purchaseToken = purchase.purchaseToken,
5        productId = purchase.products.first(),
6        orderId = purchase.orderId,
7        purchaseTime = purchase.purchaseTime,
8    )
9
10    // Show clear UI to the user
11    showPendingUI(
12        message = "Your purchase is being processed. " +
13            "You'll get access once payment is confirmed.",
14    )
15}

保留中の購入の完了

支払いが最終的に確認されると、アプリは onPurchasesUpdated または queryPurchasesAsync を通じて更新された購入情報を受け取ります。purchaseState は PURCHASED になり、承認処理およびエンタイトルメントの付与を進めることができます。

ただし、注意点があります。支払いが完了した時点で、ユーザーがアプリを開いていない可能性があります。このケースには、Real-Time Developer Notifications(RTDN)を通じてバックエンドで対応すべきです。以前は保留中だったトークンに対して ONE_TIME_PRODUCT_PURCHASED または SUBSCRIPTION_PURCHASED の通知を受け取った場合、バックエンドでエンタイトルメントを更新し、ユーザーに通知する必要があります。

1// Backend notification handler
2fun handlePurchaseNotification(notification: DeveloperNotification) {
3    val purchaseToken = notification.oneTimeProductNotification?.purchaseToken
4        ?: notification.subscriptionNotification?.purchaseToken
5        ?: return
6
7    val pendingPurchase = purchaseRepository.findPendingPurchase(purchaseToken)
8    if (pendingPurchase != null) {
9        // Previously pending purchase is now complete
10        val purchaseDetails = playDeveloperApi
11            .purchases()
12            .products()
13            .get(packageName, pendingPurchase.productId, purchaseToken)
14            .execute()
15
16        if (purchaseDetails.purchaseState == 0) { // 0 = Purchased
17            entitlementRepository.grantEntitlement(
18                userId = pendingPurchase.userId,
19                productId = pendingPurchase.productId,
20            )
21            purchaseRepository.markCompleted(purchaseToken)
22
23            // Notify user that their purchase is ready
24            notificationService.sendPushNotification(
25                userId = pendingPurchase.userId,
26                title = "Purchase Complete",
27                body = "Your purchase has been confirmed. Enjoy your content!",
28            )
29        }
30    }
31}

BillingClient で保留中の購入を有効にする

保留中の購入をサポートするには、 BillingClient を構築する際に明示的に有効化する必要があります。これを行わない場合、遅延支払い方法による購入は完全に失敗します。

1val billingClient = BillingClient.newBuilder(context)
2    .setListener(purchasesUpdatedListener)
3    .enablePendingPurchases(
4        PendingPurchasesParams.newBuilder()
5            .enableOneTimeProducts()
6            .enablePrepaidPlans()
7            .build()
8    )
9    .build()

Play Billing Library 7 以降では、 enablePendingPurchases() の呼び出しが必須となっています。これを行わない場合、 BillingClient の初期化は失敗します。

ITEM_ALREADY_OWNED レスポンス:よくある混乱の原因

最も頻繁に遭遇するエッジケースの一つが BillingResponseCode.ITEM_ALREADY_OWNED です。このレスポンスは、ユーザーがすでに所有している非消耗型商品やサブスクリプションを再度購入しようとした場合に発生します。一見すると分かりやすい挙動に思えますが、実際にこれが発生するシナリオは意外なものが少なくありません。

なぜ ITEM_ALREADY_OWNED が発生するのか

最も一般的な原因は、ユーザーが意図的に同じ商品を二重購入しようとしていることではありません。問題は、以前の購入が正しく承認(acknowledge)されなかったことにあります。Google Play では購入の承認が必須であるため、未承認の購入は宙ぶらりんの状態に置かれます。つまり、ユーザーはすでに課金されているものの、購入がアプリ側で確認されていない状態です。このまま同じ商品を再度購入しようとすると、未承認の購入が残っているため、Google Play は ITEM_ALREADY_OWNED を返します。

この状況は、想像以上によく発生します。

  • アプリが購入受信後、承認前にクラッシュした
  • ネットワークエラーにより承認呼び出しが完了しなかった
  • 購入フロー中にユーザーがアプリを強制終了した
  • 承認 API 呼び出しがエラーを返し、再試行されなかった

ITEM_ALREADY_OWNED を正しく処理する

ITEM_ALREADY_OWNED に対する正しい対応は、エラーメッセージを表示することではありません。代わりに、既存の購入をクエリし、未承認の購入があればそれを処理するべきです。

1override fun onPurchasesUpdated(
2    billingResult: BillingResult,
3    purchases: List<Purchase>?
4) {
5    when (billingResult.responseCode) {
6        BillingClient.BillingResponseCode.OK -> {
7            purchases?.forEach { processPurchase(it) }
8        }
9        BillingClient.BillingResponseCode.ITEM_ALREADY_OWNED -> {
10            // Query existing purchases and process any unacknowledged ones
11            recoverUnacknowledgedPurchases()
12        }
13        BillingClient.BillingResponseCode.USER_CANCELED -> {
14            // User backed out, no action needed
15        }
16        else -> {
17            handleBillingError(billingResult)
18        }
19    }
20}
21
22private fun recoverUnacknowledgedPurchases() {
23    val params = QueryPurchasesParams.newBuilder()
24        .setProductType(BillingClient.ProductType.INAPP)
25        .build()
26
27    billingClient.queryPurchasesAsync(params) { billingResult, purchases ->
28        if (billingResult.responseCode == BillingClient.BillingResponseCode.OK) {
29            purchases.forEach { purchase ->
30                if (purchase.purchaseState == Purchase.PurchaseState.PURCHASED &&
31                    !purchase.isAcknowledged
32                ) {
33                    // Found the unacknowledged purchase, process it
34                    processPurchase(purchase)
35                }
36            }
37        }
38    }
39}

このパターンにより、フラストレーションを招くエラーを、シームレスな復旧処理へと変えることができます。以前の購入が承認に失敗していたことを、ユーザーが知る必要はありません。ユーザーの視点では、「購入」をタップすれば商品が手に入るだけです。

ITEM_ALREADY_OWNED を事前に防ぐ

最善のアプローチは、アプリ起動時に未承認の購入を処理することで、この状況を未然に防ぐことです。

1fun processUnacknowledgedPurchasesOnStartup() {
2    val inAppParams = QueryPurchasesParams.newBuilder()
3        .setProductType(BillingClient.ProductType.INAPP)
4        .build()
5
6    val subsParams = QueryPurchasesParams.newBuilder()
7        .setProductType(BillingClient.ProductType.SUBS)
8        .build()
9
10    billingClient.queryPurchasesAsync(inAppParams) { result, purchases ->
11        if (result.responseCode == BillingClient.BillingResponseCode.OK) {
12            purchases.filter {
13                it.purchaseState == Purchase.PurchaseState.PURCHASED &&
14                    !it.isAcknowledged
15            }.forEach { processPurchase(it) }
16        }
17    }
18
19    billingClient.queryPurchasesAsync(subsParams) { result, purchases ->
20        if (result.responseCode == BillingClient.BillingResponseCode.OK) {
21            purchases.filter {
22                it.purchaseState == Purchase.PurchaseState.PURCHASED &&
23                    !it.isAcknowledged
24            }.forEach { processPurchase(it) }
25        }
26    }

このメソッドは、 BillingClient が正常に接続されたタイミングで呼び出してください。これにより、取りこぼされていた購入がユーザーに問題が発生する前に回復されます。

消耗型購入: acknowledge と consume の違い

ゲーム内通貨、追加ライフ、トークンパックのような消耗型商品では、承認(acknowledgment)と消費(consumption)の違いがバグの一般的な原因になります。どちらも消耗型商品には必要ですが、目的もタイミング要件も異なります。

承認と消費のフロー

承認(Acknowledgment)は、購入されたコンテンツを配信済みであることを Google Play に通知するものです。購入から3日以内に行う必要があり、期限を過ぎると自動的に返金されます。

消費(Consumption)は、その購入をリセットし、ユーザーが同じ商品を再度購入できるようにする処理です。商品を消費しない場合、ユーザーは再購入できず、再購入を試みると ITEM_ALREADY_OWNED が返されます。

消耗型商品の場合は、購入を消費するべきです。消費処理は暗黙的に承認も行います。

1fun processConsumablePurchase(purchase: Purchase) {
2    // Verify with backend first
3    verifyPurchaseWithBackend(purchase) { isValid ->
4        if (isValid) {
5            // Grant the consumable content
6            grantConsumableContent(purchase)
7
8            // Consume the purchase (this also acknowledges it)
9            val consumeParams = ConsumeParams.newBuilder()
10                .setPurchaseToken(purchase.purchaseToken)
11                .build()
12
13            billingClient.consumeAsync(consumeParams) { billingResult, _ ->
14                if (billingResult.responseCode != BillingClient.BillingResponseCode.OK) {
15                    // Retry consumption, the user cannot rebuy until consumed
16                    scheduleConsumptionRetry(purchase.purchaseToken)
17                }
18            }
19        }
20    }
21}

複数量購入のエッジケース

Google Play は、消耗型商品の複数量購入をサポートしています。ユーザーは、1回のトランザクションで同じ消耗型商品を複数個購入できます。数量は Purchase オブジェクト内で取得できます。

1fun processConsumablePurchase(purchase: Purchase) {
2    val quantity = purchase.quantity  // Could be > 1
3
4    verifyPurchaseWithBackend(purchase) { isValid ->
5        if (isValid) {
6            // Grant the correct quantity
7            grantConsumableContent(purchase.products.first(), quantity)
8
9            val consumeParams = ConsumeParams.newBuilder()
10                .setPurchaseToken(purchase.purchaseToken)
11                .build()
12
13            billingClient.consumeAsync(consumeParams) { billingResult, _ ->
14                if (billingResult.responseCode != BillingClient.BillingResponseCode.OK) {
15                    scheduleConsumptionRetry(purchase.purchaseToken)
16                }
17            }
18        }
19    }
20}

quantity フィールドを無視して常に1ユニットしか付与しない場合、複数個を購入したユーザーは支払った分より少ないアイテムしか受け取れません。これはサポート問い合わせや返金リクエストの原因になります。

複数量購入を有効にするには、Google Play Console で該当商品に対して「Allow multi-quantity purchases(複数量購入を許可)」を有効にする必要があります。さらに、 BillingFlowParams ではユーザーが選択できる最大数量を指定できます。

1val billingFlowParams = BillingFlowParams.newBuilder()
2    .setProductDetailsParamsList(
3        listOf(
4            BillingFlowParams.ProductDetailsParams.newBuilder()
5                .setProductDetails(productDetails)
6                .build()
7        )
8    )
9    .build()

消費処理(consume)の再試行問題

たとえばネットワークエラーなどで consumeAsync の呼び出しが失敗すると、ユーザーはコンテンツを受け取っているのに、購入が消費されていない状態になります。これは次のような問題につながります。

  1. ユーザーが同じ消耗型商品を再度購入できない
  2. 承認されないまま3日が経過すると購入が返金される可能性がある(消費処理は暗黙的に承認も行いますが)

そのため、消費処理に失敗した場合に備えて、再試行メカニズムを実装するべきです。

1class ConsumptionRetryManager(
2    private val billingClient: BillingClient,
3    private val purchaseRepository: PurchaseRepository,
4) {
5    fun scheduleConsumptionRetry(purchaseToken: String) {
6        purchaseRepository.markPendingConsumption(purchaseToken)
7    }
8
9    fun retryPendingConsumptions() {
10        val pendingTokens = purchaseRepository.getPendingConsumptionTokens()
11
12        pendingTokens.forEach { token ->
13            val consumeParams = ConsumeParams.newBuilder()
14                .setPurchaseToken(token)
15                .build()
16
17            billingClient.consumeAsync(consumeParams) { billingResult, _ ->
18                if (billingResult.responseCode == BillingClient.BillingResponseCode.OK) {
19                    purchaseRepository.clearPendingConsumption(token)
20                }
21                // If still failing, it will be retried next time
22            }
23        }
24    }
25}

BillingClient が接続されるたびに、未承認購入のリカバリーロジックとあわせて retryPendingConsumptions() を呼び出してください。

サブスクリプションのダウングレードと proration(日割り計算)モード

ユーザーがサブスクリプションプランを変更する際の課金挙動は、アップグレードかダウングレードか、そしてどの proration モードを指定しているかによって異なります。特にダウングレードは、多くの開発者にとって予想外の挙動を示します。

デフォルトのダウングレード挙動

ユーザーがサブスクリプションをダウングレード(より安価なプランへ変更)した場合、デフォルトの挙動は DEFERRED です。つまり、ダウングレードは即時ではなく、次回の更新日に適用されます。現在の請求期間が終了するまでは、ユーザーは引き続き上位プランの機能を利用できます。

1fun launchDowngrade(
2    activity: Activity,
3    newProductDetails: ProductDetails,
4    newOfferToken: String,
5    currentPurchaseToken: String,
6) {
7    val billingFlowParams = BillingFlowParams.newBuilder()
8        .setProductDetailsParamsList(
9            listOf(
10                BillingFlowParams.ProductDetailsParams.newBuilder()
11                    .setProductDetails(newProductDetails)
12                    .setOfferToken(newOfferToken)
13                    .build()
14            )
15        )
16        .setSubscriptionUpdateParams(
17            BillingFlowParams.SubscriptionUpdateParams.newBuilder()
18                .setOldPurchaseToken(currentPurchaseToken)
19                .setSubscriptionReplacementMode(
20                    BillingFlowParams.SubscriptionUpdateParams
21                        .ReplacementMode.DEFERRED
22                )
23                .build()
24        )
25        .build()
26
27    billingClient.launchBillingFlow(activity, billingFlowParams)
28}

リプレースメントモードの理解

各リプレースメントモードは、課金、アクセス権、そしてユーザー体験にそれぞれ異なる影響を与えます。

モード変更が適用されるタイミング課金への影響最適な用途
IMMEDIATE_WITH_TIME_PRORATION即時残り期間分が新しいプランにクレジットされるユーザーが即時にアクセスできるアップグレード
IMMEDIATE_AND_CHARGE_PRORATED_PRICE即時残り期間分に対して日割り請求公平な課金を伴うアップグレード
IMMEDIATE_AND_CHARGE_FULL_PRICE即時新しいプランの全額が請求され、新しい請求期間が開始プレミアムへのアップグレード
DEFERRED次回更新時即時の請求なしダウングレード
IMMEDIATE_WITHOUT_PRORATION即時次回更新まで追加請求なし同価格帯プラン間の移動や上位プランのトライアル

遅延ダウングレード(Deferred)の落とし穴

遅延ダウングレードで最もよくある誤りは、購入フロー完了直後にサブスクリプション状態を確認し、新しいプランが反映されていることを期待してしまうことです。 DEFERRED モードでは、次回の更新日まで元のサブスクリプションが元のプロダクト ID のまま有効です。新しいサブスクリプションが表示されるのは更新後になります。

つまり、エンタイトルメントのチェックでは、この移行期間を考慮する必要があります。

1fun handleDowngradeResult(purchase: Purchase) {
2    // After a deferred downgrade, the purchase still reflects
3    // the OLD subscription until the next renewal
4    val currentProductId = purchase.products.first()
5
6    // Query the subscription status from your backend to check
7    // if a deferred downgrade is pending
8    checkBackendForPendingDowngrade(purchase.purchaseToken) { pendingDowngrade ->
9        if (pendingDowngrade != null) {
10            // Show UI indicating the downgrade is scheduled
11            showDowngradeScheduledUI(
12                currentPlan = currentProductId,
13                futurePlan = pendingDowngrade.newProductId,
14                effectiveDate = pendingDowngrade.effectiveDate,
15            )
16        } else {
17            // Normal subscription state
18            showSubscriptionUI(currentProductId)
19        }
20    }
21}

プラン変更時の linkedPurchaseToken

サブスクリプションのリプレースメント(アップグレードまたはダウングレード)が処理されると、新しい購入トークンが生成されます。この新しい購入には、以前のサブスクリプションを参照する linkedPurchaseToken フィールドが含まれます。重複したエンタイトルメントを作成しないよう、バックエンドでこれを正しく処理する必要があります。

1// Backend handler for subscription replacement
2fun handleSubscriptionReplacement(newPurchaseToken: String) {
3    val subscription = playDeveloperApi
4        .purchases()
5        .subscriptionsv2()
6        .get(packageName, newPurchaseToken)
7        .execute()
8
9    val linkedToken = subscription.linkedPurchaseToken
10
11    if (linkedToken != null) {
12        // This is a plan change, not a new purchase
13        val userId = userRepository.findByPurchaseToken(linkedToken)
14
15        // Update to new token
16        userRepository.updatePurchaseToken(userId, newPurchaseToken)
17
18        // Invalidate old token to prevent double-counting
19        subscriptionRepository.invalidate(linkedToken)
20    }
21
22    // Acknowledge the new purchase
23    acknowledgePurchase(newPurchaseToken)
24}

リプレースメント処理時に古い購入トークンを無効化しないことは、よくあるバグの一つであり、サブスクライバー数の水増しや誤った収益レポートにつながります。

ネットワーク障害と再試行戦略

課金処理はネットワークに依存しており、ネットワーク障害や遅延は避けられません。失敗する可能性がある重要な処理には、購入フローそのもの、承認(acknowledgment)、消費(consumption)、購入検証が含まれます。

承認の猶予期間

Google Play では、購入を承認するために3日間の猶予が与えられています。この期間内に承認しない場合、購入は自動的に返金されます。これはユーザー保護のための仕組みですが、その分、承認ロジックは一時的な障害に対して耐性を持つ必要があります。

1class AcknowledgmentManager(
2    private val billingClient: BillingClient,
3    private val purchaseRepository: PurchaseRepository,
4) {
5    fun acknowledgePurchaseWithRetry(purchase: Purchase) {
6        if (purchase.isAcknowledged) return
7
8        val params = AcknowledgePurchaseParams.newBuilder()
9            .setPurchaseToken(purchase.purchaseToken)
10            .build()
11
12        billingClient.acknowledgePurchase(params) { billingResult ->
13            when (billingResult.responseCode) {
14                BillingClient.BillingResponseCode.OK -> {
15                    purchaseRepository.markAcknowledged(purchase.purchaseToken)
16                }
17                BillingClient.BillingResponseCode.SERVICE_UNAVAILABLE,
18                BillingClient.BillingResponseCode.SERVICE_DISCONNECTED,
19                BillingClient.BillingResponseCode.ERROR -> {
20                    // Transient failure, schedule retry
21                    purchaseRepository.markPendingAcknowledgment(
22                        purchase.purchaseToken
23                    )
24                }
25                else -> {
26                    // Non-retryable error, log for investigation
27                    logAcknowledgmentFailure(purchase, billingResult)
28                }
29            }
30        }
31    }
32
33    fun retryPendingAcknowledgments() {
34        val pendingTokens = purchaseRepository.getPendingAcknowledgmentTokens()
35
36        val inAppParams = QueryPurchasesParams.newBuilder()
37            .setProductType(BillingClient.ProductType.INAPP)
38            .build()
39
40        billingClient.queryPurchasesAsync(inAppParams) { result, purchases ->
41            if (result.responseCode == BillingClient.BillingResponseCode.OK) {
42                purchases
43                    .filter { it.purchaseToken in pendingTokens }
44                    .filter { !it.isAcknowledged }
45                    .forEach { acknowledgePurchaseWithRetry(it) }
46            }
47        }
48    }
49}

BillingClient の切断

The BillingClient はいつでも切断される可能性があり、切断された状態で実行された操作は失敗します。そのため、指数バックオフを用いた再接続ロジックを実装する必要があります。

1class BillingClientManager(
2    private val context: Context,
3    private val listener: PurchasesUpdatedListener,
4) {
5    private var billingClient: BillingClient? = null
6    private var retryCount = 0
7
8    fun connect(onConnected: () -> Unit) {
9        billingClient = BillingClient.newBuilder(context)
10            .setListener(listener)
11            .enablePendingPurchases(
12                PendingPurchasesParams.newBuilder()
13                    .enableOneTimeProducts()
14                    .build()
15            )
16            .build()
17
18        billingClient?.startConnection(object : BillingClientStateListener {
19            override fun onBillingSetupFinished(billingResult: BillingResult) {
20                if (billingResult.responseCode == BillingClient.BillingResponseCode.OK) {
21                    retryCount = 0
22                    onConnected()
23                } else {
24                    retryConnection(onConnected)
25                }
26            }
27
28            override fun onBillingServiceDisconnected() {
29                retryConnection(onConnected)
30            }
31        })
32    }
33
34    private fun retryConnection(onConnected: () -> Unit) {
35        if (retryCount < MAX_RETRY_COUNT) {
36            retryCount++
37            val delayMs = (1000L * (1 shl retryCount)).coerceAtMost(MAX_RETRY_DELAY_MS)
38            handler.postDelayed({ connect(onConnected) }, delayMs)
39        }
40    }
41
42    companion object {
43        private const val MAX_RETRY_COUNT = 5
44        private const val MAX_RETRY_DELAY_MS = 30_000L
45    }
46}

RevenueCat がこれらのエッジケースをどう扱うか

これまでに説明した各エッジケースには、慎重な実装、再試行ロジック、そしてバックエンド基盤が必要です。ここで RevenueCat は、複雑さを抽象化し、これらのシナリオを自動的に処理することで大きな価値を提供します。

保留中の購入(Pending purchases)

RevenueCat は保留中の購入状態を内部で追跡し、支払いが確定したタイミングで CustomerInfo を更新します。アプリ側はエンタイトルメントを確認するだけで済みます。

1Purchases.sharedInstance.getCustomerInfoWith { customerInfo ->
2    val isPremium = customerInfo.entitlements["premium"]?.isActive == true
3
4    if (isPremium) {
5        showPremiumContent()
6    } else {
7        showSubscriptionOptions()
8    }
9}

RevenueCat のバックエンドは、Google Play からの RTDN 通知を処理します。そのため、保留中の購入が完了すると、エンタイトルメントは RevenueCat のサーバー側で更新されます。次にアプリが CustomerInfo を取得したときには、そのエンタイトルメントは有効になっています。こちら側で独自の通知処理や購入トークンの追跡を行う必要はありません。特に個人開発の場合、こうしたバックエンド基盤を一式構築するのは非常に大きなリソースが必要になります。

承認と消費

RevenueCat は承認(acknowledgment)と消費(consumption)を自動で処理します。SDK が購入を受け取ると、その購入は RevenueCat のバックエンドで検証され、RevenueCat があなたに代わって Google Play に対して承認を行います。消耗型商品については、検証後に RevenueCat が消費処理を行います。 acknowledgePurchase や consumeAsync を自分で呼び出す必要はありません。

これにより、承認失敗、消費処理の呼び忘れ、そして ITEM_ALREADY_OWNED 問題に関連する一連のバグをすべて排除できます。

サブスクリプションプランの変更

RevenueCat は、purchaseWith を通じてサブスクリプションのアップグレードおよびダウングレードのためのシンプルな API を提供しています。

1Purchases.sharedInstance.purchaseWith(
2    PurchaseParams.Builder(activity, newPackage)
3        .oldProductId(currentProductId)
4        .googleReplacementMode(GoogleReplacementMode.DEFERRED)
5        .build(),
6    onSuccess = { transaction, customerInfo ->
7        // CustomerInfo reflects the new subscription state
8        updateUI(customerInfo)
9    },
10    onError = { error, userCancelled ->
11        if (!userCancelled) {
12            showError(error)
13        }
14    }
15)

RevenueCat は、linked purchase token の処理、エンタイトルメントの移行、そして遅延ダウングレードの追跡をバックエンドで処理します。アプリ側では、現在のエンタイトルメント状態を CustomerInfo で確認するだけで済みます。

ネットワーク耐性

RevenueCat の SDK には、すべてのネットワーク操作に対する組み込みの再試行ロジックが含まれており、 CustomerInfo をローカルにキャッシュしてオフラインでも参照できるようにしています。また、接続が回復した際にはバックエンドと自動的に同期されます。これにより、デバイスがオフラインの状態でもアプリはエンタイトルメントを確認できます。

1// This works offline using cached CustomerInfo
2Purchases.sharedInstance.getCustomerInfoWith { customerInfo ->
3    val isPremium = customerInfo.entitlements["premium"]?.isActive == true
4    updateUI(isPremium)
5}

SDK は古いデータと最新データを区別し、失敗した操作を指数バックオフで再試行し、最終的にエンタイトルメントがサーバー側の状態と整合することを保証します。

まとめ

本記事では、サンプルレベルの課金実装と本番運用レベルの実装を分ける、さまざまなエッジケースについて見てきました。

これらの各シナリオには、Play Billing Library を直接使って解決するための明確な方法があります。しかし、それらをすべて実装するための累積的な工数は決して小さくありません。クライアント側の処理、バックエンドでの RTDN 処理、再試行メカニズム、そして各状態をまたいだ慎重なステート管理が必要になります。こうしたインフラを自前で構築・維持せずにサブスクリプション機能を提供したいチームにとって、RevenueCat はこれらのエッジケースを自動的に処理し、複雑な管理を行う代わりに単一の CustomerInfo オブジェクトを確認するだけで済むようにしてくれます。

課金インフラを自前で構築する場合でも、RevenueCat を利用する場合でも、これらのエッジケースを理解することは不可欠です。テスト環境では動く課金システムと、多様な市場や支払い方法にまたがる数百万のユーザーに対して安定して動作する課金システムとの違いは、まさにここにあります。

それでは、Happy coding!

— Jaewoong