Перейти к основному содержимому

Вебхуки

Веб-хуки расширений отправляются в виде запросов POST на настроенный вами URL-адрес веб-хука.

Заголовки:

  • Content-Type: application/json
  • x-webhook-event: <event-name>
  • x-webhook-token-hash: <sha256-hash>

Форма полезной нагрузки верхнего уровня:

{
"id": "uuid",
"event": "lock.time_changed",
"eventData": {},
"timestamp": "2026-02-16T01:23:45.678Z",
"app": {
"id": "app_id",
"slug": "my-app",
"displayName": "My App"
},
"sessionId": "extension_session_id",
"extension": {
"sessionId": "extension_session_id",
"enabled": true,
"config": {}
},
"wearer": {
"id": "wearer_user_id",
"username": "wearer_name"
},
"keyholder": {
"id": "keyholder_user_id",
"username": "keyholder_name"
},
"lock": {
"id": "lock_id",
"title": "Lock title",
"releaseAt": "2026-02-20T00:00:00.000Z",
"remainingSeconds": 12345,
"isLocked": true,
"isFrozen": false
}
}

Доставка, повторные попытки и оформление заказа

  • Веб-хуки ставятся в очередь и доставляются асинхронно в фоновом режиме.
  • В системе доставки используется очередь на основе Redis с обрабатываемым списком запросов для восстановления после сбоев.
  • Обработка выполняется непрерывно (примерно раз в секунду на основном рабочем узле).
  • Время ожидания HTTP-запроса на одну попытку составляет примерно 3 секунды.

Политика повторных попыток:

  • Повторная попытка доставки происходит в следующих случаях:
  • Запрос истек по времени или произошла сетевая ошибка.
  • Конечная точка возвращает 429
  • Конечная точка возвращает любой статус 5xx
  • Повторная попытка доставки для других статусов 4xx не предпринимается.
  • Задержки возврата: 15s, 60s, 150s, 600s.

Последствия для интеграторов:

  • Доставка осуществляется как минимум один раз, а не ровно один раз. Ваша конечная точка должна быть идемпотентной.
  • Порядок событий определяется по мере возможности. Не следует предполагать строгий глобальный порядок для всех событий.
  • Возвращайте быстрый ответ 2xx и обрабатывайте свою ресурсоемкую работу асинхронно.
  • В качестве ключа дедупликации можно использовать полезную нагрузку id.

Данные события

lock.time_changed

{
"event": "lock.time_changed",
"eventData": {
"source": "keyholder",
"beforeReleaseAt": "2026-02-20T00:00:00.000Z",
"afterReleaseAt": "2026-02-20T01:00:00.000Z",
"requestedDeltaSeconds": 3600,
"appliedDeltaSeconds": 3600
}
}

Примечания:

  • requestedDeltaSeconds — это запрошенное действие.
  • Фактически, был сохранен код appliedDeltaSeconds.
  • Если значение appliedDeltaSeconds невозможно определить по меткам времени выпуска, оно может быть null.

lock.frozen / lock.unfrozen

{
"event": "lock.frozen",
"eventData": {
"source": "keyholder",
"durationSeconds": 3600,
"beforeFrozen": false,
"afterFrozen": true
}
}
{
"event": "lock.unfrozen",
"eventData": {
"source": "dice",
"durationSeconds": 900,
"beforeFrozen": true,
"afterFrozen": false
}
}

События task.*

task.assigned

{
"event": "task.assigned",
"eventData": {
"runId": "tr_1739665200000",
"taskId": "task_123",
"status": "active",
"source": "keyholder",
"overriddenRunId": null
}
}

task.completed

{
"event": "task.completed",
"eventData": {
"runId": "tr_1739665200000",
"taskId": "task_123",
"status": "completed",
"pointsAwarded": 10
}
}

task.failed

{
"event": "task.failed",
"eventData": {
"runId": "tr_1739665200000",
"taskId": "task_123",
"status": "failed",
"penaltyApplied": true,
"outcomeDescription": "1 hour added"
}
}

События hygiene.*

hygiene.started

{
"event": "hygiene.started",
"eventData": {
"startedAt": "2026-02-16T02:00:00.000Z",
"expiresAt": "2026-02-16T02:30:00.000Z",
"windowSeconds": 1800,
"penaltySeconds": 3600
}
}

hygiene.resumed

{
"event": "hygiene.resumed",
"eventData": {
"completedAt": "2026-02-16T02:20:00.000Z",
"lateMinutes": 0,
"penaltyApplied": false,
"punishmentSeconds": 0,
"penaltyMode": null,
"cardsAdjusted": 0,
"actualDeltaSeconds": null
}
}

hygiene.resumed_late

{
"event": "hygiene.resumed_late",
"eventData": {
"completedAt": "2026-02-16T02:45:00.000Z",
"lateMinutes": 15,
"penaltyApplied": true,
"punishmentSeconds": 3600,
"penaltyMode": "time",
"cardsAdjusted": 0,
"actualDeltaSeconds": 3600
}
}

verification.picture_submitted

{
"event": "verification.picture_submitted",
"eventData": {
"verificationId": "verification_id",
"date": "2026-02-16",
"submittedAt": "2026-02-16T03:10:00.000Z",
"photoUrl": "https://cdn.example.com/locks/..."
}
}

dice.rolled

{
"event": "dice.rolled",
"eventData": {
"userTotalScore": 14,
"botTotalScore": 8,
"difference": 6,
"requestedAdjustmentSeconds": -3600,
"appliedAdjustmentSeconds": -3600,
"cardGameEnabled": false
}
}

lock.archive_removal

{
"event": "lock.archive_removal",
"eventData": {
"source": "archive",
"actorRole": "wearer",
"lockId": "lock_id"
}
}

Примечания:

  • source — это delete или archive.
  • actorRole — это wearer или keyholder.