SDK Платежи in-app для Flutter (версия 10.0.0)
RuStore позволяет интегрировать платежи в мобильное приложение.
-
Если не знаете с чего начать, прочтите инструкцию в сценариях использования.
-
Если вы переходите на Pay SDK с billingClient SDK, ознакомьтесь с инструкцией по переходу. Подробности о Pay SDK можно узнать тут.
Подготовка к работе
Для подключения пакета платежей к проекту выполните следующую команду.
flutter pub add flutter_rustore_pay
Обработка deeplink
Обработка deeplink в RuStore SDK позволяет эффективно взаимодействовать со сторонними приложениями, при проведении платежей через банковские приложения (СБП, SberPay, T-Pay и др.). Это позволяет перевести пользователя на экран оплаты, а после завершения транзакции — вернуть в ваше приложение.
Для настройки работы с deeplink в вашем приложении и Pay SDK, укажите deeplinkScheme с помощью sdk_pay_scheme_value в AndroidManifest.xml файле.
При использовании deeplinks указание схемы является обязатель ным. При попытке платежа без указания схемы будет возникать ошибка.
Указание deeplinkScheme:
<activity
android:name=".sample.MainActivity">
<intent-filter>
<action android:name="android.intent.action.VIEW" />
<category android:name="android.intent.category.DEFAULT" />
<category android:name="android.intent.category.BROWSABLE" />
<data android:scheme="@string/SDK_PAY_SCHEME" /> />
</intent-filter>
<meta-data
android:name="sdk_pay_scheme_value"
android:value="@string/SDK_PAY_SCHEME" />
</activity>
Не забудьте указать строку с вашей SDK_PAY_SCHEME в файле strings.xml.
Для восстановления состояния вашего приложения при возврате с deeplink добавьте в
AndroidManifest.xml атрибут android:launchMode="singleTop".
Указание launchMode:
<activity
android:name=".YourPayActivity"
android:launchMode="singleTop
android:exported="true"
android:screenOrientation="portrait"
android:windowSoftInputMode="adjustResize">
Инициализация SDK
Перед вызовом методов библиотеки необходимо выполнить её инициализацию. Сама инициализация происходит автоматически, но для работы SDK в вашем файле AndroidManifest.xml необходимо прописать console_app_id_value.
<meta-data
android:name="console_app_id_value"
android:value="@string/CONSOLE_APPLICATION_ID" />
В strings.xml замените строку с вашим app_id.
<string name="CONSOLE_APPLICATION_ID">your_app_id</string>
CONSOLE_APPLICATION_ID — идентификатор приложения из RuStore консоли.
Где в RuStore Консоль отображаются идентификаторы приложений?
- Перейдите на вкладку Приложения и выберите нужное приложение.
- Скопируйте идентификатор из URL-адреса страницы приложения — это набор цифр между
apps/и/versions. Например, для URL-адресаhttps://console.rustore.ru/apps/123456/versionsID приложения —123456.

