use errors::Result;
use {Diagnostics, Flavor, FlavorField, Translate, Translator};
#[derive(Debug, Clone, Serialize, PartialEq, Eq)]
#[serde(bound = "F::Type: ::serde::Serialize")]
pub struct RpField<F: 'static>
where
F: Flavor,
{
pub required: bool,
#[serde(skip_serializing_if = "Option::is_none")]
pub safe_ident: Option<String>,
pub ident: String,
pub comment: Vec<String>,
#[serde(rename = "type")]
pub ty: F::Type,
#[serde(skip_serializing_if = "Option::is_none")]
pub field_as: Option<String>,
}
impl<F: 'static> FlavorField for RpField<F>
where
F: Flavor,
{
fn is_discriminating(&self) -> bool {
self.required
}
}
impl<F: 'static> RpField<F>
where
F: Flavor,
{
pub fn new<S: AsRef<str>>(ident: S, ty: F::Type) -> Self {
RpField {
required: true,
safe_ident: None,
ident: ident.as_ref().to_string(),
comment: Vec::new(),
ty: ty,
field_as: None,
}
}
pub fn is_optional(&self) -> bool {
!self.required
}
pub fn is_required(&self) -> bool {
self.required
}
pub fn safe_ident(&self) -> &str {
self.safe_ident.as_ref().unwrap_or(&self.ident)
}
pub fn with_safe_ident<S: AsRef<str>>(self, safe_ident: S) -> RpField<F> {
Self {
safe_ident: Some(safe_ident.as_ref().to_string()),
..self
}
}
pub fn ident(&self) -> &str {
&self.ident
}
pub fn name(&self) -> &str {
self.field_as.as_ref().unwrap_or(&self.ident)
}
pub fn ty(&self) -> &F::Type {
&self.ty
}
pub fn display(&self) -> String {
self.name().to_owned()
}
}
impl<F: 'static, T> Translate<T> for RpField<F>
where
F: Flavor,
T: Translator<Source = F>,
{
type Source = F;
type Out = RpField<T::Target>;
fn translate(self, diag: &mut Diagnostics, translator: &T) -> Result<RpField<T::Target>> {
Ok(RpField {
required: self.required,
safe_ident: self.safe_ident,
ident: self.ident,
comment: self.comment,
ty: translator.translate_type(diag, self.ty)?,
field_as: self.field_as,
})
}
}