5.1 KiB
5.1 KiB
id, title, category, status, source_trust_level, verification_status, created_at, updated_at, tags, tech_stack, applied_in, aliases
| id | title | category | status | source_trust_level | verification_status | created_at | updated_at | tags | tech_stack | applied_in | aliases | ||||||||||||||
|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|
| android-billingclient-iap | Android Billing Client — IAP / Subscription | Coding | draft | B | conceptual | 2026-05-09 | 2026-05-09 |
|
|
|
Android Billing Client (IAP)
Google Play 결제. acknowledge / consume 안 하면 사용자 환불. 서버 검증 (Real-Time Developer Notifications + Subscription API) 필수.
📖 핵심 개념
- ProductDetails: Google Play Console 등록 상품.
- Purchase: 구매 결과. acknowledged / consumed 상태.
- Acknowledge: 3일 안 안 하면 자동 환불.
- Server: Pub/Sub RTDN 으로 갱신 / 환불 / 만료 통지.
💻 코드 패턴
의존성
implementation("com.android.billingclient:billing-ktx:7.0.0")
Setup
class BillingManager(ctx: Context) : PurchasesUpdatedListener {
private val client = BillingClient.newBuilder(ctx)
.setListener(this)
.enablePendingPurchases()
.build()
suspend fun startConnection(): BillingResult = suspendCancellableCoroutine { cont ->
client.startConnection(object : BillingClientStateListener {
override fun onBillingSetupFinished(result: BillingResult) { cont.resume(result) }
override fun onBillingServiceDisconnected() { /* retry later */ }
})
}
suspend fun queryProducts(ids: List<String>): List<ProductDetails> {
val params = QueryProductDetailsParams.newBuilder()
.setProductList(ids.map {
QueryProductDetailsParams.Product.newBuilder()
.setProductId(it)
.setProductType(BillingClient.ProductType.SUBS)
.build()
})
.build()
return client.queryProductDetails(params).productDetailsList ?: emptyList()
}
fun launchPurchase(activity: Activity, product: ProductDetails) {
val flow = BillingFlowParams.newBuilder()
.setProductDetailsParamsList(listOf(
BillingFlowParams.ProductDetailsParams.newBuilder()
.setProductDetails(product)
.setOfferToken(product.subscriptionOfferDetails!!.first().offerToken)
.build()
))
.setObfuscatedAccountId(currentUserId) // server matching
.build()
client.launchBillingFlow(activity, flow)
}
override fun onPurchasesUpdated(result: BillingResult, purchases: List<Purchase>?) {
if (result.responseCode == BillingClient.BillingResponseCode.OK && purchases != null) {
for (p in purchases) handlePurchase(p)
}
}
private fun handlePurchase(purchase: Purchase) {
if (purchase.purchaseState != Purchase.PurchaseState.PURCHASED) return
// 1) 서버 검증
scope.launch {
val verified = api.verify(purchase.purchaseToken, purchase.products.first(), currentUserId)
if (!verified) return@launch
// 2) acknowledge (subscription) or consume (consumable)
if (!purchase.isAcknowledged) {
val params = AcknowledgePurchaseParams.newBuilder()
.setPurchaseToken(purchase.purchaseToken)
.build()
client.acknowledgePurchase(params) {}
}
}
}
}
서버 검증 (Subscription API)
// Backend
import { google } from 'googleapis';
const auth = new google.auth.GoogleAuth({ keyFile: serviceAccountPath, scopes: ['https://www.googleapis.com/auth/androidpublisher'] });
const pub = google.androidpublisher({ version: 'v3', auth });
const r = await pub.purchases.subscriptionsv2.get({
packageName: 'com.example.app',
token: purchaseToken,
});
if (r.data.subscriptionState === 'SUBSCRIPTION_STATE_ACTIVE') {
await grantEntitlement(userId, productId);
}
RTDN (Real-Time Developer Notifications)
- Google Cloud Pub/Sub 구독.
- 갱신 / 환불 / 만료 / 환경 변경 notify.
- 서버가 사용자 entitlement 갱신.
🤔 의사결정 기준
| 상품 | type |
|---|---|
| 1회 영구 (광고 제거) | INAPP |
| 사용 후 소진 (코인) | INAPP + consume |
| 월/년 구독 | SUBS + offer |
| 무료 trial | SUBS + introductoryPrice offer |
❌ 안티패턴
- acknowledge 안 함: 3일 후 자동 환불.
- client-side validation 만: 우회 가능. 서버 + signature 검증.
- consume vs acknowledge 헷갈림: consumable 만 consume, 그 외 acknowledge.
- obfuscatedAccountId 미설정: 결제 ↔ 사용자 매핑 어려움.
- RTDN 없음: 환불 / 만료 모름.
- client 가 entitlement 결정: 서버가 진실. 클라는 표시.
- 동시에 여러 구독 가능: 의도와 다른 구독 활성. SkuDetails 하나만 active.
🤖 LLM 활용 힌트
- BillingClient 7+ + obfuscatedAccountId + 서버 검증 + RTDN 4종 세트.
- acknowledge / consume 분리.