Exemple #1
0
    def _generate_struct_builder(self, struct):
        fields = [
            "%s %s" %
            (fmt_var(field.name),
             fmt_type(field.data_type, struct.namespace, use_interface=True))
            for field in struct.all_required_fields
        ]
        self.emit('// New{0} returns a new {0} instance'.format(struct.name))
        signature = "func New{0}({1}) *{0}".format(struct.name,
                                                   ', '.join(fields))
        with self.block(signature):
            self.emit('s := new({0})'.format(struct.name))
            for field in struct.all_required_fields:
                field_name = fmt_var(field.name)
                self.emit("s.{0} = {0}".format(field_name))

            for field in struct.all_optional_fields:
                if field.has_default:
                    if is_primitive_type(field.data_type):
                        default = field.default
                        if is_boolean_type(field.data_type):
                            default = str(default).lower()
                        self.emit('s.{0} = {1}'.format(fmt_var(field.name),
                                                       default))
                    elif is_union_type(field.data_type):
                        self.emit('s.%s = &%s{Tagged:dropbox.Tagged{"%s"}}' %
                                  (fmt_var(field.name),
                                   fmt_type(field.data_type,
                                            struct.namespace).lstrip('*'),
                                   field.default.tag_name))
            self.emit('return s')
        self.emit()
Exemple #2
0
def fmt_type(data_type, namespace=None, use_interface=False, raw=False):
    data_type, nullable = unwrap_nullable(data_type)
    if is_list_type(data_type):
        if raw and is_primitive_type(data_type.data_type):
            return "json.RawMessage"
        return '[]%s' % fmt_type(data_type.data_type, namespace, use_interface,
                                 raw)
    if raw:
        return "json.RawMessage"
    type_name = data_type.name
    if use_interface and _needs_base_type(data_type):
        type_name = 'Is' + type_name
    if is_composite_type(data_type) and namespace is not None and \
            namespace.name != data_type.namespace.name:
        type_name = data_type.namespace.name + '.' + type_name
    if use_interface and _needs_base_type(data_type):
        return _type_table.get(data_type.__class__, type_name)
    else:
        return _type_table.get(data_type.__class__, '*' + type_name)
