Allow simple multiwallet rpc calls #18734

pull jonasschnelli wants to merge 1 commits into bitcoin:master from jonasschnelli:2020/03/multiwallet_rpc changing 3 files +223 −1
  1. jonasschnelli commented at 2:05 pm on April 22, 2020: contributor

    Would fix #18715. See discussion in #18453.

    Currently, batching RPC requests across wallets does not work due to the fact that each wallet has its own endpoint.

    This PR allows using a simple wildcard approach for the wallet endpoint (/wallet/*/). Using bitcoin-cli -rpcwallet="*" or -rpcwallet="pattern* works with this PR.

    This PR is a simple approach that allows detailed control on a per call basis. Some calls probably need special logic (rescanblockchain). Some calls make not much sense across wallets (dumpwallet or backupwallet as example).

    The output (or error/s) will be bundles as json array with a dictionary for each wallet:

     0[
     1  {
     2    "walletname": "test",
     3    "result": 0.00000000
     4  },
     5  {
     6    "walletname": "wallet",
     7    "result": 0.00000000
     8  },
     9  {
    10    "walletname": "test2",
    11    "result": 0.00000000
    12  }
    13]
    

    Alternatives:

    • implementing a generic approach on our httpserver/rpcserver layer (complicates layering)
    • try to add it to the CRPCTable (overhead for non wallet calls)
    • a dedicated “foreachwallet” RPC meta call

    Todo:

    • tests
    • documentation
  2. Allow simple multiwallet rpc calls 57b449dc25
  3. jonasschnelli added the label Wallet on Apr 22, 2020
  4. jonasschnelli added the label RPC/REST/ZMQ on Apr 22, 2020
  5. in src/wallet/rpcwallet.cpp:415 in 57b449dc25
    410@@ -355,6 +411,10 @@ static CTransactionRef SendMoney(interfaces::Chain::Lock& locked_chain, CWallet
    411 
    412 static UniValue sendtoaddress(const JSONRPCRequest& request)
    413 {
    414+    if (IsMultiwalletJSONRPCRequest(request)) {
    415+        return ExecForeachWalletJSONRPCRequest(request, &sendtoaddress);
    


    promag commented at 2:14 pm on April 22, 2020:
    Do we want to incur in double payments?
  6. promag commented at 2:18 pm on April 22, 2020: member

    I still don’t get what are the use cases beside the obvious getbalances. Not sure about mutation calls, like send.

    Does it make sense to have one unified view in qt instead? That view could show all loaded wallets + their balances, flags etc.

  7. MarcoFalke commented at 2:40 pm on April 22, 2020: member
    I think @promag has a good point. Apart from get*, list* and keypoolrefill there is not really a use case for this endpoint.
  8. jonasschnelli commented at 2:46 pm on April 22, 2020: contributor

    I think @promag has a good point. Apart from get*, list* and keypoolrefill there is not really a use case for this endpoint.

    Fair point. I think this is why I would favour this approach which can be implemented per call. I guess there are some edge cases for other calls like set* (like settxfee), import*. But even only supporting the list* and get* calls,… it would be around half of the wallet calls.

  9. MarcoFalke commented at 2:51 pm on April 22, 2020: member

    Instead of the not-quite-regex-but-still-a-pattern I’d prefer the caller to explicitly list the wallets. I think requiring that the user calls listtransactions once to get the wallet names is not too much to ask.

    Also, the returned array could be a json dict with keys being the wallet name. That is easier to parse at the client.

  10. jonasschnelli commented at 3:21 pm on April 22, 2020: contributor

    Instead of the not-quite-regex-but-still-a-pattern I’d prefer the caller to explicitly list the wallets. I think requiring that the user calls listtransactions once to get the wallet names is not too much to ask.

    Probably meh to not support a pure “*” (all wallets). Maybe “*” or a cs-list of wallets?

    Also, the returned array could be a json dict with keys being the wallet name. That is easier to parse at the client.

    Good point. A downside could be the identifiability of an error (you already don’t get the http response code).

  11. DrahtBot commented at 7:45 pm on April 22, 2020: member

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

    Conflicts

    Reviewers, this pull request conflicts with the following ones:

    • #18946 (rpcwallet: Replace boost::optional::emplace with simple assignment of T{} by MarcoFalke)
    • #18654 (rpc: separate bumpfee’s psbt creation function into psbtbumpfee by achow101)
    • #18531 (rpc: Assert that RPCArg names are equal to CRPCCommand ones by MarcoFalke)

    If you consider this pull request important, please also help to review the conflicting pull requests. Ideally, start with the one that should be merged first.

  12. jb55 commented at 8:44 pm on April 22, 2020: member

    @promag:

    I still don’t get what are the use cases beside the obvious getbalances.

    listing transactions, coins. This what I currently do in scripts like btc-coins, but the round-trips are a bit slow with many wallets

    Not sure about mutation calls, like send.

    good point, that seems like a footgun. seems like this is more useful for not-mutating calls

    Does it make sense to have one unified view in qt instead? That view could show all loaded wallets + their balances, flags etc.

    I think that would be useful, but runs into similar issues when creating transactions where you probably want to know what wallet you’re creating from. So maybe it would make sense to limit it to balances and maybe transactions in a unified qt view? The logic for this could be tricky. @MarcoFalke:

    Instead of the not-quite-regex-but-still-a-pattern I’d prefer the caller to explicitly list the wallets. I think requiring that the user calls listtransactions once to get the wallet names is not too much to ask.

    :+1:

    Also, the returned array could be a json dict with keys being the wallet name. That is easier to parse at the client.

    :+1:

  13. in src/wallet/rpcdump.cpp:97 in 57b449dc25
    91@@ -92,6 +92,10 @@ static void RescanWallet(CWallet& wallet, const WalletRescanReserver& reserver,
    92 
    93 UniValue importprivkey(const JSONRPCRequest& request)
    94 {
    95+    if (IsMultiwalletJSONRPCRequest(request)) {
    96+        return ExecForeachWalletJSONRPCRequest(request, &importprivkey);
    97+    }
    


    luke-jr commented at 9:21 pm on April 24, 2020:
    Can we avoid adding this to every wallet function?
  14. in src/wallet/rpcdump.cpp:705 in 57b449dc25
    699@@ -676,6 +700,10 @@ UniValue importwallet(const JSONRPCRequest& request)
    700 
    701 UniValue dumpprivkey(const JSONRPCRequest& request)
    702 {
    703+    if (IsMultiwalletJSONRPCRequest(request)) {
    704+        return ExecForeachWalletJSONRPCRequest(request, &dumpprivkey);
    705+    }
    


    luke-jr commented at 9:23 pm on April 24, 2020:
    Something like dumpprivkey should probably just return the first result it gets (and ignore all the failures)?
  15. luke-jr commented at 9:26 pm on April 24, 2020: member

    Any reason not to return

    0{
    1    "walletname": result,
    2    ...
    3}
    4```?
    
  16. in src/wallet/rpcwallet.cpp:120 in 57b449dc25
    115+}
    116+
    117+UniValue ExecForeachWalletJSONRPCRequest(JSONRPCRequest mutable_request, const rpcfn_type& func) {
    118+    std::string search_string = URL_DECODE(mutable_request.URI.substr(WALLET_ENDPOINT_BASE.size()));
    119+    const size_t pos = search_string.find('*');
    120+    assert(pos != std::string::npos); // we must have checked earlier for an asterisk
    


    jonatack commented at 10:58 am on May 26, 2020:

    Needed to appease test/lint/lint-assertions.sh

    0-    assert(pos != std::string::npos); // we must have checked earlier for an asterisk
    1+    CHECK_NONFATAL(pos != std::string::npos); // we must have checked earlier for an asterisk
    
  17. jonatack commented at 11:20 am on May 26, 2020: member

    Tested Concept ACK 57b449d rebased on current master

    • I really like the rpcwallet=* wildcard feature; it would be my default use as a human user.

    • The ability to specify multiple wallets with rpcwallet= would be good also, especially for RPC clients/scripts, as a logical follow-up.

    • Agree with the idea of initially adding this only for get*, list*, and maybe keypoolrefill calls.

    • Agree with the suggestions to try returning this as a JSON dict of objects with wallet names as keys.

  18. fanquake commented at 12:52 pm on June 9, 2021: member
    Going to close this as “Up for Grabs” for now.
  19. fanquake closed this on Jun 9, 2021

  20. fanquake added the label Up for grabs on Jun 9, 2021
  21. DrahtBot locked this on Aug 18, 2022

github-metadata-mirror

This is a metadata mirror of the GitHub repository bitcoin/bitcoin. This site is not affiliated with GitHub. Content is generated from a GitHub metadata backup.
generated: 2025-01-21 06:12 UTC

This site is hosted by @0xB10C
More mirrored repositories can be found on mirror.b10c.me