mirror of
https://github.com/serai-dex/serai.git
synced 2025-01-25 20:16:01 +00:00
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:
parent
c88ebe985e
commit
fadc88d2ad
12 changed files with 173 additions and 21 deletions
3
.github/workflows/tests.yml
vendored
3
.github/workflows/tests.yml
vendored
|
@ -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
12
Cargo.lock
generated
|
@ -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",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
|
|
@ -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",
|
||||||
|
|
|
@ -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" },
|
||||||
|
|
|
@ -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.
|
||||||
|
|
25
processor/scheduler/primitives/Cargo.toml
Normal file
25
processor/scheduler/primitives/Cargo.toml
Normal 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" }
|
15
processor/scheduler/primitives/LICENSE
Normal file
15
processor/scheduler/primitives/LICENSE
Normal 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/>.
|
3
processor/scheduler/primitives/README.md
Normal file
3
processor/scheduler/primitives/README.md
Normal file
|
@ -0,0 +1,3 @@
|
||||||
|
# Scheduler Primitives
|
||||||
|
|
||||||
|
Primitives for schedulers.
|
48
processor/scheduler/primitives/src/lib.rs
Normal file
48
processor/scheduler/primitives/src/lib.rs
Normal 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)
|
||||||
|
}
|
||||||
|
}
|
|
@ -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" }
|
||||||
|
|
|
@ -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());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -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
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue