Add scheduler-primitives

The main benefit is whatever scheduler is in use, we now have a single API to
receive TXs to sign (which is of value to the TX signer crate we'll inevitably
build).
This commit is contained in:
Luke Parker 2024-09-02 16:09:52 -04:00
parent c88ebe985e
commit fadc88d2ad
12 changed files with 173 additions and 21 deletions

View file

@ -42,9 +42,10 @@ jobs:
-p serai-processor-key-gen \ -p serai-processor-key-gen \
-p serai-processor-frost-attempt-manager \ -p serai-processor-frost-attempt-manager \
-p serai-processor-primitives \ -p serai-processor-primitives \
-p serai-processor-scanner \
-p serai-processor-scheduler-primitives \
-p serai-processor-utxo-scheduler-primitives \ -p serai-processor-utxo-scheduler-primitives \
-p serai-processor-transaction-chaining-scheduler \ -p serai-processor-transaction-chaining-scheduler \
-p serai-processor-scanner \
-p serai-processor \ -p serai-processor \
-p tendermint-machine \ -p tendermint-machine \
-p tributary-chain \ -p tributary-chain \

12
Cargo.lock generated
View file

@ -8679,6 +8679,16 @@ dependencies = [
"tokio", "tokio",
] ]
[[package]]
name = "serai-processor-scheduler-primitives"
version = "0.1.0"
dependencies = [
"borsh",
"group",
"parity-scale-codec",
"serai-db",
]
[[package]] [[package]]
name = "serai-processor-tests" name = "serai-processor-tests"
version = "0.1.0" version = "0.1.0"
@ -8715,11 +8725,11 @@ dependencies = [
"borsh", "borsh",
"group", "group",
"parity-scale-codec", "parity-scale-codec",
"serai-coins-primitives",
"serai-db", "serai-db",
"serai-primitives", "serai-primitives",
"serai-processor-primitives", "serai-processor-primitives",
"serai-processor-scanner", "serai-processor-scanner",
"serai-processor-scheduler-primitives",
"serai-processor-utxo-scheduler-primitives", "serai-processor-utxo-scheduler-primitives",
] ]

View file