ApplicationId, указанный вbuild.gradle, должен совпадать сapplicationIdAPK-файла, который вы публиковали в RuStore Консоль.-
Подпись
keystoreдолжна совпадать с подписью, которой было подписано приложение, опубликованное в RuStore Консоль. Убедитесь, что используемыйbuildType(пр.debug) использует такую же подпись, что и опубликованное приложение (пр.release).
В целях безопасности, SDK устанавливает android:usesCleartextTraffic="false" по умолчанию, чтобы предотвратить передачу данных по незащищённому HTTP и защитить от атак типа "Man-in-the-Middle". Если ваше приложение требует использования HTTP, вы можете изменить этот атрибут на true, но делайте это на свой страх и риск, так как это увеличивает шанс перехвата и подмены данных. Мы рекомендуем разрешать незащищённый трафик только в исключительных случаях и для доверенных доменов, предпочитая HTTPS для всех сетевых взаимодействий.
Методы SDK
Доступные публичные интеракторы:
-
PurchaseInteractor- интерактор, который позволяет работать с платежами и имеет несколько публичных методов.getPurchase(purchaseId: PurchaseId): Task<Purchase>- позволяет получить информацию о покупке по её ID.getPurchases(productType: ProductType? = null, purchaseStatus: PurchaseStatus? = null): Task<List<Purchase>>— позволяет получить покупки пользователя. Данный метод поддерживает опциональную фильтрацию по типу товаров (потребляемые, непотребляемые товары или подписки), а также по статусу покупки (поддерживаются статусыPAID,CONFIRMEDACTIVEиPAUSED). По умолчанию фильтры отключены, и вернутся все покупки пользователя (независимо от типа товара) в статусахPAID,CONFIRMEDACTIVEиPAUSED.getPurchaseAvailability(): Task<PurchaseAvailabilityResult>- возвращает результат проверки доступности работы с платежами.purchase(params: ProductPurchaseParams, preferredPurchaseType: PreferredPurchaseType = PreferredPurchaseType.ONE_STEP): Task<ProductPurchaseResult>- позволяет совершить покупку продукта с указанием желаемого типа оплаты - одностадийной (ONE_STEP) или двухстадийной (TWO_STEP). Для данного метода оплаты на платёжной шторке доступны все способы оплаты. Если параметр не задан, по умолчанию запускается с оплатой в одну стадию.
Важно!Если указан тип оплаты
TWO_STEP, будет произведена попытка запустить двухстадийную оплату, но итоговый результат напрямую будет зависеть от того, какой способ оплаты (карта, СБП и др.) будет выбран пользователем.Обратите вн имание, что тип оплаты
TWO_STEPнедоступен:-
При выборе способа оплаты СБП.
-
При покупке подписок.
Двухстадийная оплата доступна только для определенного набора способов оплаты (на текущий момент — только для карт и SberPay). Если выбран способ оплаты, который не поддерживает холдирование, то покупка будет запущена по сценарию с одной стадией.
purchaseTwoStep(params: ProductPurchaseParams): Task<ProductPurchaseResult>- запускает сценарий гарантированной двухстадийной покупки товара. При использовании данного метода пользователю на платёжной шторке доступен ограниченный набор способов оплаты - только те, которые поддерживают двухстадийную оплату. В процессе платежа сначала осуществляется холдирование денежных средств покупателя, которые списываются только после подтверждения покупки методомconfirmTwoStepPurchase.confirmTwoStepPurchase(purchaseId: PurchaseId, developerPayload: DeveloperPayload? = null)- подтверждение покупки, совершенной по двухстадийной оплате.cancelTwoStepPurchase(purchaseId: PurchaseId)- отмена покупки, совершённой по двухстадийной оплате.
-
ProductInteractor- интера ктор, который позволяет работать с продуктами:getProducts(productsId: List<ProductId>): Task<List<Product>>- позволяет получить информацию по активным продуктам, опубликованным в RuStore консоль.
ВажноДанный метод возвращает не более 1000 продуктов и работает без авторизации и наличия установленного RuStore на устройстве пользователя.
-
UserInteractor- интерактор, который позволяет получить статус авторизации пользователяUserAuthorizationStatus. У данной модели может быть 2 состояния:Authorized- пользователь авторизован в RuStoreUnauthorized- пользователь неавторизован в RuStore.
-
IntentInteractor- интерактор, который позволяет обрабатывать intent-ы:proceedIntent(intent: Intent?)- метод для обработки deeplinks, использующихся для оплаты в приложениях банков. Вызов данного метода необходим для корректного отображения шторки оплаты при возвращении в приложение из приложения банка.
-
блок
RuStoreUtils- набор открытых методов, таких как:isRuStoreInstalled- проверки наличия приложения RuStore на устройстве пользователя.openRuStoreDownloadInstruction- открывает веб-страницу для скачивания приложения RuStore.openRuStore- запускает приложение RuStore.openRuStoreAuthorization- запускает приложение RuStore для авторизации. После успешной авторизации пользователя приложение RuStore автоматически закроется.
Типы покупок
В SDK предусмотрен базовый интерфейс Purchase, который объединяет общие поля всех типов покупок.
На его основе созданы две реализации:
- ProductPurchase: для потребляемых и непотребляемых покупок.
- SubscriptionPurchase: для подписок.
Данное разделение позволяет каждому типу покупки иметь свои уникальные свойства и поведение.
abstract class Purchase {
String get purchaseId;
String get invoiceId;
String? get orderId;
PurchaseType get purchaseType;
PurchaseStatus get status;
String get description;
String? get purchaseTime; // ISO-8601
int get price;
String get amountLabel;
String get currency;
String? get developerPayload;
bool get sandbox;
}
Получение сведений о покупке
Для получения информации о покупке, используйте методgetPurchase.
RuStorePayClient.instance.purchaseInteractor.getPurchase("PURCHASE_ID").then((purchase) {
// Для получения информации о покупке определенного типа
if (purchase is ProductPurchase) {
print('Product ID: ${purchase.productId}, Status: ${purchase.status}');
} else if (purchase is SubscriptionPurchase) {
print('Product ID: ${purchase.productId}, Status: ${purchase.status}');
}
}).catchError((error) {
// Обработка ошибок
});
Метод возвращает информацию о конкретной покупке в любом статусе. Модель покупки ука зана в разделе типы покупок
Получение списка покупок
Для получения списка покупок пользователя используйте метод getPurchases.
RuStorePayClient.instance.purchaseInteractor.getPurchases().then((purchases) {
for (final purchase in purchases) {
// Для получения покупки определенного типа
if (purchase is ProductPurchase) {
print('Product ID: ${purchase.productId}, Status: ${purchase.status}');
} else if (purchase is SubscriptionPurchase) {
print('Product ID: ${purchase.productId}, Status: ${purchase.status}');
}
}
}).catchError((error) {
// Обработка ошибок
});
Данный метод позволяет фильтровать покупки по типу товаров и статусу покупки:
Типы товаров:
- Потребляемые товары -
ProductType.CONSUMABLE_PRODUCT - Непотребляемые товары -
ProductType.NON_CONSUMABLE_PRODUCT - Подписки -
ProductType.SUBSCRIPTION
Статусы покупок:
-
Для продуктов:
PAID: Успешное холдирование средств, покупка ожидает подтверждения со стороны разработчика.CONFIRMED: Покупка подтверждена, средства списаны.
-
Для подписок:
ACTIVE: Подписка активна.PAUSED: Подписка в Hold периоде (например, из-за недостатка средств на карте), продолжаются попытки списания в соответствии с настройками тарифа подписки.
По умолчанию фильтры выключены, если значения не заданы, метод вернёт все покупки пользователя в статусах PAID, CONFIRMED, ACTIVE и PAUSED, независимо от типа товара.
RuStorePayClient.instance.purchaseInteractor.getPurchases().then((purchases) {
// Для получения покупки определенного типа
for (final purchase in purchases) {
if (purchase is ProductPurchase) {
print('Product ID: ${purchase.productId}, Status: ${purchase.status}');
} else if (purchase is SubscriptionPurchase) {
print('Product ID: ${purchase.productId}, Status: ${purchase.status}');
}
}
}).catchError((error) {
// Обработка ошибок
});