Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Debug Console implementation of generate method #692

Open
wants to merge 2 commits into
base: master
Choose a base branch
from

Conversation

hernanmarino
Copy link
Contributor

@hernanmarino hernanmarino commented Dec 28, 2022

This is an implementation of a (gui console only) generate method, and fixes #55 , as described in that issue.
This changes the behaviour of bitcoin-qt while imitating the behaviour and return data from bitcoin-cli's -generate argument as implemented in bitcoin/bitcoin#19133.

The generate comand takes two optional parameters:

generate ( nblocks   ( maxtries) )
Mine a specified amount of blocks to own ( on the fly generated ) address.

Arguments:
1. nblocks    (numeric, optional, , default=1) How many blocks are generated.
2. maxtries   (numeric, optional, default=1000000) How many iterations to try.

Result:
{
  "address":          (address generated by this command)
  "blocks": [           (json array of block hashes)
     ....
]
}

The output looks similar to the following :

image

Note to reviewers / testers:

  • Consider that this is a bitcoin-qt only implementation . bitcoin-cli already had this functionality implemented in the referenced PR.
  • It is advised to run bitcoin-qt in regtest, to be able to actually generate some blocks and test the output properly

@DrahtBot
Copy link
Contributor

DrahtBot commented Dec 28, 2022

The following sections might be updated with supplementary metadata relevant to reviewers and maintainers.

Reviews

See the guideline for information on the review process.

Type Reviewers
Concept ACK jarolrod, hebasto, rserranon, LarryRuane, furszy
Approach NACK luke-jr
Stale ACK RandyMcMillan, pablomartin4btc

If your review is incorrectly listed, please react with 👎 to this comment and the bot will ignore it on the next update.

Copy link
Contributor

@pablomartin4btc pablomartin4btc left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Code Review ACK. It's good to make the -generate command available at the rpc console level in bitcoin-qt as it's already in bitcoin-cli.

src/qt/rpcconsole.cpp Outdated Show resolved Hide resolved
Copy link
Contributor

@pablomartin4btc pablomartin4btc left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

b7782a4 ACK.

src/qt/rpcconsole.cpp Outdated Show resolved Hide resolved
@RandyMcMillan
Copy link
Contributor

RandyMcMillan commented Dec 29, 2022

ConceptACK

:)


Additional logic to detect whether `-generate` is or is not followed by an integer would be great.

