Files
prysm/consensus-types/primitives/epoch.go
Preston Van Loon 2aa52fb56a Add static analyzer to discourage use of panic() (#15075)
* Implement static analysis to prevent panics

* Add nopanic to nogo

* Fix violations and add exclusions

Fix violations and add exclusions for all

* Changelog fragment

* Use pass.Report instead of pass.Reportf

* Remove strings.ToLower for checking init method name

* Add exclusion for herumi init

* Move api/client/beacon template function to init and its own file

* Fix nopanic testcase
2025-03-19 18:04:15 +00:00

161 lines
4.3 KiB
Go

package primitives
import (
"fmt"
fssz "github.com/prysmaticlabs/fastssz"
"github.com/prysmaticlabs/prysm/v5/math"
)
var _ fssz.HashRoot = (Epoch)(0)
var _ fssz.Marshaler = (*Epoch)(nil)
var _ fssz.Unmarshaler = (*Epoch)(nil)
// Epoch represents a single epoch.
type Epoch uint64
// MaxEpoch compares two epochs and returns the greater one.
func MaxEpoch(a, b Epoch) Epoch {
if a > b {
return a
}
return b
}
// Mul multiplies epoch by x.
// In case of arithmetic issues (overflow/underflow/div by zero) panic is thrown.
func (e Epoch) Mul(x uint64) Epoch {
res, err := e.SafeMul(x)
if err != nil {
panic(err.Error()) // lint:nopanic -- Panic is communicated in the godoc commentary.
}
return res
}
// SafeMul multiplies epoch by x.
// In case of arithmetic issues (overflow/underflow/div by zero) error is returned.
func (e Epoch) SafeMul(x uint64) (Epoch, error) {
res, err := math.Mul64(uint64(e), x)
return Epoch(res), err
}
// Div divides epoch by x.
// In case of arithmetic issues (overflow/underflow/div by zero) panic is thrown.
func (e Epoch) Div(x uint64) Epoch {
res, err := e.SafeDiv(x)
if err != nil {
panic(err.Error()) // lint:nopanic -- Panic is communicated in the godoc commentary.
}
return res
}
// SafeDiv divides epoch by x.
// In case of arithmetic issues (overflow/underflow/div by zero) error is returned.
func (e Epoch) SafeDiv(x uint64) (Epoch, error) {
res, err := math.Div64(uint64(e), x)
return Epoch(res), err
}
// Add increases epoch by x.
// In case of arithmetic issues (overflow/underflow/div by zero) panic is thrown.
func (e Epoch) Add(x uint64) Epoch {
res, err := e.SafeAdd(x)
if err != nil {
panic(err.Error()) // lint:nopanic -- Panic is communicated in the godoc commentary.
}
return res
}
// SafeAdd increases epoch by x.
// In case of arithmetic issues (overflow/underflow/div by zero) error is returned.
func (e Epoch) SafeAdd(x uint64) (Epoch, error) {
res, err := math.Add64(uint64(e), x)
return Epoch(res), err
}
// AddEpoch increases epoch using another epoch value.
// In case of arithmetic issues (overflow/underflow/div by zero) panic is thrown.
func (e Epoch) AddEpoch(x Epoch) Epoch {
return e.Add(uint64(x))
}
// SafeAddEpoch increases epoch using another epoch value.
// In case of arithmetic issues (overflow/underflow/div by zero) error is returned.
func (e Epoch) SafeAddEpoch(x Epoch) (Epoch, error) {
return e.SafeAdd(uint64(x))
}
// Sub subtracts x from the epoch.
// In case of arithmetic issues (overflow/underflow/div by zero) panic is thrown.
func (e Epoch) Sub(x uint64) Epoch {
res, err := e.SafeSub(x)
if err != nil {
panic(err.Error()) // lint:nopanic -- Panic is communicated in the godoc commentary.
}
return res
}
// SafeSub subtracts x from the epoch.
// In case of arithmetic issues (overflow/underflow/div by zero) error is returned.
func (e Epoch) SafeSub(x uint64) (Epoch, error) {
res, err := math.Sub64(uint64(e), x)
return Epoch(res), err
}
// Mod returns result of `epoch % x`.
// In case of arithmetic issues (overflow/underflow/div by zero) panic is thrown.
func (e Epoch) Mod(x uint64) Epoch {
res, err := e.SafeMod(x)
if err != nil {
panic(err.Error()) // lint:nopanic -- Panic is communicated in the godoc commentary.
}
return res
}
// SafeMod returns result of `epoch % x`.
// In case of arithmetic issues (overflow/underflow/div by zero) error is returned.
func (e Epoch) SafeMod(x uint64) (Epoch, error) {
res, err := math.Mod64(uint64(e), x)
return Epoch(res), err
}
// HashTreeRoot --
func (e Epoch) HashTreeRoot() ([32]byte, error) {
return fssz.HashWithDefaultHasher(e)
}
// HashTreeRootWith --
func (e Epoch) HashTreeRootWith(hh *fssz.Hasher) error {
hh.PutUint64(uint64(e))
return nil
}
// UnmarshalSSZ --
func (e *Epoch) UnmarshalSSZ(buf []byte) error {
if len(buf) != e.SizeSSZ() {
return fmt.Errorf("expected buffer of length %d received %d", e.SizeSSZ(), len(buf))
}
*e = Epoch(fssz.UnmarshallUint64(buf))
return nil
}
// MarshalSSZTo --
func (e *Epoch) MarshalSSZTo(dst []byte) ([]byte, error) {
marshalled, err := e.MarshalSSZ()
if err != nil {
return nil, err
}
return append(dst, marshalled...), nil
}
// MarshalSSZ --
func (e *Epoch) MarshalSSZ() ([]byte, error) {
marshalled := fssz.MarshalUint64([]byte{}, uint64(*e))
return marshalled, nil
}
// SizeSSZ --
func (e *Epoch) SizeSSZ() int {
return 8
}