// 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& 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.out_reference); // if the following line fails, consider using prepare_outputs_entries_for_key_offsets() for correct calculation of real out index CHECK_AND_ASSERT_THROW_MES(absolute_sorted_output_offsets_to_relative_in_place(input_to_key.key_offsets), "absolute_sorted_output_offsets_to_relative_in_place failed"); m_tx.vin.push_back(input_to_key); } } void step3_fill_outputs(const std::vector& 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_bare 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& sources) { m_tx.signatures.clear(); size_t i = 0; for(const currency::tx_source_entry& src_entr : sources) { std::vector keys_ptrs; for(const currency::tx_source_entry::output_entry& o : src_entr.outputs) { keys_ptrs.push_back(&o.stealth_address); } m_tx.signatures.push_back(currency::NLSAG_sig()); std::vector& sigs = boost::get(m_tx.signatures.back()).s; sigs.resize(src_entr.outputs.size()); generate_ring_signature(m_tx_prefix_hash, boost::get(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& 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 sources; std::vector 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 m_in_contexts; crypto::hash m_tx_prefix_hash; };