pbjson-build 0.9.0

Generates Serialize and Deserialize implementations for prost message types
Documentation
//! This module contains the code to generate Serialize and Deserialize
//! implementations for enumeration type
//!
//! An enumeration should be decode-able from the full string variant name
//! or its integer tag number, and should encode to the string representation

use super::{
    Indent, write_deserialize_end, write_deserialize_start, write_serialize_end,
    write_serialize_start,
};
use crate::descriptor::{EnumDescriptor, TypePath};
use crate::generator::write_fields_array;
use crate::resolver::Resolver;
use std::collections::HashSet;
use std::io::{Result, Write};

pub fn generate_enum<W: Write>(
    resolver: &Resolver<'_>,
    path: &TypePath,
    descriptor: &EnumDescriptor,
    writer: &mut W,
    use_integers_for_enums: bool,
    ignore_unknown_enum_variants: bool,
) -> Result<()> {
    let rust_type = resolver.rust_type(path);

    let mut seen_numbers = HashSet::new();
    let variants: Vec<_> = descriptor
        .values
        .iter()
        // Skip duplicates if we've seen the number before
        // Protobuf's `allow_alias` option permits duplicates if set
        .filter(|variant| seen_numbers.insert(variant.number()))
        .map(|variant| {
            let variant_name = variant.name.clone().unwrap();
            let variant_number = variant.number();
            let rust_variant = resolver.rust_variant(path, &variant_name);
            (variant_name, variant_number, rust_variant)
        })
        .collect();

    // Generate Serialize
    write_serialize_start(0, &rust_type, writer)?;
    if use_integers_for_enums {
        writeln!(writer, "{}let variant = match self {{", Indent(2))?;
        for (_, variant_number, rust_variant) in &variants {
            writeln!(
                writer,
                "{}Self::{} => {},",
                Indent(3),
                rust_variant,
                variant_number
            )?;
        }
        writeln!(writer, "{}}};", Indent(2))?;

        writeln!(writer, "{}serializer.serialize_i32(variant)", Indent(2))?;
    } else {
        writeln!(writer, "{}let variant = match self {{", Indent(2))?;
        for (variant_name, _, rust_variant) in &variants {
            writeln!(
                writer,
                "{}Self::{} => \"{}\",",
                Indent(3),
                rust_variant,
                variant_name
            )?;
        }
        writeln!(writer, "{}}};", Indent(2))?;

        writeln!(writer, "{}serializer.serialize_str(variant)", Indent(2))?;
    }
    write_serialize_end(0, writer)?;

    // Generate Deserialize
    write_deserialize_start(0, &rust_type, writer)?;
    write_fields_array(writer, 2, variants.iter().map(|(name, _, _)| name.as_str()))?;
    write_visitor(
        writer,
        2,
        &rust_type,
        &variants,
        ignore_unknown_enum_variants,
    )?;

    // Use deserialize_any to allow users to provide integers or strings
    writeln!(
        writer,
        "{}deserializer.deserialize_any(GeneratedVisitor)",
        Indent(2)
    )?;

    write_deserialize_end(0, writer)?;
    Ok(())
}

fn write_visitor<W: Write>(
    writer: &mut W,
    indent: usize,
    rust_type: &str,
    variants: &[(String, i32, String)],
    ignore_unknown_enum_variants: bool,
) -> Result<()> {
    // These are what needs to be done for an unknown i32 or string value.
    let (or_unknown_i32, unknown_string_return) = if ignore_unknown_enum_variants {
        // If ignore_unknown_enum_variants is set, we will return the default for the enum.
        (
            format!(".or_else(|| Some({rust_type}::default()))"),
            format!("Ok({rust_type}::default())"),
        )
    } else {
        // If ignore_unknown_enum_variants is not set, we will return an Err.
        (
            "".into(),
            "Err(serde::de::Error::unknown_variant(value, FIELDS))".into(),
        )
    };

    // Protobuf supports deserialization of enumerations both from string and integer values
    writeln!(
        writer,
        r#"{indent}struct GeneratedVisitor;

{indent}impl serde::de::Visitor<'_> for GeneratedVisitor {{
{indent}    type Value = {rust_type};

{indent}    fn expecting(&self, formatter: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {{
{indent}        write!(formatter, "expected one of: {{:?}}", &FIELDS)
{indent}    }}

{indent}    fn visit_i64<E>(self, v: i64) -> std::result::Result<Self::Value, E>
{indent}    where
{indent}        E: serde::de::Error,
{indent}    {{
{indent}        i32::try_from(v)
{indent}            .ok()
{indent}            .and_then(|x| x.try_into().ok(){or_unknown_i32})
{indent}            .ok_or_else(|| {{
{indent}                serde::de::Error::invalid_value(serde::de::Unexpected::Signed(v), &self)
{indent}            }})
{indent}    }}

{indent}    fn visit_u64<E>(self, v: u64) -> std::result::Result<Self::Value, E>
{indent}    where
{indent}        E: serde::de::Error,
{indent}    {{
{indent}        i32::try_from(v)
{indent}            .ok()
{indent}            .and_then(|x| x.try_into().ok(){or_unknown_i32})
{indent}            .ok_or_else(|| {{
{indent}                serde::de::Error::invalid_value(serde::de::Unexpected::Unsigned(v), &self)
{indent}            }})
{indent}    }}

{indent}    fn visit_str<E>(self, value: &str) -> std::result::Result<Self::Value, E>
{indent}    where
{indent}        E: serde::de::Error,
{indent}    {{"#,
        indent = Indent(indent),
        rust_type = rust_type,
    )?;

    writeln!(writer, "{}match value {{", Indent(indent + 2))?;
    for (variant_name, _, rust_variant) in variants {
        writeln!(
            writer,
            "{}\"{}\" => Ok({}::{}),",
            Indent(indent + 3),
            variant_name,
            rust_type,
            rust_variant
        )?;
    }

    writeln!(
        writer,
        "{indent}_ => {unknown_string_return},",
        indent = Indent(indent + 3)
    )?;
    writeln!(writer, "{}}}", Indent(indent + 2))?;
    writeln!(writer, "{}}}", Indent(indent + 1))?;
    writeln!(writer, "{}}}", Indent(indent))
}