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

graphql.execution.MergedField Maven / Gradle / Ivy

package graphql.execution;

import com.google.common.collect.ImmutableList;
import graphql.ExperimentalApi;
import graphql.PublicApi;
import graphql.execution.incremental.DeferredExecution;
import graphql.language.Argument;
import graphql.language.Field;
import org.jetbrains.annotations.Nullable;

import java.util.List;
import java.util.Objects;
import java.util.function.Consumer;

import static graphql.Assert.assertNotEmpty;

/**
 * This represents all Fields in a query which overlap and are merged into one.
 * This means they all represent the same field actually when the query is executed.
 *
 * Example query with more than one Field merged together:
 *
 * 
 * {@code
 *
 *      query Foo {
 *          bar
 *          ...BarFragment
 *      }
 *
 *      fragment BarFragment on Query {
 *          bar
 *      }
 * }
 * 
* * Another example: *
 * {@code
 *     {
 *          me{fistName}
 *          me{lastName}
 *     }
 * }
 * 
* * Here the field is merged together including the sub selections. * * A third example with different directives: *
 * {@code
 *     {
 *          foo @someDirective
 *          foo @anotherDirective
 *     }
 * }
 * 
* These examples make clear that you need to consider all merged fields together to have the full picture. * * The actual logic when fields can be successfully merged together is implemented in {#graphql.validation.rules.OverlappingFieldsCanBeMerged} */ @PublicApi public class MergedField { private final ImmutableList fields; private final Field singleField; private final ImmutableList deferredExecutions; private MergedField(ImmutableList fields, ImmutableList deferredExecutions) { assertNotEmpty(fields); this.fields = fields; this.singleField = fields.get(0); this.deferredExecutions = deferredExecutions; } private MergedField(Field field, DeferredExecution deferredExecution) { this(ImmutableList.of(field), deferredExecution == null ? ImmutableList.of() : ImmutableList.of(deferredExecution)); } /** * 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 the merged fields. */ public String getName() { return singleField.getName(); } /** * Returns the key of this MergedField for the overall result. * This is either an alias or the field name. * * @return the key for this MergedField. */ public String getResultKey() { return singleField.getResultKey(); } /** * The first of the merged fields. * * Because all fields are almost identically * often only one of the merged fields are used. * * @return the fist of the merged Fields */ public Field getSingleField() { return singleField; } /** * All merged fields share the same arguments. * * @return the list of arguments */ public List getArguments() { return singleField.getArguments(); } /** * All merged fields * * @return all merged fields */ public List getFields() { return fields; } /** * Get a list of all {@link DeferredExecution}s that this field is part of * * @return all defer executions. */ @ExperimentalApi public List getDeferredExecutions() { return deferredExecutions; } /** * Returns true if this field is part of a deferred execution * * @return true if this field is part of a deferred execution */ @ExperimentalApi public boolean isDeferred() { return !deferredExecutions.isEmpty(); } public static Builder newMergedField() { return new Builder(); } public static Builder newMergedField(Field field) { return new Builder().addField(field); } public static Builder newMergedField(List fields) { return new Builder().fields(fields); } static MergedField newSingletonMergedField(Field field, DeferredExecution deferredExecution) { return new MergedField(field, deferredExecution); } public MergedField transform(Consumer builderConsumer) { Builder builder = new Builder(this); builderConsumer.accept(builder); return builder.build(); } public static class Builder { private final ImmutableList.Builder fields = new ImmutableList.Builder<>(); private final ImmutableList.Builder deferredExecutions = new ImmutableList.Builder<>(); private Builder() { } private Builder(MergedField existing) { fields.addAll(existing.getFields()); deferredExecutions.addAll(existing.deferredExecutions); } public Builder fields(List fields) { this.fields.addAll(fields); return this; } public Builder addField(Field field) { this.fields.add(field); return this; } public Builder addDeferredExecutions(List deferredExecutions) { this.deferredExecutions.addAll(deferredExecutions); return this; } public Builder addDeferredExecution(@Nullable DeferredExecution deferredExecution) { if(deferredExecution != null) { this.deferredExecutions.add(deferredExecution); } return this; } public MergedField build() { return new MergedField(fields.build(), deferredExecutions.build()); } } @Override public boolean equals(Object o) { if (this == o) { return true; } if (o == null || getClass() != o.getClass()) { return false; } MergedField that = (MergedField) o; return fields.equals(that.fields); } @Override public int hashCode() { return Objects.hashCode(fields); } @Override public String toString() { return "MergedField{" + "fields=" + fields + '}'; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy