zebra_network/protocol/internal/response_status.rs
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96
//! The status of a response to an inventory request.
use std::fmt;
#[cfg(any(test, feature = "proptest-impl"))]
use proptest_derive::Arbitrary;
use InventoryResponse::*;
/// A generic peer inventory response status.
///
/// `Available` is used for inventory that is present in the response,
/// and `Missing` is used for inventory that is missing from the response.
#[derive(Copy, Clone, Debug, PartialEq, Eq, Hash)]
#[cfg_attr(any(test, feature = "proptest-impl"), derive(Arbitrary))]
pub enum InventoryResponse<A, M> {
/// An available inventory item.
Available(A),
/// A missing inventory item.
Missing(M),
}
impl<A, M> fmt::Display for InventoryResponse<A, M> {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
f.write_str(self.command())
}
}
impl<A, M> InventoryResponse<A, M> {
/// Returns the response status type as a string.
pub fn command(&self) -> &'static str {
match self {
InventoryResponse::Available(_) => "Available",
InventoryResponse::Missing(_) => "Missing",
}
}
/// Returns true if the inventory item was available.
pub fn is_available(&self) -> bool {
matches!(self, Available(_))
}
/// Returns true if the inventory item was missing.
pub fn is_missing(&self) -> bool {
matches!(self, Missing(_))
}
/// Maps a `InventoryResponse<A, M>` to `InventoryResponse<B, M>` by applying a function to a
/// contained [`Available`] value, leaving the [`Missing`] value untouched.
pub fn map_available<B, F: FnOnce(A) -> B>(self, f: F) -> InventoryResponse<B, M> {
// Based on Result::map from https://doc.rust-lang.org/src/core/result.rs.html#765
match self {
Available(a) => Available(f(a)),
Missing(m) => Missing(m),
}
}
/// Maps a `InventoryResponse<A, M>` to `InventoryResponse<A, N>` by applying a function to a
/// contained [`Missing`] value, leaving the [`Available`] value untouched.
pub fn map_missing<N, F: FnOnce(M) -> N>(self, f: F) -> InventoryResponse<A, N> {
// Based on Result::map_err from https://doc.rust-lang.org/src/core/result.rs.html#850
match self {
Available(a) => Available(a),
Missing(m) => Missing(f(m)),
}
}
/// Converts from `&InventoryResponse<A, M>` to `InventoryResponse<&A, &M>`.
pub fn as_ref(&self) -> InventoryResponse<&A, &M> {
match self {
Available(item) => Available(item),
Missing(item) => Missing(item),
}
}
}
impl<A: Clone, M: Clone> InventoryResponse<A, M> {
/// Get the available inventory item, if present.
pub fn available(&self) -> Option<A> {
if let Available(item) = self {
Some(item.clone())
} else {
None
}
}
/// Get the missing inventory item, if present.
pub fn missing(&self) -> Option<M> {
if let Missing(item) = self {
Some(item.clone())
} else {
None
}
}
}