From 316254ce38d7965bd99db85967498d541a33cfa8 Mon Sep 17 00:00:00 2001 From: laststylebender <43403528+laststylebender14@users.noreply.github.com> Date: Sat, 14 Dec 2024 01:45:11 +0530 Subject: [PATCH 1/3] feat: cache graphql query in query plan. (#3106) Co-authored-by: Tushar Mathur --- src/core/document.rs | 122 +++++++++----- src/core/graphql/request_template.rs | 45 +++++- src/core/ir/model.rs | 22 +++ src/core/jit/model.rs | 28 +++- src/core/jit/request.rs | 1 + src/core/jit/transform/graphql.rs | 109 +++++++++++++ src/core/jit/transform/input_resolver.rs | 30 +++- src/core/jit/transform/mod.rs | 2 + src/core/json/graphql.rs | 152 +++++++++++++++++- tests/core/snapshots/graphql-nested.md_0.snap | 23 +++ .../snapshots/graphql-nested.md_client.snap | 30 ++++ .../snapshots/graphql-nested.md_merged.snap | 30 ++++ .../graphql-datasource-query-directives.md | 2 +- tests/execution/graphql-nested.md | 71 ++++++++ 14 files changed, 615 insertions(+), 52 deletions(-) create mode 100644 src/core/jit/transform/graphql.rs create mode 100644 tests/core/snapshots/graphql-nested.md_0.snap create mode 100644 tests/core/snapshots/graphql-nested.md_client.snap create mode 100644 tests/core/snapshots/graphql-nested.md_merged.snap create mode 100644 tests/execution/graphql-nested.md diff --git a/src/core/document.rs b/src/core/document.rs index baaf7b9ca8..a65b307f7c 100644 --- a/src/core/document.rs +++ b/src/core/document.rs @@ -1,10 +1,12 @@ +use std::borrow::Cow; +use std::fmt::Display; + use async_graphql::parser::types::*; -use async_graphql::{Pos, Positioned}; -use async_graphql_value::{ConstValue, Name}; +use async_graphql::Positioned; +use async_graphql_value::ConstValue; -fn pos(a: A) -> Positioned { - Positioned::new(a, Pos::default()) -} +use super::jit::Directive as JitDirective; +use super::json::JsonLikeOwned; struct LineBreaker<'a> { string: &'a str, @@ -61,9 +63,12 @@ fn get_formatted_docs(docs: Option, indent: usize) -> String { formatted_docs } -pub fn print_directives<'a>(directives: impl Iterator) -> String { +pub fn print_directives<'a, T>(directives: impl Iterator) -> String +where + &'a T: Into> + 'a, +{ directives - .map(|d| print_directive(&const_directive_to_sdl(d))) + .map(|d| print_directive(d)) .collect::>() .join(" ") } @@ -102,37 +107,6 @@ fn print_schema(schema: &SchemaDefinition) -> String { ) } -fn const_directive_to_sdl(directive: &ConstDirective) -> DirectiveDefinition { - DirectiveDefinition { - description: None, - name: pos(Name::new(directive.name.node.as_str())), - arguments: directive - .arguments - .iter() - .filter_map(|(k, v)| { - if v.node != ConstValue::Null { - Some(pos(InputValueDefinition { - description: None, - name: pos(Name::new(k.node.clone())), - ty: pos(Type { - nullable: true, - base: async_graphql::parser::types::BaseType::Named(Name::new( - v.to_string(), - )), - }), - default_value: Some(pos(ConstValue::String(v.to_string()))), - directives: Vec::new(), - })) - } else { - None - } - }) - .collect(), - is_repeatable: true, - locations: vec![], - } -} - fn print_type_def(type_def: &TypeDefinition) -> String { match &type_def.kind { TypeKind::Scalar => { @@ -320,18 +294,23 @@ fn print_input_value(field: &async_graphql::parser::types::InputValueDefinition) print_default_value(field.default_value.as_ref()) ) } -fn print_directive(directive: &DirectiveDefinition) -> String { + +pub fn print_directive<'a, T>(directive: &'a T) -> String +where + &'a T: Into>, +{ + let directive: Directive<'a> = directive.into(); let args = directive - .arguments + .args .iter() - .map(|arg| format!("{}: {}", arg.node.name.node, arg.node.ty.node)) + .map(|arg| format!("{}: {}", arg.name, arg.value)) .collect::>() .join(", "); if args.is_empty() { - format!("@{}", directive.name.node) + format!("@{}", directive.name) } else { - format!("@{}({})", directive.name.node, args) + format!("@{}({})", directive.name, args) } } @@ -420,3 +399,60 @@ pub fn print(sd: ServiceDocument) -> String { sdl_string.trim_end_matches('\n').to_string() } + +pub struct Directive<'a> { + pub name: Cow<'a, str>, + pub args: Vec>, +} + +pub struct Arg<'a> { + pub name: Cow<'a, str>, + pub value: Cow<'a, str>, +} + +impl<'a> From<&'a ConstDirective> for Directive<'a> { + fn from(value: &'a ConstDirective) -> Self { + Self { + name: Cow::Borrowed(value.name.node.as_str()), + args: value + .arguments + .iter() + .filter_map(|(k, v)| { + if v.node != async_graphql_value::ConstValue::Null { + Some(Arg { + name: Cow::Borrowed(k.node.as_str()), + value: Cow::Owned(v.to_string()), + }) + } else { + None + } + }) + .collect(), + } + } +} + +impl<'a, Input: JsonLikeOwned + Display> From<&'a JitDirective> for Directive<'a> { + fn from(value: &'a JitDirective) -> Self { + let to_mustache = |s: &str| -> String { + s.strip_prefix('$') + .map(|v| format!("{{{{{}}}}}", v)) + .unwrap_or_else(|| s.to_string()) + }; + Self { + name: Cow::Borrowed(value.name.as_str()), + args: value + .arguments + .iter() + .filter_map(|(k, v)| { + if !v.is_null() { + let v_str = to_mustache(&v.to_string()); + Some(Arg { name: Cow::Borrowed(k), value: Cow::Owned(v_str) }) + } else { + None + } + }) + .collect(), + } + } +} diff --git a/src/core/graphql/request_template.rs b/src/core/graphql/request_template.rs index 1eb87f1cc8..9e849d3033 100644 --- a/src/core/graphql/request_template.rs +++ b/src/core/graphql/request_template.rs @@ -6,6 +6,7 @@ use std::hash::{Hash, Hasher}; use derive_setters::Setters; use http::header::{HeaderMap, HeaderValue}; use tailcall_hasher::TailcallHasher; +use tracing::info; use crate::core::config::{GraphQLOperationType, KeyValue}; use crate::core::has_headers::HasHeaders; @@ -14,7 +15,35 @@ use crate::core::http::Method::POST; use crate::core::ir::model::{CacheKey, IoId}; use crate::core::ir::{GraphQLOperationContext, RelatedFields}; use crate::core::mustache::Mustache; -use crate::core::path::PathGraphql; +use crate::core::path::{PathGraphql, PathString}; + +/// Represents a GraphQL selection that can either be resolved or unresolved. +#[derive(Debug, Clone)] +pub enum Selection { + /// A selection with a resolved string value. + Resolved(String), + /// A selection that contains a Mustache template to be resolved later. + UnResolved(Mustache), +} + +impl Selection { + /// Resolves the `Unresolved` variant using the provided `PathString`. + pub fn resolve(self, p: &impl PathString) -> Selection { + match self { + Selection::UnResolved(template) => Selection::Resolved(template.render(p)), + resolved => resolved, + } + } +} + +impl From for Selection { + fn from(value: Mustache) -> Self { + match value.is_const() { + true => Selection::Resolved(value.to_string()), + false => Selection::UnResolved(value), + } + } +} /// RequestTemplate for GraphQL requests (See RequestTemplate documentation) #[derive(Setters, Debug, Clone)] @@ -26,6 +55,7 @@ pub struct RequestTemplate { pub operation_arguments: Option>, pub headers: MustacheHeaders, pub related_fields: RelatedFields, + pub selection: Option, } impl RequestTemplate { @@ -85,7 +115,12 @@ impl RequestTemplate { ctx: &C, ) -> String { let operation_type = &self.operation_type; - let selection_set = ctx.selection_set(&self.related_fields).unwrap_or_default(); + + let selection_set = match &self.selection { + Some(Selection::Resolved(s)) => Cow::Borrowed(s), + Some(Selection::UnResolved(u)) => Cow::Owned(u.to_string()), + None => Cow::Owned(ctx.selection_set(&self.related_fields).unwrap_or_default()), + }; let mut operation = Cow::Borrowed(&self.operation_name); @@ -121,7 +156,10 @@ impl RequestTemplate { } } - format!(r#"{{ "query": "{operation_type} {{ {operation} {selection_set} }}" }}"#) + let query = + format!(r#"{{ "query": "{operation_type} {{ {operation} {selection_set} }}" }}"#); + info!("Query {} ", query); + query } pub fn new( @@ -149,6 +187,7 @@ impl RequestTemplate { operation_arguments, headers, related_fields, + selection: None, }) } } diff --git a/src/core/ir/model.rs b/src/core/ir/model.rs index 9a07742503..f4fbfdb563 100644 --- a/src/core/ir/model.rs +++ b/src/core/ir/model.rs @@ -131,6 +131,28 @@ impl Cache { } impl IR { + // allows to modify the IO node in the IR tree + pub fn modify_io(&mut self, io_modifier: &mut dyn FnMut(&mut IO)) { + match self { + IR::IO(io) => io_modifier(io), + IR::Cache(cache) => io_modifier(&mut cache.io), + IR::Discriminate(_, ir) | IR::Protect(_, ir) | IR::Path(ir, _) => { + ir.modify_io(io_modifier) + } + IR::Pipe(ir1, ir2) => { + ir1.modify_io(io_modifier); + ir2.modify_io(io_modifier); + } + IR::Entity(hash_map) => { + for ir in hash_map.values_mut() { + ir.modify_io(io_modifier); + } + } + IR::Map(map) => map.input.modify_io(io_modifier), + _ => {} + } + } + pub fn pipe(self, next: Self) -> Self { IR::Pipe(Box::new(self), Box::new(next)) } diff --git a/src/core/jit/model.rs b/src/core/jit/model.rs index 1b833b0ed0..9b2950a22a 100644 --- a/src/core/jit/model.rs +++ b/src/core/jit/model.rs @@ -1,6 +1,6 @@ use std::borrow::Cow; use std::collections::HashMap; -use std::fmt::{Debug, Formatter}; +use std::fmt::{Debug, Display, Formatter}; use std::num::NonZeroU64; use std::sync::Arc; @@ -13,12 +13,20 @@ use super::Error; use crate::core::blueprint::Index; use crate::core::ir::model::IR; use crate::core::ir::TypedValue; -use crate::core::json::JsonLike; +use crate::core::json::{JsonLike, JsonLikeOwned}; +use crate::core::path::PathString; use crate::core::scalar::Scalar; #[derive(Debug, Deserialize, Clone)] pub struct Variables(HashMap); +impl PathString for Variables { + fn path_string<'a, T: AsRef>(&'a self, path: &'a [T]) -> Option> { + self.get(path[0].as_ref()) + .map(|v| Cow::Owned(v.to_string())) + } +} + impl Default for Variables { fn default() -> Self { Self::new() @@ -96,6 +104,22 @@ pub struct Arg { pub default_value: Option, } +impl Display for Arg { + fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result { + let v = self + .value + .as_ref() + .map(|v| v.to_string()) + .unwrap_or_else(|| { + self.default_value + .as_ref() + .map(|v| v.to_string()) + .unwrap_or_default() + }); + write!(f, "{}: {}", self.name, v) + } +} + impl Arg { pub fn try_map( self, diff --git a/src/core/jit/request.rs b/src/core/jit/request.rs index dc08ee53d7..cbd3bc0faf 100644 --- a/src/core/jit/request.rs +++ b/src/core/jit/request.rs @@ -50,6 +50,7 @@ impl Request { .pipe(transform::AuthPlanner::new()) .pipe(transform::CheckDedupe::new()) .pipe(transform::CheckCache::new()) + .pipe(transform::GraphQL::new()) .transform(plan) .to_result() // both transformers are infallible right now diff --git a/src/core/jit/transform/graphql.rs b/src/core/jit/transform/graphql.rs new file mode 100644 index 0000000000..79139c9891 --- /dev/null +++ b/src/core/jit/transform/graphql.rs @@ -0,0 +1,109 @@ +use std::borrow::Cow; +use std::convert::Infallible; +use std::fmt::{Debug, Display}; +use std::marker::PhantomData; + +use tailcall_valid::Valid; + +use crate::core::document::print_directives; +use crate::core::ir::model::{IO, IR}; +use crate::core::jit::{Field, OperationPlan}; +use crate::core::json::JsonLikeOwned; +use crate::core::{Mustache, Transform}; + +#[derive(Default)] +pub struct GraphQL(PhantomData); + +impl GraphQL { + pub fn new() -> Self { + Self(PhantomData) + } +} + +fn compute_selection_set(base_field: &mut [Field]) { + for field in base_field.iter_mut() { + if let Some(ir) = field.ir.as_mut() { + ir.modify_io(&mut |io| { + if let IO::GraphQL { req_template, .. } = io { + if let Some(v) = format_selection_set(field.selection.iter()) { + req_template.selection = Some(Mustache::parse(&v).into()); + } + } + }); + } + compute_selection_set(field.selection.as_mut()); + } +} + +impl Transform for GraphQL { + type Value = OperationPlan; + type Error = Infallible; + + fn transform(&self, mut plan: Self::Value) -> Valid { + compute_selection_set(&mut plan.selection); + + Valid::succeed(plan) + } +} + +fn format_selection_set<'a, A: 'a + Display + JsonLikeOwned>( + selection_set: impl Iterator>, +) -> Option { + let set = selection_set + .filter(|field| !matches!(&field.ir, Some(IR::IO(_)) | Some(IR::Dynamic(_)))) + .map(|field| { + // handle @modify directive scenario. + let field_name = if let Some(IR::ContextPath(data)) = &field.ir { + data.first().cloned().unwrap_or(field.name.to_string()) + } else { + field.name.to_string() + }; + format_selection_field(field, &field_name) + }) + .collect::>(); + + if set.is_empty() { + return None; + } + + Some(format!("{{ {} }}", set.join(" "))) +} + +fn format_selection_field(field: &Field, name: &str) -> String { + let arguments = format_selection_field_arguments(field); + let selection_set = format_selection_set(field.selection.iter()); + + let mut output = format!("{}{}", name, arguments); + + if !field.directives.is_empty() { + let directives = print_directives(field.directives.iter()); + + if !directives.is_empty() { + output.push(' '); + output.push_str(&directives.escape_default().to_string()); + } + } + + if let Some(selection_set) = selection_set { + output.push(' '); + output.push_str(&selection_set); + } + + output +} + +fn format_selection_field_arguments(field: &Field) -> Cow<'static, str> { + let arguments = field + .args + .iter() + .filter(|a| a.value.is_some()) + .map(|arg| arg.to_string()) + .collect::>() + .join(","); + + if arguments.is_empty() { + Cow::Borrowed("") + } else { + Cow::Owned(format!("({})", arguments.escape_default())) + } +} diff --git a/src/core/jit/transform/input_resolver.rs b/src/core/jit/transform/input_resolver.rs index 920bcfdae1..86b5fe59ba 100644 --- a/src/core/jit/transform/input_resolver.rs +++ b/src/core/jit/transform/input_resolver.rs @@ -1,7 +1,10 @@ +use std::fmt::Display; + use async_graphql_value::{ConstValue, Value}; use super::super::{Arg, Field, OperationPlan, ResolveInputError, Variables}; use crate::core::blueprint::Index; +use crate::core::ir::model::IO; use crate::core::json::{JsonLikeOwned, JsonObjectLike}; use crate::core::Type; @@ -46,7 +49,7 @@ impl InputResolver { impl InputResolver where Input: Clone + std::fmt::Debug, - Output: Clone + JsonLikeOwned + TryFrom + std::fmt::Debug, + Output: Clone + JsonLikeOwned + TryFrom + std::fmt::Debug + Display, Input: InputResolvable, >::Error: std::fmt::Debug, { @@ -55,7 +58,7 @@ where variables: &Variables, ) -> Result, ResolveInputError> { let index = self.plan.index; - let selection = self + let mut selection = self .plan .selection .into_iter() @@ -68,6 +71,10 @@ where .map(|field| Self::resolve_field(&index, field?)) .collect::, _>>()?; + // adjust the pre-computed values in selection set like graphql query for + // @graphql directive. + Self::resolve_graphql_selection_set(&mut selection, variables); + Ok(OperationPlan { root_name: self.plan.root_name.to_string(), operation_type: self.plan.operation_type, @@ -82,6 +89,25 @@ where }) } + // resolves the variables in selection set mustache template for graphql query. + fn resolve_graphql_selection_set( + base_field: &mut [Field], + variables: &Variables, + ) { + for field in base_field.iter_mut() { + if let Some(ir) = field.ir.as_mut() { + ir.modify_io(&mut |io| { + if let IO::GraphQL { req_template, .. } = io { + if let Some(selection) = req_template.selection.take() { + req_template.selection = Some(selection.resolve(variables)); + } + } + }); + } + Self::resolve_graphql_selection_set(field.selection.as_mut(), variables); + } + } + fn resolve_field( index: &Index, field: Field, diff --git a/src/core/jit/transform/mod.rs b/src/core/jit/transform/mod.rs index 71928ddbfd..c3cd01c409 100644 --- a/src/core/jit/transform/mod.rs +++ b/src/core/jit/transform/mod.rs @@ -3,6 +3,7 @@ mod check_cache; mod check_const; mod check_dedupe; mod check_protected; +mod graphql; mod input_resolver; mod skip; @@ -11,5 +12,6 @@ pub use check_cache::*; pub use check_const::*; pub use check_dedupe::*; pub use check_protected::*; +pub use graphql::*; pub use input_resolver::*; pub use skip::*; diff --git a/src/core/json/graphql.rs b/src/core/json/graphql.rs index 5491625461..f9d4260f55 100644 --- a/src/core/json/graphql.rs +++ b/src/core/json/graphql.rs @@ -2,7 +2,7 @@ use std::borrow::Cow; use std::collections::HashMap; use async_graphql::Name; -use async_graphql_value::ConstValue; +use async_graphql_value::{ConstValue, Value}; use indexmap::IndexMap; use super::*; @@ -196,3 +196,153 @@ impl<'json> JsonLike<'json> for ConstValue { ConstValue::String(s.to_string()) } } + +impl<'json> JsonLike<'json> for Value { + type JsonObject = IndexMap; + + fn from_primitive(x: JsonPrimitive<'json>) -> Self { + match x { + JsonPrimitive::Null => Value::Null, + JsonPrimitive::Bool(x) => Value::Boolean(x), + JsonPrimitive::Str(s) => Value::String(s.to_string()), + JsonPrimitive::Number(number) => Value::Number(number), + } + } + + fn as_primitive(&self) -> Option { + let val = match self { + Value::Null => JsonPrimitive::Null, + Value::Boolean(x) => JsonPrimitive::Bool(*x), + Value::Number(number) => JsonPrimitive::Number(number.clone()), + Value::String(s) => JsonPrimitive::Str(s.as_ref()), + Value::Enum(e) => JsonPrimitive::Str(e.as_str()), + _ => return None, + }; + + Some(val) + } + + fn as_array(&self) -> Option<&Vec> { + match self { + Value::List(seq) => Some(seq), + _ => None, + } + } + + fn as_array_mut(&mut self) -> Option<&mut Vec> { + match self { + Value::List(seq) => Some(seq), + _ => None, + } + } + + fn into_array(self) -> Option> { + match self { + Value::List(seq) => Some(seq), + _ => None, + } + } + + fn as_str(&self) -> Option<&str> { + match self { + Value::String(s) => Some(s), + _ => None, + } + } + + fn as_i64(&self) -> Option { + match self { + Value::Number(n) => n.as_i64(), + _ => None, + } + } + + fn as_u64(&self) -> Option { + match self { + Value::Number(n) => n.as_u64(), + _ => None, + } + } + + fn as_f64(&self) -> Option { + match self { + Value::Number(n) => n.as_f64(), + _ => None, + } + } + + fn as_bool(&self) -> Option { + match self { + Value::Boolean(b) => Some(*b), + _ => None, + } + } + + fn is_null(&self) -> bool { + matches!(self, Value::Null) + } + + fn get_path>(&self, path: &[T]) -> Option<&Self> { + let mut val = self; + for token in path { + val = match val { + Value::List(seq) => { + let index = token.as_ref().parse::().ok()?; + seq.get(index)? + } + Value::Object(map) => map.get(token.as_ref())?, + _ => return None, + }; + } + Some(val) + } + + fn get_key(&self, path: &str) -> Option<&Self> { + match self { + Value::Object(map) => map.get(&async_graphql::Name::new(path)), + _ => None, + } + } + + fn group_by(&self, path: &[String]) -> HashMap> { + let src = gather_path_matches(self, path, vec![]); + group_by_key(src) + } + + fn null() -> Self { + Default::default() + } + + fn as_object(&self) -> Option<&Self::JsonObject> { + match self { + Value::Object(map) => Some(map), + _ => None, + } + } + + fn as_object_mut(&mut self) -> Option<&mut Self::JsonObject> { + match self { + Value::Object(map) => Some(map), + _ => None, + } + } + + fn into_object(self) -> Option { + match self { + Value::Object(map) => Some(map), + _ => None, + } + } + + fn object(obj: Self::JsonObject) -> Self { + Value::Object(obj) + } + + fn array(arr: Vec) -> Self { + Value::List(arr) + } + + fn string(s: Cow<'json, str>) -> Self { + Value::String(s.to_string()) + } +} diff --git a/tests/core/snapshots/graphql-nested.md_0.snap b/tests/core/snapshots/graphql-nested.md_0.snap new file mode 100644 index 0000000000..0b3e1b3e22 --- /dev/null +++ b/tests/core/snapshots/graphql-nested.md_0.snap @@ -0,0 +1,23 @@ +--- +source: tests/core/spec.rs +expression: response +--- +{ + "status": 200, + "headers": { + "content-type": "application/json" + }, + "body": { + "data": { + "queryNodeA": { + "name": "nodeA", + "nodeB": { + "name": "nodeB" + }, + "nodeC": { + "name": "nodeC" + } + } + } + } +} diff --git a/tests/core/snapshots/graphql-nested.md_client.snap b/tests/core/snapshots/graphql-nested.md_client.snap new file mode 100644 index 0000000000..b804ac9539 --- /dev/null +++ b/tests/core/snapshots/graphql-nested.md_client.snap @@ -0,0 +1,30 @@ +--- +source: tests/core/spec.rs +expression: formatted +--- +type NodeA { + child: NodeA + name: String + nodeB: NodeB + nodeC: NodeC +} + +type NodeB { + name: String + nodeA: NodeA + nodeC: NodeC +} + +type NodeC { + name: String + nodeA: NodeA + nodeB: NodeB +} + +type Query { + queryNodeA: NodeA +} + +schema { + query: Query +} diff --git a/tests/core/snapshots/graphql-nested.md_merged.snap b/tests/core/snapshots/graphql-nested.md_merged.snap new file mode 100644 index 0000000000..8accbbde47 --- /dev/null +++ b/tests/core/snapshots/graphql-nested.md_merged.snap @@ -0,0 +1,30 @@ +--- +source: tests/core/spec.rs +expression: formatter +--- +schema @server(hostname: "0.0.0.0", port: 8000) @upstream { + query: Query +} + +type NodeA { + name: String + nodeA: NodeA @modify(name: "child") + nodeB: NodeB + nodeC: NodeC +} + +type NodeB { + name: String + nodeA: NodeA + nodeC: NodeC +} + +type NodeC { + name: String + nodeA: NodeA + nodeB: NodeB +} + +type Query { + queryNodeA: NodeA @graphQL(url: "http://upstream/graphql", name: "nodeA") +} diff --git a/tests/execution/graphql-datasource-query-directives.md b/tests/execution/graphql-datasource-query-directives.md index f56b4b6a16..e1580c70c3 100644 --- a/tests/execution/graphql-datasource-query-directives.md +++ b/tests/execution/graphql-datasource-query-directives.md @@ -21,7 +21,7 @@ type Query { - request: method: POST url: http://upstream/graphql - textBody: '{ "query": "query { user @cascade(fields: [\\\"id\\\"]) { id @options(paging: false) } }" }' + textBody: '{ "query": "query { user @cascade(fields: [\\\"id\\\"]) { id @options(paging: false) name } }" }' response: status: 200 body: diff --git a/tests/execution/graphql-nested.md b/tests/execution/graphql-nested.md new file mode 100644 index 0000000000..bb9a277611 --- /dev/null +++ b/tests/execution/graphql-nested.md @@ -0,0 +1,71 @@ +# Complicated queries + +```graphql @config +schema @server(port: 8000, hostname: "0.0.0.0") { + query: Query +} + +type Query { + queryNodeA: NodeA @graphQL(url: "http://upstream/graphql", name: "nodeA") +} + +type NodeA { + name: String + nodeB: NodeB + nodeC: NodeC + nodeA: NodeA @modify(name: "child") +} + +type NodeB { + name: String + nodeA: NodeA + nodeC: NodeC +} + +type NodeC { + name: String + nodeA: NodeA + nodeB: NodeB +} +``` + +```yml @mock +- request: + method: POST + url: http://upstream/graphql + textBody: {"query": "query { nodeA { name nodeB { name } nodeC { name } } }"} + expectedHits: 1 + response: + status: 200 + body: + data: + nodeA: + name: nodeA + nodeB: + name: nodeB + nodeC: + name: nodeC + nodeA: + name: nodeA +``` + +```yml @test +- method: POST + url: http://localhost:8080/graphql + body: + query: | + query queryNodeA { + queryNodeA { + name + nodeA { + name + } + nodeB { + name + } + nodeC { + name + } + } + } +``` From 76fc56541460381c26de982aa8ed4a2c93d4b9c2 Mon Sep 17 00:00:00 2001 From: Kiryl Mialeshka <8974488+meskill@users.noreply.github.com> Date: Fri, 13 Dec 2024 23:53:50 +0100 Subject: [PATCH 2/3] feat: add support for separate runtime config in json & yaml formats (#3221) --- examples/jsonplaceholder.yaml | 12 + generated/.tailcallrc.graphql | 441 ------------------------------ generated/.tailcallrc.schema.json | 190 ++++++++----- src/cli/generator/source.rs | 37 ++- src/cli/tc/init.rs | 97 ++++--- src/core/config/config.rs | 116 +++++--- src/core/config/from_document.rs | 16 +- src/core/config/source.rs | 53 ++-- tailcall-typedefs/src/main.rs | 14 +- 9 files changed, 313 insertions(+), 663 deletions(-) create mode 100644 examples/jsonplaceholder.yaml diff --git a/examples/jsonplaceholder.yaml b/examples/jsonplaceholder.yaml new file mode 100644 index 0000000000..f1edf73537 --- /dev/null +++ b/examples/jsonplaceholder.yaml @@ -0,0 +1,12 @@ +# yaml-language-server: $schema=../generated/.tailcallrc.schema.json + +server: + port: 8000 + +upstream: + batch: + delay: 100 + httpCache: 42 + +links: + - src: ./jsonplaceholder.graphql diff --git a/generated/.tailcallrc.graphql b/generated/.tailcallrc.graphql index 587bcd53c7..ddc1563d6b 100644 --- a/generated/.tailcallrc.graphql +++ b/generated/.tailcallrc.graphql @@ -247,35 +247,6 @@ directive @js( name: String! ) repeatable on FIELD_DEFINITION | OBJECT -""" -The @link directive allows you to import external resources, such as configuration -– which will be merged into the config importing it –, or a .proto file – which - will be later used by `@grpc` directive –. -""" -directive @link( - """ - Custom headers for gRPC reflection server. - """ - headers: [KeyValue] - """ - The id of the link. It is used to reference the link in the schema. - """ - id: String - """ - Additional metadata pertaining to the linked resource. - """ - meta: JSON - """ - The source of the link. It can be a URL or a path to a file. If a path is provided, - it is relative to the file that imports the link. - """ - src: String - """ - The type of the link. It can be `Config`, or `Protobuf`. - """ - type: LinkType -) repeatable on SCHEMA - directive @modify( name: String omit: Boolean @@ -303,200 +274,6 @@ directive @protected( id: [String!] ) on OBJECT | FIELD_DEFINITION -""" -The `@server` directive, when applied at the schema level, offers a comprehensive -set of server configurations. It dictates how the server behaves and helps tune tailcall -for various use-cases. -""" -directive @server( - """ - `apolloTracing` exposes GraphQL query performance data, including execution time - of queries and individual resolvers. - """ - apolloTracing: Boolean - """ - `batchRequests` combines multiple requests into one, improving performance but potentially - introducing latency and complicating debugging. Use judiciously. @default `false`. - """ - batchRequests: Boolean - """ - `enableFederation` enables functionality to Tailcall server to act as a federation - subgraph. - """ - enableFederation: Boolean - enableJIT: Boolean - """ - `globalResponseTimeout` sets the maximum query duration before termination, acting - as a safeguard against long-running queries. - """ - globalResponseTimeout: Int - """ - `headers` contains key-value pairs that are included as default headers in server - responses, allowing for consistent header management across all responses. - """ - headers: Headers - """ - `hostname` sets the server hostname. - """ - hostname: String - """ - `introspection` allows clients to fetch schema information directly, aiding tools - and applications in understanding available types, fields, and operations. @default - `true`. - """ - introspection: Boolean - """ - `pipelineFlush` allows to control flushing behavior of the server pipeline. - """ - pipelineFlush: Boolean - """ - `port` sets the Tailcall running port. @default `8000`. - """ - port: Int - """ - `queryValidation` checks incoming GraphQL queries against the schema, preventing - errors from invalid queries. Can be disabled for performance. @default `false`. - """ - queryValidation: Boolean - """ - `responseValidation` Tailcall automatically validates responses from upstream services - using inferred schema. @default `false`. - """ - responseValidation: Boolean - """ - `routes` allows customization of server endpoint paths. It provides options to change - the default paths for status and GraphQL endpoints. Default values are: - status: - "/status" - graphQL: "/graphql" If not specified, these default values will be used. - """ - routes: Routes - """ - A link to an external JS file that listens on every HTTP request response event. - """ - script: ScriptOptions - """ - `showcase` enables the /showcase/graphql endpoint. - """ - showcase: Boolean - """ - This configuration defines local variables for server operations. Useful for storing - constant configurations, secrets, or shared information. - """ - vars: [KeyValue] - """ - `version` sets the HTTP version for the server. Options are `HTTP1` and `HTTP2`. - @default `HTTP1`. - """ - version: HttpVersion - """ - `workers` sets the number of worker threads. @default the number of system cores. - """ - workers: Int -) on SCHEMA - -""" -The @telemetry directive facilitates seamless integration with OpenTelemetry, enhancing -the observability of your GraphQL services powered by Tailcall. By leveraging this -directive, developers gain access to valuable insights into the performance and behavior -of their applications. -""" -directive @telemetry( - export: TelemetryExporter - """ - The list of headers that will be sent as additional attributes to telemetry exporters - Be careful about **leaking sensitive information** from requests when enabling the - headers that may contain sensitive data - """ - requestHeaders: [String!] -) on SCHEMA - -""" -The `upstream` directive allows you to control various aspects of the upstream server -connection. This includes settings like connection timeouts, keep-alive intervals, -and more. If not specified, default values are used. -""" -directive @upstream( - """ - `allowedHeaders` defines the HTTP headers allowed to be forwarded to upstream services. - If not set, no headers are forwarded, enhancing security but possibly limiting data - flow. - """ - allowedHeaders: [String!] - """ - An object that specifies the batch settings, including `maxSize` (the maximum size - of the batch), `delay` (the delay in milliseconds between each batch), and `headers` - (an array of HTTP headers to be included in the batch). - """ - batch: Batch - """ - The time in seconds that the connection will wait for a response before timing out. - """ - connectTimeout: Int - """ - The `http2Only` setting allows you to specify whether the client should always issue - HTTP2 requests, without checking if the server supports it or not. By default it - is set to `false` for all HTTP requests made by the server, but is automatically - set to true for GRPC. - """ - http2Only: Boolean - """ - Providing httpCache size enables Tailcall's HTTP caching, adhering to the [HTTP Caching - RFC](https://tools.ietf.org/html/rfc7234), to enhance performance by minimizing redundant - data fetches. Defaults to `0` if unspecified. - """ - httpCache: Int - """ - The time in seconds between each keep-alive message sent to maintain the connection. - """ - keepAliveInterval: Int - """ - The time in seconds that the connection will wait for a keep-alive message before - closing. - """ - keepAliveTimeout: Int - """ - A boolean value that determines whether keep-alive messages should be sent while - the connection is idle. - """ - keepAliveWhileIdle: Boolean - """ - onRequest field gives the ability to specify the global request interception handler. - """ - onRequest: String - """ - The time in seconds that the connection pool will wait before closing idle connections. - """ - poolIdleTimeout: Int - """ - The maximum number of idle connections that will be maintained per host. - """ - poolMaxIdlePerHost: Int - """ - The `proxy` setting defines an intermediary server through which the upstream requests - will be routed before reaching their intended endpoint. By specifying a proxy URL, - you introduce an additional layer, enabling custom routing and security policies. - """ - proxy: Proxy - """ - The time in seconds between each TCP keep-alive message sent to maintain the connection. - """ - tcpKeepAlive: Int - """ - The maximum time in seconds that the connection will wait for a response. - """ - timeout: Int - """ - The User-Agent header value to be used in HTTP requests. @default `Tailcall/1.0` - """ - userAgent: String - """ - A boolean value that determines whether to verify certificates. Setting this as `false` - will make tailcall accept self-signed certificates. NOTE: use this *only* during - development or testing. It is highly recommended to keep this enabled (`true`) in - production. - """ - verifySSL: Boolean -) on SCHEMA - """ The `@discriminate` directive is used to drive Tailcall discriminator to use a field of an object to resolve the type. For example with the directive applied on a field @@ -652,156 +429,6 @@ input Schema { Enum: [String!] } -""" -Type to configure Cross-Origin Resource Sharing (CORS) for a server. -""" -input Cors { - """ - Indicates whether the server allows credentials (e.g., cookies, authorization headers) - to be sent in cross-origin requests. - """ - allowCredentials: Boolean - """ - A list of allowed headers in cross-origin requests. This can be used to specify custom - headers that are allowed to be included in cross-origin requests. - """ - allowHeaders: [String!] - """ - A list of allowed HTTP methods in cross-origin requests. These methods specify the - actions that are permitted in cross-origin requests. - """ - allowMethods: [Method] - """ - A list of origins that are allowed to access the server's resources in cross-origin - requests. An origin can be a domain, a subdomain, or even 'null' for local file schemes. - """ - allowOrigins: [String!] - """ - Indicates whether requests from private network addresses are allowed in cross-origin - requests. Private network addresses typically include IP addresses reserved for internal - networks. - """ - allowPrivateNetwork: Boolean - """ - A list of headers that the server exposes to the browser in cross-origin responses. - Exposing certain headers allows the client-side code to access them in the response. - """ - exposeHeaders: [String!] - """ - The maximum time (in seconds) that the client should cache preflight OPTIONS requests - in order to avoid sending excessive requests to the server. - """ - maxAge: Int - """ - A list of header names that indicate the values of which might cause the server's - response to vary, potentially affecting caching. - """ - vary: [String!] -} - -input Headers { - """ - `cacheControl` sends `Cache-Control` headers in responses when activated. The `max-age` - value is the least of the values received from upstream services. @default `false`. - """ - cacheControl: Boolean - """ - `cors` allows Cross-Origin Resource Sharing (CORS) for a server. - """ - cors: Cors - """ - `headers` are key-value pairs included in every server response. Useful for setting - headers like `Access-Control-Allow-Origin` for cross-origin requests or additional - headers for downstream services. - """ - custom: [KeyValue] - """ - `experimental` allows the use of `X-*` experimental headers in the response. @default - `[]`. - """ - experimental: [String!] - """ - `setCookies` when enabled stores `set-cookie` headers and all the response will be - sent with the headers. - """ - setCookies: Boolean -} - -input Routes { - graphQL: String - status: String -} - -input ScriptOptions { - timeout: Int -} - -input Apollo { - """ - Setting `apiKey` for Apollo. - """ - apiKey: String! - """ - Setting `graphRef` for Apollo in the format @. - """ - graphRef: String! - """ - Setting `platform` for Apollo. - """ - platform: String - """ - Setting `userVersion` for Apollo. - """ - userVersion: String - """ - Setting `version` for Apollo. - """ - version: String -} - -""" -Output the opentelemetry data to otlp collector -""" -input OtlpExporter { - headers: [KeyValue] - url: String! -} - -""" -Output the telemetry metrics data to prometheus server -""" -input PrometheusExporter { - format: PrometheusFormat - path: String -} - -""" -Output the opentelemetry data to the stdout. Mostly used for debug purposes -""" -input StdoutExporter { - """ - Output to stdout in pretty human-readable format - """ - pretty: Boolean -} - -input TelemetryExporter { - stdout: StdoutExporter - otlp: OtlpExporter - prometheus: PrometheusExporter - apollo: Apollo -} - -input Batch { - delay: Int - headers: [String!] - maxSize: Int -} - -input Proxy { - url: String! -} - """ The @graphQL operator allows to specify GraphQL API server request to fetch data from. @@ -1016,22 +643,6 @@ input Cache { maxAge: Int! } -""" -The @telemetry directive facilitates seamless integration with OpenTelemetry, enhancing -the observability of your GraphQL services powered by Tailcall. By leveraging this -directive, developers gain access to valuable insights into the performance and behavior -of their applications. -""" -input Telemetry { - export: TelemetryExporter - """ - The list of headers that will be sent as additional attributes to telemetry exporters - Be careful about **leaking sensitive information** from requests when enabling the - headers that may contain sensitive data - """ - requestHeaders: [String!] -} - enum Encoding { ApplicationJson ApplicationXWwwFormUrlencoded @@ -1047,56 +658,4 @@ enum Method { OPTIONS CONNECT TRACE -} - -enum LinkType { - """ - Points to another Tailcall Configuration file. The imported configuration will be merged into the importing configuration. - """ - Config - """ - Points to a Protobuf file. The imported Protobuf file will be used by the `@grpc` directive. If your API exposes a reflection endpoint, you should set the type to `Grpc` instead. - """ - Protobuf - """ - Points to a JS file. The imported JS file will be used by the `@js` directive. - """ - Script - """ - Points to a Cert file. The imported Cert file will be used by the server to serve over HTTPS. - """ - Cert - """ - Points to a Key file. The imported Key file will be used by the server to serve over HTTPS. - """ - Key - """ - A trusted document that contains GraphQL operations (queries, mutations) that can be exposed a REST API using the `@rest` directive. - """ - Operation - """ - Points to a Htpasswd file. The imported Htpasswd file will be used by the server to authenticate users. - """ - Htpasswd - """ - Points to a Jwks file. The imported Jwks file will be used by the server to authenticate users. - """ - Jwks - """ - Points to a reflection endpoint. The imported reflection endpoint will be used by the `@grpc` directive to resolve data from gRPC services. - """ - Grpc -} - -enum HttpVersion { - HTTP1 - HTTP2 -} - -""" -Output format for prometheus data -""" -enum PrometheusFormat { - text - protobuf } \ No newline at end of file diff --git a/generated/.tailcallrc.schema.json b/generated/.tailcallrc.schema.json index e76f874e55..62a29551d1 100644 --- a/generated/.tailcallrc.schema.json +++ b/generated/.tailcallrc.schema.json @@ -1,8 +1,18 @@ { "$schema": "http://json-schema.org/draft-07/schema#", - "title": "Config", + "title": "RuntimeConfig", "type": "object", + "required": [ + "links" + ], "properties": { + "links": { + "description": "A list of all links in the schema.", + "type": "array", + "items": { + "$ref": "#/definitions/Link" + } + }, "server": { "description": "Dictates how the server behaves and helps tune tailcall for all ingress requests. Features such as request batching, SSL, HTTP2 etc. can be configured here.", "default": {}, @@ -96,10 +106,6 @@ } } }, - "Bytes": { - "title": "Bytes", - "description": "Field whose value is a sequence of bytes." - }, "Cors": { "description": "Type to configure Cross-Origin Resource Sharing (CORS) for a server.", "type": "object", @@ -169,22 +175,6 @@ } } }, - "Date": { - "title": "Date", - "description": "Field whose value conforms to the standard date format as specified in RFC 3339 (https://datatracker.ietf.org/doc/html/rfc3339)." - }, - "DateTime": { - "title": "DateTime", - "description": "Field whose value conforms to the standard datetime format as specified in RFC 3339 (https://datatracker.ietf.org/doc/html/rfc3339\")." - }, - "Email": { - "title": "Email", - "description": "Field whose value conforms to the standard internet email address format as specified in HTML Spec: https://html.spec.whatwg.org/multipage/input.html#valid-e-mail-address." - }, - "Empty": { - "title": "Empty", - "description": "Empty scalar type represents an empty value." - }, "Headers": { "type": "object", "properties": { @@ -240,30 +230,6 @@ "HTTP2" ] }, - "Int128": { - "title": "Int128", - "description": "Field whose value is a 128-bit signed integer." - }, - "Int16": { - "title": "Int16", - "description": "Field whose value is a 16-bit signed integer." - }, - "Int32": { - "title": "Int32", - "description": "Field whose value is a 32-bit signed integer." - }, - "Int64": { - "title": "Int64", - "description": "Field whose value is a 64-bit signed integer." - }, - "Int8": { - "title": "Int8", - "description": "Field whose value is an 8-bit signed integer." - }, - "JSON": { - "title": "JSON", - "description": "Field whose value conforms to the standard JSON format as specified in RFC 8259 (https://datatracker.ietf.org/doc/html/rfc8259)." - }, "KeyValue": { "type": "object", "required": [ @@ -279,6 +245,112 @@ } } }, + "Link": { + "description": "The @link directive allows you to import external resources, such as configuration – which will be merged into the config importing it –, or a .proto file – which will be later used by `@grpc` directive –.", + "type": "object", + "properties": { + "headers": { + "description": "Custom headers for gRPC reflection server.", + "type": [ + "array", + "null" + ], + "items": { + "$ref": "#/definitions/KeyValue" + } + }, + "id": { + "description": "The id of the link. It is used to reference the link in the schema.", + "type": [ + "string", + "null" + ] + }, + "meta": { + "description": "Additional metadata pertaining to the linked resource." + }, + "src": { + "description": "The source of the link. It can be a URL or a path to a file. If a path is provided, it is relative to the file that imports the link.", + "type": "string" + }, + "type": { + "description": "The type of the link. It can be `Config`, or `Protobuf`.", + "allOf": [ + { + "$ref": "#/definitions/LinkType" + } + ] + } + }, + "additionalProperties": false + }, + "LinkType": { + "oneOf": [ + { + "description": "Points to another Tailcall Configuration file. The imported configuration will be merged into the importing configuration.", + "type": "string", + "enum": [ + "Config" + ] + }, + { + "description": "Points to a Protobuf file. The imported Protobuf file will be used by the `@grpc` directive. If your API exposes a reflection endpoint, you should set the type to `Grpc` instead.", + "type": "string", + "enum": [ + "Protobuf" + ] + }, + { + "description": "Points to a JS file. The imported JS file will be used by the `@js` directive.", + "type": "string", + "enum": [ + "Script" + ] + }, + { + "description": "Points to a Cert file. The imported Cert file will be used by the server to serve over HTTPS.", + "type": "string", + "enum": [ + "Cert" + ] + }, + { + "description": "Points to a Key file. The imported Key file will be used by the server to serve over HTTPS.", + "type": "string", + "enum": [ + "Key" + ] + }, + { + "description": "A trusted document that contains GraphQL operations (queries, mutations) that can be exposed a REST API using the `@rest` directive.", + "type": "string", + "enum": [ + "Operation" + ] + }, + { + "description": "Points to a Htpasswd file. The imported Htpasswd file will be used by the server to authenticate users.", + "type": "string", + "enum": [ + "Htpasswd" + ] + }, + { + "description": "Points to a Jwks file. The imported Jwks file will be used by the server to authenticate users.", + "type": "string", + "enum": [ + "Jwks" + ] + }, + { + "description": "Points to a reflection endpoint. The imported reflection endpoint will be used by the `@grpc` directive to resolve data from gRPC services.", + "type": "string", + "enum": [ + "Grpc" + ] + } + ] + }, "Method": { "type": "string", "enum": [ @@ -311,10 +383,6 @@ } } }, - "PhoneNumber": { - "title": "PhoneNumber", - "description": "Field whose value conforms to the standard E.164 format as specified in E.164 specification (https://en.wikipedia.org/wiki/E.164)." - }, "PrometheusExporter": { "description": "Output the telemetry metrics data to prometheus server", "type": "object", @@ -612,26 +680,6 @@ } ] }, - "UInt128": { - "title": "UInt128", - "description": "Field whose value is a 128-bit unsigned integer." - }, - "UInt16": { - "title": "UInt16", - "description": "Field whose value is a 16-bit unsigned integer." - }, - "UInt32": { - "title": "UInt32", - "description": "Field whose value is a 32-bit unsigned integer." - }, - "UInt64": { - "title": "UInt64", - "description": "Field whose value is a 64-bit unsigned integer." - }, - "UInt8": { - "title": "UInt8", - "description": "Field whose value is an 8-bit unsigned integer." - }, "Upstream": { "description": "The `upstream` directive allows you to control various aspects of the upstream server connection. This includes settings like connection timeouts, keep-alive intervals, and more. If not specified, default values are used.", "type": "object", @@ -778,10 +826,6 @@ } }, "additionalProperties": false - }, - "Url": { - "title": "Url", - "description": "Field whose value conforms to the standard URL format as specified in RFC 3986 (https://datatracker.ietf.org/doc/html/rfc3986)." } } } \ No newline at end of file diff --git a/src/cli/generator/source.rs b/src/cli/generator/source.rs index 1050dc4d34..caa87eb14a 100644 --- a/src/cli/generator/source.rs +++ b/src/cli/generator/source.rs @@ -1,4 +1,5 @@ -use thiserror::Error; +use crate::core::config; +use crate::core::config::SourceError; #[derive(Clone, Copy, PartialEq, Debug)] pub enum ConfigSource { @@ -6,29 +7,25 @@ pub enum ConfigSource { Yml, } -impl ConfigSource { - fn ext(&self) -> &str { - match self { - Self::Json => "json", - Self::Yml => "yml", - } - } +impl TryFrom for ConfigSource { + type Error = SourceError; - fn ends_with(&self, file: &str) -> bool { - file.ends_with(&format!(".{}", self.ext())) + fn try_from(value: config::Source) -> Result { + match value { + config::Source::Json => Ok(Self::Json), + config::Source::Yml => Ok(Self::Yml), + config::Source::GraphQL => { + Err(SourceError::UnsupportedFileFormat(value.ext().to_string())) + } + } } +} +impl ConfigSource { /// Detect the config format from the file name - pub fn detect(name: &str) -> Result { - const ALL: &[ConfigSource] = &[ConfigSource::Json, ConfigSource::Yml]; + pub fn detect(name: &str) -> Result { + let source = config::Source::detect(name)?; - ALL.iter() - .find(|format| format.ends_with(name)) - .copied() - .ok_or(UnsupportedFileFormat(name.to_string())) + ConfigSource::try_from(source) } } - -#[derive(Debug, Error, PartialEq)] -#[error("Unsupported config extension: {0}")] -pub struct UnsupportedFileFormat(String); diff --git a/src/cli/tc/init.rs b/src/cli/tc/init.rs index abbacf9e03..b7ef91a606 100644 --- a/src/cli/tc/init.rs +++ b/src/cli/tc/init.rs @@ -1,21 +1,25 @@ use std::collections::BTreeMap; use std::path::Path; -use anyhow::Result; +use anyhow::{anyhow, Result}; use super::helpers::{GRAPHQL_RC, TAILCALL_RC, TAILCALL_RC_SCHEMA}; use crate::cli::runtime::{confirm_and_write, create_directory, select_prompt}; -use crate::core::config::{Config, Expr, Field, Resolver, RootSchema, Source}; +use crate::core::config::{ + Config, Expr, Field, Link, LinkType, Resolver, RootSchema, RuntimeConfig, Source, +}; use crate::core::merge_right::MergeRight; use crate::core::runtime::TargetRuntime; use crate::core::{config, Type}; +const SCHEMA_FILENAME: &str = "main.graphql"; + pub(super) async fn init_command(runtime: TargetRuntime, folder_path: &str) -> Result<()> { create_directory(folder_path).await?; let selection = select_prompt( "Please select the format in which you want to generate the config.", - vec![Source::GraphQL, Source::Json, Source::Yml], + vec![Source::Json, Source::Yml], )?; let tailcallrc = include_str!("../../../generated/.tailcallrc.graphql"); @@ -25,30 +29,24 @@ pub(super) async fn init_command(runtime: TargetRuntime, folder_path: &str) -> R let tailcall_rc_schema = Path::new(folder_path).join(TAILCALL_RC_SCHEMA); let graphql_rc = Path::new(folder_path).join(GRAPHQL_RC); - match selection { - Source::GraphQL => { - // .tailcallrc.graphql - confirm_and_write( - runtime.clone(), - &tailcall_rc.display().to_string(), - tailcallrc.as_bytes(), - ) - .await?; - - // .graphqlrc.yml - confirm_and_write_yml(runtime.clone(), &graphql_rc).await?; - } - - Source::Json | Source::Yml => { - // .tailcallrc.schema.json - confirm_and_write( - runtime.clone(), - &tailcall_rc_schema.display().to_string(), - tailcallrc_json.as_bytes(), - ) - .await?; - } - } + // .tailcallrc.graphql + confirm_and_write( + runtime.clone(), + &tailcall_rc.display().to_string(), + tailcallrc.as_bytes(), + ) + .await?; + + // .graphqlrc.yml + confirm_and_write_yml(runtime.clone(), &graphql_rc).await?; + + // .tailcallrc.schema.json + confirm_and_write( + runtime.clone(), + &tailcall_rc_schema.display().to_string(), + tailcallrc_json.as_bytes(), + ) + .await?; create_main(runtime.clone(), folder_path, selection).await?; @@ -97,33 +95,60 @@ fn main_config() -> Config { }; Config { - server: Default::default(), - upstream: Default::default(), schema: RootSchema { query: Some("Query".to_string()), ..Default::default() }, types: BTreeMap::from([("Query".into(), query_type)]), ..Default::default() } } +fn runtime_config() -> RuntimeConfig { + let config = RuntimeConfig::default(); + + config.links(vec![Link { + id: Some("main".to_string()), + src: SCHEMA_FILENAME.to_string(), + type_of: LinkType::Config, + ..Default::default() + }]) +} + async fn create_main( runtime: TargetRuntime, folder_path: impl AsRef, source: Source, ) -> Result<()> { let config = main_config(); - - let content = match source { - Source::GraphQL => config.to_sdl(), - Source::Json => config.to_json(true)?, - Source::Yml => config.to_yaml()?, + let runtime_config = runtime_config(); + + let runtime_config = match source { + Source::Json => runtime_config.to_json(true)?, + Source::Yml => runtime_config.to_yaml()?, + _ => { + return Err(anyhow!( + "Only json/yaml formats are supported for json configs" + )) + } }; - let path = folder_path + let schema = config.to_sdl(); + + let runtime_config_path = folder_path .as_ref() .join(format!("main.{}", source.ext())) .display() .to_string(); + let schema_path = folder_path + .as_ref() + .join(SCHEMA_FILENAME) + .display() + .to_string(); - confirm_and_write(runtime.clone(), &path, content.as_bytes()).await?; + confirm_and_write( + runtime.clone(), + &runtime_config_path, + runtime_config.as_bytes(), + ) + .await?; + confirm_and_write(runtime.clone(), &schema_path, schema.as_bytes()).await?; Ok(()) } diff --git a/src/core/config/config.rs b/src/core/config/config.rs index a58715bec1..759ee0a9aa 100644 --- a/src/core/config/config.rs +++ b/src/core/config/config.rs @@ -38,8 +38,7 @@ use crate::core::scalar::Scalar; schemars::JsonSchema, MergeRight, )] -#[serde(rename_all = "camelCase")] -pub struct Config { +pub struct RuntimeConfig { /// /// Dictates how the server behaves and helps tune tailcall for all ingress /// requests. Features such as request batching, SSL, HTTP2 etc. can be @@ -53,35 +52,51 @@ pub struct Config { #[serde(default)] pub upstream: Upstream, + /// + /// A list of all links in the schema. + pub links: Vec, + + /// Enable [opentelemetry](https://opentelemetry.io) support + #[serde(default, skip_serializing_if = "is_default")] + pub telemetry: Telemetry, +} + +#[derive(Clone, Debug, Default, Setters, PartialEq, Eq, MergeRight)] +pub struct Config { + /// + /// Dictates how the server behaves and helps tune tailcall for all ingress + /// requests. Features such as request batching, SSL, HTTP2 etc. can be + /// configured here. + pub server: Server, + + /// + /// Dictates how tailcall should handle upstream requests/responses. + /// Tuning upstream can improve performance and reliability for connections. + pub upstream: Upstream, + /// /// Specifies the entry points for query and mutation in the generated /// GraphQL schema. - #[serde(skip)] pub schema: RootSchema, /// /// A map of all the types in the schema. - #[serde(skip)] #[setters(skip)] pub types: BTreeMap, /// /// A map of all the union types in the schema. - #[serde(skip)] pub unions: BTreeMap, /// /// A map of all the enum types in the schema - #[serde(skip)] pub enums: BTreeMap, /// /// A list of all links in the schema. - #[serde(skip)] pub links: Vec, /// Enable [opentelemetry](https://opentelemetry.io) support - #[serde(default, skip_serializing_if = "is_default")] pub telemetry: Telemetry, } @@ -305,7 +320,47 @@ impl Display for GraphQLOperationType { } } +impl RuntimeConfig { + pub fn from_json(json: &str) -> Result { + Ok(serde_json::from_str(json)?) + } + + pub fn from_yaml(yaml: &str) -> Result { + Ok(serde_yaml_ng::from_str(yaml)?) + } + + pub fn from_source(source: Source, config: &str) -> Result { + match source { + Source::Json => RuntimeConfig::from_json(config), + Source::Yml => RuntimeConfig::from_yaml(config), + _ => Err(anyhow!("Only the json/yaml runtime configs are supported")), + } + } + + pub fn to_yaml(&self) -> Result { + Ok(serde_yaml_ng::to_string(self)?) + } + + pub fn to_json(&self, pretty: bool) -> Result { + if pretty { + Ok(serde_json::to_string_pretty(self)?) + } else { + Ok(serde_json::to_string(self)?) + } + } +} + impl Config { + pub fn with_runtime_config(self, runtime_config: RuntimeConfig) -> Self { + Self { + server: runtime_config.server, + upstream: runtime_config.upstream, + links: runtime_config.links, + telemetry: runtime_config.telemetry, + ..self + } + } + pub fn is_root_operation_type(&self, type_name: &str) -> bool { let type_name = type_name.to_lowercase(); @@ -335,18 +390,6 @@ impl Config { self.enums.get(name) } - pub fn to_yaml(&self) -> Result { - Ok(serde_yaml_ng::to_string(self)?) - } - - pub fn to_json(&self, pretty: bool) -> Result { - if pretty { - Ok(serde_json::to_string_pretty(self)?) - } else { - Ok(serde_json::to_string(self)?) - } - } - /// Renders current config to graphQL string pub fn to_sdl(&self) -> String { crate::core::document::print(self.into()) @@ -372,14 +415,6 @@ impl Config { || self.enums.contains_key(name) } - pub fn from_json(json: &str) -> Result { - Ok(serde_json::from_str(json)?) - } - - pub fn from_yaml(yaml: &str) -> Result { - Ok(serde_yaml_ng::from_str(yaml)?) - } - pub fn from_sdl(sdl: &str) -> Valid { let doc = async_graphql::parser::parse_schema(sdl); match doc { @@ -388,10 +423,10 @@ impl Config { } } - pub fn from_source(source: Source, schema: &str) -> Result { + pub fn from_source(source: Source, content: &str) -> Result { match source { - Source::GraphQL => Ok(Config::from_sdl(schema).to_result()?), - _ => Err(anyhow!("Only the graphql config is currently supported")), + Source::GraphQL => Ok(Config::from_sdl(content).to_result()?), + source => Ok(Config::from(RuntimeConfig::from_source(source, content)?)), } } @@ -607,13 +642,9 @@ impl Config { .add_directive(Grpc::directive_definition(generated_types)) .add_directive(Http::directive_definition(generated_types)) .add_directive(JS::directive_definition(generated_types)) - .add_directive(Link::directive_definition(generated_types)) .add_directive(Modify::directive_definition(generated_types)) .add_directive(Omit::directive_definition(generated_types)) .add_directive(Protected::directive_definition(generated_types)) - .add_directive(Server::directive_definition(generated_types)) - .add_directive(Telemetry::directive_definition(generated_types)) - .add_directive(Upstream::directive_definition(generated_types)) .add_directive(Discriminate::directive_definition(generated_types)) .add_input(GraphQL::input_definition()) .add_input(Grpc::input_definition()) @@ -621,8 +652,7 @@ impl Config { .add_input(Expr::input_definition()) .add_input(JS::input_definition()) .add_input(Modify::input_definition()) - .add_input(Cache::input_definition()) - .add_input(Telemetry::input_definition()); + .add_input(Cache::input_definition()); for scalar in Scalar::iter() { builder = builder.add_scalar(scalar.scalar_definition()); @@ -632,6 +662,18 @@ impl Config { } } +impl From for Config { + fn from(config: RuntimeConfig) -> Self { + Self { + server: config.server, + upstream: config.upstream, + links: config.links, + telemetry: config.telemetry, + ..Default::default() + } + } +} + #[derive( Clone, Debug, Serialize, Deserialize, PartialEq, Eq, Hash, Default, schemars::JsonSchema, )] diff --git a/src/core/config/from_document.rs b/src/core/config/from_document.rs index b05757900f..20096936a4 100644 --- a/src/core/config/from_document.rs +++ b/src/core/config/from_document.rs @@ -12,7 +12,7 @@ use indexmap::IndexMap; use tailcall_valid::{Valid, ValidationError, Validator}; use super::directive::{to_directive, Directive}; -use super::{Alias, Discriminate, Resolver, Telemetry, FEDERATION_DIRECTIVES}; +use super::{Alias, Discriminate, Resolver, RuntimeConfig, Telemetry, FEDERATION_DIRECTIVES}; use crate::core::config::{ self, Cache, Config, Enum, Link, Modify, Omit, Protected, RootSchema, Server, Union, Upstream, Variant, @@ -51,15 +51,11 @@ pub fn from_document(doc: ServiceDocument) -> Valid { .fuse(links(sd)) .fuse(telemetry(sd)) .map( - |(server, upstream, types, unions, enums, schema, links, telemetry)| Config { - server, - upstream, - types, - unions, - enums, - schema, - links, - telemetry, + |(server, upstream, types, unions, enums, schema, links, telemetry)| { + let runtime_config = RuntimeConfig { server, upstream, links, telemetry }; + let config = Config { types, unions, enums, schema, ..Default::default() }; + + config.with_runtime_config(runtime_config) }, ) }) diff --git a/src/core/config/source.rs b/src/core/config/source.rs index 24e6adc110..010673fdb0 100644 --- a/src/core/config/source.rs +++ b/src/core/config/source.rs @@ -1,10 +1,10 @@ +use std::path::Path; +use std::str::FromStr; + use schemars::JsonSchema; use serde::{Deserialize, Serialize}; -use tailcall_valid::{ValidationError, Validator}; use thiserror::Error; -use super::Config; - #[derive(Clone, Default, Debug, Serialize, Deserialize, JsonSchema)] #[serde(rename_all = "camelCase")] pub enum Source { @@ -27,21 +27,24 @@ impl std::fmt::Display for Source { const JSON_EXT: &str = "json"; const YML_EXT: &str = "yml"; const GRAPHQL_EXT: &str = "graphql"; -const ALL: [Source; 3] = [Source::Json, Source::Yml, Source::GraphQL]; #[derive(Debug, Error, PartialEq)] -#[error("Unsupported config extension: {0}")] -pub struct UnsupportedConfigFormat(pub String); +pub enum SourceError { + #[error("Unsupported config extension: {0}")] + UnsupportedFileFormat(String), + #[error("Cannot parse")] + InvalidPath(String), +} impl std::str::FromStr for Source { - type Err = UnsupportedConfigFormat; + type Err = SourceError; fn from_str(s: &str) -> Result { match s.to_lowercase().as_str() { "json" => Ok(Source::Json), "yml" | "yaml" => Ok(Source::Yml), "graphql" | "gql" => Ok(Source::GraphQL), - _ => Err(UnsupportedConfigFormat(s.to_string())), + _ => Err(SourceError::UnsupportedFileFormat(s.to_string())), } } } @@ -56,34 +59,12 @@ impl Source { } } - fn ends_with(&self, file: &str) -> bool { - file.ends_with(&format!(".{}", self.ext())) - } - /// Detect the config format from the file name - pub fn detect(name: &str) -> Result { - ALL.into_iter() - .find(|format| format.ends_with(name)) - .ok_or(UnsupportedConfigFormat(name.to_string())) - } - - /// Encode the config to the given format - pub fn encode(&self, config: &Config) -> Result { - match self { - Source::Yml => Ok(config.to_yaml()?), - Source::GraphQL => Ok(config.to_sdl()), - Source::Json => Ok(config.to_json(true)?), - } - } - - /// Decode the config from the given data - pub fn decode(&self, data: &str) -> Result> { - match self { - Source::Yml => Config::from_yaml(data).map_err(|e| ValidationError::new(e.to_string())), - Source::GraphQL => Config::from_sdl(data).to_result(), - Source::Json => { - Config::from_json(data).map_err(|e| ValidationError::new(e.to_string())) - } - } + pub fn detect(name: &str) -> Result { + Path::new(name) + .extension() + .and_then(|ext| ext.to_str()) + .map(Source::from_str) + .ok_or(SourceError::InvalidPath(name.to_string()))? } } diff --git a/tailcall-typedefs/src/main.rs b/tailcall-typedefs/src/main.rs index aecf725b24..f147486ff8 100644 --- a/tailcall-typedefs/src/main.rs +++ b/tailcall-typedefs/src/main.rs @@ -7,14 +7,12 @@ use std::process::exit; use std::sync::Arc; use anyhow::{anyhow, Result}; -use schemars::schema::{RootSchema, Schema}; -use schemars::Map; +use schemars::schema::RootSchema; use serde_json::{json, Value}; -use strum::IntoEnumIterator; use tailcall::cli; -use tailcall::core::config::Config; +use tailcall::core::config::RuntimeConfig; use tailcall::core::tracing::default_tracing_for_name; -use tailcall::core::{scalar, FileIO}; +use tailcall::core::FileIO; static JSON_SCHEMA_FILE: &str = "generated/.tailcallrc.schema.json"; static GRAPHQL_SCHEMA_FILE: &str = "generated/.tailcallrc.graphql"; @@ -143,11 +141,7 @@ fn get_graphql_path() -> PathBuf { } fn get_updated_json() -> Result { - let mut schema: RootSchema = schemars::schema_for!(Config); - let scalar = scalar::Scalar::iter() - .map(|scalar| (scalar.name(), scalar.schema())) - .collect::>(); - schema.definitions.extend(scalar); + let schema: RootSchema = schemars::schema_for!(RuntimeConfig); let schema = json!(schema); Ok(schema) From d8947cde4a6683e5b02e82d1941f38cbb402eabf Mon Sep 17 00:00:00 2001 From: DLillard0 <53611763+DLillard0@users.noreply.github.com> Date: Tue, 17 Dec 2024 00:13:21 +0800 Subject: [PATCH 3/3] feat(link): add `proto_paths` support (#3222) --- generated/.tailcallrc.schema.json | 10 ++++++ src/core/config/directives/link.rs | 5 +++ src/core/config/reader.rs | 8 ++++- src/core/generator/generator.rs | 1 + src/core/grpc/data_loader_request.rs | 1 + src/core/grpc/protobuf.rs | 48 ++++++++++++++++++++++++++++ src/core/grpc/request_template.rs | 1 + 7 files changed, 73 insertions(+), 1 deletion(-) diff --git a/generated/.tailcallrc.schema.json b/generated/.tailcallrc.schema.json index 62a29551d1..55fbf80df1 100644 --- a/generated/.tailcallrc.schema.json +++ b/generated/.tailcallrc.schema.json @@ -269,6 +269,16 @@ "meta": { "description": "Additional metadata pertaining to the linked resource." }, + "proto_paths": { + "description": "The proto paths to be used when resolving dependencies. Only valid when [`Link::type_of`] is [`LinkType::Protobuf`]", + "type": [ + "array", + "null" + ], + "items": { + "type": "string" + } + }, "src": { "description": "The source of the link. It can be a URL or a path to a file. If a path is provided, it is relative to the file that imports the link.", "type": "string" diff --git a/src/core/config/directives/link.rs b/src/core/config/directives/link.rs index 5063812a2f..e045caf98c 100644 --- a/src/core/config/directives/link.rs +++ b/src/core/config/directives/link.rs @@ -93,4 +93,9 @@ pub struct Link { /// Additional metadata pertaining to the linked resource. #[serde(default, skip_serializing_if = "is_default")] pub meta: Option, + /// + /// The proto paths to be used when resolving dependencies. + /// Only valid when [`Link::type_of`] is [`LinkType::Protobuf`] + #[serde(default, skip_serializing_if = "is_default")] + pub proto_paths: Option>, } diff --git a/src/core/config/reader.rs b/src/core/config/reader.rs index 675c6a9d42..5240e658e8 100644 --- a/src/core/config/reader.rs +++ b/src/core/config/reader.rs @@ -79,7 +79,13 @@ impl ConfigReader { }); } LinkType::Protobuf => { - let meta = self.proto_reader.read(path, None).await?; + let proto_paths = link.proto_paths.as_ref().map(|paths| { + paths + .iter() + .map(|p| Self::resolve_path(p, parent_dir)) + .collect::>() + }); + let meta = self.proto_reader.read(path, proto_paths.as_deref()).await?; extensions.add_proto(meta); } LinkType::Script => { diff --git a/src/core/generator/generator.rs b/src/core/generator/generator.rs index 98cb81515e..5bcee1f556 100644 --- a/src/core/generator/generator.rs +++ b/src/core/generator/generator.rs @@ -99,6 +99,7 @@ impl Generator { type_of: LinkType::Protobuf, headers: None, meta: None, + proto_paths: None, }); Ok(config) } diff --git a/src/core/grpc/data_loader_request.rs b/src/core/grpc/data_loader_request.rs index 1f49188d6f..7ee4bba124 100644 --- a/src/core/grpc/data_loader_request.rs +++ b/src/core/grpc/data_loader_request.rs @@ -74,6 +74,7 @@ mod tests { type_of: LinkType::Protobuf, headers: None, meta: None, + proto_paths: None, }]); let method = GrpcMethod { package: "greetings".to_string(), diff --git a/src/core/grpc/protobuf.rs b/src/core/grpc/protobuf.rs index 979b7afb2f..36ebdffebd 100644 --- a/src/core/grpc/protobuf.rs +++ b/src/core/grpc/protobuf.rs @@ -254,6 +254,18 @@ pub mod tests { use crate::core::config::{Config, Field, Grpc, Link, LinkType, Resolver, Type}; pub async fn get_proto_file(path: &str) -> Result { + get_proto_file_with_config(path, LinkConfig::default()).await + } + + #[derive(Default)] + pub struct LinkConfig { + proto_paths: Option>, + } + + pub async fn get_proto_file_with_config( + path: &str, + link_config: LinkConfig, + ) -> Result { let runtime = crate::core::runtime::test::init(None); let reader = ConfigReader::init(runtime); @@ -268,6 +280,7 @@ pub mod tests { type_of: LinkType::Protobuf, headers: None, meta: None, + proto_paths: link_config.proto_paths, }]); let method = GrpcMethod { package: id, service: "a".to_owned(), name: "b".to_owned() }; @@ -395,6 +408,41 @@ pub mod tests { Ok(()) } + #[tokio::test] + async fn news_proto_file_with_proto_paths() -> Result<()> { + let grpc_method = GrpcMethod::try_from("news.NewsService.GetNews").unwrap(); + + let path: &str = protobuf::NEWS_PROTO_PATHS; + let proto_paths = Some(vec![Path::new(path) + .ancestors() + .nth(2) + .unwrap() + .to_string_lossy() + .to_string()]); + let file = ProtobufSet::from_proto_file( + get_proto_file_with_config(path, LinkConfig { proto_paths }).await?, + )?; + let service = file.find_service(&grpc_method)?; + let operation = service.find_operation(&grpc_method)?; + + let input = operation.convert_input(r#"{ "id": 1 }"#)?; + + assert_eq!(input, b"\0\0\0\0\x02\x08\x01"); + + let output = b"\0\0\0\x005\x08\x01\x12\x06Note 1\x1a\tContent 1\"\x0cPost image 1"; + + let parsed = operation.convert_output::(output)?; + + assert_eq!( + serde_json::to_value(parsed)?, + json!({ + "id": 1, "title": "Note 1", "body": "Content 1", "postImage": "Post image 1", "status": "PUBLISHED" + }) + ); + + Ok(()) + } + #[tokio::test] async fn oneof_proto_file() -> Result<()> { let grpc_method = GrpcMethod::try_from("oneof.OneOfService.GetOneOf").unwrap(); diff --git a/src/core/grpc/request_template.rs b/src/core/grpc/request_template.rs index b1cb5653e4..4643c5965f 100644 --- a/src/core/grpc/request_template.rs +++ b/src/core/grpc/request_template.rs @@ -160,6 +160,7 @@ mod tests { type_of: LinkType::Protobuf, headers: None, meta: None, + proto_paths: None, }]); let method = GrpcMethod { package: id.to_string(),