Реализация удалённого открытия/закрытия замков автомобиля через мобильное приложение
Управление замками автомобиля удалённо — функция, которая на первый взгляд проще удалённого запуска двигателя, но имеет свои технические нюансы: разные типы ТБУ управляют центральным замком по-разному, задержки GSM-команды критичны когда пользователь стоит у машины, а неправильная обработка статуса «заперто/открыто» приводит к ситуации когда приложение показывает одно, а машина делает другое.
Управление через ТБУ: реле и CAN-шина
Простой способ — ТБУ с реле на провода центрального замка (сигнальный провод на закрытие/открытие). Работает на любом автомобиле, не требует CAN-адаптера. Минус: нет обратной связи о реальном состоянии замка — только «команда отправлена».
Продвинутый способ — CAN-интеграция (Teltonika FMB003 + CAN адаптер, LAWIG, Fortin EVO). ТБУ читает шину CAN и знает реальное состояние дверей, замков, капота. Обратная связь: команда закрыть → статус «двери закрыты» из CAN.
data class VehicleLockStatus(
val frontLeftDoor: DoorState,
val frontRightDoor: DoorState,
val rearLeftDoor: DoorState,
val rearRightDoor: DoorState,
val trunk: DoorState,
val centralLock: LockState,
val lastUpdated: Instant,
)
enum class DoorState { OPEN, CLOSED }
enum class LockState { LOCKED, UNLOCKED, UNKNOWN }
Команда запирания и подтверждение
Для команды запирания — упрощённое подтверждение по сравнению с запуском двигателя, но не полное отсутствие защиты:
suspend fun lockDoors(carId: Long): LockCommandResult {
val status = api.getVehicleStatus(carId)
// Предупреждение если дверь открыта — запирать с открытой дверью странно
if (status.anyDoorOpen) {
return LockCommandResult.Warning(
message = "Открыта ${status.openDoors.joinToString()}",
canProceed = true,
)
}
val command = LockCommand(
carId = carId,
action = LockAction.LOCK,
userId = currentUser.id,
timestamp = Instant.now().epochSecond,
)
return api.sendLockCommand(command)
}
UX: отклик при плохом GSM
Пользователь стоит у машины и жмёт «Закрыть» — а команда идёт через сотовую сеть и может задержаться на 5-20 секунд. Оптимистичный UI не работает здесь — нельзя показывать «Заперто» до подтверждения от ТБУ: пользователь уйдёт думая что машина заперта.
Правильная модель: кнопка → «Отправка команды...» → «Команда доставлена» → ожидание подтверждения от устройства (смена статуса замка из CAN) → «Заперто» или «Таймаут — проверьте машину».
Stream<LockCommandState> watchLockCommand(String commandId) async* {
yield LockCommandState.sending;
final delivered = await api.awaitCommandDelivery(commandId,
timeout: const Duration(seconds: 15));
if (!delivered) { yield LockCommandState.deliveryFailed; return; }
yield LockCommandState.delivered;
// Ждём изменения статуса замка
final confirmed = await vehicleStatusStream
.where((s) => s.centralLock == LockState.LOCKED)
.first
.timeout(const Duration(seconds: 30));
yield LockCommandState.confirmed;
}
Digital Key: UWB и NFC
Apple CarKey (iOS 14+) и Google Digital Car Key (Android 12+) — стандарты дистанционного открытия замков через UWB (Ultra Wideband) для passive entry и NFC для резервного режима. Требуют поддержки автопроизводителем (BMW, Hyundai, Genesis). Для кастомных приложений — CCC (Car Connectivity Consortium) Digital Key спецификация, реализация через PassKit на iOS и DigitalKeyService в Google Wallet API.
Разработка функции удалённого управления замками автомобиля с обратной связью о статусе: 3–5 недель (в составе приложения телематики). Стоимость рассчитывается индивидуально после анализа API конкретного ТБУ.







