TEST_F(TransfersContainer_addTransaction, spendingConfirmedMultisignatureOutputWithConfirmedTxSucceed) { TestTransactionBuilder tx; tx.addTestInput(TEST_OUTPUT_AMOUNT + 1, account); auto out = tx.addTestMultisignatureOutput(TEST_OUTPUT_AMOUNT, TEST_TRANSACTION_OUTPUT_GLOBAL_INDEX); ASSERT_TRUE(container.addTransaction(blockInfo(TEST_BLOCK_HEIGHT), *tx.build(), { out })); container.advanceHeight(TEST_BLOCK_HEIGHT + TEST_TRANSACTION_SPENDABLE_AGE); ASSERT_EQ(TEST_OUTPUT_AMOUNT, container.balance(ITransfersContainer::IncludeAllUnlocked)); TestTransactionBuilder spendingTx; spendingTx.addTestMultisignatureInput(TEST_OUTPUT_AMOUNT, out); ASSERT_TRUE(container.addTransaction(blockInfo(TEST_BLOCK_HEIGHT + TEST_TRANSACTION_SPENDABLE_AGE), *spendingTx.build(), {})); container.advanceHeight(TEST_BLOCK_HEIGHT + TEST_TRANSACTION_SPENDABLE_AGE*2); ASSERT_EQ(0, container.balance(ITransfersContainer::IncludeAllUnlocked)); }
TEST_F(TransfersContainer_addTransaction, handlesAddingConfirmedOutputToKey) { CryptoNote::TransactionBlockInfo blockInfo{ TEST_BLOCK_HEIGHT, 1000000 }; TestTransactionBuilder txbuilder; txbuilder.addTestInput(TEST_OUTPUT_AMOUNT + 1, account); auto outInfo = txbuilder.addTestKeyOutput(TEST_OUTPUT_AMOUNT, TEST_TRANSACTION_OUTPUT_GLOBAL_INDEX); std::vector<TransactionOutputInformationIn> outputs = { outInfo }; auto tx = txbuilder.build(); ASSERT_TRUE(container.addTransaction(blockInfo, *tx, outputs)); container.advanceHeight(1000); ASSERT_EQ(1, container.transfersCount()); ASSERT_EQ(1, container.transactionsCount()); ASSERT_EQ(0, container.balance(ITransfersContainer::IncludeAllLocked)); ASSERT_EQ(TEST_OUTPUT_AMOUNT, container.balance(ITransfersContainer::IncludeAllUnlocked)); std::vector<TransactionOutputInformation> transfers; container.getOutputs(transfers, ITransfersContainer::IncludeAllLocked); ASSERT_TRUE(transfers.empty()); transfers.clear(); container.getOutputs(transfers, ITransfersContainer::IncludeAllUnlocked); ASSERT_EQ(1, transfers.size()); transfers = container.getTransactionOutputs(tx->getTransactionHash(), ITransfersContainer::IncludeAllLocked); ASSERT_TRUE(transfers.empty()); transfers = container.getTransactionOutputs(tx->getTransactionHash(), ITransfersContainer::IncludeAllUnlocked); ASSERT_EQ(1, transfers.size()); TransactionInformation txInfo; uint64_t amountIn; uint64_t amountOut; ASSERT_TRUE(container.getTransactionInformation(tx->getTransactionHash(), txInfo, &amountIn, &amountOut)); ASSERT_EQ(blockInfo.height, txInfo.blockHeight); ASSERT_EQ(0, amountIn); ASSERT_EQ(TEST_OUTPUT_AMOUNT, amountOut); std::vector<Crypto::Hash> unconfirmedTransactions; container.getUnconfirmedTransactions(unconfirmedTransactions); ASSERT_TRUE(unconfirmedTransactions.empty()); }
std::unique_ptr<ITransactionReader> FusionTransactionBuilder::buildReader() const { assert(m_inputCount > 0); assert(m_firstInput <= m_amount); assert(m_amount > m_currency.defaultDustThreshold()); TestTransactionBuilder builder; if (m_extraSize != 0) { builder.appendExtra(BinaryArray(m_extraSize, 0)); } if (m_firstInput != 0) { builder.addTestInput(m_firstInput); } if (m_amount > m_firstInput) { builder.addTestInput(m_amount - m_firstInput - (m_inputCount - 1) * m_currency.defaultDustThreshold()); for (size_t i = 0; i < m_inputCount - 1; ++i) { builder.addTestInput(m_currency.defaultDustThreshold()); } } AccountPublicAddress address = generateAddress(); std::vector<uint64_t> outputAmounts; assert(m_amount >= m_firstOutput + m_fee); decomposeAmount(m_amount - m_firstOutput - m_fee, m_currency.defaultDustThreshold(), outputAmounts); std::sort(outputAmounts.begin(), outputAmounts.end()); if (m_firstOutput != 0) { builder.addOutput(m_firstOutput, address); } for (auto outAmount : outputAmounts) { builder.addOutput(outAmount, address); } return builder.build(); }
TEST_F(TransfersContainer_markTransactionConfirmed, confirmationTxWithNoOutputs) { addTransaction(TEST_BLOCK_HEIGHT); container.advanceHeight(TEST_BLOCK_HEIGHT + TEST_TRANSACTION_SPENDABLE_AGE); std::vector<TransactionOutputInformation> transfers; container.getOutputs(transfers, ITransfersContainer::IncludeAllUnlocked); ASSERT_EQ(1, transfers.size()); ASSERT_EQ(TEST_OUTPUT_AMOUNT, container.balance(ITransfersContainer::IncludeAllUnlocked)); TestTransactionBuilder builder; builder.addInput(account, transfers[0]); auto tx = builder.build(); ASSERT_TRUE(container.addTransaction(blockInfo(WALLET_LEGACY_UNCONFIRMED_TRANSACTION_HEIGHT), *tx, {})); ASSERT_EQ(2, container.transactionsCount()); ASSERT_EQ(1, container.transfersCount()); ASSERT_EQ(0, container.balance(ITransfersContainer::IncludeAllUnlocked)); ASSERT_TRUE(markConfirmed(tx->getTransactionHash())); ASSERT_EQ(2, container.transactionsCount()); ASSERT_EQ(1, container.transfersCount()); ASSERT_EQ(0, container.balance(ITransfersContainer::IncludeAll)); }
TEST_F(TransfersContainer_addTransaction, addingEmptyTransactionOuptutsDoesNotChaingeContainer) { CryptoNote::TransactionBlockInfo blockInfo{ WALLET_LEGACY_UNCONFIRMED_TRANSACTION_HEIGHT, 1000000 }; TestTransactionBuilder builder; builder.addTestInput(TEST_OUTPUT_AMOUNT + 1, account); builder.addTestKeyOutput(TEST_OUTPUT_AMOUNT, TEST_TRANSACTION_OUTPUT_GLOBAL_INDEX); auto tx = builder.build(); std::vector<TransactionOutputInformationIn> outputs; ASSERT_FALSE(container.addTransaction(blockInfo, *tx, outputs)); ASSERT_EQ(0, container.transfersCount()); ASSERT_EQ(0, container.transactionsCount()); ASSERT_EQ(0, container.balance(ITransfersContainer::IncludeAllLocked)); ASSERT_EQ(0, container.balance(ITransfersContainer::IncludeAllUnlocked)); std::vector<TransactionOutputInformation> transfers; container.getOutputs(transfers, ITransfersContainer::IncludeAllLocked); ASSERT_TRUE(transfers.empty()); transfers.clear(); container.getOutputs(transfers, ITransfersContainer::IncludeAllUnlocked); ASSERT_TRUE(transfers.empty()); transfers = container.getTransactionOutputs(tx->getTransactionHash(), ITransfersContainer::IncludeAll); ASSERT_TRUE(transfers.empty()); TransactionInformation txInfo; int64_t txBalance; ASSERT_FALSE(container.getTransactionInformation(tx->getTransactionHash(), txInfo, txBalance)); std::vector<Crypto::Hash> unconfirmedTransactions; container.getUnconfirmedTransactions(unconfirmedTransactions); ASSERT_TRUE(unconfirmedTransactions.empty()); }
TEST_F(TransfersContainer_balance, filtersByStateAndKeySimultaneously) { TestTransactionBuilder tx; tx.addTestInput(AMOUNT_1 + AMOUNT_2 + 1); auto outInfo1 = tx.addTestKeyOutput(AMOUNT_1, UNCONFIRMED_TRANSACTION_GLOBAL_OUTPUT_INDEX, account); auto outInfo2 = tx.addTestMultisignatureOutput(AMOUNT_2, UNCONFIRMED_TRANSACTION_GLOBAL_OUTPUT_INDEX); ASSERT_TRUE(container.addTransaction(blockInfo(WALLET_LEGACY_UNCONFIRMED_TRANSACTION_HEIGHT), *tx.build(), { outInfo1, outInfo2 })); auto tx2 = addTransaction(TEST_BLOCK_HEIGHT, AMOUNT_1 + AMOUNT_2); container.advanceHeight(TEST_CONTAINER_CURRENT_HEIGHT); ASSERT_EQ(AMOUNT_1, container.balance(ITransfersContainer::IncludeStateLocked | ITransfersContainer::IncludeTypeKey)); ASSERT_EQ(AMOUNT_2, container.balance(ITransfersContainer::IncludeStateLocked | ITransfersContainer::IncludeTypeMultisignature)); ASSERT_EQ(AMOUNT_1 + AMOUNT_2, container.balance(ITransfersContainer::IncludeStateUnlocked | ITransfersContainer::IncludeTypeKey)); }
TEST_F(TransfersContainer_addTransaction, spendingUnconfirmedOutputFails) { auto tx = addTransaction(WALLET_LEGACY_UNCONFIRMED_TRANSACTION_HEIGHT); ASSERT_EQ(1, container.transactionsCount()); ASSERT_EQ(1, container.transfersCount()); auto outputs = container.getTransactionOutputs( tx->getTransactionHash(), ITransfersContainer::IncludeAll); ASSERT_EQ(1, outputs.size()); TestTransactionBuilder spendingTx; for (const auto& t : outputs) { spendingTx.addInput(account, t); } ASSERT_ANY_THROW(container.addTransaction(blockInfo(WALLET_LEGACY_UNCONFIRMED_TRANSACTION_HEIGHT), *spendingTx.build() , {})); }
TEST_F(TransfersContainer_addTransaction, ignoresUnrelatedTransactionsWithMultisignatureInput) { TestTransactionBuilder tx; tx.addFakeMultisignatureInput(TEST_OUTPUT_AMOUNT, TEST_TRANSACTION_OUTPUT_GLOBAL_INDEX, 1); ASSERT_FALSE(container.addTransaction(blockInfo(TEST_BLOCK_HEIGHT), *tx.build(), {})); }
TEST_F(TransfersContainer_addTransaction, ignoresUnrelatedTransactionsWithKeyInput) { TestTransactionBuilder tx; tx.addTestInput(TEST_OUTPUT_AMOUNT, account); ASSERT_FALSE(container.addTransaction(blockInfo(TEST_BLOCK_HEIGHT), *tx.build(), {})); }
TEST_F(TransfersContainer_addTransaction, handlesAddingUnconfirmedOutputMultisignature) { TestTransactionBuilder tx; auto out = tx.addTestMultisignatureOutput(TEST_OUTPUT_AMOUNT, UNCONFIRMED_TRANSACTION_GLOBAL_OUTPUT_INDEX); ASSERT_TRUE(container.addTransaction(blockInfo(WALLET_LEGACY_UNCONFIRMED_TRANSACTION_HEIGHT), *tx.build(), { out })); ASSERT_EQ(1, container.transactionsCount()); ASSERT_EQ(1, container.transfersCount()); ASSERT_EQ(TEST_OUTPUT_AMOUNT, container.balance(ITransfersContainer::IncludeAll)); ASSERT_EQ(TEST_OUTPUT_AMOUNT, container.balance(ITransfersContainer::IncludeTypeMultisignature | ITransfersContainer::IncludeStateLocked)); ASSERT_EQ(0, container.balance(ITransfersContainer::IncludeTypeMultisignature | ITransfersContainer::IncludeStateUnlocked)); }
TEST_F(TransfersContainer_getOutputs, filtersByStateAndKeySimultaneously) { TestTransactionBuilder tx; tx.addTestInput(AMOUNT_1 + AMOUNT_2 + 1); auto outInfo1 = tx.addTestKeyOutput(AMOUNT_1, UNCONFIRMED_TRANSACTION_GLOBAL_OUTPUT_INDEX, account); auto outInfo2 = tx.addTestMultisignatureOutput(AMOUNT_2, UNCONFIRMED_TRANSACTION_GLOBAL_OUTPUT_INDEX); ASSERT_TRUE(container.addTransaction(blockInfo(WALLET_LEGACY_UNCONFIRMED_TRANSACTION_HEIGHT), *tx.build(), { outInfo1, outInfo2 })); auto tx2 = addTransaction(TEST_BLOCK_HEIGHT, AMOUNT_1 + AMOUNT_2); container.advanceHeight(TEST_CONTAINER_CURRENT_HEIGHT); std::vector<TransactionOutputInformation> transfers; container.getOutputs(transfers, ITransfersContainer::IncludeStateLocked | ITransfersContainer::IncludeTypeKey); ASSERT_EQ(1, transfers.size()); ASSERT_EQ(AMOUNT_1, transfers.front().amount); transfers.clear(); container.getOutputs(transfers, ITransfersContainer::IncludeStateLocked | ITransfersContainer::IncludeTypeMultisignature); ASSERT_EQ(1, transfers.size()); ASSERT_EQ(AMOUNT_2, transfers.front().amount); transfers.clear(); container.getOutputs(transfers, ITransfersContainer::IncludeStateUnlocked | ITransfersContainer::IncludeTypeKey); ASSERT_EQ(1, transfers.size()); ASSERT_EQ(AMOUNT_1 + AMOUNT_2, transfers.front().amount); }