Skip to content

Commit

Permalink
Merge pull request #3070 from apollographql/clenfest/validateOverride
Browse files Browse the repository at this point in the history
Only run override validation when field has an `@override` in at least one subgraph
  • Loading branch information
sachindshinde authored Jul 9, 2024
2 parents 3dff8a3 + 67b70c6 commit bda679f
Show file tree
Hide file tree
Showing 2 changed files with 44 additions and 9 deletions.
5 changes: 5 additions & 0 deletions .changeset/good-shoes-chew.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
---
"@apollo/composition": patch
---

Add a fast path to skip override validation for fields without any subgraph `@override`s.
48 changes: 39 additions & 9 deletions composition-js/src/merging/merge.ts
Original file line number Diff line number Diff line change
Expand Up @@ -80,6 +80,7 @@ import {
StaticArgumentsTransform,
isNullableType,
isFieldDefinition,
Post20FederationDirectiveDefinition,
} from "@apollo/federation-internals";
import { ASTNode, GraphQLError, DirectiveLocation } from "graphql";
import {
Expand Down Expand Up @@ -368,13 +369,15 @@ class Merger {
private joinDirectiveIdentityURLs = new Set<string>();
private schemaToImportNameToFeatureUrl = new Map<Schema, Map<string, FeatureUrl>>();
private fieldsWithFromContext: Set<string>;
private fieldsWithOverride: Set<string>;

constructor(readonly subgraphs: Subgraphs, readonly options: CompositionOptions) {
this.latestFedVersionUsed = this.getLatestFederationVersionUsed();
this.joinSpec = JOIN_VERSIONS.getMinimumRequiredVersion(this.latestFedVersionUsed);
this.linkSpec = LINK_VERSIONS.getMinimumRequiredVersion(this.latestFedVersionUsed);
this.inaccessibleSpec = INACCESSIBLE_VERSIONS.getMinimumRequiredVersion(this.latestFedVersionUsed);
this.fieldsWithFromContext = this.getFieldsWithFromContextDirective();
this.fieldsWithOverride = this.getFieldsWithOverrideDirective();

this.names = subgraphs.names();
this.composeDirectiveManager = new ComposeDirectiveManager(
Expand Down Expand Up @@ -1346,6 +1349,9 @@ class Merger {
*/
private validateOverride(sources: Sources<FieldDefinition<any>>, dest: FieldDefinition<any>): FieldMergeContext {
const result = new FieldMergeContext(sources);
if (!this.fieldsWithOverride.has(dest.coordinate)) {
return result;
}

// For any field, we can't have more than one @override directive
type MappedValue = {
Expand Down Expand Up @@ -3437,20 +3443,44 @@ class Merger {
}

private getFieldsWithFromContextDirective(): Set<string> {
const fieldsWithFromContext = new Set<string>();
return this.getFieldsWithAppliedDirective(
(subgraph: Subgraph) => subgraph.metadata().fromContextDirective(),
(application: Directive<SchemaElement<any,any>>) => {
const field = application.parent.parent;
assert(isFieldDefinition(field), () => `Expected ${application.parent} to be a field`);
return field;
},
);
}

private getFieldsWithOverrideDirective(): Set<string> {
return this.getFieldsWithAppliedDirective(
(subgraph: Subgraph) => subgraph.metadata().overrideDirective(),
(application: Directive<SchemaElement<any,any>>) => {
const field = application.parent;
assert(isFieldDefinition(field), () => `Expected ${application.parent} to be a field`);
return field;
}
);
}

private getFieldsWithAppliedDirective(
getDirective: (subgraph: Subgraph) => Post20FederationDirectiveDefinition<any>,
getField: (application: Directive<SchemaElement<any, any>>) => FieldDefinition<any>,
) {
const fields = new Set<string>();
for (const subgraph of this.subgraphs) {
const fromContextDirective = subgraph.metadata().fromContextDirective();
if (isFederationDirectiveDefinedInSchema(fromContextDirective)) {
for (const application of fromContextDirective.applications()) {
const field = application.parent.parent;
assert(isFieldDefinition(field), () => `Expected ${application.parent.parent} to be a field`);
const directive = getDirective(subgraph);
if (isFederationDirectiveDefinedInSchema(directive)) {
for (const application of directive.applications()) {
const field = getField(application);
const coordinate = field.coordinate;
if (!fieldsWithFromContext.has(coordinate)) {
fieldsWithFromContext.add(coordinate);
if (!fields.has(coordinate)) {
fields.add(coordinate);
}
}
}
}
return fieldsWithFromContext;
return fields;
}
}

0 comments on commit bda679f

Please sign in to comment.