mirror of
https://github.com/fluencelabs/go-libp2p-kad-dht
synced 2025-04-25 06:42:13 +00:00
83 lines
1.4 KiB
Go
83 lines
1.4 KiB
Go
package dht
|
|
|
|
import (
|
|
"bytes"
|
|
"crypto/sha256"
|
|
|
|
peer "github.com/jbenet/go-ipfs/peer"
|
|
u "github.com/jbenet/go-ipfs/util"
|
|
)
|
|
|
|
// ID for IpfsDHT should be a byte slice, to allow for simpler operations
|
|
// (xor). DHT ids are based on the peer.IDs.
|
|
//
|
|
// The type dht.ID signifies that its contents have been hashed from either a
|
|
// peer.ID or a util.Key. This unifies the keyspace
|
|
type ID []byte
|
|
|
|
func (id ID) Equal(other ID) bool {
|
|
return bytes.Equal(id, other)
|
|
}
|
|
|
|
func (id ID) Less(other interface{}) bool {
|
|
a, b := equalizeSizes(id, other.(ID))
|
|
for i := 0; i < len(a); i++ {
|
|
if a[i] != b[i] {
|
|
return a[i] < b[i]
|
|
}
|
|
}
|
|
return len(a) < len(b)
|
|
}
|
|
|
|
func (id ID) commonPrefixLen() int {
|
|
for i := 0; i < len(id); i++ {
|
|
for j := 0; j < 8; j++ {
|
|
if (id[i]>>uint8(7-j))&0x1 != 0 {
|
|
return i*8 + j
|
|
}
|
|
}
|
|
}
|
|
return len(id)*8 - 1
|
|
}
|
|
|
|
func prefLen(a, b ID) int {
|
|
return xor(a, b).commonPrefixLen()
|
|
}
|
|
|
|
func xor(a, b ID) ID {
|
|
a, b = equalizeSizes(a, b)
|
|
|
|
c := make(ID, len(a))
|
|
for i := 0; i < len(a); i++ {
|
|
c[i] = a[i] ^ b[i]
|
|
}
|
|
return c
|
|
}
|
|
|
|
func equalizeSizes(a, b ID) (ID, ID) {
|
|
la := len(a)
|
|
lb := len(b)
|
|
|
|
if la < lb {
|
|
na := make([]byte, lb)
|
|
copy(na, a)
|
|
a = na
|
|
} else if lb < la {
|
|
nb := make([]byte, la)
|
|
copy(nb, b)
|
|
b = nb
|
|
}
|
|
|
|
return a, b
|
|
}
|
|
|
|
func convertPeerID(id peer.ID) ID {
|
|
hash := sha256.Sum256(id)
|
|
return hash[:]
|
|
}
|
|
|
|
func convertKey(id u.Key) ID {
|
|
hash := sha256.Sum256([]byte(id))
|
|
return hash[:]
|
|
}
|