mirror of
https://github.com/logos-blockchain/logos-blockchain-testing.git
synced 2026-04-12 06:03:09 +00:00
135 lines
3.7 KiB
Markdown
135 lines
3.7 KiB
Markdown
# Example: New Workload & Expectation (Rust)
|
||
|
||
A minimal, end-to-end illustration of adding a custom workload and matching
|
||
expectation. This shows the shape of the traits and where to plug into the
|
||
framework; expand the logic to fit your real test.
|
||
|
||
## Workload: simple reachability probe
|
||
|
||
Key ideas:
|
||
- **name**: identifies the workload in logs.
|
||
- **expectations**: workloads can bundle defaults so callers don’t forget checks.
|
||
- **init**: derive inputs from the generated topology (e.g., pick a target node).
|
||
- **start**: drive async activity using the shared `RunContext`.
|
||
|
||
```rust,ignore
|
||
use async_trait::async_trait;
|
||
use testing_framework_core::{
|
||
scenario::{DynError, Expectation, RunContext, RunMetrics, Workload},
|
||
topology::generation::GeneratedTopology,
|
||
};
|
||
|
||
pub struct ReachabilityWorkload {
|
||
target_idx: usize,
|
||
}
|
||
|
||
impl ReachabilityWorkload {
|
||
pub fn new(target_idx: usize) -> Self {
|
||
Self { target_idx }
|
||
}
|
||
}
|
||
|
||
#[async_trait]
|
||
impl Workload for ReachabilityWorkload {
|
||
fn name(&self) -> &str {
|
||
"reachability_workload"
|
||
}
|
||
|
||
fn expectations(&self) -> Vec<Box<dyn Expectation>> {
|
||
vec![Box::new(
|
||
crate::custom_workload_example_expectation::ReachabilityExpectation::new(
|
||
self.target_idx,
|
||
),
|
||
)]
|
||
}
|
||
|
||
fn init(
|
||
&mut self,
|
||
topology: &GeneratedTopology,
|
||
_run_metrics: &RunMetrics,
|
||
) -> Result<(), DynError> {
|
||
if topology.nodes().get(self.target_idx).is_none() {
|
||
return Err(Box::new(std::io::Error::new(
|
||
std::io::ErrorKind::Other,
|
||
"no node at requested index",
|
||
)));
|
||
}
|
||
Ok(())
|
||
}
|
||
|
||
async fn start(&self, ctx: &RunContext) -> Result<(), DynError> {
|
||
let client = ctx
|
||
.node_clients()
|
||
.node_clients()
|
||
.get(self.target_idx)
|
||
.ok_or_else(|| {
|
||
Box::new(std::io::Error::new(
|
||
std::io::ErrorKind::Other,
|
||
"missing target client",
|
||
)) as DynError
|
||
})?;
|
||
|
||
// Lightweight API call to prove reachability.
|
||
client
|
||
.consensus_info()
|
||
.await
|
||
.map(|_| ())
|
||
.map_err(|e| e.into())
|
||
}
|
||
}
|
||
```
|
||
|
||
## Expectation: confirm the target stayed reachable
|
||
|
||
Key ideas:
|
||
- **start_capture**: snapshot baseline if needed (not used here).
|
||
- **evaluate**: assert the condition after workloads finish.
|
||
|
||
```rust,ignore
|
||
use async_trait::async_trait;
|
||
use testing_framework_core::scenario::{DynError, Expectation, RunContext};
|
||
|
||
pub struct ReachabilityExpectation {
|
||
target_idx: usize,
|
||
}
|
||
|
||
impl ReachabilityExpectation {
|
||
pub fn new(target_idx: usize) -> Self {
|
||
Self { target_idx }
|
||
}
|
||
}
|
||
|
||
#[async_trait]
|
||
impl Expectation for ReachabilityExpectation {
|
||
fn name(&self) -> &str {
|
||
"target_reachable"
|
||
}
|
||
|
||
async fn evaluate(&mut self, ctx: &RunContext) -> Result<(), DynError> {
|
||
let client = ctx
|
||
.node_clients()
|
||
.node_clients()
|
||
.get(self.target_idx)
|
||
.ok_or_else(|| {
|
||
Box::new(std::io::Error::new(
|
||
std::io::ErrorKind::Other,
|
||
"missing target client",
|
||
)) as DynError
|
||
})?;
|
||
|
||
client
|
||
.consensus_info()
|
||
.await
|
||
.map(|_| ())
|
||
.map_err(|e| e.into())
|
||
}
|
||
}
|
||
```
|
||
|
||
## How to wire it
|
||
- Build your scenario as usual and call `.with_workload(ReachabilityWorkload::new(0))`.
|
||
- The bundled expectation is attached automatically; you can add more with
|
||
`.with_expectation(...)` if needed.
|
||
- Keep the logic minimal and fast for smoke tests; grow it into richer probes
|
||
for deeper scenarios.
|