Upgrades, Downgrades, & Management
Some parts of a customer's subscription can be managed directly through RevenueCat, other parts can only be managed by the customer directly in the respective stores (Apple, Google, Stripe, and Amazon). Learn how to upgrade/downgrade, cancel, and refund subscriptions here!
For information about Stripe, you can read more about working with web payments here.
Upgrade or Downgrade a Subscriptionโ
The app stores do not allow developers (you) to upgrade or downgrade a subscription on behalf of a customer. Only a customer can change their subscription.
App Storeโ
There are no code changes required to support upgrades, downgrades, and crossgrades for iOS subscriptions in your app. A customer can choose to upgrade, downgrade, or crossgrade between subscriptions as often as they like.
According to Apple, when a customer changes their subscription level, access to the new product can vary depending on the change:
Upgrade. A user purchases a subscription that offers a higher level of service than their current subscription. They are immediately upgraded and receive a refund of the prorated amount of their original subscription. If youโd like users to immediately access more content or features, rank the subscription higher to make it an upgrade.
Downgrade. A user selects a subscription that offers a lower level of service than their current subscription. The subscription continues until the next renewal date, then is renewed at the lower level and price.
Crossgrade. A user switches to a new subscription of the equivalent level. If the subscriptions are the same duration, the new subscription begins immediately. If the durations are different, the new subscription goes into effect at the next renewal date.
You can refer to this blog post for more information on how to set up subscription groups in App Store Connect.
When a customer upgrades products during an introductory period (including a free trial), Apple does not cancel the introductory offer but instead keeps the introductory offer active in addition to the upgraded product. So in these cases you can expect two products in the same Subscription Group to be active simultaneously.
Google Playโ
In order to perform upgrades and downgrades for Google Play subscriptions, you will need to set the old product ID on PurchaseParams.Builder
. Replacement mode is optional and will default to WITHOUT_PRORATION
.
- Kotlin
- Java
Purchases.sharedInstance.purchase(
PurchaseParams.Builder(requireActivity(), pkg)
.oldProductId("oldProductId:oldBasePlanId")
.googleReplacementMode(GoogleReplacementMode.WITHOUT_PRORATION)
.build(),
object : PurchaseCallback {
override fun onCompleted(storeTransaction: StoreTransaction, customerInfo: CustomerInfo) {
val entitlement = customerInfo.entitlements["my_entitlement_identifier"]
if (entitlement != null && entitlement.isActive) {
// Unlock that great "pro" content
}
}
override fun onError(purchasesError: PurchasesError, b: Boolean) {
// No purchase
}
}
)
Purchases.getSharedInstance().purchase(
new PurchaseParams.Builder(activity, pkg)
.oldProductId("oldProductId:oldBasePlanId")
.googleReplacementMode(GoogleReplacementMode.WITHOUT_PRORATION)
.build(),
new PurchaseCallback() {
@Override
public void onCompleted(@NonNull StoreTransaction storeTransaction, @NonNull CustomerInfo customerInfo) {
if (customerInfo.getEntitlements().get("my_entitlement_identifier").isActive()) {
// Unlock that great "pro" content
}
}
@Override
public void onError(@NonNull PurchasesError purchasesError, boolean b) {
// No purchase
}
}
);
Google documentation provides examples of each replacement mode, behavior when upgrading with free trial or intro price offers, and recommendations for which replacement mode to use in different scenarios.
Mode | Description |
---|---|
WITHOUT_PRORATION | Old subscription is cancelled, and new subscription takes effect immediately. User is charged for the full price of the new subscription on the old subscription's expiration date. This is the default behavior. Google recommends this mode for upgrading while in a free trial. |
WITH_TIME_PRORATION | Old subscription is cancelled, and new subscription takes effect immediately. Any time remaining on the old subscription is used to push out the first payment date for the new subscription. User is charged the full price of new subscription once that prorated time has passed. The purchase will fail if this mode is used when switching between SubscriptionOption s of the same StoreProduct . |
CHARGE_FULL_PRICE | Replacement takes effect immediately, and the user is charged full price of new plan and is given a full billing cycle of subscription, plus remaining prorated time from the old plan. |
CHARGE_PRORATED_PRICE | Replacement takes effect immediately, and the billing cycle remains the same. The price difference for the remaining period is then charged to the user. Note: This option is available only for a subscription upgrade, where the price per unit of time increases. Google recommends this mode for upgrading to a more expensive tier, and for upgrading while in a free trial which will end access to the free trial. |
DEFERRED | Replacement takes effect when the old plan expires, and the new price will be charged at the same time. Google recommends this mode for downgrading to a less expensive tier, and for changing recurring period on the same tier (from monthly to annual). Important: Google Server Notifications are required to be configured for DEFERRED mode to work properly. |
Amazon Appstoreโ
Amazon does not support changing products. Customers will need to cancel their existing subscription and re-subscribe to a different product.
Rokuโ
- Upgrades: Upgrading products will cancel the previous subscription and start the new one immediately
- Downgrades: Downgrades schedule the subscription change for when the current billing period ends, then the new subscription product will take effect
Please note when your customer cancels a pending downgrade transaction through the Roku dashboard, this will not only cancel the downgrade, but will also turn off auto-renew for the existing subscription product.
Considerationsโ
PRODUCT_CHANGE events and webhooksโ
- The
expiration_at_ms
will always be for the product the customer is changing from (old product). - The
PRODUCT_CHANGE
webhook should be considered informative, and does not mean that the product change has gone into effect. When the product change goes into effect you will receive aRENEWAL
event on Apple and Stripe or aINITIAL_PURCHASE
event on Google Play.
Prorations and Revenue Metricsโ
App Storeโ
For Apple transactions, prorated revenue will not be shown in the Customer History page. However, prorated revenue will be calculated for your chart and overview data.
Google Playโ
For Google transactions, prorated revenue will be shown in the Customer History page and will be calculated for your chart and overview data.
Cancelling Subscriptionsโ
Cancelling (or unsubscribing) from subscriptions is handled differently on each platform. Cancellations are automatically detected by RevenueCat within a few hours of occurring. This detection time can be increased to near real-time by enabling Platform Server Notifications.
Store | Behavior |
---|---|
Apple | Apple does not allow developers to manage subscriptions on behalf of users. Your customers have to manually opt-out of renewal. The Apple subscription terms require users to cancel subscriptions at least 24 hours before the next renewal. |
Google Play | Google allows developers to cancel auto-renewing subscriptions on behalf of customers via Google Play Console or Google's API. For prepaid subscriptions, neither the customer nor the developer can cancel the subscription as it already has an expiration date. If you're looking to refund and revoke a subscription via RevenueCat's dashboard, see Refunds. |
Stripe | Subscriptions can be cancelled on behalf of customers via the Stripe dashboard or REST API. Refer to Stripe documentation for more info. |
Amazon | Amazon does not allow developers to cancel subscriptions on behalf of users. Once a subscription is purchased for a period, it is valid through that period and cannot be cancelled. |
Using the managementURL
to Help Customers Cancel a Subscriptionโ
Google requires developers to allow customers to cancel a subscription within apps. You can do this by displaying a link in your app that takes the user directly to the Google Play or App Store subscription management screen where they can immediately cancel their subscription. RevenueCat helps you do this by providing a managementURL
property on the CustomerInfo object in the SDK and in our REST API.
The managementURL
is a great way to allow customers to check the status of and manage their subscriptions on both iOS and Android. RevenueCat will automatically provide your app with the correct managementURL
based on the platform of the customer's device and original store they purchased their subscription through.
managementURL
is not supported on Amazon or Stripe.
- Swift
- Kotlin
Purchases.shared.getCustomerInfo { (customerInfo, error) in
let managementURL = customerInfo.managementURL
// display the managementURL in your app
}
Purchases.sharedInstance.getCustomerInfo({ error -> /* Optional error handling */ }) { customerInfo ->
val managementURL = customerInfo.managementURL
// display the managementURL in your app
}
Deleting a user from RevenueCat WILL NOT cancel their subscription. The user can still trigger the Restore Purchases method to re-sync their transactions with RevenueCat servers.