gpt4 book ai didi

javascript - 用于递归异步/等待调用的计时器包装函数

转载 作者:行者123 更新时间:2023-11-30 06:14:22 25 4
gpt4 key购买 nike

我有一个递归异步函数 getResponse(url,attempts = 0),它轮询外部 API 以获取响应并在达到 X 次重试或服务器错误后解析或退出。然而,它的内部“时钟”基于重试次数(在允许延迟以避免速率限制之后),但我也希望能够灵活地设置基于时间的计时器,这将解析函数并结束递归。理想情况下,我希望能够将基于时间的计时器包装在我的递归异步函数周围,就像这样 timed(getResponse(url),3400)

我只设法让基于时间的和基于“重试”的计时器一起工作,方法是将两个计时器打包在一个异步函数中,并将局部变量 expired 作为退出标志并设置 Promise。两个函数的竞争条件。

async function timedgetResponse (expiry = 3500,url) {
let expired = false;
async function timeout(expiry){
await new Promise(_=> setTimeout(_,expiry));
expired = true;
return false;
};

async function getResponse(url,attempts = 0){
try {
if(expired){ return false; };
const limit = 10;
if(attempts >= limit){ok: false, e:"MAX_ATTEMPTS"};
const rawRes = await fetch(url,
{
method: 'GET',
credentials: 'include',
headers: {
'Accept': 'application/json'
}
});
if (!rawRes.ok) { throw (Error('SERVER_ERROR')); };
const res = await rawRes.json();
if(!res || res.status === 0){ throw (Error(res.request)); };
return {ok: true, res: res.request};
} catch(e){
const err = e.message;
if(err === "RESPONSE_NOT_READY"){
await new Promise(_ => setTimeout(_, 333));
attempts +=1;
return getResponse(url,attempts);
} else
if(err === "SERVER_ERROR_ON_RESOLVER"){
await new Promise(_ => setTimeout(_, 10000));
attempts +=1;
return getResponse(url,attempts);
} else {
return {ok: false, e:"MISC_ERROR"};
};

};
};

const awaited = await Promise.race([
getResponse(url),
timeout(expiry)
]);

return awaited;
};

我觉得这不是一个正确的方法,如果对 timed(getResponse(url),3400) 解决方案有任何帮助,我将不胜感激。

最佳答案

我有一个功能可能会满足您的需要。我已经根据我对您的需求的解释对其进行了更新。这个想法是您将进行轮询,直到某些事情为真,即解决或您超过最大尝试次数。它具有内置的可配置延迟。

这里的想法是你将传递一个函数来包装你的 fetch 调用,它最终会解析/拒绝。

setPolling(pollFunc, freq = 1000, maxAttempts = 3)

pollFunc = 不带参数并返回最终解决或拒绝的 promise 的函数。

freq = 以毫秒为单位运行 pollFunc 的频率

maxAttempts = 放弃前的最大尝试次数

const setPolling = async (pollFunc, freq = 1000, maxAttempts = 3, _attempts = 1) => {
const wait = (delay) => new Promise(resolve=>setTimeout(resolve, delay))
try {
return await pollFunc()
} catch (e) {
if (_attempts < maxAttempts) {
await wait(freq)
return await setPolling(pollFunc, freq, maxAttempts, ++_attempts)
}
throw (e instanceof Error) ? e : new Error((typeof e !== 'undefined') ? e : 'setPolling maxAttempts exceeded!')
}
}

async function alwaysFail() {
throw new Error(`alwaysFail, failed because that's what it does!`)
}

function passAfter(x) {
let i = 0
return async ()=> {
if (i > x) return `passAfter succeeded because i(${i}) > x(${x})`
throw new Error(`i(${i++}) < x(${x})`)
}
}

setPolling(alwaysFail)
.catch((e)=>console.error(`alwaysFail, failed!\n${e.message}\n${e.stack}`))

setPolling(passAfter(5), 500, 10)
.then((res)=>console.log(`passAfter, succeeded!\n${res}`))
.catch((e)=>console.error(`passAfter, failed!\n${e.message}\n${e.stack}`))

关于javascript - 用于递归异步/等待调用的计时器包装函数,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/57128460/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com