-
Notifications
You must be signed in to change notification settings - Fork 791
/
Copy pathkbucket.js
71 lines (55 loc) · 1.67 KB
/
kbucket.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
const { EventEmitter } = require('events')
const Buffer = require('safe-buffer').Buffer
const _KBucket = require('k-bucket')
const KBUCKET_SIZE = 16
const KBUCKET_CONCURRENCY = 3
class KBucket extends EventEmitter {
constructor (id) {
super()
this._peers = new Map()
this._kbucket = new _KBucket({
localNodeId: id,
numberOfNodesPerKBucket: KBUCKET_SIZE,
numberOfNodesToPing: KBUCKET_CONCURRENCY
})
this._kbucket.on('added', (peer) => {
KBucket.getKeys(peer).forEach((key) => this._peers.set(key, peer))
this.emit('added', peer)
})
this._kbucket.on('removed', (peer) => {
KBucket.getKeys(peer).forEach((key) => this._peers.delete(key, peer))
this.emit('removed', peer)
})
this._kbucket.on('ping', (...args) => this.emit('ping', ...args))
}
static getKeys (obj) {
if (Buffer.isBuffer(obj)) return [ obj.toString('hex') ]
if (typeof obj === 'string') return [ obj ]
const keys = []
if (Buffer.isBuffer(obj.id)) keys.push(obj.id.toString('hex'))
if (obj.address && obj.port) keys.push(`${obj.address}:${obj.port}`)
return keys
}
add (peer) {
const isExists = KBucket.getKeys(peer).some((key) => this._peers.has(key))
if (!isExists) this._kbucket.add(peer)
}
get (obj) {
for (let key of KBucket.getKeys(obj)) {
const peer = this._peers.get(key)
if (peer !== undefined) return peer
}
return null
}
getAll () {
return this._kbucket.toArray()
}
closest (id) {
return this._kbucket.closest(id, KBUCKET_SIZE)
}
remove (obj) {
const peer = this.get(obj)
if (peer !== null) this._kbucket.remove(peer.id)
}
}
module.exports = KBucket