prost_reflect/dynamic/text_format/mod.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 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212
mod format;
#[cfg(feature = "text-format")]
mod parse;
#[cfg(feature = "text-format")]
pub use self::parse::ParseError;
#[cfg(feature = "text-format")]
use crate::{DynamicMessage, MessageDescriptor};
pub(super) use self::format::Writer;
/// Options to control printing of the protobuf text format.
///
/// Used by [`DynamicMessage::to_text_format_with_options()`].
#[derive(Debug, Clone)]
#[cfg_attr(docsrs, doc(cfg(feature = "text-format")))]
pub struct FormatOptions {
pretty: bool,
skip_unknown_fields: bool,
expand_any: bool,
skip_default_fields: bool,
}
#[cfg(feature = "text-format")]
impl DynamicMessage {
/// Parse a [`DynamicMessage`] from the given message encoded using the [text format](https://developers.google.com/protocol-buffers/docs/text-format-spec).
///
/// # Examples
///
/// ```
/// # use prost::Message;
/// # use prost_reflect::{DynamicMessage, DescriptorPool, Value};
/// # let pool = DescriptorPool::decode(include_bytes!("../../file_descriptor_set.bin").as_ref()).unwrap();
/// # let message_descriptor = pool.get_message_by_name("package.MyMessage").unwrap();
/// let dynamic_message = DynamicMessage::parse_text_format(message_descriptor, "foo: 150").unwrap();
/// assert_eq!(dynamic_message.get_field_by_name("foo").unwrap().as_ref(), &Value::I32(150));
/// ```
#[cfg_attr(docsrs, doc(cfg(feature = "text-format")))]
pub fn parse_text_format(desc: MessageDescriptor, input: &str) -> Result<Self, ParseError> {
let mut message = DynamicMessage::new(desc);
message.merge_text_format(input)?;
Ok(message)
}
/// Merges the given message encoded using the [text format](https://developers.google.com/protocol-buffers/docs/text-format-spec) into this message.
///
/// # Examples
///
/// ```
/// # use prost::Message;
/// # use prost_reflect::{DynamicMessage, DescriptorPool, Value};
/// # let pool = DescriptorPool::decode(include_bytes!("../../file_descriptor_set.bin").as_ref()).unwrap();
/// # let message_descriptor = pool.get_message_by_name("package.MyMessage").unwrap();
/// let mut dynamic_message = DynamicMessage::new(message_descriptor);
/// dynamic_message.merge_text_format("foo: 150").unwrap();
/// assert_eq!(dynamic_message.get_field_by_name("foo").unwrap().as_ref(), &Value::I32(150));
/// ```
#[cfg_attr(docsrs, doc(cfg(feature = "text-format")))]
pub fn merge_text_format(&mut self, input: &str) -> Result<(), ParseError> {
parse::Parser::new(input)
.parse_message(self)
.map_err(|kind| ParseError::new(kind, input))
}
/// Formats this dynamic message using the protobuf text format, with default options.
///
/// # Examples
///
/// ```
/// # use prost::Message;
/// # use prost_types::FileDescriptorSet;
/// # use prost_reflect::{DynamicMessage, DescriptorPool, Value, text_format::FormatOptions};
/// # let pool = DescriptorPool::decode(include_bytes!("../../file_descriptor_set.bin").as_ref()).unwrap();
/// # let message_descriptor = pool.get_message_by_name("package.MyMessage").unwrap();
/// let dynamic_message = DynamicMessage::decode(message_descriptor, b"\x08\x96\x01\x1a\x02\x10\x42".as_ref()).unwrap();
/// assert_eq!(dynamic_message.to_text_format(), "foo:150,nested{bar:66}");
/// ```
#[cfg_attr(docsrs, doc(cfg(feature = "text-format")))]
pub fn to_text_format(&self) -> String {
self.to_text_format_with_options(&FormatOptions::new())
}
/// Formats this dynamic message using the protobuf text format, with custom options.
///
/// # Examples
///
/// ```
/// # use prost::Message;
/// # use prost_types::FileDescriptorSet;
/// # use prost_reflect::{DynamicMessage, DescriptorPool, Value, text_format::FormatOptions};
/// # let pool = DescriptorPool::decode(include_bytes!("../../file_descriptor_set.bin").as_ref()).unwrap();
/// # let message_descriptor = pool.get_message_by_name("package.MyMessage").unwrap();
/// let dynamic_message = DynamicMessage::decode(message_descriptor, b"\x08\x96\x01\x1a\x02\x10\x42".as_ref()).unwrap();
/// let options = FormatOptions::new().pretty(true);
/// assert_eq!(dynamic_message.to_text_format_with_options(&options), "foo: 150\nnested {\n bar: 66\n}");
/// ```
#[cfg_attr(docsrs, doc(cfg(feature = "text-format")))]
pub fn to_text_format_with_options(&self, options: &FormatOptions) -> String {
let mut result = String::new();
format::Writer::new(options.clone(), &mut result)
.fmt_message(self)
.expect("writing to string cannot fail");
result
}
}
impl FormatOptions {
/// Creates new instance of [`FormatOptions`] with default options.
pub fn new() -> Self {
FormatOptions::default()
}
/// Whether to prettify the format output.
///
/// If set to `true`, each field will be printed on a new line, and nested messages will be indented.
///
/// The default value is `false`.
pub fn pretty(mut self, yes: bool) -> Self {
self.pretty = yes;
self
}
/// Whether to include unknown fields in the output.
///
/// If set to `false`, unknown fields will be printed. The protobuf format does not include type information,
/// so the formatter will attempt to infer types.
///
/// The default value is `true`.
///
/// # Examples
///
/// ```
/// # use prost::Message;
/// # use prost_types::FileDescriptorSet;
/// # use prost_reflect::{DynamicMessage, DescriptorPool, Value, text_format::FormatOptions};
/// # let pool = DescriptorPool::decode(include_bytes!("../../file_descriptor_set.bin").as_ref()).unwrap();
/// # let message_descriptor = pool.get_message_by_name("google.protobuf.Empty").unwrap();
/// let dynamic_message = DynamicMessage::decode(message_descriptor, b"\x08\x96\x01\x1a\x02\x10\x42".as_ref()).unwrap();
/// assert_eq!(dynamic_message.to_text_format(), "");
/// let options = FormatOptions::new().skip_unknown_fields(false);
/// assert_eq!(dynamic_message.to_text_format_with_options(&options), "1:150,3{2:66}");
/// ```
#[cfg(feature = "text-format")]
pub fn skip_unknown_fields(mut self, yes: bool) -> Self {
self.skip_unknown_fields = yes;
self
}
/// Whether to skip fields which have their default value.
///
/// If `true`, any fields for which [`has_field`][DynamicMessage::has_field] returns `false` will
/// not be included. If `false`, they will be included with their default value.
///
/// The default value is `true`.
#[cfg(feature = "text-format")]
pub fn skip_default_fields(mut self, yes: bool) -> Self {
self.skip_default_fields = yes;
self
}
/// Whether to use the expanded form of the `google.protobuf.Any` type.
///
/// If set to `true`, `Any` fields will use an expanded form:
///
/// ```textproto
/// [type.googleapis.com/package.MyMessage] {
/// foo: 150
/// }
/// ```
///
/// If set to `false`, the normal text format representation will be used:
///
/// ```textproto
/// type_url: "type.googleapis.com/package.MyMessage"
/// value: "\x08\x96\x01"
/// ```
///
/// The default value is `true`.
///
/// # Examples
///
/// ```
/// # use prost::Message;
/// # use prost_types::FileDescriptorSet;
/// # use prost_reflect::{DynamicMessage, DescriptorPool, Value, text_format::FormatOptions, bytes::Bytes};
/// # let pool = DescriptorPool::decode(include_bytes!("../../file_descriptor_set.bin").as_ref()).unwrap();
/// let message_descriptor = pool.get_message_by_name("google.protobuf.Any").unwrap();
/// let mut dynamic_message = DynamicMessage::new(message_descriptor);
/// dynamic_message.set_field_by_name("type_url", Value::String("type.googleapis.com/package.MyMessage".to_owned()));
/// dynamic_message.set_field_by_name("value", Value::Bytes(Bytes::from_static(b"\x08\x96\x01\x1a\x02\x10\x42".as_ref())));
///
/// assert_eq!(dynamic_message.to_text_format(), "[type.googleapis.com/package.MyMessage]{foo:150,nested{bar:66}}");
/// let options = FormatOptions::new().expand_any(false);
/// assert_eq!(dynamic_message.to_text_format_with_options(&options), r#"type_url:"type.googleapis.com/package.MyMessage",value:"\010\226\001\032\002\020B""#);
/// ```
#[cfg(feature = "text-format")]
pub fn expand_any(mut self, yes: bool) -> Self {
self.expand_any = yes;
self
}
}
impl Default for FormatOptions {
fn default() -> Self {
FormatOptions {
pretty: false,
skip_unknown_fields: true,
expand_any: true,
skip_default_fields: true,
}
}
}