5#include <bitcoin-build-config.h>
64 argsman.
AddArg(
"outmultisig=VALUE:REQUIRED:PUBKEYS:PUBKEY1:PUBKEY2:....[:FLAGS]",
"Add Pay To n-of-m Multi-sig output to TX. n = REQUIRED, m = PUBKEYS. "
65 "Optionally add the \"W\" flag to produce a pay-to-witness-script-hash output. "
67 argsman.
AddArg(
"outpubkey=VALUE:PUBKEY[:FLAGS]",
"Add pay-to-pubkey output to TX. "
68 "Optionally add the \"W\" flag to produce a pay-to-witness-pubkey-hash output. "
70 argsman.
AddArg(
"outscript=VALUE:SCRIPT[:FLAGS]",
"Add raw script output to TX. "
71 "Optionally add the \"W\" flag to produce a pay-to-witness-script-hash output. "
73 argsman.
AddArg(
"replaceable(=N)",
"Sets Replace-By-Fee (RBF) opt-in sequence number for input N. "
74 "If N is not provided, the command attempts to opt-in all available inputs for RBF. "
76 argsman.
AddArg(
"sign=SIGHASH-FLAGS",
"Add zero or more signatures to transaction. "
77 "This command requires JSON registers:"
78 "prevtxs=JSON object, "
79 "privatekeys=JSON object. "
95 tfm::format(std::cerr,
"Error parsing command line arguments: %s\n", error);
102 }
catch (
const std::exception& e) {
111 std::string strUsage = CLIENT_NAME
" bitcoin-tx utility version " +
FormatFullVersion() +
"\n";
117 "The bitcoin-tx tool is used for creating and modifying bitcoin transactions.\n\n"
118 "bitcoin-tx can be used with \"<hex-tx> [commands]\" to update a hex-encoded bitcoin transaction, or with \"-create [commands]\" to create a hex-encoded bitcoin transaction.\n"
120 "Usage: bitcoin-tx [options] <hex-tx> [commands]\n"
121 "or: bitcoin-tx [options] -create [commands]\n"
129 tfm::format(std::cerr,
"Error: too few parameters\n");
140 if (!val.
read(rawJson)) {
141 std::string strErr =
"Cannot parse JSON for key " + key;
142 throw std::runtime_error(strErr);
151 size_t pos = strInput.find(
':');
152 if ((pos == std::string::npos) ||
154 (pos == (strInput.size() - 1)))
155 throw std::runtime_error(
"Register input requires NAME:VALUE");
157 std::string key = strInput.substr(0, pos);
158 std::string valStr = strInput.substr(pos + 1, std::string::npos);
166 size_t pos = strInput.find(
':');
167 if ((pos == std::string::npos) ||
169 (pos == (strInput.size() - 1)))
170 throw std::runtime_error(
"Register load requires NAME:FILENAME");
172 std::string key = strInput.substr(0, pos);
173 std::string filename = strInput.substr(pos + 1, std::string::npos);
177 std::string strErr =
"Cannot open file " + filename;
178 throw std::runtime_error(strErr);
183 while ((!feof(f)) && (!ferror(f))) {
185 int bread = fread(buf, 1,
sizeof(buf), f);
189 valStr.insert(valStr.size(), buf, bread);
192 int error = ferror(f);
196 std::string strErr =
"Error reading file " + filename;
197 throw std::runtime_error(strErr);
206 if (std::optional<CAmount> parsed =
ParseMoney(strValue)) {
207 return parsed.value();
209 throw std::runtime_error(
"invalid TX output value");
215 const auto ver{ToIntegral<uint32_t>(cmdVal)};
217 throw std::runtime_error(
"Invalid TX version requested: '" + cmdVal +
"'");
224 const auto locktime{ToIntegral<uint32_t>(cmdVal)};
226 throw std::runtime_error(
"Invalid TX locktime requested: '" + cmdVal +
"'");
233 const auto idx{ToIntegral<uint32_t>(strInIdx)};
234 if (strInIdx !=
"" && (!idx || *idx >= tx.
vin.size())) {
235 throw std::runtime_error(
"Invalid TX input index '" + strInIdx +
"'");
241 if (strInIdx ==
"" || cnt == *idx) {
254 if (!parsed.has_value()) {
255 throw std::runtime_error(err +
" '" + int_str +
"'");
257 return parsed.value();
262 std::vector<std::string> vStrInputParts =
SplitString(strInput,
':');
265 if (vStrInputParts.size()<2)
266 throw std::runtime_error(
"TX input missing separator");
271 throw std::runtime_error(
"invalid TX input txid");
274 static const unsigned int minTxOutSz = 9;
278 const std::string& strVout = vStrInputParts[1];
279 const auto vout{ToIntegral<uint32_t>(strVout)};
280 if (!vout || *vout > maxVout) {
281 throw std::runtime_error(
"invalid TX input vout '" + strVout +
"'");
286 if (vStrInputParts.size() > 2) {
287 nSequenceIn = TrimAndParse<uint32_t>(vStrInputParts.at(2),
"invalid TX sequence id");
292 tx.
vin.push_back(txin);
298 std::vector<std::string> vStrInputParts =
SplitString(strInput,
':');
300 if (vStrInputParts.size() != 2)
301 throw std::runtime_error(
"TX output missing or too many separators");
307 const std::string& strAddr = vStrInputParts[1];
310 throw std::runtime_error(
"invalid TX output address");
315 CTxOut txout(value, scriptPubKey);
316 tx.
vout.push_back(txout);
322 std::vector<std::string> vStrInputParts =
SplitString(strInput,
':');
324 if (vStrInputParts.size() < 2 || vStrInputParts.size() > 3)
325 throw std::runtime_error(
"TX output missing or too many separators");
333 throw std::runtime_error(
"invalid TX output pubkey");
337 bool bSegWit =
false;
338 bool bScriptHash =
false;
339 if (vStrInputParts.size() == 3) {
340 const std::string&
flags = vStrInputParts[2];
341 bSegWit = (
flags.find(
'W') != std::string::npos);
342 bScriptHash = (
flags.find(
'S') != std::string::npos);
347 throw std::runtime_error(
"Uncompressed pubkeys are not useable for SegWit outputs");
358 CTxOut txout(value, scriptPubKey);
359 tx.
vout.push_back(txout);
365 std::vector<std::string> vStrInputParts =
SplitString(strInput,
':');
368 if (vStrInputParts.size()<3)
369 throw std::runtime_error(
"Not enough multisig parameters");
375 const uint32_t required{TrimAndParse<uint32_t>(vStrInputParts.at(1),
"invalid multisig required number")};
378 const uint32_t numkeys{TrimAndParse<uint32_t>(vStrInputParts.at(2),
"invalid multisig total number")};
381 if (vStrInputParts.size() < numkeys + 3)
382 throw std::runtime_error(
"incorrect number of multisig pubkeys");
385 throw std::runtime_error(
"multisig parameter mismatch. Required " \
389 std::vector<CPubKey> pubkeys;
390 for(
int pos = 1; pos <= int(numkeys); pos++) {
393 throw std::runtime_error(
"invalid TX output pubkey");
394 pubkeys.push_back(pubkey);
398 bool bSegWit =
false;
399 bool bScriptHash =
false;
400 if (vStrInputParts.size() == numkeys + 4) {
401 const std::string&
flags = vStrInputParts.back();
402 bSegWit = (
flags.find(
'W') != std::string::npos);
403 bScriptHash = (
flags.find(
'S') != std::string::npos);
405 else if (vStrInputParts.size() > numkeys + 4) {
407 throw std::runtime_error(
"Too many parameters");
413 for (
const CPubKey& pubkey : pubkeys) {
414 if (!pubkey.IsCompressed()) {
415 throw std::runtime_error(
"Uncompressed pubkeys are not useable for SegWit outputs");
431 CTxOut txout(value, scriptPubKey);
432 tx.
vout.push_back(txout);
440 size_t pos = strInput.find(
':');
443 throw std::runtime_error(
"TX output value not specified");
445 if (pos == std::string::npos) {
454 const std::string strData{strInput.substr(pos, std::string::npos)};
457 throw std::runtime_error(
"invalid TX output data");
462 tx.
vout.push_back(txout);
468 std::vector<std::string> vStrInputParts =
SplitString(strInput,
':');
469 if (vStrInputParts.size() < 2)
470 throw std::runtime_error(
"TX output missing separator");
476 const std::string& strScript = vStrInputParts[1];
480 bool bSegWit =
false;
481 bool bScriptHash =
false;
482 if (vStrInputParts.size() == 3) {
483 const std::string&
flags = vStrInputParts.back();
484 bSegWit = (
flags.find(
'W') != std::string::npos);
485 bScriptHash = (
flags.find(
'S') != std::string::npos);
505 CTxOut txout(value, scriptPubKey);
506 tx.
vout.push_back(txout);
511 const auto idx{ToIntegral<uint32_t>(strInIdx)};
512 if (!idx || idx >= tx.
vin.size()) {
513 throw std::runtime_error(
"Invalid TX input index '" + strInIdx +
"'");
515 tx.
vin.erase(tx.
vin.begin() + *idx);
520 const auto idx{ToIntegral<uint32_t>(strOutIdx)};
521 if (!idx || idx >= tx.
vout.size()) {
522 throw std::runtime_error(
"Invalid TX output index '" + strOutIdx +
"'");
524 tx.
vout.erase(tx.
vout.begin() + *idx);
558 throw std::runtime_error(
"Amount is not a number or string");
561 throw std::runtime_error(
"Invalid amount");
563 throw std::runtime_error(
"Amount out of range");
573 throw std::runtime_error(strName +
" must be hexadecimal string (not '" + strHex +
"')");
583 throw std::runtime_error(
"unknown sighash flag/sign option");
593 throw std::runtime_error(
"privatekeys register variable must be set.");
597 for (
unsigned int kidx = 0; kidx < keysObj.
size(); kidx++) {
598 if (!keysObj[kidx].isStr())
599 throw std::runtime_error(
"privatekey not a std::string");
602 throw std::runtime_error(
"privatekey not valid");
609 throw std::runtime_error(
"prevtxs register variable must be set.");
612 for (
unsigned int previdx = 0; previdx < prevtxsObj.
size(); previdx++) {
613 const UniValue& prevOut = prevtxsObj[previdx];
615 throw std::runtime_error(
"expected prevtxs internal object");
617 std::map<std::string, UniValue::VType> types = {
623 throw std::runtime_error(
"prevtxs internal object typecheck fail");
627 throw std::runtime_error(
"txid must be hexadecimal string (not '" + prevOut[
"txid"].get_str() +
"')");
630 const int nOut = prevOut[
"vout"].
getInt<
int>();
632 throw std::runtime_error(
"vout cannot be negative");
635 std::vector<unsigned char> pkData(
ParseHexUV(prevOut[
"scriptPubKey"],
"scriptPubKey"));
636 CScript scriptPubKey(pkData.begin(), pkData.end());
641 std::string err(
"Previous output scriptPubKey mismatch:\n");
644 throw std::runtime_error(err);
649 if (prevOut.
exists(
"amount")) {
659 prevOut.
exists(
"redeemScript")) {
660 UniValue v = prevOut[
"redeemScript"];
661 std::vector<unsigned char> rsData(
ParseHexUV(v,
"redeemScript"));
662 CScript redeemScript(rsData.begin(), rsData.end());
670 bool fHashSingle = ((nHashType & ~SIGHASH_ANYONECANPAY) ==
SIGHASH_SINGLE);
673 for (
unsigned int i = 0; i < mergedTx.vin.size(); i++) {
674 CTxIn& txin = mergedTx.vin[i];
684 if (!fHashSingle || (i < mergedTx.vout.size()))
688 throw std::runtime_error(
strprintf(
"Missing amount for CTxOut with scriptPubKey=%s",
HexStr(prevPubKey)));
698 const std::string& commandVal)
700 std::unique_ptr<ECC_Context> ecc;
704 else if (
command ==
"locktime")
706 else if (
command ==
"replaceable") {
719 else if (
command ==
"outpubkey") {
722 }
else if (
command ==
"outmultisig") {
725 }
else if (
command ==
"outscript")
742 throw std::runtime_error(
"unknown command");
750 std::string jsonOutput = entry.
write(4);
783 while (!feof(stdin)) {
784 size_t bread = fread(buf, 1,
sizeof(buf), stdin);
785 ret.append(buf, bread);
786 if (bread <
sizeof(buf))
791 throw std::runtime_error(
"error reading stdin");
814 throw std::runtime_error(
"too few parameters");
817 std::string strHexTx(argv[1]);
822 throw std::runtime_error(
"invalid transaction encoding");
828 for (
int i = startArg; i < argc; i++) {
829 std::string arg = argv[i];
830 std::string key, value;
831 size_t eqpos = arg.find(
'=');
832 if (eqpos == std::string::npos)
835 key = arg.substr(0, eqpos);
836 value = arg.substr(eqpos + 1);
844 catch (
const std::exception& e) {
845 strPrint = std::string(
"error: ") + e.what();
868 catch (
const std::exception& e) {
880 catch (
const std::exception& e) {
bool IsValidDestination(const CTxDestination &dest)
Check whether a CTxDestination corresponds to one with an address.
CScript GetScriptForDestination(const CTxDestination &dest)
Generate a Bitcoin scriptPubKey for the given CTxDestination.
std::variant< CNoDestination, PubKeyDestination, PKHash, ScriptHash, WitnessV0ScriptHash, WitnessV0KeyHash, WitnessV1Taproot, PayToAnchor, WitnessUnknown > CTxDestination
A txout script categorized into standard templates.
static constexpr CAmount MAX_MONEY
No amount larger than this (in satoshi) is valid.
bool MoneyRange(const CAmount &nValue)
int64_t CAmount
Amount in satoshis (Can be negative)
bool HelpRequested(const ArgsManager &args)
void SetupHelpOptions(ArgsManager &args)
Add help options to the args manager.
bool IsSwitchChar(char c)
static bool findSighashFlags(int &flags, const std::string &flagStr)
static void OutputTxHash(const CTransaction &tx)
static const unsigned int N_SIGHASH_OPTS
static void MutateTxSign(CMutableTransaction &tx, const std::string &flagStr)
static const int CONTINUE_EXECUTION
static std::string readStdin()
static void OutputTxJSON(const CTransaction &tx)
static void RegisterSet(const std::string &strInput)
static void RegisterSetJson(const std::string &key, const std::string &rawJson)
static CAmount ExtractAndValidateValue(const std::string &strValue)
static std::vector< unsigned char > ParseHexUV(const UniValue &v, const std::string &strName)
static void MutateTxDelOutput(CMutableTransaction &tx, const std::string &strOutIdx)
static const struct @0 sighashOptions[N_SIGHASH_OPTS]
static CAmount AmountFromValue(const UniValue &value)
static void MutateTx(CMutableTransaction &tx, const std::string &command, const std::string &commandVal)
static T TrimAndParse(const std::string &int_str, const std::string &err)
static void MutateTxAddOutPubKey(CMutableTransaction &tx, const std::string &strInput)
static void MutateTxRBFOptIn(CMutableTransaction &tx, const std::string &strInIdx)
static void MutateTxAddOutData(CMutableTransaction &tx, const std::string &strInput)
static void MutateTxVersion(CMutableTransaction &tx, const std::string &cmdVal)
static void MutateTxAddOutAddr(CMutableTransaction &tx, const std::string &strInput)
static int CommandLineRawTx(int argc, char *argv[])
const TranslateFn G_TRANSLATION_FUN
Translate string to current locale using Qt.
static void OutputTxHex(const CTransaction &tx)
static void RegisterLoad(const std::string &strInput)
static void MutateTxDelInput(CMutableTransaction &tx, const std::string &strInIdx)
static int AppInitRawTx(int argc, char *argv[])
static void MutateTxAddInput(CMutableTransaction &tx, const std::string &strInput)
static std::map< std::string, UniValue > registers
static void SetupBitcoinTxArgs(ArgsManager &argsman)
static void MutateTxAddOutMultiSig(CMutableTransaction &tx, const std::string &strInput)
static void MutateTxAddOutScript(CMutableTransaction &tx, const std::string &strInput)
static void MutateTxLocktime(CMutableTransaction &tx, const std::string &cmdVal)
static void OutputTx(const CTransaction &tx)
void SelectParams(const ChainType chain)
Sets the params returned by Params() to those for the given chain type.
void SetupChainParamsBaseOptions(ArgsManager &argsman)
Set the arguments for chainparams.
bool ParseParameters(int argc, const char *const argv[], std::string &error) EXCLUSIVE_LOCKS_REQUIRED(!cs_args)
ChainType GetChainType() const EXCLUSIVE_LOCKS_REQUIRED(!cs_args)
Returns the appropriate chain type from the program arguments.
@ ALLOW_ANY
disable validation
void AddArg(const std::string &name, const std::string &help, unsigned int flags, const OptionsCategory &cat) EXCLUSIVE_LOCKS_REQUIRED(!cs_args)
Add argument.
bool GetBoolArg(const std::string &strArg, bool fDefault) const EXCLUSIVE_LOCKS_REQUIRED(!cs_args)
Return boolean argument or default value.
std::string GetHelpMessage() const EXCLUSIVE_LOCKS_REQUIRED(!cs_args)
Get the help string.
CCoinsView that adds a memory cache for transactions to another CCoinsView.
void AddCoin(const COutPoint &outpoint, Coin &&coin, bool possible_overwrite)
Add a coin.
const Coin & AccessCoin(const COutPoint &output) const
Return a reference to Coin in the cache, or coinEmpty if not found.
Abstract view on the open txout dataset.
An encapsulated private key.
bool IsValid() const
Check whether this private key is valid.
An outpoint - a combination of a transaction hash and an index n into its vout.
An encapsulated public key.
bool IsCompressed() const
Check whether this is a compressed public key.
bool IsFullyValid() const
fully validate whether this is a valid public key (more expensive than IsValid())
Serialized script, used inside transaction inputs and outputs.
bool IsPayToScriptHash() const
bool IsPayToWitnessScriptHash() const
The basic transaction that is broadcasted on the network and contained in blocks.
const Txid & GetHash() const LIFETIMEBOUND
An input of a transaction.
static const uint32_t SEQUENCE_FINAL
Setting nSequence to this value for every input in a transaction disables nLockTime/IsFinalTx().
An output of a transaction.
CTxOut out
unspent transaction output
bool IsSpent() const
Either this coin never existed (see e.g.
uint32_t nHeight
at which height this containing transaction was included in the active block chain
RAII class initializing and deinitializing global state for elliptic curve support.
Fillable signing provider that keeps keys in an address->secret map.
virtual bool AddCScript(const CScript &redeemScript)
virtual bool AddKey(const CKey &key)
A signature creator for transactions.
bool checkObject(const std::map< std::string, UniValue::VType > &memberTypes) const
std::string write(unsigned int prettyIndent=0, unsigned int indentLevel=0) const
const std::string & getValStr() const
bool read(std::string_view raw)
bool exists(const std::string &key) const
std::string GetHex() const
static std::optional< transaction_identifier > FromHex(std::string_view hex)
std::string FormatFullVersion()
static const unsigned int MAX_BLOCK_WEIGHT
The maximum allowed weight for a block, see BIP 141 (network rule)
static const int WITNESS_SCALE_FACTOR
std::string EncodeHexTx(const CTransaction &tx)
CScript ParseScript(const std::string &s)
bool DecodeHexTx(CMutableTransaction &tx, const std::string &hex_tx, bool try_no_witness, bool try_witness)
void TxToUniv(const CTransaction &tx, const uint256 &block_hash, UniValue &entry, bool include_hex, const CTxUndo *txundo, TxVerbosity verbosity, std::function< bool(const CTxOut &)> is_change_func)
std::string ScriptToAsmStr(const CScript &script, const bool fAttemptSighashDecode)
Create the assembly string representation of a CScript object.
void PrintExceptionContinue(const std::exception *pex, std::string_view thread_name)
#define T(expected, seed, data)
std::string HexStr(const std::span< const uint8_t > s)
Convert a span of bytes to a lower-case hexadecimal string.
@ SIGHASH_DEFAULT
Taproot only; implied when sighash byte is missing, and equivalent to SIGHASH_ALL.
CTxDestination DecodeDestination(const std::string &str, std::string &error_msg, std::vector< int > *error_locations)
CKey DecodeSecret(const std::string &str)
std::string LicenseInfo()
Returns licensing information (for -version)
std::optional< CAmount > ParseMoney(const std::string &money_string)
Parse an amount denoted in full coins.
FILE * fopen(const fs::path &p, const char *mode)
std::vector< std::string > SplitString(std::string_view str, char sep)
std::string_view TrimStringView(std::string_view str, std::string_view pattern=" \f\n\r\t\v")
std::string ToString(const T &t)
Locale-independent version of std::to_string.
std::string TrimString(std::string_view str, std::string_view pattern=" \f\n\r\t\v")
static constexpr decltype(CTransaction::version) TX_MAX_STANDARD_VERSION
static const unsigned int MAX_SCRIPT_ELEMENT_SIZE
static const int MAX_SCRIPT_SIZE
static const int MAX_PUBKEYS_PER_MULTISIG
bool ProduceSignature(const SigningProvider &provider, const BaseSignatureCreator &creator, const CScript &fromPubKey, SignatureData &sigdata)
Produce a script signature using a generic signature creator.
void UpdateInput(CTxIn &input, const SignatureData &data)
SignatureData DataFromTransaction(const CMutableTransaction &tx, unsigned int nIn, const CTxOut &txout)
Extract signature data from a transaction input, and insert it.
CScript GetScriptForMultisig(int nRequired, const std::vector< CPubKey > &keys)
Generate a multisig script.
CScript GetScriptForRawPubKey(const CPubKey &pubKey)
Generate a P2PK script for the given pubkey.
std::vector< Byte > ParseHex(std::string_view hex_str)
Like TryParseHex, but returns an empty vector on invalid input.
A mutable version of CTransaction.
std::vector< CTxOut > vout
CScriptWitness scriptWitness
The scriptWitness of an input. Contains complete signatures or the traditional partial signatures for...
std::function< std::string(const char *)> TranslateFn
Translate a message to the native language of the user.
static constexpr uint32_t MAX_BIP125_RBF_SEQUENCE
bool ParseFixedPoint(std::string_view val, int decimals, int64_t *amount_out)
Parse number as fixed point according to JSON number syntax.
bool IsHex(std::string_view str)
std::string FormatParagraph(std::string_view in, size_t width, size_t indent)
Format a paragraph of text to a fixed width, adding spaces for indentation to any added line.