forked from lthn/blockchain
157 lines
6 KiB
C++
157 lines
6 KiB
C++
// Copyright (c) 2014-2018 Zano Project
|
|
// Copyright (c) 2014-2018 The Louisdor Project
|
|
// Distributed under the MIT/X11 software license, see the accompanying
|
|
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
|
|
|
|
#pragma once
|
|
|
|
struct tx_builder
|
|
{
|
|
void step1_init(size_t version = TRANSACTION_VERSION_PRE_HF4, uint64_t unlock_time = 0)
|
|
{
|
|
m_tx = AUTO_VAL_INIT(m_tx);
|
|
m_in_contexts.clear();
|
|
|
|
m_tx.version = version;
|
|
set_tx_unlock_time(m_tx, unlock_time);
|
|
|
|
m_tx_key = currency::keypair::generate();
|
|
add_tx_pub_key_to_extra(m_tx, m_tx_key.pub);
|
|
}
|
|
|
|
void step2_fill_inputs(const currency::account_keys& sender_account_keys, const std::vector<currency::tx_source_entry>& sources)
|
|
{
|
|
for(const currency::tx_source_entry& src_entr : sources)
|
|
{
|
|
m_in_contexts.push_back(currency::keypair());
|
|
currency::keypair& in_ephemeral = m_in_contexts.back();
|
|
crypto::key_image img;
|
|
generate_key_image_helper(sender_account_keys, src_entr.real_out_tx_key, src_entr.real_output_in_tx_index, in_ephemeral, img);
|
|
|
|
// put key image into tx input
|
|
currency::txin_to_key input_to_key = AUTO_VAL_INIT(input_to_key);
|
|
input_to_key.amount = src_entr.amount;
|
|
input_to_key.k_image = img;
|
|
|
|
// fill outputs array and use relative offsets
|
|
for(const currency::tx_source_entry::output_entry& out_entry : src_entr.outputs)
|
|
input_to_key.key_offsets.push_back(out_entry.first);
|
|
|
|
input_to_key.key_offsets = currency::absolute_output_offsets_to_relative(input_to_key.key_offsets);
|
|
m_tx.vin.push_back(input_to_key);
|
|
}
|
|
}
|
|
|
|
void step3_fill_outputs(const std::vector<currency::tx_destination_entry>& destinations, uint8_t mix_attr = CURRENCY_TO_KEY_OUT_RELAXED, uint32_t minimum_sigs = UINT32_MAX)
|
|
{
|
|
size_t output_index = 0;
|
|
for(const currency::tx_destination_entry& dst_entr : destinations)
|
|
{
|
|
CHECK_AND_ASSERT_MES(!dst_entr.addr.empty(), void(0), "Destination entry #" << output_index << " contains empty addr list");
|
|
|
|
currency::tx_out out = AUTO_VAL_INIT(out);
|
|
out.amount = dst_entr.amount;
|
|
|
|
if (dst_entr.addr.size() == 1)
|
|
{
|
|
// one destination address - create txout_to_key
|
|
crypto::public_key out_eph_public_key = currency::null_pkey; // null_pkey means "burn" money
|
|
currency::account_public_address addr = dst_entr.addr.front();
|
|
if (addr.view_public_key != currency::null_pkey && addr.spend_public_key != currency::null_pkey)
|
|
{
|
|
crypto::key_derivation derivation = AUTO_VAL_INIT(derivation);
|
|
crypto::generate_key_derivation(addr.view_public_key, m_tx_key.sec, derivation);
|
|
crypto::derive_public_key(derivation, output_index, addr.spend_public_key, out_eph_public_key);
|
|
}
|
|
currency::txout_to_key tk = AUTO_VAL_INIT(tk);
|
|
tk.mix_attr = mix_attr;
|
|
tk.key = out_eph_public_key;
|
|
out.target = tk;
|
|
}
|
|
else
|
|
{
|
|
// many destination addresses - create txout_multisig
|
|
currency::txout_multisig ms = AUTO_VAL_INIT(ms);
|
|
ms.minimum_sigs = (minimum_sigs == UINT32_MAX) ? dst_entr.addr.size() : minimum_sigs;
|
|
for (auto& addr : dst_entr.addr)
|
|
{
|
|
crypto::key_derivation derivation = AUTO_VAL_INIT(derivation);
|
|
bool r = crypto::generate_key_derivation(addr.view_public_key, m_tx_key.sec, derivation);
|
|
CHECK_AND_ASSERT_MES(r, void(0), "generate_key_derivation failed for ms output #" << output_index);
|
|
crypto::public_key out_eph_public_key = AUTO_VAL_INIT(out_eph_public_key);
|
|
r = crypto::derive_public_key(derivation, output_index, addr.spend_public_key, out_eph_public_key);
|
|
CHECK_AND_ASSERT_MES(r, void(0), "derive_public_key failed for ms output #" << output_index);
|
|
ms.keys.push_back(out_eph_public_key);
|
|
}
|
|
out.target = ms;
|
|
}
|
|
|
|
m_tx.vout.push_back(out);
|
|
output_index++;
|
|
}
|
|
}
|
|
|
|
void step4_calc_hash()
|
|
{
|
|
get_transaction_prefix_hash(m_tx, m_tx_prefix_hash);
|
|
}
|
|
|
|
void step5_sign(const std::vector<currency::tx_source_entry>& sources)
|
|
{
|
|
m_tx.signatures.clear();
|
|
|
|
size_t i = 0;
|
|
for(const currency::tx_source_entry& src_entr : sources)
|
|
{
|
|
std::vector<const crypto::public_key*> keys_ptrs;
|
|
for(const currency::tx_source_entry::output_entry& o : src_entr.outputs)
|
|
{
|
|
keys_ptrs.push_back(&o.second);
|
|
}
|
|
|
|
m_tx.signatures.push_back(std::vector<crypto::signature>());
|
|
std::vector<crypto::signature>& sigs = m_tx.signatures.back();
|
|
sigs.resize(src_entr.outputs.size());
|
|
generate_ring_signature(m_tx_prefix_hash, boost::get<currency::txin_to_key>(m_tx.vin[i]).k_image, keys_ptrs, m_in_contexts[i].sec, src_entr.real_output, sigs.data());
|
|
i++;
|
|
}
|
|
}
|
|
|
|
static currency::transaction make_simple_tx_with_unlock_time(const std::vector<test_event_entry>& events,
|
|
const currency::block& blk_head, const currency::account_base& from, const currency::account_base& to,
|
|
uint64_t amount, uint64_t unlock_time, bool check_for_unlock_time = true)
|
|
{
|
|
std::vector<currency::tx_source_entry> sources;
|
|
std::vector<currency::tx_destination_entry> destinations;
|
|
fill_tx_sources_and_destinations(events, blk_head, from, to, amount, TESTS_DEFAULT_FEE, 0, sources, destinations, true, check_for_unlock_time);
|
|
|
|
tx_builder builder;
|
|
builder.step1_init(TRANSACTION_VERSION_PRE_HF4, unlock_time);
|
|
builder.step2_fill_inputs(from.get_keys(), sources);
|
|
builder.step3_fill_outputs(destinations);
|
|
builder.step4_calc_hash();
|
|
builder.step5_sign(sources);
|
|
return builder.m_tx;
|
|
};
|
|
|
|
static crypto::public_key generate_invalid_pub_key()
|
|
{
|
|
for (int i = 0; i <= 0xFF; ++i)
|
|
{
|
|
crypto::public_key key;
|
|
memset(&key, i, sizeof(crypto::public_key));
|
|
if (!crypto::check_key(key))
|
|
{
|
|
return key;
|
|
}
|
|
}
|
|
|
|
throw std::runtime_error("invalid public key wasn't found");
|
|
return crypto::public_key();
|
|
}
|
|
|
|
currency::transaction m_tx;
|
|
currency::keypair m_tx_key;
|
|
std::vector<currency::keypair> m_in_contexts;
|
|
crypto::hash m_tx_prefix_hash;
|
|
};
|