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

graphql.normalized.ExecutableNormalizedField Maven / Gradle / Ivy

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

import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableMap;
import graphql.Assert;
import graphql.Internal;
import graphql.Mutable;
import graphql.collect.ImmutableKit;
import graphql.introspection.Introspection;
import graphql.language.Argument;
import graphql.schema.GraphQLFieldDefinition;
import graphql.schema.GraphQLInterfaceType;
import graphql.schema.GraphQLObjectType;
import graphql.schema.GraphQLOutputType;
import graphql.schema.GraphQLSchema;
import graphql.schema.GraphQLType;
import graphql.schema.GraphQLUnionType;
import graphql.util.FpKit;
import org.jetbrains.annotations.NotNull;
import org.jetbrains.annotations.Nullable;

import java.util.ArrayList;
import java.util.Collection;
import java.util.LinkedHashMap;
import java.util.LinkedHashSet;
import java.util.LinkedList;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.function.Consumer;

import static graphql.Assert.assertNotNull;
import static graphql.Assert.assertTrue;
import static graphql.schema.GraphQLTypeUtil.simplePrint;
import static graphql.schema.GraphQLTypeUtil.unwrapAll;
import static java.util.stream.Collectors.joining;
import static java.util.stream.Collectors.toList;
import static java.util.stream.Collectors.toSet;

/**
 * Intentionally Mutable
 */
@Internal
@Mutable
public class ExecutableNormalizedField {
    private final String alias;
    private final ImmutableMap normalizedArguments;
    private final LinkedHashMap resolvedArguments;
    private final ImmutableList astArguments;

    // Mutable List on purpose: it is modified after creation
    private final LinkedHashSet objectTypeNames;
    private final ArrayList children;
    private ExecutableNormalizedField parent;

    private final String fieldName;
    private final int level;


    private ExecutableNormalizedField(Builder builder) {
        this.alias = builder.alias;
        this.resolvedArguments = builder.resolvedArguments;
        this.normalizedArguments = builder.normalizedArguments;
        this.astArguments = builder.astArguments;
        this.objectTypeNames = builder.objectTypeNames;
        this.fieldName = assertNotNull(builder.fieldName);
        this.children = builder.children;
        this.level = builder.level;
        this.parent = builder.parent;
    }

