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

graphql.execution.ValuesResolver Maven / Gradle / Ivy

There is a newer version: 230521-nf-execution
Show newest version
package graphql.execution;


import graphql.Internal;
import graphql.language.Argument;
import graphql.language.ArrayValue;
import graphql.language.NullValue;
import graphql.language.ObjectField;
import graphql.language.ObjectValue;
import graphql.language.Value;
import graphql.language.VariableDefinition;
import graphql.language.VariableReference;
import graphql.schema.Coercing;
import graphql.schema.CoercingParseValueException;
import graphql.schema.GraphQLArgument;
import graphql.schema.GraphQLEnumType;
import graphql.schema.GraphQLInputObjectField;
import graphql.schema.GraphQLInputObjectType;
import graphql.schema.GraphQLList;
import graphql.schema.GraphQLNonNull;
import graphql.schema.GraphQLScalarType;
import graphql.schema.GraphQLSchema;
import graphql.schema.GraphQLType;
import graphql.schema.visibility.GraphqlFieldVisibility;

import java.util.ArrayList;
import java.util.Collections;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import java.util.stream.Collectors;

import static graphql.Assert.assertShouldNeverHappen;
import static graphql.schema.visibility.DefaultGraphqlFieldVisibility.DEFAULT_FIELD_VISIBILITY;

@Internal
public class ValuesResolver {

