mirror of
https://gitlab.com/pulsechaincom/erigon-pulse.git
synced 2025-01-16 07:48:20 +00:00
78 lines
1.5 KiB
Go
78 lines
1.5 KiB
Go
package observer
|
|
|
|
import (
|
|
"context"
|
|
"crypto/ecdsa"
|
|
"time"
|
|
|
|
"github.com/ledgerwatch/erigon/crypto"
|
|
"github.com/ledgerwatch/erigon/p2p/enode"
|
|
"github.com/ledgerwatch/log/v3"
|
|
)
|
|
|
|
func keygen(
|
|
parentContext context.Context,
|
|
targetKey *ecdsa.PublicKey,
|
|
timeout time.Duration,
|
|
concurrencyLimit uint,
|
|
logger log.Logger,
|
|
) []*ecdsa.PublicKey {
|
|
ctx, cancel := context.WithTimeout(parentContext, timeout)
|
|
defer cancel()
|
|
|
|
targetID := enode.PubkeyToIDV4(targetKey)
|
|
cpus := concurrencyLimit
|
|
|
|
type result struct {
|
|
key *ecdsa.PublicKey
|
|
distance int
|
|
}
|
|
|
|
generatedKeys := make(chan result, cpus)
|
|
|
|
for i := uint(0); i < cpus; i++ {
|
|
go func() {
|
|
for ctx.Err() == nil {
|
|
keyPair, err := crypto.GenerateKey()
|
|
if err != nil {
|
|
logger.Error("keygen has failed to generate a key", "err", err)
|
|
break
|
|
}
|
|
|
|
key := &keyPair.PublicKey
|
|
id := enode.PubkeyToIDV4(key)
|
|
distance := enode.LogDist(targetID, id)
|
|
|
|
select {
|
|
case generatedKeys <- result{key, distance}:
|
|
case <-ctx.Done():
|
|
break
|
|
}
|
|
}
|
|
}()
|
|
}
|
|
|
|
keysAtDist := make(map[int]*ecdsa.PublicKey)
|
|
|
|
for ctx.Err() == nil {
|
|
select {
|
|
case res := <-generatedKeys:
|
|
keysAtDist[res.distance] = res.key
|
|
case <-ctx.Done():
|
|
break
|
|
}
|
|
}
|
|
|
|
keys := valuesOfIntToPubkeyMap(keysAtDist)
|
|
|
|
return keys
|
|
}
|
|
|
|
func valuesOfIntToPubkeyMap(m map[int]*ecdsa.PublicKey) []*ecdsa.PublicKey {
|
|
values := make([]*ecdsa.PublicKey, 0, len(m))
|
|
for _, value := range m {
|
|
values = append(values, value)
|
|
}
|
|
return values
|
|
}
|