revenuecat-migrate: migrate to RevenueCat or upgrade the SDK
Use this skill when the user wants to either adopt RevenueCat in an app that already ships in app purchases, or upgrade the RevenueCat SDK across a major version.
These two paths share some concepts but have different risks. Identify which one applies before touching code.
1. Detect the platform
Inspect the working directory and pick the first match, from top to bottom:
- React Native: has a entry, or as a dependency → read
platforms/react-native.md
. If is also a dependency, note it as an Expo project.
- Flutter: exists at the project root → read .
- Kotlin Multiplatform: contains a multiplatform source sets block, or depends on
com.revenuecat.purchases:purchases-kmp*
→ read .
- Android (native): applies (and is not KMP) → read .
- iOS (native): , , , or at the project root → read .
If several match (e.g. an
folder inside a Flutter project), pick the
outermost project, the one that owns the build. If still ambiguous, ask the user which platform they want to configure.
2. Identify the migration path
Ask the user (or infer from the codebase):
- Path A: adoption. The app already has working in app purchases implemented directly against StoreKit or Google Play Billing. RevenueCat is being added on top.
- Path B: version upgrade. The app already uses RevenueCat, and the user wants to bump from one major version to the next (e.g. v4 to v5, v7 to v8).
Both paths can happen at once (e.g. adopt RC today on the latest major version). Run Path A first, then Path B if needed.
3. Shared concepts
Observer mode (Path A)
Observer mode is the key lever for adopting RevenueCat without rewriting purchase code. The SDK observes transactions that your existing StoreKit / Billing code processes, sends them to the RevenueCat backend for validation, and updates subscriber state, but does not initiate or finish the transactions. Your existing purchase UI, receipt validation, and transaction finishing stay in place.
Set this at configure time:
- iOS: set
purchasesAreCompletedBy: .myApp
together with storeKitVersion: .storeKit1
(or ) on . They are separate parameters, not a single associated value.
- Android:
purchasesAreCompletedBy(PurchasesAreCompletedBy.MY_APP)
on PurchasesConfiguration.Builder
.
- Flutter: pass
const PurchasesAreCompletedByMyApp(storeKitVersion: StoreKitVersion.storeKit2)
to .
- React Native: pass
purchasesAreCompletedBy: { type: PURCHASES_ARE_COMPLETED_BY_TYPE.MY_APP, storeKitVersion: STOREKIT_VERSION.STOREKIT_2 }
in the configure call.
The default is RevenueCat completed (
/
), where the SDK owns the full flow.
Once stable in observer mode, you can optionally cut over to full RevenueCat mode later by removing your own purchase plumbing and dropping the
override.
Do not double process transactions
When
is set to
, RevenueCat does
not finish transactions on iOS or acknowledge on Android. Your existing code must continue to do that. If you remove the
flag while leaving your old transaction finishing code in place, transactions get acknowledged twice and subscriber state can appear inconsistent.
Exactly one side must own finishing / acknowledging. Pick a side and remove the other.
User continuity (Path A)
If the app already has its own authentication system, call
Purchases.logIn(existingAppUserID)
once RevenueCat is configured. This attaches the prior purchase history to the right RevenueCat user on ingestion. Without this step, existing purchases get recorded against an anonymous RC user and cannot be matched to the app's actual user records later.
Only skip this if the app has no notion of authenticated users.
Version bumps change required fields (Path B)
Major version upgrades change configuration shape, drop deprecated APIs, and shift default behavior in ways that move with each release. This skill does not duplicate the per-version diff. Read the canonical sources from the SDK repo:
- CHANGELOG: in the relevant SDK repo on GitHub. Walk entries from your installed version up to the target.
- Migration guides: search the SDK repo for files matching or a directory. Major bumps usually ship a dedicated guide there. The release notes for the major version on the repo's GitHub releases page typically link to it.
- Release notes: each major version's release notes on the repo's GitHub releases page summarize the breaking changes.
Treat the SDK repo's docs as authoritative. Any version-specific diff written here would drift out of date. The platform file under
for your target lists the exact repo to consult.
Plan then migrate
Work in this order on every platform:
- Bump the SDK to the new major version in a branch.
- Fix compile errors using the CHANGELOG deprecations and removals as a guide.
- Fix runtime behavior by reading the SDK logs on first launch.
- Run the existing test suite and manual sandbox scenarios before merging.
4. Implementation
Read the platform file that matches detection:
platforms/react-native.md
Each platform file covers both migration paths for that platform.
5. Verify
Do not declare migration done until:
- The app builds on the new SDK version with no warnings from deprecated APIs you care about.
- A sandbox purchase succeeds and the transaction shows up on the RevenueCat dashboard Sandbox view with the expected appUserID.
- An existing subscriber from before the migration opens the app, and their entitlement state is correct. For Path A this proves the observer mode ingest worked. For Path B this proves the version bump did not drop state.
- You have removed the debug log level override before shipping.