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
use std::fmt::Debug;
use peace_resources::{resources::ts::SetUp, type_reg::untagged::DataType, Resources};
use serde::{Serialize, Serializer};
use crate::{ParamsResolveError, ValueResolutionCtx};
/// Type erased mapping function.
///
/// This is used by Peace to hold type-erased mapping functions, and is not
/// intended to be implemented by users or implementors.
pub trait MappingFn: Debug + DataType {
/// Type that is output by the function.
type Output;
/// Maps data in resources to the output type.
///
/// The data being accessed is defined by the implementation of this
/// function.
///
/// # Parameters
///
/// * `resources`: Resources to resolve values from.
/// * `value_resolution_ctx`: Fields traversed during this value resolution.
fn map(
&self,
resources: &Resources<SetUp>,
value_resolution_ctx: &mut ValueResolutionCtx,
) -> Result<Self::Output, ParamsResolveError>;
/// Maps data in resources to the output type.
///
/// The data being accessed is defined by the implementation of this
/// function.
///
/// # Parameters
///
/// * `resources`: Resources to resolve values from.
/// * `value_resolution_ctx`: Fields traversed during this value resolution.
fn try_map(
&self,
resources: &Resources<SetUp>,
value_resolution_ctx: &mut ValueResolutionCtx,
) -> Result<Option<Self::Output>, ParamsResolveError>;
/// Returns whether this mapping function actually holds the function logic.
///
/// Deserialized mapping functions will not hold any function logic, and
/// Peace uses this function to determine if this is an empty `MappingFn`.
fn is_valued(&self) -> bool;
}
impl<T> Clone for Box<dyn MappingFn<Output = T>> {
fn clone(&self) -> Self {
dyn_clone::clone_box(&**self)
}
}
impl<'a, T> Serialize for dyn MappingFn<Output = T> + 'a {
fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
where
S: Serializer,
{
// Sadly the following doesn't work, it says the lifetime of:
// `&'1 self` must outlive `'static`
//
// let data_type: &(dyn DataType + 'a) = &self;
// Serialize::serialize(data_type, serializer)
// so we have to depend on `erased_serde` directly
erased_serde::serialize(self, serializer)
}
}