![generate-gui-console](https://user-images.githubusercontent.com/152159/209985593-0fe468fc-038b-44d2-90fc-303ecc3944d4.png)

Slightly better syntax suggestion in the error message would be excellent. 

@hernanmarino
Copy link
Contributor Author

hernanmarino commented Dec 30, 2022

Hi @RandyMcMillan . Thank you for taking the time to look at my PR. The changes I implemented are actually on bitcoin-qt and not on bitcoin-cli . I updated the PR description to make it clearer. I believe the issues you describe are not present here, but you are welcome to test them, please let me know if you find anything.

src/qt/rpcconsole.cpp Outdated Show resolved Hide resolved
src/qt/rpcconsole.cpp Outdated Show resolved Hide resolved
src/qt/rpcconsole.cpp Outdated Show resolved Hide resolved
src/qt/rpcconsole.cpp Outdated Show resolved Hide resolved
@jarolrod jarolrod added Feature UX All about "how to get things done" labels Jan 3, 2023
Copy link
Contributor

@pablomartin4btc pablomartin4btc left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Tested ACK 9b0b762.
Checked output with different params (amount of blocks, tries), compared with outputs from bitcoin-cli as well, all good!

Copy link
Contributor

@pablomartin4btc pablomartin4btc left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Is it possible for someone to link this PR to the issue #55 if appropriate please?

Copy link
Member

@jarolrod jarolrod left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Concept ACK

Did some light testing, seems to work as intended.

Some window dressing, will review more deeply soon. First, please squash the commits so that you have a clean commit history; and give the commit a nice descriptive name like: qt: add generate command to gui console

After you've squashed the commits, run the clang-format-diff script to clean up some of the code formatting issues here, for convenience you could just run the following:

git diff -U0 HEAD~1.. | ./contrib/devtools/clang-format-diff.py -p1 -i -v

Additionally, the command could use a help generate, currently this results in a parse error as there is no help for it.

Copy link
Member

@furszy furszy left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Concept ACK

Could re-write it, without the regex stuff, a bit simpler:

std::vector<std::string> split_command = SplitString(executableCommand, " ");
if (!split_command.empty() && split_command[0] == "generate") {
    // Remove last "\n"
    std::string last_param = split_command[split_command.size()-1];
    split_command[split_command.size()-1] = last_param.substr(0, last_param.size() - 1);

    // Generate address
    std::string address_result;
    if (!RPCConsole::RPCExecuteCommandLine(m_node, address_result, "getnewaddress\n", nullptr, wallet_model)) {
        Q_EMIT reply(RPCConsole::CMD_ERROR, QString("Error: cannot get new address"));
         return;
    }

    // Craft command
    const std::string blocks_num = split_command.size() > 1 ? split_command[1] : "1";
    const std::string max_tries = split_command.size() > 2 ? split_command[2] : "";
    executableCommand = "generatetoaddress " + blocks_num + " \"" + address_result + "\" " + max_tries;
}

@hernanmarino
Copy link
Contributor Author

Thanks for all the insights. In the first days of the next week I'll try to incorporate the feedback and squash the commits.

@hebasto hebasto changed the title GUI: Debug Console implementation of generate method Debug Console implementation of generate method Jan 15, 2023
@hebasto
Copy link
Member

hebasto commented Jan 15, 2023

Concept ACK.

@hernanmarino You could also link the initial issue in the PR description.

@rserranon
Copy link

Concept ACK

Could re-write it, without the regex stuff, a bit simpler:

std::vector<std::string> split_command = SplitString(executableCommand, " ");
if (!split_command.empty() && split_command[0] == "generate") {
    // Remove last "\n"
    std::string last_param = split_command[split_command.size()-1];
    split_command[split_command.size()-1] = last_param.substr(0, last_param.size() - 1);

    // Generate address
    std::string address_result;
    if (!RPCConsole::RPCExecuteCommandLine(m_node, address_result, "getnewaddress\n", nullptr, wallet_model)) {
        Q_EMIT reply(RPCConsole::CMD_ERROR, QString("Error: cannot get new address"));
         return;
    }

    // Craft command
    const std::string blocks_num = split_command.size() > 1 ? split_command[1] : "1";
    const std::string max_tries = split_command.size() > 2 ? split_command[2] : "";
    executableCommand = "generatetoaddress " + blocks_num + " \"" + address_result + "\" " + max_tries;
}

Does it make sense to include parsing logic, generate a new address, and then craft the command to call the generatetoaddress, inside the RPCExecutor::request function, or it should be done on a separate function?

@furszy
Copy link
Member

furszy commented Jan 16, 2023

Does it make sense to include parsing logic, generate a new address, and then craft the command to call the generatetoaddress, inside the RPCExecutor::request function, or it should be done on a separate function?

A separate function returning the command string would be better.

@RandyMcMillan
Copy link
Contributor

RandyMcMillan commented Jan 25, 2023

tested on top of a62231b

Tested ConceptACK 9b0b762

macOS: Arm64

Screen Shot 2023-01-24 at 7 12 54 PM

@RandyMcMillan
Copy link
Contributor

RandyMcMillan commented Jan 25, 2023

tested on top of a62231b

Tested ConceptACK 9b0b762

macOS: x86_64

Screen Shot 2023-01-24 at 7 31 04 PM

@RandyMcMillan
Copy link
Contributor

RandyMcMillan commented Jan 25, 2023

@hernanmarino - @rserranon 's suggestion is compelling...

https://github.com/RandyMcMillan/gui/tree/612a0c14173d681a04021581851422ba832236ee

Applied as a patch on top of this PR:
https://github.com/RandyMcMillan/gui/blob/612a0c14173d681a04021581851422ba832236ee/rserranon.patch

I haven't fully tested the patch but works on MacOS x86_64

@hernanmarino
Copy link
Contributor Author

Thanks for the feedback and testing. I ' m already working on an update consider this and all feedback received, will update soon.

@hernanmarino - @rserranon 's suggestion is compelling...

RandyMcMillan/gui@612a0c1

Applied as a patch on top of this PR: RandyMcMillan/gui@612a0c1/rserranon.patch

I haven't fully tested the patch but works on MacOS x86_64

@hernanmarino
Copy link
Contributor Author

Summary of changes just pushed:

  • Changed the regex for a simpler string splitter, while still mantaining the syntax and several argument separators (beyond whitespace) supported by the console. Code now runs 180X faster (thanks @furszy and @andrewtoth for the suggestion)
  • Moved the implementation to a separate method that deals with these console-only commands (@rserranon)Also moved the already implemented "help-console" command handling to this method.
  • Added "help generate" code
  • Cleaned some code formatting issues according to clang-format ( thanks @jarolrod for the last two :) )
  • Squashed all changes on a single commit

@hernanmarino
Copy link
Contributor Author

Approach NACK. If we're going to go this route, we can just add back the RPC method... :/

