lnd/itest/lnd_psbt_test.go

1343 lines
43 KiB
Go
Raw Normal View History

2020-03-31 09:13:21 +02:00
package itest
import (
"bytes"
"testing"
"time"
2022-04-28 13:39:28 +02:00
"github.com/btcsuite/btcd/btcec/v2"
"github.com/btcsuite/btcd/btcec/v2/ecdsa"
2022-04-28 13:39:28 +02:00
"github.com/btcsuite/btcd/btcec/v2/schnorr"
"github.com/btcsuite/btcd/btcutil"
"github.com/btcsuite/btcd/btcutil/hdkeychain"
"github.com/btcsuite/btcd/btcutil/psbt"
"github.com/btcsuite/btcd/txscript"
2020-03-31 09:13:21 +02:00
"github.com/btcsuite/btcd/wire"
"github.com/lightningnetwork/lnd/funding"
2022-04-28 13:39:28 +02:00
"github.com/lightningnetwork/lnd/input"
"github.com/lightningnetwork/lnd/keychain"
2020-03-31 09:13:21 +02:00
"github.com/lightningnetwork/lnd/lnrpc"
2022-01-05 11:04:24 +01:00
"github.com/lightningnetwork/lnd/lnrpc/signrpc"
"github.com/lightningnetwork/lnd/lnrpc/walletrpc"
"github.com/lightningnetwork/lnd/lntest"
"github.com/lightningnetwork/lnd/lntest/node"
"github.com/stretchr/testify/require"
2020-03-31 09:13:21 +02:00
)
// testPsbtChanFunding makes sure a channel can be opened between carol and dave
// by using a Partially Signed Bitcoin Transaction that funds the channel
// multisig funding output.
func testPsbtChanFunding(ht *lntest.HarnessTest) {
const (
burnAddr = "bcrt1qxsnqpdc842lu8c0xlllgvejt6rhy49u6fmpgyz"
)
testCases := []struct {
name string
commitmentType lnrpc.CommitmentType
private bool
}{
{
name: "anchors",
commitmentType: lnrpc.CommitmentType_ANCHORS,
private: false,
},
{
name: "simple taproot",
commitmentType: lnrpc.CommitmentType_SIMPLE_TAPROOT,
// Set this to true once simple taproot channels can be
// announced to the network.
private: true,
},
}
for _, tc := range testCases {
tc := tc
success := ht.T.Run(tc.name, func(tt *testing.T) {
st := ht.Subtest(tt)
args := lntest.NodeArgsForCommitType(tc.commitmentType)
// First, we'll create two new nodes that we'll use to
// open channels between for this test. Dave gets some
// coins that will be used to fund the PSBT, just to
// make sure that Carol has an empty wallet.
carol := st.NewNode("carol", args)
dave := st.NewNode("dave", args)
// We just send enough funds to satisfy the anchor
// channel reserve for 5 channels (50k sats).
st.FundCoins(50_000, carol)
st.FundCoins(50_000, dave)
st.RunTestCase(&lntest.TestCase{
Name: tc.name,
TestFunc: func(sst *lntest.HarnessTest) {
runPsbtChanFunding(
sst, carol, dave, tc.private,
tc.commitmentType,
)
},
})
// Empty out the wallets so there aren't any lingering
// coins.
sendAllCoinsConfirm(st, carol, burnAddr)
sendAllCoinsConfirm(st, dave, burnAddr)
// Now we test the second scenario. Again, we just send
// enough funds to satisfy the anchor channel reserve
// for 5 channels (50k sats).
st.FundCoins(50_000, carol)
st.FundCoins(50_000, dave)
st.RunTestCase(&lntest.TestCase{
Name: tc.name,
TestFunc: func(sst *lntest.HarnessTest) {
runPsbtChanFundingExternal(
sst, carol, dave, tc.private,
tc.commitmentType,
)
},
})
// Empty out the wallets a last time, so there aren't
// any lingering coins.
sendAllCoinsConfirm(st, carol, burnAddr)
sendAllCoinsConfirm(st, dave, burnAddr)
// The last test case tests the anchor channel reserve
// itself, so we need empty wallets.
st.RunTestCase(&lntest.TestCase{
Name: tc.name,
TestFunc: func(sst *lntest.HarnessTest) {
runPsbtChanFundingSingleStep(
sst, carol, dave, tc.private,
tc.commitmentType,
)
},
})
})
if !success {
// Log failure time to help relate the lnd logs to the
// failure.
ht.Logf("Failure time: %v", time.Now().Format(
"2006-01-02 15:04:05.000",
))
break
}
}
2021-10-14 15:42:59 +02:00
}
// runPsbtChanFunding makes sure a channel can be opened between carol and dave
// by using a Partially Signed Bitcoin Transaction that funds the channel
// multisig funding output.
func runPsbtChanFunding(ht *lntest.HarnessTest, carol, dave *node.HarnessNode,
private bool, commitType lnrpc.CommitmentType) {
2021-10-14 15:42:59 +02:00
const chanSize = funding.MaxBtcFundingAmount
2022-08-11 06:49:11 +02:00
ht.FundCoins(btcutil.SatoshiPerBitcoin, dave)
2020-03-31 09:13:21 +02:00
// Before we start the test, we'll ensure both sides are connected so
// the funding flow can be properly executed.
2022-08-11 06:49:11 +02:00
alice := ht.Alice
ht.EnsureConnected(carol, dave)
ht.EnsureConnected(carol, alice)
2020-03-31 09:13:21 +02:00
// At this point, we can begin our PSBT channel funding workflow. We'll
// start by generating a pending channel ID externally that will be used
// to track this new funding type.
2022-08-11 06:49:11 +02:00
pendingChanID := ht.Random32Bytes()
2020-03-31 09:13:21 +02:00
2020-07-09 02:13:41 +02:00
// We'll also test batch funding of two channels so we need another ID.
2022-08-11 06:49:11 +02:00
pendingChanID2 := ht.Random32Bytes()
2020-07-09 02:13:41 +02:00
2020-03-31 09:13:21 +02:00
// Now that we have the pending channel ID, Carol will open the channel
2020-07-09 02:13:41 +02:00
// by specifying a PSBT shim. We use the NoPublish flag here to avoid
// publishing the whole batch TX too early.
2022-08-11 06:49:11 +02:00
chanUpdates, tempPsbt := ht.OpenChannelPsbt(
carol, dave, lntest.OpenChannelParams{
2020-03-31 09:13:21 +02:00
Amt: chanSize,
FundingShim: &lnrpc.FundingShim{
Shim: &lnrpc.FundingShim_PsbtShim{
PsbtShim: &lnrpc.PsbtShim{
2022-08-11 06:49:11 +02:00
PendingChanId: pendingChanID,
2020-07-09 02:13:41 +02:00
NoPublish: true,
2020-03-31 09:13:21 +02:00
},
},
},
Private: private,
CommitmentType: commitType,
2020-03-31 09:13:21 +02:00
},
)
// Let's add a second channel to the batch. This time between Carol and
// Alice. We will publish the batch TX once this channel funding is
// complete.
2022-08-11 06:49:11 +02:00
chanUpdates2, psbtBytes2 := ht.OpenChannelPsbt(
carol, alice, lntest.OpenChannelParams{
2020-07-09 02:13:41 +02:00
Amt: chanSize,
FundingShim: &lnrpc.FundingShim{
Shim: &lnrpc.FundingShim_PsbtShim{
PsbtShim: &lnrpc.PsbtShim{
2022-08-11 06:49:11 +02:00
PendingChanId: pendingChanID2,
2020-07-09 02:13:41 +02:00
NoPublish: false,
BasePsbt: tempPsbt,
2020-07-09 02:13:41 +02:00
},
},
},
// We haven't started Alice with the explicit params to
// support the current commit type, so we'll just use
// the default for this channel. That also allows us to
// test batches of different channel types.
2020-07-09 02:13:41 +02:00
},
)
2020-03-31 09:13:21 +02:00
// We'll now ask Dave's wallet to fund the PSBT for us. This will return
// a packet with inputs and outputs set but without any witness data.
// This is exactly what we need for the next step.
fundReq := &walletrpc.FundPsbtRequest{
Template: &walletrpc.FundPsbtRequest_Psbt{
Psbt: psbtBytes2,
},
Fees: &walletrpc.FundPsbtRequest_SatPerVbyte{
SatPerVbyte: 2,
},
2020-03-31 09:13:21 +02:00
}
2022-08-11 06:49:11 +02:00
fundResp := dave.RPC.FundPsbt(fundReq)
2020-03-31 09:13:21 +02:00
// We have a PSBT that has no witness data yet, which is exactly what we
2020-07-09 02:13:41 +02:00
// need for the next step: Verify the PSBT with the funding intents.
2022-08-11 06:49:11 +02:00
carol.RPC.FundingStateStep(&lnrpc.FundingTransitionMsg{
2020-03-31 09:13:21 +02:00
Trigger: &lnrpc.FundingTransitionMsg_PsbtVerify{
PsbtVerify: &lnrpc.FundingPsbtVerify{
2022-08-11 06:49:11 +02:00
PendingChanId: pendingChanID,
FundedPsbt: fundResp.FundedPsbt,
2020-03-31 09:13:21 +02:00
},
},
})
2022-08-11 06:49:11 +02:00
carol.RPC.FundingStateStep(&lnrpc.FundingTransitionMsg{
2020-07-09 02:13:41 +02:00
Trigger: &lnrpc.FundingTransitionMsg_PsbtVerify{
PsbtVerify: &lnrpc.FundingPsbtVerify{
2022-08-11 06:49:11 +02:00
PendingChanId: pendingChanID2,
FundedPsbt: fundResp.FundedPsbt,
2020-07-09 02:13:41 +02:00
},
},
})
2020-03-31 09:13:21 +02:00
// Now we'll ask Dave's wallet to sign the PSBT so we can finish the
// funding flow.
finalizeReq := &walletrpc.FinalizePsbtRequest{
FundedPsbt: fundResp.FundedPsbt,
2020-03-31 09:13:21 +02:00
}
2022-08-11 06:49:11 +02:00
finalizeRes := dave.RPC.FinalizePsbt(finalizeReq)
2020-03-31 09:13:21 +02:00
// We've signed our PSBT now, let's pass it to the intent again.
2022-08-11 06:49:11 +02:00
carol.RPC.FundingStateStep(&lnrpc.FundingTransitionMsg{
2020-03-31 09:13:21 +02:00
Trigger: &lnrpc.FundingTransitionMsg_PsbtFinalize{
PsbtFinalize: &lnrpc.FundingPsbtFinalize{
2022-08-11 06:49:11 +02:00
PendingChanId: pendingChanID,
SignedPsbt: finalizeRes.SignedPsbt,
2020-03-31 09:13:21 +02:00
},
},
})
// Consume the "channel pending" update. This waits until the funding
2020-07-09 02:13:41 +02:00
// transaction was fully compiled.
2022-08-11 06:49:11 +02:00
updateResp := ht.ReceiveOpenChannelUpdate(chanUpdates)
2020-03-31 09:13:21 +02:00
upd, ok := updateResp.Update.(*lnrpc.OpenStatusUpdate_ChanPending)
2022-08-11 06:49:11 +02:00
require.True(ht, ok)
2020-03-31 09:13:21 +02:00
chanPoint := &lnrpc.ChannelPoint{
FundingTxid: &lnrpc.ChannelPoint_FundingTxidBytes{
FundingTxidBytes: upd.ChanPending.Txid,
},
OutputIndex: upd.ChanPending.OutputIndex,
}
2020-07-09 02:13:41 +02:00
// No transaction should have been published yet.
2022-08-11 06:49:11 +02:00
ht.Miner.AssertNumTxsInMempool(0)
2020-07-09 02:13:41 +02:00
// Let's progress the second channel now. This time we'll use the raw
// wire format transaction directly.
2022-08-11 06:49:11 +02:00
carol.RPC.FundingStateStep(&lnrpc.FundingTransitionMsg{
2020-07-09 02:13:41 +02:00
Trigger: &lnrpc.FundingTransitionMsg_PsbtFinalize{
PsbtFinalize: &lnrpc.FundingPsbtFinalize{
2022-08-11 06:49:11 +02:00
PendingChanId: pendingChanID2,
FinalRawTx: finalizeRes.RawFinalTx,
2020-07-09 02:13:41 +02:00
},
},
})
// Consume the "channel pending" update for the second channel. This
// waits until the funding transaction was fully compiled and in this
// case published.
2022-08-11 06:49:11 +02:00
updateResp2 := ht.ReceiveOpenChannelUpdate(chanUpdates2)
2020-07-09 02:13:41 +02:00
upd2, ok := updateResp2.Update.(*lnrpc.OpenStatusUpdate_ChanPending)
2022-08-11 06:49:11 +02:00
require.True(ht, ok)
2020-07-09 02:13:41 +02:00
chanPoint2 := &lnrpc.ChannelPoint{
FundingTxid: &lnrpc.ChannelPoint_FundingTxidBytes{
FundingTxidBytes: upd2.ChanPending.Txid,
},
OutputIndex: upd2.ChanPending.OutputIndex,
}
2020-03-31 09:13:21 +02:00
// Great, now we can mine a block to get the transaction confirmed, then
// wait for the new channel to be propagated through the network.
var finalTx wire.MsgTx
2022-08-11 06:49:11 +02:00
err := finalTx.Deserialize(bytes.NewReader(finalizeRes.RawFinalTx))
require.NoError(ht, err)
txHash := finalTx.TxHash()
2022-08-11 06:49:11 +02:00
block := ht.MineBlocksAndAssertNumTxes(6, 1)[0]
ht.Miner.AssertTxInBlock(block, &txHash)
ht.AssertTopologyChannelOpen(carol, chanPoint)
ht.AssertTopologyChannelOpen(carol, chanPoint2)
2020-03-31 09:13:21 +02:00
// With the channel open, ensure that it is counted towards Carol's
// total channel balance.
2022-08-11 06:49:11 +02:00
balRes := carol.RPC.ChannelBalance()
require.NotZero(ht, balRes.LocalBalance.Sat)
2020-03-31 09:13:21 +02:00
// Next, to make sure the channel functions as normal, we'll make some
// payments within the channel.
payAmt := btcutil.Amount(100000)
invoice := &lnrpc.Invoice{
Memo: "new chans",
Value: int64(payAmt),
}
2022-08-11 06:49:11 +02:00
resp := dave.RPC.AddInvoice(invoice)
ht.CompletePaymentRequests(carol, []string{resp.PaymentRequest})
// TODO(yy): remove the sleep once the following bug is fixed. When the
// payment is reported as settled by Carol, it's expected the
// commitment dance is finished and all subsequent states have been
// updated. Yet we'd receive the error `cannot co-op close channel with
// active htlcs` or `link failed to shutdown` if we close the channel.
// We need to investigate the order of settling the payments and
// updating commitments to understand and fix .
time.Sleep(2 * time.Second)
2020-03-31 09:13:21 +02:00
// To conclude, we'll close the newly created channel between Carol and
// Dave. This function will also block until the channel is closed and
// will additionally assert the relevant channel closing post
// conditions.
2022-08-11 06:49:11 +02:00
ht.CloseChannel(carol, chanPoint)
ht.CloseChannel(carol, chanPoint2)
2020-03-31 09:13:21 +02:00
}
// runPsbtChanFundingExternal makes sure a channel can be opened between carol
// and dave by using a Partially Signed Bitcoin Transaction that funds the
// channel multisig funding output and is fully funded by an external third
// party.
func runPsbtChanFundingExternal(ht *lntest.HarnessTest, carol,
dave *node.HarnessNode, private bool, commitType lnrpc.CommitmentType) {
const chanSize = funding.MaxBtcFundingAmount
// Before we start the test, we'll ensure both sides are connected so
// the funding flow can be properly executed.
alice := ht.Alice
ht.EnsureConnected(carol, dave)
ht.EnsureConnected(carol, alice)
// At this point, we can begin our PSBT channel funding workflow. We'll
// start by generating a pending channel ID externally that will be used
// to track this new funding type.
pendingChanID := ht.Random32Bytes()
// We'll also test batch funding of two channels so we need another ID.
pendingChanID2 := ht.Random32Bytes()
// Now that we have the pending channel ID, Carol will open the channel
// by specifying a PSBT shim. We use the NoPublish flag here to avoid
// publishing the whole batch TX too early.
chanUpdates, tempPsbt := ht.OpenChannelPsbt(
carol, dave, lntest.OpenChannelParams{
Amt: chanSize,
FundingShim: &lnrpc.FundingShim{
Shim: &lnrpc.FundingShim_PsbtShim{
PsbtShim: &lnrpc.PsbtShim{
PendingChanId: pendingChanID,
NoPublish: true,
},
},
},
Private: private,
CommitmentType: commitType,
},
)
// Let's add a second channel to the batch. This time between Carol and
// Alice. We will publish the batch TX once this channel funding is
// complete.
chanUpdates2, psbtBytes2 := ht.OpenChannelPsbt(
carol, alice, lntest.OpenChannelParams{
Amt: chanSize,
FundingShim: &lnrpc.FundingShim{
Shim: &lnrpc.FundingShim_PsbtShim{
PsbtShim: &lnrpc.PsbtShim{
PendingChanId: pendingChanID2,
NoPublish: true,
BasePsbt: tempPsbt,
},
},
},
// We haven't started Alice with the explicit params to
// support the current commit type, so we'll just use
// the default for this channel. That also allows us to
// test batches of different channel types.
},
)
// We'll now ask Alice's wallet to fund the PSBT for us. This will
// return a packet with inputs and outputs set but without any witness
// data. This is exactly what we need for the next step.
fundReq := &walletrpc.FundPsbtRequest{
Template: &walletrpc.FundPsbtRequest_Psbt{
Psbt: psbtBytes2,
},
Fees: &walletrpc.FundPsbtRequest_SatPerVbyte{
SatPerVbyte: 2,
},
}
fundResp := alice.RPC.FundPsbt(fundReq)
// We have a PSBT that has no witness data yet, which is exactly what we
// need for the next step: Verify the PSBT with the funding intents.
// We tell the PSBT intent to skip the finalize step because we know the
// final transaction will not be broadcast by Carol herself but by
// Alice. And we assume that Alice is a third party that is not in
// direct communication with Carol and won't send the signed TX to her
// before broadcasting it. So we cannot call the finalize step but
// instead just tell lnd to wait for a TX to be published/confirmed.
carol.RPC.FundingStateStep(&lnrpc.FundingTransitionMsg{
Trigger: &lnrpc.FundingTransitionMsg_PsbtVerify{
PsbtVerify: &lnrpc.FundingPsbtVerify{
PendingChanId: pendingChanID,
FundedPsbt: fundResp.FundedPsbt,
SkipFinalize: true,
},
},
})
carol.RPC.FundingStateStep(&lnrpc.FundingTransitionMsg{
Trigger: &lnrpc.FundingTransitionMsg_PsbtVerify{
PsbtVerify: &lnrpc.FundingPsbtVerify{
PendingChanId: pendingChanID2,
FundedPsbt: fundResp.FundedPsbt,
SkipFinalize: true,
},
},
})
// Consume the "channel pending" update. This waits until the funding
// transaction was fully compiled for both channels.
updateResp := ht.ReceiveOpenChannelUpdate(chanUpdates)
upd, ok := updateResp.Update.(*lnrpc.OpenStatusUpdate_ChanPending)
require.True(ht, ok)
chanPoint := &lnrpc.ChannelPoint{
FundingTxid: &lnrpc.ChannelPoint_FundingTxidBytes{
FundingTxidBytes: upd.ChanPending.Txid,
},
OutputIndex: upd.ChanPending.OutputIndex,
}
updateResp2 := ht.ReceiveOpenChannelUpdate(chanUpdates2)
upd2, ok := updateResp2.Update.(*lnrpc.OpenStatusUpdate_ChanPending)
require.True(ht, ok)
chanPoint2 := &lnrpc.ChannelPoint{
FundingTxid: &lnrpc.ChannelPoint_FundingTxidBytes{
FundingTxidBytes: upd2.ChanPending.Txid,
},
OutputIndex: upd2.ChanPending.OutputIndex,
}
ht.AssertNumPendingOpenChannels(carol, 2)
// Now we'll ask Alice's wallet to sign the PSBT so we can finish the
// funding flow.
finalizeReq := &walletrpc.FinalizePsbtRequest{
FundedPsbt: fundResp.FundedPsbt,
}
finalizeRes := alice.RPC.FinalizePsbt(finalizeReq)
// No transaction should have been published yet.
ht.Miner.AssertNumTxsInMempool(0)
// Great, now let's publish the final raw transaction.
var finalTx wire.MsgTx
err := finalTx.Deserialize(bytes.NewReader(finalizeRes.RawFinalTx))
require.NoError(ht, err)
txHash := finalTx.TxHash()
_, err = ht.Miner.Client.SendRawTransaction(&finalTx, false)
require.NoError(ht, err)
// Now we can mine a block to get the transaction confirmed, then wait
// for the new channel to be propagated through the network.
block := ht.MineBlocksAndAssertNumTxes(6, 1)[0]
ht.Miner.AssertTxInBlock(block, &txHash)
ht.AssertTopologyChannelOpen(carol, chanPoint)
ht.AssertTopologyChannelOpen(carol, chanPoint2)
// With the channel open, ensure that it is counted towards Carol's
// total channel balance.
balRes := carol.RPC.ChannelBalance()
require.NotZero(ht, balRes.LocalBalance.Sat)
// Next, to make sure the channel functions as normal, we'll make some
// payments within the channel.
payAmt := btcutil.Amount(100000)
invoice := &lnrpc.Invoice{
Memo: "new chans",
Value: int64(payAmt),
}
resp := dave.RPC.AddInvoice(invoice)
ht.CompletePaymentRequests(carol, []string{resp.PaymentRequest})
// TODO(yy): remove the sleep once the following bug is fixed. When the
// payment is reported as settled by Carol, it's expected the
// commitment dance is finished and all subsequent states have been
// updated. Yet we'd receive the error `cannot co-op close channel with
// active htlcs` or `link failed to shutdown` if we close the channel.
// We need to investigate the order of settling the payments and
// updating commitments to understand and fix .
time.Sleep(2 * time.Second)
// To conclude, we'll close the newly created channel between Carol and
// Dave. This function will also block until the channels are closed and
// will additionally assert the relevant channel closing post
// conditions.
ht.CloseChannel(carol, chanPoint)
ht.CloseChannel(carol, chanPoint2)
}
// runPsbtChanFundingSingleStep checks whether PSBT funding works also when
// the wallet of both nodes are empty and one of them uses PSBT and an external
// wallet to fund the channel while creating reserve output in the same
// transaction.
func runPsbtChanFundingSingleStep(ht *lntest.HarnessTest, carol,
dave *node.HarnessNode, private bool, commitType lnrpc.CommitmentType) {
const chanSize = funding.MaxBtcFundingAmount
alice := ht.Alice
ht.FundCoins(btcutil.SatoshiPerBitcoin, alice)
// Get new address for anchor reserve.
req := &lnrpc.NewAddressRequest{
Type: lnrpc.AddressType_WITNESS_PUBKEY_HASH,
}
addrResp := carol.RPC.NewAddress(req)
reserveAddr, err := btcutil.DecodeAddress(
addrResp.Address, harnessNetParams,
)
require.NoError(ht, err)
reserveAddrScript, err := txscript.PayToAddrScript(reserveAddr)
require.NoError(ht, err)
// Before we start the test, we'll ensure both sides are connected so
// the funding flow can be properly executed.
ht.EnsureConnected(carol, dave)
// At this point, we can begin our PSBT channel funding workflow. We'll
// start by generating a pending channel ID externally that will be used
// to track this new funding type.
pendingChanID := ht.Random32Bytes()
// Now that we have the pending channel ID, Carol will open the channel
// by specifying a PSBT shim.
chanUpdates, tempPsbt := ht.OpenChannelPsbt(
carol, dave, lntest.OpenChannelParams{
Amt: chanSize,
FundingShim: &lnrpc.FundingShim{
Shim: &lnrpc.FundingShim_PsbtShim{
PsbtShim: &lnrpc.PsbtShim{
PendingChanId: pendingChanID,
NoPublish: false,
},
},
},
Private: private,
CommitmentType: commitType,
},
)
decodedPsbt, err := psbt.NewFromRawBytes(
bytes.NewReader(tempPsbt), false,
)
require.NoError(ht, err)
reserveTxOut := wire.TxOut{
Value: 10000,
PkScript: reserveAddrScript,
}
decodedPsbt.UnsignedTx.TxOut = append(
decodedPsbt.UnsignedTx.TxOut, &reserveTxOut,
)
decodedPsbt.Outputs = append(decodedPsbt.Outputs, psbt.POutput{})
var psbtBytes bytes.Buffer
err = decodedPsbt.Serialize(&psbtBytes)
require.NoError(ht, err)
fundReq := &walletrpc.FundPsbtRequest{
Template: &walletrpc.FundPsbtRequest_Psbt{
Psbt: psbtBytes.Bytes(),
},
Fees: &walletrpc.FundPsbtRequest_SatPerVbyte{
SatPerVbyte: 2,
},
}
fundResp := alice.RPC.FundPsbt(fundReq)
// Make sure the wallets are actually empty
ht.AssertNumUTXOsUnconfirmed(alice, 0)
ht.AssertNumUTXOsUnconfirmed(dave, 0)
// We have a PSBT that has no witness data yet, which is exactly what we
// need for the next step: Verify the PSBT with the funding intents.
carol.RPC.FundingStateStep(&lnrpc.FundingTransitionMsg{
Trigger: &lnrpc.FundingTransitionMsg_PsbtVerify{
PsbtVerify: &lnrpc.FundingPsbtVerify{
PendingChanId: pendingChanID,
FundedPsbt: fundResp.FundedPsbt,
},
},
})
// Now we'll ask Alice's wallet to sign the PSBT so we can finish the
// funding flow.
finalizeReq := &walletrpc.FinalizePsbtRequest{
FundedPsbt: fundResp.FundedPsbt,
}
finalizeRes := alice.RPC.FinalizePsbt(finalizeReq)
// We've signed our PSBT now, let's pass it to the intent again.
carol.RPC.FundingStateStep(&lnrpc.FundingTransitionMsg{
Trigger: &lnrpc.FundingTransitionMsg_PsbtFinalize{
PsbtFinalize: &lnrpc.FundingPsbtFinalize{
PendingChanId: pendingChanID,
SignedPsbt: finalizeRes.SignedPsbt,
},
},
})
// Consume the "channel pending" update. This waits until the funding
// transaction was fully compiled.
updateResp := ht.ReceiveOpenChannelUpdate(chanUpdates)
upd, ok := updateResp.Update.(*lnrpc.OpenStatusUpdate_ChanPending)
require.True(ht, ok)
chanPoint := &lnrpc.ChannelPoint{
FundingTxid: &lnrpc.ChannelPoint_FundingTxidBytes{
FundingTxidBytes: upd.ChanPending.Txid,
},
OutputIndex: upd.ChanPending.OutputIndex,
}
var finalTx wire.MsgTx
err = finalTx.Deserialize(bytes.NewReader(finalizeRes.RawFinalTx))
require.NoError(ht, err)
txHash := finalTx.TxHash()
block := ht.MineBlocksAndAssertNumTxes(6, 1)[0]
ht.Miner.AssertTxInBlock(block, &txHash)
ht.AssertTopologyChannelOpen(carol, chanPoint)
// Next, to make sure the channel functions as normal, we'll make some
// payments within the channel.
payAmt := btcutil.Amount(100000)
invoice := &lnrpc.Invoice{
Memo: "new chans",
Value: int64(payAmt),
}
resp := dave.RPC.AddInvoice(invoice)
ht.CompletePaymentRequests(carol, []string{resp.PaymentRequest})
// TODO(yy): remove the sleep once the following bug is fixed. When the
// payment is reported as settled by Carol, it's expected the
// commitment dance is finished and all subsequent states have been
// updated. Yet we'd receive the error `cannot co-op close channel with
// active htlcs` or `link failed to shutdown` if we close the channel.
// We need to investigate the order of settling the payments and
// updating commitments to understand and fix .
time.Sleep(2 * time.Second)
// To conclude, we'll close the newly created channel between Carol and
// Dave. This function will also block until the channel is closed and
// will additionally assert the relevant channel closing post
// conditions.
ht.CloseChannel(carol, chanPoint)
}
2022-04-28 13:19:50 +02:00
// testSignPsbt tests that the SignPsbt RPC works correctly.
func testSignPsbt(ht *lntest.HarnessTest) {
2022-08-11 07:07:33 +02:00
runSignPsbtSegWitV0P2WKH(ht, ht.Alice)
runSignPsbtSegWitV0NP2WKH(ht, ht.Alice)
runSignPsbtSegWitV1KeySpendBip86(ht, ht.Alice)
runSignPsbtSegWitV1KeySpendRootHash(ht, ht.Alice)
runSignPsbtSegWitV1ScriptSpend(ht, ht.Alice)
// The above tests all make sure we can sign for keys that aren't in
// the wallet. But we also want to make sure we can fund and then sign
// PSBTs from our wallet.
runFundAndSignPsbt(ht, ht.Alice)
2022-01-05 11:04:24 +01:00
}
2022-04-28 13:19:50 +02:00
// runSignPsbtSegWitV0P2WKH tests that the SignPsbt RPC works correctly for a
// SegWit v0 p2wkh input.
func runSignPsbtSegWitV0P2WKH(ht *lntest.HarnessTest, alice *node.HarnessNode) {
2022-01-05 11:04:24 +01:00
// We test that we can sign a PSBT that spends funds from an input that
// the wallet doesn't know about. To set up that test case, we first
// derive an address manually that the wallet won't be watching on
// chain. We can do that by exporting the account xpub of lnd's main
// account.
2022-08-11 07:07:33 +02:00
accounts := alice.RPC.ListAccounts(&walletrpc.ListAccountsRequest{})
require.NotEmpty(ht, accounts.Accounts)
2022-01-05 11:04:24 +01:00
// We also need to parse the accounts, so we have easy access to the
// parsed derivation paths.
parsedAccounts, err := walletrpc.AccountsToWatchOnly(accounts.Accounts)
2022-08-11 07:07:33 +02:00
require.NoError(ht, err)
2022-01-05 11:04:24 +01:00
account := parsedAccounts[0]
xpub, err := hdkeychain.NewKeyFromString(account.Xpub)
2022-08-11 07:07:33 +02:00
require.NoError(ht, err)
2022-01-05 11:04:24 +01:00
const (
changeIndex = 1
addrIndex = 1337
)
fullDerivationPath := []uint32{
hdkeychain.HardenedKeyStart + account.Purpose,
hdkeychain.HardenedKeyStart + account.CoinType,
hdkeychain.HardenedKeyStart + account.Account,
changeIndex,
addrIndex,
}
// Let's simulate a change address.
change, err := xpub.DeriveNonStandard(changeIndex) // nolint:staticcheck
2022-08-11 07:07:33 +02:00
require.NoError(ht, err)
2022-01-05 11:04:24 +01:00
// At an index that we are certainly not watching in the wallet.
addrKey, err := change.DeriveNonStandard(addrIndex) // nolint:staticcheck
2022-08-11 07:07:33 +02:00
require.NoError(ht, err)
2022-01-05 11:04:24 +01:00
addrPubKey, err := addrKey.ECPubKey()
2022-08-11 07:07:33 +02:00
require.NoError(ht, err)
2022-01-05 11:04:24 +01:00
pubKeyHash := btcutil.Hash160(addrPubKey.SerializeCompressed())
witnessAddr, err := btcutil.NewAddressWitnessPubKeyHash(
pubKeyHash, harnessNetParams,
)
2022-08-11 07:07:33 +02:00
require.NoError(ht, err)
2022-01-05 11:04:24 +01:00
pkScript, err := txscript.PayToAddrScript(witnessAddr)
2022-08-11 07:07:33 +02:00
require.NoError(ht, err)
2022-01-05 11:04:24 +01:00
2022-04-28 13:19:50 +02:00
// Send some funds to the output and then try to get a signature through
// the SignPsbt RPC to spend that output again.
assertPsbtSpend(
2022-08-11 07:07:33 +02:00
ht, alice, pkScript,
2022-04-28 13:19:50 +02:00
func(packet *psbt.Packet) {
in := &packet.Inputs[0]
in.Bip32Derivation = []*psbt.Bip32Derivation{{
PubKey: addrPubKey.SerializeCompressed(),
Bip32Path: fullDerivationPath,
}}
in.SighashType = txscript.SigHashAll
},
func(packet *psbt.Packet) {
2022-08-11 07:07:33 +02:00
require.Len(ht, packet.Inputs, 1)
require.Len(ht, packet.Inputs[0].PartialSigs, 1)
2022-04-28 13:19:50 +02:00
partialSig := packet.Inputs[0].PartialSigs[0]
require.Equal(
2022-08-11 07:07:33 +02:00
ht, partialSig.PubKey,
2022-04-28 13:19:50 +02:00
addrPubKey.SerializeCompressed(),
)
require.Greater(
2022-08-11 07:07:33 +02:00
ht, len(partialSig.Signature), ecdsa.MinSigLen,
2022-04-28 13:19:50 +02:00
)
},
)
}
// runSignPsbtSegWitV0NP2WKH tests that the SignPsbt RPC works correctly for a
// SegWit v0 np2wkh input.
func runSignPsbtSegWitV0NP2WKH(ht *lntest.HarnessTest,
2022-10-25 18:55:26 +02:00
alice *node.HarnessNode) {
// We test that we can sign a PSBT that spends funds from an input that
// the wallet doesn't know about. To set up that test case, we first
// derive an address manually that the wallet won't be watching on
// chain. We can do that by exporting the account xpub of lnd's main
// account.
2022-08-11 07:07:33 +02:00
accounts := alice.RPC.ListAccounts(&walletrpc.ListAccountsRequest{})
require.NotEmpty(ht, accounts.Accounts)
// We also need to parse the accounts, so we have easy access to the
// parsed derivation paths.
parsedAccounts, err := walletrpc.AccountsToWatchOnly(accounts.Accounts)
2022-08-11 07:07:33 +02:00
require.NoError(ht, err)
account := parsedAccounts[0]
xpub, err := hdkeychain.NewKeyFromString(account.Xpub)
2022-08-11 07:07:33 +02:00
require.NoError(ht, err)
const (
changeIndex = 1
addrIndex = 1337
)
fullDerivationPath := []uint32{
hdkeychain.HardenedKeyStart + account.Purpose,
hdkeychain.HardenedKeyStart + account.CoinType,
hdkeychain.HardenedKeyStart + account.Account,
changeIndex,
addrIndex,
}
// Let's simulate a change address.
change, err := xpub.DeriveNonStandard(changeIndex) // nolint:staticcheck
2022-08-11 07:07:33 +02:00
require.NoError(ht, err)
// At an index that we are certainly not watching in the wallet.
addrKey, err := change.DeriveNonStandard(addrIndex) // nolint:staticcheck
2022-08-11 07:07:33 +02:00
require.NoError(ht, err)
addrPubKey, err := addrKey.ECPubKey()
2022-08-11 07:07:33 +02:00
require.NoError(ht, err)
pubKeyHash := btcutil.Hash160(addrPubKey.SerializeCompressed())
witnessAddr, err := btcutil.NewAddressWitnessPubKeyHash(
pubKeyHash, harnessNetParams,
)
2022-08-11 07:07:33 +02:00
require.NoError(ht, err)
witnessProgram, err := txscript.PayToAddrScript(witnessAddr)
2022-08-11 07:07:33 +02:00
require.NoError(ht, err)
np2wkhAddr, err := btcutil.NewAddressScriptHash(
witnessProgram, harnessNetParams,
)
2022-08-11 07:07:33 +02:00
require.NoError(ht, err)
pkScript, err := txscript.PayToAddrScript(np2wkhAddr)
2022-08-11 07:07:33 +02:00
require.NoError(ht, err)
// Send some funds to the output and then try to get a signature through
// the SignPsbt RPC to spend that output again.
assertPsbtSpend(
2022-08-11 07:07:33 +02:00
ht, alice, pkScript,
func(packet *psbt.Packet) {
in := &packet.Inputs[0]
in.RedeemScript = witnessProgram
in.Bip32Derivation = []*psbt.Bip32Derivation{{
PubKey: addrPubKey.SerializeCompressed(),
Bip32Path: fullDerivationPath,
}}
in.SighashType = txscript.SigHashAll
},
func(packet *psbt.Packet) {
2022-08-11 07:07:33 +02:00
require.Len(ht, packet.Inputs, 1)
require.Len(ht, packet.Inputs[0].PartialSigs, 1)
partialSig := packet.Inputs[0].PartialSigs[0]
require.Equal(
2022-08-11 07:07:33 +02:00
ht, partialSig.PubKey,
addrPubKey.SerializeCompressed(),
)
require.Greater(
2022-08-11 07:07:33 +02:00
ht, len(partialSig.Signature), ecdsa.MinSigLen,
)
},
)
}
2022-04-28 13:39:28 +02:00
// runSignPsbtSegWitV1KeySpendBip86 tests that the SignPsbt RPC works correctly
// for a SegWit v1 p2tr key spend BIP-0086 input.
func runSignPsbtSegWitV1KeySpendBip86(ht *lntest.HarnessTest,
2022-08-11 07:07:33 +02:00
alice *node.HarnessNode) {
2022-04-28 13:39:28 +02:00
// Derive a key we can use for signing.
2022-08-11 07:07:33 +02:00
keyDesc, internalKey, fullDerivationPath := deriveInternalKey(ht, alice)
2022-04-28 13:39:28 +02:00
// Our taproot key is a BIP0086 key spend only construction that just
// commits to the internal key and no root hash.
taprootKey := txscript.ComputeTaprootKeyNoScript(internalKey)
tapScriptAddr, err := btcutil.NewAddressTaproot(
schnorr.SerializePubKey(taprootKey), harnessNetParams,
)
2022-08-11 07:07:33 +02:00
require.NoError(ht, err)
2022-04-28 13:39:28 +02:00
p2trPkScript, err := txscript.PayToAddrScript(tapScriptAddr)
2022-08-11 07:07:33 +02:00
require.NoError(ht, err)
2022-04-28 13:39:28 +02:00
// Send some funds to the output and then try to get a signature through
// the SignPsbt RPC to spend that output again.
assertPsbtSpend(
2022-08-11 07:07:33 +02:00
ht, alice, p2trPkScript,
2022-04-28 13:39:28 +02:00
func(packet *psbt.Packet) {
in := &packet.Inputs[0]
in.Bip32Derivation = []*psbt.Bip32Derivation{{
PubKey: keyDesc.RawKeyBytes,
Bip32Path: fullDerivationPath,
}}
in.TaprootBip32Derivation = []*psbt.TaprootBip32Derivation{{
XOnlyPubKey: keyDesc.RawKeyBytes[1:],
Bip32Path: fullDerivationPath,
}}
in.SighashType = txscript.SigHashDefault
},
func(packet *psbt.Packet) {
2022-08-11 07:07:33 +02:00
require.Len(ht, packet.Inputs, 1)
2022-04-28 13:39:28 +02:00
require.Len(
2022-08-11 07:07:33 +02:00
ht, packet.Inputs[0].TaprootKeySpendSig, 64,
2022-04-28 13:39:28 +02:00
)
},
)
}
// runSignPsbtSegWitV1KeySpendRootHash tests that the SignPsbt RPC works
// correctly for a SegWit v1 p2tr key spend that also commits to a script tree
// root hash.
func runSignPsbtSegWitV1KeySpendRootHash(ht *lntest.HarnessTest,
2022-08-11 07:07:33 +02:00
alice *node.HarnessNode) {
2022-04-28 13:39:28 +02:00
// Derive a key we can use for signing.
2022-08-11 07:07:33 +02:00
keyDesc, internalKey, fullDerivationPath := deriveInternalKey(ht, alice)
2022-04-28 13:39:28 +02:00
// Let's create a taproot script output now. This is a hash lock with a
// simple preimage of "foobar".
2022-08-11 07:07:33 +02:00
leaf1 := testScriptHashLock(ht.T, []byte("foobar"))
2022-04-28 13:39:28 +02:00
rootHash := leaf1.TapHash()
taprootKey := txscript.ComputeTaprootOutputKey(internalKey, rootHash[:])
tapScriptAddr, err := btcutil.NewAddressTaproot(
schnorr.SerializePubKey(taprootKey), harnessNetParams,
)
2022-08-11 07:07:33 +02:00
require.NoError(ht, err)
2022-04-28 13:39:28 +02:00
p2trPkScript, err := txscript.PayToAddrScript(tapScriptAddr)
2022-08-11 07:07:33 +02:00
require.NoError(ht, err)
2022-04-28 13:39:28 +02:00
// Send some funds to the output and then try to get a signature through
// the SignPsbt RPC to spend that output again.
assertPsbtSpend(
2022-08-11 07:07:33 +02:00
ht, alice, p2trPkScript,
2022-04-28 13:39:28 +02:00
func(packet *psbt.Packet) {
in := &packet.Inputs[0]
in.Bip32Derivation = []*psbt.Bip32Derivation{{
PubKey: keyDesc.RawKeyBytes,
Bip32Path: fullDerivationPath,
}}
in.TaprootBip32Derivation = []*psbt.TaprootBip32Derivation{{
XOnlyPubKey: keyDesc.RawKeyBytes[1:],
Bip32Path: fullDerivationPath,
}}
in.TaprootMerkleRoot = rootHash[:]
in.SighashType = txscript.SigHashDefault
},
func(packet *psbt.Packet) {
2022-08-11 07:07:33 +02:00
require.Len(ht, packet.Inputs, 1)
2022-04-28 13:39:28 +02:00
require.Len(
2022-08-11 07:07:33 +02:00
ht, packet.Inputs[0].TaprootKeySpendSig, 64,
2022-04-28 13:39:28 +02:00
)
},
)
}
// runSignPsbtSegWitV1ScriptSpend tests that the SignPsbt RPC works correctly
// for a SegWit v1 p2tr script spend.
func runSignPsbtSegWitV1ScriptSpend(ht *lntest.HarnessTest,
2022-08-11 07:07:33 +02:00
alice *node.HarnessNode) {
2022-04-28 13:39:28 +02:00
// Derive a key we can use for signing.
2022-08-11 07:07:33 +02:00
keyDesc, internalKey, fullDerivationPath := deriveInternalKey(ht, alice)
2022-04-28 13:39:28 +02:00
// Let's create a taproot script output now. This is a hash lock with a
// simple preimage of "foobar".
2022-08-11 07:07:33 +02:00
leaf1 := testScriptSchnorrSig(ht.T, internalKey)
2022-04-28 13:39:28 +02:00
rootHash := leaf1.TapHash()
taprootKey := txscript.ComputeTaprootOutputKey(internalKey, rootHash[:])
tapScriptAddr, err := btcutil.NewAddressTaproot(
schnorr.SerializePubKey(taprootKey), harnessNetParams,
)
2022-08-11 07:07:33 +02:00
require.NoError(ht, err)
2022-04-28 13:39:28 +02:00
p2trPkScript, err := txscript.PayToAddrScript(tapScriptAddr)
2022-08-11 07:07:33 +02:00
require.NoError(ht, err)
2022-04-28 13:39:28 +02:00
// We need to assemble the control block to be able to spend through the
// script path.
tapscript := input.TapscriptPartialReveal(internalKey, leaf1, nil)
controlBlockBytes, err := tapscript.ControlBlock.ToBytes()
2022-08-11 07:07:33 +02:00
require.NoError(ht, err)
2022-04-28 13:39:28 +02:00
// Send some funds to the output and then try to get a signature through
// the SignPsbt RPC to spend that output again.
assertPsbtSpend(
2022-08-11 07:07:33 +02:00
ht, alice, p2trPkScript,
2022-04-28 13:39:28 +02:00
func(packet *psbt.Packet) {
in := &packet.Inputs[0]
in.Bip32Derivation = []*psbt.Bip32Derivation{{
PubKey: keyDesc.RawKeyBytes,
Bip32Path: fullDerivationPath,
}}
in.TaprootBip32Derivation = []*psbt.TaprootBip32Derivation{{
XOnlyPubKey: keyDesc.RawKeyBytes[1:],
Bip32Path: fullDerivationPath,
LeafHashes: [][]byte{rootHash[:]},
}}
in.SighashType = txscript.SigHashDefault
in.TaprootLeafScript = []*psbt.TaprootTapLeafScript{{
ControlBlock: controlBlockBytes,
Script: leaf1.Script,
LeafVersion: leaf1.LeafVersion,
}}
},
func(packet *psbt.Packet) {
2022-08-11 07:07:33 +02:00
require.Len(ht, packet.Inputs, 1)
2022-04-28 13:39:28 +02:00
require.Len(
2022-08-11 07:07:33 +02:00
ht, packet.Inputs[0].TaprootScriptSpendSig, 1,
2022-04-28 13:39:28 +02:00
)
scriptSpendSig := packet.Inputs[0].TaprootScriptSpendSig[0]
2022-08-11 07:07:33 +02:00
require.Len(ht, scriptSpendSig.Signature, 64)
2022-04-28 13:39:28 +02:00
},
)
}
// runFundAndSignPsbt makes sure we can sign PSBTs that were funded by our
// internal wallet.
func runFundAndSignPsbt(ht *lntest.HarnessTest, alice *node.HarnessNode) {
2022-08-11 07:07:33 +02:00
alice.AddToLogf("================ runFundAndSignPsbt ===============")
// We'll be using a "main" address where we send the funds to and from
// several times.
2022-08-11 07:07:33 +02:00
mainAddrResp := alice.RPC.NewAddress(&lnrpc.NewAddressRequest{
Type: lnrpc.AddressType_WITNESS_PUBKEY_HASH,
})
fundOutputs := map[string]uint64{
mainAddrResp.Address: 999000,
}
spendAddrTypes := []lnrpc.AddressType{
lnrpc.AddressType_NESTED_PUBKEY_HASH,
lnrpc.AddressType_WITNESS_PUBKEY_HASH,
lnrpc.AddressType_TAPROOT_PUBKEY,
}
changeAddrTypes := []walletrpc.ChangeAddressType{
walletrpc.ChangeAddressType_CHANGE_ADDRESS_TYPE_UNSPECIFIED,
walletrpc.ChangeAddressType_CHANGE_ADDRESS_TYPE_P2TR,
}
for _, addrType := range spendAddrTypes {
for _, changeType := range changeAddrTypes {
ht.Logf("testing with address type %s and"+
"change address type %s", addrType, changeType)
// First, spend all the coins in the wallet to an
// address of the given type so that UTXO will be picked
// when funding a PSBT.
sendAllCoinsToAddrType(ht, alice, addrType)
// Let's fund a PSBT now where we want to send a few
// sats to our main address.
assertPsbtFundSignSpend(
ht, alice, fundOutputs, changeType, false,
)
// Send all coins back to a single address once again.
sendAllCoinsToAddrType(ht, alice, addrType)
// And now make sure the alternate way of signing a
// PSBT, which is calling FinalizePsbt directly, also
// works for this address type.
assertPsbtFundSignSpend(
ht, alice, fundOutputs, changeType, true,
)
}
}
}
2022-04-28 13:19:50 +02:00
// assertPsbtSpend creates an output with the given pkScript on chain and then
// attempts to create a sweep transaction that is signed using the SignPsbt RPC
// that spends that output again.
func assertPsbtSpend(ht *lntest.HarnessTest, alice *node.HarnessNode,
2022-08-11 07:07:33 +02:00
pkScript []byte, decorateUnsigned func(*psbt.Packet),
verifySigned func(*psbt.Packet)) {
2022-04-28 13:19:50 +02:00
2022-01-05 11:04:24 +01:00
// Let's send some coins to that address now.
utxo := &wire.TxOut{
Value: 600_000,
PkScript: pkScript,
}
2022-08-11 07:07:33 +02:00
req := &walletrpc.SendOutputsRequest{
Outputs: []*signrpc.TxOut{{
Value: utxo.Value,
PkScript: utxo.PkScript,
}},
MinConfs: 0,
SpendUnconfirmed: true,
SatPerKw: 2500,
}
resp := alice.RPC.SendOutputs(req)
2022-01-05 11:04:24 +01:00
prevTx := wire.NewMsgTx(2)
2022-08-11 07:07:33 +02:00
err := prevTx.Deserialize(bytes.NewReader(resp.RawTx))
require.NoError(ht, err)
2022-01-05 11:04:24 +01:00
prevOut := -1
for idx, txOut := range prevTx.TxOut {
if bytes.Equal(txOut.PkScript, pkScript) {
prevOut = idx
}
}
2022-08-11 07:07:33 +02:00
require.Greater(ht, prevOut, -1)
2022-01-05 11:04:24 +01:00
// Okay, we have everything we need to create a PSBT now.
pendingTx := &wire.MsgTx{
Version: 2,
TxIn: []*wire.TxIn{{
PreviousOutPoint: wire.OutPoint{
Hash: prevTx.TxHash(),
Index: uint32(prevOut),
},
}},
// We send to the same address again, but deduct some fees.
TxOut: []*wire.TxOut{{
Value: utxo.Value - 600,
PkScript: utxo.PkScript,
}},
}
packet, err := psbt.NewFromUnsignedTx(pendingTx)
2022-08-11 07:07:33 +02:00
require.NoError(ht, err)
2022-01-05 11:04:24 +01:00
// We first try to sign the psbt without the necessary input data
// which should fail with the expected error.
var buf bytes.Buffer
err = packet.Serialize(&buf)
require.NoError(ht, err)
signReq := &walletrpc.SignPsbtRequest{FundedPsbt: buf.Bytes()}
err = alice.RPC.SignPsbtErr(signReq)
require.ErrorContains(ht, err, "input (index=0) doesn't specify "+
"any UTXO info", "error does not match")
2022-01-05 11:04:24 +01:00
// Now let's add the meta information that we need for signing.
packet.Inputs[0].WitnessUtxo = utxo
packet.Inputs[0].NonWitnessUtxo = prevTx
2022-04-28 13:19:50 +02:00
decorateUnsigned(packet)
2022-01-05 11:04:24 +01:00
// That's it, we should be able to sign the PSBT now.
buf.Reset()
2022-01-05 11:04:24 +01:00
err = packet.Serialize(&buf)
2022-08-11 07:07:33 +02:00
require.NoError(ht, err)
2022-01-05 11:04:24 +01:00
signReq = &walletrpc.SignPsbtRequest{FundedPsbt: buf.Bytes()}
2022-08-11 07:07:33 +02:00
signResp := alice.RPC.SignPsbt(signReq)
2022-01-05 11:04:24 +01:00
// Let's make sure we have a partial signature.
signedPacket, err := psbt.NewFromRawBytes(
bytes.NewReader(signResp.SignedPsbt), false,
)
2022-08-11 07:07:33 +02:00
require.NoError(ht, err)
2022-01-05 11:04:24 +01:00
2022-04-28 13:19:50 +02:00
// Allow the caller to also verify (and potentially move) some of the
// returned fields.
verifySigned(signedPacket)
2022-01-05 11:04:24 +01:00
// We should be able to finalize the PSBT and extract the final TX now.
err = psbt.MaybeFinalizeAll(signedPacket)
2022-08-11 07:07:33 +02:00
require.NoError(ht, err)
2022-01-05 11:04:24 +01:00
finalTx, err := psbt.Extract(signedPacket)
2022-08-11 07:07:33 +02:00
require.NoError(ht, err)
2022-01-05 11:04:24 +01:00
// Make sure we can also sign a second time. This makes sure any key
// tweaking that happened for the signing didn't affect any keys in the
// cache.
2022-08-11 07:07:33 +02:00
r := &walletrpc.SignPsbtRequest{FundedPsbt: buf.Bytes()}
signResp2 := alice.RPC.SignPsbt(r)
signedPacket2, err := psbt.NewFromRawBytes(
bytes.NewReader(signResp2.SignedPsbt), false,
)
2022-08-11 07:07:33 +02:00
require.NoError(ht, err)
verifySigned(signedPacket2)
2022-01-05 11:04:24 +01:00
buf.Reset()
err = finalTx.Serialize(&buf)
2022-08-11 07:07:33 +02:00
require.NoError(ht, err)
2022-01-05 11:04:24 +01:00
// Publish the second transaction and then mine both of them.
2022-08-11 07:07:33 +02:00
txReq := &walletrpc.Transaction{TxHex: buf.Bytes()}
alice.RPC.PublishTransaction(txReq)
2022-01-05 11:04:24 +01:00
// Mine one block which should contain two transactions.
2022-08-11 07:07:33 +02:00
block := ht.MineBlocksAndAssertNumTxes(1, 2)[0]
2022-01-05 11:04:24 +01:00
firstTxHash := prevTx.TxHash()
secondTxHash := finalTx.TxHash()
2022-08-11 07:07:33 +02:00
ht.Miner.AssertTxInBlock(block, &firstTxHash)
ht.Miner.AssertTxInBlock(block, &secondTxHash)
2022-01-05 11:04:24 +01:00
}
// assertPsbtFundSignSpend funds a PSBT from the internal wallet and then
// attempts to sign it by using the SignPsbt or FinalizePsbt method.
func assertPsbtFundSignSpend(ht *lntest.HarnessTest, alice *node.HarnessNode,
fundOutputs map[string]uint64, changeType walletrpc.ChangeAddressType,
useFinalize bool) {
2022-08-11 07:07:33 +02:00
fundResp := alice.RPC.FundPsbt(&walletrpc.FundPsbtRequest{
Template: &walletrpc.FundPsbtRequest_Raw{
Raw: &walletrpc.TxTemplate{
Outputs: fundOutputs,
},
},
2022-08-11 07:07:33 +02:00
Fees: &walletrpc.FundPsbtRequest_SatPerVbyte{
SatPerVbyte: 2,
},
MinConfs: 1,
ChangeType: changeType,
2022-08-11 07:07:33 +02:00
},
)
2022-08-11 07:07:33 +02:00
require.GreaterOrEqual(ht, fundResp.ChangeOutputIndex, int32(-1))
// Make sure our change output has all the meta information required for
// signing.
fundedPacket, err := psbt.NewFromRawBytes(
bytes.NewReader(fundResp.FundedPsbt), false,
)
require.NoError(ht, err)
pOut := fundedPacket.Outputs[fundResp.ChangeOutputIndex]
require.NotEmpty(ht, pOut.Bip32Derivation)
derivation := pOut.Bip32Derivation[0]
_, err = btcec.ParsePubKey(derivation.PubKey)
require.NoError(ht, err)
require.Len(ht, derivation.Bip32Path, 5)
// Ensure we get the change output properly decorated with all the new
// Taproot related fields, if it is a Taproot output.
if changeType == walletrpc.ChangeAddressType_CHANGE_ADDRESS_TYPE_P2TR {
require.NotEmpty(ht, pOut.TaprootBip32Derivation)
require.NotEmpty(ht, pOut.TaprootInternalKey)
trDerivation := pOut.TaprootBip32Derivation[0]
require.Equal(
ht, trDerivation.XOnlyPubKey, pOut.TaprootInternalKey,
)
_, err := schnorr.ParsePubKey(pOut.TaprootInternalKey)
require.NoError(ht, err)
}
var signedPsbt []byte
if useFinalize {
2022-08-11 07:07:33 +02:00
finalizeResp := alice.RPC.FinalizePsbt(
&walletrpc.FinalizePsbtRequest{
FundedPsbt: fundResp.FundedPsbt,
},
)
signedPsbt = finalizeResp.SignedPsbt
} else {
2022-08-11 07:07:33 +02:00
signResp := alice.RPC.SignPsbt(
&walletrpc.SignPsbtRequest{
FundedPsbt: fundResp.FundedPsbt,
},
)
signedPsbt = signResp.SignedPsbt
}
// Let's make sure we have a partial signature.
signedPacket, err := psbt.NewFromRawBytes(
bytes.NewReader(signedPsbt), false,
)
2022-08-11 07:07:33 +02:00
require.NoError(ht, err)
// We should be able to finalize the PSBT and extract the final
// TX now.
err = psbt.MaybeFinalizeAll(signedPacket)
2022-08-11 07:07:33 +02:00
require.NoError(ht, err)
finalTx, err := psbt.Extract(signedPacket)
2022-08-11 07:07:33 +02:00
require.NoError(ht, err)
// Check type of the change script depending on the change address
// type we provided in FundPsbt.
changeScript := finalTx.TxOut[fundResp.ChangeOutputIndex].PkScript
assertChangeScriptType(ht, changeScript, changeType)
var buf bytes.Buffer
err = finalTx.Serialize(&buf)
2022-08-11 07:07:33 +02:00
require.NoError(ht, err)
// Publish the second transaction and then mine both of them.
2022-08-11 07:07:33 +02:00
alice.RPC.PublishTransaction(&walletrpc.Transaction{
TxHex: buf.Bytes(),
})
2022-08-11 07:07:33 +02:00
// Mine one block which should contain one transaction.
block := ht.MineBlocksAndAssertNumTxes(1, 1)[0]
finalTxHash := finalTx.TxHash()
2022-08-11 07:07:33 +02:00
ht.Miner.AssertTxInBlock(block, &finalTxHash)
}
// assertChangeScriptType checks if the given script has the right type given
// the change address type we used in FundPsbt. By default, the script should
// be a P2WPKH one.
func assertChangeScriptType(ht *lntest.HarnessTest, script []byte,
fundChangeType walletrpc.ChangeAddressType) {
switch fundChangeType {
case walletrpc.ChangeAddressType_CHANGE_ADDRESS_TYPE_P2TR:
require.True(ht, txscript.IsPayToTaproot(script))
default:
require.True(ht, txscript.IsPayToWitnessPubKeyHash(script))
}
}
2022-08-11 07:07:33 +02:00
// deriveInternalKey derives a signing key and returns its descriptor, full
// derivation path and parsed public key.
func deriveInternalKey(ht *lntest.HarnessTest,
2022-08-11 07:07:33 +02:00
alice *node.HarnessNode) (*signrpc.KeyDescriptor, *btcec.PublicKey,
[]uint32) {
// For the next step, we need a public key. Let's use a special family
// for this.
req := &walletrpc.KeyReq{KeyFamily: testTaprootKeyFamily}
keyDesc := alice.RPC.DeriveNextKey(req)
// The DeriveNextKey returns a key from the internal 1017 scope.
fullDerivationPath := []uint32{
hdkeychain.HardenedKeyStart + keychain.BIP0043Purpose,
hdkeychain.HardenedKeyStart + harnessNetParams.HDCoinType,
hdkeychain.HardenedKeyStart + uint32(keyDesc.KeyLoc.KeyFamily),
0,
uint32(keyDesc.KeyLoc.KeyIndex),
}
parsedPubKey, err := btcec.ParsePubKey(keyDesc.RawKeyBytes)
require.NoError(ht, err)
return keyDesc, parsedPubKey, fullDerivationPath
}
// sendAllCoinsToAddrType sweeps all coins from the wallet and sends them to a
// new address of the given type.
func sendAllCoinsToAddrType(ht *lntest.HarnessTest,
2022-08-11 07:07:33 +02:00
hn *node.HarnessNode, addrType lnrpc.AddressType) {
2022-08-11 07:07:33 +02:00
resp := hn.RPC.NewAddress(&lnrpc.NewAddressRequest{
Type: addrType,
})
2022-08-11 07:07:33 +02:00
hn.RPC.SendCoins(&lnrpc.SendCoinsRequest{
Addr: resp.Address,
SendAll: true,
SpendUnconfirmed: true,
})
2022-08-11 07:07:33 +02:00
ht.MineBlocksAndAssertNumTxes(1, 1)
ht.WaitForBlockchainSync(hn)
}