Migrating from Adapty to Qonversion
We have prepared this guide to help you move your infrastructure seamlessly from Adapty to Qonversion. Although this guide is comprehensive and should enable you to transfer the entire infrastructure smoothly, feel free to reach out to support if you have any questions.
Configure dashboard
First and foremost, register at Qonversion. Registration will take you through an onboarding process, where you'll have to fill out information about your project and insert links β all to help you add your project to Qonversion. We highly recommend following this flow and not skipping any settings during registration. We'll also cover the scenario if you don't fill these out.
Store Settings
First, let's set up your Stores in the Qonversion dashboard. To do this, gather the necessary data from Adapty.
Apple App Store settings
To configure the Apple App Store in Qonversion, you will need the following fields:
- App-Specific Shared Secret: Copy this value from the App Store Connect shared secret field in the App Settings -> iOS SDK section in the Adapty dashboard. Here's a more detailed guide on how to obtain this.
- App Store ID: You'll find your App Store ID in the URL of your App Store Connect account. For example, if your URL is something like
https://apps.apple.com/us/app/example-app/id123456789
, then 123456789 is your appβs Apple ID.
Google Play Console Settings
Here's what you have at Adapty:
To set up your Google Play account in Qonversion, you will need the following:
- Android Package Name: This is the applicationID found in your app-level build.gradle file. You can either copy it from there or use the Package Name from your Adapty settings.
- Service Account Credentials JSON: Copy the value from the Service account key file shown in the screenshot above, or attach the same file you used with Adapty. Here's more on how you can get Google Play Service Account Key.
Configure Stores in Qonversion
Once you have all the necessary data from Adapty, follow the steps to fill in the required Stores in Qonversion
Products, Entitlements and Remote Configs
After you've filled the store data, it's time to set up Remote Configs, Products, and Entitlements. Here's another way to describe these elements:
Adapty | Qonversion |
---|---|
Access levels | Entitlements |
Products | Products |
Paywalls | Offerings |
Entitlements (Access levels)
Let's start with Entitlements. Similar to how you set up Access levels in Adapty, you need to set up Entitlements in Qonversion.
Hereβs how it was done in Adapty:
Hereβs how to do it in Qonversion:
Navigate to the Entitlements and click the Create Entitlement button.
Products
Next, you need to create Products similar to how you created in Adapty:
To create Products in Qonversion, go to Entitlements β Products and click Create Product.
-
Qonversion Product ID is your unique product identifier in Qonversion that corresponds to a unique product on the Apple App Store and Google Play Store. Qonversion SDK will use it to make purchases.
-
AppStore Product ID β product identifier on Apple App Store. Learn more on how to create subscriptions on iOS.
-
Google Play Product ID β product identifier on Google Play Console. Learn more on how to set up Android in-app products .
-
Google Play Base Plan ID - identifier of the base plan for Google Play Product. This is used for subscription products. If you're creating an in-app, leave this field empty for in-apps. In Qonversion, you will set up different products for the different base plans of the same product.
-
Associated Entitlements - choose the entitlements that should be granted once this product is purchased.
Offerings
Offerings are designed as a remote configuration to present particular products on specific paywalls. Moreover, you can change the products without additional app releases.
For example, you can offer one set of products on a paywall immediately after an onboarding flow is completed and another with discounts if a user has not converted later..
Server-to-server notifications
To seamlessly implement server-to-server notifications, use Adapty S2S URLs as a temporary proxy in our system. This allows Qonversion to process new requests while still sending notifications to your existing users.
Configure Apple App Store
- Navigate to your Qonversion project settings. Copy the server-to-server notification URL for Apple App Store.
- Sign in to App Store Connect and select your app.
- Navigate to the App Information section.
- Copy the Adapty URL you used for server-to-server notifications and replace it with the URL from step 1.
- Set up the Adapty URL as a Proxy URL in Qonversion in project settings.
See more information here.
Google Developer Notifications
-
-
Navigate to project settings, then click the "Connect to Google" button.
-
You need to copy generated Google Developer Notifications Topic ID.
- Open the Google Play Console for your application, then navigate to Monetization setup.
- In the Real-time developer notifications tab, insert your topic ID, click on Save changes, and then the Send Test Notification button.
It takes up to 24 hours for your Google service credentials to work correctly. Make sure to wait for at least 24 hours after you have created the credentials before connecting to Google Real-Time Developer Notifications.
See more information here.
Install SDKs
Check out the links below for the SDK you need and follow the steps provided:
β iOS SDK
β Android SDK
β Flutter SDK
β React Native SDK
β Unity SDK
β Cordova Plugin
β Capacitor Plugin
β Web
β Kids Mode Qonversion SDKs
Read more about Installing the SDKs.
When you're done with the setup, it's time to get into the code!
Launch SDK
Firstly, you need to initialize the SDK. Here's how you did it in Adapty:
func application(_ application: UIApplication, didFinishLaunchingWithOptions launchOptions: [UIApplication.LaunchOptionsKey: Any]?) -> Bool {
let configurationBuilder =
Adapty.Configuration
.Builder(withAPIKey: "PUBLIC_SDK_KEY")
.with(observerMode: false)
Adapty.activate(with: configurationBuilder) { error in
// handle the error
}
}
public class App extends Application {
@Override
public void onCreate() {
super.onCreate()
Adapty.activate(
applicationContext,
new AdaptyConfig.Builder("PUBLIC_SDK_KEY")
.withObserverMode(false)
.build()
);
}
}
class App : Application() {
override fun onCreate() {
super.onCreate()
Adapty.activate(
applicationContext,
AdaptyConfig.Builder("PUBLIC_SDK_KEY")
.withObserverMode(false)
.build()
}
}
// Few different steps with iOS Info.plist and AndroidManifest.xml files.
// That can be removed while migrating to Qonversion.
// And then call:
try {
Adapty().activate();
} on AdaptyError catch (adaptyError) {}
} catch (e) {}
adapty.activate('PUBLIC_SDK_KEY', {
observerMode: false
});
// Few different steps with iOS Info.plist and AndroidManifest.xml files.
// Create a script which will be responsible for listening of Adapty events.
// Few other steps
Here's how you'll do it for Qonversion.
import Qonversion
func application(_ application: UIApplication, didFinishLaunchingWithOptions launchOptions: [UIApplicationLaunchOptionsKey: Any]?) -> Bool {
let config = Qonversion.Configuration(projectKey: "projectKey", launchMode: .subscriptionManagement)
Qonversion.initWithConfig(config)
return true
}
#import "Qonversion.h"
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
QONConfiguration *configuration = [[QONConfiguration alloc] initWithProjectKey:@"projectKey" launchMode:QONLaunchModeSubscriptionManagement];
[Qonversion initWithConfig:configuration];
return YES;
}
import com.qonversion.android.sdk.Qonversion;
import com.qonversion.android.sdk.QonversionConfig;
import com.qonversion.android.sdk.dto.QLaunchMode;
public class App extends Application {
@Override
public void onCreate() {
super.onCreate();
final QonversionConfig qonversionConfig = new QonversionConfig.Builder(
this,
"projectKey",
QLaunchMode.SubscriptionManagement
).build();
Qonversion.initialize(qonversionConfig);
}
}
import com.qonversion.android.sdk.Qonversion
import com.qonversion.android.sdk.QonversionConfig
import com.qonversion.android.sdk.dto.QLaunchMode
public class App : Application {
override fun onCreate() {
super.onCreate()
val qonversionConfig = QonversionConfig.Builder(
this,
"projectKey",
QLaunchMode.SubscriptionManagement
).build()
Qonversion.initialize(qonversionConfig)
}
}
import 'package:qonversion_flutter/qonversion_flutter.dart';
final config = new QonversionConfigBuilder(
'projectKey',
QLaunchMode.subscriptionManagement
).build();
Qonversion.initialize(config);
import Qonversion, {
QonversionConfigBuilder,
LaunchMode,
} from 'react-native-qonversion';
const config = new QonversionConfigBuilder(
'projectKey',
LaunchMode.SUBSCRIPTION_MANAGEMENT
).build();
Qonversion.initialize(config);
using QonversionUnity;
private void Start()
{
QonversionConfig config = new QonversionConfigBuilder(
"projectKey",
LaunchMode.SubscriptionManagement
).Build();
Qonversion.Initialize(config);
}
const config = new Qonversion.ConfigBuilder(
'projectKey',
Qonversion.LaunchMode.SUBSCRIPTION_MANAGEMENT,
).build();
Qonversion.initialize(config);
import Qonversion, {
QonversionConfigBuilder,
LaunchMode,
} from '@qonversion/capacitor-plugin';
const config = new QonversionConfigBuilder(
'projectKey',
LaunchMode.SUBSCRIPTION_MANAGEMENT
).build();
Qonversion.initialize(config);
Displaying products
In Adapty you did it using paywalls.
Adapty.getPaywallProducts(paywall: paywall) { result in
switch result {
case let .success(products):
// the requested products array
case let .failure(error):
// handle the error
}
}
Adapty.getPaywallProducts(paywall) { result ->
when (result) {
is AdaptyResult.Success -> {
val products = result.value
// the requested products
}
is AdaptyResult.Error -> {
val error = result.error
// handle the error
}
}
}
Adapty.getPaywallProducts(paywall, result -> {
if (result instanceof AdaptyResult.Success) {
List<AdaptyPaywallProduct> products = ((AdaptyResult.Success<List<AdaptyPaywallProduct>>) result).getValue();
// the requested products
} else if (result instanceof AdaptyResult.Error) {
AdaptyError error = ((AdaptyResult.Error) result).getError();
// handle the error
}
});
try {
final products = await Adapty().getPaywallProducts(paywall: paywall);
// the requested products array
} on AdaptyError catch (adaptyError) {
// handle the error
} catch (e) {
}
Adapty.GetPaywallProducts(paywall, (products, error) => {
if(error != null) {
// handle the error
return;
}
// products - the requested products array
});
try {
// ...paywall
const products = await adapty.getPaywallProducts(paywall);
// the requested products list
} catch (error) {
// handle the error
}
In Qonversion you can simply fetch products and display them using offerings:
Qonversion.shared().offerings { (offerings, error) in
if let products = offerings?.main?.products {
// Display products for sale
}
}
[[Qonversion sharedInstance] offerings:^(QONOfferings * _Nullable offerings, NSError * _Nullable error) {
if (offerings.main.products.count > 0) {
// Display products for sale
}
}];
Qonversion.getSharedInstance().offerings(new QonversionOfferingsCallback() {
@Override
public void onSuccess(@NotNull QOfferings offerings) {
if (offerings.getMain() != null && !offerings.getMain().getProducts().isEmpty()) {
// Display products for sale
}
}
@Override
public void onError(@NotNull QonversionError error) {
// handle error here
}
});
Qonversion.shared.offerings(object: QonversionOfferingsCallback {
override fun onSuccess(offerings: QOfferings) {
val mainOffering = offerings.main
if (mainOffering != null && mainOffering.products.isNotEmpty()) {
// Display products for sale
}
}
override fun onError(error: QonversionError) {
// handle error here
}
})
try {
final QOfferings offerings = await Qonversion.getSharedInstance().offerings();
final List<QProduct> products = offerings.main.products;
if (products.isNotEmpty) {
// Display your products
}
} catch (e) {
print(e);
}
try {
const offerings = await Qonversion.getSharedInstance().offerings();
if (offerings.main != null && offerings.main.products.length > 0) {
// Display products for sale
}
} catch (e) {
// handle error here
}
Qonversion.GetSharedInstance().Offerings((offerings, error) =>
{
if (error == null)
{
Offering mainOffering = offerings.Main;
if(mainOffering != null)
{
List<Product> products = mainOffering.Products;
if (products.Any())
{
// Display your products
}
}
}
else
{
// Handle the error
Debug.Log("Error" + error.ToString());
}
});
try {
const offerings = await Qonversion.getSharedInstance().offerings();
if (offerings.main != null && offerings.main.products.length > 0) {
// Display products for sale
}
} catch (e) {
// handle error here
}
try {
const offerings = await Qonversion.getSharedInstance().offerings();
if (offerings.main != null && offerings.main.products.length > 0) {
// Display products for sale
}
} catch (e) {
// handle error here
}
Making purchases
Adapty:
Adapty.makePurchase(product: product) { result in
switch result {
case let .success(info):
if info.profile.accessLevels["YOUR_ACCESS_LEVEL"]?.isActive ?? false {
// successful purchase
}
case let .failure(error):
// handle the error
}
}
Adapty.makePurchase(activity, product) { result ->
when (result) {
is AdaptyResult.Success -> {
val info = result.value
//NOTE: info is null in case of cross-grade with DEFERRED proration mode
val profile = info?.profile
if (profile?.accessLevels?.get("YOUR_ACCESS_LEVEL")?.isActive == true) {
// grant access to premium features
}
}
is AdaptyResult.Error -> {
val error = result.error
// handle the error
}
}
}
Adapty.makePurchase(activity, product, result -> {
if (result instanceof AdaptyResult.Success) {
AdaptyPurchasedInfo info = ((AdaptyResult.Success<AdaptyPurchasedInfo>) result).getValue();
//NOTE: info is null in case of cross-grade with DEFERRED proration mode
AdaptyProfile profile = info != null ? info.getProfile() : null;
if (profile != null) {
AdaptyProfile.AccessLevel premium = profile.getAccessLevels().get("YOUR_ACCESS_LEVEL");
if (premium != null && premium.isActive()) {
// successful purchase
}
}
} else if (result instanceof AdaptyResult.Error) {
AdaptyError error = ((AdaptyResult.Error) result).getError();
// handle the error
}
});
try {
final profile = await Adapty().makePurchase(product: product);
if (profile?.accessLevels['YOUR_ACCESS_LEVEL']?.isActive ?? false) {
// successful purchase
}
} on AdaptyError catch (adaptyError) {
// handle the error
} catch (e) {
}
Adapty.MakePurchase(product, (profile, error) => {
if(error != null) {
// handle error
return;
}
var accessLevel = profile.AccessLevels["YOUR_ACCESS_LEVEL"];
if (accessLevel != null && accessLevel.IsActive) {
// grant access to features
}
});
try {
const profile = await adapty.makePurchase(product);
const isSubscribed = profile?.accessLevels['YOUR_ACCESS_LEVEL']?.isActive;
if (isSubscribed) {
// grant access to features in accordance with access level
}
} catch (error) {
// handle the error
}
Qonversion:
Qonversion.shared().purchaseProduct(product) { (entitlements, error, isCancelled) in
if let premium: Qonversion.Entitlement = entitlements["premium"], premium.isActive {
// Flow for success state
}
}
[[Qonversion sharedInstance] purchaseProduct:product completion:^(NSDictionary<NSString *,QONEntitlement *> * _Nonnull entitlements,
NSError * _Nullable error,
BOOL cancelled) {
QONEntitlement *premiumEntitlement = entitlements[@"premium"];
if (premiumEntitlement && premiumEntitlement.isActive) {
// Flow for success state
}
}];
Qonversion.getSharedInstance().purchase(this, product, new QonversionEntitlementsCallback() {
@Override
public void onSuccess(@NotNull Map<String, QEntitlement> entitlements) {
QEntitlement premiumEntitlement = entitlements.get("premium");
if (premiumEntitlement != null && premiumEntitlement.isActive()) {
// Handle active entitlement here
}
}
@Override
public void onError(@NotNull QonversionError error) {
// Handle error here
if (error.getCode() == QonversionErrorCode.CanceledPurchase) {
// Purchase canceled by the user
}
}
});
Qonversion.shared.purchase(this, product, callback = object: QonversionEntitlementsCallback {
override fun onSuccess(entitlements: Map<String, QEntitlement>) {
val premiumEntitlement = entitlements["premium"]
if (premiumEntitlement != null && premiumEntitlement.isActive) {
// Handle active entitlement here
}
}
override fun onError(error: QonversionError) {
// Handle error here
if (error.code === QonversionErrorCode.CanceledPurchase) {
// Purchase canceled by the user
}
}
})
try {
final Map<String, QEntitlement> entitlements = await Qonversion.getSharedInstance().purchaseProduct(product);
} on QPurchaseException catch (e) {
if (e.isUserCancelled) {
// Purchase canceled by the user
}
print(e);
}
try {
const entitlements: Map<string, Entitlement> = await Qonversion.getSharedInstance().purchaseProduct(product);
} catch (e) {
if (e.userCanceled) {
// Purchase canceled by the user
}
console.log(e);
}
Qonversion.GetSharedInstance().PurchaseProduct(product, (entitlements, error, isCancelled) =>
{
if (error == null)
{
if (entitlements.TryGetValue("premium", out Entitlement premium) && premium.IsActive)
{
// Handle the active entitlement here
}
}
else
{
// Handle the error
Debug.Log("Error" + error.ToString());
}
});
try {
const entitlements = await Qonversion.getSharedInstance().purchaseProduct(product);
} catch (e) {
if (e.userCanceled) {
// Purchase canceled by the user
}
console.log(e);
}
try {
const entitlements: Map<string, Entitlement> = await Qonversion.getSharedInstance().purchaseProduct(product);
} catch (e) {
if (e.userCanceled) {
// Purchase canceled by the user
}
console.log(e);
}
Restore purchases
If your app has a restore purchases feature on paywalls, just use the Qonversion restore()
function when users want to restore their purchases.
Qonversion.shared().restore { [weak self] (entitlements, error) in
if let error = error {
// Handle error
}
if let entitlement: Qonversion.Entitlement = entitlements["plus"], entitlement.isActive {
// Restored and entitlement is active
}
[[Qonversion sharedInstance] restore:^(NSDictionary<NSString *, QONEntitlement *> * _Nonnull result, NSError * _Nullable error) {
if (error) {
// Handle error
}
QONEntitlement *entitlement = result[@"active"];
if (entitlement && entitlement.isActive) {
// Restored and entitlement is active
}
}];
Qonversion.getSharedInstance().restore(new QonversionEntitlementsCallback() {
@Override
public void onSuccess(@NotNull Map<String, QEntitlement> entitlements) {
QEntitlement premiumEntitlement = entitlements.get("premium");
if (premiumEntitlement != null && premiumEntitlement.isActive()) {
// handle active entitlement here
}
}
@Override
public void onError(@NotNull QonversionError error) {
// handle error here
}
});
Qonversion.shared.restore(object : QonversionEntitlementsCallback {
override fun onSuccess(entitlements: Map<String, QEntitlement>) {
val premiumEntitlement = entitlements["premium"]
if (premiumEntitlement != null && premiumEntitlement.isActive) {
// handle active entitlement here
}
}
override fun onError(error: QonversionError) {
// handle error here
}
})
try {
final Map<String, QEntitlement> entitlements = await Qonversion.getSharedInstance().restore();
} catch (e) {
print(e);
}
try {
const entitlements: Map<string, Entitlement> = await Qonversion.getSharedInstance().restore();
} catch (e) {
console.log(e);
}
Qonversion.GetSharedInstance().Restore((entitlements, error) =>
{
if (error == null)
{
// Handle entitlements here
}
else
{
// Handle the error
Debug.Log("Error" + error.ToString());
}
});
try {
const entitlements = await Qonversion.getSharedInstance().restore();
} catch (e) {
console.log(e);
}
try {
const entitlements: Map<string, Entitlement> = await Qonversion.getSharedInstance().restore();
} catch (e) {
console.log(e);
}
User status
To check the customer status in Qonversion, you need to use Entitlements. In Adapty, you used profile and access levels.
Adapty:
Adapty.getProfile { result in
if let profile = try? result.get() {
// check the access
profile.accessLevels["YOUR_ACCESS_LEVEL"]?.isActive ?? false {
// grant access to premium features
}
}
}
Adapty.getProfile { result ->
when (result) {
is AdaptyResult.Success -> {
val profile = result.value
// check the access
}
is AdaptyResult.Error -> {
val error = result.error
// handle the error
}
}
}
Adapty.getProfile(result -> {
if (result instanceof AdaptyResult.Success) {
AdaptyProfile profile = ((AdaptyResult.Success<AdaptyProfile>) result).getValue();
// check the access
} else if (result instanceof AdaptyResult.Error) {
AdaptyError error = ((AdaptyResult.Error) result).getError();
// handle the error
}
});
try {
final profile = await Adapty().getProfile();
// check the access
} on AdaptyError catch (adaptyError) {
// handle the error
} catch (e) {
}
Adapty.GetProfile((profile, error) => {
if (error != null) {
// handle the error
return;
}
// check the access
});
try {
const profile = await adapty.getProfile();
} catch (error) {
// handle the error
}
Qonversion:
Qonversion.shared().checkEntitlements { (entitlements, error) in
if let error = error {
// handle error
return
}
if let premium: Qonversion.Entitlement = entitlements["premium"], premium.isActive {
// unlock feature
}
}
[[Qonversion sharedInstance] checkEntitlements:^(NSDictionary<NSString *, QONEntitlement *> * _Nonnull entitlements,
NSError * _Nullable error) {
QONEntitlement *premiumEntitlement = entitlements[@"premium"];
if (premiumEntitlement && premiumEntitlement.isActive) {
// unlock feature
}
}];
Qonversion.getSharedInstance().checkEntitlements(new QonversionEntitlementsCallback() {
@Override
public void onSuccess(@NotNull Map<String, QEntitlement> entitlements) {
final QEntitlement premiumEntitlement = entitlements.get("premium");
if (premiumEntitlement != null && premiumEntitlement.isActive()) {
// unlock feature
}
}
@Override
public void onError(@NotNull QonversionError error) {
// handle error here
}
});
Qonversion.shared.checkEntitlements(object: QonversionEntitlementsCallback {
override fun onSuccess(entitlements: Map<String, QEntitlement>) {
val premiumEntitlement = entitlements["premium"]
if (premiumEntitlement != null && premiumEntitlement.isActive) {
// unlock feature
}
}
override fun onError(error: QonversionError) {
// handle error here
}
})
try {
final Map<String, QEntitlement> entitlements = await Qonversion.getSharedInstance().checkEntitlements();
final premium = entitlements['premium'];
if (premium != null && premium.isActive) {
// unlock feature
}
} catch (e) {
print(e);
}
try {
const entitlements = await Qonversion.getSharedInstance().checkEntitlements();
const premiumEntitlement = entitlements.get('premium');
if (premiumEntitlement != null) {
// unlock feature
}
} catch (e) {
// handle error here
}
Qonversion.GetSharedInstance().CheckEntitlements((entitlements, error) =>
{
if (error == null)
{
if (entitlements.TryGetValue("premium", out Entitlement premium) && premium.IsActive)
{
// unlock feature
}
}
else
{
// Handle the error
Debug.Log("Error" + error.ToString());
}
});
try {
const entitlements = await Qonversion.getSharedInstance().checkEntitlements();
const premiumEntitlement = entitlements.get('premium');
if (premiumEntitlement != null) {
// unlock feature
}
} catch (e) {
// handle error here
}
try {
const entitlements = await Qonversion.getSharedInstance().checkEntitlements();
const premiumEntitlement = entitlements.get('premium');
if (premiumEntitlement != null) {
// unlock feature
}
} catch (e) {
// handle error here
}
Authorization system
Cross-device and cross-platform access
Set up this section only if you need cross-device and cross-platform access and your project includes an authorization system.
Qonversion lets you identify your signed-in users and unlock premium access across multiple devices. Use the identify()
method to link a user to your signed-in subscriber. Call this method every time you want to use User Identity. For example, when a user logs in.
User Identity provides a convenient way of managing premium access of your existing subscribers, including the following cases:
- A user reinstalls your app for any reason. Using the same User ID allows you to provide premium access linked to previously purchased products.
- A user logs in on several devices. You can provide premium access based on a subscription purchased on one of his devices.
- A user logs in on iOS and Android versions of your app. You can provide premium access based on a subscription purchased on one of the platforms.
Logging in
When a user logs into his account, call identify()
.
Qonversion.shared().identify("your_custom_user_id")
// or the following option, if you want to get notified about the result.
Qonversion.shared().identify("your_custom_user_id") { user, error in
// use user if necessary
}
[[Qonversion sharedInstance] identify:@"your_custom_user_id"];
// or the following option, if you want to get notified about the result.
[[Qonversion sharedInstance] identify:@"your_custom_user_id" completion:^(QONUser * _Nullable user, NSError * _Nullable error) {
// use user if necessary
}];
Qonversion.getSharedInstance().identify("your_custom_user_id");
// or the following option, if you want to get notified about the result.
Qonversion.getSharedInstance().identify("your_custom_user_id", new QonversionUserCallback() {
@Override
public void onSuccess(@NonNull QUser user) {
// use user if necessary
}
@Override
public void onError(@NonNull QonversionError error) {
// handle error here
}
});
Qonversion.shared.identify("your_custom_user_id")
// or the following option, if you want to get notified about the result.
Qonversion.shared.identify("your_custom_user_id", object : QonversionUserCallback {
override fun onSuccess(user: QUser) {
// use user if necessary
}
override fun onError(error: QonversionError) {
// handle error here
}
})
try {
final userInfo = await Qonversion.getSharedInstance().identify("your_custom_user_id");
// use userInfo if necessary
} catch (e) {
// handle error here
}
try {
const userInfo = await Qonversion.getSharedInstance().identify('your_custom_user_id');
// use userInfo if necessary
} catch (e) {
// handle error here
}
Qonversion.GetSharedInstance().Identify("your_custom_user_id")
// or the following option, if you want to get notified about the result.
Qonversion.GetSharedInstance().Identify("your_custom_user_id", (userInfo, error) =>
{
if (error == null)
{
// use userInfo if necessary
}
else
{
// Handle the error
Debug.Log("Error" + error.ToString());
}
});
try {
const userInfo = await Qonversion.getSharedInstance().identify('your_custom_user_id');
// use userInfo if necessary
} catch (e) {
// handle error here
}
try {
const userInfo = await Qonversion.getSharedInstance().identify('your_custom_user_id');
// use userInfo if necessary
} catch (e) {
// handle error here
}
curl --location --request POST 'https://api.qonversion.io/v3/identities/new_identities_user' \
--header 'Authorization: Bearer ' \
--header 'Content-Type: application/json' \
--data-raw '{
"user_id": "QON_38a2d811afd54a433587620f8696266e"
}'
{
"id": "new_identities_user",
"user_id": "QON_38a2d811afd54a433587620f8696266e"
}
Use the unique user ID stored in your database
Always use unique user ID values. Otherwise, a user can get matched to another user's entitlements status.
To check entitlements for identified users, you can call the checkEntitlements
function, which we discussed above.
Logging out
You need to call the logout()
method to handle entitlements for an unauthorized user.
Call this method when a user logs out within your app:
Qonversion.shared().logout()
[[Qonversion sharedInstance] logout];
Qonversion.getSharedInstance().logout();
Qonversion.shared.logout()
Qonversion.getSharedInstance().logout();
Qonversion.getSharedInstance().logout();
Qonversion.GetSharedInstance().Logout();
Qonversion.getSharedInstance().logout();
Qonversion.getSharedInstance().logout();
When a user logs back into his account, don't forget to use identify()
method again.
User-base migration
Those steps have to be done to keep the system working smoothly for your current user base. The necessity of steps depends on the complexity of your system. The simplest system is represented by a single platform infrastructure (Apple, Google, or Stripe). The complex one includes a few platforms with authorization system, your backend infrastructure, and WebHooks handler.
Client-side migration
Follow these steps to sync user status and retrieve the latest data from Google and Apple on your current device. Π‘all the syncHistoricalData()
method right after Qonversion SDK initialization to synchronize all the device-related data and ensure that none of the entitlements has been missed.
User-database migration
Once you have the app version with Qonversion SDK up and running, it's time to proceed to the server-side migration.
- Prepare files with AppStore with Base64 encoded receipt data or Google Play with purchase tokens.
If you do not have receipt data or purchase tokens on your side but are already running your subscription management with a third-party vendor, you should ask the vendorβs support team for the required data file. - Then simply share the files with us using our support chat and provide CSV files.
Release
After finishing all the steps, you're ready to release your app. For complex systems with cross-platform access and authorization setups, make sure your Qonversion team has verified the user database migration.
You're good to go
Check out the detailed Adapty and Qonversion feature comparison and don't hesitate to contact us if you have questions.
Updated about 2 months ago