graphql.schema.DataFetchingFieldSelectionSetImpl Maven / Gradle / Ivy
package graphql.schema;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableSet;
import graphql.Internal;
import graphql.normalized.ExecutableNormalizedField;
import java.io.File;
import java.nio.file.FileSystems;
import java.nio.file.Path;
import java.nio.file.PathMatcher;
import java.nio.file.Paths;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.LinkedHashMap;
import java.util.LinkedHashSet;
import java.util.List;
import java.util.Map;
import java.util.Objects;
import java.util.Set;
import java.util.function.Supplier;
import java.util.stream.Collectors;
import java.util.stream.Stream;
import static graphql.Assert.assertNotNull;
import static graphql.util.FpKit.newList;
import static java.util.Collections.emptyList;
@Internal
public class DataFetchingFieldSelectionSetImpl implements DataFetchingFieldSelectionSet {
private final static String SEP = "/";
private final static boolean UNIXY = SEP.equals(File.separator);
private final static DataFetchingFieldSelectionSet NOOP = new DataFetchingFieldSelectionSet() {
@Override
public boolean contains(String fieldGlobPattern) {
return false;
}
@Override
public boolean containsAnyOf(String fieldGlobPattern, String... fieldGlobPatterns) {
return false;
}
@Override
public boolean containsAllOf(String fieldGlobPattern, String... fieldGlobPatterns) {
return false;
}
@Override
public List getFields() {
return emptyList();
}
@Override
public List getImmediateFields() {
return emptyList();
}
@Override
public List getFields(String fieldGlobPattern, String... fieldGlobPatterns) {
return Collections.emptyList();
}
@Override
public Map> getFieldsGroupedByResultKey() {
return Collections.emptyMap();
}
@Override
public Map> getFieldsGroupedByResultKey(String fieldGlobPattern, String... fieldGlobPatterns) {
return Collections.emptyMap();
}
};
public static DataFetchingFieldSelectionSet newCollector(GraphQLSchema schema, GraphQLOutputType fieldType, Supplier normalizedFieldSupplier) {
if (!GraphQLTypeUtil.isLeaf(fieldType)) {
return new DataFetchingFieldSelectionSetImpl(normalizedFieldSupplier, schema);
} else {
// we can only collect fields on object types and interfaces and unions.
return NOOP;
}
}
private final Supplier normalizedFieldSupplier;
private volatile boolean computedValues;
// we have multiple entries in this map so that we can do glob matching in multiple ways
// however it needs to be normalised back to a set of unique fields when give back out to
// the caller.
private Map> normalisedSelectionSetFields;
private List immediateFields;
private Set flattenedFieldsForGlobSearching;
private final GraphQLSchema schema;
private DataFetchingFieldSelectionSetImpl(Supplier normalizedFieldSupplier, GraphQLSchema schema) {
this.schema = schema;
this.normalizedFieldSupplier = normalizedFieldSupplier;
this.computedValues = false;
}
@Override
public boolean contains(String fieldGlobPattern) {
if (fieldGlobPattern == null || fieldGlobPattern.isEmpty()) {
return false;
}
computeValuesLazily();
fieldGlobPattern = removeLeadingSlash(fieldGlobPattern);
PathMatcher globMatcher = globMatcher(fieldGlobPattern);
for (String flattenedField : flattenedFieldsForGlobSearching) {
flattenedField = osAppropriate(flattenedField);
Path path = Paths.get(flattenedField);
if (globMatcher.matches(path)) {
return true;
}
}
return false;
}
private String osAppropriate(String flattenedField) {
if (UNIXY) {
return flattenedField;
} else {
return flattenedField.replace(SEP, "\\");
}
}
@Override
public boolean containsAnyOf(String fieldGlobPattern, String... fieldGlobPatterns) {
assertNotNull(fieldGlobPattern);
assertNotNull(fieldGlobPatterns);
for (String globPattern : mkIterable(fieldGlobPattern, fieldGlobPatterns)) {
if (contains(globPattern)) {
return true;
}
}
return false;
}
@Override
public boolean containsAllOf(String fieldGlobPattern, String... fieldGlobPatterns) {
assertNotNull(fieldGlobPattern);
assertNotNull(fieldGlobPatterns);
for (String globPattern : mkIterable(fieldGlobPattern, fieldGlobPatterns)) {
if (!contains(globPattern)) {
return false;
}
}
return true;
}
@Override
public List getFields(String fieldGlobPattern, String... fieldGlobPatterns) {
if (fieldGlobPattern == null || fieldGlobPattern.isEmpty()) {
return emptyList();
}
computeValuesLazily();
List targetNames = new ArrayList<>();
for (String flattenedField : flattenedFieldsForGlobSearching) {
for (String globPattern : mkIterable(fieldGlobPattern, fieldGlobPatterns)) {
PathMatcher globMatcher = globMatcher(globPattern);
Path path = Paths.get(flattenedField);
if (globMatcher.matches(path)) {
targetNames.add(flattenedField);
}
}
}
return toSetSemanticsList(targetNames.stream()
.flatMap(name -> normalisedSelectionSetFields.getOrDefault(name, emptyList()).stream()));
}
@Override
public List getFields() {
computeValuesLazily();
return toSetSemanticsList(normalisedSelectionSetFields.values().stream()
.flatMap(Collection::stream));
}
private List toSetSemanticsList(Stream stream) {
return ImmutableList.copyOf(stream
.collect(ImmutableSet.toImmutableSet()));
}
@Override
public List getImmediateFields() {
computeValuesLazily();
return immediateFields;
}
@Override
public Map> getFieldsGroupedByResultKey() {
return getFields().stream().collect(Collectors.groupingBy(SelectedField::getResultKey));
}
@Override
public Map> getFieldsGroupedByResultKey(String fieldGlobPattern, String... fieldGlobPatterns) {
return getFields(fieldGlobPattern, fieldGlobPatterns).stream().collect(Collectors.groupingBy(SelectedField::getResultKey));
}
private void computeValuesLazily() {
if (computedValues) {
return;
}
// this supplier is a once only thread synced call - so do it outside this lock
// if only to have only 1 lock in action at a time
ExecutableNormalizedField currentNormalisedField = normalizedFieldSupplier.get();
synchronized (this) {
if (computedValues) {
return;
}
flattenedFieldsForGlobSearching = new LinkedHashSet<>();
normalisedSelectionSetFields = new LinkedHashMap<>();
ImmutableList.Builder immediateFieldsBuilder = ImmutableList.builder();
traverseSubSelectedFields(currentNormalisedField, immediateFieldsBuilder, "", "", true);
immediateFields = immediateFieldsBuilder.build();
computedValues = true;
}
}
private void traverseSubSelectedFields(ExecutableNormalizedField currentNormalisedField, ImmutableList.Builder immediateFieldsBuilder, String qualifiedFieldPrefix, String simpleFieldPrefix, boolean firstLevel) {
List children = currentNormalisedField.getChildren();
for (ExecutableNormalizedField normalizedSubSelectedField : children) {
String typeQualifiedName = mkTypeQualifiedName(normalizedSubSelectedField);
String simpleName = normalizedSubSelectedField.getName();
String globQualifiedName = mkFieldGlobName(qualifiedFieldPrefix, typeQualifiedName);
String globSimpleName = mkFieldGlobName(simpleFieldPrefix, simpleName);
flattenedFieldsForGlobSearching.add(globQualifiedName);
// put in entries for the simple names - eg `Invoice.payments/Payment.amount` becomes `payments/amount`
flattenedFieldsForGlobSearching.add(globSimpleName);
SelectedFieldImpl selectedField = new SelectedFieldImpl(globSimpleName, globQualifiedName, normalizedSubSelectedField, schema);
if (firstLevel) {
immediateFieldsBuilder.add(selectedField);
}
normalisedSelectionSetFields.computeIfAbsent(globQualifiedName, newList()).add(selectedField);
normalisedSelectionSetFields.computeIfAbsent(globSimpleName, newList()).add(selectedField);
if (normalizedSubSelectedField.hasChildren()) {
traverseSubSelectedFields(normalizedSubSelectedField, immediateFieldsBuilder, globQualifiedName, globSimpleName, false);
}
}
}
private String removeLeadingSlash(String fieldGlobPattern) {
if (fieldGlobPattern.startsWith(SEP)) {
fieldGlobPattern = fieldGlobPattern.substring(1);
}
return fieldGlobPattern;
}
private static String mkTypeQualifiedName(ExecutableNormalizedField executableNormalizedField) {
return executableNormalizedField.objectTypeNamesToString() + "." + executableNormalizedField.getName();
}
private static String mkFieldGlobName(String fieldPrefix, String fieldName) {
return (!fieldPrefix.isEmpty() ? fieldPrefix + SEP : "") + fieldName;
}
private static PathMatcher globMatcher(String fieldGlobPattern) {
return FileSystems.getDefault().getPathMatcher("glob:" + fieldGlobPattern);
}
private List mkIterable(String fieldGlobPattern, String[] fieldGlobPatterns) {
List l = new ArrayList<>();
l.add(fieldGlobPattern);
Collections.addAll(l, fieldGlobPatterns);
return l;
}
@Override
public String toString() {
if (!computedValues) {
return "notComputed";
}
return String.join("\n", flattenedFieldsForGlobSearching);
}
private static class SelectedFieldImpl implements SelectedField {
private final String qualifiedName;
private final String fullyQualifiedName;
private final DataFetchingFieldSelectionSet selectionSet;
private final ExecutableNormalizedField executableNormalizedField;
private final GraphQLSchema schema;
private SelectedFieldImpl(String simpleQualifiedName, String fullyQualifiedName, ExecutableNormalizedField executableNormalizedField, GraphQLSchema schema) {
this.schema = schema;
this.qualifiedName = simpleQualifiedName;
this.fullyQualifiedName = fullyQualifiedName;
this.executableNormalizedField = executableNormalizedField;
this.selectionSet = new DataFetchingFieldSelectionSetImpl(() -> executableNormalizedField, schema);
}
private SelectedField mkParent(ExecutableNormalizedField executableNormalizedField) {
String parentSimpleQualifiedName = beforeLastSlash(qualifiedName);
String parentFullyQualifiedName = beforeLastSlash(fullyQualifiedName);
return executableNormalizedField.getParent() == null ? null :
new SelectedFieldImpl(parentSimpleQualifiedName, parentFullyQualifiedName, executableNormalizedField.getParent(), schema);
}
private String beforeLastSlash(String name) {
int index = name.lastIndexOf("/");
if (index > 0) {
return name.substring(0, index);
}
return "";
}
@Override
public String getName() {
return executableNormalizedField.getName();
}
@Override
public String getQualifiedName() {
return qualifiedName;
}
@Override
public String getFullyQualifiedName() {
return fullyQualifiedName;
}
@Override
public List getFieldDefinitions() {
return executableNormalizedField.getFieldDefinitions(schema);
}
@Override
public GraphQLOutputType getType() {
return executableNormalizedField.getType(schema);
}
@Override
public List getObjectTypes() {
return this.schema.getTypes(executableNormalizedField.getObjectTypeNames());
}
@Override
public List getObjectTypeNames() {
return ImmutableList.copyOf(executableNormalizedField.getObjectTypeNames());
}
@Override
public Map getArguments() {
return executableNormalizedField.getResolvedArguments();
}
@Override
public int getLevel() {
return executableNormalizedField.getLevel();
}
@Override
public boolean isConditional() {
return executableNormalizedField.isConditional(this.schema);
}
@Override
public String getAlias() {
return executableNormalizedField.getAlias();
}
@Override
public String getResultKey() {
return executableNormalizedField.getResultKey();
}
@Override
public SelectedField getParentField() {
// lazy
return mkParent(executableNormalizedField);
}
@Override
public DataFetchingFieldSelectionSet getSelectionSet() {
return selectionSet;
}
// a selected field is the same as another selected field if its the same ExecutableNF
@Override
public boolean equals(Object o) {
if (this == o) {
return true;
}
if (o == null || getClass() != o.getClass()) {
return false;
}
SelectedFieldImpl that = (SelectedFieldImpl) o;
return executableNormalizedField.equals(that.executableNormalizedField);
}
@Override
public int hashCode() {
return Objects.hash(executableNormalizedField);
}
@Override
public String toString() {
return getFullyQualifiedName();
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy