2022-07-23 00:38:00 +00:00
|
|
|
use crate::{parser, types};
|
|
|
|
use case::CaseExt;
|
|
|
|
use proc_macro2::{Ident, TokenStream};
|
|
|
|
use quote::quote;
|
|
|
|
|
2022-07-24 00:02:57 +00:00
|
|
|
use super::{api_group::GroupGeneration, util};
|
2022-07-23 00:38:00 +00:00
|
|
|
|
2022-07-24 00:02:57 +00:00
|
|
|
pub fn generate_groups(groups: Vec<parser::ApiGroup>, resp_derives: Vec<String>) -> TokenStream {
|
2022-07-23 00:38:00 +00:00
|
|
|
let gr = groups
|
2022-07-24 00:02:57 +00:00
|
|
|
.into_iter()
|
2022-07-23 00:38:00 +00:00
|
|
|
// implemented manually
|
|
|
|
.filter(|group| group.name != "authentication")
|
2022-07-24 00:02:57 +00:00
|
|
|
.map(|group| GroupGeneration::new(group, resp_derives.clone()))
|
2022-07-23 00:38:00 +00:00
|
|
|
.map(generate_group);
|
|
|
|
|
|
|
|
quote! {
|
|
|
|
#(#gr)*
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-07-24 00:02:57 +00:00
|
|
|
fn generate_group(group: GroupGeneration) -> TokenStream {
|
2022-07-23 00:38:00 +00:00
|
|
|
let group = group.generate();
|
|
|
|
|
|
|
|
quote! {
|
|
|
|
#group
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-07-24 00:02:57 +00:00
|
|
|
#[derive(Debug)]
|
|
|
|
pub struct StructGenerator<'a> {
|
|
|
|
type_: &'a parser::TypeWithName,
|
|
|
|
group: &'a GroupGeneration,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a> StructGenerator<'a> {
|
|
|
|
pub fn new(type_: &'a parser::TypeWithName, group: &'a GroupGeneration) -> Self {
|
|
|
|
Self { type_, group }
|
|
|
|
}
|
|
|
|
|
2022-07-23 00:38:00 +00:00
|
|
|
pub fn generate_struct(&self) -> TokenStream {
|
2022-07-24 00:02:57 +00:00
|
|
|
let fields = self
|
|
|
|
.type_
|
|
|
|
.types
|
|
|
|
.iter()
|
|
|
|
.map(|obj| obj.generate_struct_field());
|
|
|
|
let name = util::to_ident(&self.type_.name);
|
|
|
|
let derives = self.group.response_derives(vec![]);
|
2022-07-23 00:38:00 +00:00
|
|
|
|
|
|
|
quote! {
|
2022-07-24 00:02:57 +00:00
|
|
|
#derives
|
2022-07-23 00:38:00 +00:00
|
|
|
pub struct #name {
|
|
|
|
#(#fields,)*
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl types::Type {
|
|
|
|
pub fn generate_struct_field(&self) -> TokenStream {
|
|
|
|
let name_snake = self.name_snake();
|
|
|
|
let type_ = self.owned_type_ident();
|
|
|
|
let orig_name = self.name();
|
|
|
|
|
|
|
|
util::add_docs(
|
|
|
|
&self.get_type_info().description,
|
|
|
|
quote! {
|
|
|
|
#[serde(rename = #orig_name)]
|
|
|
|
pub #name_snake: #type_
|
|
|
|
},
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn owned_type_ident(&self) -> TokenStream {
|
|
|
|
let owned_type = match self {
|
2022-07-24 01:06:45 +00:00
|
|
|
types::Type::Number(_) => quote! { i64 },
|
2022-07-23 00:38:00 +00:00
|
|
|
types::Type::Float(_) => quote! { f32 },
|
|
|
|
types::Type::Bool(_) => quote! { bool },
|
|
|
|
types::Type::String(_) => quote! { String },
|
|
|
|
types::Type::StringArray(_) => quote! { String },
|
|
|
|
types::Type::Object(obj) => match &obj.ref_type {
|
|
|
|
types::RefType::String(str) => {
|
|
|
|
let str_ident = &util::to_ident(str);
|
|
|
|
quote! { #str_ident }
|
|
|
|
}
|
|
|
|
types::RefType::Map(key, value) => {
|
|
|
|
let key_ident = util::to_ident(key);
|
|
|
|
let value_ident = util::to_ident(value);
|
|
|
|
quote! { std::collections::HashMap<#key_ident, #value_ident> }
|
|
|
|
}
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
if self.is_list() {
|
|
|
|
quote! { std::vec::Vec<#owned_type> }
|
|
|
|
} else {
|
|
|
|
owned_type
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn name(&self) -> String {
|
|
|
|
self.get_type_info().name.clone()
|
|
|
|
}
|
|
|
|
|
|
|
|
fn name_snake(&self) -> Ident {
|
|
|
|
util::to_ident(&self.name().to_snake())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-07-24 00:02:57 +00:00
|
|
|
#[derive(Debug)]
|
|
|
|
pub struct EnumGeneration<'a> {
|
|
|
|
enum_: &'a parser::Enum,
|
|
|
|
group: &'a GroupGeneration,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a> EnumGeneration<'a> {
|
|
|
|
pub fn new(enum_: &'a parser::Enum, group: &'a GroupGeneration) -> Self {
|
|
|
|
Self { enum_, group }
|
|
|
|
}
|
|
|
|
|
2022-07-23 00:38:00 +00:00
|
|
|
pub fn generate(&self) -> TokenStream {
|
2022-07-24 00:02:57 +00:00
|
|
|
let values = self
|
|
|
|
.enum_
|
|
|
|
.values
|
|
|
|
.iter()
|
|
|
|
.map(|enum_value| enum_value.generate());
|
|
|
|
let name = util::to_ident(&self.enum_.name);
|
|
|
|
let derives = self.group.response_derives(vec!["PartialEq", "Eq"]);
|
2022-07-23 00:38:00 +00:00
|
|
|
|
|
|
|
quote! {
|
|
|
|
#[allow(clippy::enum_variant_names)]
|
2022-07-24 00:02:57 +00:00
|
|
|
#derives
|
2022-07-23 00:38:00 +00:00
|
|
|
pub enum #name {
|
|
|
|
#(#values,)*
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl parser::EnumValue {
|
|
|
|
fn generate(&self) -> TokenStream {
|
|
|
|
util::add_docs(&self.description, self.generate_field())
|
|
|
|
}
|
|
|
|
|
|
|
|
fn generate_field(&self) -> TokenStream {
|
|
|
|
let orig_name = self.original_value.clone();
|
|
|
|
|
|
|
|
// special enum value which does not follow conventions
|
|
|
|
if orig_name == "\"/path/to/download/to\"" {
|
|
|
|
quote! {
|
|
|
|
PathToDownloadTo(String)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
let name_camel = self.name_camel();
|
|
|
|
quote! {
|
|
|
|
#[serde(rename = #orig_name)]
|
|
|
|
#name_camel
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn name_camel(&self) -> Ident {
|
|
|
|
util::to_ident(&self.value.to_camel())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl types::Type {
|
|
|
|
pub fn generate_optional_builder_method_with_docs(&self) -> TokenStream {
|
|
|
|
util::add_docs(
|
|
|
|
&self.get_type_info().description,
|
|
|
|
self.generate_optional_builder_method(),
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn borrowed_type_ident(&self) -> Ident {
|
|
|
|
util::to_ident(&self.to_borrowed_type())
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn to_parameter(&self) -> TokenStream {
|
|
|
|
let name_snake = self.name_snake();
|
|
|
|
let borrowed_type = self.borrowed_type();
|
|
|
|
|
|
|
|
quote! { #name_snake: #borrowed_type }
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn generate_form_builder(&self, add_to: TokenStream) -> TokenStream {
|
|
|
|
let name_str = self.name();
|
|
|
|
let name_snake = self.name_snake();
|
|
|
|
|
|
|
|
quote! {
|
|
|
|
#add_to = #add_to.text(#name_str, #name_snake.to_string());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn generate_optional_builder_method(&self) -> TokenStream {
|
|
|
|
let name_snake = self.name_snake();
|
|
|
|
let borrowed_type = self.borrowed_type();
|
|
|
|
let form_builder = self.generate_form_builder(quote! { self.form });
|
|
|
|
|
|
|
|
quote! {
|
|
|
|
pub fn #name_snake(mut self, #name_snake: #borrowed_type) -> Self {
|
|
|
|
#form_builder;
|
|
|
|
self
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn borrowed_type(&self) -> TokenStream {
|
|
|
|
let type_ = self.borrowed_type_ident();
|
|
|
|
if self.should_borrow() {
|
|
|
|
quote! { &#type_ }
|
|
|
|
} else {
|
|
|
|
quote! { #type_ }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|