Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: implement and calculate the current key period using pallas #1541

Merged
merged 24 commits into from
Mar 12, 2024
Merged
Show file tree
Hide file tree
Changes from 7 commits
Commits
Show all changes
24 commits
Select commit Hold shift + click to select a range
a1e7679
feat: add kes period
falcucci Feb 22, 2024
72ec63d
refactor: remove fallback field usage
falcucci Feb 22, 2024
a166a10
refactor: streamline code formatting in various modules
falcucci Feb 22, 2024
96dcd34
chore: update dependency sources to `txpipe` repository
falcucci Feb 28, 2024
583a7f1
merge 'main'
falcucci Feb 28, 2024
2390899
refactor: improved error handling and configuration handling
falcucci Feb 28, 2024
8343f83
chore: improve error messages for config extraction process
falcucci Feb 28, 2024
c1a1537
fix: improve slots_per_kes_period validation and error messaging
falcucci Feb 29, 2024
5afcb59
feat: refactor kes period calculation in chain module
falcucci Feb 29, 2024
4783f80
chore: update specific points in `getchainpoint` request
falcucci Feb 29, 2024
b2ee2bd
chore: refactor method signature for `calculate_kes_period`
falcucci Feb 29, 2024
cc4109c
feat: refactor return types and error handling in functions
falcucci Mar 1, 2024
17f06ee
refactor: implementation logic for `slots_per_kes_period` being `0`
falcucci Mar 1, 2024
9079e07
docs: enhance documentation for kes period calculation formula
falcucci Mar 1, 2024
691b337
docs: correct variable descriptions
falcucci Mar 1, 2024
af349bf
refactor: standardize variable naming conventions across files
falcucci Mar 1, 2024
9218952
Merge branch 'main' into feat/kes_period
falcucci Mar 1, 2024
f20259d
refactor: chainobserver imports in infrastructure codebase
falcucci Mar 1, 2024
5284e56
merge 'main'
falcucci Mar 11, 2024
1d6648c
chore: update version number and dependencies
falcucci Mar 11, 2024
f9ee712
docs: standardize code comments and examples clarity
falcucci Mar 11, 2024
3d368d6
docs: revise comments for kes period calculations across files
falcucci Mar 11, 2024
7e51687
docs: update documentation and refactor code in observer module
falcucci Mar 11, 2024
4e87a39
feat: refactor `calculate_kes_period` function in pallas observer
falcucci Mar 11, 2024
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
18 changes: 6 additions & 12 deletions Cargo.lock

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

18 changes: 12 additions & 6 deletions mithril-common/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -39,11 +39,16 @@ minicbor = { version = "0.20", features = [
"derive",
], optional = true }
nom = "7.1.3"
pallas-addresses = { version = "0.23.0", optional = true }
pallas-codec = { version = "0.23.0", optional = true }
pallas-network = { version = "0.23.0", optional = true }
pallas-primitives = { version = "0.23.0", optional = true }
pallas-traverse = { version = "0.23.0", optional = true }
# pallas-addresses = { version = "0.23.0", optional = true }
pallas-addresses = { git = "https://github.com/txpipe/pallas.git", branch = "main", optional = true }
# pallas-codec = { version = "0.23.0", optional = true }
pallas-codec = { git = "https://github.com/txpipe/pallas.git", branch = "main", optional = true }
# pallas-network = { version = "0.23.0", optional = true }
pallas-network = { git = "https://github.com/txpipe/pallas.git", branch = "main", optional = true }
# pallas-primitives = { version = "0.23.0", optional = true }
pallas-primitives = { git = "https://github.com/txpipe/pallas.git", branch = "main", optional = true }
# pallas-traverse = { version = "0.23.0", optional = true }
pallas-traverse = { git = "https://github.com/txpipe/pallas.git", branch = "main", optional = true }
rand_chacha = "0.3.1"
rand_core = "0.6.4"
rayon = "1.8.1"
Expand Down Expand Up @@ -83,7 +88,8 @@ wasm-bindgen = "0.2.90"
[dev-dependencies]
criterion = { version = "0.5.1", features = ["html_reports", "async_tokio"] }
mockall = "0.12.1"
pallas-crypto = "0.23.0"
# pallas-crypto = "0.23.0"
pallas-crypto = { git = "https://github.com/txpipe/pallas.git", branch = "main" }
rand_core = { version = "0.6.4", features = ["getrandom"] }
reqwest = { version = "0.11.23", features = ["json"] }
slog-async = "2.8.0"
Expand Down
13 changes: 2 additions & 11 deletions mithril-common/src/chain_observer/builder.rs
Original file line number Diff line number Diff line change
Expand Up @@ -75,17 +75,8 @@ impl ChainObserverBuilder {
.to_owned(),
))),
ChainObserverType::Pallas => {
let fallback = CardanoCliChainObserver::new(
self.cardano_cli_runner
.as_ref()
.ok_or(ChainObserverBuilderError::MissingCardanoCliRunner)?
.to_owned(),
);
let observer = PallasChainObserver::new(
&self.cardano_node_socket_path,
self.cardano_network,
fallback,
);
let observer =
PallasChainObserver::new(&self.cardano_node_socket_path, self.cardano_network);
Ok(Arc::new(observer))
}
#[cfg(any(test, feature = "test_tools"))]
Expand Down
171 changes: 128 additions & 43 deletions mithril-common/src/chain_observer/pallas_observer.rs
Original file line number Diff line number Diff line change
Expand Up @@ -28,13 +28,11 @@ use crate::{
};

