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

[ZCash] Implement Zcash sync process and Orchard inputs spending #27018

Open
wants to merge 5 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
20 changes: 16 additions & 4 deletions components/brave_wallet/browser/BUILD.gn
Original file line number Diff line number Diff line change
Expand Up @@ -224,14 +224,20 @@ static_library("browser") {
"wallet_data_files_installer.cc",
"wallet_data_files_installer.h",
"wallet_data_files_installer_delegate.h",
"zcash/zcash_action_context.cc",
"zcash/zcash_action_context.h",
"zcash/zcash_block_tracker.cc",
"zcash/zcash_block_tracker.h",
"zcash/zcash_complete_transaction_task.cc",
"zcash/zcash_complete_transaction_task.h",
"zcash/zcash_create_transparent_transaction_task.cc",
"zcash/zcash_create_transparent_transaction_task.h",
"zcash/zcash_discover_next_unused_zcash_address_task.cc",
"zcash/zcash_discover_next_unused_zcash_address_task.h",
"zcash/zcash_get_transparent_utxos_context.cc",
"zcash/zcash_get_transparent_utxos_context.h",
"zcash/zcash_get_zcash_chain_tip_status_task.cc",
"zcash/zcash_get_zcash_chain_tip_status_task.h",
"zcash/zcash_grpc_utils.cc",
"zcash/zcash_grpc_utils.h",
"zcash/zcash_resolve_balance_task.cc",
Expand All @@ -242,8 +248,6 @@ static_library("browser") {
"zcash/zcash_serializer.h",
"zcash/zcash_transaction.cc",
"zcash/zcash_transaction.h",
"zcash/zcash_transaction_complete_manager.cc",
"zcash/zcash_transaction_complete_manager.h",
"zcash/zcash_transaction_utils.cc",
"zcash/zcash_transaction_utils.h",
"zcash/zcash_tx_manager.cc",
Expand Down Expand Up @@ -312,10 +316,18 @@ static_library("browser") {

if (enable_orchard) {
sources += [
"zcash/zcash_create_shield_transaction_task.cc",
"zcash/zcash_create_shield_transaction_task.h",
"zcash/zcash_blocks_batch_scan_task.cc",
"zcash/zcash_blocks_batch_scan_task.h",
"zcash/zcash_create_orchard_to_orchard_transaction_task.cc",
"zcash/zcash_create_orchard_to_orchard_transaction_task.h",
"zcash/zcash_create_transparent_to_orchard_transaction_task.cc",
"zcash/zcash_create_transparent_to_orchard_transaction_task.h",
"zcash/zcash_scan_blocks_task.cc",
"zcash/zcash_scan_blocks_task.h",
"zcash/zcash_shield_sync_service.cc",
"zcash/zcash_shield_sync_service.h",
"zcash/zcash_verify_chain_state_task.cc",
"zcash/zcash_verify_chain_state_task.h",
]

deps += [
Expand Down
4 changes: 4 additions & 0 deletions components/brave_wallet/browser/internal/hd_key_zip32.cc
Original file line number Diff line number Diff line change
Expand Up @@ -39,4 +39,8 @@ OrchardFullViewKey HDKeyZip32::GetFullViewKey() {
return orchard_extended_spending_key_->GetFullViewKey();
}

OrchardSpendingKey HDKeyZip32::GetSpendingKey() {
return orchard_extended_spending_key_->GetSpendingKey();
}

} // namespace brave_wallet
2 changes: 2 additions & 0 deletions components/brave_wallet/browser/internal/hd_key_zip32.h
Original file line number Diff line number Diff line change
Expand Up @@ -42,6 +42,8 @@ class HDKeyZip32 {
// Full view key(fvk) is used to decode incoming transactions
OrchardFullViewKey GetFullViewKey();

OrchardSpendingKey GetSpendingKey();

private:
explicit HDKeyZip32(std::unique_ptr<orchard::OrchardExtendedSpendingKey> key);
// Extended spending key is a root key of an account, all other keys can be
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -127,4 +127,40 @@ TEST(HDKeyZip32Test, FullViewKey) {
}
}

// Gen script:
// https://github.com/zcash/zcash-test-vectors/blob/1ac6808080e302ba2c7b02333d6fa3713f7d1f0e/zcash_test_vectors/orchard/zip32.py#L1
// Result:
// https://github.com/zcash/zcash-test-vectors/blob/1ac6808080e302ba2c7b02333d6fa3713f7d1f0e/test-vectors/rust/orchard_zip32.rs#L1
TEST(HDKeyZip32Test, SpendingKey) {
auto hd_key = HDKeyZip32::GenerateFromSeed(std::vector<uint8_t>(
{0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0x0a,
0x0b, 0x0c, 0x0d, 0x0e, 0x0f, 0x10, 0x11, 0x12, 0x13, 0x14, 0x15,
0x16, 0x17, 0x18, 0x19, 0x1a, 0x1b, 0x1c, 0x1d, 0x1e, 0x1f}));

{
constexpr OrchardSpendingKey expected = {
0x98, 0xd7, 0x03, 0xfc, 0xb4, 0x05, 0x04, 0xc9, 0x5b, 0x3b, 0x6e,
0xd1, 0x0e, 0xcd, 0x50, 0x08, 0x2c, 0xff, 0x97, 0xdf, 0xd1, 0xdd,
0x9a, 0xa0, 0x91, 0x3c, 0x78, 0xf9, 0x77, 0xc9, 0x62, 0xaf};
EXPECT_EQ(expected, hd_key->DeriveHardenedChild(1)->GetSpendingKey());
}
{
constexpr OrchardSpendingKey expected = {
0x98, 0xd7, 0x03, 0xfc, 0xb4, 0x05, 0x04, 0xc9, 0x5b, 0x3b, 0x6e,
0xd1, 0x0e, 0xcd, 0x50, 0x08, 0x2c, 0xff, 0x97, 0xdf, 0xd1, 0xdd,
0x9a, 0xa0, 0x91, 0x3c, 0x78, 0xf9, 0x77, 0xc9, 0x62, 0xaf};
EXPECT_EQ(expected, hd_key->DeriveHardenedChild(1)->GetSpendingKey());
}

{
constexpr OrchardSpendingKey expected = {
0x99, 0xaf, 0xd8, 0x89, 0x4b, 0xaa, 0xd5, 0x87, 0x84, 0xd0, 0xec,
0x08, 0xf5, 0x14, 0x8e, 0xe2, 0xc2, 0xa1, 0x7b, 0x2b, 0x29, 0x4b,
0x08, 0xef, 0x9e, 0x0a, 0x0c, 0xf1, 0x4b, 0xcc, 0x09, 0x20};
EXPECT_EQ(expected, hd_key->DeriveHardenedChild(1)
->DeriveHardenedChild(2)
->GetSpendingKey());
}
}

} // namespace brave_wallet
Original file line number Diff line number Diff line change
Expand Up @@ -22,12 +22,13 @@ std::optional<size_t> OrchardBundleManager::random_seed_for_testing_ =
// static
std::unique_ptr<OrchardBundleManager> OrchardBundleManager::Create(
base::span<const uint8_t> tree_state,
const OrchardSpendsBundle& spends_bundle,
const std::vector<OrchardOutput>& orchard_outputs) {
if (orchard_outputs.empty()) {
return nullptr;
}
auto bundle = orchard::OrchardUnauthorizedBundle::Create(
tree_state, orchard_outputs, random_seed_for_testing_);
tree_state, spends_bundle, orchard_outputs, random_seed_for_testing_);
if (!bundle) {
return nullptr;
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -45,6 +45,7 @@ class OrchardBundleManager {
// Returns in unauthorized state
static std::unique_ptr<OrchardBundleManager> Create(
base::span<const uint8_t> tree_state,
const OrchardSpendsBundle& spends_bundle,
const std::vector<OrchardOutput>& orchard_outputs);

static void OverrideRandomSeedForTesting(size_t seed) {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -27,8 +27,9 @@ TEST(OrchardBundleManagerTest, SingleOutput) {
127, 239, 163, 246, 227, 18, 158, 164, 223, 176, 169, 233, 135, 3, 166,
61, 171, 146, 149, 137, 214, 220, 81, 201, 112, 249, 53, 179}});

auto unauthorized_state =
OrchardBundleManager::Create(std::vector<uint8_t>(), std::move(outputs));
OrchardSpendsBundle orchard_spends_bundle;
auto unauthorized_state = OrchardBundleManager::Create(
std::vector<uint8_t>(), orchard_spends_bundle, std::move(outputs));
EXPECT_TRUE(unauthorized_state);
// Unauthorized state doesn't have raw tx bytes
EXPECT_FALSE(unauthorized_state->GetRawTxBytes());
Expand Down Expand Up @@ -324,8 +325,9 @@ TEST(OrchardBundleManagerTest, MultiplyOutputs) {
0x91, 0xd7, 0x34, 0xdf, 0x12, 0xd0, 0x46, 0xc9, 0x69, 0x75, 0x13,
0x30, 0xbb, 0xf4, 0x93, 0xa2, 0x41, 0xec, 0x4b, 0x88, 0xbc}});

auto unauthorized_state =
OrchardBundleManager::Create(std::vector<uint8_t>(), std::move(outputs));
OrchardSpendsBundle orchard_spends_bundle;
auto unauthorized_state = OrchardBundleManager::Create(
std::vector<uint8_t>(), orchard_spends_bundle, std::move(outputs));
EXPECT_TRUE(unauthorized_state);
// Unauthorized state doesn't have raw tx bytes
EXPECT_FALSE(unauthorized_state->GetRawTxBytes());
Expand Down Expand Up @@ -602,9 +604,11 @@ TEST(OrchardBundleManagerTest, MultiplyOutputs) {
TEST(OrchardBundleManagerTest, NoOutputs) {
OrchardBundleManager::OverrideRandomSeedForTesting(0);

OrchardSpendsBundle orchard_spends_bundle;
std::vector<OrchardOutput> outputs;
auto unauthorized_state = OrchardBundleManager::Create(
std::vector<uint8_t>(), std::vector<OrchardOutput>());
std::vector<uint8_t>(), orchard_spends_bundle,
std::vector<OrchardOutput>());
EXPECT_FALSE(unauthorized_state);
}

Expand Down
16 changes: 16 additions & 0 deletions components/brave_wallet/browser/internal/orchard_sync_state.cc
Original file line number Diff line number Diff line change
Expand Up @@ -134,6 +134,22 @@ OrchardSyncState::CalculateWitnessForCheckpoint(
return base::ok(std::move(result));
}

base::expected<std::optional<uint32_t>, OrchardStorage::Error>
OrchardSyncState::GetLatestShardIndex(const mojom::AccountIdPtr& account_id) {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
return storage_.GetLatestShardIndex(account_id);
}

base::expected<std::optional<uint32_t>, OrchardStorage::Error>
OrchardSyncState::GetMaxCheckpointedHeight(
const mojom::AccountIdPtr& account_id,
uint32_t chain_tip_height,
uint32_t min_confirmations) {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
return storage_.GetMaxCheckpointedHeight(account_id, chain_tip_height,
min_confirmations);
}

void OrchardSyncState::ResetDatabase() {
DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
storage_.ResetDatabase();
Expand Down
14 changes: 11 additions & 3 deletions components/brave_wallet/browser/internal/orchard_sync_state.h
Original file line number Diff line number Diff line change
Expand Up @@ -27,7 +27,7 @@ namespace brave_wallet {
class OrchardSyncState {
public:
explicit OrchardSyncState(const base::FilePath& path_to_database);
~OrchardSyncState();
virtual ~OrchardSyncState();

base::expected<OrchardStorage::AccountMeta, OrchardStorage::Error>
RegisterAccount(const mojom::AccountIdPtr& account_id,
Expand All @@ -42,7 +42,7 @@ class OrchardSyncState {
uint32_t reorg_block_id,
const std::string& reorg_block_hash);

base::expected<std::vector<OrchardNote>, OrchardStorage::Error>
virtual base::expected<std::vector<OrchardNote>, OrchardStorage::Error>
GetSpendableNotes(const mojom::AccountIdPtr& account_id);

base::expected<std::vector<OrchardNoteSpend>, OrchardStorage::Error>
Expand All @@ -56,14 +56,22 @@ class OrchardSyncState {
const uint32_t latest_scanned_block,
const std::string& latest_scanned_block_hash);

base::expected<std::optional<uint32_t>, OrchardStorage::Error>
GetLatestShardIndex(const mojom::AccountIdPtr& account_id);

virtual base::expected<std::optional<uint32_t>, OrchardStorage::Error>
GetMaxCheckpointedHeight(const mojom::AccountIdPtr& account_id,
uint32_t chain_tip_height,
uint32_t min_confirmations);

// Clears sync data related to the account except it's birthday.
base::expected<OrchardStorage::Result, OrchardStorage::Error>
ResetAccountSyncState(const mojom::AccountIdPtr& account_id);

// Drops underlying database.
void ResetDatabase();

base::expected<std::vector<OrchardInput>, OrchardStorage::Error>
virtual base::expected<std::vector<OrchardInput>, OrchardStorage::Error>
CalculateWitnessForCheckpoint(const mojom::AccountIdPtr& account_id,
const std::vector<OrchardInput>& notes,
uint32_t checkpoint_position);
Expand Down
11 changes: 11 additions & 0 deletions components/brave_wallet/browser/keyring_service.cc
Original file line number Diff line number Diff line change
Expand Up @@ -2212,6 +2212,17 @@ std::optional<OrchardFullViewKey> KeyringService::GetOrchardFullViewKey(

return zcash_keyring->GetOrchardFullViewKey(account_id->account_index);
}

std::optional<OrchardSpendingKey> KeyringService::GetOrchardSpendingKey(
const mojom::AccountIdPtr& account_id) {
auto* zcash_keyring = GetZCashKeyringById(account_id->keyring_id);
if (!zcash_keyring) {
return std::nullopt;
}

return zcash_keyring->GetOrchardSpendingKey(account_id->account_index);
}

#endif

void KeyringService::UpdateNextUnusedAddressForBitcoinAccount(
Expand Down
2 changes: 2 additions & 0 deletions components/brave_wallet/browser/keyring_service.h
Original file line number Diff line number Diff line change
Expand Up @@ -251,6 +251,8 @@ class KeyringService : public mojom::KeyringService {
const mojom::ZCashKeyIdPtr& key_id);
std::optional<OrchardFullViewKey> GetOrchardFullViewKey(
const mojom::AccountIdPtr& account_id);
std::optional<OrchardSpendingKey> GetOrchardSpendingKey(
const mojom::AccountIdPtr& account_id);
#endif

const std::vector<mojom::AccountInfoPtr>& GetAllAccountInfos();
Expand Down
5 changes: 5 additions & 0 deletions components/brave_wallet/browser/test/BUILD.gn
Original file line number Diff line number Diff line change
Expand Up @@ -105,6 +105,8 @@ source_set("brave_wallet_unit_tests") {
"//brave/components/brave_wallet/browser/unstoppable_domains_dns_resolve_unittest.cc",
"//brave/components/brave_wallet/browser/unstoppable_domains_multichain_calls_unittest.cc",
"//brave/components/brave_wallet/browser/wallet_data_files_installer_unittest.cc",
"//brave/components/brave_wallet/browser/zcash/zcash_create_orchard_to_orchard_transaction_task_unittest.cc",
"//brave/components/brave_wallet/browser/zcash/zcash_create_transparent_to_orchard_transaction_task_unittest.cc",
"//brave/components/brave_wallet/browser/zcash/zcash_grpc_utils_unittest.cc",
"//brave/components/brave_wallet/browser/zcash/zcash_keyring_unittest.cc",
"//brave/components/brave_wallet/browser/zcash/zcash_serializer_unittest.cc",
Expand Down Expand Up @@ -163,7 +165,10 @@ source_set("brave_wallet_unit_tests") {
"//brave/components/brave_wallet/browser/internal/orchard_bundle_manager_unittest.cc",
"//brave/components/brave_wallet/browser/internal/orchard_storage/orchard_storage_unittest.cc",
"//brave/components/brave_wallet/browser/internal/orchard_sync_state_unittest.cc",
"//brave/components/brave_wallet/browser/zcash/zcash_blocks_batch_scan_task_unittest.cc",
"//brave/components/brave_wallet/browser/zcash/zcash_scan_blocks_task_unittest.cc",
"//brave/components/brave_wallet/browser/zcash/zcash_shield_sync_service_unittest.cc",
"//brave/components/brave_wallet/browser/zcash/zcash_verify_chain_state_task_unittest.cc",
]
deps += [
"//brave/components/brave_wallet/browser/internal:orchard_bundle",
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -34,6 +34,8 @@ class OrchardExtendedSpendingKey {
uint32_t div_index,
OrchardAddressKind kind) = 0;

virtual OrchardSpendingKey GetSpendingKey() = 0;

virtual OrchardFullViewKey GetFullViewKey() = 0;
};

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -53,4 +53,8 @@ OrchardFullViewKey OrchardExtendedSpendingKeyImpl::GetFullViewKey() {
return cxx_extended_spending_key_->full_view_key();
}

OrchardSpendingKey OrchardExtendedSpendingKeyImpl::GetSpendingKey() {
return cxx_extended_spending_key_->spending_key();
}

} // namespace brave_wallet::orchard
Original file line number Diff line number Diff line change
Expand Up @@ -42,6 +42,8 @@ class OrchardExtendedSpendingKeyImpl : public OrchardExtendedSpendingKey {

OrchardFullViewKey GetFullViewKey() override;

OrchardSpendingKey GetSpendingKey() override;

private:
// Extended spending key is a root key of an account, all other keys can be
// derived from esk
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -30,6 +30,7 @@ class OrchardUnauthorizedBundle {
// Creates OrchardUnauthorizedBundle without shielded inputs
static std::unique_ptr<OrchardUnauthorizedBundle> Create(
base::span<const uint8_t> tree_state,
const ::brave_wallet::OrchardSpendsBundle& orchard_spends,
const std::vector<::brave_wallet::OrchardOutput>& orchard_outputs,
std::optional<size_t> random_seed_for_testing);

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -27,8 +27,27 @@ OrchardUnauthorizedBundleImpl::~OrchardUnauthorizedBundleImpl() = default;
// static
std::unique_ptr<OrchardUnauthorizedBundle> OrchardUnauthorizedBundle::Create(
base::span<const uint8_t> tree_state,
const ::brave_wallet::OrchardSpendsBundle& orchard_spends,
const std::vector<::brave_wallet::OrchardOutput>& orchard_outputs,
std::optional<size_t> random_seed_for_testing) {
::rust::Vec<orchard::CxxOrchardSpend> spends;
for (const auto& input : orchard_spends.inputs) {
if (!input.witness) {
return nullptr;
}

auto& note = input.note;

orchard::CxxMerklePath merkle_path;
merkle_path.position = input.witness->position;
for (const auto& merkle_hash : input.witness->merkle_path) {
merkle_path.auth_path.push_back(orchard::CxxMerkleHash{merkle_hash});
}
spends.push_back(orchard::CxxOrchardSpend{
orchard_spends.fvk, orchard_spends.sk, note.amount, note.addr, note.rho,
note.seed, std::move(merkle_path)});
}

::rust::Vec<orchard::CxxOrchardOutput> outputs;
for (const auto& output : orchard_outputs) {
outputs.push_back(orchard::CxxOrchardOutput{
Expand All @@ -39,8 +58,7 @@ std::unique_ptr<OrchardUnauthorizedBundle> OrchardUnauthorizedBundle::Create(
CHECK_IS_TEST();
auto bundle_result = create_testing_orchard_bundle(
::rust::Slice<const uint8_t>{tree_state.data(), tree_state.size()},
::rust::Vec<::brave_wallet::orchard::CxxOrchardSpend>(),
std::move(outputs), random_seed_for_testing.value());
std::move(spends), std::move(outputs), random_seed_for_testing.value());
if (!bundle_result->is_ok()) {
return nullptr;
}
Expand All @@ -50,8 +68,7 @@ std::unique_ptr<OrchardUnauthorizedBundle> OrchardUnauthorizedBundle::Create(
} else {
auto bundle_result = create_orchard_bundle(
::rust::Slice<const uint8_t>{tree_state.data(), tree_state.size()},
::rust::Vec<::brave_wallet::orchard::CxxOrchardSpend>(),
std::move(outputs));
std::move(spends), std::move(outputs));
if (!bundle_result->is_ok()) {
return nullptr;
}
Expand Down
31 changes: 31 additions & 0 deletions components/brave_wallet/browser/zcash/zcash_action_context.cc
Original file line number Diff line number Diff line change
@@ -0,0 +1,31 @@
/* Copyright (c) 2024 The Brave Authors. All rights reserved.
* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this file,
* You can obtain one at https://mozilla.org/MPL/2.0/. */

#include "brave/components/brave_wallet/browser/zcash/zcash_action_context.h"

namespace brave_wallet {

ZCashActionContext::ZCashActionContext(
ZCashRpc& zcash_rpc,
#if BUILDFLAG(ENABLE_ORCHARD)
base::SequenceBound<OrchardSyncState>& sync_state,
#endif // BUILDFLAG(ENABLE_ORCHARD)
const mojom::AccountIdPtr& account_id,
const std::string& chain_id)
: zcash_rpc(zcash_rpc),
#if BUILDFLAG(ENABLE_ORCHARD)
sync_state(sync_state),
#endif // BUILDFLAG(ENABLE_ORCHARD)
account_id(account_id.Clone()),
chain_id(chain_id) {
}

ZCashActionContext& ZCashActionContext::operator=(ZCashActionContext&&) =
default;
ZCashActionContext::ZCashActionContext(ZCashActionContext&&) = default;

ZCashActionContext::~ZCashActionContext() = default;

} // namespace brave_wallet
Loading
Loading