Live Activities: как мы сделали обновление без разрешения пользователя

Страницы:  1

Ответить
 

Professor Seleznov


pic
Привет! Меня зовут Сергей, я тимлид iOS-команды в Банки.ру. В разработке уже 11 лет — успел поработать и на аутсорсе, и в продуктовых финтех-компаниях. Мы в Банки.ру делаем приложение, которое помогает людям сравнить финансовые продукты от разных банков и страховых компаний и выбрать продукт с лучшими условиями.
Если вы iOS-разработчик и планируете внедрять Live Activities в своё приложение — эта статья для вас. Особенно если обновления LA инициируются событиями на сервере, а не действиями пользователя в приложении. Когда пользователь нажимает кнопку или запускает таймер — приложение само знает об этом и может обновить LA напрямую из кода. Но когда банк одобряет кредит или меняется статус заказа — это происходит на бэкенде, и только сервер знает о событии. В таких случаях без пушей не обойтись. Мы наступили на несколько граблей, нашли неочевидное решение и хотим сохранить вам пару недель отладки.
С чего всё началось
Одним прекрасным осенним днём наши продакт-оунеры пришли с задачей: сделать Live Activity, которая будет вести пользователя по этапам оформления банковского продукта. Например, это может быть оформление кредита (проверка данных → одобрение → подписание документов → выдача средств) или оформление карты (заявка → проверка → изготовление → доставка). В среднем у таких процессов 4-6 шагов, каждый из которых может занимать от нескольких секунд до нескольких минут. Каждый шаг — новое состояние на экране блокировки и в Dynamic Island.
Задача казалась понятной. Мы открыли документацию Apple и пошли разбираться.
Что говорит документация
Опыта с Live Activities у команды не было, поэтому начали с нуля. Apple описывает два способа запускать и обновлять LA:
1.Напрямую через кодactivity.request(...) и activity.update(...) вызываются прямо в приложении.
2.Через пуш-уведомления:
     — Push-to-Start token — для запуска LA без открытия приложения
     — Update token — для обновления и завершения LA
Для нашего приложения первый способ (обновление через код напрямую) сразу не подходил.
Причина в архитектуре: приложение не знает, когда наступает следующий шаг. Это решает бэкенд на основе ответа от банка или страховой компании. Мы не можем предсказать, через 10 секунд придёт одобрение или через 5 минут. У нас также нет background mode, который позволял бы приложению самостоятельно просыпаться и проверять статус — мы специально избегаем фоновой активности для экономии батареи. Пуш-уведомления решают обе проблемы: сервер сам знает о событии и сам инициирует обновление. Архитектурно это чище — один источник правды на бэкенде.
Обновление напрямую из кода было бы предпочтительнее в другом сценарии: например, если бы шаги определялись локально в приложении (таймер обратного отсчёта, прогресс загрузки файла, счётчик калорий в фитнес-приложении). В таких случаях приложение само контролирует данные и может обновлять LA без участия сервера. Но для нашего случая — когда состояние меняется на бэкенде в непредсказуемые моменты — пуши были единственным рабочим вариантом.
Мы реализовали всё строго по рекомендациям Apple: получаем Push-to-Start token, отправляем на сервер, сервер запускает LA. Получаем Update token, отправляем на сервер, сервер обновляет LA через APNs. Красиво. Чисто. По книжке.
И тут всё пошло не так.
Проблема, которую мы не ожидали
При первых двух запусках Live Activity iOS показывает системные кнопки прямо под активностью. В первый раз — «Запретить / Разрешить», во второй — «Запретить / Разрешить всегда». Только с третьего запуска эти кнопки исчезают (если пользователь дважды нажал «Разрешить»).
pic
И вот в чём ловушка: Update token генерируется и отправляется на сервер только после того, как пользователь нажмёт «Разрешить».
До этого момента — никакого токена. На практике это выглядело так: пользователь начинает оформление, видит первый шаг в Live Activity — и на этом LA замирает. Само оформление продолжается, заявка обрабатывается, но пользователь об этом не знает — следующие шаги в Live Activity не появляются. Чтобы LA начала обновляться, нужно специально выйти на Lock Screen и нажать «Разрешить».
Очевидно, что пользователь сам до этого не додумается. А отвлекать его от оформления ради разрешения на LA — плохой UX. Мы зашли в тупик.
Подсказка пришла от продакт-оунеров
И тут нас выручили те самые люди, которые принесли задачу. Продакты заметили: в Яндекс Картах Live Activity обновляется даже без нажатия на «Разрешить». Как?
Мы начали копать: гуглили, засыпали ИИ вопросами и тестировали приложение Яндекс Карт.
Оказалось, что ограничение касается только обновления через Update token. Если обновлять LA напрямую из кода — разрешение не нужно. Кнопки по-прежнему висят под активностью, но они не блокируют обновление.
Механизм понятен. Но есть проблема: данные о следующем шаге живут на бэкенде. А обновлять мы теперь хотим из кода, не через пуш.
Два варианта решения
Мы рассмотрели два подхода, взвесив плюсы и минусы:
Вариант 1. Background task (пуллинг)
Периодически будить приложение в фоне и спрашивать у сервера: «Есть ли новый шаг?». Если есть — обновить LA из кода.
  •  Потребляет батарею в фоне
  • Требует отдельного метода для пуллинга
  • Пуш-сервер должен хранить текущее состояние шагов, чтобы отдать его по запросу.
Сейчас он так не работает: просто получает триггер и сразу шлёт шаг. Значит нужна серьёзная переработка сервера.
Вариант 2. Silent push
Сервер сам присылает данные нового шага внутри silent push (content-available: 1). Приложение просыпается на секунду, читает данные из payload и обновляет LA из кода.
  • Не нагружает батарею
  • Пуш-сервер уже умеет отправлять данные о шагах — логику менять почти не нужно
  • Единственный минус: нет гарантии доставки
