forked from uber-node/lb_pool
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathpool_request_set.js
91 lines (75 loc) · 2.85 KB
/
pool_request_set.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
// Copyright 2013 Voxer IP LLC. All rights reserved.
var Stream = require("stream");
// PoolRequestSet - an object to track server requests and handle retries
//
// pool: a pool of endpoints
// options: {
// attempts: number of tries
// maxHangups: number of 'socket hang ups' before giving up (2)
// timeout: request timeout in ms
// maxAborts: number of 'aborted' before giving up (2)
// retryDelay: minimum ms to wait before first retry using exponential backoff (20)
// }
// callback: function (err, response, body) {}
function PoolRequestSet(pool, options, callback) {
this.options = options || {};
this.pool = pool;
this.callback = callback;
if (this.options.data instanceof Stream || this.options.end === false) {
this.max_attempts = 1;
} else {
this.max_attempts = options.max_attempts || Math.min(pool.options.max_retries + 1, Math.max(pool.length, 2));
}
this.attempts_remaining = this.max_attempts;
this.max_timeouts = options.max_timeouts || 1; // no retries on timeouts by default
this.timeouts = 0;
this.max_hangups = options.max_hangups || 2;
this.hangups = 0;
this.max_aborts = options.max_aborts || 2;
this.aborts = 0;
this.duration = null;
if (!options.retry_delay && options.retry_delay !== 0) {
options.retry_delay = 20;
}
this.delay = options.retry_delay;
}
PoolRequestSet.prototype.handle_response = function (err, response, body) {
var delay;
this.attempts_remaining--;
if (err) {
delay = Math.round(Math.random() * Math.pow(2, this.max_attempts - this.attempts_remaining) * this.delay);
err.delay = delay; // stash delay here so "retrying" listeners can understand the delay
if (err.reason === "socket hang up") {
this.hangups++;
} else if (err.reason === "aborted") {
this.aborts++;
} else if (err.reason === "timed_out") {
this.timeouts++;
}
if (this.attempts_remaining > 0 && err.reason !== "full" && err.reason !== "unhealthy" &&
this.hangups < this.max_hangups && this.aborts < this.max_aborts && this.timeouts < this.max_timeouts) {
this.pool.on_retry(err);
if (delay > 0) {
setTimeout(this.do_request.bind(this), delay);
} else {
this.do_request();
}
return;
}
}
if (this.callback) {
this.callback(err, response, body);
this.callback = null;
}
};
PoolRequestSet.prototype.do_request = function () {
var endpoint = this.pool.get_endpoint(this.options),
self = this;
return endpoint.request(this.options, function (err, res, body, duration) {
self.duration = duration;
self.handle_response(err, res, body);
});
};
module.exports = function init() {
return PoolRequestSet;
};