Exemple #3
0
    def _generate_union_helper(self, u):
        name = u.name
        namespace = u.namespace
        # Unions can be inherited, but don't need to be polymorphic.
        # So let's flatten out all the inherited fields.
        fields = u.all_fields
        if is_struct_type(u) and u.has_enumerated_subtypes():
            name = fmt_var(name, export=False) + 'Union'
            fields = u.get_enumerated_subtypes()

        with self.block('type %s struct' % name):
            self.emit('dropbox.Tagged')
            for field in fields:
                if is_void_type(field.data_type):
                    continue
                self._generate_field(field,
                                     union_field=True,
                                     namespace=namespace)
        self.emit()
        self.emit('// Valid tag values for %s' % fmt_var(u.name))
        with self.block('const', delim=('(', ')')):
            for field in fields:
                self.emit('%s%s = "%s"' %
                          (fmt_var(u.name), fmt_var(field.name), field.name))
        self.emit()

        num_void_fields = sum([is_void_type(f.data_type) for f in fields])
        # Simple structure, no need in UnmarshalJSON
        if len(fields) == num_void_fields:
            return

        self.emit('// UnmarshalJSON deserializes into a %s instance' % name)
        with self.block('func (u *%s) UnmarshalJSON(body []byte) error' %
                        name):
            with self.block('type wrap struct'):
                self.emit('dropbox.Tagged')
                for field in fields:
                    if is_void_type(field.data_type) or \
                            is_primitive_type(field.data_type):
                        continue
                    self._generate_field(field,
                                         union_field=True,
                                         namespace=namespace,
                                         raw=True)
            self.emit('var w wrap')
            self.emit('var err error')
            with self.block('if err = json.Unmarshal(body, &w); err != nil'):
                self.emit('return err')
            self.emit('u.Tag = w.Tag')
            with self.block('switch u.Tag'):
                for field in fields:
                    if is_void_type(field.data_type):
                        continue
                    field_name = fmt_var(field.name)
                    with self.block('case "%s":' % field.name,
                                    delim=(None, None)):
                        if is_union_type(field.data_type):
                            self.emit(
                                'err = json.Unmarshal(w.{0}, &u.{0})'.format(
                                    field_name))
                        elif is_struct_type(field.data_type) and \
                            field.data_type.has_enumerated_subtypes():
                            self.emit(
                                "u.{0}, err = Is{1}FromJSON(body)".format(
                                    field_name, field.data_type.name))
                        else:
                            self.emit(
                                'err = json.Unmarshal(body, &u.{0})'.format(
                                    field_name))
                    with self.block("if err != nil"):
                        self.emit("return err")
            self.emit('return nil')
        self.emit()
 def _impl_serde_for_union(self, union):
     type_name = self.enum_name(union)
     with self._impl_deserialize(type_name):
         self.emit(u'// union deserializer')
         self.emit(u'use serde::de::{self, MapAccess, Visitor};')
         self.emit(u'struct EnumVisitor;')
         with self.block(u'impl<\'de> Visitor<\'de> for EnumVisitor'):
             self.emit(u'type Value = {};'.format(type_name))
             with self.emit_rust_function_def(
                     u'expecting',
                 [u'&self', u'f: &mut ::std::fmt::Formatter<\'_>'],
                     u'::std::fmt::Result'):
                 self.emit(u'f.write_str("a {} structure")'.format(
                     union.name))
             with self.emit_rust_function_def(
                     u'visit_map<V: MapAccess<\'de>>',
                 [u'self', u'mut map: V'],
                     u'Result<Self::Value, V::Error>'):
                 with self.block(u'let tag: &str = match map.next_key()?',
                                 after=';'):
                     self.emit(u'Some(".tag") => map.next_value()?,')
                     self.emit(
                         u'_ => return Err(de::Error::missing_field(".tag"))'
                     )
                 with self.block(u'match tag'):
                     for field in union.all_fields:
                         if field.catch_all:
                             # Handle the 'Other' variant at the end.
                             continue
                         variant_name = self.enum_variant_name(field)
                         ultimate_type = ir.unwrap(field.data_type)[0]
                         if isinstance(field.data_type, ir.Void):
                             with self.block(u'"{}" =>'.format(field.name)):
                                 self.emit(
                                     u'crate::eat_json_fields(&mut map)?;')
                                 self.emit(u'Ok({}::{})'.format(
                                     type_name, variant_name))
                         elif isinstance(ultimate_type, ir.Struct) \
                                 and not ultimate_type.has_enumerated_subtypes():
                             if isinstance(
                                     ir.unwrap_aliases(field.data_type)[0],
                                     ir.Nullable):
                                 # A nullable here means we might have more fields that can be
                                 # deserialized into the inner type, or we might have nothing,
                                 # meaning None.
                                 if not ultimate_type.all_required_fields:
                                     raise RuntimeError(
                                         '{}.{}: an optional struct with no'
                                         ' required fields is ambiguous'.
                                         format(union.name, field.name))
                                 self.emit(
                                     u'"{}" => Ok({}::{}({}::internal_deserialize_opt('
                                     u'map, true)?)),'.format(
                                         field.name, type_name,
                                         variant_name,
                                         self._rust_type(ultimate_type)))
                             else:
                                 self.emit(
                                     u'"{}" => Ok({}::{}({}::internal_deserialize(map)?)),'
                                     .format(
                                         field.name, type_name,
                                         variant_name,
                                         self._rust_type(field.data_type)))
                         else:
                             with self.block(u'"{}" =>'.format(field.name)):
                                 with self.block(u'match map.next_key()?'):
                                     self.emit(
                                         u'Some("{}") => Ok({}::{}(map.next_value()?)),'
                                         .format(field.name, type_name,
                                                 variant_name))
                                     if isinstance(
                                             ir.unwrap_aliases(
                                                 field.data_type)[0],
                                             ir.Nullable):
                                         # if it's null, the field can be omitted entirely
                                         self.emit(
                                             u'None => Ok({}::{}(None)),'.
                                             format(type_name,
                                                    variant_name))
                                     else:
                                         self.emit(
                                             u'None => Err('
                                             u'de::Error::missing_field("{}")),'
                                             .format(field.name))
                                     self.emit(
                                         u'_ => Err(de::Error::unknown_field('
                                         u'tag, VARIANTS))')
                     if not union.closed:
                         with self.block(u'_ =>'):
                             self.emit(
                                 u'crate::eat_json_fields(&mut map)?;')
                             self.emit(u'Ok({}::Other)'.format(type_name))
                     else:
                         self.emit(
                             u'_ => Err(de::Error::unknown_variant(tag, VARIANTS))'
                         )
         self.generate_multiline_list(
             list(u'"{}"'.format(field.name) for field in union.all_fields),
             before='const VARIANTS: &[&str] = &',
             after=';',
             delim=(u'[', u']'),
         )
         self.emit(
             u'deserializer.deserialize_struct("{}", VARIANTS, EnumVisitor)'
             .format(union.name))
     self.emit()
     with self._impl_serialize(type_name):
         self.emit(u'// union serializer')
         if len(union.all_fields) == 1 and union.all_fields[0].catch_all:
             # special case: an open union with no variants defined.
             self.emit(u'#![allow(unused_variables)]')
             self.emit(
                 u'Err(::serde::ser::Error::custom("cannot serialize an open union with '
                 u'no defined variants"))')
         else:
             self.emit(u'use serde::ser::SerializeStruct;')
             with self.block(u'match *self'):
                 for field in union.all_fields:
                     if field.catch_all:
                         # Handle the 'Other' variant at the end.
                         continue
                     variant_name = self.enum_variant_name(field)
                     if isinstance(field.data_type, ir.Void):
                         with self.block(u'{}::{} =>'.format(
                                 type_name, variant_name)):
                             self.emit(u'// unit')
                             self.emit(
                                 u'let mut s = serializer.serialize_struct("{}", 1)?;'
                                 .format(union.name))
                             self.emit(u's.serialize_field(".tag", "{}")?;'.
                                       format(field.name))
                             self.emit(u's.end()')
                     else:
                         ultimate_type = ir.unwrap(field.data_type)[0]
                         needs_x = not (isinstance(field.data_type,
                                                   ir.Struct)
                                        and not field.data_type.all_fields)
                         ref_x = 'ref x' if needs_x else '_'
                         with self.block(u'{}::{}({}) =>'.format(
                                 type_name, variant_name, ref_x)):
                             if self.is_enum_type(ultimate_type):
                                 # Inner type is a union or polymorphic struct; need to always
                                 # emit another nesting level.
                                 self.emit(
                                     u'// union or polymporphic struct')
                                 self.emit(
                                     u'let mut s = serializer.serialize_struct("{}", 2)?;'
                                     .format(union.name))
                                 self.emit(
                                     u's.serialize_field(".tag", "{}")?;'.
                                     format(field.name))
                                 self.emit(
                                     u's.serialize_field("{}", x)?;'.format(
                                         field.name))
                                 self.emit(u's.end()')
                             elif isinstance(
                                     ir.unwrap_aliases(field.data_type)[0],
                                     ir.Nullable):
                                 self.emit(
                                     u'// nullable (struct or primitive)')
                                 # If it's nullable and the value is None, just emit the tag and
                                 # nothing else, otherwise emit the fields directly at the same
                                 # level.
                                 num_fields = 1 if ir.is_primitive_type(ultimate_type) \
                                     else len(ultimate_type.all_fields) + 1
                                 self.emit(
                                     u'let n = if x.is_some() {{ {} }} else {{ 1 }};'
                                     .format(num_fields + 1))
                                 self.emit(
                                     u'let mut s = serializer.serialize_struct("{}", n)?;'
                                     .format(union.name))
                                 self.emit(
                                     u's.serialize_field(".tag", "{}")?;'.
                                     format(field.name))
                                 with self.block(u'if let Some(ref x) = x'):
                                     if ir.is_primitive_type(ultimate_type):
                                         self.emit(
                                             u's.serialize_field("{}", &x)?;'
                                             .format(field.name))
                                     else:
                                         self.emit(
                                             u'x.internal_serialize::<S>(&mut s)?;'
                                         )
                                 self.emit(u's.end()')
                             elif isinstance(ultimate_type, ir.Struct):
                                 self.emit(u'// struct')
                                 self.emit(
                                     u'let mut s = serializer.serialize_struct("{}", {})?;'
                                     .format(
                                         union.name,
                                         len(ultimate_type.all_fields) + 1))
                                 self.emit(
                                     u's.serialize_field(".tag", "{}")?;'.
                                     format(field.name))
                                 if ultimate_type.all_fields:
                                     self.emit(
                                         u'x.internal_serialize::<S>(&mut s)?;'
                                     )
                                 self.emit(u's.end()')
                             else:
                                 self.emit(u'// primitive')
                                 self.emit(
                                     u'let mut s = serializer.serialize_struct("{}", 2)?;'
                                     .format(union.name))
                                 self.emit(
                                     u's.serialize_field(".tag", "{}")?;'.
                                     format(field.name))
                                 self.emit(
                                     u's.serialize_field("{}", x)?;'.format(
                                         field.name))
                                 self.emit(u's.end()')
                 if not union.closed:
                     self.emit(
                         u'{}::Other => Err(::serde::ser::Error::custom('
                         u'"cannot serialize \'Other\' variant"))'.format(
                             type_name))
     self.emit()
    def _impl_serde_for_struct(self, struct):
        """
        Emit internal_deserialize() and possibly internal_deserialize_opt().
        internal_deserialize[_opt] takes a map and deserializes it into the struct. It reads the
        fields in whatever order; missing fields will be given their default value, or an error
        returned if they have no default. Errors will also be raised if a field is present more
        than once.
        The _opt deserializer returns a None if it reads exactly zero map keys, and is used for
        cases where the JSON has a tag, but omits all the fields to signify a null value. It is
        only emitted for types which have at least one required field, because if all fields are
        optional, there's no way to differentiate between a null value and one where all fields
        are default.
        """

        type_name = self.struct_name(struct)
        field_list_name = u'{}_FIELDS'.format(fmt_shouting_snake(struct.name))
        self.generate_multiline_list(
            list(u'"{}"'.format(field.name) for field in struct.all_fields),
            before='const {}: &[&str] = &'.format(field_list_name),
            after=';',
            delim=(u'[', u']'))
        # Only emit the _opt deserializer if there are required fields.
        optional = len(struct.all_required_fields) > 0
        with self._impl_struct(struct):
            if optional:
                # Convenience wrapper around _opt for the more common case where the struct is
                # NOT optional.
                with self.emit_rust_function_def(
                        u'internal_deserialize<\'de, V: ::serde::de::MapAccess<\'de>>',
                    [u'map: V'],
                        u'Result<{}, V::Error>'.format(type_name),
                        access=u'pub(crate)'):
                    self.emit(u'Self::internal_deserialize_opt(map, false)'
                              u'.map(Option::unwrap)')
                self.emit()
            else:
                self.emit(u'// no _opt deserializer')
            with self.emit_rust_function_def(
                (u'internal_deserialize_opt'
                 if optional else u'internal_deserialize') +
                    u'<\'de, V: ::serde::de::MapAccess<\'de>>',
                [u'mut map: V'] + ([u'optional: bool'] if optional else []),
                (u'Result<Option<{}>, V::Error>'
                 if optional else u'Result<{}, V::Error>').format(type_name),
                    access=u'pub(crate)'):
                for field in struct.all_fields:
                    self.emit(u'let mut field_{} = None;'.format(
                        self.field_name(field)))
                if optional:
                    self.emit(u'let mut nothing = true;')
                with self.block(
                        u'while let Some(key) = map.next_key::<&str>()?'):
                    if optional:
                        self.emit(u'nothing = false;')
                    with self.block(u'match key'):
                        for field in struct.all_fields:
                            field_name = self.field_name(field)
                            with self.block(u'"{}" =>'.format(field.name)):
                                with self.block(
                                        u'if field_{}.is_some()'.format(
                                            field_name)):
                                    self.emit(
                                        u'return Err(::serde::de::Error::duplicate_field('
                                        u'"{}"));'.format(field.name))
                                self.emit(
                                    u'field_{} = Some(map.next_value()?);'.
                                    format(field_name))
                        with self.block(u'_ =>'):
                            self.emit(u'// unknown field allowed and ignored')
                            self.emit(
                                u'map.next_value::<::serde_json::Value>()?;')
                if optional:
                    with self.block(u'if optional && nothing'):
                        self.emit(u'return Ok(None);')
                with self.block(u'let result = {}'.format(type_name),
                                delim=(u'{', u'};')):
                    for field in struct.all_fields:
                        field_name = self.field_name(field)
                        if isinstance(field.data_type, ir.Nullable):
                            self.emit(u'{}: field_{},'.format(
                                field_name, field_name))
                        elif field.has_default:
                            if isinstance(field.data_type, ir.String) \
                                    and not field.default:
                                self.emit(
                                    u'{}: field_{}.unwrap_or_else(String::new),'
                                    .format(field_name, field_name))
                            elif ir.is_primitive_type(
                                    ir.unwrap_aliases(field.data_type)[0]):
                                self.emit(
                                    u'{}: field_{}.unwrap_or({}),'.format(
                                        field_name, field_name,
                                        self._default_value(field)))
                            else:
                                self.emit(
                                    u'{}: field_{}.unwrap_or_else(|| {}),'.
                                    format(field_name, field_name,
                                           self._default_value(field)))
                        else:
                            self.emit(
                                u'{}: field_{}.ok_or_else(|| '
                                u'::serde::de::Error::missing_field("{}"))?,'.
                                format(field_name, field_name, field.name))
                if optional:
                    self.emit(u'Ok(Some(result))')
                else:
                    self.emit(u'Ok(result)')
            if struct.all_fields:
                self.emit()
                with self.emit_rust_function_def(
                        u'internal_serialize<S: ::serde::ser::Serializer>',
                    [u'&self', u's: &mut S::SerializeStruct'],
                        u'Result<(), S::Error>',
                        access=u'pub(crate)'):
                    self.emit(u'use serde::ser::SerializeStruct;')
                    self.generate_multiline_list(list(
                        u's.serialize_field("{}", &self.{})'.format(
                            field.name, self.field_name(field))
                        for field in struct.all_fields),
                                                 delim=(u'', u''),
                                                 sep='?;',
                                                 skip_last_sep=True)
        self.emit()
        with self._impl_deserialize(self.struct_name(struct)):
            self.emit(u'// struct deserializer')
            self.emit(u'use serde::de::{MapAccess, Visitor};')
            self.emit(u'struct StructVisitor;')
            with self.block(u'impl<\'de> Visitor<\'de> for StructVisitor'):
                self.emit(u'type Value = {};'.format(type_name))
                with self.emit_rust_function_def(
                        u'expecting',
                    [u'&self', u'f: &mut ::std::fmt::Formatter<\'_>'],
                        u'::std::fmt::Result'):
                    self.emit(u'f.write_str("a {} struct")'.format(
                        struct.name))
                with self.emit_rust_function_def(
                        u'visit_map<V: MapAccess<\'de>>', [u'self', u'map: V'],
                        u'Result<Self::Value, V::Error>'):
                    self.emit(
                        u'{}::internal_deserialize(map)'.format(type_name))
            self.emit(
                u'deserializer.deserialize_struct("{}", {}, StructVisitor)'.
                format(struct.name, field_list_name))
        self.emit()
        with self._impl_serialize(type_name):
            self.emit(u'// struct serializer')
            self.emit(u'use serde::ser::SerializeStruct;')
            if not struct.all_fields:
                self.emit(
                    u'serializer.serialize_struct("{}", 0)?.end()'.format(
                        struct.name))
            else:
                self.emit(
                    u'let mut s = serializer.serialize_struct("{}", {})?;'.
                    format(struct.name, len(struct.all_fields)))
                self.emit(u'self.internal_serialize::<S>(&mut s)?;')
                self.emit(u's.end()')
        self.emit()