Skip to content

Commit

Permalink
sanitize default values
Browse files Browse the repository at this point in the history
  • Loading branch information
tafia committed Feb 14, 2017
1 parent f54c456 commit 0716e7b
Show file tree
Hide file tree
Showing 2 changed files with 86 additions and 54 deletions.
4 changes: 2 additions & 2 deletions codegen/src/parser.rs
Original file line number Diff line number Diff line change
Expand Up @@ -53,8 +53,8 @@ named!(reserved_names<Vec<String>>,
named!(key_val<(&str, &str)>,
do_parse!(tag!("[") >> many0!(br) >>
key: word_ref >> many0!(br) >> tag!("=") >> many0!(br) >>
value: word_ref >> many0!(br) >> tag!("]") >> many0!(br) >>
((key, value)) ));
value: map_res!(is_not!("]"), str::from_utf8) >> tag!("]") >> many0!(br) >>
((key, value.trim())) ));

named!(frequency<Frequency>,
alt!(tag!("optional") => { |_| Frequency::Optional } |
Expand Down
136 changes: 84 additions & 52 deletions codegen/src/types.rs
Original file line number Diff line number Diff line change
Expand Up @@ -58,24 +58,6 @@ pub enum FieldType {

impl FieldType {

fn is_numeric(&self) -> bool {
match *self {
FieldType::Int32 |
FieldType::Int64 |
FieldType::Uint32 |
FieldType::Uint64 |
FieldType::Sint32 |
FieldType::Sint64 |
FieldType::Fixed64 |
FieldType::Sfixed64 |
FieldType::Double |
FieldType::Fixed32 |
FieldType::Sfixed32 |
FieldType::Float => true,
_ => false,
}
}

fn is_cow(&self) -> bool {
match *self {
FieldType::Bytes | FieldType::String_ => true,
Expand Down Expand Up @@ -142,21 +124,21 @@ impl FieldType {

fn regular_default<'a, 'b>(&'a self, msgs: &'b [Message], enums: &'b [Enumerator]) -> Option<&'b str> {
match *self {
FieldType::Int32 => Some("0"),
FieldType::Sint32 => Some("0"),
FieldType::Int64 => Some("0"),
FieldType::Sint64 => Some("0"),
FieldType::Uint32 => Some("0"),
FieldType::Uint64 => Some("0"),
FieldType::Int32 => Some("0i32"),
FieldType::Sint32 => Some("0i32"),
FieldType::Int64 => Some("0i64"),
FieldType::Sint64 => Some("0i64"),
FieldType::Uint32 => Some("0u32"),
FieldType::Uint64 => Some("0u64"),
FieldType::Bool => Some("false"),
FieldType::Fixed32 => Some("0"),
FieldType::Sfixed32 => Some("0"),
FieldType::Float => Some("0.0"),
FieldType::Fixed64 => Some("0"),
FieldType::Sfixed64 => Some("0"),
FieldType::Double => Some("0.0"),
FieldType::String_ => Some(""),
FieldType::Bytes => Some(""),
FieldType::Fixed32 => Some("0u32"),
FieldType::Sfixed32 => Some("0i32"),
FieldType::Float => Some("0f32"),
FieldType::Fixed64 => Some("0u64"),
FieldType::Sfixed64 => Some("0i64"),
FieldType::Double => Some("0f64"),
FieldType::String_ => Some("Cow::Borrowed(\"\")"),
FieldType::Bytes => Some("Cow::Borrowed(b\"\")"),
FieldType::Enum(_) => self
.find_enum(msgs, enums)
.and_then(|e| e.fields.iter().find(|&&(_, i)| i == 0))
Expand Down Expand Up @@ -374,6 +356,33 @@ impl Field {
}
}

fn sanitize_default(&mut self, msgs: &[Message], enums: &[Enumerator]) {
if let Some(ref mut d) = self.default {
*d = match &*self.typ.rust_type(msgs, enums) {
"u32" => format!("{}u32", *d),
"u64" => format!("{}u64", *d),
"i32" => format!("{}i32", *d),
"i64" => format!("{}i64", *d),
"f32" => match &*d.to_lowercase() {
"inf" => "::std::f32::INFINITY".to_string(),
"-inf" => "::std::f32::NEG_INFINITY".to_string(),
"nan" => "::std::f32::NAN".to_string(),
_ => format!("{}f32", *d),
},
"f64" => match &*d.to_lowercase() {
"inf" => "::std::f64::INFINITY".to_string(),
"-inf" => "::std::f64::NEG_INFINITY".to_string(),
"nan" => "::std::f64::NAN".to_string(),
_ => format!("{}f64", *d),
},
"Cow<'a, str>" => format!("Cow::Borrowed({})", d),
"Cow<'a, [u8]>" => format!("Cow::Borrowed(b{})", d),
"bool" => format!("{}", d.parse::<bool>().unwrap()),
e => format!("{}::{}", e, d), // enum, as message and map do not have defaults
}
}
}

fn has_regular_default(&self, msgs: &[Message], enums: &[Enumerator]) -> bool {
self.default.is_none()
|| self.default.as_ref().map(|d| &**d) == self.typ.regular_default(msgs, enums)
Expand Down Expand Up @@ -452,7 +461,7 @@ impl Field {
}
}
Some(d) => {
write!(w, "if self.{} == {} {{ 0 }} else {{ {} + {} }}",
writeln!(w, "if self.{} == {} {{ 0 }} else {{ {} + {} }}",
self.name, d, tag_size, self.typ.get_size(&format!("&self.{}", self.name)))?;
}
}
Expand Down Expand Up @@ -760,6 +769,41 @@ impl Message {
m.set_enums(msgs);
}
}

fn set_map_required(&mut self) {
for f in self.fields.iter_mut()
.chain(self.oneofs.iter_mut().flat_map(|o| o.fields.iter_mut()))
{
if let FieldType::Map(_) = f.typ {
f.frequency = Frequency::Required;
}
}
for m in &mut self.messages {
m.set_map_required();
}
}

fn set_repeated_as_packed(&mut self) {
for f in self.fields.iter_mut()
.chain(self.oneofs.iter_mut().flat_map(|o| o.fields.iter_mut()))
{
if f.packed.is_none() {
if let Frequency::Repeated = f.frequency {
f.packed = Some(true);
}
}
}
}

fn sanitize_defaults(&mut self, msgs: &[Message], enums: &[Enumerator]) {
for f in self.fields.iter_mut()
.chain(self.oneofs.iter_mut().flat_map(|o| o.fields.iter_mut())) {
f.sanitize_default(msgs, enums);
}
for m in &mut self.messages {
m.sanitize_defaults(msgs, enums);
}
}
}

#[derive(Debug, Clone)]
Expand Down Expand Up @@ -1020,31 +1064,19 @@ impl FileDescriptor {
fn set_defaults(&mut self) {
// set map fields as required (they are equivalent to repeated message)
for m in &mut self.messages {
for f in m.fields.iter_mut()
.chain(m.oneofs.iter_mut().flat_map(|o| o.fields.iter_mut()))
{
if let FieldType::Map(_) = f.typ {
f.frequency = Frequency::Required;
}
}
m.set_map_required();
}
// if proto3, then changes several defaults
if let Syntax::Proto3 = self.syntax {
for m in &mut self.messages {
for f in m.fields.iter_mut()
.chain(m.oneofs.iter_mut().flat_map(|o| o.fields.iter_mut()))
{
if f.packed.is_none() {
if let Frequency::Repeated = f.frequency {
f.packed = Some(true);
}
}
if f.default.is_none() && f.typ.is_numeric() {
f.default = Some("0".to_string());
}
}
m.set_repeated_as_packed();
}
}
// this is very inefficient but we don't care ...
let msgs = self.messages.clone();
for m in &mut self.messages {
m.sanitize_defaults(&msgs, &self.enums);
}
}

fn set_enums(&mut self) {
Expand Down

0 comments on commit 0716e7b

Please sign in to comment.