use super::model::{try_inspect, Datum, Datums};
use super::CardanoCliChainObserver;

/// A runner that uses Pallas library to interact with a Cardano node using N2C Ouroboros mini-protocols
pub struct PallasChainObserver {
socket: PathBuf,
network: CardanoNetwork,
fallback: CardanoCliChainObserver,
}

impl From<anyhow::Error> for ChainObserverError {
Expand All @@ -44,12 +42,11 @@ impl From<anyhow::Error> for ChainObserverError {
}

impl PallasChainObserver {
/// Creates a new PallasObserver while accepting a fallback CliRunner
pub fn new(socket: &Path, network: CardanoNetwork, fallback: CardanoCliChainObserver) -> Self {
/// Creates a new PallasObserver
pub fn new(socket: &Path, network: CardanoNetwork) -> Self {
Self {
socket: socket.to_owned(),
network,
fallback,
}
}

Expand All @@ -61,11 +58,6 @@ impl PallasChainObserver {
Ok(client)
}

/// Returns a reference to the fallback `CardanoCliChainObserver` instance.
fn get_fallback(&self) -> &CardanoCliChainObserver {
&self.fallback
}

/// Creates and returns a new `NodeClient`, handling any potential errors.
async fn get_client(&self) -> StdResult<NodeClient> {
self.new_client()
Expand Down Expand Up @@ -239,6 +231,45 @@ impl PallasChainObserver {
Ok(Some(stake_distribution))
}

/// Fetches chain point and genesis config through the local statequery.
/// The KES period is calculated afterwards.
async fn calculate_kes_period(
&self,
client: &mut NodeClient,
) -> Result<Option<KESPeriod>, ChainObserverError> {
let statequery = client.statequery();

statequery
.acquire(None)
.await
.map_err(|err| anyhow!(err))
.with_context(|| "PallasChainObserver failed to acquire statequery")?;

let chain_point = queries_v16::get_chain_point(statequery)
.await
.map_err(|err| anyhow!(err))
.with_context(|| "PallasChainObserver failed to get chain point")?;

let era = queries_v16::get_current_era(statequery)
.await
.map_err(|err| anyhow!(err))
.with_context(|| "PallasChainObserver failed to get current era")?;

let genesis_config = queries_v16::get_genesis_config(statequery, era)
.await
.map_err(|err| anyhow!(err))
.with_context(|| "PallasChainObserver failed to get genesis config")?;

let config = genesis_config
.first()
.with_context(|| "PallasChainObserver failed to extract the config")?;

let slots_per_kes_period = config.slots_per_kes_period as u64;
let current_kes_period = chain_point.slot_or_default() / slots_per_kes_period;
falcucci marked this conversation as resolved.
Show resolved Hide resolved

Ok(Some(current_kes_period as u32))
}

/// Processes a state query with the `NodeClient`, releasing the state query.
async fn process_statequery(&self, client: &mut NodeClient) -> StdResult<()> {
let statequery = client.statequery();
Expand Down Expand Up @@ -322,30 +353,42 @@ impl ChainObserver for PallasChainObserver {

async fn get_current_kes_period(
&self,
opcert: &OpCert,
_opcert: &OpCert,
) -> Result<Option<KESPeriod>, ChainObserverError> {
let fallback = self.get_fallback();
fallback.get_current_kes_period(opcert).await
let mut client = self.get_client().await?;

let current_kes_period = self.calculate_kes_period(&mut client).await?;

self.post_process_statequery(&mut client).await?;

client.abort().await;

Ok(current_kes_period)
}
}

#[cfg(test)]
mod tests {
use std::fs;

use kes_summed_ed25519::{kes::Sum6Kes, traits::KesSk};
use pallas_codec::utils::{AnyCbor, AnyUInt, KeyValuePairs, TagWrap};
use pallas_crypto::hash::Hash;
use pallas_network::miniprotocols::localstate::{
queries_v16::{
BlockQuery, HardForkQuery, LedgerQuery, Request, Snapshots, StakeSnapshot, Value,
use pallas_network::miniprotocols::{
localstate::{
queries_v16::{
BlockQuery, Fraction, Genesis, HardForkQuery, LedgerQuery, Request, Snapshots,
StakeSnapshot, SystemStart, Value,
},
ClientQueryRequest,
},
ClientQueryRequest,
Point,
};
use tokio::net::UnixListener;

use super::*;
use crate::test_utils::TempDir;
use crate::{chain_observer::test_cli_runner::TestCliRunner, CardanoNetwork};
use crate::{crypto_helper::ColdKeyGenerator, CardanoNetwork};

fn get_fake_utxo_by_address() -> UTxOByAddress {
let tx_hex = "1e4e5cf2889d52f1745b941090f04a65dea6ce56c5e5e66e69f65c8e36347c17";
Expand Down Expand Up @@ -437,6 +480,28 @@ mod tests {
}
}

fn get_fake_genesis_config() -> Vec<Genesis> {
let genesis = Genesis {
system_start: SystemStart {
year: 2021,
day_of_year: 150,
picoseconds_of_day: 0,
},
network_magic: 42,
network_id: 42,
active_slots_coefficient: Fraction { num: 6, dem: 10 },
security_param: 2160,
epoch_length: 432000,
slots_per_kes_period: 129600,
max_kes_evolutions: 62,
slot_length: 1,
update_quorum: 5,
max_lovelace_supply: AnyUInt::MajorByte(2),
};

vec![genesis]
}

/// pallas responses mock server.
async fn mock_server(server: &mut pallas_network::facades::NodeServer) -> AnyCbor {
let query: queries_v16::Request =
Expand All @@ -446,12 +511,18 @@ mod tests {
};

match query {
Request::GetChainPoint => {
AnyCbor::from_encode(Point::Specific(52851885, vec![1, 2, 3]))
}
Request::LedgerQuery(LedgerQuery::HardForkQuery(HardForkQuery::GetCurrentEra)) => {
AnyCbor::from_encode(4)
}
Request::LedgerQuery(LedgerQuery::BlockQuery(_, BlockQuery::GetEpochNo)) => {
AnyCbor::from_encode([8])
}
Request::LedgerQuery(LedgerQuery::BlockQuery(_, BlockQuery::GetGenesisConfig)) => {
AnyCbor::from_encode(get_fake_genesis_config())
}
Request::LedgerQuery(LedgerQuery::BlockQuery(_, BlockQuery::GetUTxOByAddress(_))) => {
AnyCbor::from_encode(get_fake_utxo_by_address())
}
Expand Down Expand Up @@ -488,21 +559,20 @@ mod tests {

let result = mock_server(&mut server).await;
server.statequery().send_result(result).await.unwrap();

let result = mock_server(&mut server).await;
server.statequery().send_result(result).await.unwrap();
}
})
}

#[tokio::test]
async fn get_current_epoch_with_fallback() {
let socket_path = create_temp_dir("get_current_epoch_with_fallback").join("node.socket");
async fn get_current_epoch() {
let socket_path = create_temp_dir("get_current_epoch").join("node.socket");
let server = setup_server(socket_path.clone()).await;
let client = tokio::spawn(async move {
let fallback = CardanoCliChainObserver::new(Box::<TestCliRunner>::default());
let observer = PallasChainObserver::new(
socket_path.as_path(),
CardanoNetwork::TestNet(10),
fallback,
);
let observer =
PallasChainObserver::new(socket_path.as_path(), CardanoNetwork::TestNet(10));
observer.get_current_epoch().await.unwrap().unwrap()
});

Expand All @@ -512,16 +582,12 @@ mod tests {
}

#[tokio::test]
async fn get_current_datums_with_fallback() {
let socket_path = create_temp_dir("get_current_datums_with_fallback").join("node.socket");
async fn get_current_datums() {
let socket_path = create_temp_dir("get_current_datums").join("node.socket");
let server = setup_server(socket_path.clone()).await;
let client = tokio::spawn(async move {
let fallback = CardanoCliChainObserver::new(Box::<TestCliRunner>::default());
let observer = PallasChainObserver::new(
socket_path.as_path(),
CardanoNetwork::TestNet(10),
fallback,
);
let observer =
PallasChainObserver::new(socket_path.as_path(), CardanoNetwork::TestNet(10));
let address =
"addr_test1vr80076l3x5uw6n94nwhgmv7ssgy6muzf47ugn6z0l92rhg2mgtu0".to_string();
observer.get_current_datums(&address).await.unwrap()
Expand All @@ -533,17 +599,12 @@ mod tests {
}

#[tokio::test]
async fn get_current_stake_distribution_with_fallback() {
let socket_path =
create_temp_dir("get_current_stake_distribution_with_fallback").join("node.socket");
async fn get_current_stake_distribution() {
let socket_path = create_temp_dir("get_current_stake_distribution").join("node.socket");
let server = setup_server(socket_path.clone()).await;
let client = tokio::spawn(async move {
let fallback = CardanoCliChainObserver::new(Box::<TestCliRunner>::default());
let observer = super::PallasChainObserver::new(
socket_path.as_path(),
CardanoNetwork::TestNet(10),
fallback,
);
let observer =
super::PallasChainObserver::new(socket_path.as_path(), CardanoNetwork::TestNet(10));
observer.get_current_stake_distribution().await.unwrap()
});

Expand All @@ -566,4 +627,28 @@ mod tests {

assert_eq!(expected_stake_distribution, computed_stake_distribution);
}

#[tokio::test]
async fn get_current_kes_period() {
let socket_path = create_temp_dir("get_current_kes_period").join("node.socket");
falcucci marked this conversation as resolved.
Show resolved Hide resolved
let server = setup_server(socket_path.clone()).await;
let client = tokio::spawn(async move {
let observer =
super::PallasChainObserver::new(socket_path.as_path(), CardanoNetwork::TestNet(10));

let keypair = ColdKeyGenerator::create_deterministic_keypair([0u8; 32]);
let mut dummy_key_buffer = [0u8; Sum6Kes::SIZE + 4];
let mut dummy_seed = [0u8; 32];
let (_, kes_verification_key) = Sum6Kes::keygen(&mut dummy_key_buffer, &mut dummy_seed);
let operational_certificate = OpCert::new(kes_verification_key, 0, 0, keypair);
observer
.get_current_kes_period(&operational_certificate)
.await
.unwrap()
});

let (_, client_res) = tokio::join!(server, client);
let kes_period = client_res.unwrap().unwrap();
assert_eq!(407, kes_period);
falcucci marked this conversation as resolved.
Show resolved Hide resolved
}
}
Loading
Loading