    /**
     * Determines whether this NF needs a fragment to select the field. However, it considers the parent
     * output type when determining whether it needs a fragment.
     * 

* Consider the following schema * *

     * interface Animal {
     *     name: String
     *     parent: Animal
     * }
     * type Cat implements Animal {
     *     name: String
     *     parent: Cat
     * }
     * type Dog implements Animal {
     *     name: String
     *     parent: Dog
     *     isGoodBoy: Boolean
     * }
     * type Query {
     *     animal: Animal
     * }
     * 
*

* and the following query * *

     * {
     *     animal {
     *         parent {
     *             name
     *         }
     *     }
     * }
     * 
*

* Then we would get the following normalized operation tree * *

     * -Query.animal: Animal
     * --[Cat, Dog].parent: Cat, Dog
     * ---[Cat, Dog].name: String
     * 
*

* If we simply checked the {@link #parent}'s {@link #getFieldDefinitions(GraphQLSchema)} that would * point us to {@code Cat.parent} and {@code Dog.parent} whose output types would incorrectly answer * our question whether this is conditional? *

* We MUST consider that the output type of the {@code parent} field is {@code Animal} and * NOT {@code Cat} or {@code Dog} as their respective impls would say. * * @param schema - the graphql schema in play * * @return true if the field is conditional */ public boolean isConditional(@NotNull GraphQLSchema schema) { if (parent == null) { return false; } /** * checking if we have an interface which can be used as an unconditional parent type */ ImmutableList parentTypes = ImmutableKit.map(parent.getFieldDefinitions(schema), fd -> unwrapAll(fd.getType())); Set interfacesImplementedByAllParents = null; for (GraphQLType parentType : parentTypes) { List toAdd = new ArrayList<>(); if (parentType instanceof GraphQLObjectType) { toAdd.addAll((List) ((GraphQLObjectType) parentType).getInterfaces()); } else if (parentType instanceof GraphQLInterfaceType) { toAdd.add((GraphQLInterfaceType) parentType); toAdd.addAll((List) ((GraphQLInterfaceType) parentType).getInterfaces()); } if (interfacesImplementedByAllParents == null) { interfacesImplementedByAllParents = new LinkedHashSet<>(toAdd); } else { interfacesImplementedByAllParents.retainAll(toAdd); } } for (GraphQLInterfaceType parentInterfaceType : interfacesImplementedByAllParents) { List implementations = schema.getImplementations(parentInterfaceType); // __typename if (this.fieldName.equals(Introspection.TypeNameMetaFieldDef.getName()) && implementations.size() == objectTypeNames.size()) { return false; } if (parentInterfaceType.getField(fieldName) == null) { continue; } if (implementations.size() == objectTypeNames.size()) { return false; } } /** *__typename is the only field in a union type that CAN be NOT conditional */ List fieldDefinitions = parent.getFieldDefinitions(schema); if (unwrapAll(fieldDefinitions.get(0).getType()) instanceof GraphQLUnionType) { GraphQLUnionType parentOutputTypeAsUnion = (GraphQLUnionType) unwrapAll(fieldDefinitions.get(0).getType()); if (this.fieldName.equals(Introspection.TypeNameMetaFieldDef.getName()) && objectTypeNames.size() == parentOutputTypeAsUnion.getTypes().size()) { return false; // Not conditional } } /** * This means there is no Union or Interface which could serve as unconditional parent */ if (objectTypeNames.size() > 1) { return true; // Conditional } if (parent.objectTypeNames.size() > 1) { return true; } GraphQLObjectType oneObjectType = (GraphQLObjectType) schema.getType(objectTypeNames.iterator().next()); return unwrapAll(parent.getFieldDefinitions(schema).get(0).getType()) != oneObjectType; } public boolean hasChildren() { return children.size() > 0; } public GraphQLOutputType getType(GraphQLSchema schema) { List fieldDefinitions = getFieldDefinitions(schema); Set fieldTypes = fieldDefinitions.stream().map(fd -> simplePrint(fd.getType())).collect(toSet()); Assert.assertTrue(fieldTypes.size() == 1, () -> "More than one type ... use getTypes"); return fieldDefinitions.get(0).getType(); } public List getTypes(GraphQLSchema schema) { List fieldTypes = ImmutableKit.map(getFieldDefinitions(schema), fd -> fd.getType()); return fieldTypes; } public List getFieldDefinitions(GraphQLSchema schema) { GraphQLFieldDefinition fieldDefinition = resolveIntrospectionField(schema, objectTypeNames, fieldName); if (fieldDefinition != null) { return ImmutableList.of(fieldDefinition); } ImmutableList.Builder builder = ImmutableList.builder(); for (String objectTypeName : objectTypeNames) { GraphQLObjectType type = (GraphQLObjectType) assertNotNull(schema.getType(objectTypeName)); builder.add(assertNotNull(type.getField(fieldName), () -> String.format("no field %s found for type %s", fieldName, objectTypeNames.iterator().next()))); } return builder.build(); } private static GraphQLFieldDefinition resolveIntrospectionField(GraphQLSchema schema, Set objectTypeNames, String fieldName) { if (fieldName.equals(schema.getIntrospectionTypenameFieldDefinition().getName())) { return schema.getIntrospectionTypenameFieldDefinition(); } else if (objectTypeNames.size() == 1 && objectTypeNames.iterator().next().equals(schema.getQueryType().getName())) { if (fieldName.equals(schema.getIntrospectionSchemaFieldDefinition().getName())) { return schema.getIntrospectionSchemaFieldDefinition(); } else if (fieldName.equals(schema.getIntrospectionTypeFieldDefinition().getName())) { return schema.getIntrospectionTypeFieldDefinition(); } } return null; } public void addObjectTypeNames(Collection objectTypeNames) { this.objectTypeNames.addAll(objectTypeNames); } public void setObjectTypeNames(Collection objectTypeNames) { this.objectTypeNames.clear(); this.objectTypeNames.addAll(objectTypeNames); } public void addChild(ExecutableNormalizedField executableNormalizedField) { this.children.add(executableNormalizedField); } public void clearChildren() { this.children.clear(); } /** * All merged fields have the same name. *

* WARNING: This is not always the key in the execution result, because of possible aliases. See {@link #getResultKey()} * * @return the name of of the merged fields. */ public String getName() { return getFieldName(); } /** * Returns the key of this MergedFieldWithType for the overall result. * This is either an alias or the FieldWTC name. * * @return the key for this MergedFieldWithType. */ public String getResultKey() { if (alias != null) { return alias; } return getName(); } public String getAlias() { return alias; } public ImmutableList getAstArguments() { return astArguments; } public NormalizedInputValue getNormalizedArgument(String name) { return normalizedArguments.get(name); } public ImmutableMap getNormalizedArguments() { return normalizedArguments; } public LinkedHashMap getResolvedArguments() { return resolvedArguments; } public static Builder newNormalizedField() { return new Builder(); } public String getFieldName() { return fieldName; } public ExecutableNormalizedField transform(Consumer builderConsumer) { Builder builder = new Builder(this); builderConsumer.accept(builder); return builder.build(); } /** * @return Warning: returns a Mutable Set. No defensive copy is made for performance reasons. */ public Set getObjectTypeNames() { return objectTypeNames; } public String getSingleObjectTypeName() { return objectTypeNames.iterator().next(); } public String printDetails() { StringBuilder result = new StringBuilder(); if (getAlias() != null) { result.append(getAlias()).append(": "); } return result + objectTypeNamesToString() + "." + fieldName; } public String objectTypeNamesToString() { if (objectTypeNames.size() == 1) { return objectTypeNames.iterator().next(); } else { return objectTypeNames.toString(); } } public List getListOfResultKeys() { LinkedList list = new LinkedList<>(); ExecutableNormalizedField current = this; while (current != null) { list.addFirst(current.getResultKey()); current = current.parent; } return list; } public List getChildren() { return children; } public List getChildrenWithSameResultKey(String resultKey) { return FpKit.filterList(children, child -> child.getResultKey().equals(resultKey)); } public List getChildren(int includingRelativeLevel) { List result = new ArrayList<>(); assertTrue(includingRelativeLevel >= 1, () -> "relative level must be >= 1"); this.getChildren().forEach(child -> { traverseImpl(child, result::add, 1, includingRelativeLevel); }); return result; } /** * This returns the child fields that can be used if the object is of the specified object type * * @param objectTypeName the object type * * @return a list of child fields that would apply to that object type */ public List getChildren(String objectTypeName) { return children.stream() .filter(cld -> cld.objectTypeNames.contains(objectTypeName)) .collect(toList()); } public int getLevel() { return level; } public ExecutableNormalizedField getParent() { return parent; } public void replaceParent(ExecutableNormalizedField newParent) { this.parent = newParent; } @Override public String toString() { return "NormalizedField{" + objectTypeNamesToString() + "." + fieldName + ", alias=" + alias + ", level=" + level + ", children=" + children.stream().map(ExecutableNormalizedField::toString).collect(joining("\n")) + '}'; } public void traverseSubTree(Consumer consumer) { this.getChildren().forEach(child -> { traverseImpl(child, consumer, 1, Integer.MAX_VALUE); }); } private void traverseImpl(ExecutableNormalizedField root, Consumer consumer, int curRelativeLevel, int abortAfter) { if (curRelativeLevel > abortAfter) { return; } consumer.accept(root); root.getChildren().forEach(child -> { traverseImpl(child, consumer, curRelativeLevel + 1, abortAfter); }); } public static class Builder { private LinkedHashSet objectTypeNames = new LinkedHashSet<>(); private String fieldName; private ArrayList children = new ArrayList<>(); private int level; private ExecutableNormalizedField parent; private String alias; private ImmutableMap normalizedArguments = ImmutableKit.emptyMap(); private LinkedHashMap resolvedArguments = new LinkedHashMap<>(); private ImmutableList astArguments = ImmutableKit.emptyList(); private Builder() { } private Builder(ExecutableNormalizedField existing) { this.alias = existing.alias; this.normalizedArguments = existing.normalizedArguments; this.astArguments = existing.astArguments; this.resolvedArguments = existing.resolvedArguments; this.objectTypeNames = new LinkedHashSet<>(existing.getObjectTypeNames()); this.fieldName = existing.getFieldName(); this.children = new ArrayList<>(existing.children); this.level = existing.getLevel(); this.parent = existing.getParent(); } public Builder clearObjectTypesNames() { this.objectTypeNames.clear(); return this; } public Builder objectTypeNames(List objectTypeNames) { this.objectTypeNames.addAll(objectTypeNames); return this; } public Builder alias(String alias) { this.alias = alias; return this; } public Builder normalizedArguments(@Nullable Map arguments) { this.normalizedArguments = arguments == null ? ImmutableKit.emptyMap() : ImmutableMap.copyOf(arguments); return this; } public Builder resolvedArguments(@Nullable Map arguments) { this.resolvedArguments = arguments == null ? new LinkedHashMap<>() : new LinkedHashMap<>(arguments); return this; } public Builder astArguments(@NotNull List astArguments) { this.astArguments = ImmutableList.copyOf(astArguments); return this; } public Builder fieldName(String fieldName) { this.fieldName = fieldName; return this; } public Builder children(List children) { this.children.clear(); this.children.addAll(children); return this; } public Builder level(int level) { this.level = level; return this; } public Builder parent(ExecutableNormalizedField parent) { this.parent = parent; return this; } public ExecutableNormalizedField build() { return new ExecutableNormalizedField(this); } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy