Укажите хотя бы один Intent filter
: INTENT_FILTER_DRIVER_MANAGER
или INTENT_FILTER_VIRTUAL_DRIVER_MANAGER
.
Пример объявленного сервиса:
<service
android:name="ru.mycompany.drivers.MyPaySystemService"
android:enabled="true"
android:exported="true"
android:icon="@drawable/logo"
android:label="@string/service_name">
<intent-filter>
<action android:name="ru.evotor.devices.drivers.DriverManager" />
<action android:name="ru.evotor.devices.drivers.PaySystemService" />
</intent-filter>
<meta-data
android:name="vendor_name"
android:value="Ingenico" />
<meta-data
android:name="model_name"
android:value="IPP320" />
<meta-data
android:name="usb_device"
android:value="VID_1947PID_40" />
<meta-data
android:name="virtual_device"
android:value="false" />
<meta-data
android:name="settings_activity"
android:value="" />
<meta-data
android:name="device_categories"
android:value="PAYSYSTEM" />
</service>
Где:
vendor_name
– наименование производителя, которое будет отображаться при подключении устройства.
model_name
– наименование модели устройства.
INTENT_FILTER_DRIVER_MANAGER
- используется для драйверов, требующих подключения USB- оборудования. Помимо Intent filter
в meta-data
необходимо указать характеристики VendorID
и ProductID
целевого устройства:
<meta-data
android:name="usb_device"
android:value="VID_1659PID_8963" />
VendorID
и ProductID
необходимо указывать десятичными числами.Можно указать несколько устройств: VID_1659PID_8963|VID_123PID_456|VID_1659PID_8964
.
Экземпляр драйвера будет автоматически создан/удалён при подключении/отключении указанного оборудования к смарт-терминалу. При наличии нескольких подходящих драйверов, пользователю будет предложен выбор.
INTENT_FILTER_VIRTUAL_DRIVER_MANAGER
– используется для драйверов, не требующих подключения USB-оборудования (сетевое, bluetooth и др.). В meta-data
необходимо указать, что драйвер является виртуальным:
<meta-data
android:name="virtual_device"
android:value="true" />
Экземпляр такого драйвера пользователь может создать исключительно вручную через настройки оборудования. В этом случае, все работы по подключению к нужному устройству берёт на себя производитель драйвера.
Чтобы устройство было распознано как банковский терминал, используйте Intent filter
:
INTENT_FILTER_PAY_SYSTEM
Вместе с этим необходимо указать в meta-data категорию устройства:
<meta-data
android:name="device_categories"
android:value="PAYSYSTEM" />
SCALES|PRICEPRINTER|CASHDRAWER|OTHER
(весы | принтер чеков | денежный ящик | другое оборудование).Укажите в манифесте приложения у сервиса:
android:icon
– картинка устройства, которая будет отображаться пользователю при инициализации устройства;
android:label
– имя драйвера, которое будет отображаться пользователю при инициализации устройства
Пример отображения иконки и имени драйвера
Можно задать activity
настроек:
<meta-data
android:name="settings_activity"
android:value="ru.mycompany.drivers.MySettingsActivity" />
Указанная activity
должна находиться в текущем package
и будет вызвана при первом подключении устройства или при выборе устройства в меню настроек оборудования.
Версия драйвера (versionCode
и versionName
) берётся из build.gradle
:
defaultConfig {
applicationId "ru.mycompany.drivers.mypaysystem"
minSdkVersion 23
targetSdkVersion 24
versionCode 2
versionName "1.0.1"
}
В реализации метода подключения к сервису для всех action
, указанных в интент-фильтрах укажите соответствующие Binder’ы:
для INTENT_FILTER_DRIVER_MANAGER
– класс наследник ru.evotor.devices.drivers.IUsbDriverManagerService.Stub
;
для INTENT_FILTER_VIRTUAL_DRIVER_MANAGER
– класс наследник ru.evotor.devices.drivers.IVirtualDriverManagerService.Stub
;
для INTENT_FILTER_PAY_SYSTEM
– класс наследник ru.evotor.devices.drivers.IPaySystemDriverService.Stub
;
Пример:
public class MyDeviceService extends Service {
private final Map<Integer, MyDevice> instances = new HashMap<>();
private volatile AtomicInteger newDeviceIndex = new AtomicInteger(0);
@Nullable
@Override
public IBinder onBind(Intent intent) {
String action = intent.getAction();
switch (action) {
case Constants.INTENT_FILTER_DRIVER_MANAGER:
return new MyDriverManagerStub(MyDeviceService.this);
case Constants.INTENT_FILTER_PAY_SYSTEM:
return new MyScalesStub(MyDeviceService.this);
default:
return null;
}
}
public int createNewDevice(UsbDevice usbDevice) {
int currentIndex = newDeviceIndex.getAndIncrement();
instances.put(currentIndex, new MyDevice(getApplicationContext(), usbDevice));
return currentIndex;
}
public MyDevice getMyDevice(int instanceId) {
return instances.get(instanceId);
}
public void destroy(int instanceId) {
getMyDevice(instanceId).destroy();
instances.remove(instanceId);
}
}
В этом же сервисе можно определить Map
для хранения списка нескольких активных экземпляров драйверов, т.к. обращаться к нему придётся из всех указанных Stub’ов.
Если нельзя выполнить действие, используйте исключение (Exception
) одного из следующих типов:
BadParcelableException
;IllegalArgumentException
;IllegalStateException
;NullPointerException
;SecurityException
;NetworkOnMainThreadException
.Пример:
throw new IllegalStateException();
IUsbDriverManagerService.Stub
– класс для управления драйверами usb-устройств: подключение и отключение устройств происходят здесь. Требуется реализовать методы addUsbDevice
и destroy
.
import ru.evotor.devices.drivers.IUsbDriverManagerService;
public class MyDriverManagerStub extends IUsbDriverManagerService.Stub {
private MyDeviceService myDeviceService;
public MyDriverManagerStub(MyDeviceService myDeviceService) {
this.myDeviceService = myDeviceService;
}
@Override
public int addUsbDevice(UsbDevice usbDevice, String usbPortPath) throws RemoteException {
return myDeviceService.createNewDevice(usbDevice);
}
@Override
public void destroy(int instanceId) throws RemoteException {
myDeviceService.destroy(instanceId);
}
}
Метод addUsbDevice
в IUsbDriverManagerService
принимает на вход:
UsbDevice
, для которого он создан;
некоторый строковый идентификатор номера физического usb-порта для сохранения настроек оборудования и восстановления их после перезагрузки терминала. В этот момент у приложения-драйвера уже есть permission
для работы с этим устройством.
Метод addUsbDevice
возвращает номер экземпляра драйвера внутри приложения. По этому номеру будет происходить обращение к конкретному драйверу.
Метод destroy
в IUsbDriverManagerService
принимает на вход номер экземпляра драйвера. Вызов этого метода уведомляет об отключении от устройства. В этот момент у приложения-драйвера уже нет permission
для работы с этим устройством, само устройство уже может быть удалено из смарт-терминала.
IVirtualDriverManagerService.Stub
– класс для управления драйверами виртуальных устройств:
подключение и отключение устройств происходят здесь. Требуется реализовать методы addNewVirtualDevice
, recreateNewVirtualDevice
и destroy
.
import ru.evotor.devices.drivers.IVirtualDriverManagerService;
public class MyDriverManagerStub extends IVirtualDriverManagerService.Stub {
private MyDeviceService myDeviceService;
public MyDriverManagerStub(MyDeviceService myDeviceService) {
this.myDeviceService = myDeviceService;
}
@Override
public int addNewVirtualDevice() throws RemoteException {
return myDeviceService.createNewDevice(usbDevice);
}
@Override
public void recreateNewVirtualDevice(int instanceId) throws RemoteException {
myDeviceService.recreateNewVirtualDevice(instanceId);
}
@Override
public void destroy(int i) throws RemoteException {
myDeviceService.destroy(instanceId);
}
}
addNewVirtualDevice
возвращает номер экземпляра драйвера внутри приложения. По этому номеру будет происходить обращение к конкретному драйверу.
Метод recreateNewVirtualDevice
принимает на вход номер экземпляра драйвера внутри приложения.
Метод destroy
принимает на вход номер экземпляра драйвера. Вызов этого метода уведомляет приложение об отключении от устройства.
Для вновь созданного экземпляра драйвера (а виртуальные устройства могут создаваться только вручную пользователем через меню настроек оборудования) будет вызван метод addNewVirtualDevice
.
Метод recreateNewVirtualDevice
будет вызван для тех устройств, которые уже создавались пользователем ранее, но в данный момент подключения к таким драйверам нет. Например, после перезагрузки смарт-терминала, перезапуска сервиса работы с оборудованием или обновления приложения-драйвера.
Метод destroy
будет вызван для устройства, которое пользователь вручную удалил из списка оборудования.
IPaySystemDriverService.Stub
– класс для работы с конкретными экземплярами банковских терминалов.
import ru.evotor.devices.drivers.IPaySystemDriverService;
import ru.evotor.devices.drivers.paysystem.PayResult;
import ru.evotor.devices.drivers.paysystem.PayInfo;
public class MyPaySystemStub extends IPaySystemDriverService.Stub {
private MyPaySystemService paySystemService;
public MyPaySystemStub(MyPaySystemService paySystemService) {
this.paySystemService = paySystemService;
}
@Override
public PayResult payment(int instanceId, PayInfo payInfo) throws RemoteException {
return paySystemService.getPaySystem(instanceId).payment(payInfo);
}
@Override
public PayResult cancelPayment(int instanceId, PayInfo payInfo, String rrn) throws RemoteException {
return paySystemService.getPaySystem(instanceId).cancelPayment(payInfo, rrn);
}
@Override
public PayResult payback(int instanceId, PayInfo payInfo, String rrn) throws RemoteException {
return paySystemService.getPaySystem(instanceId).payback(payInfo, rrn);
}
@Override
public PayResult cancelPayback(int instanceId, PayInfo payInfo, String rrn) throws RemoteException {
return paySystemService.getPaySystem(instanceId).cancelPayback(payInfo, rrn);
}
@Override
public PayResult closeSession(int instanceId) throws RemoteException {
return paySystemService.getPaySystem(instanceId).closeSession();
}
@Override
public void openServiceMenu(int instanceId) throws RemoteException {
paySystemService.getPaySystem(instanceId).openServiceMenu();
}
@Override
public String getBankName(int instanceId) throws RemoteException {
return paySystemService.getPaySystem(instanceId).getBankName();
}
@Override
public String getTerminalNumberAsString(int instanceId) throws RemoteException {
return paySystemService.getPaySystem(instanceId).getTerminalNumberAsString();
}
@Override
public String getTerminalID(int instanceId) throws RemoteException {
return paySystemService.getPaySystem(instanceId).getTerminalID();
}
@Override
public String getMerchNumber(int instanceId) throws RemoteException {
return paySystemService.getPaySystem(instanceId).getMerchEngName();
}
@Override
public String getMerchCategoryCode(int instanceId) throws RemoteException {
return paySystemService.getPaySystem(instanceId).getMerchCategoryCode();
}
@Override
public String getMerchEngName(int instanceId) throws RemoteException {
return paySystemService.getPaySystem(instanceId).getMerchEngName();
}
@Override
public String getCashier(int instanceId) throws RemoteException {
return paySystemService.getPaySystem(instanceId).getCashier();
}
@Override
public String getServerIP(int instanceId) throws RemoteException {
return paySystemService.getPaySystem(instanceId).getServerIP();
}
@Override
public boolean isNotNeedRRN(int instanceId) throws RemoteException {
return paySystemService.getPaySystem(instanceId).isNotNeedRRN();
}
}
Все методы принимают на вход номер экземпляра драйвера.
Метод оплаты принимает на вход информацию об оплате (сумму), методы возврата и отмены дополнительно к этому принимают на вход РРН прошлой операции.
Пример для банковских терминалов, работающих через USB:
public class MyDevice implements IPaySystem {
@Override
public PayResult payment(PayInfo payInfo) {
//TODO Ваш код
}
@Override
public PayResult cancelPayment(PayInfo payInfo, String s) {
//TODO Ваш код
}
@Override
public PayResult payback(PayInfo payInfo, String s) {
//TODO Ваш код
}
@Override
public PayResult cancelPayback(PayInfo payInfo, String s) {
//TODO Ваш код
}
@Override
public PayResult closeSession() {
//TODO Ваш код
}
@Override
public void openServiceMenu() {
//TODO Ваш код
}
@Override
public String getBankName() {
//TODO Ваш код
}
@Override
public String getTerminalNumberAsString() {
//TODO Ваш код
}
@Override
public String getTerminalID() {
//TODO Ваш код
}
@Override
public String getMerchNumber() {
//TODO Ваш код
}
@Override
public String getMerchCategoryCode() {
//TODO Ваш код
}
@Override
public String getMerchEngName() {
//TODO Ваш код
}
@Override
public String getCashier() {
//TODO Ваш код
}
@Override
public String getServerIP() {
//TODO Ваш код
}
@Override
public boolean isNotNeedRRN() {
//TODO Ваш код
}
}
Для устройств других категорий реализуйте соответствующие интерфейсы.
На рынке появляются новые способы безналичной оплаты, помимо оплаты картой, такие как оплата по «СБП», оплата «Улыбкой» (по биометрии) и так далее. В некоторых случаях все эти способы оплаты реализованы в одном драйвере платёжных систем, например, драйвер Сбербанка. Так же комиссия при выполнении операции может отличаться в зависимости способа платежа. Мы дали возможность сторонним разработчикам драйверов и платежных приложений передавать в товароучётные системы через наше облако подробную детализацию по способам этих платежей.
Чтобы передать расширенную информацию о способе оплаты, заполните поле cashlessInfo
объекта PayResult:
CashlessInfo(
@NotNull CashlessMethod method,
@NotNull String description,
@NotNull String uuid
)
Где:
method (Enum)
- Способ безналичной оплаты. description (String)
- Расширенное описание способа безналичной оплаты. Например: SberPay, MtsPay, AliPay.uuid (String)
- Уникальный uuid
способа безналичной оплаты. Формат обязательно должен быть uuid4
, при создании объекта будет добавлена валидация на проверку формата uuid
.Пример чека с заполняемой в CashlessInfo
информацией приведен в статье: https://developer.evotor.ru/docs/doc_java_extended_api_data.html.
Загрузите приложение на смарт-терминал, чтобы работать с драйвером.
Перед публикацией драйвера, сотрудники Эвотора проверяют:
иконку драйвера, установленного на смарт-терминале;
На иконке должен быть указан производитель оборудования, модель или список моделей устройств, на которые распространяется драйвер, а также версия драйвера.
Если драйвер работает со всеми моделями производителя, достаточно указать название производителя.
Версия драйвера должна отличаться от предыдущей и должна совпадать с версией, указанной на странице драйвера в Эвотор.Маркете.
обмен сообщениями между драйвером и Облаком Эвотор.
Для проверки обмена сообщениями вам потребуется произвести по пять тестовых транзакций и сообщить сотрудникам Эвотора IMEI смарт-терминала и время проведения транзакций. Необходимо выполнить следующие транзакции: