Если verifyToken является asyn c, разве это не означает, что он вызывается, немедленно возвращает Promise, а затем узел продолжает свое выполнение? Как асинхронные методы работают в качестве промежуточного программного обеспечения?
Вы правы, что функция async
немедленно возвращает обещание, как только оно достигает await
внутри. Но, хотя Express не обращает никакого внимания на возвращаемое обещание, оно все равно может работать просто отлично из-за дизайна Express.
Express предполагает в своем дизайне необходимость асинхронного промежуточное программное обеспечение (используя обещания или нет). Таким образом, он вызывает ваше промежуточное ПО, передает ему объекты req
и res
и обратный вызов next
, а затем ничего не делает. Буквально ничего. Это делается с дальнейшей обработкой на этом этапе, пока вы не выполните одну из двух вещей - либо отправите ответ с чем-то вроде res.send()
, либо позвоните next()
. Если вы отправите ответ, это говорит само за себя. Если вы вызываете next()
, то это говорит Express следовать цепочке обработчиков для этого запроса, найти следующий соответствующий обработчик запроса и вызвать его.
Итак, вот шаги, когда этот маршрут соответствует:
async function verifyToken(req, res, next) {
try {
await API call...
next() // <-- continue to the next middleware if the api call was successful
} catch (err) {
// catch exceptions from api call
}
}
this.router.get("/test", verifyToken, (req, res) => {
res.send("GET TEST");
});
- nodejs первый вызов
verifyToken()
и его передача req, res, next
. - Эта функция достигает
await
, выполняет вызов API, приостанавливает дальнейшее выполнение функция, а затем немедленно возвращает ожидающее обещание - Возврат возвращается в Express, и он просто возвращается к любому сетевому событию, сообщившему ему о входящем запросе, и больше там ничего не делать.
- Затем на некоторое время ничего не происходит.
- Затем ваш вызов API завершается в какой-то момент, и ожидаемое обещание разрешается. Это возобновляет выполнение функции в строке кода после
await
. - Это вызывает
next()
, который возвращается в Express и, используя данные в замыкании для обратного вызова next()
, он знает, какой обработчик вызывать следующим, и он вызывает обработчик окончательного запроса на вашем маршруте, который вызывает res.send()
. - Этот обработчик возвращается и Express готово. Поскольку вы не вызывали
next()
из последнего обработчика, Express больше не будет выполнять работу над этим запросом.
Express был разработан некоторое время go и, таким образом, не врожденно обещает. Но он был спроектирован с учетом асинхронной работы, поэтому обещания по-прежнему могут отлично использоваться с инфраструктурой - вы просто подключаете обработчики событий обещаний к соответствующим обратным вызовам Express. Поскольку express не следит за обещанием, что ваша функция async
вернулась, обязательно убедитесь, что вы ловите все отклонения самостоятельно, потому что Express не поймает их для вас.
Существует двоюродная сестра Express по имени koa, которая более разработана для обещаний и имеет автоматическую c обработку ошибок для отклоненных обещаний.