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

graphql.validation.rules.DeferDirectiveOnValidOperation Maven / Gradle / Ivy

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

import graphql.Directives;
import graphql.ExperimentalApi;
import graphql.language.Argument;
import graphql.language.BooleanValue;
import graphql.language.Directive;
import graphql.language.Node;
import graphql.language.OperationDefinition;
import graphql.language.VariableReference;
import graphql.validation.AbstractRule;
import graphql.validation.ValidationContext;
import graphql.validation.ValidationErrorCollector;

import java.util.List;
import java.util.Optional;

import static graphql.language.OperationDefinition.Operation.SUBSCRIPTION;
import static graphql.validation.ValidationErrorType.MisplacedDirective;

/**
 * Defer Directive is Used On Valid Operations
 *
 * A GraphQL document is only valid if defer directives are not used on subscription types.
 *
 * See proposed spec:spec/Section 5 -- Validation.md ### Defer And Stream Directives Are Used On Valid Operations
 *
 */
@ExperimentalApi
public class DeferDirectiveOnValidOperation extends AbstractRule {
    public DeferDirectiveOnValidOperation(ValidationContext validationContext, ValidationErrorCollector validationErrorCollector) {
        super(validationContext, validationErrorCollector);
        this.setVisitFragmentSpreads(true);
    }


    @Override
    public void checkDirective(Directive directive, List ancestors) {
        // ExperimentalApi.ENABLE_INCREMENTAL_SUPPORT must be true
        if (!isExperimentalApiKeyEnabled(ExperimentalApi.ENABLE_INCREMENTAL_SUPPORT)) {
            return;
        }

        if (!Directives.DeferDirective.getName().equals(directive.getName())) {
            return;
        }
        // check if the directive is on allowed operation
        Optional operationDefinition = getOperationDefinition(ancestors);
        if (operationDefinition.isPresent() &&
                SUBSCRIPTION.equals(operationDefinition.get().getOperation()) &&
                !ifArgumentMightBeFalse(directive) ){
            String message = i18n(MisplacedDirective, "IncrementalDirective.notAllowedSubscriptionOperation", directive.getName());
            addError(MisplacedDirective, directive.getSourceLocation(), message);
        }
    }

    /**
     * Extract from ancestors the OperationDefinition using the document ancestor.
     * @param ancestors list of ancestors
     * @return Optional of OperationDefinition
     */
    private Optional getOperationDefinition(List ancestors) {
        return ancestors.stream()
                .filter(doc -> doc instanceof OperationDefinition)
                .map((def -> (OperationDefinition) def))
                .findFirst();
    }

    private Boolean ifArgumentMightBeFalse(Directive directive) {
        Argument ifArgument = directive.getArgumentsByName().get("if");
        if (ifArgument == null) {
            return false;
        }
        if(ifArgument.getValue() instanceof BooleanValue){
            return !((BooleanValue) ifArgument.getValue()).isValue();
        }
        if(ifArgument.getValue() instanceof VariableReference){
            return true;
        }
        return false;
    }

}





© 2015 - 2025 Weber Informatics LLC | Privacy Policy