mirror of
https://github.com/kjuulh/dagger-rs.git
synced 2025-04-20 04:29:50 +02:00
432 lines
13 KiB
Rust
432 lines
13 KiB
Rust
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<String> {
|
|
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<rust::Tokens> {
|
|
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<rust::Tokens> {
|
|
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::<Vec<String>>());
|
|
});
|
|
}
|
|
}
|
|
}
|
|
|
|
Some(quote! {
|
|
query = query.arg($(quoted(name)), $(n));
|
|
})
|
|
})
|
|
})
|
|
.flatten()
|
|
.collect::<Vec<_>>();
|
|
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<rust::Tokens> {
|
|
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::<Vec<_>>();
|
|
|
|
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::<Vec<_>>();
|
|
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 <func>_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<rust::Tokens> {
|
|
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::<Vec<_>>();
|
|
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::<Vec<_>>())
|
|
.map(|t| {
|
|
t.into_iter()
|
|
.filter(|t| type_ref_is_optional(&t.input_value.type_))
|
|
.collect::<Vec<_>>()
|
|
})
|
|
.pipe(|t| render_optional_field_args(funcs, t))
|
|
.flatten()
|
|
}
|
|
|
|
pub fn write_comment_line(content: &str) -> Option<rust::Tokens> {
|
|
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<rust::Tokens> {
|
|
let lines = desc.trim().split("\n");
|
|
|
|
let formatted_lines = lines
|
|
.into_iter()
|
|
.map(write_comment_line)
|
|
.collect::<Vec<_>>();
|
|
|
|
if formatted_lines.len() > 0 {
|
|
Some(quote! {
|
|
$(for line in formatted_lines join($['\r']) => $line)
|
|
})
|
|
} else {
|
|
None
|
|
}
|
|
}
|