|
|
package apitypes
import ( "database/sql/driver" "encoding/base64" "encoding/json" "errors" "fmt" "math/big" "strconv" "strings"
ethCommon "github.com/ethereum/go-ethereum/common" "github.com/hermeznetwork/hermez-node/common" "github.com/iden3/go-iden3-crypto/babyjub" )
// BigIntStr is used to scan/value *big.Int directly into strings from/to sql DBs.
// It assumes that *big.Int are inserted/fetched to/from the DB using the BigIntMeddler meddler
// defined at github.com/hermeznetwork/hermez-node/db
type BigIntStr string
// NewBigIntStr creates a *BigIntStr from a *big.Int.
// If the provided bigInt is nil the returned *BigIntStr will also be nil
func NewBigIntStr(bigInt *big.Int) *BigIntStr { if bigInt == nil { return nil } bigIntStr := BigIntStr(bigInt.String()) return &bigIntStr }
// Scan implements Scanner for database/sql
func (b *BigIntStr) Scan(src interface{}) error { // decode base64 src
var decoded []byte var err error if srcStr, ok := src.(string); ok { // src is a string
decoded, err = base64.StdEncoding.DecodeString(srcStr) } else if srcBytes, ok := src.([]byte); ok { // src is []byte
decoded, err = base64.StdEncoding.DecodeString(string(srcBytes)) } else { // unexpected src
return fmt.Errorf("can't scan %T into apitypes.BigIntStr", src) } if err != nil { return err } // decoded bytes to *big.Int
bigInt := &big.Int{} bigInt = bigInt.SetBytes(decoded) // *big.Int to BigIntStr
bigIntStr := NewBigIntStr(bigInt) if bigIntStr == nil { return nil } *b = *bigIntStr return nil }
// Value implements valuer for database/sql
func (b BigIntStr) Value() (driver.Value, error) { // string to *big.Int
bigInt := &big.Int{} bigInt, ok := bigInt.SetString(string(b), 10) if !ok || bigInt == nil { return nil, errors.New("invalid representation of a *big.Int") } // *big.Int to base64
return base64.StdEncoding.EncodeToString(bigInt.Bytes()), nil }
// StrBigInt is used to unmarshal BigIntStr directly into an alias of big.Int
type StrBigInt big.Int
// UnmarshalText unmarshals a StrBigInt
func (s *StrBigInt) UnmarshalText(text []byte) error { bi, ok := (*big.Int)(s).SetString(string(text), 10) if !ok { return fmt.Errorf("could not unmarshal %s into a StrBigInt", text) } *s = StrBigInt(*bi) return nil }
// CollectedFees is used to retrieve common.batch.CollectedFee from the DB
type CollectedFees map[common.TokenID]BigIntStr
// UnmarshalJSON unmarshals a json representation of map[common.TokenID]*big.Int
func (c *CollectedFees) UnmarshalJSON(text []byte) error { bigIntMap := make(map[common.TokenID]*big.Int) if err := json.Unmarshal(text, &bigIntMap); err != nil { return err } bStrMap := make(map[common.TokenID]BigIntStr) for k, v := range bigIntMap { bStr := NewBigIntStr(v) bStrMap[k] = *bStr } *c = CollectedFees(bStrMap) return nil }
// HezEthAddr is used to scan/value Ethereum Address directly into strings that follow the Ethereum address hez fotmat (^hez:0x[a-fA-F0-9]{40}$) from/to sql DBs.
// It assumes that Ethereum Address are inserted/fetched to/from the DB using the default Scan/Value interface
type HezEthAddr string
// NewHezEthAddr creates a HezEthAddr from an Ethereum addr
func NewHezEthAddr(addr ethCommon.Address) HezEthAddr { return HezEthAddr("hez:" + addr.String()) }
// ToEthAddr returns an Ethereum Address created from HezEthAddr
func (a HezEthAddr) ToEthAddr() (ethCommon.Address, error) { addrStr := strings.TrimPrefix(string(a), "hez:") var addr ethCommon.Address return addr, addr.UnmarshalText([]byte(addrStr)) }
// Scan implements Scanner for database/sql
func (a *HezEthAddr) Scan(src interface{}) error { ethAddr := ðCommon.Address{} if err := ethAddr.Scan(src); err != nil { return err } if ethAddr == nil { return nil } *a = NewHezEthAddr(*ethAddr) return nil }
// Value implements valuer for database/sql
func (a HezEthAddr) Value() (driver.Value, error) { ethAddr, err := a.ToEthAddr() if err != nil { return nil, err } return ethAddr.Value() }
// StrHezEthAddr is used to unmarshal HezEthAddr directly into an alias of ethCommon.Address
type StrHezEthAddr ethCommon.Address
// UnmarshalText unmarshals a StrHezEthAddr
func (s *StrHezEthAddr) UnmarshalText(text []byte) error { withoutHez := strings.TrimPrefix(string(text), "hez:") var addr ethCommon.Address if err := addr.UnmarshalText([]byte(withoutHez)); err != nil { return err } *s = StrHezEthAddr(addr) return nil }
// HezBJJ is used to scan/value *babyjub.PublicKey directly into strings that follow the BJJ public key hez fotmat (^hez:[A-Za-z0-9_-]{44}$) from/to sql DBs.
// It assumes that *babyjub.PublicKey are inserted/fetched to/from the DB using the default Scan/Value interface
type HezBJJ string
// NewHezBJJ creates a HezBJJ from a *babyjub.PublicKey.
// Calling this method with a nil bjj causes panic
func NewHezBJJ(bjj *babyjub.PublicKey) HezBJJ { pkComp := [32]byte(bjj.Compress()) sum := pkComp[0] for i := 1; i < len(pkComp); i++ { sum += pkComp[i] } bjjSum := append(pkComp[:], sum) return HezBJJ("hez:" + base64.RawURLEncoding.EncodeToString(bjjSum)) }
func hezStrToBJJ(s string) (*babyjub.PublicKey, error) { const decodedLen = 33 const encodedLen = 44 formatErr := errors.New("invalid BJJ format. Must follow this regex: ^hez:[A-Za-z0-9_-]{44}$") encoded := strings.TrimPrefix(s, "hez:") if len(encoded) != encodedLen { return nil, formatErr } decoded, err := base64.RawURLEncoding.DecodeString(encoded) if err != nil { return nil, formatErr } if len(decoded) != decodedLen { return nil, formatErr } bjjBytes := [decodedLen - 1]byte{} copy(bjjBytes[:decodedLen-1], decoded[:decodedLen-1]) sum := bjjBytes[0] for i := 1; i < len(bjjBytes); i++ { sum += bjjBytes[i] } if decoded[decodedLen-1] != sum { return nil, errors.New("checksum verification failed") } bjjComp := babyjub.PublicKeyComp(bjjBytes) return bjjComp.Decompress() }
// ToBJJ returns a *babyjub.PublicKey created from HezBJJ
func (b HezBJJ) ToBJJ() (*babyjub.PublicKey, error) { return hezStrToBJJ(string(b)) }
// Scan implements Scanner for database/sql
func (b *HezBJJ) Scan(src interface{}) error { bjj := &babyjub.PublicKey{} if err := bjj.Scan(src); err != nil { return err } if bjj == nil { return nil } *b = NewHezBJJ(bjj) return nil }
// Value implements valuer for database/sql
func (b HezBJJ) Value() (driver.Value, error) { bjj, err := b.ToBJJ() if err != nil { return nil, err } return bjj.Value() }
// StrHezBJJ is used to unmarshal HezBJJ directly into an alias of babyjub.PublicKey
type StrHezBJJ babyjub.PublicKey
// UnmarshalText unmarshals a StrHezBJJ
func (s *StrHezBJJ) UnmarshalText(text []byte) error { bjj, err := hezStrToBJJ(string(text)) if err != nil { return err } *s = StrHezBJJ(*bjj) return nil }
// HezIdx is used to value common.Idx directly into strings that follow the Idx key hez fotmat (hez:tokenSymbol:idx) to sql DBs.
// Note that this can only be used to insert to DB since there is no way to automaticaly read from the DB since it needs the tokenSymbol
type HezIdx string
// StrHezIdx is used to unmarshal HezIdx directly into an alias of common.Idx
type StrHezIdx common.Idx
// UnmarshalText unmarshals a StrHezIdx
func (s *StrHezIdx) UnmarshalText(text []byte) error { withoutHez := strings.TrimPrefix(string(text), "hez:") splitted := strings.Split(withoutHez, ":") const expectedLen = 2 if len(splitted) != expectedLen { return fmt.Errorf("can not unmarshal %s into StrHezIdx", text) } idxInt, err := strconv.Atoi(splitted[1]) if err != nil { return err } *s = StrHezIdx(common.Idx(idxInt)) return nil }
|