Skip to content

Commit

Permalink
Merge branch 'dev' into geal/subgraph-request-id
Browse files Browse the repository at this point in the history
  • Loading branch information
Geal authored Oct 4, 2024
2 parents 33fea07 + 5a1249c commit c702347
Show file tree
Hide file tree
Showing 22 changed files with 862 additions and 218 deletions.
23 changes: 0 additions & 23 deletions .github/workflows/github_projects_tagger.yml

This file was deleted.

35 changes: 35 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -90,6 +90,41 @@ By [@goto-bus-stop](https://github.com/goto-bus-stop) in https://github.com/apol

## 🚀 Features

### Entity cache invalidation preview ([PR #5889](https://github.com/apollographql/router/pull/5889))

> ⚠️ This is a preview for an [Enterprise feature](https://www.apollographql.com/blog/platform/evaluating-apollo-router-understanding-free-and-open-vs-commercial-features/) of the Apollo Router. It requires an organization with a [GraphOS Enterprise plan](https://www.apollographql.com/pricing/). If your organization doesn't currently have an Enterprise plan, you can test out this functionality with a [free Enterprise trial](https://studio.apollographql.com/signup?type=enterprise-trial).
>
> As a preview feature, it's subject to our [Preview launch stage](https://www.apollographql.com/docs/resources/product-launch-stages/#preview) expectations and configuration and performance may change in future releases.

As a follow up to the Entity cache preview that was published in the router 1.46.0 release, we're introducing a new feature that allows you to invalidate cached entries.

This introduces two ways to invalidate cached entries:
- through an HTTP endpoint exposed by the router
- via GraphQL response `extensions` returned from subgraph requests

The invalidation endpoint can be defined in the router's configuration, as follows:

```yaml
preview_entity_cache:
enabled: true

# global invalidation configuration
invalidation:
# address of the invalidation endpoint
# this should only be exposed to internal networks
listen: "127.0.0.1:3000"
path: "/invalidation"
```

Invalidation requests can target cached entries by:
- subgraph name
- subgraph name and type name
- subgraph name, type name and entity key

You can learn more about invalidation in the [documentation](https://www.apollographql.com/docs/router/configuration/entity-caching#entity-cache-invalidation).

By [@bnjjj](https://github.com/bnjjj),[@bryncooke](https://github.com/bryncooke), [@garypen](https://github.com/garypen), [@Geal](https://github.com/Geal), [@IvanGoncharov](https://github.com/IvanGoncharov) in https://github.com/apollographql/router/pull/5889

### Support aliasing standard attributes for telemetry ([Issue #5930](https://github.com/apollographql/router/issues/5930))

The router now supports creating aliases for standard attributes for telemetry.
Expand Down
31 changes: 31 additions & 0 deletions apollo-federation/src/error/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,37 @@ use lazy_static::lazy_static;

use crate::subgraph::spec::FederationSpecError;

/// Break out of the current function, returning an internal error.
#[macro_export]
macro_rules! internal_error {
( $( $arg:tt )+ ) => {
return Err($crate::error::FederationError::internal(format!( $( $arg )+ )).into());
}
}

/// A safe assertion: in debug mode, it panicks on failure, and in production, it returns an
/// internal error.
///
/// Treat this as an assertion. It must only be used for conditions that *should never happen*
/// in normal operation.
#[macro_export]
macro_rules! ensure {
( $expr:expr, $( $arg:tt )+ ) => {
#[cfg(debug_assertions)]
{
if false {
return Err($crate::error::FederationError::internal("ensure!() must be used in a function that returns a Result").into());
}
assert!($expr, $( $arg )+);
}

#[cfg(not(debug_assertions))]
if !$expr {
$crate::internal_error!( $( $arg )+ );
}
}
}

// What we really needed here was the string representations in enum form, this isn't meant to
// replace AST components.
#[derive(Clone, Debug, strum_macros::Display)]
Expand Down
148 changes: 64 additions & 84 deletions apollo-federation/src/operation/merging.rs
Original file line number Diff line number Diff line change
Expand Up @@ -15,7 +15,9 @@ use super::NamedFragments;
use super::Selection;
use super::SelectionSet;
use super::SelectionValue;
use crate::ensure;
use crate::error::FederationError;
use crate::internal_error;

impl<'a> FieldSelectionValue<'a> {
/// Merges the given field selections into this one.
Expand All @@ -36,31 +38,29 @@ impl<'a> FieldSelectionValue<'a> {
let mut selection_sets = vec![];
for other in others {
let other_field = &other.field;
if other_field.schema != self_field.schema {
return Err(FederationError::internal(
"Cannot merge field selections from different schemas",
));
}
if other_field.field_position != self_field.field_position {
return Err(FederationError::internal(format!(
"Cannot merge field selection for field \"{}\" into a field selection for field \"{}\"",
other_field.field_position,
self_field.field_position,
)));
}
ensure!(
other_field.schema == self_field.schema,
"Cannot merge field selections from different schemas",
);
ensure!(
other_field.field_position == self_field.field_position,
"Cannot merge field selection for field \"{}\" into a field selection for field \"{}\"",
other_field.field_position,
self_field.field_position,
);
if self.get().selection_set.is_some() {
let Some(other_selection_set) = &other.selection_set else {
return Err(FederationError::internal(format!(
internal_error!(
"Field \"{}\" has composite type but not a selection set",
other_field.field_position,
)));
);
};
selection_sets.push(other_selection_set);
} else if other.selection_set.is_some() {
return Err(FederationError::internal(format!(
internal_error!(
"Field \"{}\" has non-composite type but also has a selection set",
other_field.field_position,
)));
);
}
}
if let Some(self_selection_set) = self.get_selection_set_mut() {
Expand All @@ -87,22 +87,16 @@ impl<'a> InlineFragmentSelectionValue<'a> {
let mut selection_sets = vec![];
for other in others {
let other_inline_fragment = &other.inline_fragment;
if other_inline_fragment.schema != self_inline_fragment.schema {
return Err(FederationError::internal(
"Cannot merge inline fragment from different schemas",
));
}
if other_inline_fragment.parent_type_position
!= self_inline_fragment.parent_type_position
{
return Err(FederationError::internal(
format!(
"Cannot merge inline fragment of parent type \"{}\" into an inline fragment of parent type \"{}\"",
other_inline_fragment.parent_type_position,
self_inline_fragment.parent_type_position,
),
));
}
ensure!(
other_inline_fragment.schema == self_inline_fragment.schema,
"Cannot merge inline fragment from different schemas",
);
ensure!(
other_inline_fragment.parent_type_position == self_inline_fragment.parent_type_position,
"Cannot merge inline fragment of parent type \"{}\" into an inline fragment of parent type \"{}\"",
other_inline_fragment.parent_type_position,
self_inline_fragment.parent_type_position,
);
selection_sets.push(&other.selection_set);
}
self.get_selection_set_mut()
Expand All @@ -127,11 +121,10 @@ impl<'a> FragmentSpreadSelectionValue<'a> {
let self_fragment_spread = &self.get().spread;
for other in others {
let other_fragment_spread = &other.spread;
if other_fragment_spread.schema != self_fragment_spread.schema {
return Err(FederationError::internal(
"Cannot merge fragment spread from different schemas",
));
}
ensure!(
other_fragment_spread.schema == self_fragment_spread.schema,
"Cannot merge fragment spread from different schemas",
);
// Nothing to do since the fragment spread is already part of the selection set.
// Fragment spreads are uniquely identified by fragment name and applied directives.
// Since there is already an entry for the same fragment spread, there is no point
Expand All @@ -157,20 +150,16 @@ impl SelectionSet {
) -> Result<(), FederationError> {
let mut selections_to_merge = vec![];
for other in others {
if other.schema != self.schema {
return Err(FederationError::internal(
"Cannot merge selection sets from different schemas",
));
}
if other.type_position != self.type_position {
return Err(FederationError::internal(
format!(
"Cannot merge selection set for type \"{}\" into a selection set for type \"{}\"",
other.type_position,
self.type_position,
),
));
}
ensure!(
other.schema == self.schema,
"Cannot merge selection sets from different schemas",
);
ensure!(
other.type_position == self.type_position,
"Cannot merge selection set for type \"{}\" into a selection set for type \"{}\"",
other.type_position,
self.type_position,
);
selections_to_merge.extend(other.selections.values());
}
self.merge_selections_into(selections_to_merge.into_iter())
Expand Down Expand Up @@ -198,12 +187,10 @@ impl SelectionSet {
selection_map::Entry::Occupied(existing) => match existing.get() {
Selection::Field(self_field_selection) => {
let Selection::Field(other_field_selection) = other_selection else {
return Err(FederationError::internal(
format!(
"Field selection key for field \"{}\" references non-field selection",
self_field_selection.field.field_position,
),
));
internal_error!(
"Field selection key for field \"{}\" references non-field selection",
self_field_selection.field.field_position,
);
};
fields
.entry(other_key)
Expand All @@ -214,12 +201,10 @@ impl SelectionSet {
let Selection::FragmentSpread(other_fragment_spread_selection) =
other_selection
else {
return Err(FederationError::internal(
format!(
"Fragment spread selection key for fragment \"{}\" references non-field selection",
self_fragment_spread_selection.spread.fragment_name,
),
));
internal_error!(
"Fragment spread selection key for fragment \"{}\" references non-field selection",
self_fragment_spread_selection.spread.fragment_name,
);
};
fragment_spreads
.entry(other_key)
Expand All @@ -230,17 +215,15 @@ impl SelectionSet {
let Selection::InlineFragment(other_inline_fragment_selection) =
other_selection
else {
return Err(FederationError::internal(
format!(
"Inline fragment selection key under parent type \"{}\" {}references non-field selection",
self_inline_fragment_selection.inline_fragment.parent_type_position,
self_inline_fragment_selection.inline_fragment.type_condition_position.clone()
.map_or_else(
String::new,
|cond| format!("(type condition: {}) ", cond),
),
),
));
internal_error!(
"Inline fragment selection key under parent type \"{}\" {}references non-field selection",
self_inline_fragment_selection.inline_fragment.parent_type_position,
self_inline_fragment_selection.inline_fragment.type_condition_position.clone()
.map_or_else(
String::new,
|cond| format!("(type condition: {}) ", cond),
),
);
};
inline_fragments
.entry(other_key)
Expand Down Expand Up @@ -306,9 +289,8 @@ impl SelectionSet {
&mut self,
selection: &Selection,
) -> Result<(), FederationError> {
debug_assert_eq!(
&self.schema,
selection.schema(),
ensure!(
self.schema == *selection.schema(),
"In order to add selection it needs to point to the same schema"
);
self.merge_selections_into(std::iter::once(selection))
Expand All @@ -328,12 +310,12 @@ impl SelectionSet {
&mut self,
selection_set: &SelectionSet,
) -> Result<(), FederationError> {
debug_assert_eq!(
self.schema, selection_set.schema,
ensure!(
self.schema == selection_set.schema,
"In order to add selection set it needs to point to the same schema."
);
debug_assert_eq!(
self.type_position, selection_set.type_position,
ensure!(
self.type_position == selection_set.type_position,
"In order to add selection set it needs to point to the same type position"
);
self.merge_into(std::iter::once(selection_set))
Expand Down Expand Up @@ -386,9 +368,7 @@ pub(crate) fn merge_selection_sets(
mut selection_sets: Vec<SelectionSet>,
) -> Result<SelectionSet, FederationError> {
let Some((first, remainder)) = selection_sets.split_first_mut() else {
return Err(FederationError::internal(
"merge_selection_sets(): must have at least one selection set",
));
internal_error!("merge_selection_sets(): must have at least one selection set");
};
first.merge_into(remainder.iter())?;

Expand Down
14 changes: 14 additions & 0 deletions apollo-federation/src/operation/optimize.rs
Original file line number Diff line number Diff line change
Expand Up @@ -1684,6 +1684,20 @@ impl FragmentGenerator {
SelectionValue::InlineFragment(mut candidate) => {
self.visit_selection_set(candidate.get_selection_set_mut())?;

// XXX(@goto-bus-stop): This is temporary to support mismatch testing with JS!
// JS federation does not consider fragments without a type condition.
if candidate
.get()
.inline_fragment
.type_condition_position
.is_none()
{
new_selection_set.add_local_selection(&Selection::InlineFragment(
Arc::clone(candidate.get()),
))?;
continue;
}

let directives = &candidate.get().inline_fragment.directives;
let skip_include = directives
.iter()
Expand Down
9 changes: 5 additions & 4 deletions apollo-federation/src/operation/rebase.rs
Original file line number Diff line number Diff line change
Expand Up @@ -22,6 +22,7 @@ use super::Selection;
use super::SelectionId;
use super::SelectionSet;
use super::TYPENAME_FIELD;
use crate::ensure;
use crate::error::FederationError;
use crate::schema::position::CompositeTypeDefinitionPosition;
use crate::schema::position::OutputTypeDefinitionPosition;
Expand Down Expand Up @@ -376,12 +377,12 @@ impl FragmentSpread {
}
.into());
};
debug_assert_eq!(
*schema, self.schema,
ensure!(
*schema == self.schema,
"Fragment spread should only be rebased within the same subgraph"
);
debug_assert_eq!(
*schema, named_fragment.schema,
ensure!(
*schema == named_fragment.schema,
"Referenced named fragment should've been rebased for the subgraph"
);
if runtime_types_intersect(
Expand Down
Loading

0 comments on commit c702347

Please sign in to comment.