All Downloads are FREE. Search and download functionalities are using the official Maven repository.

graphql.normalized.ExecutableNormalizedOperation Maven / Gradle / Ivy

package graphql.normalized;

import com.google.common.collect.ImmutableListMultimap;
import graphql.Assert;
import graphql.PublicApi;
import graphql.execution.MergedField;
import graphql.execution.ResultPath;
import graphql.execution.directives.QueryDirectives;
import graphql.language.Field;
import graphql.language.OperationDefinition;
import graphql.schema.FieldCoordinates;
import graphql.schema.GraphQLFieldsContainer;

import java.util.List;
import java.util.Map;

/**
 * A {@link ExecutableNormalizedOperation} represent how the text of a graphql operation (sometimes known colloquially as a query)
 * will be executed at runtime according to the graphql specification.  It handles complex mechanisms like merging
 * duplicate fields into one and also detecting when the types of a given field may actually be for more than one possible object
 * type.
 * 

* An operation consists of a list of {@link ExecutableNormalizedField}s in a parent child hierarchy */ @PublicApi public class ExecutableNormalizedOperation { private final OperationDefinition.Operation operation; private final String operationName; private final List topLevelFields; private final ImmutableListMultimap fieldToNormalizedField; private final Map normalizedFieldToMergedField; private final Map normalizedFieldToQueryDirectives; private final ImmutableListMultimap coordinatesToNormalizedFields; public ExecutableNormalizedOperation( OperationDefinition.Operation operation, String operationName, List topLevelFields, ImmutableListMultimap fieldToNormalizedField, Map normalizedFieldToMergedField, Map normalizedFieldToQueryDirectives, ImmutableListMultimap coordinatesToNormalizedFields ) { this.operation = operation; this.operationName = operationName; this.topLevelFields = topLevelFields; this.fieldToNormalizedField = fieldToNormalizedField; this.normalizedFieldToMergedField = normalizedFieldToMergedField; this.normalizedFieldToQueryDirectives = normalizedFieldToQueryDirectives; this.coordinatesToNormalizedFields = coordinatesToNormalizedFields; } /** * @return operation AST being executed */ public OperationDefinition.Operation getOperation() { return operation; } /** * @return the operation name, which can be null */ public String getOperationName() { return operationName; } /** * This multimap shows how a given {@link ExecutableNormalizedField} maps to a one or more field coordinate in the schema * * @return a multimap of fields to schema field coordinates */ public ImmutableListMultimap getCoordinatesToNormalizedFields() { return coordinatesToNormalizedFields; } /** * @return a list of the top level {@link ExecutableNormalizedField}s in this operation. */ public List getTopLevelFields() { return topLevelFields; } /** * This is a multimap and the size of it reflects all the normalized fields in the operation * * @return an immutable list multimap of {@link Field} to {@link ExecutableNormalizedField} */ public ImmutableListMultimap getFieldToNormalizedField() { return fieldToNormalizedField; } /** * Looks up one or more {@link ExecutableNormalizedField}s given a {@link Field} AST element in the operation * * @param field the field to look up * * @return zero, one or more possible {@link ExecutableNormalizedField}s that represent that field */ public List getNormalizedFields(Field field) { return fieldToNormalizedField.get(field); } /** * @return a map of {@link ExecutableNormalizedField} to {@link MergedField}s */ public Map getNormalizedFieldToMergedField() { return normalizedFieldToMergedField; } /** * Looks up the {@link MergedField} given a {@link ExecutableNormalizedField} * * @param executableNormalizedField the field to use the key * * @return a {@link MergedField} or null if its not present */ public MergedField getMergedField(ExecutableNormalizedField executableNormalizedField) { return normalizedFieldToMergedField.get(executableNormalizedField); } /** * @return a map of {@link ExecutableNormalizedField} to its {@link QueryDirectives} */ public Map getNormalizedFieldToQueryDirectives() { return normalizedFieldToQueryDirectives; } /** * This looks up the {@link QueryDirectives} associated with the given {@link ExecutableNormalizedField} * * @param executableNormalizedField the executable normalised field in question * * @return the fields query directives or null */ public QueryDirectives getQueryDirectives(ExecutableNormalizedField executableNormalizedField) { return normalizedFieldToQueryDirectives.get(executableNormalizedField); } /** * This will find a {@link ExecutableNormalizedField} given a merged field and a result path. If this does not find a field it will assert with an exception * * @param mergedField the merged field * @param fieldsContainer the containing type of that field * @param resultPath the result path in play * * @return the ExecutableNormalizedField */ public ExecutableNormalizedField getNormalizedField(MergedField mergedField, GraphQLFieldsContainer fieldsContainer, ResultPath resultPath) { List executableNormalizedFields = fieldToNormalizedField.get(mergedField.getSingleField()); List keysOnlyPath = resultPath.getKeysOnly(); for (ExecutableNormalizedField executableNormalizedField : executableNormalizedFields) { if (executableNormalizedField.getListOfResultKeys().equals(keysOnlyPath)) { if (executableNormalizedField.getObjectTypeNames().contains(fieldsContainer.getName())) { return executableNormalizedField; } } } return Assert.assertShouldNeverHappen("normalized field not found"); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy