mirror of
https://github.com/serai-dex/serai.git
synced 2025-12-10 21:19:24 +00:00
87 lines
2.7 KiB
Rust
87 lines
2.7 KiB
Rust
use core::future::Future;
|
|
use std::sync::Arc;
|
|
|
|
use serai_client_serai::{
|
|
abi::primitives::{network_id::ExternalNetworkId, instructions::SignedBatch},
|
|
RpcError, Serai,
|
|
};
|
|
|
|
use serai_db::{Get, DbTxn, Db, create_db};
|
|
use serai_task::ContinuallyRan;
|
|
|
|
use crate::SignedBatches;
|
|
|
|
create_db!(
|
|
CoordinatorSubstrate {
|
|
LastPublishedBatch: (network: ExternalNetworkId) -> u32,
|
|
BatchesToPublish: (network: ExternalNetworkId, batch: u32) -> SignedBatch,
|
|
}
|
|
);
|
|
|
|
/// Publish `SignedBatch`s from `SignedBatches` onto Serai.
|
|
pub struct PublishBatchTask<D: Db> {
|
|
db: D,
|
|
serai: Arc<Serai>,
|
|
network: ExternalNetworkId,
|
|
}
|
|
|
|
impl<D: Db> PublishBatchTask<D> {
|
|
/// Create a task to publish `SignedBatch`s onto Serai.
|
|
pub fn new(db: D, serai: Arc<Serai>, network: ExternalNetworkId) -> Self {
|
|
Self { db, serai, network }
|
|
}
|
|
}
|
|
|
|
impl<D: Db> ContinuallyRan for PublishBatchTask<D> {
|
|
type Error = RpcError;
|
|
|
|
fn run_iteration(&mut self) -> impl Send + Future<Output = Result<bool, Self::Error>> {
|
|
async move {
|
|
// Read from SignedBatches, which is sequential, into our own mapping
|
|
loop {
|
|
let mut txn = self.db.txn();
|
|
let Some(batch) = SignedBatches::try_recv(&mut txn, self.network) else {
|
|
break;
|
|
};
|
|
|
|
// If this is a Batch not yet published, save it into our unordered mapping
|
|
if LastPublishedBatch::get(&txn, self.network) < Some(batch.batch.id()) {
|
|
BatchesToPublish::set(&mut txn, self.network, batch.batch.id(), &batch);
|
|
}
|
|
|
|
txn.commit();
|
|
}
|
|
|
|
// Synchronize our last published batch with the Serai network's
|
|
let next_to_publish = {
|
|
let mut txn = self.db.txn();
|
|
let last_batch = crate::last_indexed_batch_id(&txn, self.network);
|
|
let mut our_last_batch = LastPublishedBatch::get(&txn, self.network);
|
|
while our_last_batch < last_batch {
|
|
let next_batch = our_last_batch.map(|batch| batch + 1).unwrap_or(0);
|
|
// Clean up the Batch to publish since it's already been published
|
|
BatchesToPublish::take(&mut txn, self.network, next_batch);
|
|
our_last_batch = Some(next_batch);
|
|
}
|
|
if let Some(last_batch) = our_last_batch {
|
|
LastPublishedBatch::set(&mut txn, self.network, &last_batch);
|
|
}
|
|
txn.commit();
|
|
last_batch.map(|batch| batch + 1).unwrap_or(0)
|
|
};
|
|
|
|
let made_progress =
|
|
if let Some(batch) = BatchesToPublish::get(&self.db, self.network, next_to_publish) {
|
|
self
|
|
.serai
|
|
.publish_transaction(&serai_client_serai::InInstructions::execute_batch(batch))
|
|
.await?;
|
|
true
|
|
} else {
|
|
false
|
|
};
|
|
Ok(made_progress)
|
|
}
|
|
}
|
|
}
|