use convert_case::{Case, Casing}; use dagger_core::introspection::{FullTypeFields, TypeRef}; use genco::prelude::rust; use genco::quote; use genco::tokens::quoted; use crate::functions::{ type_field_has_optional, type_ref_is_list, type_ref_is_list_of_objects, type_ref_is_object, type_ref_is_optional, type_ref_is_scalar, CommonFunctions, Scalar, }; use crate::utility::OptionExt; use super::templates::object_tmpl::render_optional_field_args; pub fn format_name(s: &str) -> String { s.to_case(Case::Pascal) } pub fn format_struct_name(s: &str) -> String { s.to_case(Case::Snake) } pub fn field_options_struct_name(field: &FullTypeFields) -> Option { field .parent_type .as_ref() .map(|p| p.name.as_ref().map(|n| format_name(n))) .flatten() .zip(field.name.as_ref().map(|n| format_name(n))) .map(|(parent_name, field_name)| format!("{parent_name}{field_name}Opts")) } pub fn format_function(funcs: &CommonFunctions, field: &FullTypeFields) -> Option { let is_async = field.type_.pipe(|t| &t.type_ref).pipe(|t| { if type_ref_is_object(&t) || type_ref_is_list_of_objects(&t) { return None; } else { return Some(quote! { async }); }; }); let signature = quote! { pub $(is_async) fn $(field.name.pipe(|n | format_struct_name(n))) }; let lifecycle = format_optional_args(funcs, field) .pipe(|(_, contains_lifecycle)| contains_lifecycle) .and_then(|c| { if *c { Some(quote! { <'a> }) } else { None } }); let args = format_function_args(funcs, field, lifecycle.as_ref()); let output_type = field .type_ .pipe(|t| &t.type_ref) .pipe(|t| render_output_type(funcs, t)); if let Some((args, desc, true)) = args { let required_args = format_required_function_args(funcs, field); Some(quote! { $(field.description.pipe(|d| format_struct_comment(d))) $(&desc) $(&signature)( $(required_args) ) -> $(output_type.as_ref()) { let mut query = self.selection.select($(quoted(field.name.as_ref()))); $(render_required_args(funcs, field)) $(render_execution(funcs, field)) } $(field.description.pipe(|d| format_struct_comment(d))) $(&desc) $(&signature)_opts$(lifecycle)( $args ) -> $(output_type) { let mut query = self.selection.select($(quoted(field.name.as_ref()))); $(render_required_args(funcs, field)) $(render_optional_args(funcs, field)) $(render_execution(funcs, field)) } }) } else { Some(quote! { $(field.description.pipe(|d| format_struct_comment(d))) $(if let Some((_, desc, _)) = &args => $desc) $(signature)( $(if let Some((args, _, _)) = &args => $args) ) -> $(output_type) { let mut query = self.selection.select($(quoted(field.name.as_ref()))); $(render_required_args(funcs, field)) $(render_optional_args(funcs, field)) $(render_execution(funcs, field)) } }) } } fn render_required_args(_funcs: &CommonFunctions, field: &FullTypeFields) -> Option { if let Some(args) = field.args.as_ref() { let args = args .into_iter() .map(|a| { a.as_ref().and_then(|s| { if type_ref_is_optional(&s.input_value.type_) { return None; } let n = format_struct_name(&s.input_value.name); let name = &s.input_value.name; if type_ref_is_scalar(&s.input_value.type_) { if let Scalar::String = Scalar::from(&*s.input_value.type_.of_type.as_ref().unwrap().clone()) { return Some(quote! { query = query.arg($(quoted(name)), $(&n).into()); }); } } if type_ref_is_list(&s.input_value.type_) { let inner = *s .input_value .type_ .of_type .as_ref() .unwrap() .clone() .of_type .as_ref() .unwrap() .clone(); println!("type: {:?}", inner); if type_ref_is_scalar(&inner) { if let Scalar::String = Scalar::from(&*inner.of_type.as_ref().unwrap().clone()) { return Some(quote! { query = query.arg($(quoted(name)), $(&n).into_iter().map(|i| i.into()).collect::>()); }); } } } Some(quote! { query = query.arg($(quoted(name)), $(n)); }) }) }) .flatten() .collect::>(); let required_args = quote! { $(for arg in args join ($['\r']) => $arg) }; Some(required_args) } else { None } } fn render_optional_args(_funcs: &CommonFunctions, field: &FullTypeFields) -> Option { if let Some(args) = field.args.as_ref() { let args = args .into_iter() .map(|a| { a.as_ref().and_then(|s| { if !type_ref_is_optional(&s.input_value.type_) { return None; } let n = format_struct_name(&s.input_value.name); let name = &s.input_value.name; Some(quote! { if let Some($(&n)) = opts.$(&n) { query = query.arg($(quoted(name)), $(&n)); } }) }) }) .flatten() .collect::>(); if args.len() == 0 { return None; } let required_args = quote! { $(for arg in args join ($['\r']) => $arg) }; Some(required_args) } else { None } } fn render_output_type(funcs: &CommonFunctions, type_ref: &TypeRef) -> rust::Tokens { let output_type = funcs.format_output_type(type_ref); if type_ref_is_object(type_ref) || type_ref_is_list_of_objects(type_ref) { return quote! { $(output_type) }; } quote! { eyre::Result<$output_type> } } fn render_execution(funcs: &CommonFunctions, field: &FullTypeFields) -> rust::Tokens { if let Some(true) = field.type_.pipe(|t| type_ref_is_object(&t.type_ref)) { let output_type = funcs.format_output_type(&field.type_.as_ref().unwrap().type_ref); return quote! { return $(output_type) { proc: self.proc.clone(), selection: query, conn: self.conn.clone(), } }; } if let Some(true) = field .type_ .pipe(|t| type_ref_is_list_of_objects(&t.type_ref)) { let output_type = funcs.format_output_type( &field .type_ .as_ref() .unwrap() .type_ref .of_type .as_ref() .unwrap() .of_type .as_ref() .unwrap(), ); return quote! { return vec![$(output_type) { proc: self.proc.clone(), selection: query, conn: self.conn.clone(), }] }; } let graphql_client = rust::import("crate::client", "graphql_client"); quote! { query.execute(&$graphql_client(&self.conn)).await } } fn format_function_args( funcs: &CommonFunctions, field: &FullTypeFields, lifecycle: Option<&rust::Tokens>, ) -> Option<(rust::Tokens, rust::Tokens, bool)> { let mut argument_description = Vec::new(); if let Some(args) = field.args.as_ref() { let args = args .into_iter() .map(|a| { a.as_ref().and_then(|s| { if type_ref_is_optional(&s.input_value.type_) { return None; } let t = funcs.format_input_type(&s.input_value.type_); let n = format_struct_name(&s.input_value.name); if let Some(desc) = s.input_value.description.as_ref().and_then(|d| { if d != "" { Some(write_comment_line(&format!("* `{n}` - {}", d))) } else { None } }) { argument_description.push(quote! { $(desc) }); } Some(quote! { $(n): $(t), }) }) }) .flatten() .collect::>(); let required_args = quote! { &self, $(for arg in args join ($['\r']) => $arg) }; if type_field_has_optional(field) { let field_name = field_options_struct_name(field); argument_description.push(quote! { $(field_name.pipe(|_| write_comment_line(&format!("* `opt` - optional argument, see inner type for documentation, use _opts to use")))) }); let description = quote! { $(if argument_description.len() > 0 => $(format!("/// "))) $(if argument_description.len() > 0 => $(format!("/// # Arguments"))) $(if argument_description.len() > 0 => $(format!("/// "))) $(for arg_desc in argument_description join ($['\r']) => $arg_desc) }; Some(( quote! { $(required_args) opts: $(field_name)$(lifecycle) }, description, true, )) } else { let description = quote! { $(if argument_description.len() > 0 => $(format!("/// "))) $(if argument_description.len() > 0 => $(format!("/// # Arguments"))) $(if argument_description.len() > 0 => $(format!("/// "))) $(for arg_desc in argument_description join ($['\r']) => $arg_desc) }; Some((required_args, description, false)) } } else { None } } fn format_required_function_args( funcs: &CommonFunctions, field: &FullTypeFields, ) -> Option { if let Some(args) = field.args.as_ref() { let args = args .into_iter() .map(|a| { a.as_ref().and_then(|s| { if type_ref_is_optional(&s.input_value.type_) { return None; } let t = funcs.format_input_type(&s.input_value.type_); let n = format_struct_name(&s.input_value.name); Some(quote! { $(n): $(t), }) }) }) .flatten() .collect::>(); let required_args = quote! { &self, $(for arg in args join ($['\r']) => $arg) }; Some(required_args) } else { None } } pub fn format_optional_args( funcs: &CommonFunctions, field: &FullTypeFields, ) -> Option<(rust::Tokens, bool)> { field .args .pipe(|t| t.into_iter().flatten().collect::>()) .map(|t| { t.into_iter() .filter(|t| type_ref_is_optional(&t.input_value.type_)) .collect::>() }) .pipe(|t| render_optional_field_args(funcs, t)) .flatten() } pub fn write_comment_line(content: &str) -> Option { let cnt = content.trim(); if cnt == "" { return None; } let mut tokens = rust::Tokens::new(); for line in content.split('\n') { tokens.append(format!("/// {}", line.trim())); tokens.push(); } Some(tokens) } pub fn format_struct_comment(desc: &str) -> Option { let lines = desc.trim().split("\n"); let formatted_lines = lines .into_iter() .map(write_comment_line) .collect::>(); if formatted_lines.len() > 0 { Some(quote! { $(for line in formatted_lines join($['\r']) => $line) }) } else { None } }