Skip to content

Commit

Permalink
FullNetworkTx implementation
Browse files Browse the repository at this point in the history
FullNetworkTx for MVP-0 Auctions

  * we are currently only concerned with `BidTx`
  * there is some very basic coverage of essential functionality
  • Loading branch information
tbro committed Jun 20, 2024
1 parent 1c82997 commit 0d8eeec
Show file tree
Hide file tree
Showing 3 changed files with 291 additions and 58 deletions.
269 changes: 219 additions & 50 deletions sequencer/src/auction.rs
Original file line number Diff line number Diff line change
@@ -1,39 +1,41 @@
use crate::{
eth_signature_key::EthKeyPair,
state::{FeeAccount, FeeAmount},
NamespaceId, Transaction, ValidatedState,
};
use hotshot_types::data::ViewNumber;
use std::num::NonZeroU64;
use ark_serialize::CanonicalSerialize;
use committable::{Commitment, Committable};
use derivative::Derivative;
use ethers::types::Signature;
use hotshot::types::SignatureKey;
use hotshot_types::{data::ViewNumber, traits::signature_key::BuilderSignatureKey};
use rand::RngCore;
use serde::{Deserialize, Serialize};
use std::{
collections::{HashMap, HashSet},
num::NonZeroU64,
str::FromStr,
};
use url::Url;

/// State Machine for Auction
#[derive(Clone, Copy)]
struct Auction {
/// current phase.
phase: AuctionPhase,
/// A structure to find which Phase should be the current one.
// This could probably be an enum with some restructuring.
possible_phases: [AuctionPhase; 3],
}

