332 lines
10 KiB
Go
332 lines
10 KiB
Go
package rest
|
|
|
|
import (
|
|
"bytes"
|
|
"fmt"
|
|
"io/ioutil"
|
|
"net/http"
|
|
|
|
"github.com/cosmos/cosmos-sdk/client/context"
|
|
"github.com/cosmos/cosmos-sdk/crypto/keys"
|
|
sdk "github.com/cosmos/cosmos-sdk/types"
|
|
"github.com/cosmos/cosmos-sdk/wire"
|
|
authcliCtx "github.com/cosmos/cosmos-sdk/x/auth/client/context"
|
|
"github.com/cosmos/cosmos-sdk/x/stake"
|
|
"github.com/cosmos/cosmos-sdk/x/stake/types"
|
|
|
|
"github.com/gorilla/mux"
|
|
|
|
ctypes "github.com/tendermint/tendermint/rpc/core/types"
|
|
)
|
|
|
|
func registerTxRoutes(cliCtx context.CLIContext, r *mux.Router, cdc *wire.Codec, kb keys.Keybase) {
|
|
r.HandleFunc(
|
|
"/stake/delegators/{delegatorAddr}/delegations",
|
|
delegationsRequestHandlerFn(cdc, kb, cliCtx),
|
|
).Methods("POST")
|
|
}
|
|
|
|
type msgDelegationsInput struct {
|
|
DelegatorAddr string `json:"delegator_addr"` // in bech32
|
|
ValidatorAddr string `json:"validator_addr"` // in bech32
|
|
Delegation sdk.Coin `json:"delegation"`
|
|
}
|
|
type msgBeginRedelegateInput struct {
|
|
DelegatorAddr string `json:"delegator_addr"` // in bech32
|
|
ValidatorSrcAddr string `json:"validator_src_addr"` // in bech32
|
|
ValidatorDstAddr string `json:"validator_dst_addr"` // in bech32
|
|
SharesAmount string `json:"shares"`
|
|
}
|
|
type msgCompleteRedelegateInput struct {
|
|
DelegatorAddr string `json:"delegator_addr"` // in bech32
|
|
ValidatorSrcAddr string `json:"validator_src_addr"` // in bech32
|
|
ValidatorDstAddr string `json:"validator_dst_addr"` // in bech32
|
|
}
|
|
type msgBeginUnbondingInput struct {
|
|
DelegatorAddr string `json:"delegator_addr"` // in bech32
|
|
ValidatorAddr string `json:"validator_addr"` // in bech32
|
|
SharesAmount string `json:"shares"`
|
|
}
|
|
type msgCompleteUnbondingInput struct {
|
|
DelegatorAddr string `json:"delegator_addr"` // in bech32
|
|
ValidatorAddr string `json:"validator_addr"` // in bech32
|
|
}
|
|
|
|
// the request body for edit delegations
|
|
type EditDelegationsBody struct {
|
|
LocalAccountName string `json:"name"`
|
|
Password string `json:"password"`
|
|
ChainID string `json:"chain_id"`
|
|
AccountNumber int64 `json:"account_number"`
|
|
Sequence int64 `json:"sequence"`
|
|
Gas int64 `json:"gas"`
|
|
Delegations []msgDelegationsInput `json:"delegations"`
|
|
BeginUnbondings []msgBeginUnbondingInput `json:"begin_unbondings"`
|
|
CompleteUnbondings []msgCompleteUnbondingInput `json:"complete_unbondings"`
|
|
BeginRedelegates []msgBeginRedelegateInput `json:"begin_redelegates"`
|
|
CompleteRedelegates []msgCompleteRedelegateInput `json:"complete_redelegates"`
|
|
}
|
|
|
|
// nolint: gocyclo
|
|
// TODO: Split this up into several smaller functions, and remove the above nolint
|
|
// TODO: use sdk.ValAddress instead of sdk.AccAddress for validators in messages
|
|
func delegationsRequestHandlerFn(cdc *wire.Codec, kb keys.Keybase, cliCtx context.CLIContext) http.HandlerFunc {
|
|
return func(w http.ResponseWriter, r *http.Request) {
|
|
var m EditDelegationsBody
|
|
|
|
body, err := ioutil.ReadAll(r.Body)
|
|
if err != nil {
|
|
w.WriteHeader(http.StatusBadRequest)
|
|
w.Write([]byte(err.Error()))
|
|
return
|
|
}
|
|
|
|
err = cdc.UnmarshalJSON(body, &m)
|
|
if err != nil {
|
|
w.WriteHeader(http.StatusBadRequest)
|
|
w.Write([]byte(err.Error()))
|
|
return
|
|
}
|
|
|
|
info, err := kb.Get(m.LocalAccountName)
|
|
if err != nil {
|
|
w.WriteHeader(http.StatusUnauthorized)
|
|
w.Write([]byte(err.Error()))
|
|
return
|
|
}
|
|
|
|
// build messages
|
|
messages := make([]sdk.Msg, len(m.Delegations)+
|
|
len(m.BeginRedelegates)+
|
|
len(m.CompleteRedelegates)+
|
|
len(m.BeginUnbondings)+
|
|
len(m.CompleteUnbondings))
|
|
|
|
i := 0
|
|
for _, msg := range m.Delegations {
|
|
delegatorAddr, err := sdk.AccAddressFromBech32(msg.DelegatorAddr)
|
|
if err != nil {
|
|
w.WriteHeader(http.StatusInternalServerError)
|
|
w.Write([]byte(fmt.Sprintf("Couldn't decode delegator. Error: %s", err.Error())))
|
|
return
|
|
}
|
|
|
|
validatorAddr, err := sdk.AccAddressFromBech32(msg.ValidatorAddr)
|
|
if err != nil {
|
|
w.WriteHeader(http.StatusInternalServerError)
|
|
w.Write([]byte(fmt.Sprintf("Couldn't decode validator. Error: %s", err.Error())))
|
|
return
|
|
}
|
|
|
|
if !bytes.Equal(info.GetPubKey().Address(), delegatorAddr) {
|
|
w.WriteHeader(http.StatusUnauthorized)
|
|
w.Write([]byte("Must use own delegator address"))
|
|
return
|
|
}
|
|
|
|
messages[i] = stake.MsgDelegate{
|
|
DelegatorAddr: delegatorAddr,
|
|
ValidatorAddr: validatorAddr,
|
|
Delegation: msg.Delegation,
|
|
}
|
|
|
|
i++
|
|
}
|
|
|
|
for _, msg := range m.BeginRedelegates {
|
|
delegatorAddr, err := sdk.AccAddressFromBech32(msg.DelegatorAddr)
|
|
if err != nil {
|
|
w.WriteHeader(http.StatusInternalServerError)
|
|
w.Write([]byte(fmt.Sprintf("Couldn't decode delegator. Error: %s", err.Error())))
|
|
return
|
|
}
|
|
|
|
if !bytes.Equal(info.GetPubKey().Address(), delegatorAddr) {
|
|
w.WriteHeader(http.StatusUnauthorized)
|
|
w.Write([]byte("Must use own delegator address"))
|
|
return
|
|
}
|
|
|
|
validatorSrcAddr, err := sdk.AccAddressFromBech32(msg.ValidatorSrcAddr)
|
|
if err != nil {
|
|
w.WriteHeader(http.StatusInternalServerError)
|
|
w.Write([]byte(fmt.Sprintf("Couldn't decode validator. Error: %s", err.Error())))
|
|
return
|
|
}
|
|
validatorDstAddr, err := sdk.AccAddressFromBech32(msg.ValidatorDstAddr)
|
|
if err != nil {
|
|
w.WriteHeader(http.StatusInternalServerError)
|
|
w.Write([]byte(fmt.Sprintf("Couldn't decode validator. Error: %s", err.Error())))
|
|
return
|
|
}
|
|
|
|
shares, err := sdk.NewRatFromDecimal(msg.SharesAmount, types.MaxBondDenominatorPrecision)
|
|
if err != nil {
|
|
w.WriteHeader(http.StatusInternalServerError)
|
|
w.Write([]byte(fmt.Sprintf("Couldn't decode shares amount. Error: %s", err.Error())))
|
|
return
|
|
}
|
|
|
|
messages[i] = stake.MsgBeginRedelegate{
|
|
DelegatorAddr: delegatorAddr,
|
|
ValidatorSrcAddr: validatorSrcAddr,
|
|
ValidatorDstAddr: validatorDstAddr,
|
|
SharesAmount: shares,
|
|
}
|
|
|
|
i++
|
|
}
|
|
|
|
for _, msg := range m.CompleteRedelegates {
|
|
delegatorAddr, err := sdk.AccAddressFromBech32(msg.DelegatorAddr)
|
|
if err != nil {
|
|
w.WriteHeader(http.StatusInternalServerError)
|
|
w.Write([]byte(fmt.Sprintf("Couldn't decode delegator. Error: %s", err.Error())))
|
|
return
|
|
}
|
|
|
|
validatorSrcAddr, err := sdk.AccAddressFromBech32(msg.ValidatorSrcAddr)
|
|
if err != nil {
|
|
w.WriteHeader(http.StatusInternalServerError)
|
|
w.Write([]byte(fmt.Sprintf("Couldn't decode validator. Error: %s", err.Error())))
|
|
return
|
|
}
|
|
validatorDstAddr, err := sdk.AccAddressFromBech32(msg.ValidatorDstAddr)
|
|
if err != nil {
|
|
w.WriteHeader(http.StatusInternalServerError)
|
|
w.Write([]byte(fmt.Sprintf("Couldn't decode validator. Error: %s", err.Error())))
|
|
return
|
|
}
|
|
|
|
if !bytes.Equal(info.GetPubKey().Address(), delegatorAddr) {
|
|
w.WriteHeader(http.StatusUnauthorized)
|
|
w.Write([]byte("Must use own delegator address"))
|
|
return
|
|
}
|
|
|
|
messages[i] = stake.MsgCompleteRedelegate{
|
|
DelegatorAddr: delegatorAddr,
|
|
ValidatorSrcAddr: validatorSrcAddr,
|
|
ValidatorDstAddr: validatorDstAddr,
|
|
}
|
|
|
|
i++
|
|
}
|
|
|
|
for _, msg := range m.BeginUnbondings {
|
|
delegatorAddr, err := sdk.AccAddressFromBech32(msg.DelegatorAddr)
|
|
if err != nil {
|
|
w.WriteHeader(http.StatusInternalServerError)
|
|
w.Write([]byte(fmt.Sprintf("Couldn't decode delegator. Error: %s", err.Error())))
|
|
return
|
|
}
|
|
|
|
if !bytes.Equal(info.GetPubKey().Address(), delegatorAddr) {
|
|
w.WriteHeader(http.StatusUnauthorized)
|
|
w.Write([]byte("Must use own delegator address"))
|
|
return
|
|
}
|
|
|
|
validatorAddr, err := sdk.AccAddressFromBech32(msg.ValidatorAddr)
|
|
if err != nil {
|
|
w.WriteHeader(http.StatusInternalServerError)
|
|
w.Write([]byte(fmt.Sprintf("Couldn't decode validator. Error: %s", err.Error())))
|
|
return
|
|
}
|
|
|
|
shares, err := sdk.NewRatFromDecimal(msg.SharesAmount, types.MaxBondDenominatorPrecision)
|
|
if err != nil {
|
|
w.WriteHeader(http.StatusInternalServerError)
|
|
w.Write([]byte(fmt.Sprintf("Couldn't decode shares amount. Error: %s", err.Error())))
|
|
return
|
|
}
|
|
|
|
messages[i] = stake.MsgBeginUnbonding{
|
|
DelegatorAddr: delegatorAddr,
|
|
ValidatorAddr: validatorAddr,
|
|
SharesAmount: shares,
|
|
}
|
|
|
|
i++
|
|
}
|
|
|
|
for _, msg := range m.CompleteUnbondings {
|
|
delegatorAddr, err := sdk.AccAddressFromBech32(msg.DelegatorAddr)
|
|
if err != nil {
|
|
w.WriteHeader(http.StatusInternalServerError)
|
|
w.Write([]byte(fmt.Sprintf("Couldn't decode delegator. Error: %s", err.Error())))
|
|
return
|
|
}
|
|
|
|
validatorAddr, err := sdk.AccAddressFromBech32(msg.ValidatorAddr)
|
|
if err != nil {
|
|
w.WriteHeader(http.StatusInternalServerError)
|
|
w.Write([]byte(fmt.Sprintf("Couldn't decode validator. Error: %s", err.Error())))
|
|
return
|
|
}
|
|
|
|
if !bytes.Equal(info.GetPubKey().Address(), delegatorAddr) {
|
|
w.WriteHeader(http.StatusUnauthorized)
|
|
w.Write([]byte("Must use own delegator address"))
|
|
return
|
|
}
|
|
|
|
messages[i] = stake.MsgCompleteUnbonding{
|
|
DelegatorAddr: delegatorAddr,
|
|
ValidatorAddr: validatorAddr,
|
|
}
|
|
|
|
i++
|
|
}
|
|
|
|
txCtx := authcliCtx.TxContext{
|
|
Codec: cdc,
|
|
ChainID: m.ChainID,
|
|
Gas: m.Gas,
|
|
}
|
|
|
|
// sign messages
|
|
signedTxs := make([][]byte, len(messages[:]))
|
|
for i, msg := range messages {
|
|
// increment sequence for each message
|
|
txCtx = txCtx.WithAccountNumber(m.AccountNumber)
|
|
txCtx = txCtx.WithSequence(m.Sequence)
|
|
|
|
m.Sequence++
|
|
|
|
txBytes, err := txCtx.BuildAndSign(m.LocalAccountName, m.Password, []sdk.Msg{msg})
|
|
if err != nil {
|
|
w.WriteHeader(http.StatusUnauthorized)
|
|
w.Write([]byte(err.Error()))
|
|
return
|
|
}
|
|
|
|
signedTxs[i] = txBytes
|
|
}
|
|
|
|
// send
|
|
// XXX the operation might not be atomic if a tx fails
|
|
// should we have a sdk.MultiMsg type to make sending atomic?
|
|
results := make([]*ctypes.ResultBroadcastTxCommit, len(signedTxs[:]))
|
|
for i, txBytes := range signedTxs {
|
|
res, err := cliCtx.BroadcastTx(txBytes)
|
|
if err != nil {
|
|
w.WriteHeader(http.StatusInternalServerError)
|
|
w.Write([]byte(err.Error()))
|
|
return
|
|
}
|
|
|
|
results[i] = res
|
|
}
|
|
|
|
output, err := wire.MarshalJSONIndent(cdc, results[:])
|
|
if err != nil {
|
|
w.WriteHeader(http.StatusInternalServerError)
|
|
w.Write([]byte(err.Error()))
|
|
return
|
|
}
|
|
|
|
w.Write(output)
|
|
}
|
|
}
|