Abraham对Google Calendar API的回答:&;Backend error &;代码503正好描述了我的情况。当循环执行创建或删除日历条目的代码时,我在随机位置得到503。
然而,我不知道如何遵循他从谷歌引用的建议,即捕获错误并使用指数回退重试交易。
下面的代码是一个循环,将8个新事件放入我的日历。它随机经历503个错误,这些错误是从Google API而不是我自己的代码抛出的。很多时候,它的工作没有错误。
Google API代码从我的循环中异步运行,因此在我的循环完成之前,没有任何Google操作实际执行。当异步代码抛出503时,我的代码周围的try-catch
块不会触发。我不能在没有try
的情况下将catch
放入回调函数中,这将缩小catch
的范围以排除Google的代码。
有什么建议吗?
/* Special date string format for all-day Google Calendar events.
Time zone independent.
*/
Date.prototype.yyyy_mm_dd = function() {
var yyyy= this.getFullYear().toString();
var mm = (this.getMonth()+101).toString().slice(-2); //get leading 0
var dd = (this.getDate()+100).toString().slice(-2);
return yyyy+'-'+mm+'-'+dd;
}
var fastevent = {
'summary': 'Fast',
'organizer': {
'self': true,
'displayName': 'Wes Rishel',
'email': 'wrishel@gmail.com'},
'start': {'date': 'zzzz'}, // filled in for each instance
'end': {'date': 'zzzz'},
'colorId': '11',
}
function addFastEvents() {
try {
var eventDate = calendar.getLastFastDate() || new Date;
for (var eventCount = 0; eventCount < 8; eventCount++) {
// advance to next Tuesday or Friday
eventDate=eventDate.addDays(
[2, 1, 3, 2, 1, 4, 3][eventDate.getDay()]
);
fastevent.start.date = eventDate.yyyy_mm_dd();
fastevent.end.date = fastevent.start.date;
var request = gapi.client.calendar.events.insert({
'calendarId': 'primary',
'resource': fastevent
});
request.execute(function(fastevent) {});
calendar.getPage(eventDate);
calendar.setCellStyle(eventDate, 'fastingweekdaydata');
} // for
} catch(e) {
p(e.message, e.name)
}
}
指数回退是一种奇特的说法,表示在每次尝试时,在放弃请求之前,您将以指数方式增加等待时间。
实现指数回退
指数回退是网络的标准错误处理策略客户端定期重试失败请求的应用程序随着时间的推移。如果有大量的请求或繁重的网络流量导致服务器返回的错误呈指数级增长后退可能是处理这些错误的好策略
下面是一个JS演示代码,可能会给你一个想法:
console.log = consoleLog;
exponentialBackoff(sometimesFails, 10, 100, function(result) {
console.log('the result is',result);
});
// A function that keeps trying, "toTry" until it returns true or has
// tried "max" number of times. First retry has a delay of "delay".
// "callback" is called upon success.
function exponentialBackoff(toTry, max, delay, callback) {
console.log('max',max,'next delay',delay);
var result = toTry();
if (result) {
callback(result);
} else {
if (max > 0) {
setTimeout(function() {
exponentialBackoff(toTry, --max, delay * 2, callback);
}, delay);
} else {
console.log('we give up');
}
}
}
function sometimesFails() {
var percentFail = 0.8;
return Math.random() >= 0.8;
}
function consoleLog() {
var args = [].slice.apply(arguments);
document.querySelector('#result').innerHTML += 'n' + args.join(' - ');
}