lnwallet/interface_test: extract local utility functions

In preparation for extending the testPublishTransaction test, shorten it
by moving utility methods out of the local scope.
This commit is contained in:
Johan T. Halseth 2019-09-19 14:59:07 +02:00
parent 18f88cbd8d
commit 7897b96e6a
No known key found for this signature in database
GPG Key ID: 15BAADA29DA20D26

@ -1418,15 +1418,31 @@ func testTransactionSubscriptions(miner *rpctest.Harness,
} }
} }
// testPublishTransaction checks that PublishTransaction returns the // scriptFromKey creates a P2WKH script from the given pubkey.
// expected error types in case the transaction being published func scriptFromKey(pubkey *btcec.PublicKey) ([]byte, error) {
// conflicts with the current mempool or chain. pubkeyHash := btcutil.Hash160(pubkey.SerializeCompressed())
func testPublishTransaction(r *rpctest.Harness, keyAddr, err := btcutil.NewAddressWitnessPubKeyHash(
alice, _ *lnwallet.LightningWallet, t *testing.T) { pubkeyHash, &chaincfg.RegressionNetParams,
)
if err != nil {
return nil, fmt.Errorf("unable to create addr: %v", err)
}
keyScript, err := txscript.PayToAddrScript(keyAddr)
if err != nil {
return nil, fmt.Errorf("unable to generate script: %v", err)
}
return keyScript, nil
}
// mineAndAssert mines a block and ensures the passed TX is part of that block.
func mineAndAssert(r *rpctest.Harness, tx *wire.MsgTx) error {
txid := tx.TxHash()
err := waitForMempoolTx(r, &txid)
if err != nil {
return fmt.Errorf("tx not relayed to miner: %v", err)
}
// mineAndAssert mines a block and ensures the passed TX
// is part of that block.
mineAndAssert := func(tx *wire.MsgTx) error {
blockHashes, err := r.Node.Generate(1) blockHashes, err := r.Node.Generate(1)
if err != nil { if err != nil {
return fmt.Errorf("unable to generate block: %v", err) return fmt.Errorf("unable to generate block: %v", err)
@ -1447,45 +1463,22 @@ func testPublishTransaction(r *rpctest.Harness,
return fmt.Errorf("incorrect transaction was mined") return fmt.Errorf("incorrect transaction was mined")
} }
// Sleep for a second before returning, to make sure the // Sleep for a second before returning, to make sure the block has
// block has propagated. // propagated.
time.Sleep(1 * time.Second) time.Sleep(1 * time.Second)
return nil return nil
} }
// Generate a pubkey, and pay-to-addr script. // txFromOutput takes a tx paying to fromPubKey, and creates a new tx that
pubKey, err := alice.DeriveNextKey( // spends the output from this tx, to an address derived from payToPubKey.
keychain.KeyFamilyMultiSig, func txFromOutput(tx *wire.MsgTx, signer input.Signer, fromPubKey,
) payToPubKey *btcec.PublicKey, txFee btcutil.Amount) (
if err != nil { *wire.MsgTx, error) {
t.Fatalf("unable to obtain public key: %v", err)
}
pubkeyHash := btcutil.Hash160(pubKey.PubKey.SerializeCompressed())
keyAddr, err := btcutil.NewAddressWitnessPubKeyHash(pubkeyHash,
&chaincfg.RegressionNetParams)
if err != nil {
t.Fatalf("unable to create addr: %v", err)
}
keyScript, err := txscript.PayToAddrScript(keyAddr)
if err != nil {
t.Fatalf("unable to generate script: %v", err)
}
// txFromOutput takes a tx, and creates a new tx that spends // Generate the script we want to spend from.
// the output from this tx, to an address derived from payToPubKey. keyScript, err := scriptFromKey(fromPubKey)
// NB: assumes that the output from tx is paid to pubKey.
txFromOutput := func(tx *wire.MsgTx, payToPubKey *btcec.PublicKey,
txFee btcutil.Amount) *wire.MsgTx {
// Create a script to pay to.
payToPubkeyHash := btcutil.Hash160(payToPubKey.SerializeCompressed())
payToKeyAddr, err := btcutil.NewAddressWitnessPubKeyHash(payToPubkeyHash,
&chaincfg.RegressionNetParams)
if err != nil { if err != nil {
t.Fatalf("unable to create addr: %v", err) return nil, fmt.Errorf("unable to generate script: %v", err)
}
payToScript, err := txscript.PayToAddrScript(payToKeyAddr)
if err != nil {
t.Fatalf("unable to generate script: %v", err)
} }
// We assume the output was paid to the keyScript made earlier. // We assume the output was paid to the keyScript made earlier.
@ -1497,9 +1490,10 @@ func testPublishTransaction(r *rpctest.Harness,
} }
outputValue := tx.TxOut[outputIndex].Value outputValue := tx.TxOut[outputIndex].Value
// With the index located, we can create a transaction spending // With the index located, we can create a transaction spending the
// the referenced output. // referenced output.
tx1 := wire.NewMsgTx(2) tx1 := wire.NewMsgTx(2)
tx1.AddTxIn(&wire.TxIn{ tx1.AddTxIn(&wire.TxIn{
PreviousOutPoint: wire.OutPoint{ PreviousOutPoint: wire.OutPoint{
Hash: tx.TxHash(), Hash: tx.TxHash(),
@ -1508,16 +1502,22 @@ func testPublishTransaction(r *rpctest.Harness,
// We don't support RBF, so set sequence to max. // We don't support RBF, so set sequence to max.
Sequence: wire.MaxTxInSequenceNum, Sequence: wire.MaxTxInSequenceNum,
}) })
// Create a script to pay to.
payToScript, err := scriptFromKey(payToPubKey)
if err != nil {
return nil, fmt.Errorf("unable to generate script: %v", err)
}
tx1.AddTxOut(&wire.TxOut{ tx1.AddTxOut(&wire.TxOut{
Value: outputValue - int64(txFee), Value: outputValue - int64(txFee),
PkScript: payToScript, PkScript: payToScript,
}) })
// Now we can populate the sign descriptor which we'll use to // Now we can populate the sign descriptor which we'll use to generate
// generate the signature. // the signature.
signDesc := &input.SignDescriptor{ signDesc := &input.SignDescriptor{
KeyDesc: keychain.KeyDescriptor{ KeyDesc: keychain.KeyDescriptor{
PubKey: pubKey.PubKey, PubKey: fromPubKey,
}, },
WitnessScript: keyScript, WitnessScript: keyScript,
Output: tx.TxOut[outputIndex], Output: tx.TxOut[outputIndex],
@ -1526,40 +1526,47 @@ func testPublishTransaction(r *rpctest.Harness,
InputIndex: 0, // Has only one input. InputIndex: 0, // Has only one input.
} }
// With the descriptor created, we use it to generate a // With the descriptor created, we use it to generate a signature, then
// signature, then manually create a valid witness stack we'll // manually create a valid witness stack we'll use for signing.
// use for signing. spendSig, err := signer.SignOutputRaw(tx1, signDesc)
spendSig, err := alice.Cfg.Signer.SignOutputRaw(tx1, signDesc)
if err != nil { if err != nil {
t.Fatalf("unable to generate signature: %v", err) return nil, fmt.Errorf("unable to generate signature: %v", err)
} }
witness := make([][]byte, 2) witness := make([][]byte, 2)
witness[0] = append(spendSig, byte(txscript.SigHashAll)) witness[0] = append(spendSig, byte(txscript.SigHashAll))
witness[1] = pubKey.PubKey.SerializeCompressed() witness[1] = fromPubKey.SerializeCompressed()
tx1.TxIn[0].Witness = witness tx1.TxIn[0].Witness = witness
// Finally, attempt to validate the completed transaction. This // Finally, attempt to validate the completed transaction. This should
// should succeed if the wallet was able to properly generate // succeed if the wallet was able to properly generate the proper
// the proper private key. // private key.
vm, err := txscript.NewEngine(keyScript, vm, err := txscript.NewEngine(
tx1, 0, txscript.StandardVerifyFlags, nil, keyScript, tx1, 0, txscript.StandardVerifyFlags, nil,
nil, outputValue) nil, outputValue,
)
if err != nil { if err != nil {
t.Fatalf("unable to create engine: %v", err) return nil, fmt.Errorf("unable to create engine: %v", err)
} }
if err := vm.Execute(); err != nil { if err := vm.Execute(); err != nil {
t.Fatalf("spend is invalid: %v", err) return nil, fmt.Errorf("spend is invalid: %v", err)
}
return tx1
} }
// newTx sends coins from Alice's wallet, mines this transaction, return tx1, nil
// and creates a new, unconfirmed tx that spends this output to }
// pubKey.
newTx := func() *wire.MsgTx {
// With the script fully assembled, instruct the wallet to fund // newTx sends coins from Alice's wallet, mines this transaction, and creates a
// the output with a newly created transaction. // new, unconfirmed tx that spends this output to pubKey.
func newTx(t *testing.T, r *rpctest.Harness, pubKey *btcec.PublicKey,
alice *lnwallet.LightningWallet, rbf bool) *wire.MsgTx {
t.Helper()
keyScript, err := scriptFromKey(pubKey)
if err != nil {
t.Fatalf("unable to generate script: %v", err)
}
// Instruct the wallet to fund the output with a newly created
// transaction.
newOutput := &wire.TxOut{ newOutput := &wire.TxOut{
Value: btcutil.SatoshiPerBitcoin, Value: btcutil.SatoshiPerBitcoin,
PkScript: keyScript, PkScript: keyScript,
@ -1568,27 +1575,40 @@ func testPublishTransaction(r *rpctest.Harness,
if err != nil { if err != nil {
t.Fatalf("unable to create output: %v", err) t.Fatalf("unable to create output: %v", err)
} }
txid := tx.TxHash()
// Query for the transaction generated above so we can located // Query for the transaction generated above so we can located the
// the index of our output. // index of our output.
err = waitForMempoolTx(r, &txid) if err := mineAndAssert(r, tx); err != nil {
if err != nil {
t.Fatalf("tx not relayed to miner: %v", err)
}
if err := mineAndAssert(tx); err != nil {
t.Fatalf("unable to mine tx: %v", err) t.Fatalf("unable to mine tx: %v", err)
} }
// Create a new unconfirmed tx that spends this output.
txFee := btcutil.Amount(0.1 * btcutil.SatoshiPerBitcoin) txFee := btcutil.Amount(0.1 * btcutil.SatoshiPerBitcoin)
tx1 := txFromOutput(tx, pubKey.PubKey, txFee) tx1, err := txFromOutput(
tx, alice.Cfg.Signer, pubKey, pubKey, txFee,
)
if err != nil {
t.Fatal(err)
}
return tx1 return tx1
} }
// We will first check that publishing a transaction already // testPublishTransaction checks that PublishTransaction returns the expected
// in the mempool does NOT return an error. Create the tx. // error types in case the transaction being published conflicts with the
tx1 := newTx() // current mempool or chain.
func testPublishTransaction(r *rpctest.Harness,
alice, _ *lnwallet.LightningWallet, t *testing.T) {
// Generate a pubkey, and pay-to-addr script.
keyDesc, err := alice.DeriveNextKey(keychain.KeyFamilyMultiSig)
if err != nil {
t.Fatalf("unable to obtain public key: %v", err)
}
// We will first check that publishing a transaction already in the
// mempool does NOT return an error. Create the tx.
tx1 := newTx(t, r, keyDesc.PubKey, alice, false)
// Publish the transaction. // Publish the transaction.
if err := alice.PublishTransaction(tx1); err != nil { if err := alice.PublishTransaction(tx1); err != nil {
@ -1601,9 +1621,8 @@ func testPublishTransaction(r *rpctest.Harness,
t.Fatalf("tx not relayed to miner: %v", err) t.Fatalf("tx not relayed to miner: %v", err)
} }
// Publish the exact same transaction again. This should // Publish the exact same transaction again. This should not return an
// not return an error, even though the transaction is // error, even though the transaction is already in the mempool.
// already in the mempool.
if err := alice.PublishTransaction(tx1); err != nil { if err := alice.PublishTransaction(tx1); err != nil {
t.Fatalf("unable to publish: %v", err) t.Fatalf("unable to publish: %v", err)
} }
@ -1613,62 +1632,56 @@ func testPublishTransaction(r *rpctest.Harness,
t.Fatalf("unable to generate block: %v", err) t.Fatalf("unable to generate block: %v", err)
} }
// We'll now test that we don't get an error if we try // We'll now test that we don't get an error if we try to publish a
// to publish a transaction that is already mined. // transaction that is already mined.
// //
// Create a new transaction. We must do this to properly // Create a new transaction. We must do this to properly test the
// test the reject messages from our peers. They might // reject messages from our peers. They might only send us a reject
// only send us a reject message for a given tx once, // message for a given tx once, so we create a new to make sure it is
// so we create a new to make sure it is not just // not just immediately rejected.
// immediately rejected. tx2 := newTx(t, r, keyDesc.PubKey, alice, false)
tx2 := newTx()
// Publish this tx. // Publish this tx.
if err := alice.PublishTransaction(tx2); err != nil { if err := alice.PublishTransaction(tx2); err != nil {
t.Fatalf("unable to publish: %v", err) t.Fatalf("unable to publish: %v", err)
} }
txid2 := tx2.TxHash()
err = waitForMempoolTx(r, &txid2)
if err != nil {
t.Fatalf("tx not relayed to miner: %v", err)
}
// Mine the transaction. // Mine the transaction.
if err := mineAndAssert(tx2); err != nil { if err := mineAndAssert(r, tx2); err != nil {
t.Fatalf("unable to mine tx: %v", err) t.Fatalf("unable to mine tx: %v", err)
} }
// Publish the transaction again. It is already mined, // Publish the transaction again. It is already mined, and we don't
// and we don't expect this to return an error. // expect this to return an error.
if err := alice.PublishTransaction(tx2); err != nil { if err := alice.PublishTransaction(tx2); err != nil {
t.Fatalf("unable to publish: %v", err) t.Fatalf("unable to publish: %v", err)
} }
// Now we'll try to double spend an output with a different // Now we'll try to double spend an output with a different
// transaction. Create a new tx and publish it. This is // transaction. Create a new tx and publish it. This is the output
// the output we'll try to double spend. // we'll try to double spend.
tx3 := newTx() tx3 := newTx(t, r, keyDesc.PubKey, alice, false)
if err := alice.PublishTransaction(tx3); err != nil { if err := alice.PublishTransaction(tx3); err != nil {
t.Fatalf("unable to publish: %v", err) t.Fatalf("unable to publish: %v", err)
} }
txid3 := tx3.TxHash()
err = waitForMempoolTx(r, &txid3)
if err != nil {
t.Fatalf("tx not relayed to miner: %v", err)
}
// Mine the transaction. // Mine the transaction.
if err := mineAndAssert(tx3); err != nil { if err := mineAndAssert(r, tx3); err != nil {
t.Fatalf("unable to mine tx: %v", err) t.Fatalf("unable to mine tx: %v", err)
} }
// Now we create a transaction that spends the output // Now we create a transaction that spends the output from the tx just
// from the tx just mined. This should be accepted // mined.
// into the mempool.
txFee := btcutil.Amount(0.05 * btcutil.SatoshiPerBitcoin) txFee := btcutil.Amount(0.05 * btcutil.SatoshiPerBitcoin)
tx4 := txFromOutput(tx3, pubKey.PubKey, txFee) tx4, err := txFromOutput(
tx3, alice.Cfg.Signer, keyDesc.PubKey, keyDesc.PubKey,
txFee,
)
if err != nil {
t.Fatal(err)
}
// This should be accepted into the mempool.
if err := alice.PublishTransaction(tx4); err != nil { if err := alice.PublishTransaction(tx4); err != nil {
t.Fatalf("unable to publish: %v", err) t.Fatalf("unable to publish: %v", err)
} }
@ -1679,50 +1692,63 @@ func testPublishTransaction(r *rpctest.Harness,
t.Fatalf("tx not relayed to miner: %v", err) t.Fatalf("tx not relayed to miner: %v", err)
} }
// Create a new key we'll pay to, to ensure we create // Create a new key we'll pay to, to ensure we create a unique
// a unique transaction. // transaction.
pubKey2, err := alice.DeriveNextKey( keyDesc2, err := alice.DeriveNextKey(
keychain.KeyFamilyMultiSig, keychain.KeyFamilyMultiSig,
) )
if err != nil { if err != nil {
t.Fatalf("unable to obtain public key: %v", err) t.Fatalf("unable to obtain public key: %v", err)
} }
// Create a new transaction that spends the output from // Create a new transaction that spends the output from tx3, and that
// tx3, and that pays to a different address. We expect // pays to a different address. We expect this to be rejected because
// this to be rejected because it is a double spend. // it is a double spend.
tx5 := txFromOutput(tx3, pubKey2.PubKey, txFee) tx5, err := txFromOutput(
if err := alice.PublishTransaction(tx5); err != lnwallet.ErrDoubleSpend { tx3, alice.Cfg.Signer, keyDesc.PubKey, keyDesc2.PubKey,
txFee,
)
if err != nil {
t.Fatal(err)
}
err = alice.PublishTransaction(tx5)
if err != lnwallet.ErrDoubleSpend {
t.Fatalf("expected ErrDoubleSpend, got: %v", err) t.Fatalf("expected ErrDoubleSpend, got: %v", err)
} }
// Create another transaction that spends the same output, // Create another transaction that spends the same output, but has a
// but has a higher fee. We expect also this tx to be // higher fee. We expect also this tx to be rejected, since the
// rejected, since the sequence number of tx3 is set to Max, // sequence number of tx3 is set to Max, indicating it is not
// indicating it is not replacable. // replacable.
pubKey3, err := alice.DeriveNextKey( pubKey3, err := alice.DeriveNextKey(keychain.KeyFamilyMultiSig)
keychain.KeyFamilyMultiSig,
)
if err != nil { if err != nil {
t.Fatalf("unable to obtain public key: %v", err) t.Fatalf("unable to obtain public key: %v", err)
} }
tx6 := txFromOutput(tx3, pubKey3.PubKey, 3*txFee) tx6, err := txFromOutput(
tx3, alice.Cfg.Signer, keyDesc.PubKey,
pubKey3.PubKey, 2*txFee,
)
if err != nil {
t.Fatal(err)
}
// Expect rejection. // Expect rejection.
if err := alice.PublishTransaction(tx6); err != lnwallet.ErrDoubleSpend { err = alice.PublishTransaction(tx6)
if err != lnwallet.ErrDoubleSpend {
t.Fatalf("expected ErrDoubleSpend, got: %v", err) t.Fatalf("expected ErrDoubleSpend, got: %v", err)
} }
// At last we try to spend an output already spent by a // At last we try to spend an output already spent by a confirmed
// confirmed transaction. // transaction.
// TODO(halseth): we currently skip this test for neutrino, // TODO(halseth): we currently skip this test for neutrino, as the
// as the backing btcd node will consider the tx being an // backing btcd node will consider the tx being an orphan, and will
// orphan, and will accept it. Should look into if this is // accept it. Should look into if this is the behavior also for
// the behavior also for bitcoind, and update test // bitcoind, and update test accordingly.
// accordingly.
if alice.BackEnd() != "neutrino" { if alice.BackEnd() != "neutrino" {
// Mine the tx spending tx3. // Mine the tx spending tx3.
if err := mineAndAssert(tx4); err != nil { if err := mineAndAssert(r, tx4); err != nil {
t.Fatalf("unable to mine tx: %v", err) t.Fatalf("unable to mine tx: %v", err)
} }
@ -1733,16 +1759,24 @@ func testPublishTransaction(r *rpctest.Harness,
if err != nil { if err != nil {
t.Fatalf("unable to obtain public key: %v", err) t.Fatalf("unable to obtain public key: %v", err)
} }
tx7 := txFromOutput(tx3, pubKey4.PubKey, txFee) tx7, err := txFromOutput(
tx3, alice.Cfg.Signer, keyDesc.PubKey,
pubKey4.PubKey, txFee,
)
if err != nil {
t.Fatal(err)
}
// Expect rejection. // Expect rejection.
if err := alice.PublishTransaction(tx7); err != lnwallet.ErrDoubleSpend { err = alice.PublishTransaction(tx7)
if err != lnwallet.ErrDoubleSpend {
t.Fatalf("expected ErrDoubleSpend, got: %v", err) t.Fatalf("expected ErrDoubleSpend, got: %v", err)
} }
} }
// TODO(halseth): test replaceable transactions when btcd // TODO(halseth): test replaceable transactions when btcd gets RBF
// gets RBF support. // support.
} }
func testSignOutputUsingTweaks(r *rpctest.Harness, func testSignOutputUsingTweaks(r *rpctest.Harness,