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

feat: bitcoin + stacks blockchain sync on startup #1087

Draft
wants to merge 17 commits into
base: main
Choose a base branch
from
Draft
Show file tree
Hide file tree
Changes from 13 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
2 changes: 0 additions & 2 deletions signer/migrations/0003__create_tables.sql
Original file line number Diff line number Diff line change
Expand Up @@ -26,8 +26,6 @@ CREATE TABLE sbtc_signer.stacks_blocks (
bitcoin_anchor BYTEA NOT NULL,
created_at TIMESTAMPTZ DEFAULT CURRENT_TIMESTAMP NOT NULL
);
-- Index to serve queries filtering on `parent_hash`. This is commonly used when
cylewitruk marked this conversation as resolved.
Show resolved Hide resolved
-- "walking" the chain in recursive CTE's.

CREATE TABLE sbtc_signer.deposit_requests (
txid BYTEA NOT NULL,
Expand Down
295 changes: 156 additions & 139 deletions signer/src/block_observer.rs
Original file line number Diff line number Diff line change
Expand Up @@ -28,6 +28,7 @@ use crate::context::SbtcLimits;
use crate::context::SignerEvent;
use crate::emily_client::EmilyInteract;
use crate::error::Error;
use crate::stacks::api::GetNakamotoStartHeight as _;
use crate::stacks::api::StacksInteract;
use crate::stacks::api::TenureBlocks;
use crate::storage;
Expand All @@ -39,6 +40,7 @@ use bitcoin::Amount;
use bitcoin::BlockHash;
use bitcoin::ScriptBuf;
use bitcoin::Transaction;
use clarity::types::chainstate::StacksAddress;
use futures::stream::Stream;
use futures::stream::StreamExt;
use sbtc::deposits::CreateDepositRequest;
Expand Down Expand Up @@ -276,20 +278,50 @@ impl<C: Context, B> BlockObserver<C, B> {
/// Process the bitcoin block. Also process all recent stacks blocks.
#[tracing::instrument(skip_all, fields(block_hash = %block.block_hash()))]
async fn process_bitcoin_block(&self, block: bitcoin::Block) -> Result<(), Error> {
let storage = self.context.get_storage_mut();
let bitcoin_client = self.context.get_bitcoin_client();
let stacks_client = self.context.get_stacks_client();

tracing::info!("processing bitcoin block");

let until_bitcoin_height = stacks_client
cylewitruk marked this conversation as resolved.
Show resolved Hide resolved
.get_pox_info()
.await?
.nakamoto_start_height()
.ok_or(Error::MissingNakamotoStartHeight)?;

// Get the current tenure info (incl. tip details) from the Stacks node.
let stacks_client = self.context.get_stacks_client();
let tenure_info = stacks_client.get_tenure_info().await?;

// While we do do this at startup to do the "heavy lifting" of syncing
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
// While we do do this at startup to do the "heavy lifting" of syncing
// While we do this at startup to do the "heavy lifting" of syncing

// potentially from a long chain, we also do this here to ensure that
// we don't miss any Stacks blocks given the high frequency of
// Nakamoto blocks.
//
// Maybe this can be removed in the future if we can be sure that our
// Stacks event observer will always pick up this information before
// we get here, but for now it's a good safety net.
tracing::debug!("fetching unknown ancestral blocks from stacks-core");
let stacks_blocks = crate::stacks::api::fetch_unknown_ancestors(
&stacks_client,
&self.context.get_storage(),
&storage,
tenure_info.tip_block_id,
until_bitcoin_height,
)
.await?;

self.write_stacks_blocks(&stacks_blocks).await?;
self.write_bitcoin_block(&block).await?;
// Write the Stacks blocks and header information. This method will also
// extract relevant sBTC information from the blocks.
write_stacks_blocks(
&storage,
&self.context.config().signer.deployer,
&stacks_blocks,
)
.await?;

// Finally, write the Bitcoin block and any sBTC-related transactions.
write_bitcoin_block(&storage, &bitcoin_client, &block).await?;

tracing::debug!("finished processing bitcoin block");
Ok(())
Expand Down Expand Up @@ -367,125 +399,6 @@ impl<C: Context, B> BlockObserver<C, B> {
Ok(())
}

/// Extract all BTC transactions from the block where one of the UTXOs
/// can be spent by the signers.
///
/// # Note
///
/// When using the postgres storage, we need to make sure that this
/// function is called after the `Self::write_bitcoin_block` function
/// because of the foreign key constraints.
pub async fn extract_sbtc_transactions(
Copy link
Member Author

@cylewitruk cylewitruk Dec 10, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Note: these are just moved to top-level fn's in the file so they can be used elsewhere without needing to instantiate a block observer instance. Maybe they should even be in a separate file.

&self,
block_hash: BlockHash,
txs: &[Transaction],
) -> Result<(), Error> {
let db = self.context.get_storage_mut();
// We store all the scriptPubKeys associated with the signers'
// aggregate public key. Let's get the last years worth of them.
let signer_script_pubkeys: HashSet<ScriptBuf> = db
.get_signers_script_pubkeys()
.await?
.into_iter()
.map(ScriptBuf::from_bytes)
.collect();

let btc_rpc = self.context.get_bitcoin_client();
// Look through all the UTXOs in the given transaction slice and
// keep the transactions where a UTXO is locked with a
// `scriptPubKey` controlled by the signers.
let mut sbtc_txs = Vec::new();
for tx in txs {
tracing::debug!(txid = %tx.compute_txid(), "attempting to extract sbtc transaction");
// If any of the outputs are spent to one of the signers'
// addresses, then we care about it
let outputs_spent_to_signers = tx
.output
.iter()
.any(|tx_out| signer_script_pubkeys.contains(&tx_out.script_pubkey));

if !outputs_spent_to_signers {
continue;
}

// This function is called after we have received a
// notification of a bitcoin block, and we are iterating
// through all of the transactions within that block. This
// means the `get_tx_info` call below should not fail.
let txid = tx.compute_txid();
let tx_info = btc_rpc
.get_tx_info(&txid, &block_hash)
.await?
.ok_or(Error::BitcoinTxMissing(txid, None))?;

// sBTC transactions have as first txin a signers spendable output
let tx_type = if tx_info.is_signer_created(&signer_script_pubkeys) {
model::TransactionType::SbtcTransaction
} else {
model::TransactionType::Donation
};

let txid = tx.compute_txid();
sbtc_txs.push(model::Transaction {
txid: txid.to_byte_array(),
tx: bitcoin::consensus::serialize(&tx),
tx_type,
block_hash: block_hash.to_byte_array(),
});

for prevout in tx_info.to_inputs(&signer_script_pubkeys) {
db.write_tx_prevout(&prevout).await?;
}

for output in tx_info.to_outputs(&signer_script_pubkeys) {
db.write_tx_output(&output).await?;
}
}

// Write these transactions into storage.
db.write_bitcoin_transactions(sbtc_txs).await?;
Ok(())
}

/// Write the given stacks blocks to the database.
///
/// This function also extracts sBTC Stacks transactions from the given
/// blocks and stores them into the database.
async fn write_stacks_blocks(&self, tenures: &[TenureBlocks]) -> Result<(), Error> {
let deployer = &self.context.config().signer.deployer;
let txs = tenures
.iter()
.flat_map(|tenure| {
storage::postgres::extract_relevant_transactions(tenure.blocks(), deployer)
})
.collect::<Vec<_>>();

let headers = tenures
.iter()
.flat_map(TenureBlocks::as_stacks_blocks)
.collect::<Vec<_>>();

let storage = self.context.get_storage_mut();
storage.write_stacks_block_headers(headers).await?;
storage.write_stacks_transactions(txs).await?;
Ok(())
}

/// Write the bitcoin block to the database. We also write any
/// transactions that are spend to any of the signers `scriptPubKey`s
async fn write_bitcoin_block(&self, block: &bitcoin::Block) -> Result<(), Error> {
let db_block = model::BitcoinBlock::from(block);

self.context
.get_storage_mut()
.write_bitcoin_block(&db_block)
.await?;
self.extract_sbtc_transactions(block.block_hash(), &block.txdata)
.await?;

Ok(())
}

/// Update the sBTC peg limits from Emily
async fn update_sbtc_limits(&self) -> Result<(), Error> {
let limits = self.context.get_emily_client().get_limits().await?;
Expand Down Expand Up @@ -525,6 +438,125 @@ impl<C: Context, B> BlockObserver<C, B> {
}
}

/// Takes a list of Nakamoto [`TenureBlocks`] and writes the Stacks block
/// headers and transactions to the database.
pub async fn write_stacks_blocks(
storage: &impl DbWrite,
deployer: &StacksAddress,
tenures: &[TenureBlocks],
) -> Result<(), Error> {
let txs = tenures
.iter()
.flat_map(|tenure| {
storage::postgres::extract_relevant_transactions(tenure.blocks(), deployer)
})
.collect::<Vec<_>>();

let headers = tenures
.iter()
.flat_map(TenureBlocks::as_stacks_blocks)
.collect::<Vec<_>>();

storage.write_stacks_block_headers(headers).await?;
storage.write_stacks_transactions(txs).await?;
Ok(())
}

/// Write the bitcoin block to the database. We also write any
/// transactions that are spend to any of the signers `scriptPubKey`s
async fn write_bitcoin_block(
storage: &(impl DbRead + DbWrite),
bitcoin_client: &impl BitcoinInteract,
block: &bitcoin::Block,
) -> Result<(), Error> {
let db_block = model::BitcoinBlock::from(block);

storage.write_bitcoin_block(&db_block).await?;

extract_sbtc_transactions(storage, bitcoin_client, block.block_hash(), &block.txdata).await?;

Ok(())
}

/// Extract all BTC transactions from the block where one of the UTXOs
/// can be spent by the signers.
///
/// # Note
///
/// When using the postgres storage, we need to make sure that this
/// function is called after the `Self::write_bitcoin_block` function
/// because of the foreign key constraints.
pub async fn extract_sbtc_transactions(
storage: &(impl DbWrite + DbRead),
bitcoin_client: &impl BitcoinInteract,
block_hash: BlockHash,
txs: &[Transaction],
) -> Result<(), Error> {
// We store all the scriptPubKeys associated with the signers'
// aggregate public key. Let's get the last years worth of them.
let signer_script_pubkeys: HashSet<ScriptBuf> = storage
.get_signers_script_pubkeys()
.await?
.into_iter()
.map(ScriptBuf::from_bytes)
.collect();

// Look through all the UTXOs in the given transaction slice and
// keep the transactions where a UTXO is locked with a
// `scriptPubKey` controlled by the signers.
let mut sbtc_txs = Vec::new();
for tx in txs {
tracing::debug!(txid = %tx.compute_txid(), "attempting to extract sbtc transaction");
// If any of the outputs are spent to one of the signers'
// addresses, then we care about it
let outputs_spent_to_signers = tx
.output
.iter()
.any(|tx_out| signer_script_pubkeys.contains(&tx_out.script_pubkey));

if !outputs_spent_to_signers {
continue;
}

// This function is called after we have received a
// notification of a bitcoin block, and we are iterating
// through all of the transactions within that block. This
// means the `get_tx_info` call below should not fail.
let txid = tx.compute_txid();
let tx_info = bitcoin_client
.get_tx_info(&txid, &block_hash)
.await?
.ok_or(Error::BitcoinTxMissing(txid, None))?;

// sBTC transactions have as first txin a signers spendable output
let tx_type = if tx_info.is_signer_created(&signer_script_pubkeys) {
model::TransactionType::SbtcTransaction
} else {
model::TransactionType::Donation
};

let txid = tx.compute_txid();
sbtc_txs.push(model::Transaction {
txid: txid.to_byte_array(),
tx: bitcoin::consensus::serialize(&tx),
tx_type,
block_hash: block_hash.to_byte_array(),
});

for prevout in tx_info.to_inputs(&signer_script_pubkeys) {
storage.write_tx_prevout(&prevout).await?;
}

for output in tx_info.to_outputs(&signer_script_pubkeys) {
storage.write_tx_output(&output).await?;
}
}

// Write these transactions into storage.
storage.write_bitcoin_transactions(sbtc_txs).await?;
Ok(())
}

#[cfg(test)]
mod tests {
use bitcoin::Amount;
Expand Down Expand Up @@ -875,24 +907,10 @@ mod tests {
test_harness.add_deposit(txid0, response0);
test_harness.add_deposit(txid1, response1);

let ctx = TestContext::builder()
.with_storage(storage.clone())
.with_stacks_client(test_harness.clone())
.with_emily_client(test_harness.clone())
.with_bitcoin_client(test_harness.clone())
.build();

let block_observer = BlockObserver {
context: ctx,
bitcoin_blocks: (),
horizon: 1,
};

// First we try extracting the transactions from a block that does
// not contain any transactions spent to the signers
let txs = [tx_setup1.tx.clone()];
block_observer
.extract_sbtc_transactions(block_hash, &txs)
extract_sbtc_transactions(&storage, &test_harness, block_hash, &txs)
.await
.unwrap();

Expand All @@ -911,8 +929,7 @@ mod tests {
// Now we try again, but we include the transaction that spends to
// the signer. This one should turn out differently.
let txs = [tx_setup0.tx.clone(), tx_setup1.tx.clone()];
block_observer
.extract_sbtc_transactions(block_hash, &txs)
extract_sbtc_transactions(&storage, &test_harness, block_hash, &txs)
.await
.unwrap();

Expand Down
Loading
Loading