2020-06-26 14:58:47 +00:00
|
|
|
/*
|
2021-11-12 20:30:09 +00:00
|
|
|
Package rand defines methods of obtaining random number generators.
|
2020-06-26 14:58:47 +00:00
|
|
|
|
|
|
|
One is expected to use randomness from this package only, without introducing any other packages.
|
|
|
|
This limits the scope of code that needs to be hardened.
|
|
|
|
|
|
|
|
There are two modes, one for deterministic and another non-deterministic randomness:
|
|
|
|
1. If deterministic pseudo-random generator is enough, use:
|
|
|
|
|
2021-09-15 22:55:11 +00:00
|
|
|
import "github.com/prysmaticlabs/prysm/crypto/rand"
|
2020-06-26 14:58:47 +00:00
|
|
|
randGen := rand.NewDeterministicGenerator()
|
|
|
|
randGen.Intn(32) // or any other func defined in math.rand API
|
|
|
|
|
|
|
|
In this mode, only seed is generated using cryptographically secure source (crypto/rand). So,
|
|
|
|
once seed is obtained, and generator is seeded, the next generations are deterministic, thus fast.
|
2021-11-12 20:30:09 +00:00
|
|
|
However given that we only seed this 63 bits from crypto/rand and use math/rand to generate the outputs,
|
|
|
|
this method is not cryptographically secure. This is directly stated in the math/rand package,
|
|
|
|
https://github.com/golang/go/blob/release-branch.go1.17/src/math/rand/rand.go#L15. For any security
|
|
|
|
sensitive work this particular generator is NOT to be used.
|
2020-06-26 14:58:47 +00:00
|
|
|
|
|
|
|
2. For cryptographically secure non-deterministic mode (CSPRNG), use:
|
|
|
|
|
2021-09-15 22:55:11 +00:00
|
|
|
import "github.com/prysmaticlabs/prysm/crypto/rand"
|
2020-06-26 14:58:47 +00:00
|
|
|
randGen := rand.NewGenerator()
|
|
|
|
randGen.Intn(32) // or any other func defined in math.rand API
|
|
|
|
|
|
|
|
Again, any of the functions from `math/rand` can be used, however, they all use custom source
|
|
|
|
of randomness (crypto/rand), on every step. This makes randomness non-deterministic. However,
|
|
|
|
you take a performance hit -- as it is an order of magnitude slower.
|
|
|
|
*/
|
|
|
|
package rand
|
|
|
|
|
|
|
|
import (
|
|
|
|
"crypto/rand"
|
|
|
|
"encoding/binary"
|
|
|
|
mrand "math/rand"
|
2020-11-11 02:57:07 +00:00
|
|
|
"sync"
|
2020-06-26 14:58:47 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
type source struct{}
|
|
|
|
|
2020-11-11 02:57:07 +00:00
|
|
|
var lock sync.RWMutex
|
2021-08-15 15:24:13 +00:00
|
|
|
var _ mrand.Source64 = (*source)(nil) /* #nosec G404 */
|
2020-06-26 14:58:47 +00:00
|
|
|
|
|
|
|
// Seed does nothing when crypto/rand is used as source.
|
2020-10-12 08:11:05 +00:00
|
|
|
func (s *source) Seed(_ int64) {}
|
2020-06-26 14:58:47 +00:00
|
|
|
|
|
|
|
// Int63 returns uniformly-distributed random (as in CSPRNG) int64 value within [0, 1<<63) range.
|
|
|
|
// Panics if random generator reader cannot return data.
|
|
|
|
func (s *source) Int63() int64 {
|
|
|
|
return int64(s.Uint64() & ^uint64(1<<63))
|
|
|
|
}
|
|
|
|
|
|
|
|
// Uint64 returns uniformly-distributed random (as in CSPRNG) uint64 value within [0, 1<<64) range.
|
|
|
|
// Panics if random generator reader cannot return data.
|
|
|
|
func (s *source) Uint64() (val uint64) {
|
2020-11-11 02:57:07 +00:00
|
|
|
lock.RLock()
|
|
|
|
defer lock.RUnlock()
|
2020-06-26 14:58:47 +00:00
|
|
|
if err := binary.Read(rand.Reader, binary.BigEndian, &val); err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// Rand is alias for underlying random generator.
|
2021-08-15 15:24:13 +00:00
|
|
|
type Rand = mrand.Rand /* #nosec G404 */
|
2020-06-26 14:58:47 +00:00
|
|
|
|
|
|
|
// NewGenerator returns a new generator that uses random values from crypto/rand as a source
|
|
|
|
// (cryptographically secure random number generator).
|
|
|
|
// Panics if crypto/rand input cannot be read.
|
|
|
|
// Use it for everything where crypto secure non-deterministic randomness is required. Performance
|
|
|
|
// takes a hit, so use sparingly.
|
|
|
|
func NewGenerator() *Rand {
|
2021-08-15 15:24:13 +00:00
|
|
|
return mrand.New(&source{}) /* #nosec G404 */
|
2020-06-26 14:58:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// NewDeterministicGenerator returns a random generator which is only seeded with crypto/rand,
|
|
|
|
// but is deterministic otherwise (given seed, produces given results, deterministically).
|
|
|
|
// Panics if crypto/rand input cannot be read.
|
|
|
|
// Use this method for performance, where deterministic pseudo-random behaviour is enough.
|
2021-11-12 20:30:09 +00:00
|
|
|
// Otherwise, rely on NewGenerator(). This method is not cryptographically secure as outputs
|
|
|
|
// can be potentially predicted even without knowledge of the underlying seed.
|
2020-06-26 14:58:47 +00:00
|
|
|
func NewDeterministicGenerator() *Rand {
|
|
|
|
randGen := NewGenerator()
|
2021-08-15 15:24:13 +00:00
|
|
|
return mrand.New(mrand.NewSource(randGen.Int63())) /* #nosec G404 */
|
2020-06-26 14:58:47 +00:00
|
|
|
}
|