Files
ark/server/internal/interface/grpc/handlers/parser.go
Pietralberto Mazza 403a82e25e Rename vtxo is_oor to is_pending (#385)
* Rename vtxo is_oor > is_pending

* Clean swaggers
2024-11-20 02:36:35 +01:00

292 lines
6.9 KiB
Go

package handlers
import (
"encoding/hex"
"fmt"
arkv1 "github.com/ark-network/ark/api-spec/protobuf/gen/ark/v1"
"github.com/ark-network/ark/common"
"github.com/ark-network/ark/common/note"
"github.com/ark-network/ark/common/tree"
"github.com/ark-network/ark/server/internal/core/application"
"github.com/ark-network/ark/server/internal/core/domain"
"github.com/ark-network/ark/server/internal/core/ports"
"github.com/btcsuite/btcd/btcec/v2/schnorr"
"github.com/btcsuite/btcd/chaincfg/chainhash"
"github.com/btcsuite/btcd/txscript"
)
// From interface type to app type
func parseAddress(addr string) (*common.Address, error) {
if len(addr) <= 0 {
return nil, fmt.Errorf("missing address")
}
return common.DecodeAddress(addr)
}
func parseAsyncPaymentInputs(ins []*arkv1.AsyncPaymentInput) ([]application.AsyncPaymentInput, error) {
if len(ins) <= 0 {
return nil, fmt.Errorf("missing inputs")
}
inputs := make([]application.AsyncPaymentInput, 0, len(ins))
for _, input := range ins {
forfeitLeafHash, err := chainhash.NewHashFromStr(input.GetForfeitLeafHash())
if err != nil {
return nil, fmt.Errorf("invalid forfeit leaf hash: %s", err)
}
inputs = append(inputs, application.AsyncPaymentInput{
Input: ports.Input{
VtxoKey: domain.VtxoKey{
Txid: input.GetInput().GetOutpoint().GetTxid(),
VOut: input.GetInput().GetOutpoint().GetVout(),
},
Descriptor: input.GetInput().GetDescriptor_(),
},
ForfeitLeafHash: *forfeitLeafHash,
})
}
return inputs, nil
}
func parseNotes(notes []string) ([]note.Note, error) {
if len(notes) <= 0 {
return nil, fmt.Errorf("missing notes")
}
notesParsed := make([]note.Note, 0, len(notes))
for _, noteStr := range notes {
n, err := note.NewFromString(noteStr)
if err != nil {
return nil, fmt.Errorf("invalid note: %s", err)
}
notesParsed = append(notesParsed, *n)
}
return notesParsed, nil
}
func parseInputs(ins []*arkv1.Input) ([]ports.Input, error) {
if len(ins) <= 0 {
return nil, fmt.Errorf("missing inputs")
}
inputs := make([]ports.Input, 0, len(ins))
for _, input := range ins {
inputs = append(inputs, ports.Input{
VtxoKey: domain.VtxoKey{
Txid: input.GetOutpoint().GetTxid(),
VOut: input.GetOutpoint().GetVout(),
},
Descriptor: input.GetDescriptor_(),
})
}
return inputs, nil
}
func parseReceiver(out *arkv1.Output) (domain.Receiver, error) {
decodedAddr, err := common.DecodeAddress(out.GetAddress())
if err != nil {
// onchain address
return domain.Receiver{
Amount: out.GetAmount(),
OnchainAddress: out.GetAddress(),
}, nil
}
return domain.Receiver{
Amount: out.GetAmount(),
Pubkey: hex.EncodeToString(schnorr.SerializePubKey(decodedAddr.VtxoTapKey)),
}, nil
}
func parseReceivers(outs []*arkv1.Output) ([]domain.Receiver, error) {
receivers := make([]domain.Receiver, 0, len(outs))
for _, out := range outs {
if out.GetAmount() == 0 {
return nil, fmt.Errorf("missing output amount")
}
if len(out.GetAddress()) <= 0 {
return nil, fmt.Errorf("missing output destination")
}
rcv, err := parseReceiver(out)
if err != nil {
return nil, err
}
receivers = append(receivers, rcv)
}
return receivers, nil
}
// From app type to interface type
type vtxoList []domain.Vtxo
func (v vtxoList) toProto() []*arkv1.Vtxo {
list := make([]*arkv1.Vtxo, 0, len(v))
for _, vv := range v {
list = append(list, &arkv1.Vtxo{
Outpoint: &arkv1.Outpoint{
Txid: vv.Txid,
Vout: vv.VOut,
},
Amount: vv.Amount,
RoundTxid: vv.RoundTxid,
Spent: vv.Spent,
ExpireAt: vv.ExpireAt,
SpentBy: vv.SpentBy,
Swept: vv.Swept,
RedeemTx: vv.RedeemTx,
IsPending: len(vv.RedeemTx) > 0,
Pubkey: vv.Pubkey,
CreatedAt: vv.CreatedAt,
})
}
return list
}
type vtxoKeyList []domain.VtxoKey
func (v vtxoKeyList) toProto() []*arkv1.Outpoint {
list := make([]*arkv1.Outpoint, 0, len(v))
for _, vtxoKey := range v {
list = append(list, &arkv1.Outpoint{
Txid: vtxoKey.Txid,
Vout: vtxoKey.VOut,
})
}
return list
}
type congestionTree tree.CongestionTree
func (t congestionTree) toProto() *arkv1.Tree {
levels := make([]*arkv1.TreeLevel, 0, len(t))
for _, level := range t {
levelProto := &arkv1.TreeLevel{
Nodes: make([]*arkv1.Node, 0, len(level)),
}
for _, node := range level {
levelProto.Nodes = append(levelProto.Nodes, &arkv1.Node{
Txid: node.Txid,
Tx: node.Tx,
ParentTxid: node.ParentTxid,
})
}
levels = append(levels, levelProto)
}
return &arkv1.Tree{
Levels: levels,
}
}
type stage domain.Stage
func (s stage) toProto() arkv1.RoundStage {
if s.Failed {
return arkv1.RoundStage_ROUND_STAGE_FAILED
}
switch s.Code {
case domain.RegistrationStage:
return arkv1.RoundStage_ROUND_STAGE_REGISTRATION
case domain.FinalizationStage:
if s.Ended {
return arkv1.RoundStage_ROUND_STAGE_FINALIZED
}
return arkv1.RoundStage_ROUND_STAGE_FINALIZATION
default:
return arkv1.RoundStage_ROUND_STAGE_UNSPECIFIED
}
}
func parseSignedVtxoOutpoints(signedVtxoOutpoints []*arkv1.SignedVtxoOutpoint) ([]application.SignedVtxoOutpoint, error) {
if len(signedVtxoOutpoints) <= 0 {
return nil, fmt.Errorf("missing signed vtxo outpoints")
}
parsed := make([]application.SignedVtxoOutpoint, 0, len(signedVtxoOutpoints))
for _, signedVtxo := range signedVtxoOutpoints {
outpoint := signedVtxo.GetOutpoint()
if outpoint == nil {
return nil, fmt.Errorf("missing outpoint")
}
txid := outpoint.GetTxid()
if len(txid) <= 0 {
return nil, fmt.Errorf("missing txid")
}
proof := signedVtxo.GetProof()
if proof == nil {
return nil, fmt.Errorf("missing proof")
}
controlBlockHex := proof.GetControlBlock()
if len(controlBlockHex) <= 0 {
return nil, fmt.Errorf("missing control block")
}
controlBlockBytes, err := hex.DecodeString(controlBlockHex)
if err != nil {
return nil, fmt.Errorf("invalid control block: %s", err)
}
controlBlock, err := txscript.ParseControlBlock(controlBlockBytes)
if err != nil {
return nil, fmt.Errorf("invalid control block: %s", err)
}
signatureHex := proof.GetSignature()
if len(signatureHex) <= 0 {
return nil, fmt.Errorf("missing signature")
}
signatureBytes, err := hex.DecodeString(signatureHex)
if err != nil {
return nil, fmt.Errorf("invalid signature: %s", err)
}
signature, err := schnorr.ParseSignature(signatureBytes)
if err != nil {
return nil, fmt.Errorf("invalid signature: %s", err)
}
scriptHex := proof.GetScript()
if len(scriptHex) <= 0 {
return nil, fmt.Errorf("missing script")
}
scriptBytes, err := hex.DecodeString(scriptHex)
if err != nil {
return nil, fmt.Errorf("invalid script: %s", err)
}
vout := outpoint.GetVout()
parsed = append(parsed, application.SignedVtxoOutpoint{
Outpoint: domain.VtxoKey{
Txid: txid,
VOut: vout,
},
Proof: application.OwnershipProof{
ControlBlock: controlBlock,
Script: scriptBytes,
Signature: signature,
},
})
}
return parsed, nil
}