forked from openstf/stf
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathsrv.js
125 lines (104 loc) · 2.7 KB
/
srv.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
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
var url = require('url')
var util = require('util')
var Promise = require('bluebird')
var dns = Promise.promisifyAll(require('dns'))
var srv = module.exports = Object.create(null)
function groupByPriority(records) {
function sortByPriority(a, b) {
return a.priority - b.priority
}
return records.sort(sortByPriority).reduce(function(acc, record) {
if (acc.length) {
var last = acc[acc.length - 1]
if (last[0].priority !== record.priority) {
acc.push([record])
}
else {
last.push(record)
}
}
else {
acc.push([record])
}
return acc
}, [])
}
function shuffleWeighted(records) {
function sortByWeight(a, b) {
return b.weight - a.weight
}
function totalWeight(records) {
return records.reduce(function(sum, record) {
return sum + record.weight
}, 0)
}
function pick(records, sum) {
var rand = Math.random() * sum
var counter = 0
for (var i = 0, l = records.length; i < l; ++i) {
counter += records[i].weight
if (rand < counter) {
var picked = records.splice(i, 1)
return picked.concat(pick(records, sum - picked[0].weight))
}
}
return []
}
return pick(records.sort(sortByWeight), totalWeight(records))
}
function flatten(groupedRecords) {
return groupedRecords.reduce(function(acc, group) {
return acc.concat(group)
}, [])
}
function NEXT() {
Error.call(this)
this.name = 'NEXT'
Error.captureStackTrace(this, NEXT)
}
util.inherits(NEXT, Error)
srv.NEXT = NEXT
srv.sort = function(records) {
return flatten(groupByPriority(records).map(shuffleWeighted))
}
srv.resolve = function(domain) {
var parsedUrl = url.parse(domain)
if (!parsedUrl.protocol) {
return Promise.reject(new Error(util.format(
'Must include protocol in "%s"'
, domain
)))
}
if (/^srv\+/.test(parsedUrl.protocol)) {
parsedUrl.protocol = parsedUrl.protocol.substr(4)
return dns.resolveSrvAsync(parsedUrl.hostname)
.then(module.exports.sort)
.then(function(records) {
return records.map(function(record) {
parsedUrl.host = util.format('%s:%d', record.name, record.port)
parsedUrl.hostname = record.name
parsedUrl.port = record.port
record.url = url.format(parsedUrl)
return record
})
})
}
else {
return Promise.resolve([{
url: domain
, name: parsedUrl.hostname
, port: parsedUrl.port
}])
}
}
srv.attempt = function(records, fn) {
function next(i) {
if (i >= records.length) {
throw new Error('No more records left to try')
}
return fn(records[i]).catch(srv.NEXT, function() {
return next(i + 1)
})
}
return next(0)
}