mirror of
https://github.com/aljazceru/ark.git
synced 2025-12-17 12:14:21 +01:00
Co-authored-by: Marco Argentieri <tiero@users.noreply.github.com> * Add claim command * Persist pending data in sqlite repo * Remove debug log * Return pending data at interface level * Fix unlocking btc wallet after restart * Lint & Fix whitelist permissions * Fix send command for covenant * Update client/covenantless/claim.go Signed-off-by: Marco Argentieri <3596602+tiero@users.noreply.github.com> * Fix * Pay for min relay fee instead of estimating fees for redeem and unconf forfeit txs * Add support for pending payments (coventanless) * Fixes * Fixes * Improve verbosity * Fix coin selection * Fix --------- Signed-off-by: Marco Argentieri <3596602+tiero@users.noreply.github.com> Co-authored-by: louisinger <louis@vulpem.com> Co-authored-by: Marco Argentieri <tiero@users.noreply.github.com> Co-authored-by: Marco Argentieri <3596602+tiero@users.noreply.github.com>
236 lines
4.9 KiB
Go
236 lines
4.9 KiB
Go
package bitcointree
|
|
|
|
import (
|
|
"bytes"
|
|
"fmt"
|
|
|
|
"github.com/ark-network/ark/common"
|
|
"github.com/btcsuite/btcd/btcec/v2/schnorr"
|
|
"github.com/btcsuite/btcd/txscript"
|
|
"github.com/decred/dcrd/dcrec/secp256k1/v4"
|
|
)
|
|
|
|
type Closure interface {
|
|
Leaf() (*txscript.TapLeaf, error)
|
|
Decode(script []byte) (bool, error)
|
|
}
|
|
|
|
type CSVSigClosure struct {
|
|
Pubkey *secp256k1.PublicKey
|
|
Seconds uint
|
|
}
|
|
|
|
type MultisigClosure struct {
|
|
Pubkey *secp256k1.PublicKey
|
|
AspPubkey *secp256k1.PublicKey
|
|
}
|
|
|
|
func DecodeClosure(script []byte) (Closure, error) {
|
|
var closure Closure
|
|
|
|
closure = &CSVSigClosure{}
|
|
if valid, err := closure.Decode(script); err == nil && valid {
|
|
return closure, nil
|
|
}
|
|
|
|
closure = &MultisigClosure{}
|
|
if valid, err := closure.Decode(script); err == nil && valid {
|
|
return closure, nil
|
|
}
|
|
|
|
return nil, fmt.Errorf("invalid closure script")
|
|
|
|
}
|
|
|
|
func (f *MultisigClosure) Leaf() (*txscript.TapLeaf, error) {
|
|
aspKeyBytes := schnorr.SerializePubKey(f.AspPubkey)
|
|
userKeyBytes := schnorr.SerializePubKey(f.Pubkey)
|
|
|
|
script, err := txscript.NewScriptBuilder().AddData(aspKeyBytes).
|
|
AddOp(txscript.OP_CHECKSIGVERIFY).AddData(userKeyBytes).
|
|
AddOp(txscript.OP_CHECKSIG).Script()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
tapLeaf := txscript.NewBaseTapLeaf(script)
|
|
return &tapLeaf, nil
|
|
}
|
|
|
|
func (f *MultisigClosure) Decode(script []byte) (bool, error) {
|
|
valid, aspPubKey, err := decodeChecksigScript(script)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
|
|
if !valid {
|
|
return false, nil
|
|
}
|
|
|
|
valid, pubkey, err := decodeChecksigScript(script[33:])
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
|
|
if !valid {
|
|
return false, nil
|
|
}
|
|
|
|
f.Pubkey = pubkey
|
|
f.AspPubkey = aspPubKey
|
|
|
|
rebuilt, err := f.Leaf()
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
|
|
if !bytes.Equal(rebuilt.Script, script) {
|
|
return false, nil
|
|
}
|
|
|
|
return true, nil
|
|
}
|
|
|
|
func (d *CSVSigClosure) Leaf() (*txscript.TapLeaf, error) {
|
|
script, err := encodeCsvWithChecksigScript(d.Pubkey, d.Seconds)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
tapLeaf := txscript.NewBaseTapLeaf(script)
|
|
return &tapLeaf, nil
|
|
}
|
|
|
|
func (d *CSVSigClosure) Decode(script []byte) (bool, error) {
|
|
csvIndex := bytes.Index(
|
|
script, []byte{txscript.OP_CHECKSEQUENCEVERIFY, txscript.OP_DROP},
|
|
)
|
|
if csvIndex == -1 || csvIndex == 0 {
|
|
return false, nil
|
|
}
|
|
|
|
sequence := script[1:csvIndex]
|
|
|
|
seconds, err := common.BIP68Decode(sequence)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
|
|
checksigScript := script[csvIndex+2:]
|
|
valid, pubkey, err := decodeChecksigScript(checksigScript)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
|
|
if !valid {
|
|
return false, nil
|
|
}
|
|
|
|
rebuilt, err := encodeCsvWithChecksigScript(pubkey, seconds)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
|
|
if !bytes.Equal(rebuilt, script) {
|
|
return false, nil
|
|
}
|
|
|
|
d.Pubkey = pubkey
|
|
d.Seconds = seconds
|
|
|
|
return valid, nil
|
|
}
|
|
|
|
func ComputeVtxoTaprootScript(
|
|
userPubkey, aspPubkey *secp256k1.PublicKey, exitDelay uint,
|
|
) (*secp256k1.PublicKey, *txscript.TapscriptProof, error) {
|
|
redeemClosure := &CSVSigClosure{
|
|
Pubkey: userPubkey,
|
|
Seconds: exitDelay,
|
|
}
|
|
|
|
forfeitClosure := &MultisigClosure{
|
|
Pubkey: userPubkey,
|
|
AspPubkey: aspPubkey,
|
|
}
|
|
|
|
redeemLeaf, err := redeemClosure.Leaf()
|
|
if err != nil {
|
|
return nil, nil, err
|
|
}
|
|
|
|
forfeitLeaf, err := forfeitClosure.Leaf()
|
|
if err != nil {
|
|
return nil, nil, err
|
|
}
|
|
|
|
vtxoTaprootTree := txscript.AssembleTaprootScriptTree(
|
|
*redeemLeaf, *forfeitLeaf,
|
|
)
|
|
root := vtxoTaprootTree.RootNode.TapHash()
|
|
|
|
unspendableKey := UnspendableKey()
|
|
vtxoTaprootKey := txscript.ComputeTaprootOutputKey(unspendableKey, root[:])
|
|
|
|
redeemLeafHash := redeemLeaf.TapHash()
|
|
proofIndex := vtxoTaprootTree.LeafProofIndex[redeemLeafHash]
|
|
proof := vtxoTaprootTree.LeafMerkleProofs[proofIndex]
|
|
|
|
return vtxoTaprootKey, &proof, nil
|
|
}
|
|
|
|
func decodeChecksigScript(script []byte) (bool, *secp256k1.PublicKey, error) {
|
|
data32Index := bytes.Index(script, []byte{txscript.OP_DATA_32})
|
|
if data32Index == -1 {
|
|
return false, nil, nil
|
|
}
|
|
|
|
key := script[data32Index+1 : data32Index+33]
|
|
if len(key) != 32 {
|
|
return false, nil, nil
|
|
}
|
|
|
|
pubkey, err := schnorr.ParsePubKey(key)
|
|
if err != nil {
|
|
return false, nil, err
|
|
}
|
|
|
|
return true, pubkey, nil
|
|
}
|
|
|
|
// checkSequenceVerifyScript without checksig
|
|
func encodeCsvScript(seconds uint) ([]byte, error) {
|
|
sequence, err := common.BIP68Encode(seconds)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return txscript.NewScriptBuilder().AddData(sequence).AddOps([]byte{
|
|
txscript.OP_CHECKSEQUENCEVERIFY,
|
|
txscript.OP_DROP,
|
|
}).Script()
|
|
}
|
|
|
|
// checkSequenceVerifyScript + checksig
|
|
func encodeCsvWithChecksigScript(
|
|
pubkey *secp256k1.PublicKey, seconds uint,
|
|
) ([]byte, error) {
|
|
script, err := encodeChecksigScript(pubkey)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
csvScript, err := encodeCsvScript(seconds)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return append(csvScript, script...), nil
|
|
}
|
|
|
|
func encodeChecksigScript(pubkey *secp256k1.PublicKey) ([]byte, error) {
|
|
key := schnorr.SerializePubKey(pubkey)
|
|
return txscript.NewScriptBuilder().AddData(key).
|
|
AddOp(txscript.OP_CHECKSIG).Script()
|
|
}
|