blob: e9b017d92dff4aacc49500e8a4891902c86d462a [file] [log] [blame]
use std::cmp;
use inside::protobuf_crate_path;
use message::RustTypeMessage;
use protobuf::descriptor::*;
use protobuf_name::ProtobufAbsolutePath;
use rust_name::RustIdent;
use scope::RootScope;
use scope::WithScope;
use strx::capitalize;
use Customize;
use super::well_known_types::is_well_known_type_full;
// Represent subset of rust types used in generated code
#[derive(Debug, Clone, PartialEq, Eq)]
pub(crate) enum RustType {
// integer: signed?, size in bits
Int(bool, u32),
// param is size in bits
Float(u32),
Bool,
Vec(Box<RustType>),
HashMap(Box<RustType>, Box<RustType>),
String,
// [T], not &[T]
Slice(Box<RustType>),
// str, not &str
Str,
Option(Box<RustType>),
SingularField(Box<RustType>),
SingularPtrField(Box<RustType>),
RepeatedField(Box<RustType>),
// Box<T>
Uniq(Box<RustType>),
// &T
Ref(Box<RustType>),
// protobuf message
Message(RustTypeMessage),
// protobuf enum, not any enum
Enum(String, RustIdent),
// oneof enum
Oneof(String),
// bytes::Bytes
Bytes,
// chars::Chars
Chars,
// group
Group,
}
impl RustType {
#[inline]
pub(crate) fn to_code(&self, customize: &Customize) -> String {
match *self {
RustType::Int(true, bits) => format!("i{}", bits),
RustType::Int(false, bits) => format!("u{}", bits),
RustType::Float(bits) => format!("f{}", bits),
RustType::Bool => format!("bool"),
RustType::Vec(ref param) => format!("::std::vec::Vec<{}>", param.to_code(customize)),
RustType::HashMap(ref key, ref value) => format!(
"::std::collections::HashMap<{}, {}>",
key.to_code(customize),
value.to_code(customize)
),
RustType::String => format!("::std::string::String"),
RustType::Slice(ref param) => format!("[{}]", param.to_code(customize)),
RustType::Str => format!("str"),
RustType::Option(ref param) => {
format!("::std::option::Option<{}>", param.to_code(customize))
}
RustType::SingularField(ref param) => format!(
"{}::SingularField<{}>",
protobuf_crate_path(customize),
param.to_code(customize)
),
RustType::SingularPtrField(ref param) => format!(
"{}::SingularPtrField<{}>",
protobuf_crate_path(customize),
param.to_code(customize)
),
RustType::RepeatedField(ref param) => format!(
"{}::RepeatedField<{}>",
protobuf_crate_path(customize),
param.to_code(customize)
),
RustType::Uniq(ref param) => format!("::std::boxed::Box<{}>", param.to_code(customize)),
RustType::Ref(ref param) => format!("&{}", param.to_code(customize)),
RustType::Message(ref name) => format!("{}", name),
RustType::Enum(ref name, _) | RustType::Oneof(ref name) => format!("{}", name),
RustType::Group => format!("<group>"),
RustType::Bytes => format!("::bytes::Bytes"),
RustType::Chars => format!("{}::Chars", protobuf_crate_path(customize)),
}
}
}
impl RustType {
pub fn u8() -> RustType {
RustType::Int(false, 8)
}
/// Type is rust primitive?
pub fn is_primitive(&self) -> bool {
match *self {
RustType::Int(..) | RustType::Float(..) | RustType::Bool => true,
_ => false,
}
}
pub fn is_u8(&self) -> bool {
match *self {
RustType::Int(false, 8) => true,
_ => false,
}
}
pub fn is_copy(&self) -> bool {
if self.is_primitive() {
true
} else if let RustType::Enum(..) = *self {
true
} else {
false
}
}
fn is_str(&self) -> bool {
match *self {
RustType::Str => true,
_ => false,
}
}
fn is_string(&self) -> bool {
match *self {
RustType::String => true,
_ => false,
}
}
fn is_slice(&self) -> Option<&RustType> {
match *self {
RustType::Slice(ref v) => Some(&**v),
_ => None,
}
}
fn is_slice_u8(&self) -> bool {
match self.is_slice() {
Some(t) => t.is_u8(),
None => false,
}
}
fn is_message(&self) -> bool {
match *self {
RustType::Message(..) => true,
_ => false,
}
}
fn is_enum(&self) -> bool {
match *self {
RustType::Enum(..) => true,
_ => false,
}
}
pub fn is_ref(&self) -> bool {
match *self {
RustType::Ref(..) => true,
_ => false,
}
}
// default value for type
pub fn default_value(&self, customize: &Customize) -> String {
match *self {
RustType::Ref(ref t) if t.is_str() => "\"\"".to_string(),
RustType::Ref(ref t) if t.is_slice().is_some() => "&[]".to_string(),
RustType::Int(..) => "0".to_string(),
RustType::Float(..) => "0.".to_string(),
RustType::Bool => "false".to_string(),
RustType::Vec(..) => "::std::vec::Vec::new()".to_string(),
RustType::HashMap(..) => "::std::collections::HashMap::new()".to_string(),
RustType::String => "::std::string::String::new()".to_string(),
RustType::Bytes => "::bytes::Bytes::new()".to_string(),
RustType::Chars => format!("{}::Chars::new()", protobuf_crate_path(customize)),
RustType::Option(..) => "::std::option::Option::None".to_string(),
RustType::SingularField(..) => {
format!("{}::SingularField::none()", protobuf_crate_path(customize))
}
RustType::SingularPtrField(..) => format!(
"{}::SingularPtrField::none()",
protobuf_crate_path(customize)
),
RustType::RepeatedField(..) => {
format!("{}::RepeatedField::new()", protobuf_crate_path(customize))
}
RustType::Message(ref name) => format!("{}::new()", name),
RustType::Ref(ref m) if m.is_message() => match **m {
RustType::Message(ref name) => name.default_instance(customize),
_ => unreachable!(),
},
// Note: default value of enum type may not be equal to default value of field
RustType::Enum(ref name, ref default) => format!("{}::{}", name, default),
_ => panic!("cannot create default value for: {:?}", *self),
}
}
pub fn default_value_typed(self, customize: &Customize) -> RustValueTyped {
RustValueTyped {
value: self.default_value(customize),
rust_type: self,
}
}
/// Emit a code to clear a variable `v`
pub fn clear(&self, v: &str, customize: &Customize) -> String {
match *self {
RustType::Option(..) => format!("{} = ::std::option::Option::None", v),
RustType::Vec(..)
| RustType::Bytes
| RustType::String
| RustType::RepeatedField(..)
| RustType::SingularField(..)
| RustType::SingularPtrField(..)
| RustType::HashMap(..) => format!("{}.clear()", v),
RustType::Chars => format!(
"{}::Clear::clear(&mut {})",
protobuf_crate_path(customize),
v
),
RustType::Bool | RustType::Float(..) | RustType::Int(..) | RustType::Enum(..) => {
format!("{} = {}", v, self.default_value(customize))
}
ref ty => panic!("cannot clear type: {:?}", ty),
}
}
// wrap value in storage type
pub fn wrap_value(&self, value: &str, customize: &Customize) -> String {
match *self {
RustType::Option(..) => format!("::std::option::Option::Some({})", value),
RustType::SingularField(..) => format!(
"{}::SingularField::some({})",
protobuf_crate_path(customize),
value
),
RustType::SingularPtrField(..) => format!(
"{}::SingularPtrField::some({})",
protobuf_crate_path(customize),
value
),
_ => panic!("not a wrapper type: {:?}", *self),
}
}
// expression to convert `v` of type `self` to type `target`
pub fn into_target(&self, target: &RustType, v: &str, customize: &Customize) -> String {
self.try_into_target(target, v, customize)
.expect(&format!("failed to convert {:?} into {:?}", self, target))
}
fn try_into_target(
&self,
target: &RustType,
v: &str,
customize: &Customize,
) -> Result<String, ()> {
match (self, target) {
(x, y) if x == y => return Ok(format!("{}", v)),
(&RustType::Ref(ref x), y) if **x == *y => return Ok(format!("*{}", v)),
(x, &RustType::Uniq(ref y)) if *x == **y => {
return Ok(format!("::std::boxed::Box::new({})", v))
}
(&RustType::Uniq(ref x), y) if **x == *y => return Ok(format!("*{}", v)),
(&RustType::String, &RustType::Ref(ref t)) if **t == RustType::Str => {
return Ok(format!("&{}", v))
}
(&RustType::Chars, &RustType::Ref(ref t)) if **t == RustType::Str => {
return Ok(format!("&{}", v))
}
(&RustType::Ref(ref t1), &RustType::Ref(ref t2)) if t1.is_string() && t2.is_str() => {
return Ok(format!("&{}", v))
}
(&RustType::Ref(ref t1), &RustType::String)
if match **t1 {
RustType::Str => true,
_ => false,
} =>
{
return Ok(format!("{}.to_owned()", v))
}
(&RustType::Ref(ref t1), &RustType::Chars)
if match **t1 {
RustType::Str => true,
_ => false,
// TODO: from_static
} =>
{
return Ok(format!(
"<{}::Chars as ::std::convert::From<_>>::from({}.to_owned())",
protobuf_crate_path(customize),
v
))
}
(&RustType::Ref(ref t1), &RustType::Vec(ref t2))
if match (&**t1, &**t2) {
(&RustType::Slice(ref x), ref y) => **x == **y,
_ => false,
} =>
{
return Ok(format!("{}.to_vec()", v))
}
(&RustType::Ref(ref t1), &RustType::Bytes) if t1.is_slice_u8() => {
return Ok(format!(
"<::bytes::Bytes as ::std::convert::From<_>>::from({}.to_vec())",
v
))
}
(&RustType::Vec(ref x), &RustType::Ref(ref t))
if match **t {
RustType::Slice(ref y) => x == y,
_ => false,
} =>
{
return Ok(format!("&{}", v))
}
(&RustType::Bytes, &RustType::Ref(ref t))
if match **t {
RustType::Slice(ref y) => **y == RustType::u8(),
_ => false,
} =>
{
return Ok(format!("&{}", v))
}
(&RustType::Ref(ref t1), &RustType::Ref(ref t2))
if match (&**t1, &**t2) {
(&RustType::Vec(ref x), &RustType::Slice(ref y)) => x == y,
_ => false,
} =>
{
return Ok(format!("&{}", v))
}
(&RustType::Enum(..), &RustType::Int(true, 32)) => {
return Ok(format!(
"{}::ProtobufEnum::value(&{})",
protobuf_crate_path(customize),
v
))
}
(&RustType::Ref(ref t), &RustType::Int(true, 32)) if t.is_enum() => {
return Ok(format!(
"{}::ProtobufEnum::value({})",
protobuf_crate_path(customize),
v
))
}
_ => (),
};
if let &RustType::Ref(ref s) = self {
if let Ok(conv) = s.try_into_target(target, v, customize) {
return Ok(conv);
}
}
Err(())
}
/// Type to view data of this type
pub fn ref_type(&self) -> RustType {
RustType::Ref(Box::new(match self {
&RustType::String | &RustType::Chars => RustType::Str,
&RustType::Vec(ref p) | &RustType::RepeatedField(ref p) => RustType::Slice(p.clone()),
&RustType::Bytes => RustType::Slice(Box::new(RustType::u8())),
&RustType::Message(ref p) => RustType::Message(p.clone()),
x => panic!("no ref type for {:?}", x),
}))
}
pub fn elem_type(&self) -> RustType {
match self {
&RustType::Option(ref ty) => (**ty).clone(),
x => panic!("cannot get elem type of {:?}", x),
}
}
// type of `v` in `for v in xxx`
pub fn iter_elem_type(&self) -> RustType {
match self {
&RustType::Vec(ref ty)
| &RustType::Option(ref ty)
| &RustType::RepeatedField(ref ty)
| &RustType::SingularField(ref ty)
| &RustType::SingularPtrField(ref ty) => RustType::Ref(ty.clone()),
x => panic!("cannot iterate {:?}", x),
}
}
pub fn value(self, value: String) -> RustValueTyped {
RustValueTyped {
value: value,
rust_type: self,
}
}
}
/// Representation of an expression in code generator: text and type
pub(crate) struct RustValueTyped {
pub value: String,
pub rust_type: RustType,
}
impl RustValueTyped {
pub fn into_type(&self, target: RustType, customize: &Customize) -> RustValueTyped {
let target_value = self.rust_type.into_target(&target, &self.value, customize);
RustValueTyped {
value: target_value,
rust_type: target,
}
}
pub fn boxed(self, customize: &Customize) -> RustValueTyped {
self.into_type(RustType::Uniq(Box::new(self.rust_type.clone())), customize)
}
}
// protobuf type name for protobuf base type
pub fn protobuf_name(field_type: FieldDescriptorProto_Type) -> &'static str {
match field_type {
FieldDescriptorProto_Type::TYPE_DOUBLE => "double",
FieldDescriptorProto_Type::TYPE_FLOAT => "float",
FieldDescriptorProto_Type::TYPE_INT32 => "int32",
FieldDescriptorProto_Type::TYPE_INT64 => "int64",
FieldDescriptorProto_Type::TYPE_UINT32 => "uint32",
FieldDescriptorProto_Type::TYPE_UINT64 => "uint64",
FieldDescriptorProto_Type::TYPE_SINT32 => "sint32",
FieldDescriptorProto_Type::TYPE_SINT64 => "sint64",
FieldDescriptorProto_Type::TYPE_FIXED32 => "fixed32",
FieldDescriptorProto_Type::TYPE_FIXED64 => "fixed64",
FieldDescriptorProto_Type::TYPE_SFIXED32 => "sfixed32",
FieldDescriptorProto_Type::TYPE_SFIXED64 => "sfixed64",
FieldDescriptorProto_Type::TYPE_BOOL => "bool",
FieldDescriptorProto_Type::TYPE_STRING => "string",
FieldDescriptorProto_Type::TYPE_BYTES => "bytes",
FieldDescriptorProto_Type::TYPE_ENUM => "enum",
FieldDescriptorProto_Type::TYPE_MESSAGE => "message",
FieldDescriptorProto_Type::TYPE_GROUP => "group",
}
}
// rust type for protobuf base type
pub(crate) fn rust_name(field_type: FieldDescriptorProto_Type) -> RustType {
match field_type {
FieldDescriptorProto_Type::TYPE_DOUBLE => RustType::Float(64),
FieldDescriptorProto_Type::TYPE_FLOAT => RustType::Float(32),
FieldDescriptorProto_Type::TYPE_INT32 => RustType::Int(true, 32),
FieldDescriptorProto_Type::TYPE_INT64 => RustType::Int(true, 64),
FieldDescriptorProto_Type::TYPE_UINT32 => RustType::Int(false, 32),
FieldDescriptorProto_Type::TYPE_UINT64 => RustType::Int(false, 64),
FieldDescriptorProto_Type::TYPE_SINT32 => RustType::Int(true, 32),
FieldDescriptorProto_Type::TYPE_SINT64 => RustType::Int(true, 64),
FieldDescriptorProto_Type::TYPE_FIXED32 => RustType::Int(false, 32),
FieldDescriptorProto_Type::TYPE_FIXED64 => RustType::Int(false, 64),
FieldDescriptorProto_Type::TYPE_SFIXED32 => RustType::Int(true, 32),
FieldDescriptorProto_Type::TYPE_SFIXED64 => RustType::Int(true, 64),
FieldDescriptorProto_Type::TYPE_BOOL => RustType::Bool,
FieldDescriptorProto_Type::TYPE_STRING => RustType::String,
FieldDescriptorProto_Type::TYPE_BYTES => RustType::Vec(Box::new(RustType::Int(false, 8))),
FieldDescriptorProto_Type::TYPE_ENUM
| FieldDescriptorProto_Type::TYPE_GROUP
| FieldDescriptorProto_Type::TYPE_MESSAGE => {
panic!("there is no rust name for {:?}", field_type)
}
}
}
fn file_last_component(file: &str) -> &str {
let bs = file.rfind('\\').map(|i| i + 1).unwrap_or(0);
let fs = file.rfind('/').map(|i| i + 1).unwrap_or(0);
&file[cmp::max(fs, bs)..]
}
#[cfg(test)]
#[test]
fn test_file_last_component() {
assert_eq!("ab.proto", file_last_component("ab.proto"));
assert_eq!("ab.proto", file_last_component("xx/ab.proto"));
assert_eq!("ab.proto", file_last_component("xx\\ab.proto"));
assert_eq!("ab.proto", file_last_component("yy\\xx\\ab.proto"));
}
fn is_descriptor_proto(file: &FileDescriptorProto) -> bool {
file.get_package() == "google.protobuf"
&& file_last_component(file.get_name()) == "descriptor.proto"
}
pub(crate) fn type_name_to_rust_relative(
type_name: &ProtobufAbsolutePath,
file: &FileDescriptorProto,
subm: bool,
root_scope: &RootScope,
customize: &Customize,
) -> String {
let message_or_enum = root_scope.find_message_or_enum(type_name);
if message_or_enum.get_scope().get_file_descriptor().get_name() == file.get_name() {
// field type is a message or enum declared in the same file
if subm {
format!("super::{}", message_or_enum.rust_name())
} else {
format!("{}", message_or_enum.rust_name())
}
} else if let Some(name) = is_well_known_type_full(&type_name.path) {
// Well-known types are included in rust-protobuf library
// https://developers.google.com/protocol-buffers/docs/reference/google.protobuf
format!(
"{}::well_known_types::{}",
protobuf_crate_path(customize),
name
)
} else if is_descriptor_proto(message_or_enum.get_file_descriptor()) {
// Messages defined in descriptor.proto
format!(
"{}::descriptor::{}",
protobuf_crate_path(customize),
message_or_enum.name_to_package()
)
} else {
if subm {
format!("super::super::{}", message_or_enum.rust_fq_name())
} else {
format!("super::{}", message_or_enum.rust_fq_name())
}
}
}
#[derive(Copy, Clone, Debug, PartialEq, Eq)]
pub enum PrimitiveTypeVariant {
Default,
Carllerche,
}
pub enum _CarllercheBytesType {
Bytes,
Chars,
}
// ProtobufType trait name
pub enum ProtobufTypeGen {
Primitive(FieldDescriptorProto_Type, PrimitiveTypeVariant),
Message(String),
Enum(String),
}
impl ProtobufTypeGen {
pub fn rust_type(&self, customize: &Customize) -> String {
match self {
&ProtobufTypeGen::Primitive(t, PrimitiveTypeVariant::Default) => format!(
"{}::types::ProtobufType{}",
protobuf_crate_path(customize),
capitalize(protobuf_name(t))
),
&ProtobufTypeGen::Primitive(
FieldDescriptorProto_Type::TYPE_BYTES,
PrimitiveTypeVariant::Carllerche,
) => format!(
"{}::types::ProtobufTypeCarllercheBytes",
protobuf_crate_path(customize)
),
&ProtobufTypeGen::Primitive(
FieldDescriptorProto_Type::TYPE_STRING,
PrimitiveTypeVariant::Carllerche,
) => format!(
"{}::types::ProtobufTypeCarllercheChars",
protobuf_crate_path(customize)
),
&ProtobufTypeGen::Primitive(.., PrimitiveTypeVariant::Carllerche) => unreachable!(),
&ProtobufTypeGen::Message(ref name) => format!(
"{}::types::ProtobufTypeMessage<{}>",
protobuf_crate_path(customize),
name
),
&ProtobufTypeGen::Enum(ref name) => format!(
"{}::types::ProtobufTypeEnum<{}>",
protobuf_crate_path(customize),
name
),
}
}
}