Наш выбор пал на Silent push.
Но остался один нетривиальный вопрос: как в обработчике silent push понять, какую именно LA нужно обновить? Их может быть несколько. Об этой хитрости – чуть позже.
Итоговая схема работы
Решение получилось элегантным:
pic
  • Если у пуш-сервера есть Update token для LA — обновление идёт через него (стандартный механизм Apple).
  • Если Update token ещё не получен (пользователь не нажал «Разрешить») — обновление идёт через silent push, приложение обновляет LA из кода.
Два механизма работают параллельно и автоматически дополняют друг друга.
Отступление: когда ИИ уверенно ошибается
Небольшая история в сторону. Пока мы разбирались с silent push, я проконсультировался с DeepSeek. Он категорически отверг эту идею. Говорил: «Забудь. Так никогда работать не будет. Используй исключительно Update token». Настаивал на этом несмотря на все мои аргументы.
Что ж. Всё работает
Как это устроено внутри
Время разобраться в деталях. Самый нетривиальный вопрос звучит так: когда приходит silent push с данными нового шага, как приложение понимает, какую именно LA нужно обновить?
Ответ прямолинейный — по activityId:
let activity = Activity<Attributes>.activities.first(where: { $0.id == activityId })
Но откуда пуш-сервер знает этот activityId? Чтобы ответить, нужно посмотреть на всю систему целиком.
В ней три участника:
  • Сервис оформления продукта (далее — сервис оформления) — знает про шаги: что и когда должно отобразиться в LA.
  • Пуш-сервер — умеет формировать payload (json с данными пуша) и отправлять запросы в APNs. Хранит таблицы с токенами и отправленными сообщениями.
  • Мобильное приложение — запускает LA и знает её activityId.
Проблема в том, что сервис оформления и мобильное приложение ничего не знают друг о друге напрямую. Они общаются только через пуш-сервер. А activityId — это идентификатор, который генерирует сама iOS на устройстве в момент запуска LA. Сервис оформления о нём ничего не знает.
Нужен общий ключ, по которому все три участника смогут договориться, что речь идёт об одном и том же LA.
Таким ключом стал uuid — идентификатор, который генерирует сервис оформления в момент запуска первого шага. Для конкретного LA этот uuid постоянный: все последующие шаги идут с тем же значением.
Вот как всё работает по шагам.
Шаг 1. Запуск LA
Сервис оформления решает показать первый шаг, генерирует uuid и отправляет его на пуш-сервер вместе с данными шага. Пуш-сервер кладёт uuid в структуру attributes — это неизменяемая часть LA, которая задаётся один раз при запуске и больше не обновляется. Затем отправляет Push-to-Start нотификацию в APNs и создаёт запись в своей таблице:
uuid → (пусто)
iOS получает Push-to-Start и создаёт LA.
Шаг 2. Приложение регистрирует LA
Даже без разрешения пользователя приложение получает доступ к только что созданной LA. Приложение читает из неё два идентификатора: activityId, сгенерированный iOS, и uuid, который пуш-сервер положил в attributes.
for await activity in Activity<Attributes>.activityUpdates {
sendToServer(activityId: activity.id, uuid: activity.attributes.uuid)
}
Приложение отправляет оба значения на пуш-сервер. Тот находит запись по uuid и дописывает activityId:
uuid → activityId
Теперь пуш-сервер знает, какой activityId соответствует этому uuid.
Если в какой-то момент пользователь нажмёт «Разрешить», сгенерируется Update token. Приложение отправит его на пуш-сервер в связке с теми же activityId и uuid. Запись пополнится:
uuid → activityId → update token
Шаг 3. Отправка следующего шага
Когда сервис оформления хочет показать второй шаг, он отправляет его на пуш-сервер с тем же uuid, что был в первом. Пуш-сервер находит запись по uuid и смотрит: есть ли update token?
  • Есть→ отправляет обновление стандартным способом через APNs с update token.
  • Нет →формирует silent push, кладёт в него activityId из записи и данные нового шага (content-state), и отправляет в APNs.
Первый вариант простой, нас интересует второй вариант.
Шаг 4. Приложение обновляет LA из кода
Приложение просыпается в фоне(даже если не было запущено) получив silent push и читает из него activityId. Находит нужную LA среди всех активных и обновляет её напрямую из кода — без какого-либо разрешения:
func updateLiveActivity(activityId: String, userInfo: [AnyHashable: Any]) {
guard let activity = Activity<Attributes>.activities.first(where: { $0.id == activityId }) else {
return
}
guard let contentState = userInfo["content-state"] as? [String: Any],
let data = try? JSONSerialization.data(withJSONObject: contentState),
let updatedContentState = try? JSONDecoder().decode(Attributes.ContentState.self, from: data) else {
return
}
Task { @MainActor in
await activity.update(using: updatedContentState, alertConfiguration: nil)
}
}
LA обновляется. Пользователь видит новый шаг.
Что получил пользователь в итоге
С точки зрения пользователя всё выглядит так:
  • Кнопки «Разрешить / Запретить» появляются — шаги всё равно обновляются. Без каких-либо действий с его стороны.
  • Если пользователь нажал «Разрешить» — приложение автоматически переходит на стандартный механизм через Update token.
pic
Пользователь не знает о том, что внутри два разных механизма доставки. Он просто видит, что всё работает.
Именно это и было целью.
Спасибо, что дочитали статью. Буду рад пообщаться в комментариях!-Источник
 
Loading...
Error