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

graphql.schema.GraphQLDirective Maven / Gradle / Ivy

package graphql.schema;


import com.google.common.collect.ImmutableList;
import graphql.PublicApi;
import graphql.language.DirectiveDefinition;
import graphql.util.TraversalControl;
import graphql.util.TraverserContext;

import java.util.Collections;
import java.util.EnumSet;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import java.util.function.Consumer;
import java.util.function.UnaryOperator;

import static graphql.Assert.assertNotEmpty;
import static graphql.Assert.assertNotNull;
import static graphql.Assert.assertValidName;
import static graphql.introspection.Introspection.DirectiveLocation;
import static graphql.util.FpKit.getByName;

/**
 * A directive can be used to modify the behavior of a graphql field or type.
 * 

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

* A directive has a definition, that is what arguments it takes, and it can also be applied * to other schema elements. 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 GraphQLDirective implements GraphQLNamedSchemaElement { private final String name; private final boolean repeatable; private final String description; private final EnumSet locations; private final ImmutableList arguments; private final DirectiveDefinition definition; public static final String CHILD_ARGUMENTS = "arguments"; private GraphQLDirective(String name, String description, boolean repeatable, EnumSet locations, List arguments, DirectiveDefinition definition) { assertValidName(name); assertNotNull(arguments, () -> "arguments can't be null"); assertNotEmpty(locations, () -> "locations can't be empty"); this.name = name; this.description = description; this.repeatable = repeatable; this.locations = locations; this.arguments = ImmutableList.copyOf(arguments); this.definition = definition; } @Override public String getName() { return name; } public boolean isRepeatable() { return repeatable; } public boolean isNonRepeatable() { return !repeatable; } public List getArguments() { return arguments; } public GraphQLArgument getArgument(String name) { for (GraphQLArgument argument : arguments) { if (argument.getName().equals(name)) { return argument; } } return null; } public EnumSet validLocations() { return EnumSet.copyOf(locations); } public String getDescription() { return description; } public DirectiveDefinition getDefinition() { return definition; } @Override public String toString() { return "GraphQLDirective{" + "name='" + name + '\'' + ", repeatable='" + repeatable + '\'' + ", arguments=" + arguments + ", locations=" + locations + '}'; } /** * This helps you transform the current GraphQLDirective 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 GraphQLDirective transform(Consumer builderConsumer) { Builder builder = newDirective(this); builderConsumer.accept(builder); return builder.build(); } @Override public GraphQLSchemaElement copy() { return newDirective(this).build(); } @Override public TraversalControl accept(TraverserContext context, GraphQLTypeVisitor visitor) { return visitor.visitGraphQLDirective(this, context); } @Override public List getChildren() { return ImmutableList.copyOf(arguments); } @Override public SchemaElementChildrenContainer getChildrenWithTypeReferences() { return SchemaElementChildrenContainer.newSchemaElementChildrenContainer() .children(CHILD_ARGUMENTS, arguments) .build(); } @Override public GraphQLDirective withNewChildren(SchemaElementChildrenContainer newChildren) { return transform(builder -> builder.replaceArguments(newChildren.getChildren(CHILD_ARGUMENTS)) ); } /** * This method can be used to turn a directive that was being use as an applied directive into one. * @return an {@link GraphQLAppliedDirective} */ public GraphQLAppliedDirective toAppliedDirective() { GraphQLAppliedDirective.Builder builder = GraphQLAppliedDirective.newDirective(); builder.name(this.name); for (GraphQLArgument argument : arguments) { builder.argument(argument.toAppliedArgument()); } return builder.build(); } /** * {@inheritDoc} */ @Override public final boolean equals(Object o) { return super.equals(o); } /** * {@inheritDoc} */ @Override public final int hashCode() { return super.hashCode(); } public static Builder newDirective() { return new Builder(); } public static Builder newDirective(GraphQLDirective existing) { return new Builder(existing); } public static class Builder extends GraphqlTypeBuilder { private EnumSet locations = EnumSet.noneOf(DirectiveLocation.class); private final Map arguments = new LinkedHashMap<>(); private DirectiveDefinition definition; private boolean repeatable = false; public Builder() { } public Builder(GraphQLDirective existing) { this.name = existing.getName(); this.description = existing.getDescription(); this.repeatable = existing.isRepeatable(); this.locations = existing.validLocations(); this.arguments.putAll(getByName(existing.getArguments(), GraphQLArgument::getName)); } public Builder repeatable(boolean repeatable) { this.repeatable = repeatable; return this; } public Builder validLocations(DirectiveLocation... validLocations) { Collections.addAll(locations, validLocations); return this; } public Builder validLocation(DirectiveLocation validLocation) { locations.add(validLocation); return this; } public Builder clearValidLocations() { locations = EnumSet.noneOf(DirectiveLocation.class); return this; } public Builder argument(GraphQLArgument argument) { assertNotNull(argument, () -> "argument must not be null"); arguments.put(argument.getName(), argument); return this; } public Builder replaceArguments(List arguments) { assertNotNull(arguments, () -> "arguments must not be null"); this.arguments.clear(); for (GraphQLArgument argument : arguments) { this.arguments.put(argument.getName(), argument); } return this; } /** * Take an argument builder in a function definition and apply. Can be used in a jdk8 lambda * e.g.: *

         *     {@code
         *      argument(a -> a.name("argumentName"))
         *     }
         * 
* * @param builderFunction a supplier for the builder impl * * @return this */ public Builder argument(UnaryOperator builderFunction) { GraphQLArgument.Builder builder = GraphQLArgument.newArgument(); builder = builderFunction.apply(builder); return argument(builder); } /** * Same effect as the argument(GraphQLArgument). Builder.build() is called * from within * * @param builder an un-built/incomplete GraphQLArgument * * @return this */ public Builder argument(GraphQLArgument.Builder builder) { return argument(builder.build()); } /** * This is used to clear all the arguments in the builder so far. * * @return the builder */ public Builder clearArguments() { arguments.clear(); return this; } public Builder definition(DirectiveDefinition definition) { this.definition = definition; return this; } // -- the following are repeated to avoid a binary incompatibility problem -- @Override public Builder name(String name) { return super.name(name); } @Override public Builder description(String description) { return super.description(description); } public GraphQLDirective build() { return new GraphQLDirective( name, description, repeatable, locations, sort(arguments, GraphQLDirective.class, GraphQLArgument.class), definition); } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy