Найти тему
Develop

Алиса не любит ждать... Навык в Яндекс Диалогах должен ответить в течение 3 секунд!

Время ожидания ответа от навыка — 3 секунды. За это время нужно успеть подготовить ответ, либо сообщить, что надо подождать. В публикации рассмотрена облачная функция Яндекс на Node JS.

Если вы хотите научить Алису чему-то новому, то можно разработать навык в Яндекс Диалогах. Для запуска нового навыка достаточно сказать: "Алиса, запусти навык <ИМЯ НАВЫКА>".

В качестве бэкэнда можно использовать:

  • Webhook URL - свой обработчик POST-запроса в WEB API;
  • функцию в Яндекс Облаке.
Страница настройки навыка в Яндекс Диалогах
Страница настройки навыка в Яндекс Диалогах

При разработке навыков есть один нюанс - если ваш вебхук или функция не ответит в течение 3 секундАлиса сообщит пользователю, что навык не отвечает. То есть, запрос к базе данных, обход выборки, обращение к сторонним сервисам - на всё это 3 секунды. Если говорить про 1С, то запросы к базе данных более 3 секунд - обычное дело.

Ни оптимизация запросов к базе данных, ни увеличение производительности железа не могут гарантировать обработку запроса в течение 3 секунд. Сервер может быть перегружен, могут быть неполадки в сети, на скорость сторонних сервисов мы вообще повлиять не можем.

Перед программистом стоит задача - запустить обработчик и параллельно таймер обратного отсчета. Если обработчик не уложился в указанное время, то сообщить пользователю, что требуется еще время. Только пользователь может продолжить диалог. Поэтому надо попросить пользователя переспросить. К примеру, вот так: "Я подготовлю ответ через несколько секунд, только спросите меня: "Алиса, готово?"

В моём случае использовалась функция в Яндекс Облаке на Node JS. С помощью async, await, promise и callback решить проблему "3 секунд" не удалось. После продолжительных поисков был обнаружен специальный метод Promise.race(), который предназначен как раз для этих случаев. Как обычно, всё уже давно придумано)

Promis.race() позволяет одновременно запустить несколько асинхронных обработчиков и получить результат первого завершившегося. В нашем случае первый обработчик - двухсекундный таймер (секунду оставим на обработку), второй - запрос необходимой информации от стороннего сервиса. Если первым вернул результат таймер, то сообщаем, что времени не хватило. Если первым успел отработать запрос к стороннему сервису, то сообщаем обработанный ответ.

Пример кода получения курса доллара на Node JS:

const timeout = new Promise((resolve, reject) => {
setTimeout(() => {
resolve('Не успела, спросите меня позже)');
}, 2000);
});
const getInfo = new Promise(async (resolve, reject) => {
let url = 'https://www.cbr-xml-daily.ru/daily_json.js'
let res = await fetch(url)
let resJson = await res.json()
resolve('Курс доллара ' + resJson['Valute']['USD']['Value'])
});
let raceRes = await Promise.race([timeout, getInfo])