impl Auction {
fn update(mut self, view: ViewNumber) {
self.phase = self
.possible_phases
.into_iter()
.find(|phase| (view > phase.start && view < phase.end))
.unwrap();
}
fn recv_bid(&self, view: ViewNumber, bid: BidTx) {
self.update(view);
match self.phase.kind {
AuctionPhaseKind::Bid => self.send_txn(bid.as_txn()),
_ => unimplemented!(),
}
}
fn send_txn(&self, txn: Transaction) {
unimplemented!();
}
#[derive(Debug, Clone, Eq, PartialEq, Deserialize, Serialize, Hash)]
struct SequencerKey;

// for MVP-0(-1) (JIT)
// Sum of all sequencing fee match current check
// builder signature no longer includes payload
// new fee info flag (fee or bid)

pub struct MarketplaceResults {
/// Slot these results are for
slot: Slot,
/// Bids that did not win, intially all bids are added to this,

Check warning on line 32 in sequencer/src/auction.rs

View workflow job for this annotation

GitHub Actions / typos

"intially" should be "initially".
/// and then the winning bids are removed
pending_bids: Vec<BidTx>,
/// Map of winning sequencer public keys to the bundle of namespaces they bid for
winner_map: HashMap<SequencerKey, HashSet<NamespaceId>>,
/// Whether refunds have been processed for this slot
refunds_processed: bool,
}

// - needs to be configured in genesis block
Expand Down Expand Up @@ -70,21 +72,42 @@ enum AuctionPhaseKind {
Sequence,
}

struct SignatureKey;
struct Slot;
#[derive(Debug, Default, Clone, Copy, Eq, PartialEq, Deserialize, Serialize, Hash)]
pub struct Slot(u64);
#[derive(Debug, Clone, Eq, PartialEq, Deserialize, Serialize, Hash)]
pub enum FullNetworkTx {
Bid(BidTx),
}

/// A transaction to bid for the sequencing rights of a namespace
impl FullNetworkTx {
pub fn execute(&self, state: &ValidatedState) -> Result<(), (ExecutionError, FullNetworkTx)> {
dbg!("execute");
match self {
Self::Bid(bid) => bid.execute(state),
}
}
}

#[derive(Debug, Clone, Eq, PartialEq, Deserialize, Serialize, Hash)]
struct BidTx {
/// Signature over all other data in this struct
signature: FeeAccount,
body: BidTxBody,
signature: Signature,
}

/// A transaction to bid for the sequencing rights of a namespace
#[derive(Debug, Clone, Eq, PartialEq, Deserialize, Serialize, Hash)]
struct BidTxBody {
/// Account responsible for the signature
account: FeeAccount,
/// Fee to be sequenced in the network. Different than the bid_amount fee
// FULL_NETWORK_GAS * MINIMUM_GAS_PRICE
gas_price: FeeAmount,
/// The bid amount designated in Wei. This is different than
/// the sequencing fee (gas price) for this transaction
bid_amount: FeeAmount,
// TODO What is the correct type?
/// The public key of this sequencer
public_key: SignatureKey,
public_key: SequencerKey,
/// The URL the HotShot leader will use to request a bundle
/// from this sequencer if they win the auction
url: Url,
Expand All @@ -94,28 +117,128 @@ struct BidTx {
bundle: Vec<NamespaceId>,
}

fn get_phase() -> AuctionPhaseKind {
AuctionPhaseKind::Bid
#[derive(Debug, Clone, Eq, PartialEq, Deserialize, Serialize, Hash)]
struct ShortBidTx {
signature: Signature,
/// Account responsible for the signature
account: FeeAccount,
url: Url,
}

fn store_in_validated_state() {
unimplemented!();
// TODO consider a committable derive macro
impl Committable for BidTxBody {
fn tag() -> String {
"BID_TX".to_string()
}

fn commit(&self) -> Commitment<Self> {
let comm = committable::RawCommitmentBuilder::new(&Self::tag())
.fixed_size_field("account", &self.account.to_fixed_bytes())
.fixed_size_field("gas_price", &self.gas_price.to_fixed_bytes())
.fixed_size_field("bid_amount", &self.bid_amount.to_fixed_bytes())
.var_size_field("url", self.url.as_str().as_ref())
.u64_field("slot", self.slot.0)
.var_size_field(
"bundle",
// TODO what is the correct way to serialize `Vec<NamespaceId>`
&bincode::serialize(&self.bundle.as_slice()).unwrap(),
);
comm.finalize()
}
}

impl BidTx {
// maybe better implemented as a From on Transaction
fn as_txn(&self) -> Transaction {
unimplemented!();
impl Default for BidTxBody {
fn default() -> Self {
let message = ";)";
let mut commitment = [0u8; 32];
commitment[..message.len()].copy_from_slice(message.as_bytes());
let (account, key) = FeeAccount::generated_from_seed_indexed([0; 32], 0);
let nsid = NamespaceId::from(999);
Self {
url: Url::from_str("htts://sequencer:3939/request_budle").unwrap(),
account,
public_key: SequencerKey,
gas_price: FeeAmount::default(),
bid_amount: FeeAmount::default(),
slot: Slot::default(),
bundle: vec![nsid],
}
}
}
impl Default for BidTx {
fn default() -> Self {
let body = BidTxBody::default();
let commitment = body.commit();
let (account, key) = FeeAccount::generated_from_seed_indexed([0; 32], 0);
let signature = FeeAccount::sign_builder_message(&key, &commitment.as_ref()).unwrap();
Self { signature, body }
}
//
fn execute(&self, state: ValidatedState) -> Option<BidExecutionResult> {
}

#[derive(Debug, Clone, Eq, PartialEq, Deserialize, Serialize, Hash)]
pub enum ExecutionError {
InvalidSignature,
InvalidPhase,
}

// TODO consider moving common functionality to trait.
impl BidTx {
/// Executes `BidTx`.
/// * verify signature
/// * charge fee
/// * store state
// The rational behind the `Err` is to provide not only what
// failed, but for which varient. The entire Tx is probably

Check warning on line 191 in sequencer/src/auction.rs

View workflow job for this annotation

GitHub Actions / typos

"varient" should be "variant".
// overkill, but we can narrow down how much we want to know about
// Failed Tx in the future. Maybe we just want its name.
pub fn execute(&self, state: &ValidatedState) -> Result<(), (ExecutionError, FullNetworkTx)> {
if get_phase() != AuctionPhaseKind::Bid {
return None;
return Err((
ExecutionError::InvalidPhase,
FullNetworkTx::Bid(self.clone()),
));
}
store_in_validated_state();
self.verify()
.map_err(|e| (e, FullNetworkTx::Bid(self.clone())))?;

// self.charge().map_err;
store_in_marketplace_state();

// TODO what do we return in good result?
Ok(())
}
// fn charge(&self) {}
/// Verify signature
fn verify(&self) -> Result<(), ExecutionError> {
if !self
.body
.account
.validate_builder_signature(&self.signature, self.body.commit().as_ref())
{
return Err(ExecutionError::InvalidSignature);
};

Ok(())
}
}

pub fn get_phase() -> AuctionPhaseKind {
AuctionPhaseKind::Bid
}

fn store_in_marketplace_state() {
unimplemented!();
}

enum ExecutionResult<T, E, K> {
Ok(T),
Err(E, K),
}

// enum ExecutionResult {
// Bid(String),
// }

/// A solution to one auction of sequencing rights for namespaces
struct AuctionResultTx {
auction: AuctionId,
Expand All @@ -124,12 +247,58 @@ struct AuctionResultTx {
signature: Signature,
}

// Not sure if needed
// Seems like sequencer could just check for refund flags on events so
// we probably don't need an explicit transaction type.
struct BidRefundTx {
nonce: Nonce,
txns_to_refund: Vec<BidTx>,
// I don't think we need the list b/c we have them in state
// txns_to_refund: Vec<BidTx>,
signature: Signature,
}

/// Nonce for special (auction) transactions
struct Nonce(u64);

pub fn mock_full_network_txs() -> Vec<FullNetworkTx> {
let x = FullNetworkTx::Bid(BidTx::default());
dbg!(&x);
vec![x]
}

mod test {
use super::*;

impl BidTx {
fn mock(account: FeeAccount, key: EthKeyPair) -> Self {
let body = BidTxBody::default();
let commitment = body.commit();
let signature = FeeAccount::sign_builder_message(&key, commitment.as_ref()).unwrap();
Self { signature, body }
}
}

#[test]
fn test_default_bidtx_body() {
let a = FeeAccount::default();

let message = ";)";
let mut commitment = [0u8; 32];
commitment[..message.len()].copy_from_slice(message.as_bytes());
let key = FeeAccount::generated_from_seed_indexed([0; 32], 0).1;
let signature = FeeAccount::sign_builder_message(&key, &commitment).unwrap();
let bid = BidTxBody::default();
dbg!(&bid);
}
#[test]
fn test_mock_full_network_txs() {
let x = mock_full_network_txs();
dbg!(&x);
}
#[test]
fn test_sign_and_verify_mock_bid() {
let account = FeeAccount::default();
let key = FeeAccount::generated_from_seed_indexed([0; 32], 0).1;
let bidtx = BidTx::mock(account, key);
bidtx.verify().unwrap();
}
}
16 changes: 16 additions & 0 deletions sequencer/src/header.rs
Original file line number Diff line number Diff line change
@@ -1,4 +1,7 @@
use std::collections::HashSet;

use crate::{
auction::{mock_full_network_txs, FullNetworkTx, Slot},
block::NsTable,
chain_config::ResolvableChainConfig,
eth_signature_key::BuilderSignature,
Expand Down Expand Up @@ -95,6 +98,11 @@ pub struct Header {
/// that `fee_info` is correct without relying on the signature. Thus, this signature is not
/// included in the header commitment.
pub builder_signature: Option<BuilderSignature>,
// pub full_network_txs: Vec<FullNetworkTx>,
// /// refund flag set at the beginning of new slots
// /// In extreme cases, more than one slot may need to be refunded,
// /// hence this data structure
// pub refund_bids: HashSet<Slot>,
}

impl Committable for Header {
Expand Down Expand Up @@ -133,6 +141,14 @@ impl Committable for Header {
}

impl Header {
// TODO move to BlockHeader
pub fn get_full_network_txs(&self) -> Vec<FullNetworkTx> {
// TODO unmock
mock_full_network_txs()
}
pub fn get_refund_bids(&self) -> HashSet<Slot> {
unimplemented!();
}
#[allow(clippy::too_many_arguments)]
fn from_info(
payload_commitment: VidCommitment,
Expand Down
Loading

0 comments on commit 0d8eeec

Please sign in to comment.