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