I understand your point of view, but the reasons mentioned in the issues ( #55 bitcoin/bitcoin#16000 ), some of them mentioned by Pablo, encouraged me to work on this. Also, it evolved to (a litlle bit) more than bringing back the RPC call.

@DrahtBot DrahtBot requested review from LarryRuane and removed request for LarryRuane October 11, 2023 17:29
@hernanmarino
Copy link
Contributor Author

@maflcko @jonatack @Sjors since you opened or ACK'ed the original issues I kindly request your review on this PR.

@DrahtBot DrahtBot requested review from LarryRuane and luke-jr and removed request for LarryRuane October 15, 2023 21:20
@Sjors
Copy link
Member

Sjors commented Oct 23, 2023

Approach NACK. If we're going to go this route, we can just add back the RPC method... :/

IIUC the reason the RPC method was removed (and replaced with bitcoin-cli -generate) is because it had a wallet dependency. The GUI has wallet dependencies all over the place, so I don't see a problem adding this.

@DrahtBot DrahtBot requested review from LarryRuane and removed request for LarryRuane October 23, 2023 05:27
return true;
}
// Fail if we are on mainnet, to avoid generating addresses for blocks that will not be generated
if (Params().GetChainType() == ChainType::MAIN) {
Copy link
Member

@Sjors Sjors Oct 23, 2023

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Better to fail on all networks except regtest. It might be possible to mine on a custom signet too, but anyone who knows how to make such a network also knows how to use the RPC.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Agree, I'll change it soon

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

To access the chain type, use the node interface. See 03d67301e081ecf3123372901b115ee5e29d7c79. So we don't violate the layers division.

@DrahtBot DrahtBot requested review from LarryRuane and removed request for LarryRuane October 23, 2023 05:30
@hebasto
Copy link
Member

hebasto commented Nov 21, 2023

@hernanmarino
Copy link
Contributor Author

Rebased on top of master to fix CI errors.

Copy link
Member

@furszy furszy left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Executing generate 1 on a node without a wallet leads to a program crash. And left some other review notes.

Will read the historical context for the generate command deletion before proceed.

return true;
}
// Fail if we are on mainnet, to avoid generating addresses for blocks that will not be generated
if (Params().GetChainType() == ChainType::MAIN) {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This should pass through the model.

};

/**
* Small and fast parser supporting console command syntax, with limited functionality.
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The function iterates over the entire input command string twice. It isn't really "fast".
Could remove this line altogether.

Comment on lines +444 to +450
if (!RPCExecutor::executeConsoleOnlyCommand(command.toStdString(), wallet_model)) {
// Send to the RPC command parser if not console-only
if (!RPCConsole::RPCExecuteCommandLine(m_node, result, executableCommand, nullptr, wallet_model)) {
Q_EMIT reply(RPCConsole::CMD_ERROR, QString("Parse error: unbalanced ' or \""));
return;
}
Q_EMIT reply(RPCConsole::CMD_REPLY, QString::fromStdString(result));
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

There is an indentation issue here.

}

/**
* Catches console-only command before a RPC call is executed
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This shouldn't be here. The function's docstrings shouldn't explain the caller workflow behavior.
Something like "Executes gui-console-only commands" would be preferable.

return true;
}
// Fail if we are on mainnet, to avoid generating addresses for blocks that will not be generated
if (Params().GetChainType() == ChainType::MAIN) {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

To access the chain type, use the node interface. See 03d67301e081ecf3123372901b115ee5e29d7c79. So we don't violate the layers division.

return true;
}

// Catch the console-only generate command with 2 or less parameters before RPC call is executed .
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Same as above, this shouldn't be here. It is explaining the caller's workflow inside executeConsoleGenerate. There is no other RPC calls at this point.

Comment on lines 99 to 105
bool executeConsoleHelpConsole(const std::vector<std::string>& parsed_command, const WalletModel* wallet_model, const bool exec_help = false);
bool executeConsoleOnlyCommand(const std::string& command, const WalletModel* wallet_model);
// std::map mapping strings to methods member of RPCExecutor class
// Keys must be strings with commands and (optionally) parameters in "canonical" form (separated by single space)
// Keys should match the beggining of user input commands (user commands can have more parameters than the key)
std::map<std::string, bool (RPCExecutor::*)(const std::vector<std::string>&, const WalletModel*, const bool)> m_method_map{
{"help-console", &RPCExecutor::executeConsoleHelpConsole}};
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

In case you retouch this; these functions should return a util::Result<QString> instead of a boolean. This way, the returned string can be emitted at the caller side (same as it is done for the RPC commands). This will improve the existing structure that has all functions returning true on all code paths.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
Feature UX All about "how to get things done"
Projects
None yet
Development

Successfully merging this pull request may close these issues.

Add generate method to debug console