diff --git a/browser/brave_wallet/ethereum_provider_impl_unittest.cc b/browser/brave_wallet/ethereum_provider_impl_unittest.cc index 0e3bc07be4f5..becdcebc6101 100644 --- a/browser/brave_wallet/ethereum_provider_impl_unittest.cc +++ b/browser/brave_wallet/ethereum_provider_impl_unittest.cc @@ -125,6 +125,15 @@ std::vector DecodeHexHash(const std::string& hash_hex) { return hash; } +absl::optional ToValue(const network::ResourceRequest& request) { + base::StringPiece request_string(request.request_body->elements() + ->at(0) + .As() + .AsStringPiece()); + return base::JSONReader::Read(request_string, + base::JSONParserOptions::JSON_PARSE_RFC); +} + } // namespace class TestEventsListener : public brave_wallet::mojom::EventsListener { @@ -2171,6 +2180,70 @@ TEST_F(EthereumProviderImplUnitTest, EthSubscribeLogs) { EXPECT_FALSE(provider_->eth_logs_tracker_.IsRunning()); } +TEST_F(EthereumProviderImplUnitTest, EthSubscribeLogsFiltered) { + CreateWallet(); + url_loader_factory_.SetInterceptor( + base::BindLambdaForTesting([&](const network::ResourceRequest& request) { + url_loader_factory_.ClearResponses(); + + std::string header_value; + EXPECT_TRUE(request.headers.GetHeader("X-Eth-Method", &header_value)); + + if (header_value == "eth_getLogs") { + const absl::optional req_body_payload = + base::JSONReader::Read( + R"({"id":1,"jsonrpc":"2.0","method":"eth_getLogs","params": +[{"address":["0x1111", "0x1112"],"fromBlock":"0x2211","toBlock":"0xab65", +"topics":["0x2edc","0xb832","0x8dc8"]}]})", + base::JSON_PARSE_CHROMIUM_EXTENSIONS | + base::JSONParserOptions::JSON_PARSE_RFC); + + const auto payload = ToValue(request); + EXPECT_EQ(*payload, req_body_payload.value()); + } + url_loader_factory_.AddResponse( + request.url.spec(), + R"({"id":1,"jsonrpc":"2.0","result":[{"address":"0x91", + "blockHash":"0xe8","blockNumber":"0x10","data":"0x0067", + "logIndex":"0x0","removed":false, + "topics":["0x4b","0x06e","0x085"], + "transactionHash":"0x22f7","transactionIndex":"0x0"}]})"); + })); + + // Logs subscription with parameters + std::string request_payload_json = + R"({"id":1,"jsonrpc:": "2.0","method":"eth_subscribe", + "params": ["logs", {"address": ["0x1111", "0x1112"], "fromBlock": "0x2211", + "toBlock": "0xab65", "topics": ["0x2edc", "0xb832", "0x8dc8"]}]})"; + absl::optional request_payload = base::JSONReader::Read( + request_payload_json, base::JSON_PARSE_CHROMIUM_EXTENSIONS | + base::JSONParserOptions::JSON_PARSE_RFC); + std::string error_message; + auto response = CommonRequestOrSendAsync(request_payload.value()); + EXPECT_EQ(response.first, false); + EXPECT_TRUE(response.second.is_string()); + std::string subscription = *response.second.GetIfString(); + browser_task_environment_.FastForwardBy( + base::Seconds(kLogTrackerDefaultTimeInSeconds)); + EXPECT_TRUE(observer_->MessageEventFired()); + base::Value rv = observer_->GetLastMessage(); + ASSERT_TRUE(rv.is_dict()); + + std::string* address = rv.GetDict().FindString("address"); + EXPECT_EQ(*address, "0x91"); + + // The first unsubscribe should not stop the block tracker + request_payload_json = base::StringPrintf(R"({"id":1,"jsonrpc:": "2.0", + "method":"eth_unsubscribe", + "params": ["%s"]})", + subscription.c_str()); + request_payload = base::JSONReader::Read( + request_payload_json, base::JSON_PARSE_CHROMIUM_EXTENSIONS | + base::JSONParserOptions::JSON_PARSE_RFC); + response = CommonRequestOrSendAsync(request_payload.value()); + EXPECT_FALSE(provider_->eth_logs_tracker_.IsRunning()); +} + TEST_F(EthereumProviderImplUnitTest, Web3ClientVersion) { std::string expected_version = base::StringPrintf( "BraveWallet/v%s", version_info::GetBraveChromiumVersionNumber().c_str()); diff --git a/components/brave_wallet/browser/asset_discovery_manager.cc b/components/brave_wallet/browser/asset_discovery_manager.cc index 067b4080839c..ce1f19ee4b80 100644 --- a/components/brave_wallet/browser/asset_discovery_manager.cc +++ b/components/brave_wallet/browser/asset_discovery_manager.cc @@ -300,9 +300,13 @@ void AssetDiscoveryManager::OnGetEthTokenRegistry( weak_ptr_factory_.GetWeakPtr(), base::OwnedRef(std::move(tokens_to_search)), triggered_by_accounts_added, chain_id); - json_rpc_service_->EthGetLogs(chain_id, from_block, to_block, - std::move(contract_addresses_to_search), - std::move(topics), std::move(callback)); + base::Value::Dict filtering; + filtering.Set("fromBlock", from_block); + filtering.Set("toBlock", to_block); + filtering.Set("address", std::move(contract_addresses_to_search)); + filtering.Set("topics", std::move(topics)); + json_rpc_service_->EthGetLogs(chain_id, std::move(filtering), + std::move(callback)); } void AssetDiscoveryManager::OnGetTokenTransferLogs( diff --git a/components/brave_wallet/browser/eth_logs_tracker.cc b/components/brave_wallet/browser/eth_logs_tracker.cc index 8eca83c5154b..ef7eea287468 100644 --- a/components/brave_wallet/browser/eth_logs_tracker.cc +++ b/components/brave_wallet/browser/eth_logs_tracker.cc @@ -6,7 +6,7 @@ #include "brave/components/brave_wallet/browser/eth_logs_tracker.h" #include -#include +#include namespace brave_wallet { @@ -31,6 +31,16 @@ bool EthLogsTracker::IsRunning() const { return timer_.IsRunning(); } +void EthLogsTracker::AddSubscriber(const std::string& subscription_id, + base::Value::Dict filter) { + eth_logs_subscription_info_.insert(std::pair( + subscription_id, std::move(filter))); +} + +void EthLogsTracker::RemoveSubscriber(const std::string& subscription_id) { + eth_logs_subscription_info_.erase(subscription_id); +} + void EthLogsTracker::AddObserver(EthLogsTracker::Observer* observer) { observers_.AddObserver(observer); } @@ -42,18 +52,22 @@ void EthLogsTracker::RemoveObserver(EthLogsTracker::Observer* observer) { void EthLogsTracker::GetLogs() { const auto chain_id = json_rpc_service_->GetChainId(mojom::CoinType::ETH); - json_rpc_service_->EthGetLogs( - chain_id, {}, {}, {}, {}, - base::BindOnce(&EthLogsTracker::OnGetLogs, weak_factory_.GetWeakPtr())); + for (auto const& esi : std::as_const(eth_logs_subscription_info_)) { + json_rpc_service_->EthGetLogs( + chain_id, esi.second.Clone(), + base::BindOnce(&EthLogsTracker::OnGetLogs, weak_factory_.GetWeakPtr(), + esi.first)); + } } -void EthLogsTracker::OnGetLogs([[maybe_unused]] const std::vector& logs, +void EthLogsTracker::OnGetLogs(const std::string& subscription, + [[maybe_unused]] const std::vector& logs, base::Value rawlogs, mojom::ProviderError error, const std::string& error_message) { if (error == mojom::ProviderError::kSuccess && rawlogs.is_dict()) { for (auto& observer : observers_) - observer.OnLogsReceived(rawlogs.Clone()); + observer.OnLogsReceived(subscription, rawlogs.Clone()); } else { LOG(ERROR) << "OnGetLogs failed"; } diff --git a/components/brave_wallet/browser/eth_logs_tracker.h b/components/brave_wallet/browser/eth_logs_tracker.h index 02111b497713..d39a81c1848f 100644 --- a/components/brave_wallet/browser/eth_logs_tracker.h +++ b/components/brave_wallet/browser/eth_logs_tracker.h @@ -6,6 +6,7 @@ #ifndef BRAVE_COMPONENTS_BRAVE_WALLET_BROWSER_ETH_LOGS_TRACKER_H_ #define BRAVE_COMPONENTS_BRAVE_WALLET_BROWSER_ETH_LOGS_TRACKER_H_ +#include #include #include @@ -34,7 +35,8 @@ class EthLogsTracker { class Observer : public base::CheckedObserver { public: - virtual void OnLogsReceived(base::Value rawlogs) = 0; + virtual void OnLogsReceived(const std::string& subscription, + base::Value rawlogs) = 0; }; // If timer is already running, it will be replaced with new interval @@ -42,12 +44,17 @@ class EthLogsTracker { void Stop(); bool IsRunning() const; + void AddSubscriber(const std::string& subscription_id, + base::Value::Dict filter); + void RemoveSubscriber(const std::string& subscription_id); + void AddObserver(Observer* observer); void RemoveObserver(Observer* observer); private: void GetLogs(); - void OnGetLogs(const std::vector& logs, + void OnGetLogs(const std::string& subscription, + const std::vector& logs, base::Value rawlogs, mojom::ProviderError error, const std::string& error_message); @@ -55,6 +62,8 @@ class EthLogsTracker { base::RepeatingTimer timer_; raw_ptr json_rpc_service_ = nullptr; + std::map eth_logs_subscription_info_; + base::ObserverList observers_; base::WeakPtrFactory weak_factory_{this}; diff --git a/components/brave_wallet/browser/eth_requests.cc b/components/brave_wallet/browser/eth_requests.cc index 108863b9c5d4..184ef2feb26f 100644 --- a/components/brave_wallet/browser/eth_requests.cc +++ b/components/brave_wallet/browser/eth_requests.cc @@ -336,27 +336,8 @@ std::string eth_getFilterLogs(const std::string& filter_id) { return GetJsonRpcString("eth_getFilterLogs", filter_id); } -std::string eth_getLogs(const std::string& from_block_quantity_tag, - const std::string& to_block_quantity_tag, - base::Value::List addresses, - base::Value::List topics, - const std::string& block_hash) { +std::string eth_getLogs(base::Value::Dict filter_options) { base::Value::List params; - base::Value::Dict filter_options; - // The `address` filter option accepts either a single address, or a list of - // addresses (See spec: - // https://ethereum.org/en/developers/docs/apis/json-rpc/#eth_getlogs). At - // time of writing Infura's documentation suggests they only support a single - // address, however we have verified they also support a list. - if (!addresses.empty()) { - filter_options.Set("address", std::move(addresses)); - } - AddKeyIfNotEmpty(&filter_options, "fromBlock", from_block_quantity_tag); - AddKeyIfNotEmpty(&filter_options, "toBlock", to_block_quantity_tag); - if (!topics.empty()) { - filter_options.Set("topics", std::move(topics)); - } - AddKeyIfNotEmpty(&filter_options, "blockhash", block_hash); params.Append(std::move(filter_options)); base::Value::Dict dictionary = GetJsonRpcDictionary("eth_getLogs", std::move(params)); diff --git a/components/brave_wallet/browser/eth_requests.h b/components/brave_wallet/browser/eth_requests.h index 4501c3c384c7..399b939d4391 100644 --- a/components/brave_wallet/browser/eth_requests.h +++ b/components/brave_wallet/browser/eth_requests.h @@ -190,11 +190,7 @@ std::string eth_getFilterChanges(const std::string& filter_id); // Returns an array of all logs matching filter with given id. std::string eth_getFilterLogs(const std::string& filter_id); // Returns an array of all logs matching a given filter object. -std::string eth_getLogs(const std::string& from_block_quantity_tag, - const std::string& to_block_quantity_tag, - base::Value::List addresses, - base::Value::List topics, - const std::string& block_hash); +std::string eth_getLogs(base::Value::Dict filter_options); // Returns the hash of the current block, the seedHash, and the boundary // condition to be met (“target”). std::string eth_getWork(); diff --git a/components/brave_wallet/browser/eth_requests_unittest.cc b/components/brave_wallet/browser/eth_requests_unittest.cc index e1c80b223f9e..1fc718f26a26 100644 --- a/components/brave_wallet/browser/eth_requests_unittest.cc +++ b/components/brave_wallet/browser/eth_requests_unittest.cc @@ -337,10 +337,17 @@ TEST(EthRequestUnitTest, eth_getLogs) { base::Value::List addresses; addresses.Append(base::Value("0x8888f1f195afa192cfee860698584c030f4c9db1")); + + base::Value::Dict filtering; + filtering.Set("fromBlock", "0x1"); + filtering.Set("toBlock", "0x2"); + filtering.Set("address", std::move(addresses)); + filtering.Set("topics", std::move(topics)); + filtering.Set( + "blockhash", + "0xb903239f8543d04b5dc1ba6579132b143087c68db1b2168786408fcbce568238"); ASSERT_EQ( - eth_getLogs( - "0x1", "0x2", std::move(addresses), std::move(topics), - "0xb903239f8543d04b5dc1ba6579132b143087c68db1b2168786408fcbce568238"), + eth_getLogs(std::move(filtering)), R"({"id":1,"jsonrpc":"2.0","method":"eth_getLogs","params":[{"address":["0x8888f1f195afa192cfee860698584c030f4c9db1"],"blockhash":"0xb903239f8543d04b5dc1ba6579132b143087c68db1b2168786408fcbce568238","fromBlock":"0x1","toBlock":"0x2","topics":["0x000000000000000000000000a94f5374fce5edbc8e2a8697c15331677e6ebf0b",["0x000000000000000000000000a94f5374fce5edbc8e2a8697c15331677e6ebf0b","0x0000000000000000000000000aff3454fce5edbc8cca8697c15331677e6ebccc"]]}]})"); // NOLINT } diff --git a/components/brave_wallet/browser/ethereum_provider_impl.cc b/components/brave_wallet/browser/ethereum_provider_impl.cc index a3449a980969..5a950f5db932 100644 --- a/components/brave_wallet/browser/ethereum_provider_impl.cc +++ b/components/brave_wallet/browser/ethereum_provider_impl.cc @@ -578,9 +578,11 @@ void EthereumProviderImpl::RecoverAddress(const std::string& message, false); } -void EthereumProviderImpl::EthSubscribe(const std::string& event_type, - RequestCallback callback, - base::Value id) { +void EthereumProviderImpl::EthSubscribe( + const std::string& event_type, + absl::optional filter, + RequestCallback callback, + base::Value id) { const auto generateHexBytes = [](std::vector& subscriptions) { std::vector bytes(16); crypto::RandBytes(&bytes.front(), bytes.size()); @@ -597,11 +599,15 @@ void EthereumProviderImpl::EthSubscribe(const std::string& event_type, } std::move(callback).Run(std::move(id), base::Value(std::get<1>(gen_res)), false, "", false); - } else if (event_type == kEthSubscribeLogs) { + } else if (event_type == kEthSubscribeLogs && filter) { const auto gen_res = generateHexBytes(eth_log_subscriptions_); + if (std::get<0>(gen_res)) { eth_logs_tracker_.Start(base::Seconds(kLogTrackerDefaultTimeInSeconds)); } + + eth_logs_tracker_.AddSubscriber(std::get<1>(gen_res), std::move(*filter)); + std::move(callback).Run(std::move(id), base::Value(std::get<1>(gen_res)), false, "", false); } else { @@ -641,9 +647,10 @@ bool EthereumProviderImpl::UnsubscribeBlockObserver( bool EthereumProviderImpl::UnsubscribeLogObserver( const std::string& subscription_id) { if (base::Erase(eth_log_subscriptions_, subscription_id)) { - if (eth_log_subscriptions_.empty()) + eth_logs_tracker_.RemoveSubscriber(subscription_id); + if (eth_log_subscriptions_.empty()) { eth_logs_tracker_.Stop(); - + } return true; } return false; @@ -1237,12 +1244,15 @@ void EthereumProviderImpl::CommonRequestOrSendAsync(base::ValueView input_value, Web3ClientVersion(std::move(callback), std::move(id)); } else if (method == kEthSubscribe) { std::string event_type; - if (!ParseEthSubscribeParams(normalized_json_request, &event_type)) { + base::Value::Dict filter; + if (!ParseEthSubscribeParams(normalized_json_request, &event_type, + &filter)) { SendErrorOnRequest(error, error_message, std::move(callback), std::move(id)); return; } - EthSubscribe(event_type, std::move(callback), std::move(id)); + EthSubscribe(event_type, std::move(filter), std::move(callback), + std::move(id)); } else if (method == kEthUnsubscribe) { std::string subscription_id; if (!ParseEthUnsubscribeParams(normalized_json_request, &subscription_id)) { @@ -1705,7 +1715,8 @@ void EthereumProviderImpl::OnGetBlockByNumber( void EthereumProviderImpl::OnNewBlock(uint256_t block_num) {} -void EthereumProviderImpl::OnLogsReceived(base::Value rawlogs) { +void EthereumProviderImpl::OnLogsReceived(const std::string& subscription, + base::Value rawlogs) { if (!rawlogs.is_dict() || !events_listener_.is_bound()) { return; } @@ -1718,8 +1729,7 @@ void EthereumProviderImpl::OnLogsReceived(base::Value rawlogs) { } for (auto& results_item : *results) { - for (const auto& subscription_id : eth_log_subscriptions_) - events_listener_->MessageEvent(subscription_id, results_item.Clone()); + events_listener_->MessageEvent(subscription, results_item.Clone()); } } diff --git a/components/brave_wallet/browser/ethereum_provider_impl.h b/components/brave_wallet/browser/ethereum_provider_impl.h index 50e3614d8786..59d4fcfc7125 100644 --- a/components/brave_wallet/browser/ethereum_provider_impl.h +++ b/components/brave_wallet/browser/ethereum_provider_impl.h @@ -89,6 +89,7 @@ class EthereumProviderImpl final base::Value id); void EthSubscribe(const std::string& event_type, + absl::optional filter, RequestCallback callback, base::Value id); void EthUnsubscribe(const std::string& subscription_id, @@ -173,6 +174,8 @@ class EthereumProviderImpl final RequestEthereumPermissionsWithAccounts); FRIEND_TEST_ALL_PREFIXES(EthereumProviderImplUnitTest, EthSubscribe); FRIEND_TEST_ALL_PREFIXES(EthereumProviderImplUnitTest, EthSubscribeLogs); + FRIEND_TEST_ALL_PREFIXES(EthereumProviderImplUnitTest, + EthSubscribeLogsFiltered); friend class EthereumProviderImplUnitTest; // mojom::BraveWalletProvider: @@ -378,7 +381,8 @@ class EthereumProviderImpl final bool UnsubscribeBlockObserver(const std::string& subscription_id); // EthLogsTracker::Observer: - void OnLogsReceived(base::Value rawlogs) override; + void OnLogsReceived(const std::string& subscription, + base::Value rawlogs) override; bool UnsubscribeLogObserver(const std::string& subscription_id); raw_ptr host_content_settings_map_ = nullptr; diff --git a/components/brave_wallet/browser/json_rpc_service.cc b/components/brave_wallet/browser/json_rpc_service.cc index cb99db088931..122de90681f9 100644 --- a/components/brave_wallet/browser/json_rpc_service.cc +++ b/components/brave_wallet/browser/json_rpc_service.cc @@ -2168,10 +2168,7 @@ void JsonRpcService::GetERC1155TokenBalance( } void JsonRpcService::EthGetLogs(const std::string& chain_id, - const std::string& from_block, - const std::string& to_block, - base::Value::List contract_addresses, - base::Value::List topics, + base::Value::Dict filter_options, EthGetLogsCallback callback) { auto network_url = GetNetworkURL(prefs_, chain_id, mojom::CoinType::ETH); if (!network_url.is_valid()) { @@ -2184,10 +2181,8 @@ void JsonRpcService::EthGetLogs(const std::string& chain_id, auto internal_callback = base::BindOnce(&JsonRpcService::OnEthGetLogs, weak_ptr_factory_.GetWeakPtr(), std::move(callback)); - RequestInternal( - eth::eth_getLogs(from_block, to_block, std::move(contract_addresses), - std::move(topics), ""), - true, network_url, std::move(internal_callback)); + RequestInternal(eth::eth_getLogs(std::move(filter_options)), true, + network_url, std::move(internal_callback)); } void JsonRpcService::OnEthGetLogs(EthGetLogsCallback callback, diff --git a/components/brave_wallet/browser/json_rpc_service.h b/components/brave_wallet/browser/json_rpc_service.h index 95a33bb3589b..b09ad3391b31 100644 --- a/components/brave_wallet/browser/json_rpc_service.h +++ b/components/brave_wallet/browser/json_rpc_service.h @@ -330,10 +330,7 @@ class JsonRpcService : public KeyedService, public mojom::JsonRpcService { GetEthTokenUriCallback callback); void EthGetLogs(const std::string& chain_id, - const std::string& from_block, - const std::string& to_block, - base::Value::List addresses, - base::Value::List topics, + base::Value::Dict filter_options, EthGetLogsCallback callback); void OnEthGetLogs(EthGetLogsCallback callback, diff --git a/components/brave_wallet/browser/json_rpc_service_unittest.cc b/components/brave_wallet/browser/json_rpc_service_unittest.cc index 037a05b07b6c..f3b3c5c4b234 100644 --- a/components/brave_wallet/browser/json_rpc_service_unittest.cc +++ b/components/brave_wallet/browser/json_rpc_service_unittest.cc @@ -1160,9 +1160,13 @@ class JsonRpcServiceUnitTest : public testing::Test { mojom::ProviderError expected_error, const std::string& expected_error_message) { base::RunLoop run_loop; + base::Value::Dict params; + params.Set("fromBlock", from_block); + params.Set("toBlock", to_block); + params.Set("address", std::move(contract_addresses)); + params.Set("topics", std::move(topics)); json_rpc_service_->EthGetLogs( - chain_id, from_block, to_block, std::move(contract_addresses), - std::move(topics), + chain_id, std::move(params), base::BindLambdaForTesting( [&](const std::vector& logs, base::Value rawlogs, mojom::ProviderError error, const std::string& error_message) { diff --git a/components/brave_wallet/common/eth_request_helper.cc b/components/brave_wallet/common/eth_request_helper.cc index 4619bc9fffb8..4b049daeec20 100644 --- a/components/brave_wallet/common/eth_request_helper.cc +++ b/components/brave_wallet/common/eth_request_helper.cc @@ -677,19 +677,32 @@ bool ParseEthSendRawTransactionParams(const std::string& json, return true; } -bool ParseEthSubscribeParams(const std::string& json, std::string* event_type) { - if (!event_type) +bool ParseEthSubscribeParams(const std::string& json, + std::string* event_type, + base::Value::Dict* filter) { + if (filter == nullptr) { return false; + } auto list = GetParamsList(json); - if (!list || list->size() != 1) + if (!list || list->empty() || list->size() > 2) { return false; + } - const std::string* event_type_str = (*list)[0].GetIfString(); - if (!event_type_str) + if (!(*list)[0].is_string()) { return false; + } + + *event_type = (*list)[0].GetString(); + + if (list->size() == 2) { + if (!(*list)[1].is_dict()) { + return false; + } + + *filter = std::move((*list)[1].GetDict()); + } - *event_type = *event_type_str; return true; } diff --git a/components/brave_wallet/common/eth_request_helper.h b/components/brave_wallet/common/eth_request_helper.h index 76519990f21d..8425a2124afa 100644 --- a/components/brave_wallet/common/eth_request_helper.h +++ b/components/brave_wallet/common/eth_request_helper.h @@ -74,7 +74,9 @@ bool ParseRequestPermissionsParams( bool ParseEthSendRawTransactionParams(const std::string& json, std::string* signed_transaction); -bool ParseEthSubscribeParams(const std::string& json, std::string* event_type); +bool ParseEthSubscribeParams(const std::string& json, + std::string* event_type, + base::Value::Dict* filter); bool ParseEthUnsubscribeParams(const std::string& json, std::string* subscription_id);