sui_indexer_alt/
bootstrap.rsuse std::time::Duration;
use anyhow::{bail, Context, Result};
use diesel::{OptionalExtension, QueryDsl, SelectableHelper};
use diesel_async::RunQueryDsl;
use sui_indexer_alt_framework::types::{
full_checkpoint_content::CheckpointData,
sui_system_state::{get_sui_system_state, SuiSystemStateTrait},
transaction::{TransactionDataAPI, TransactionKind},
};
use sui_indexer_alt_schema::{
checkpoints::StoredGenesis,
epochs::StoredEpochStart,
schema::{kv_epoch_starts, kv_genesis},
};
use tokio_util::sync::CancellationToken;
use tracing::info;
use crate::Indexer;
pub async fn bootstrap(
indexer: &Indexer,
retry_interval: Duration,
cancel: CancellationToken,
) -> Result<StoredGenesis> {
info!("Bootstrapping indexer with genesis information");
let Ok(mut conn) = indexer.db().connect().await else {
bail!("Bootstrap failed to get connection for DB");
};
if let Some(genesis) = kv_genesis::table
.select(StoredGenesis::as_select())
.first(&mut conn)
.await
.optional()?
{
info!(
chain = genesis.chain()?.as_str(),
protocol = ?genesis.initial_protocol_version(),
"Indexer already bootstrapped",
);
return Ok(genesis);
}
let genesis_checkpoint = indexer
.ingestion_client()
.wait_for(0, retry_interval, &cancel)
.await
.context("Failed to fetch genesis checkpoint")?;
let CheckpointData {
checkpoint_summary,
transactions,
..
} = genesis_checkpoint.as_ref();
let Some(genesis_transaction) = transactions.iter().find(|tx| {
matches!(
tx.transaction.intent_message().value.kind(),
TransactionKind::Genesis(_)
)
}) else {
bail!("Could not find Genesis transaction");
};
let system_state = get_sui_system_state(&genesis_transaction.output_objects.as_slice())
.context("Failed to get Genesis SystemState")?;
let genesis = StoredGenesis {
genesis_digest: checkpoint_summary.digest().inner().to_vec(),
initial_protocol_version: system_state.protocol_version() as i64,
};
let epoch_start = StoredEpochStart {
epoch: 0,
protocol_version: system_state.protocol_version() as i64,
cp_lo: 0,
start_timestamp_ms: system_state.epoch_start_timestamp_ms() as i64,
reference_gas_price: system_state.reference_gas_price() as i64,
system_state: bcs::to_bytes(&system_state).context("Failed to serialize SystemState")?,
};
info!(
chain = genesis.chain()?.as_str(),
protocol = ?genesis.initial_protocol_version(),
"Bootstrapped indexer",
);
diesel::insert_into(kv_genesis::table)
.values(&genesis)
.on_conflict_do_nothing()
.execute(&mut conn)
.await
.context("Failed to write genesis record")?;
diesel::insert_into(kv_epoch_starts::table)
.values(&epoch_start)
.on_conflict_do_nothing()
.execute(&mut conn)
.await
.context("Failed to write genesis epoch start record")?;
Ok(genesis)
}