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

graphql.analysis.QueryComplexityCalculator Maven / Gradle / Ivy

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

import graphql.PublicApi;
import graphql.execution.CoercedVariables;
import graphql.language.Document;
import graphql.schema.GraphQLSchema;

import java.util.LinkedHashMap;
import java.util.Map;

import static graphql.Assert.assertNotNull;
import static java.util.Optional.ofNullable;

/**
 * This can calculate the complexity of an operation using the specified {@link FieldComplexityCalculator} you pass
 * into it.
 */
@PublicApi
public class QueryComplexityCalculator {

    private final FieldComplexityCalculator fieldComplexityCalculator;
    private final GraphQLSchema schema;
    private final Document document;
    private final String operationName;
    private final CoercedVariables variables;

    public QueryComplexityCalculator(Builder builder) {
        this.fieldComplexityCalculator = assertNotNull(builder.fieldComplexityCalculator, () -> "fieldComplexityCalculator can't be null");
        this.schema = assertNotNull(builder.schema, () -> "schema can't be null");
        this.document = assertNotNull(builder.document, () -> "document can't be null");
        this.variables = assertNotNull(builder.variables, () -> "variables can't be null");
        this.operationName = builder.operationName;
    }


    public int calculate() {
        Map valuesByParent = calculateByParents();
        return valuesByParent.getOrDefault(null, 0);
    }

    /**
     * @return a map that shows the field complexity for each field level in the operation
     */
    public Map calculateByParents() {
        QueryTraverser queryTraverser = QueryTraverser.newQueryTraverser()
                .schema(this.schema)
                .document(this.document)
                .operationName(this.operationName)
                .coercedVariables(this.variables)
                .build();


        Map valuesByParent = new LinkedHashMap<>();
        queryTraverser.visitPostOrder(new QueryVisitorStub() {
            @Override
            public void visitField(QueryVisitorFieldEnvironment env) {
                int childComplexity = valuesByParent.getOrDefault(env, 0);
                int value = calculateComplexity(env, childComplexity);

                QueryVisitorFieldEnvironment parentEnvironment = env.getParentEnvironment();
                valuesByParent.compute(parentEnvironment, (key, oldValue) -> {
                            Integer currentValue = ofNullable(oldValue).orElse(0);
                            return currentValue + value;
                        }
                );
            }
        });

        return valuesByParent;
    }

    private int calculateComplexity(QueryVisitorFieldEnvironment queryVisitorFieldEnvironment, int childComplexity) {
        if (queryVisitorFieldEnvironment.isTypeNameIntrospectionField()) {
            return 0;
        }
        FieldComplexityEnvironment fieldComplexityEnvironment = convertEnv(queryVisitorFieldEnvironment);
        return fieldComplexityCalculator.calculate(fieldComplexityEnvironment, childComplexity);
    }

    private FieldComplexityEnvironment convertEnv(QueryVisitorFieldEnvironment queryVisitorFieldEnvironment) {
        FieldComplexityEnvironment parentEnv = null;
        if (queryVisitorFieldEnvironment.getParentEnvironment() != null) {
            parentEnv = convertEnv(queryVisitorFieldEnvironment.getParentEnvironment());
        }
        return new FieldComplexityEnvironment(
                queryVisitorFieldEnvironment.getField(),
                queryVisitorFieldEnvironment.getFieldDefinition(),
                queryVisitorFieldEnvironment.getFieldsContainer(),
                queryVisitorFieldEnvironment.getArguments(),
                parentEnv
        );
    }

    public static Builder newCalculator() {
        return new Builder();
    }

    public static class Builder {
        private FieldComplexityCalculator fieldComplexityCalculator;
        private GraphQLSchema schema;
        private Document document;
        private String operationName;
        private CoercedVariables variables = CoercedVariables.emptyVariables();

        public Builder schema(GraphQLSchema graphQLSchema) {
            this.schema = graphQLSchema;
            return this;
        }

        public Builder fieldComplexityCalculator(FieldComplexityCalculator complexityCalculator) {
            this.fieldComplexityCalculator = complexityCalculator;
            return this;
        }

        public Builder document(Document document) {
            this.document = document;
            return this;
        }

        public Builder operationName(String operationName) {
            this.operationName = operationName;
            return this;
        }

        public Builder variables(CoercedVariables variables) {
            this.variables = variables;
            return this;
        }

        public QueryComplexityCalculator build() {
            return new QueryComplexityCalculator(this);
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy