zcash_client_sqlite: Fix incorrect note deduplication in `v_transactions`
The `v_transactions` view is built upon the set of received notes, received note values being added to the balance for the transaction and spent notes being deducted from this balance. This fixes an error wherein if multiple identically-valued notes were spent in a transaction, only one of those notes' values was being counted as having been spent.
This commit is contained in:
parent
7a47f8494c
commit
2873dd4b60
|
@ -435,7 +435,8 @@ mod tests {
|
||||||
"CREATE VIEW v_transactions AS
|
"CREATE VIEW v_transactions AS
|
||||||
WITH
|
WITH
|
||||||
notes AS (
|
notes AS (
|
||||||
SELECT sapling_received_notes.account AS account_id,
|
SELECT sapling_received_notes.id_note AS id,
|
||||||
|
sapling_received_notes.account AS account_id,
|
||||||
transactions.block AS block,
|
transactions.block AS block,
|
||||||
transactions.txid AS txid,
|
transactions.txid AS txid,
|
||||||
2 AS pool,
|
2 AS pool,
|
||||||
|
@ -457,7 +458,8 @@ mod tests {
|
||||||
JOIN transactions
|
JOIN transactions
|
||||||
ON transactions.id_tx = sapling_received_notes.tx
|
ON transactions.id_tx = sapling_received_notes.tx
|
||||||
UNION
|
UNION
|
||||||
SELECT utxos.received_by_account AS account_id,
|
SELECT utxos.id_utxo AS id,
|
||||||
|
utxos.received_by_account AS account_id,
|
||||||
utxos.height AS block,
|
utxos.height AS block,
|
||||||
utxos.prevout_txid AS txid,
|
utxos.prevout_txid AS txid,
|
||||||
0 AS pool,
|
0 AS pool,
|
||||||
|
@ -467,7 +469,8 @@ mod tests {
|
||||||
0 AS memo_present
|
0 AS memo_present
|
||||||
FROM utxos
|
FROM utxos
|
||||||
UNION
|
UNION
|
||||||
SELECT sapling_received_notes.account AS account_id,
|
SELECT sapling_received_notes.id_note AS id,
|
||||||
|
sapling_received_notes.account AS account_id,
|
||||||
transactions.block AS block,
|
transactions.block AS block,
|
||||||
transactions.txid AS txid,
|
transactions.txid AS txid,
|
||||||
2 AS pool,
|
2 AS pool,
|
||||||
|
@ -479,7 +482,8 @@ mod tests {
|
||||||
JOIN transactions
|
JOIN transactions
|
||||||
ON transactions.id_tx = sapling_received_notes.spent
|
ON transactions.id_tx = sapling_received_notes.spent
|
||||||
UNION
|
UNION
|
||||||
SELECT utxos.received_by_account AS account_id,
|
SELECT utxos.id_utxo AS id,
|
||||||
|
utxos.received_by_account AS account_id,
|
||||||
transactions.block AS block,
|
transactions.block AS block,
|
||||||
transactions.txid AS txid,
|
transactions.txid AS txid,
|
||||||
0 AS pool,
|
0 AS pool,
|
||||||
|
|
|
@ -12,6 +12,7 @@ mod ufvk_support;
|
||||||
mod utxos_table;
|
mod utxos_table;
|
||||||
mod v_sapling_shard_unscanned_ranges;
|
mod v_sapling_shard_unscanned_ranges;
|
||||||
mod v_transactions_net;
|
mod v_transactions_net;
|
||||||
|
mod v_transactions_note_uniqueness;
|
||||||
mod v_transactions_shielding_balance;
|
mod v_transactions_shielding_balance;
|
||||||
mod v_transactions_transparent_history;
|
mod v_transactions_transparent_history;
|
||||||
mod v_tx_outputs_use_legacy_false;
|
mod v_tx_outputs_use_legacy_false;
|
||||||
|
@ -48,6 +49,8 @@ pub(super) fn all_migrations<P: consensus::Parameters + 'static>(
|
||||||
// v_sapling_shard_unscanned_ranges v_tx_outputs_use_legacy_false
|
// v_sapling_shard_unscanned_ranges v_tx_outputs_use_legacy_false
|
||||||
// | |
|
// | |
|
||||||
// wallet_summaries v_transactions_shielding_balance
|
// wallet_summaries v_transactions_shielding_balance
|
||||||
|
// |
|
||||||
|
// v_transactions_note_uniqueness
|
||||||
vec![
|
vec![
|
||||||
Box::new(initial_setup::Migration {}),
|
Box::new(initial_setup::Migration {}),
|
||||||
Box::new(utxos_table::Migration {}),
|
Box::new(utxos_table::Migration {}),
|
||||||
|
@ -82,5 +85,6 @@ pub(super) fn all_migrations<P: consensus::Parameters + 'static>(
|
||||||
Box::new(v_transactions_transparent_history::Migration),
|
Box::new(v_transactions_transparent_history::Migration),
|
||||||
Box::new(v_tx_outputs_use_legacy_false::Migration),
|
Box::new(v_tx_outputs_use_legacy_false::Migration),
|
||||||
Box::new(v_transactions_shielding_balance::Migration),
|
Box::new(v_transactions_shielding_balance::Migration),
|
||||||
|
Box::new(v_transactions_note_uniqueness::Migration),
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,160 @@
|
||||||
|
//! This migration fixes a bug in `v_transactions` where distinct but otherwise identical notes
|
||||||
|
//! were being incorrectly deduplicated.
|
||||||
|
|
||||||
|
use std::collections::HashSet;
|
||||||
|
|
||||||
|
use schemer_rusqlite::RusqliteMigration;
|
||||||
|
use uuid::Uuid;
|
||||||
|
|
||||||
|
use crate::wallet::init::WalletMigrationError;
|
||||||
|
|
||||||
|
use super::v_transactions_shielding_balance;
|
||||||
|
|
||||||
|
pub(super) const MIGRATION_ID: Uuid = Uuid::from_u128(0xdba47c86_13b5_4601_94b2_0cde0abe1e45);
|
||||||
|
|
||||||
|
pub(super) struct Migration;
|
||||||
|
|
||||||
|
impl schemer::Migration for Migration {
|
||||||
|
fn id(&self) -> Uuid {
|
||||||
|
MIGRATION_ID
|
||||||
|
}
|
||||||
|
|
||||||
|
fn dependencies(&self) -> HashSet<Uuid> {
|
||||||
|
[v_transactions_shielding_balance::MIGRATION_ID]
|
||||||
|
.into_iter()
|
||||||
|
.collect()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn description(&self) -> &'static str {
|
||||||
|
"Fixes a bug in v_transactions that was omitting value from identically-valued notes."
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl RusqliteMigration for Migration {
|
||||||
|
type Error = WalletMigrationError;
|
||||||
|
|
||||||
|
fn up(&self, transaction: &rusqlite::Transaction) -> Result<(), Self::Error> {
|
||||||
|
transaction.execute_batch(
|
||||||
|
"DROP VIEW v_transactions;
|
||||||
|
CREATE VIEW v_transactions AS
|
||||||
|
WITH
|
||||||
|
notes AS (
|
||||||
|
SELECT sapling_received_notes.id_note AS id,
|
||||||
|
sapling_received_notes.account AS account_id,
|
||||||
|
transactions.block AS block,
|
||||||
|
transactions.txid AS txid,
|
||||||
|
2 AS pool,
|
||||||
|
sapling_received_notes.value AS value,
|
||||||
|
CASE
|
||||||
|
WHEN sapling_received_notes.is_change THEN 1
|
||||||
|
ELSE 0
|
||||||
|
END AS is_change,
|
||||||
|
CASE
|
||||||
|
WHEN sapling_received_notes.is_change THEN 0
|
||||||
|
ELSE 1
|
||||||
|
END AS received_count,
|
||||||
|
CASE
|
||||||
|
WHEN (sapling_received_notes.memo IS NULL OR sapling_received_notes.memo = X'F6')
|
||||||
|
THEN 0
|
||||||
|
ELSE 1
|
||||||
|
END AS memo_present
|
||||||
|
FROM sapling_received_notes
|
||||||
|
JOIN transactions
|
||||||
|
ON transactions.id_tx = sapling_received_notes.tx
|
||||||
|
UNION
|
||||||
|
SELECT utxos.id_utxo AS id,
|
||||||
|
utxos.received_by_account AS account_id,
|
||||||
|
utxos.height AS block,
|
||||||
|
utxos.prevout_txid AS txid,
|
||||||
|
0 AS pool,
|
||||||
|
utxos.value_zat AS value,
|
||||||
|
0 AS is_change,
|
||||||
|
1 AS received_count,
|
||||||
|
0 AS memo_present
|
||||||
|
FROM utxos
|
||||||
|
UNION
|
||||||
|
SELECT sapling_received_notes.id_note AS id,
|
||||||
|
sapling_received_notes.account AS account_id,
|
||||||
|
transactions.block AS block,
|
||||||
|
transactions.txid AS txid,
|
||||||
|
2 AS pool,
|
||||||
|
-sapling_received_notes.value AS value,
|
||||||
|
0 AS is_change,
|
||||||
|
0 AS received_count,
|
||||||
|
0 AS memo_present
|
||||||
|
FROM sapling_received_notes
|
||||||
|
JOIN transactions
|
||||||
|
ON transactions.id_tx = sapling_received_notes.spent
|
||||||
|
UNION
|
||||||
|
SELECT utxos.id_utxo AS id,
|
||||||
|
utxos.received_by_account AS account_id,
|
||||||
|
transactions.block AS block,
|
||||||
|
transactions.txid AS txid,
|
||||||
|
0 AS pool,
|
||||||
|
-utxos.value_zat AS value,
|
||||||
|
0 AS is_change,
|
||||||
|
0 AS received_count,
|
||||||
|
0 AS memo_present
|
||||||
|
FROM utxos
|
||||||
|
JOIN transactions
|
||||||
|
ON transactions.id_tx = utxos.spent_in_tx
|
||||||
|
),
|
||||||
|
sent_note_counts AS (
|
||||||
|
SELECT sent_notes.from_account AS account_id,
|
||||||
|
transactions.txid AS txid,
|
||||||
|
COUNT(DISTINCT sent_notes.id_note) as sent_notes,
|
||||||
|
SUM(
|
||||||
|
CASE
|
||||||
|
WHEN (sent_notes.memo IS NULL OR sent_notes.memo = X'F6' OR sapling_received_notes.tx IS NOT NULL)
|
||||||
|
THEN 0
|
||||||
|
ELSE 1
|
||||||
|
END
|
||||||
|
) AS memo_count
|
||||||
|
FROM sent_notes
|
||||||
|
JOIN transactions
|
||||||
|
ON transactions.id_tx = sent_notes.tx
|
||||||
|
LEFT JOIN sapling_received_notes
|
||||||
|
ON (sent_notes.tx, sent_notes.output_pool, sent_notes.output_index) =
|
||||||
|
(sapling_received_notes.tx, 2, sapling_received_notes.output_index)
|
||||||
|
WHERE COALESCE(sapling_received_notes.is_change, 0) = 0
|
||||||
|
GROUP BY account_id, txid
|
||||||
|
),
|
||||||
|
blocks_max_height AS (
|
||||||
|
SELECT MAX(blocks.height) as max_height FROM blocks
|
||||||
|
)
|
||||||
|
SELECT notes.account_id AS account_id,
|
||||||
|
notes.block AS mined_height,
|
||||||
|
notes.txid AS txid,
|
||||||
|
transactions.tx_index AS tx_index,
|
||||||
|
transactions.expiry_height AS expiry_height,
|
||||||
|
transactions.raw AS raw,
|
||||||
|
SUM(notes.value) AS account_balance_delta,
|
||||||
|
transactions.fee AS fee_paid,
|
||||||
|
SUM(notes.is_change) > 0 AS has_change,
|
||||||
|
MAX(COALESCE(sent_note_counts.sent_notes, 0)) AS sent_note_count,
|
||||||
|
SUM(notes.received_count) AS received_note_count,
|
||||||
|
SUM(notes.memo_present) + MAX(COALESCE(sent_note_counts.memo_count, 0)) AS memo_count,
|
||||||
|
blocks.time AS block_time,
|
||||||
|
(
|
||||||
|
blocks.height IS NULL
|
||||||
|
AND transactions.expiry_height BETWEEN 1 AND blocks_max_height.max_height
|
||||||
|
) AS expired_unmined
|
||||||
|
FROM notes
|
||||||
|
LEFT JOIN transactions
|
||||||
|
ON notes.txid = transactions.txid
|
||||||
|
JOIN blocks_max_height
|
||||||
|
LEFT JOIN blocks ON blocks.height = notes.block
|
||||||
|
LEFT JOIN sent_note_counts
|
||||||
|
ON sent_note_counts.account_id = notes.account_id
|
||||||
|
AND sent_note_counts.txid = notes.txid
|
||||||
|
GROUP BY notes.account_id, notes.txid;"
|
||||||
|
)?;
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn down(&self, _transaction: &rusqlite::Transaction) -> Result<(), Self::Error> {
|
||||||
|
panic!("This migration cannot be reverted.");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
Loading…
Reference in New Issue