-
Notifications
You must be signed in to change notification settings - Fork 211
/
mock_oracle.go
112 lines (89 loc) · 2.47 KB
/
mock_oracle.go
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
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
package hare
import (
"errors"
"github.com/spacemeshos/go-spacemesh/common/types"
"github.com/spacemeshos/go-spacemesh/log"
"hash/fnv"
"math"
"sync"
)
type registrable interface {
Register(isHonest bool, id string)
Unregister(isHonest bool, id string)
}
type hasherU32 struct {
}
func newHasherU32() *hasherU32 {
h := new(hasherU32)
return h
}
func (h *hasherU32) Hash(values ...[]byte) uint32 {
fnv := fnv.New32()
for _, b := range values {
fnv.Write(b)
}
return fnv.Sum32()
}
func (h *hasherU32) MaxValue() uint32 {
return math.MaxUint32
}
type mockHashOracle struct {
clients map[string]struct{}
mutex sync.RWMutex
hasher *hasherU32
}
func (mho *mockHashOracle) IsIdentityActiveOnConsensusView(edID string, layer types.LayerID) (bool, error) {
return true, nil
}
func newMockHashOracle(expectedSize int) *mockHashOracle {
mock := new(mockHashOracle)
mock.clients = make(map[string]struct{}, expectedSize)
mock.hasher = newHasherU32()
return mock
}
func (mho *mockHashOracle) Register(client string) {
mho.mutex.Lock()
if _, exist := mho.clients[client]; exist {
mho.mutex.Unlock()
return
}
mho.clients[client] = struct{}{}
mho.mutex.Unlock()
}
func (mho *mockHashOracle) Unregister(client string) {
mho.mutex.Lock()
delete(mho.clients, client)
mho.mutex.Unlock()
}
// Calculates the threshold for the given committee size
func (mho *mockHashOracle) calcThreshold(committeeSize int) uint32 {
mho.mutex.RLock()
numClients := len(mho.clients)
mho.mutex.RUnlock()
if numClients == 0 {
log.Error("Called calcThreshold with 0 clients registered")
return 0
}
if committeeSize > numClients {
/*log.Error("Requested for a committee bigger than the number of registered clients. Expected at least %v clients Actual: %v",
committeeSize, numClients)*/
return 0
}
return uint32(uint64(committeeSize) * uint64(mho.hasher.MaxValue()) / uint64(numClients))
}
// Eligible if a proof is valid for a given committee size
func (mho *mockHashOracle) Eligible(layer types.LayerID, round int32, committeeSize int, id types.NodeID, sig []byte) (bool, error) {
if sig == nil {
log.Warning("Oracle query with proof=nil. Returning false")
return false, errors.New("sig is nil")
}
// calculate hash of proof
proofHash := mho.hasher.Hash(sig)
if proofHash <= mho.calcThreshold(committeeSize) { // check threshold
return true, nil
}
return false, nil
}
func (mho *mockHashOracle) Proof(layer types.LayerID, round int32) ([]byte, error) {
return []byte{}, nil
}