The core protocol of WoopChain
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
woop/api/service/explorer/structs.go

182 lines
4.6 KiB

package explorer
import (
"bytes"
"encoding/hex"
"math/big"
"strconv"
"github.com/ethereum/go-ethereum/common"
core2 "github.com/harmony-one/harmony/core"
"github.com/harmony-one/harmony/core/types"
common2 "github.com/harmony-one/harmony/internal/common"
"github.com/harmony-one/harmony/internal/utils"
staking "github.com/harmony-one/harmony/staking/types"
)
/*
* All the code here is work of progress for the sprint.
*/
// Tx types ...
const (
Received = "RECEIVED"
Sent = "SENT"
)
// TxRecord ...
type TxRecord struct {
Hash string
Type string
Timestamp string
}
// TxRecords ...
type TxRecords []*TxRecord
// Data ...
type Data struct {
Addresses []string `json:"Addresses"`
}
// Address ...
type Address struct {
ID string `json:"id"`
Balance *big.Int `json:"balance"`
TXs TxRecords `json:"txs"`
StakingTXs TxRecords `json:"staking_txs"`
}
// Transaction ...
type Transaction struct {
ID string `json:"id"`
Timestamp string `json:"timestamp"`
From string `json:"from"`
To string `json:"to"`
Value *big.Int `json:"value"`
Bytes string `json:"bytes"`
Data string `json:"data"`
GasFee *big.Int `json:"gasFee"`
FromShard uint32 `json:"fromShard"`
ToShard uint32 `json:"toShard"`
Type string `json:"type"`
}
// GetTransaction ...
func GetTransaction(tx *types.Transaction, addressBlock *types.Block) (*Transaction, error) {
msg, err := tx.AsMessage(types.NewEIP155Signer(tx.ChainID()))
if err != nil {
utils.Logger().Error().Err(err).Msg("Error when parsing tx into message")
}
gasFee := big.NewInt(0)
gasFee = gasFee.Mul(tx.GasPrice(), new(big.Int).SetUint64(tx.GasLimit()))
to := ""
if msg.To() != nil {
if to, err = common2.AddressToBech32(*msg.To()); err != nil {
return nil, err
}
}
from := ""
if from, err = common2.AddressToBech32(msg.From()); err != nil {
return nil, err
}
return &Transaction{
ID: tx.Hash().Hex(),
Timestamp: strconv.Itoa(int(addressBlock.Time().Int64() * 1000)),
From: from,
To: to,
Value: msg.Value(),
Bytes: strconv.Itoa(int(tx.Size())),
Data: hex.EncodeToString(tx.Data()),
GasFee: gasFee,
FromShard: tx.ShardID(),
ToShard: tx.ToShardID(),
Type: "",
}, nil
}
// StakingTransaction ...
type StakingTransaction struct {
Transaction
}
// GetStakingTransaction ...
func GetStakingTransaction(tx *staking.StakingTransaction, addressBlock *types.Block) (*StakingTransaction, error) {
msg, err := core2.StakingToMessage(tx, addressBlock.Header().Number())
if err != nil {
utils.Logger().Error().Err(err).Msg("Error when parsing tx into message")
return nil, err
}
gasFee := big.NewInt(0)
gasFee = gasFee.Mul(tx.GasPrice(), new(big.Int).SetUint64(tx.GasLimit()))
var toAddress *common.Address
// Populate to address of delegate and undelegate staking txns
// This is needed for supporting received txns support correctly for staking txns history api
// For other staking txns, there is no to address.
switch tx.StakingType() {
case staking.DirectiveDelegate:
stkMsg, err := staking.RLPDecodeStakeMsg(tx.Data(), staking.DirectiveDelegate)
if err != nil {
return nil, err
}
if _, ok := stkMsg.(*staking.Delegate); !ok {
return nil, core2.ErrInvalidMsgForStakingDirective
}
delegateMsg := stkMsg.(*staking.Delegate)
if !bytes.Equal(msg.From().Bytes()[:], delegateMsg.DelegatorAddress.Bytes()[:]) {
return nil, core2.ErrInvalidSender
}
toAddress = &delegateMsg.ValidatorAddress
case staking.DirectiveUndelegate:
stkMsg, err := staking.RLPDecodeStakeMsg(tx.Data(), staking.DirectiveUndelegate)
if err != nil {
return nil, err
}
if _, ok := stkMsg.(*staking.Undelegate); !ok {
return nil, core2.ErrInvalidMsgForStakingDirective
}
undelegateMsg := stkMsg.(*staking.Undelegate)
if !bytes.Equal(msg.From().Bytes()[:], undelegateMsg.DelegatorAddress.Bytes()[:]) {
return nil, core2.ErrInvalidSender
}
toAddress = &undelegateMsg.ValidatorAddress
default:
break
}
to := ""
if toAddress != nil {
if to, err = common2.AddressToBech32(*toAddress); err != nil {
return nil, err
}
}
from := ""
if from, err = common2.AddressToBech32(msg.From()); err != nil {
return nil, err
}
txn := Transaction{
ID: tx.Hash().Hex(),
Timestamp: strconv.Itoa(int(addressBlock.Time().Int64() * 1000)),
From: from,
To: to,
Value: msg.Value(),
Bytes: strconv.Itoa(int(tx.Size())),
Data: hex.EncodeToString(msg.Data()),
GasFee: gasFee,
FromShard: tx.ShardID(),
ToShard: 0,
Type: string(msg.Type()),
}
return &StakingTransaction{
Transaction: txn,
}, nil
}