    /**
     * The http://facebook.github.io/graphql/#sec-Coercing-Variable-Values says :
     *
     * 
     * 1. Let coercedValues be an empty unordered Map.
     * 2. Let variableDefinitions be the variables defined by operation.
     * 3. For each variableDefinition in variableDefinitions:
     *      a. Let variableName be the name of variableDefinition.
     *      b. Let variableType be the expected type of variableDefinition.
     *      c. Let defaultValue be the default value for variableDefinition.
     *      d. Let value be the value provided in variableValues for the name variableName.
     *      e. If value does not exist (was not provided in variableValues):
     *          i. If defaultValue exists (including null):
     *              1. Add an entry to coercedValues named variableName with the value defaultValue.
     *          ii. Otherwise if variableType is a Non‐Nullable type, throw a query error.
     *          iii. Otherwise, continue to the next variable definition.
     *      f. Otherwise, if value cannot be coerced according to the input coercion rules of variableType, throw a query error.
     *      g. Let coercedValue be the result of coercing value according to the input coercion rules of variableType.
     *      h. Add an entry to coercedValues named variableName with the value coercedValue.
     * 4. Return coercedValues.
     * 
* * @param schema the schema * @param variableDefinitions the variable definitions * @param variableValues the supplied variables * * @return coerced variable values as a map */ public Map coerceArgumentValues(GraphQLSchema schema, List variableDefinitions, Map variableValues) { GraphqlFieldVisibility fieldVisibility = schema.getFieldVisibility(); Map coercedValues = new LinkedHashMap<>(); for (VariableDefinition variableDefinition : variableDefinitions) { String variableName = variableDefinition.getName(); GraphQLType variableType = TypeFromAST.getTypeFromAST(schema, variableDefinition.getType()); // 3.e if (!variableValues.containsKey(variableName)) { Value defaultValue = variableDefinition.getDefaultValue(); if (defaultValue != null) { // 3.e.i Object coercedValue = coerceValueAst(fieldVisibility, variableType, variableDefinition.getDefaultValue(), null); coercedValues.put(variableName, coercedValue); } else if (isNonNullType(variableType)) { // 3.e.ii throw new NonNullableValueCoercedAsNullException(variableDefinition, variableType); } } else { Object value = variableValues.get(variableName); // 3.f Object coercedValue = getVariableValue(fieldVisibility, variableDefinition, variableType, value); // 3.g coercedValues.put(variableName, coercedValue); } } return coercedValues; } private Object getVariableValue(GraphqlFieldVisibility fieldVisibility, VariableDefinition variableDefinition, GraphQLType variableType, Object value) { if (value == null && variableDefinition.getDefaultValue() != null) { return coerceValueAst(fieldVisibility, variableType, variableDefinition.getDefaultValue(), null); } return coerceValue(fieldVisibility, variableDefinition, variableDefinition.getName(), variableType, value); } private boolean isNonNullType(GraphQLType variableType) { return variableType instanceof GraphQLNonNull; } public Map getArgumentValues(List argumentTypes, List arguments, Map variables) { return getArgumentValues(DEFAULT_FIELD_VISIBILITY, argumentTypes, arguments, variables); } public Map getArgumentValues(GraphqlFieldVisibility fieldVisibility, List argumentTypes, List arguments, Map variables) { if (argumentTypes.isEmpty()) { return Collections.emptyMap(); } Map result = new LinkedHashMap<>(); Map argumentMap = argumentMap(arguments); for (GraphQLArgument fieldArgument : argumentTypes) { String argName = fieldArgument.getName(); Argument argument = argumentMap.get(argName); Object value; if (argument != null) { value = coerceValueAst(fieldVisibility, fieldArgument.getType(), argument.getValue(), variables); } else { value = fieldArgument.getDefaultValue(); } // only put an arg into the result IF they specified a variable at all or // the default value ended up being something non null if (argumentMap.containsKey(argName) || value != null) { result.put(argName, value); } } return result; } private Map argumentMap(List arguments) { Map result = new LinkedHashMap<>(); for (Argument argument : arguments) { result.put(argument.getName(), argument); } return result; } @SuppressWarnings("unchecked") private Object coerceValue(GraphqlFieldVisibility fieldVisibility, VariableDefinition variableDefinition, String inputName, GraphQLType graphQLType, Object value) { try { if (graphQLType instanceof GraphQLNonNull) { Object returnValue = coerceValue(fieldVisibility, variableDefinition, inputName, ((GraphQLNonNull) graphQLType).getWrappedType(), value); if (returnValue == null) { throw new NonNullableValueCoercedAsNullException(variableDefinition, graphQLType); } return returnValue; } if (value == null) { return null; } if (graphQLType instanceof GraphQLScalarType) { return coerceValueForScalar((GraphQLScalarType) graphQLType, value); } else if (graphQLType instanceof GraphQLEnumType) { return coerceValueForEnum((GraphQLEnumType) graphQLType, value); } else if (graphQLType instanceof GraphQLList) { return coerceValueForList(fieldVisibility, variableDefinition, inputName, (GraphQLList) graphQLType, value); } else if (graphQLType instanceof GraphQLInputObjectType) { if (value instanceof Map) { return coerceValueForInputObjectType(fieldVisibility, variableDefinition, (GraphQLInputObjectType) graphQLType, (Map) value); } else { throw new CoercingParseValueException( "Expected type 'Map' but was '" + value.getClass().getSimpleName() + "'. Variables for input objects must be an instance of type 'Map'." ); } } else { return assertShouldNeverHappen("unhandled type %s", graphQLType); } } catch (CoercingParseValueException e) { if (e.getLocations() != null) { throw e; } throw new CoercingParseValueException( "Variable '" + inputName + "' has an invalid value. " + e.getMessage(), e.getCause(), variableDefinition.getSourceLocation() ); } } private Object coerceValueForInputObjectType(GraphqlFieldVisibility fieldVisibility, VariableDefinition variableDefinition, GraphQLInputObjectType inputObjectType, Map input) { Map result = new LinkedHashMap<>(); List fields = fieldVisibility.getFieldDefinitions(inputObjectType); List fieldNames = fields.stream().map(GraphQLInputObjectField::getName).collect(Collectors.toList()); for (String inputFieldName : input.keySet()) { if (!fieldNames.contains(inputFieldName)) { throw new InputMapDefinesTooManyFieldsException(inputObjectType, inputFieldName); } } for (GraphQLInputObjectField inputField : fields) { if (input.containsKey(inputField.getName()) || alwaysHasValue(inputField)) { Object value = coerceValue(fieldVisibility, variableDefinition, inputField.getName(), inputField.getType(), input.get(inputField.getName())); result.put(inputField.getName(), value == null ? inputField.getDefaultValue() : value); } } return result; } private boolean alwaysHasValue(GraphQLInputObjectField inputField) { return inputField.getDefaultValue() != null || inputField.getType() instanceof GraphQLNonNull; } private Object coerceValueForScalar(GraphQLScalarType graphQLScalarType, Object value) { return graphQLScalarType.getCoercing().parseValue(value); } private Object coerceValueForEnum(GraphQLEnumType graphQLEnumType, Object value) { return graphQLEnumType.getCoercing().parseValue(value); } private List coerceValueForList(GraphqlFieldVisibility fieldVisibility, VariableDefinition variableDefinition, String inputName, GraphQLList graphQLList, Object value) { if (value instanceof Iterable) { List result = new ArrayList<>(); for (Object val : (Iterable) value) { result.add(coerceValue(fieldVisibility, variableDefinition, inputName, graphQLList.getWrappedType(), val)); } return result; } else { return Collections.singletonList(coerceValue(fieldVisibility, variableDefinition, inputName, graphQLList.getWrappedType(), value)); } } private Object coerceValueAst(GraphqlFieldVisibility fieldVisibility, GraphQLType type, Value inputValue, Map variables) { if (inputValue instanceof VariableReference) { return variables.get(((VariableReference) inputValue).getName()); } if (inputValue instanceof NullValue) { return null; } if (type instanceof GraphQLScalarType) { return parseLiteral(inputValue, ((GraphQLScalarType) type).getCoercing()); } if (type instanceof GraphQLNonNull) { return coerceValueAst(fieldVisibility, ((GraphQLNonNull) type).getWrappedType(), inputValue, variables); } if (type instanceof GraphQLInputObjectType) { return coerceValueAstForInputObject(fieldVisibility, (GraphQLInputObjectType) type, (ObjectValue) inputValue, variables); } if (type instanceof GraphQLEnumType) { return parseLiteral(inputValue, ((GraphQLEnumType) type).getCoercing()); } if (type instanceof GraphQLList) { return coerceValueAstForList(fieldVisibility, (GraphQLList) type, inputValue, variables); } return null; } private Object parseLiteral(Value inputValue, Coercing coercing) { // the CoercingParseLiteralException exception that could happen here has been validated earlier via ValidationUtil return coercing.parseLiteral(inputValue); } private Object coerceValueAstForList(GraphqlFieldVisibility fieldVisibility, GraphQLList graphQLList, Value value, Map variables) { if (value instanceof ArrayValue) { ArrayValue arrayValue = (ArrayValue) value; List result = new ArrayList<>(); for (Value singleValue : arrayValue.getValues()) { result.add(coerceValueAst(fieldVisibility, graphQLList.getWrappedType(), singleValue, variables)); } return result; } else { return Collections.singletonList(coerceValueAst(fieldVisibility, graphQLList.getWrappedType(), value, variables)); } } private Object coerceValueAstForInputObject(GraphqlFieldVisibility fieldVisibility, GraphQLInputObjectType type, ObjectValue inputValue, Map variables) { Map result = new LinkedHashMap<>(); Map inputValueFieldsByName = mapObjectValueFieldsByName(inputValue); List inputFields = fieldVisibility.getFieldDefinitions(type); for (GraphQLInputObjectField inputTypeField : inputFields) { if (inputValueFieldsByName.containsKey(inputTypeField.getName())) { boolean putObjectInMap = true; ObjectField field = inputValueFieldsByName.get(inputTypeField.getName()); Value fieldInputValue = field.getValue(); Object fieldObject = null; if (fieldInputValue instanceof VariableReference) { String varName = ((VariableReference) fieldInputValue).getName(); if (!variables.containsKey(varName)) { putObjectInMap = false; } else { fieldObject = variables.get(varName); } } else { fieldObject = coerceValueAst(fieldVisibility, inputTypeField.getType(), fieldInputValue, variables); } if (fieldObject == null) { if (!field.getValue().isEqualTo(NullValue.Null)) { fieldObject = inputTypeField.getDefaultValue(); } } if (putObjectInMap) { result.put(field.getName(), fieldObject); } else { assertNonNullInputField(inputTypeField); } } else if (inputTypeField.getDefaultValue() != null) { result.put(inputTypeField.getName(), inputTypeField.getDefaultValue()); } else { assertNonNullInputField(inputTypeField); } } return result; } private void assertNonNullInputField(GraphQLInputObjectField inputTypeField) { if (inputTypeField.getType() instanceof GraphQLNonNull) { throw new NonNullableValueCoercedAsNullException(inputTypeField); } } private Map mapObjectValueFieldsByName(ObjectValue inputValue) { Map inputValueFieldsByName = new LinkedHashMap<>(); for (ObjectField objectField : inputValue.getObjectFields()) { inputValueFieldsByName.put(objectField.getName(), objectField); } return inputValueFieldsByName; } }