Browse Source

lots of fixes, emits stablity/wakeup events

session-estimator
Mathias Buus 4 years ago
parent
commit
0fba9d92e2
  1. 100
      index.js

100
index.js

@ -8,6 +8,8 @@ const sodium = require('sodium-universal')
const { EventEmitter } = require('events') const { EventEmitter } = require('events')
const TICK_INTERVAL = 5000 const TICK_INTERVAL = 5000
const SLEEPING_INTERVAL = 3 * TICK_INTERVAL
const STABLE_TICKS = 240 // if nothing major bad happens in ~20mins we can consider this node stable (if nat is friendly)
const REFRESH_TICKS = 60 // refresh every ~5min when idle const REFRESH_TICKS = 60 // refresh every ~5min when idle
const RECENT_NODE = 20 // we've heard from a node less than 1min ago const RECENT_NODE = 20 // we've heard from a node less than 1min ago
const OLD_NODE = 360 // if an node has been around more than 30 min we consider it old... const OLD_NODE = 360 // if an node has been around more than 30 min we consider it old...
@ -28,6 +30,10 @@ class Request {
this.value = m.value this.value = m.value
} }
get update () {
return this.token !== null
}
error (code) { error (code) {
this.dht._reply(this.rpc, this.tid, this.target, code, null, false, this.from) this.dht._reply(this.rpc, this.tid, this.target, code, null, false, this.from)
} }
@ -62,7 +68,10 @@ class DHT extends EventEmitter {
this._bootstrapping = this.bootstrap() this._bootstrapping = this.bootstrap()
this._secrets = [randomBytes(32), randomBytes(32)] this._secrets = [randomBytes(32), randomBytes(32)]
this._tick = (Math.random() * 1024) | 0 // random offset it this._tick = (Math.random() * 1024) | 0 // random offset it
this._refreshTick = REFRESH_TICKS this._rotateSecrets = false
this._lastTick = Date.now()
this._refreshTick = this._tick + REFRESH_TICKS
this._stableTick = this._tick + STABLE_TICKS
this._tickInterval = setInterval(this._ontick.bind(this), TICK_INTERVAL) this._tickInterval = setInterval(this._ontick.bind(this), TICK_INTERVAL)
this.table.on('row', (row) => row.on('full', (node) => this._onfullrow(node, row))) this.table.on('row', (row) => row.on('full', (node) => this._onfullrow(node, row)))
@ -172,32 +181,6 @@ class DHT extends EventEmitter {
this._backgroundQuery(node ? node.id : this.table.id, 'find_node', null) this._backgroundQuery(node ? node.id : this.table.id, 'find_node', null)
} }
_tally (onlyIp) {
const sum = new Map()
let result = null
let node = this.nodes.latest
let cnt = 0
let good = 0
for (; node && cnt < 10; node = node.prev) {
const to = node.to.host + ':' + (onlyIp ? 0 : node.to.port)
const hits = 1 + (sum.get(to) || 0)
if (hits > good) {
good = hits
result = node.to
}
sum.set(to, hits)
cnt++
}
// We want at least 3 samples all with the same ip:port from
// different remotes (the to field) to be consider it consistent
// If we get >=3 samples with conflicting info we are not (or under attack) (Subject for tweaking)
const bad = cnt - good
return bad < 3 && good >= 3 ? result : null
}
_pingSome () { _pingSome () {
let cnt = this.rpc.inflightRequests > 2 ? 3 : 5 let cnt = this.rpc.inflightRequests > 2 ? 3 : 5
let oldest = this.nodes.oldest let oldest = this.nodes.oldest
@ -221,26 +204,55 @@ class DHT extends EventEmitter {
} }
_check (node) { _check (node) {
this.ping(node).catch(() => this._removeNode(node)) this.ping(node)
.then(
m => this._maybeRemoveNode(node, m.nodeId),
() => this._removeNode(node)
)
} }
_token (peer, i) { _token (peer, i) {
this._rotateSecrets = true
const out = Buffer.allocUnsafe(32) const out = Buffer.allocUnsafe(32)
sodium.crypto_generichash(out, Buffer.from(peer.host), this._secrets[i]) sodium.crypto_generichash(out, Buffer.from(peer.host), this._secrets[i])
return out return out
} }
_ontick () { _ontick () {
// rotate secrets if (this._rotateSecrets) {
const tmp = this._secrets[0] const tmp = this._secrets[0]
this._secrets[0] = this._secrets[1] this._secrets[0] = this._secrets[1]
this._secrets[1] = tmp this._secrets[1] = tmp
sodium.randombytes_buf(tmp) sodium.randombytes_buf(tmp)
}
const time = Date.now()
if (time - this._lastTick > SLEEPING_INTERVAL) {
this._stableTick = 0 // never stable
this._tick += 2 * OLD_NODE // bump the tick enough that everything appears old.
this._tick += 8 - (this._tick & 7) - 2 // triggers a series of pings in two ticks
this._refreshTick = this._tick + 1 // triggers a refresh next tick (allow network time to wake up also)
this.emit('wakeup')
} else {
this._tick++
}
this._lastTick = time
if (!this.bootstrapped) return if (!this.bootstrapped) return
this._tick++
if ((this._tick & 7) === 0) this._pingSome() if (this._tick === this._stableTick) {
if (((this._tick & 63) === 0 && this.nodes.length < 20) || this._tick === this._refreshTick) this.refresh() this.emit('stable')
}
if ((this._tick & 7) === 0) {
this._pingSome()
}
if (((this._tick & 63) === 0 && this.nodes.length < this.table.k) || this._tick >= this._refreshTick) {
this.refresh()
}
} }
_onfullrow (newNode, row) { _onfullrow (newNode, row) {
@ -271,7 +283,8 @@ class DHT extends EventEmitter {
this._repinging++ this._repinging++
this.ping(oldNode).then(onsuccess, onswap) this.ping(oldNode).then(onsuccess, onswap)
function onsuccess () { function onsuccess (m) {
if (m.nodeId === null || !m.nodeId.equals(oldNode.id)) return onswap()
self._repinging-- self._repinging--
self._repingMaybe() self._repingMaybe()
} }
@ -299,21 +312,25 @@ class DHT extends EventEmitter {
} }
_addNode (node) { _addNode (node) {
if (this.nodes.has(node)) return if (this.nodes.has(node) || node.id.equals(this.table.id)) return
node.added = node.seen = this._tick node.added = node.seen = this._tick
this.nodes.add(node) if (this.table.add(node)) this.nodes.add(node)
this.table.add(node)
this.emit('add-node', node) this.emit('add-node', node)
} }
_maybeRemoveNode (node, expectedId) {
if (expectedId !== null && expectedId.equals(node.id)) return
this._removeNode(node)
}
_removeNode (node) { _removeNode (node) {
if (!this.nodes.has(node)) return if (!this.nodes.has(node)) return
this.nodes.remove(node)
this.table.remove(node.id) this.table.remove(node.id)
this.nodes.remove(node)
this.emit('remove-node', node) this.emit('remove-node', node)
} }
@ -359,6 +376,7 @@ class DHT extends EventEmitter {
return return
} }
// empty dht reply back
if (req.command === 'find_node') { if (req.command === 'find_node') {
this._reply(this.rpc, req.tid, req.target, 0, null, false, req.from) this._reply(this.rpc, req.tid, req.target, 0, null, false, req.from)
return return

Loading…
Cancel
Save