@ -74,9 +74,10 @@ members = [
"processor/frost-attempt-manager", "processor/frost-attempt-manager",
"processor/primitives", "processor/primitives",
"processor/scanner",
"processor/scheduler/primitives",
"processor/scheduler/utxo/primitives", "processor/scheduler/utxo/primitives",
"processor/scheduler/utxo/transaction-chaining", "processor/scheduler/utxo/transaction-chaining",
"processor/scanner",
"processor", "processor",
"coordinator/tributary/tendermint", "coordinator/tributary/tendermint",

View file

@ -49,9 +49,10 @@ exceptions = [
{ allow = ["AGPL-3.0"], name = "serai-processor-key-gen" }, { allow = ["AGPL-3.0"], name = "serai-processor-key-gen" },
{ allow = ["AGPL-3.0"], name = "serai-processor-frost-attempt-manager" }, { allow = ["AGPL-3.0"], name = "serai-processor-frost-attempt-manager" },
{ allow = ["AGPL-3.0"], name = "serai-processor-utxo-primitives" },
{ allow = ["AGPL-3.0"], name = "serai-processor-transaction-chaining-scheduler" },
{ allow = ["AGPL-3.0"], name = "serai-processor-scanner" }, { allow = ["AGPL-3.0"], name = "serai-processor-scanner" },
{ allow = ["AGPL-3.0"], name = "serai-processor-scheduler-primitives" },
{ allow = ["AGPL-3.0"], name = "serai-processor-utxo-scheduler-primitives" },
{ allow = ["AGPL-3.0"], name = "serai-processor-transaction-chaining-scheduler" },
{ allow = ["AGPL-3.0"], name = "serai-processor" }, { allow = ["AGPL-3.0"], name = "serai-processor" },
{ allow = ["AGPL-3.0"], name = "tributary-chain" }, { allow = ["AGPL-3.0"], name = "tributary-chain" },

View file

@ -241,8 +241,12 @@ pub trait Scheduler<S: ScannerFeed>: 'static + Send {
/// ///
/// When a key is activated, the existing multisig should retain its outputs and utility for a /// When a key is activated, the existing multisig should retain its outputs and utility for a
/// certain time period. With `flush_key`, all outputs should be directed towards fulfilling some /// certain time period. With `flush_key`, all outputs should be directed towards fulfilling some
/// obligation or the `new_key`. Every output MUST be connected to an Eventuality. If a key no /// obligation or the `new_key`. Every output held by the retiring key MUST be connected to an
/// longer has active Eventualities, it MUST be able to be retired. /// Eventuality. If a key no longer has active Eventualities, it MUST be able to be retired
/// without losing any coins.
///
/// If the retiring key has any unfulfilled payments associated with it, those MUST be made
/// the responsibility of the new key.
fn flush_key(&mut self, txn: &mut impl DbTxn, retiring_key: KeyFor<S>, new_key: KeyFor<S>); fn flush_key(&mut self, txn: &mut impl DbTxn, retiring_key: KeyFor<S>, new_key: KeyFor<S>);
/// Retire a key as it'll no longer be used. /// Retire a key as it'll no longer be used.

View file

@ -0,0 +1,25 @@
[package]
name = "serai-processor-scheduler-primitives"
version = "0.1.0"
description = "Primitives for schedulers for the Serai processor"
license = "AGPL-3.0-only"
repository = "https://github.com/serai-dex/serai/tree/develop/processor/scheduler/primitives"
authors = ["Luke Parker <lukeparker5132@gmail.com>"]
keywords = []
edition = "2021"
publish = false
[package.metadata.docs.rs]
all-features = true
rustdoc-args = ["--cfg", "docsrs"]
[lints]
workspace = true
[dependencies]
group = { version = "0.13", default-features = false }
scale = { package = "parity-scale-codec", version = "3", default-features = false, features = ["std"] }
borsh = { version = "1", default-features = false, features = ["std", "derive", "de_strict_order"] }
serai-db = { path = "../../../common/db" }

View file

@ -0,0 +1,15 @@
AGPL-3.0-only license
Copyright (c) 2024 Luke Parker
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU Affero General Public License Version 3 as
published by the Free Software Foundation.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU Affero General Public License for more details.
You should have received a copy of the GNU Affero General Public License
along with this program. If not, see <http://www.gnu.org/licenses/>.

View file

@ -0,0 +1,3 @@
# Scheduler Primitives
Primitives for schedulers.

View file

@ -0,0 +1,48 @@
#![cfg_attr(docsrs, feature(doc_auto_cfg))]
#![doc = include_str!("../README.md")]
#![deny(missing_docs)]
use core::marker::PhantomData;
use std::io;
use group::GroupEncoding;
use serai_db::DbTxn;
/// A signable transaction.
pub trait SignableTransaction: 'static + Sized + Send + Sync {
/// Read a `SignableTransaction`.
fn read(reader: &mut impl io::Read) -> io::Result<Self>;
/// Write a `SignableTransaction`.
fn write(&self, writer: &mut impl io::Write) -> io::Result<()>;
}
mod db {
use serai_db::{Get, DbTxn, create_db, db_channel};
db_channel! {
SchedulerPrimitives {
TransactionsToSign: (key: &[u8]) -> Vec<u8>,
}
}
}
/// The transactions to sign, as scheduled by a Scheduler.
pub struct TransactionsToSign<T>(PhantomData<T>);
impl<T: SignableTransaction> TransactionsToSign<T> {
/// Send a transaction to sign.
pub fn send(txn: &mut impl DbTxn, key: &impl GroupEncoding, tx: &T) {
let mut buf = Vec::with_capacity(128);
tx.write(&mut buf).unwrap();
db::TransactionsToSign::send(txn, key.to_bytes().as_ref(), &buf);
}
/// Try to receive a transaction to sign.
pub fn try_recv(txn: &mut impl DbTxn, key: &impl GroupEncoding) -> Option<T> {
let tx = db::TransactionsToSign::try_recv(txn, key.to_bytes().as_ref())?;
let mut tx = tx.as_slice();
let res = T::read(&mut tx).unwrap();
assert!(tx.is_empty());
Some(res)
}
}

View file

@ -26,10 +26,10 @@ scale = { package = "parity-scale-codec", version = "3", default-features = fals
borsh = { version = "1", default-features = false, features = ["std", "derive", "de_strict_order"] } borsh = { version = "1", default-features = false, features = ["std", "derive", "de_strict_order"] }
serai-primitives = { path = "../../../../substrate/primitives", default-features = false, features = ["std"] } serai-primitives = { path = "../../../../substrate/primitives", default-features = false, features = ["std"] }
serai-coins-primitives = { path = "../../../../substrate/coins/primitives", default-features = false, features = ["std"] }
serai-db = { path = "../../../../common/db" } serai-db = { path = "../../../../common/db" }
primitives = { package = "serai-processor-primitives", path = "../../../primitives" } primitives = { package = "serai-processor-primitives", path = "../../../primitives" }
scheduler-primitives = { package = "serai-processor-utxo-scheduler-primitives", path = "../primitives" } scheduler-primitives = { package = "serai-processor-scheduler-primitives", path = "../../primitives" }
utxo-scheduler-primitives = { package = "serai-processor-utxo-scheduler-primitives", path = "../primitives" }
scanner = { package = "serai-processor-scanner", path = "../../../scanner" } scanner = { package = "serai-processor-scanner", path = "../../../scanner" }

View file

@ -2,7 +2,7 @@ use core::marker::PhantomData;
use group::GroupEncoding; use group::GroupEncoding;
use serai_primitives::Coin; use serai_primitives::{Coin, Amount};
use serai_db::{Get, DbTxn, create_db}; use serai_db::{Get, DbTxn, create_db};
@ -11,12 +11,23 @@ use scanner::{ScannerFeed, KeyFor, OutputFor};
create_db! { create_db! {
TransactionChainingScheduler { TransactionChainingScheduler {
OperatingCosts: (coin: Coin) -> Amount,
SerializedOutputs: (key: &[u8], coin: Coin) -> Vec<u8>, SerializedOutputs: (key: &[u8], coin: Coin) -> Vec<u8>,
// We should be immediately able to schedule the fulfillment of payments, yet this may not be
// possible if we're in the middle of a multisig rotation (as our output set will be split)
SerializedQueuedPayments: (key: &[u8]) > Vec<u8>,
} }
} }
pub(crate) struct Db<S: ScannerFeed>(PhantomData<S>); pub(crate) struct Db<S: ScannerFeed>(PhantomData<S>);
impl<S: ScannerFeed> Db<S> { impl<S: ScannerFeed> Db<S> {
pub(crate) fn operating_costs(getter: &impl Get, coin: Coin) -> Amount {
OperatingCosts::get(getter, coin).unwrap_or(Amount(0))
}
pub(crate) fn set_operating_costs(txn: &mut impl DbTxn, coin: Coin, amount: Amount) {
OperatingCosts::set(txn, coin, &amount)
}
pub(crate) fn outputs( pub(crate) fn outputs(
getter: &impl Get, getter: &impl Get,
key: KeyFor<S>, key: KeyFor<S>,
@ -46,4 +57,17 @@ impl<S: ScannerFeed> Db<S> {
pub(crate) fn del_outputs(txn: &mut impl DbTxn, key: KeyFor<S>, coin: Coin) { pub(crate) fn del_outputs(txn: &mut impl DbTxn, key: KeyFor<S>, coin: Coin) {
SerializedOutputs::del(txn, key.to_bytes().as_ref(), coin); SerializedOutputs::del(txn, key.to_bytes().as_ref(), coin);
} }
pub(crate) fn queued_payments(
getter: &impl Get,
key: KeyFor<S>,
) -> Option<Vec<Payment<S>>> {
todo!("TODO")
}
pub(crate) fn set_queued_payments(txn: &mut impl DbTxn, key: KeyFor<S>, queued: Vec<Payment<S>>) {
todo!("TODO")
}
pub(crate) fn del_outputs(txn: &mut impl DbTxn, key: KeyFor<S>) {
SerializedQueuedPayments::del(txn, key.to_bytes().as_ref());
}
} }

View file

@ -5,6 +5,8 @@
use core::marker::PhantomData; use core::marker::PhantomData;
use std::collections::HashMap; use std::collections::HashMap;
use group::GroupEncoding;
use serai_primitives::Coin; use serai_primitives::Coin;
use serai_db::DbTxn; use serai_db::DbTxn;
@ -15,6 +17,7 @@ use scanner::{
Scheduler as SchedulerTrait, Scheduler as SchedulerTrait,
}; };
use scheduler_primitives::*; use scheduler_primitives::*;
use utxo_scheduler_primitives::*;
mod db; mod db;
use db::Db; use db::Db;
@ -25,7 +28,7 @@ pub struct PlannedTransaction<S: ScannerFeed, T> {
signable: T, signable: T,
/// The outputs we'll receive from this. /// The outputs we'll receive from this.
effected_received_outputs: OutputFor<S>, effected_received_outputs: OutputFor<S>,
/// The Evtnuality to watch for. /// The Eventuality to watch for.
eventuality: EventualityFor<S>, eventuality: EventualityFor<S>,
} }
@ -60,13 +63,13 @@ impl<S: ScannerFeed, T, P: TransactionPlanner<S, PlannedTransaction = PlannedTra
impl< impl<
S: ScannerFeed, S: ScannerFeed,
T: 'static + Send + Sync, T: 'static + Send + Sync + SignableTransaction,
P: TransactionPlanner<S, PlannedTransaction = PlannedTransaction<S, T>>, P: TransactionPlanner<S, PlannedTransaction = PlannedTransaction<S, T>>,
> SchedulerTrait<S> for Scheduler<S, T, P> > SchedulerTrait<S> for Scheduler<S, T, P>
{ {
fn activate_key(&mut self, txn: &mut impl DbTxn, key: KeyFor<S>) { fn activate_key(&mut self, txn: &mut impl DbTxn, key: KeyFor<S>) {
for coin in S::NETWORK.coins() { for coin in S::NETWORK.coins() {
Db::<S>::set_outputs(txn, key, *coin, &vec![]); Db::<S>::set_outputs(txn, key, *coin, &[]);
} }
} }
@ -98,22 +101,27 @@ impl<
{ {
let mut planned_txs = vec![]; let mut planned_txs = vec![];
for forward in update.forwards() { for forward in update.forwards() {
let forward_to_key = active_keys.last().unwrap(); let key = forward.key();
assert_eq!(forward_to_key.1, LifetimeStage::Active);
assert_eq!(active_keys.len(), 2);
assert_eq!(active_keys[0].1, LifetimeStage::Forwarding);
assert_eq!(active_keys[1].1, LifetimeStage::Active);
let forward_to_key = active_keys[1].0;
let Some(plan) = P::plan_transaction_with_fee_amortization( let Some(plan) = P::plan_transaction_with_fee_amortization(
// This uses 0 for the operating costs as we don't incur any here // This uses 0 for the operating costs as we don't incur any here
&mut 0, &mut 0,
fee_rates[&forward.balance().coin], fee_rates[&forward.balance().coin],
vec![forward.clone()], vec![forward.clone()],
vec![Payment::new(P::forwarding_address(forward_to_key.0), forward.balance(), None)], vec![Payment::new(P::forwarding_address(forward_to_key), forward.balance(), None)],
None, None,
) else { ) else {
continue; continue;
}; };
planned_txs.push(plan); planned_txs.push((key, plan));
} }
for to_return in update.returns() { for to_return in update.returns() {
let key = to_return.output().key();
let out_instruction = let out_instruction =
Payment::new(to_return.address().clone(), to_return.output().balance(), None); Payment::new(to_return.address().clone(), to_return.output().balance(), None);
let Some(plan) = P::plan_transaction_with_fee_amortization( let Some(plan) = P::plan_transaction_with_fee_amortization(
@ -126,12 +134,24 @@ impl<
) else { ) else {
continue; continue;
}; };
planned_txs.push(plan); planned_txs.push((key, plan));
} }
// TODO: Send the transactions off for signing let mut eventualities = HashMap::new();
// TODO: Return the eventualities for (key, planned_tx) in planned_txs {
todo!("TODO") // Send the transactions off for signing
TransactionsToSign::<T>::send(txn, &key, &planned_tx.signable);
// Insert the eventualities into the result
eventualities
.entry(key.to_bytes().as_ref().to_vec())
.or_insert(Vec::with_capacity(1))
.push(planned_tx.eventuality);
}
// TODO: Fulfill any payments we prior couldn't
eventualities
} }
} }