// Copyright (c) 2018 The btcsuite developers // Use of this source code is governed by an ISC // license that can be found in the LICENSE file. package wallet import ( "bytes" "io/ioutil" "os" "testing" "time" "github.com/btcsuite/btcd/chaincfg" "github.com/btcsuite/btcd/chaincfg/chainhash" "github.com/btcsuite/btcd/txscript" "github.com/btcsuite/btcd/wire" "github.com/btcsuite/btcutil/hdkeychain" "github.com/btcsuite/btcwallet/waddrmgr" "github.com/btcsuite/btcwallet/walletdb" _ "github.com/btcsuite/btcwallet/walletdb/bdb" "github.com/btcsuite/btcwallet/wtxmgr" ) // TestTxToOutput checks that no new address is added to he database if we // request a dry run of the txToOutputs call. It also makes sure a subsequent // non-dry run call produces a similar transaction to the dry-run. func TestTxToOutputsDryRun(t *testing.T) { // Set up a wallet. dir, err := ioutil.TempDir("", "createtx_test") if err != nil { t.Fatalf("Failed to create db dir: %v", err) } defer os.RemoveAll(dir) seed, err := hdkeychain.GenerateSeed(hdkeychain.MinSeedBytes) if err != nil { t.Fatalf("unable to create seed: %v", err) } pubPass := []byte("hello") privPass := []byte("world") loader := NewLoader(&chaincfg.TestNet3Params, dir, 250) w, err := loader.CreateNewWallet(pubPass, privPass, seed, time.Now()) if err != nil { t.Fatalf("unable to create wallet: %v", err) } chainClient := &mockChainClient{} w.chainClient = chainClient if err := w.Unlock(privPass, time.After(10*time.Minute)); err != nil { t.Fatalf("unable to unlock wallet: %v", err) } // Create an address we can use to send some coins to. addr, err := w.CurrentAddress(0, waddrmgr.KeyScopeBIP0044) if err != nil { t.Fatalf("unable to get current address: %v", addr) } p2shAddr, err := txscript.PayToAddrScript(addr) if err != nil { t.Fatalf("unable to convert wallet address to p2sh: %v", err) } // Add an output paying to the wallet's address to the database. txOut := wire.NewTxOut(100000, p2shAddr) incomingTx := &wire.MsgTx{ TxIn: []*wire.TxIn{ {}, }, TxOut: []*wire.TxOut{ txOut, }, } var b bytes.Buffer if err := incomingTx.Serialize(&b); err != nil { t.Fatalf("unable to serialize tx: %v", err) } txBytes := b.Bytes() rec, err := wtxmgr.NewTxRecord(txBytes, time.Now()) if err != nil { t.Fatalf("unable to create tx record: %v", err) } // The block meta will be inserted to tell the wallet this is a // confirmed transaction. blockHash, _ := chainhash.NewHashFromStr( "00000000000000017188b968a371bab95aa43522665353b646e41865abae02a4") block := &wtxmgr.BlockMeta{ Block: wtxmgr.Block{Hash: *blockHash, Height: 276425}, Time: time.Unix(1387737310, 0), } if err := walletdb.Update(w.db, func(tx walletdb.ReadWriteTx) error { ns := tx.ReadWriteBucket(wtxmgrNamespaceKey) err = w.TxStore.InsertTx(ns, rec, block) if err != nil { return err } err = w.TxStore.AddCredit(ns, rec, block, 0, false) if err != nil { return err } return nil }); err != nil { t.Fatalf("failed inserting tx: %v", err) } // Now tell the wallet to create a transaction paying to the specified // outputs. txOuts := []*wire.TxOut{ { PkScript: p2shAddr, Value: 10000, }, { PkScript: p2shAddr, Value: 20000, }, } // First do a few dry-runs, making sure the number of addresses in the // database us not inflated. dryRunTx, err := w.txToOutputs(txOuts, 0, 1, 1000, true) if err != nil { t.Fatalf("unable to author tx: %v", err) } change := dryRunTx.Tx.TxOut[dryRunTx.ChangeIndex] addresses, err := w.AccountAddresses(0) if err != nil { t.Fatalf("unable to get addresses: %v", err) } if len(addresses) != 1 { t.Fatalf("expected 1 address, found %v", len(addresses)) } dryRunTx2, err := w.txToOutputs(txOuts, 0, 1, 1000, true) if err != nil { t.Fatalf("unable to author tx: %v", err) } change2 := dryRunTx2.Tx.TxOut[dryRunTx2.ChangeIndex] addresses, err = w.AccountAddresses(0) if err != nil { t.Fatalf("unable to get addresses: %v", err) } if len(addresses) != 1 { t.Fatalf("expected 1 address, found %v", len(addresses)) } // The two dry-run TXs should be invalid, since they don't have // signatures. err = validateMsgTx( dryRunTx.Tx, dryRunTx.PrevScripts, dryRunTx.PrevInputValues, ) if err == nil { t.Fatalf("Expected tx to be invalid") } err = validateMsgTx( dryRunTx2.Tx, dryRunTx2.PrevScripts, dryRunTx2.PrevInputValues, ) if err == nil { t.Fatalf("Expected tx to be invalid") } // Now we do a proper, non-dry run. This should add a change address // to the database. tx, err := w.txToOutputs(txOuts, 0, 1, 1000, false) if err != nil { t.Fatalf("unable to author tx: %v", err) } change3 := tx.Tx.TxOut[tx.ChangeIndex] addresses, err = w.AccountAddresses(0) if err != nil { t.Fatalf("unable to get addresses: %v", err) } if len(addresses) != 2 { t.Fatalf("expected 2 addresses, found %v", len(addresses)) } err = validateMsgTx(tx.Tx, tx.PrevScripts, tx.PrevInputValues) if err != nil { t.Fatalf("Expected tx to be valid: %v", err) } // Finally, we check that all the transaction were using the same // change address. if !bytes.Equal(change.PkScript, change2.PkScript) { t.Fatalf("first dry-run using different change address " + "than second") } if !bytes.Equal(change2.PkScript, change3.PkScript) { t.Fatalf("dry-run using different change address " + "than wet run") } }