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

graphql.schema.GraphQLArgument Maven / Gradle / Ivy

package graphql.schema;


import graphql.DirectivesUtil;
import graphql.GraphQLContext;
import graphql.PublicApi;
import graphql.language.InputValueDefinition;
import graphql.language.Value;
import graphql.util.TraversalControl;
import graphql.util.TraverserContext;
import org.jetbrains.annotations.NotNull;
import org.jetbrains.annotations.Nullable;

import java.util.ArrayList;
import java.util.List;
import java.util.Locale;
import java.util.Map;
import java.util.function.Consumer;

import static graphql.Assert.assertNotNull;
import static graphql.Assert.assertValidName;
import static graphql.execution.ValuesResolver.getInputValueImpl;

/**
 * This defines an argument that can be supplied to a graphql field (via {@link GraphQLFieldDefinition}.
 * 

* Fields can be thought of as "functions" that take arguments and return a value. *

* See https://graphql.org/learn/queries/#arguments for more details on the concept. *

* {@link GraphQLArgument} is used in two contexts, one context is graphql queries where it represents the arguments that can be * set on a field and the other is in Schema Definition Language (SDL) where it can be used to represent the argument value instances * that have been supplied on a {@link GraphQLDirective}. *

* The difference is the 'value' and 'defaultValue' properties. In a query argument, the 'value' is never in the GraphQLArgument * object but rather in the AST direct or in the query variables map and the 'defaultValue' represents a value to use if both of these are * not present. You can think of them like a descriptor of what shape an argument might have. *

* However, with directives on SDL elements, the value is specified in AST only and transferred into the GraphQLArgument object and the * 'defaultValue' comes instead from the directive definition elsewhere in the SDL. You can think of them as 'instances' of arguments, their shape and their * specific value on that directive. *

* Originally graphql-java re-used the {@link GraphQLDirective} and {@link GraphQLArgument} * classes to do both purposes. This was a modelling mistake. New {@link GraphQLAppliedDirective} and {@link GraphQLAppliedDirectiveArgument} * classes have been introduced to better model when a directive is applied to a schema element, * as opposed to its schema definition itself. */ @PublicApi public class GraphQLArgument implements GraphQLNamedSchemaElement, GraphQLInputValueDefinition { private final String name; private final String description; private final String deprecationReason; private final GraphQLInputType originalType; private GraphQLInputType replacedType; private final InputValueWithState defaultValue; private final InputValueWithState value; private final InputValueDefinition definition; private final DirectivesUtil.DirectivesHolder directivesHolder; public static final String CHILD_TYPE = "type"; private GraphQLArgument(String name, String description, GraphQLInputType type, InputValueWithState defaultValue, InputValueWithState value, InputValueDefinition definition, List directives, List appliedDirectives, String deprecationReason) { assertValidName(name); assertNotNull(type, () -> "type can't be null"); this.name = name; this.description = description; this.originalType = type; this.defaultValue = defaultValue; this.value = value; this.definition = definition; this.deprecationReason = deprecationReason; this.directivesHolder = DirectivesUtil.DirectivesHolder.create(directives, appliedDirectives); } void replaceType(GraphQLInputType type) { this.replacedType = type; } @Override public String getName() { return name; } public GraphQLInputType getType() { return replacedType != null ? replacedType : originalType; } /** * The default value of this argument. * * @return a {@link InputValueWithState} that represents the arguments default value */ public @NotNull InputValueWithState getArgumentDefaultValue() { return defaultValue; } public boolean hasSetDefaultValue() { return defaultValue.isSet(); } public boolean hasSetValue() { return value.isSet(); } /** * This is only used for applied directives, that is when this argument is on a {@link GraphQLDirective} applied to a schema or query element * * @return an input value with state for an applied directive * * @deprecated use {@link GraphQLAppliedDirectiveArgument} instead */ @Deprecated(since = "2022-02-24") public @NotNull InputValueWithState getArgumentValue() { return value; } /** * This static helper method will give out a java value based on the semantics captured * in the {@link InputValueWithState} from {@link GraphQLArgument#getArgumentValue()} * * Note : You MUST only call this on a {@link GraphQLArgument} that is part of a fully formed schema. We need * all of the types to be resolved in order for this work correctly. * * Note: This method will return null if the value is not set or explicitly set to null. If you you to know the difference * when "not set" and "set to null" then you cant use this method. Rather you should use {@link GraphQLArgument#getArgumentValue()} * and use the {@link InputValueWithState#isNotSet()} methods to decide how to handle those values. * * @param argument the fully formed {@link GraphQLArgument} * @param the type you want it cast as * * @return a value of type T which is the java value of the argument * * @deprecated use {@link GraphQLAppliedDirectiveArgument} instead */ @Deprecated(since = "2022-02-24") public static T getArgumentValue(GraphQLArgument argument) { return getInputValueImpl(argument.getType(), argument.getArgumentValue(), GraphQLContext.getDefault(), Locale.getDefault()); } /** * This static helper method will give out a java value based on the semantics captured * in the {@link InputValueWithState} from {@link GraphQLArgument#getArgumentDefaultValue()} * * Note : You MUST only call this on a {@link GraphQLArgument} that is part of a fully formed schema. We need * all of the types to be resolved in order for this work correctly. * * Note: This method will return null if the value is not set or explicitly set to null. If you you to know the difference * when "not set" and "set to null" then you cant use this method. Rather you should use {@link GraphQLArgument#getArgumentDefaultValue()} * and use the {@link InputValueWithState#isNotSet()} methods to decide how to handle those values. * * @param argument the fully formed {@link GraphQLArgument} * @param the type you want it cast as * * @return a value of type T which is the java value of the argument default */ public static T getArgumentDefaultValue(GraphQLArgument argument) { return getInputValueImpl(argument.getType(), argument.getArgumentDefaultValue(), GraphQLContext.getDefault(), Locale.getDefault()); } public String getDescription() { return description; } public String getDeprecationReason() { return deprecationReason; } public boolean isDeprecated() { return deprecationReason != null; } public InputValueDefinition getDefinition() { return definition; } @Override public List getDirectives() { return directivesHolder.getDirectives(); } @Override public Map getDirectivesByName() { return directivesHolder.getDirectivesByName(); } @Override public Map> getAllDirectivesByName() { return directivesHolder.getAllDirectivesByName(); } @Override public GraphQLDirective getDirective(String directiveName) { return directivesHolder.getDirective(directiveName); } @Override public List getAppliedDirectives() { return directivesHolder.getAppliedDirectives(); } @Override public Map> getAllAppliedDirectivesByName() { return directivesHolder.getAllAppliedDirectivesByName(); } @Override public GraphQLAppliedDirective getAppliedDirective(String directiveName) { return directivesHolder.getAppliedDirective(directiveName); } @Override public List getChildren() { List children = new ArrayList<>(); children.add(getType()); children.addAll(directivesHolder.getDirectives()); children.addAll(directivesHolder.getAppliedDirectives()); return children; } @Override public SchemaElementChildrenContainer getChildrenWithTypeReferences() { return SchemaElementChildrenContainer.newSchemaElementChildrenContainer() .child(CHILD_TYPE, originalType) .children(CHILD_DIRECTIVES, directivesHolder.getDirectives()) .children(CHILD_APPLIED_DIRECTIVES, directivesHolder.getAppliedDirectives()) .build(); } @Override public GraphQLArgument withNewChildren(SchemaElementChildrenContainer newChildren) { return transform(builder -> builder.type(newChildren.getChildOrNull(CHILD_TYPE)) .replaceDirectives(newChildren.getChildren(CHILD_DIRECTIVES)) .replaceAppliedDirectives(newChildren.getChildren(CHILD_APPLIED_DIRECTIVES)) ); } @Override public GraphQLSchemaElement copy() { return newArgument(this).build(); } /** * {@inheritDoc} */ @Override public final boolean equals(Object o) { return super.equals(o); } /** * {@inheritDoc} */ @Override public final int hashCode() { return super.hashCode(); } /** * This helps you transform the current GraphQLArgument into another one by starting a builder with all * the current values and allows you to transform it how you want. * * @param builderConsumer the consumer code that will be given a builder to transform * * @return a new field based on calling build on that builder */ public GraphQLArgument transform(Consumer builderConsumer) { Builder builder = newArgument(this); builderConsumer.accept(builder); return builder.build(); } public static Builder newArgument() { return new Builder(); } public static Builder newArgument(GraphQLArgument existing) { return new Builder(existing); } @Override public TraversalControl accept(TraverserContext context, GraphQLTypeVisitor visitor) { return visitor.visitGraphQLArgument(this, context); } @Override public String toString() { return "GraphQLArgument{" + "name='" + name + '\'' + ", value=" + value + ", defaultValue=" + defaultValue + ", type=" + getType() + '}'; } /** * This method can be used to turn an argument that was being use as an applied argument into one. * * @return an {@link GraphQLAppliedDirectiveArgument} */ public GraphQLAppliedDirectiveArgument toAppliedArgument() { return GraphQLAppliedDirectiveArgument.newArgument() .name(name) .type(getType()) .inputValueWithState(value) .build(); } public static class Builder extends GraphqlDirectivesContainerTypeBuilder { private GraphQLInputType type; private InputValueWithState defaultValue = InputValueWithState.NOT_SET; private InputValueWithState value = InputValueWithState.NOT_SET; private String deprecationReason; private InputValueDefinition definition; public Builder() { } public Builder(GraphQLArgument existing) { this.name = existing.getName(); this.type = existing.originalType; this.value = existing.getArgumentValue(); this.defaultValue = existing.defaultValue; this.description = existing.getDescription(); this.definition = existing.getDefinition(); this.deprecationReason = existing.deprecationReason; copyExistingDirectives(existing); } public Builder definition(InputValueDefinition definition) { this.definition = definition; return this; } public Builder deprecate(String deprecationReason) { this.deprecationReason = deprecationReason; return this; } public Builder type(GraphQLInputType type) { this.type = type; return this; } /** * A legacy method that sets a default value into the argument * * @param defaultValue a default value * * @return this builder * * @deprecated use {@link #defaultValueLiteral(Value)} or {@link #defaultValueProgrammatic(Object)} */ @Deprecated(since = "2021-05-10") public Builder defaultValue(Object defaultValue) { this.defaultValue = InputValueWithState.newInternalValue(defaultValue); return this; } /** * @param defaultValue can't be null as a `null` is represented a @{@link graphql.language.NullValue} Literal * * @return this builder */ public Builder defaultValueLiteral(@NotNull Value defaultValue) { this.defaultValue = InputValueWithState.newLiteralValue(defaultValue); return this; } /** * @param defaultValue Can be null to represent null value * * @return this builder */ public Builder defaultValueProgrammatic(@Nullable Object defaultValue) { this.defaultValue = InputValueWithState.newExternalValue(defaultValue); return this; } /** * Removes the defaultValue to represent a missing default value (which is different from null) * * @return this builder */ public Builder clearDefaultValue() { this.defaultValue = InputValueWithState.NOT_SET; return this; } /** * A legacy method for setting an arguments value * * @param value the argument value * * @return this builder * * @deprecated use {@link #valueLiteral(Value)} or {@link #valueProgrammatic(Object)} */ @Deprecated(since = "2021-05-10") public Builder value(@Nullable Object value) { this.value = InputValueWithState.newInternalValue(value); return this; } /** * Sets a literal AST value as the arguments value * * @param value can't be null as a `null` is represented a @{@link graphql.language.NullValue} Literal * * @return this builder * * @deprecated use {@link GraphQLAppliedDirectiveArgument} methods instead */ @Deprecated(since = "2022-02-24") public Builder valueLiteral(@NotNull Value value) { this.value = InputValueWithState.newLiteralValue(value); return this; } /** * @param value values can be null to represent null value * * @return this builder * * @deprecated use {@link GraphQLAppliedDirectiveArgument} methods instead */ @Deprecated(since = "2022-02-24") public Builder valueProgrammatic(@Nullable Object value) { this.value = InputValueWithState.newExternalValue(value); return this; } /** * Removes the value to represent a missing value (which is different from null) * * @return this builder * * @deprecated use {@link GraphQLAppliedDirectiveArgument} methods instead */ @Deprecated(since = "2022-02-24") public Builder clearValue() { this.value = InputValueWithState.NOT_SET; return this; } // -- the following are repeated to avoid a binary incompatibility problem -- @Override public Builder replaceDirectives(List directives) { return super.replaceDirectives(directives); } @Override public Builder withDirectives(GraphQLDirective... directives) { return super.withDirectives(directives); } @Override public Builder withDirective(GraphQLDirective directive) { return super.withDirective(directive); } @Override public Builder withDirective(GraphQLDirective.Builder builder) { return super.withDirective(builder); } @Override public Builder clearDirectives() { return super.clearDirectives(); } @Override public Builder name(String name) { return super.name(name); } @Override public Builder description(String description) { return super.description(description); } public GraphQLArgument build() { assertNotNull(type, () -> "type can't be null"); return new GraphQLArgument( name, description, type, defaultValue, value, definition, sort(directives, GraphQLArgument.class, GraphQLDirective.class), sort(appliedDirectives, GraphQLScalarType.class, GraphQLAppliedDirective.class), deprecationReason ); } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy