#![allow(clippy::items_after_statements)]
use std::rc::Rc;
use std::string::{String, ToString};
use std::sync::{Arc, RwLock};
use std::vec::Vec;
use miden_client::Word;
use miden_client::note::ToInputNoteCommitments;
use miden_client::store::{AccountSmtForest, StoreError, TransactionFilter};
use miden_client::transaction::{
TransactionDetails,
TransactionId,
TransactionRecord,
TransactionScript,
TransactionStatus,
TransactionStoreUpdate,
};
use miden_client::utils::{Deserializable as _, Serializable as _};
use rusqlite::types::Value;
use rusqlite::{Connection, Transaction, params};
use super::SqliteStore;
use super::note::apply_note_updates_tx;
use super::sync::add_note_tag_tx;
use crate::sql_error::SqlResultExt;
use crate::{insert_sql, subst};
pub(crate) const UPSERT_TRANSACTION_QUERY: &str = insert_sql!(
transactions {
id,
details,
script_root,
block_num,
status_variant,
status
} | REPLACE
);
pub(crate) const INSERT_TRANSACTION_SCRIPT_QUERY: &str =
insert_sql!(transaction_scripts { script_root, script } | IGNORE);
struct SerializedTransactionData {
id: String,
script_root: Option<Vec<u8>>,
tx_script: Option<Vec<u8>>,
details: Vec<u8>,
block_num: u32,
status_variant: u8,
status: Vec<u8>,
}
struct SerializedTransactionParts {
id: String,
tx_script: Option<Vec<u8>>,
details: Vec<u8>,
status: Vec<u8>,
}
impl SqliteStore {
pub fn get_transactions(
conn: &mut Connection,
filter: &TransactionFilter,
) -> Result<Vec<TransactionRecord>, StoreError> {
match filter {
TransactionFilter::Ids(ids) => {
let id_strings =
ids.iter().map(|id| Value::Text(id.to_string())).collect::<Vec<_>>();
conn.prepare(filter.to_query().as_ref())
.into_store_error()?
.query_map(params![Rc::new(id_strings)], parse_transaction_columns)
.into_store_error()?
.map(|result| Ok(result.into_store_error()?).and_then(parse_transaction))
.collect::<Result<Vec<TransactionRecord>, _>>()
},
_ => {
conn.prepare(filter.to_query().as_ref())
.into_store_error()?
.query_map([], parse_transaction_columns)
.into_store_error()?
.map(|result| Ok(result.into_store_error()?).and_then(parse_transaction))
.collect::<Result<Vec<TransactionRecord>, _>>()
},
}
}
pub fn apply_transaction(
conn: &mut Connection,
smt_forest: &Arc<RwLock<AccountSmtForest>>,
tx_update: &TransactionStoreUpdate,
) -> Result<(), StoreError> {
let executed_transaction = tx_update.executed_transaction();
let updated_fungible_assets = Self::get_account_fungible_assets_for_delta(
conn,
executed_transaction.account_id(),
executed_transaction.account_delta(),
)?;
let old_map_roots = Self::get_storage_map_roots_for_delta(
conn,
executed_transaction.account_id(),
executed_transaction.account_delta(),
)?;
let tx = conn.transaction().into_store_error()?;
let nullifiers: Vec<Word> = executed_transaction
.input_notes()
.iter()
.map(|x| x.nullifier().as_word())
.collect();
let output_notes = executed_transaction.output_notes();
let details = TransactionDetails {
account_id: executed_transaction.account_id(),
init_account_state: executed_transaction.initial_account().initial_commitment(),
final_account_state: executed_transaction.final_account().to_commitment(),
input_note_nullifiers: nullifiers,
output_notes: output_notes.clone(),
block_num: executed_transaction.block_header().block_num(),
submission_height: tx_update.submission_height(),
expiration_block_num: executed_transaction.expiration_block_num(),
creation_timestamp: super::current_timestamp_u64(),
};
let transaction_record = TransactionRecord::new(
executed_transaction.id(),
details,
executed_transaction.tx_args().tx_script().cloned(),
TransactionStatus::Pending,
);
upsert_transaction_record(&tx, &transaction_record)?;
let mut smt_forest = smt_forest.write().expect("smt_forest write lock not poisoned");
Self::apply_account_delta(
&tx,
&mut smt_forest,
&executed_transaction.initial_account().into(),
executed_transaction.final_account(),
updated_fungible_assets,
&old_map_roots,
executed_transaction.account_delta(),
)?;
drop(smt_forest);
apply_note_updates_tx(&tx, tx_update.note_updates())?;
for tag_record in tx_update.new_tags() {
add_note_tag_tx(&tx, tag_record)?;
}
tx.commit().into_store_error()?;
Ok(())
}
}
pub(crate) fn upsert_transaction_record(
tx: &Transaction<'_>,
transaction: &TransactionRecord,
) -> Result<(), StoreError> {
let SerializedTransactionData {
id,
script_root,
tx_script,
details,
block_num,
status_variant,
status,
} = serialize_transaction_data(transaction);
if let Some(root) = script_root.clone() {
tx.execute(INSERT_TRANSACTION_SCRIPT_QUERY, params![root, tx_script])
.into_store_error()?;
}
tx.execute(
UPSERT_TRANSACTION_QUERY,
params![id, details, script_root, block_num, status_variant, status],
)
.into_store_error()?;
Ok(())
}
fn serialize_transaction_data(transaction_record: &TransactionRecord) -> SerializedTransactionData {
let transaction_id: String = transaction_record.id.to_hex();
let script_root = transaction_record.script.as_ref().map(|script| script.root().to_bytes());
let tx_script = transaction_record.script.as_ref().map(TransactionScript::to_bytes);
SerializedTransactionData {
id: transaction_id,
script_root,
tx_script,
details: transaction_record.details.to_bytes(),
block_num: transaction_record.details.block_num.as_u32(),
status_variant: transaction_record.status.variant() as u8,
status: transaction_record.status.to_bytes(),
}
}
fn parse_transaction_columns(
row: &rusqlite::Row<'_>,
) -> Result<SerializedTransactionParts, rusqlite::Error> {
let id: String = row.get(0)?;
let tx_script: Option<Vec<u8>> = row.get(1)?;
let details: Vec<u8> = row.get(2)?;
let status: Vec<u8> = row.get(3)?;
Ok(SerializedTransactionParts { id, tx_script, details, status })
}
fn parse_transaction(
serialized_transaction: SerializedTransactionParts,
) -> Result<TransactionRecord, StoreError> {
let SerializedTransactionParts { id, tx_script, details, status } = serialized_transaction;
let id: Word = id.as_str().try_into()?;
let script: Option<TransactionScript> = tx_script
.map(|script| TransactionScript::read_from_bytes(&script))
.transpose()?;
Ok(TransactionRecord {
id: TransactionId::from_raw(id),
details: TransactionDetails::read_from_bytes(&details)?,
script,
status: TransactionStatus::read_from_bytes(&status)?,
})
}