graphql.execution.ValuesResolver Maven / Gradle / Ivy
package graphql.execution;
import graphql.Assert;
import graphql.GraphQLContext;
import graphql.Internal;
import graphql.collect.ImmutableKit;
import graphql.execution.values.InputInterceptor;
import graphql.i18n.I18n;
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.normalized.NormalizedInputValue;
import graphql.schema.CoercingParseLiteralException;
import graphql.schema.CoercingParseValueException;
import graphql.schema.GraphQLArgument;
import graphql.schema.GraphQLCodeRegistry;
import graphql.schema.GraphQLEnumType;
import graphql.schema.GraphQLInputObjectType;
import graphql.schema.GraphQLInputType;
import graphql.schema.GraphQLList;
import graphql.schema.GraphQLScalarType;
import graphql.schema.GraphQLSchema;
import graphql.schema.GraphQLType;
import graphql.schema.InputValueWithState;
import graphql.schema.visibility.GraphqlFieldVisibility;
import org.jetbrains.annotations.NotNull;
import org.jetbrains.annotations.Nullable;
import java.util.ArrayList;
import java.util.Collections;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Locale;
import java.util.Map;
import static graphql.Assert.assertShouldNeverHappen;
import static graphql.Assert.assertTrue;
import static graphql.execution.ValuesResolver.ValueMode.NORMALIZED;
import static graphql.execution.ValuesResolverConversion.externalValueToInternalValueImpl;
import static graphql.schema.GraphQLTypeUtil.isList;
import static graphql.schema.GraphQLTypeUtil.isNonNull;
import static graphql.schema.GraphQLTypeUtil.simplePrint;
import static graphql.schema.GraphQLTypeUtil.unwrapOne;
import static graphql.schema.visibility.DefaultGraphqlFieldVisibility.DEFAULT_FIELD_VISIBILITY;
@SuppressWarnings("rawtypes")
@Internal
public class ValuesResolver {
private ValuesResolver() {
}
public enum ValueMode {
LITERAL,
NORMALIZED
}
/**
* This method coerces the "raw" variables values provided to the engine. The coerced values will be used to
* provide arguments to {@link graphql.schema.DataFetchingEnvironment}
*
* This method is called once per execution and also performs validation.
*
* @param schema the schema
* @param variableDefinitions the variable definitions
* @param rawVariables the supplied variables
* @param graphqlContext the GraphqlContext to use
* @param locale the Locale to use
*
* @return coerced variable values as a map
*/
public static CoercedVariables coerceVariableValues(GraphQLSchema schema,
List variableDefinitions,
RawVariables rawVariables,
GraphQLContext graphqlContext,
Locale locale) throws CoercingParseValueException, NonNullableValueCoercedAsNullException {
InputInterceptor inputInterceptor = graphqlContext.get(InputInterceptor.class);
return ValuesResolverConversion.externalValueToInternalValueForVariables(
inputInterceptor,
schema,
variableDefinitions,
rawVariables,
graphqlContext,
locale);
}
/**
* Normalized variables values are Literals with type information. No validation here!
*
* @param schema the schema to use
* @param variableDefinitions the list of variable definitions
* @param rawVariables the raw variables
* @param graphqlContext the GraphqlContext to use
* @param locale the Locale to use
*
* @return a map of the normalised values
*/
public static Map getNormalizedVariableValues(
GraphQLSchema schema,
List variableDefinitions,
RawVariables rawVariables,
GraphQLContext graphqlContext,
Locale locale
) {
GraphqlFieldVisibility fieldVisibility = schema.getCodeRegistry().getFieldVisibility();
Map result = new LinkedHashMap<>();
for (VariableDefinition variableDefinition : variableDefinitions) {
String variableName = variableDefinition.getName();
GraphQLType variableType = TypeFromAST.getTypeFromAST(schema, variableDefinition.getType());
assertTrue(variableType instanceof GraphQLInputType);
// can be NullValue
Value defaultValue = variableDefinition.getDefaultValue();
boolean hasValue = rawVariables.containsKey(variableName);
Object value = rawVariables.get(variableName);
if (!hasValue && defaultValue != null) {
result.put(variableName, new NormalizedInputValue(simplePrint(variableType), defaultValue));
} else if (isNonNull(variableType) && (!hasValue || value == null)) {
return assertShouldNeverHappen("variable values are expected to be valid");
} else if (hasValue) {
if (value == null) {
result.put(variableName, new NormalizedInputValue(simplePrint(variableType), null));
} else {
Object literal = ValuesResolverConversion.externalValueToLiteral(fieldVisibility, value, (GraphQLInputType) variableType, NORMALIZED, graphqlContext, locale);
result.put(variableName, new NormalizedInputValue(simplePrint(variableType), literal));
}
}
}
return result;
}
/**
* This is not used for validation: the argument literals are all validated and the variables are validated (when coerced)
*
* @param argumentTypes the list of argument types
* @param arguments the AST arguments
* @param coercedVariables the coerced variables
* @param graphqlContext the GraphqlContext to use
* @param locale the Locale to use
*
* @return a map of named argument values
*/
public static Map getArgumentValues(
List argumentTypes,
List arguments,
CoercedVariables coercedVariables,
GraphQLContext graphqlContext,
Locale locale
) {
InputInterceptor inputInterceptor = graphqlContext.get(InputInterceptor.class);
return getArgumentValuesImpl(inputInterceptor, DEFAULT_FIELD_VISIBILITY, argumentTypes, arguments, coercedVariables, graphqlContext, locale);
}
/**
* No validation as the arguments are assumed valid
*
* @param argumentTypes the list of argument types
* @param arguments the AST arguments
* @param normalizedVariables the normalised variables
*
* @return a map of named normalised values
*/
public static Map getNormalizedArgumentValues(
List argumentTypes,
List arguments,
Map normalizedVariables
) {
if (argumentTypes.isEmpty()) {
return ImmutableKit.emptyMap();
}
Map result = new LinkedHashMap<>();
Map argumentMap = argumentMap(arguments);
for (GraphQLArgument argumentDefinition : argumentTypes) {
String argumentName = argumentDefinition.getName();
Argument argument = argumentMap.get(argumentName);
if (argument == null) {
continue;
}
// If a variable doesn't exist then we can't put it into the result Map
if (isVariableAbsent(argument.getValue(), normalizedVariables)) {
continue;
}
GraphQLInputType argumentType = argumentDefinition.getType();
Object value = literalToNormalizedValue(DEFAULT_FIELD_VISIBILITY, argumentType, argument.getValue(), normalizedVariables);
result.put(argumentName, new NormalizedInputValue(simplePrint(argumentType), value));
}
return result;
}
public static Map getArgumentValues(
GraphQLCodeRegistry codeRegistry,
List argumentTypes,
List arguments,
CoercedVariables coercedVariables,
GraphQLContext graphqlContext,
Locale locale
) {
InputInterceptor inputInterceptor = graphqlContext.get(InputInterceptor.class);
return getArgumentValuesImpl(inputInterceptor, codeRegistry.getFieldVisibility(), argumentTypes, arguments, coercedVariables, graphqlContext, locale);
}
/**
* Takes a value which can be in different states (internal, literal, external value) and converts into Literal
*
* This assumes the value is valid!
*
* @param fieldVisibility the field visibility to use
* @param inputValueWithState the input value
* @param type the type of input value
* @param graphqlContext the GraphqlContext to use
* @param locale the Locale to use
*
* @return a value converted to a literal
*/
public static Value> valueToLiteral(
@NotNull GraphqlFieldVisibility fieldVisibility,
@NotNull InputValueWithState inputValueWithState,
@NotNull GraphQLType type,
GraphQLContext graphqlContext,
Locale locale
) {
return (Value>) ValuesResolverConversion.valueToLiteralImpl(
fieldVisibility,
inputValueWithState,
type,
ValueMode.LITERAL,
graphqlContext,
locale);
}
public static Value> valueToLiteral(
@NotNull InputValueWithState inputValueWithState,
@NotNull GraphQLType type,
GraphQLContext graphqlContext,
Locale locale
) {
return (Value>) ValuesResolverConversion.valueToLiteralImpl(
DEFAULT_FIELD_VISIBILITY,
inputValueWithState,
type,
ValueMode.LITERAL,
graphqlContext,
locale);
}
public static Object valueToInternalValue(
InputValueWithState inputValueWithState,
GraphQLInputType inputType,
GraphQLContext graphqlContext,
Locale locale
) throws CoercingParseValueException, CoercingParseLiteralException {
InputInterceptor inputInterceptor = graphqlContext.get(InputInterceptor.class);
return ValuesResolverConversion.valueToInternalValueImpl(
inputInterceptor,
inputValueWithState,
inputType,
graphqlContext,
locale);
}
/**
* Converts an external value to an internal value
*
* @param fieldVisibility the field visibility to use
* @param externalValue the input external value
* @param type the type of input value
* @param graphqlContext the GraphqlContext to use
* @param locale the Locale to use
*
* @return a value converted to an internal value
*/
public static Object externalValueToInternalValue(
GraphqlFieldVisibility fieldVisibility,
Object externalValue,
GraphQLInputType type,
GraphQLContext graphqlContext,
Locale locale
) {
InputInterceptor inputInterceptor = graphqlContext.get(InputInterceptor.class);
return externalValueToInternalValueImpl(
inputInterceptor,
fieldVisibility,
type,
externalValue,
graphqlContext,
locale);
}
@Nullable
@SuppressWarnings("unchecked")
public static T getInputValueImpl(
GraphQLInputType inputType,
InputValueWithState inputValue,
GraphQLContext graphqlContext,
Locale locale
) {
if (inputValue.isNotSet()) {
return null;
}
return (T) valueToInternalValue(
inputValue,
inputType,
graphqlContext,
locale);
}
private static Map getArgumentValuesImpl(
InputInterceptor inputInterceptor,
GraphqlFieldVisibility fieldVisibility,
List argumentTypes,
List arguments,
CoercedVariables coercedVariables,
GraphQLContext graphqlContext,
Locale locale
) {
if (argumentTypes.isEmpty()) {
return ImmutableKit.emptyMap();
}
Map coercedValues = new LinkedHashMap<>();
Map argumentMap = argumentMap(arguments);
for (GraphQLArgument argumentDefinition : argumentTypes) {
GraphQLInputType argumentType = argumentDefinition.getType();
String argumentName = argumentDefinition.getName();
Argument argument = argumentMap.get(argumentName);
InputValueWithState defaultValue = argumentDefinition.getArgumentDefaultValue();
boolean hasValue = argument != null;
Object value;
Value argumentValue = argument != null ? argument.getValue() : null;
if (argumentValue instanceof VariableReference) {
String variableName = ((VariableReference) argumentValue).getName();
hasValue = coercedVariables.containsKey(variableName);
value = coercedVariables.get(variableName);
} else {
value = argumentValue;
}
if (!hasValue && argumentDefinition.hasSetDefaultValue()) {
Object coercedDefaultValue = ValuesResolverConversion.defaultValueToInternalValue(
inputInterceptor,
fieldVisibility,
defaultValue,
argumentType,
graphqlContext,
locale);
coercedValues.put(argumentName, coercedDefaultValue);
} else if (isNonNull(argumentType) && (!hasValue || ValuesResolverConversion.isNullValue(value))) {
throw new NonNullableValueCoercedAsNullException(argumentDefinition);
} else if (hasValue) {
if (ValuesResolverConversion.isNullValue(value)) {
coercedValues.put(argumentName, value);
} else if (argumentValue instanceof VariableReference) {
coercedValues.put(argumentName, value);
} else {
value = ValuesResolverConversion.literalToInternalValue(inputInterceptor,
fieldVisibility,
argumentType,
argument.getValue(),
coercedVariables,
graphqlContext,
locale);
coercedValues.put(argumentName, value);
}
// @oneOf input must be checked now that all variables and literals have been converted
if (argumentType instanceof GraphQLInputObjectType) {
GraphQLInputObjectType inputObjectType = (GraphQLInputObjectType) argumentType;
if (inputObjectType.isOneOf() && ! ValuesResolverConversion.isNullValue(value)) {
validateOneOfInputTypes(inputObjectType, argumentValue, argumentName, value, locale);
}
}
}
}
return coercedValues;
}
@SuppressWarnings("unchecked")
private static void validateOneOfInputTypes(GraphQLInputObjectType oneOfInputType, Value argumentValue, String argumentName, Object inputValue, Locale locale) {
Assert.assertTrue(inputValue instanceof Map, () -> String.format("The coerced argument %s GraphQLInputObjectType is unexpectedly not a map", argumentName));
Map objectMap = (Map) inputValue;
int mapSize;
if (argumentValue instanceof ObjectValue) {
mapSize = ((ObjectValue) argumentValue).getObjectFields().size();
} else {
mapSize = objectMap.size();
}
if (mapSize != 1) {
String msg = I18n.i18n(I18n.BundleType.Execution, locale)
.msg("Execution.handleOneOfNotOneFieldError", oneOfInputType.getName());
throw new OneOfTooManyKeysException(msg);
}
String fieldName = objectMap.keySet().iterator().next();
if (objectMap.get(fieldName) == null) {
String msg = I18n.i18n(I18n.BundleType.Execution, locale)
.msg("Execution.handleOneOfValueIsNullError", oneOfInputType.getName() + "." + fieldName);
throw new OneOfNullValueException(msg);
}
}
private static Map argumentMap(List arguments) {
Map result = new LinkedHashMap<>(arguments.size());
for (Argument argument : arguments) {
result.put(argument.getName(), argument);
}
return result;
}
public static Object literalToNormalizedValue(GraphqlFieldVisibility fieldVisibility,
GraphQLType type,
Value inputValue,
Map normalizedVariables
) {
if (inputValue instanceof VariableReference) {
String varName = ((VariableReference) inputValue).getName();
return normalizedVariables.get(varName).getValue();
}
if (inputValue instanceof NullValue) {
return null;
}
if (type instanceof GraphQLScalarType) {
return inputValue;
}
if (isNonNull(type)) {
return literalToNormalizedValue(
fieldVisibility,
unwrapOne(type),
inputValue,
normalizedVariables);
}
if (type instanceof GraphQLInputObjectType) {
return literalToNormalizedValueForInputObject(
fieldVisibility,
(GraphQLInputObjectType) type,
(ObjectValue) inputValue,
normalizedVariables);
}
if (type instanceof GraphQLEnumType) {
return inputValue;
}
if (isList(type)) {
return literalToNormalizedValueForList(
fieldVisibility,
(GraphQLList) type,
inputValue,
normalizedVariables);
}
return null;
}
private static Object literalToNormalizedValueForInputObject(GraphqlFieldVisibility fieldVisibility,
GraphQLInputObjectType type,
ObjectValue inputObjectLiteral,
Map normalizedVariables) {
Map result = new LinkedHashMap<>();
for (ObjectField field : inputObjectLiteral.getObjectFields()) {
// If a variable doesn't exist then we can't put it into the result Map
if (isVariableAbsent(field.getValue(), normalizedVariables)) {
continue;
}
GraphQLInputType fieldType = type.getField(field.getName()).getType();
Object fieldValue = literalToNormalizedValue(
fieldVisibility,
fieldType,
field.getValue(),
normalizedVariables);
result.put(field.getName(), new NormalizedInputValue(simplePrint(fieldType), fieldValue));
}
return result;
}
private static List