ceremonyclient/node/p2p/bloom_utils.go

103 lines
2.3 KiB
Go
Raw Permalink Normal View History

2023-09-03 23:47:09 +00:00
package p2p
import (
"fmt"
"math/big"
2024-01-03 07:31:42 +00:00
"sort"
2023-09-03 23:47:09 +00:00
"golang.org/x/crypto/sha3"
)
func GetOnesIndices(input []byte) []int {
2023-09-03 23:47:09 +00:00
var indices []int
for i, b := range input {
for j := 0; j < 8; j++ {
if (b>>j)&1 == 1 {
indices = append(indices, i*8+j)
}
}
}
return indices
}
2024-01-03 07:31:42 +00:00
// GetBloomFilter returns a bloom filter based on the data, however
2023-09-03 23:47:09 +00:00
// it assumes bitLength is a multiple of 32. If the filter size is not
// conformant, this will generate biased indices.
2024-01-03 07:31:42 +00:00
func GetBloomFilter(data []byte, bitLength int, k int) []byte {
size := big.NewInt(int64(bitLength)).BitLen() - 1
2023-09-03 23:47:09 +00:00
digest := sha3.Sum256(data)
output := make([]byte, bitLength/8)
outputBI := big.NewInt(0)
digestBI := new(big.Int).SetBytes(digest[:])
2023-09-03 23:47:09 +00:00
for i := 0; i < k; i++ {
position := uint(0)
2024-01-03 07:31:42 +00:00
for j := size * i; j < size*(i+1); j++ {
position = position<<1 | (digestBI.Bit(j))
}
if outputBI.Bit(int(position)) != 1 {
outputBI.SetBit(outputBI, int(position), 1)
} else if k*size <= 32 {
2023-09-03 23:47:09 +00:00
// we need to extend the search
k++
} else {
fmt.Printf(
"digest %+x cannot be used as bloom index, panicking\n",
digest,
)
panic(
"could not generate bloom filter index, k offset cannot be adjusted",
)
}
}
outputBI.FillBytes(output)
return output
}
2024-01-03 07:31:42 +00:00
// GetBloomFilterIndices returns the indices of a bloom filter, in increasing
// order, assuming bitLength is a multiple of 32 as in GetBloomFilter.
func GetBloomFilterIndices(data []byte, bitLength int, k int) []byte {
size := big.NewInt(int64(bitLength)).BitLen() - 1
h := sha3.NewShake256()
_, err := h.Write(data)
if err != nil {
panic(err)
}
digest := make([]byte, size*k)
2024-01-03 07:31:42 +00:00
_, err = h.Read(digest)
if err != nil {
panic(err)
}
indices := []string{}
for i := 0; i < k; i++ {
position := make([]byte, size/8)
for j := (size / 8) * i; j < (size/8)*(i+1); j++ {
position[j%(size/8)] = digest[j]
}
found := false
for _, ext := range indices {
if ext == string(position) {
k++
found = true
break
}
}
if !found {
p := sort.SearchStrings(indices, string(position))
if len(indices) > p {
indices = append(indices[:p+1], indices[p:]...)
indices[p] = string(position)
} else {
indices = append(indices, string(position))
}
}
}
output := ""
for _, idx := range indices {
output += idx
}
return []byte(output)
}