graphql.introspection.Introspection Maven / Gradle / Ivy
package graphql.introspection;
import graphql.Assert;
import graphql.Internal;
import graphql.PublicApi;
import graphql.language.AstPrinter;
import graphql.language.AstValueHelper;
import graphql.schema.DataFetcher;
import graphql.schema.DataFetchingEnvironment;
import graphql.schema.FieldCoordinates;
import graphql.schema.GraphQLArgument;
import graphql.schema.GraphQLCodeRegistry;
import graphql.schema.GraphQLCompositeType;
import graphql.schema.GraphQLDirective;
import graphql.schema.GraphQLEnumType;
import graphql.schema.GraphQLEnumValueDefinition;
import graphql.schema.GraphQLFieldDefinition;
import graphql.schema.GraphQLFieldsContainer;
import graphql.schema.GraphQLInputObjectField;
import graphql.schema.GraphQLInputObjectType;
import graphql.schema.GraphQLInputType;
import graphql.schema.GraphQLInterfaceType;
import graphql.schema.GraphQLList;
import graphql.schema.GraphQLModifiedType;
import graphql.schema.GraphQLNonNull;
import graphql.schema.GraphQLObjectType;
import graphql.schema.GraphQLScalarType;
import graphql.schema.GraphQLSchema;
import graphql.schema.GraphQLTypeUtil;
import graphql.schema.GraphQLUnionType;
import graphql.schema.visibility.GraphqlFieldVisibility;
import java.util.ArrayList;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import static graphql.Assert.assertTrue;
import static graphql.Scalars.GraphQLBoolean;
import static graphql.Scalars.GraphQLString;
import static graphql.schema.FieldCoordinates.coordinates;
import static graphql.schema.FieldCoordinates.systemCoordinates;
import static graphql.schema.GraphQLArgument.newArgument;
import static graphql.schema.GraphQLFieldDefinition.newFieldDefinition;
import static graphql.schema.GraphQLList.list;
import static graphql.schema.GraphQLNonNull.nonNull;
import static graphql.schema.GraphQLObjectType.newObject;
import static graphql.schema.GraphQLTypeReference.typeRef;
import static graphql.schema.GraphQLTypeUtil.simplePrint;
@PublicApi
public class Introspection {
private static final Map introspectionDataFetchers = new LinkedHashMap<>();
private static void register(GraphQLFieldsContainer parentType, String fieldName, DataFetcher dataFetcher) {
introspectionDataFetchers.put(coordinates(parentType.getName(), fieldName), dataFetcher);
}
@Internal
public static void addCodeForIntrospectionTypes(GraphQLCodeRegistry.Builder codeRegistry) {
// place the system __ fields into the mix. They have no parent types
codeRegistry.systemDataFetcher(systemCoordinates(SchemaMetaFieldDef.getName()), SchemaMetaFieldDefDataFetcher);
codeRegistry.systemDataFetcher(systemCoordinates(TypeNameMetaFieldDef.getName()), TypeNameMetaFieldDefDataFetcher);
codeRegistry.systemDataFetcher(systemCoordinates(TypeMetaFieldDef.getName()), TypeMetaFieldDefDataFetcher);
introspectionDataFetchers.forEach(codeRegistry::dataFetcher);
}
public enum TypeKind {
SCALAR,
OBJECT,
INTERFACE,
UNION,
ENUM,
INPUT_OBJECT,
LIST,
NON_NULL
}
public static final GraphQLEnumType __TypeKind = GraphQLEnumType.newEnum()
.name("__TypeKind")
.description("An enum describing what kind of type a given __Type is")
.value("SCALAR", TypeKind.SCALAR, "Indicates this type is a scalar.")
.value("OBJECT", TypeKind.OBJECT, "Indicates this type is an object. `fields` and `interfaces` are valid fields.")
.value("INTERFACE", TypeKind.INTERFACE, "Indicates this type is an interface. `fields` and `possibleTypes` are valid fields.")
.value("UNION", TypeKind.UNION, "Indicates this type is a union. `possibleTypes` is a valid field.")
.value("ENUM", TypeKind.ENUM, "Indicates this type is an enum. `enumValues` is a valid field.")
.value("INPUT_OBJECT", TypeKind.INPUT_OBJECT, "Indicates this type is an input object. `inputFields` is a valid field.")
.value("LIST", TypeKind.LIST, "Indicates this type is a list. `ofType` is a valid field.")
.value("NON_NULL", TypeKind.NON_NULL, "Indicates this type is a non-null. `ofType` is a valid field.")
.build();
public static final DataFetcher kindDataFetcher = environment -> {
Object type = environment.getSource();
if (type instanceof GraphQLScalarType) {
return TypeKind.SCALAR;
} else if (type instanceof GraphQLObjectType) {
return TypeKind.OBJECT;
} else if (type instanceof GraphQLInterfaceType) {
return TypeKind.INTERFACE;
} else if (type instanceof GraphQLUnionType) {
return TypeKind.UNION;
} else if (type instanceof GraphQLEnumType) {
return TypeKind.ENUM;
} else if (type instanceof GraphQLInputObjectType) {
return TypeKind.INPUT_OBJECT;
} else if (type instanceof GraphQLList) {
return TypeKind.LIST;
} else if (type instanceof GraphQLNonNull) {
return TypeKind.NON_NULL;
} else {
return Assert.assertShouldNeverHappen("Unknown kind of type: %s", type);
}
};
public static final GraphQLObjectType __InputValue = newObject()
.name("__InputValue")
.field(newFieldDefinition()
.name("name")
.type(nonNull(GraphQLString)))
.field(newFieldDefinition()
.name("description")
.type(GraphQLString))
.field(newFieldDefinition()
.name("type")
.type(nonNull(typeRef("__Type"))))
.field(newFieldDefinition()
.name("defaultValue")
.type(GraphQLString))
.build();
static {
register(__InputValue, "defaultValue", environment -> {
if (environment.getSource() instanceof GraphQLArgument) {
GraphQLArgument inputField = environment.getSource();
return inputField.getDefaultValue() != null ? print(inputField.getDefaultValue(), inputField.getType()) : null;
} else if (environment.getSource() instanceof GraphQLInputObjectField) {
GraphQLInputObjectField inputField = environment.getSource();
return inputField.getDefaultValue() != null ? print(inputField.getDefaultValue(), inputField.getType()) : null;
}
return null;
});
}
private static String print(Object value, GraphQLInputType type) {
return AstPrinter.printAst(AstValueHelper.astFromValue(value, type));
}
public static final GraphQLObjectType __Field = newObject()
.name("__Field")
.field(newFieldDefinition()
.name("name")
.type(nonNull(GraphQLString)))
.field(newFieldDefinition()
.name("description")
.type(GraphQLString))
.field(newFieldDefinition()
.name("args")
.type(nonNull(list(nonNull(__InputValue)))))
.field(newFieldDefinition()
.name("type")
.type(nonNull(typeRef("__Type"))))
.field(newFieldDefinition()
.name("isDeprecated")
.type(nonNull(GraphQLBoolean)))
.field(newFieldDefinition()
.name("deprecationReason")
.type(GraphQLString))
.build();
static {
register(__Field, "args", environment -> {
Object type = environment.getSource();
return ((GraphQLFieldDefinition) type).getArguments();
});
register(__Field, "isDeprecated", environment -> {
Object type = environment.getSource();
return ((GraphQLFieldDefinition) type).isDeprecated();
});
}
public static final GraphQLObjectType __EnumValue = newObject()
.name("__EnumValue")
.field(newFieldDefinition()
.name("name")
.type(nonNull(GraphQLString)))
.field(newFieldDefinition()
.name("description")
.type(GraphQLString))
.field(newFieldDefinition()
.name("isDeprecated")
.type(nonNull(GraphQLBoolean)))
.field(newFieldDefinition()
.name("deprecationReason")
.type(GraphQLString))
.build();
static {
register(__EnumValue, "isDeprecated", environment -> {
GraphQLEnumValueDefinition enumValue = environment.getSource();
return enumValue.isDeprecated();
});
}
public static final DataFetcher fieldsFetcher = environment -> {
Object type = environment.getSource();
Boolean includeDeprecated = environment.getArgument("includeDeprecated");
if (type instanceof GraphQLFieldsContainer) {
GraphQLFieldsContainer fieldsContainer = (GraphQLFieldsContainer) type;
List fieldDefinitions = environment
.getGraphQLSchema()
.getFieldVisibility()
.getFieldDefinitions(fieldsContainer);
if (includeDeprecated) return fieldDefinitions;
List filtered = new ArrayList<>(fieldDefinitions);
for (GraphQLFieldDefinition fieldDefinition : fieldDefinitions) {
if (fieldDefinition.isDeprecated()) filtered.remove(fieldDefinition);
}
return filtered;
}
return null;
};
public static final DataFetcher interfacesFetcher = environment -> {
Object type = environment.getSource();
if (type instanceof GraphQLObjectType) {
return ((GraphQLObjectType) type).getInterfaces();
}
return null;
};
public static final DataFetcher possibleTypesFetcher = environment -> {
Object type = environment.getSource();
if (type instanceof GraphQLInterfaceType) {
return environment.getGraphQLSchema().getImplementations((GraphQLInterfaceType) type);
}
if (type instanceof GraphQLUnionType) {
return ((GraphQLUnionType) type).getTypes();
}
return null;
};
public static final DataFetcher enumValuesTypesFetcher = environment -> {
Object type = environment.getSource();
Boolean includeDeprecated = environment.getArgument("includeDeprecated");
if (type instanceof GraphQLEnumType) {
List values = ((GraphQLEnumType) type).getValues();
if (includeDeprecated) return values;
List filtered = new ArrayList<>(values);
for (GraphQLEnumValueDefinition valueDefinition : values) {
if (valueDefinition.isDeprecated()) filtered.remove(valueDefinition);
}
return filtered;
}
return null;
};
public static final DataFetcher inputFieldsFetcher = environment -> {
Object type = environment.getSource();
if (type instanceof GraphQLInputObjectType) {
GraphqlFieldVisibility fieldVisibility = environment
.getGraphQLSchema()
.getFieldVisibility();
return fieldVisibility.getFieldDefinitions((GraphQLInputObjectType) type);
}
return null;
};
public static final DataFetcher OfTypeFetcher = environment -> {
Object type = environment.getSource();
if (type instanceof GraphQLModifiedType) {
return GraphQLTypeUtil.unwrapOne((GraphQLModifiedType) type);
}
return null;
};
public static final GraphQLObjectType __Type = newObject()
.name("__Type")
.field(newFieldDefinition()
.name("kind")
.type(nonNull(__TypeKind)))
.field(newFieldDefinition()
.name("name")
.type(GraphQLString))
.field(newFieldDefinition()
.name("description")
.type(GraphQLString))
.field(newFieldDefinition()
.name("fields")
.type(list(nonNull(__Field)))
.argument(newArgument()
.name("includeDeprecated")
.type(GraphQLBoolean)
.defaultValue(false)))
.field(newFieldDefinition()
.name("interfaces")
.type(list(nonNull(typeRef("__Type")))))
.field(newFieldDefinition()
.name("possibleTypes")
.type(list(nonNull(typeRef("__Type")))))
.field(newFieldDefinition()
.name("enumValues")
.type(list(nonNull(__EnumValue)))
.argument(newArgument()
.name("includeDeprecated")
.type(GraphQLBoolean)
.defaultValue(false)))
.field(newFieldDefinition()
.name("inputFields")
.type(list(nonNull(__InputValue))))
.field(newFieldDefinition()
.name("ofType")
.type(typeRef("__Type")))
.build();
static {
register(__Type, "kind", kindDataFetcher);
register(__Type, "fields", fieldsFetcher);
register(__Type, "interfaces", interfacesFetcher);
register(__Type, "possibleTypes", possibleTypesFetcher);
register(__Type, "enumValues", enumValuesTypesFetcher);
register(__Type, "inputFields", inputFieldsFetcher);
register(__Type, "ofType", OfTypeFetcher);
}
public enum DirectiveLocation {
QUERY,
MUTATION,
SUBSCRIPTION,
FIELD,
FRAGMENT_DEFINITION,
FRAGMENT_SPREAD,
INLINE_FRAGMENT,
//
// schema SDL places
//
SCHEMA,
SCALAR,
OBJECT,
FIELD_DEFINITION,
ARGUMENT_DEFINITION,
INTERFACE,
UNION,
ENUM,
ENUM_VALUE,
INPUT_OBJECT,
INPUT_FIELD_DEFINITION
}
public static final GraphQLEnumType __DirectiveLocation = GraphQLEnumType.newEnum()
.name("__DirectiveLocation")
.description("An enum describing valid locations where a directive can be placed")
.value("QUERY", DirectiveLocation.QUERY, "Indicates the directive is valid on queries.")
.value("MUTATION", DirectiveLocation.MUTATION, "Indicates the directive is valid on mutations.")
.value("FIELD", DirectiveLocation.FIELD, "Indicates the directive is valid on fields.")
.value("FRAGMENT_DEFINITION", DirectiveLocation.FRAGMENT_DEFINITION, "Indicates the directive is valid on fragment definitions.")
.value("FRAGMENT_SPREAD", DirectiveLocation.FRAGMENT_SPREAD, "Indicates the directive is valid on fragment spreads.")
.value("INLINE_FRAGMENT", DirectiveLocation.INLINE_FRAGMENT, "Indicates the directive is valid on inline fragments.")
//
// from schema SDL PR https://github.com/facebook/graphql/pull/90
//
.value("SCHEMA", DirectiveLocation.SCHEMA, "Indicates the directive is valid on a schema SDL definition.")
.value("SCALAR", DirectiveLocation.SCALAR, "Indicates the directive is valid on a scalar SDL definition.")
.value("OBJECT", DirectiveLocation.OBJECT, "Indicates the directive is valid on an object SDL definition.")
.value("FIELD_DEFINITION", DirectiveLocation.FIELD_DEFINITION, "Indicates the directive is valid on a field SDL definition.")
.value("ARGUMENT_DEFINITION", DirectiveLocation.ARGUMENT_DEFINITION, "Indicates the directive is valid on a field argument SDL definition.")
.value("INTERFACE", DirectiveLocation.INTERFACE, "Indicates the directive is valid on an interface SDL definition.")
.value("UNION", DirectiveLocation.UNION, "Indicates the directive is valid on an union SDL definition.")
.value("ENUM", DirectiveLocation.ENUM, "Indicates the directive is valid on an enum SDL definition.")
.value("ENUM_VALUE", DirectiveLocation.ENUM_VALUE, "Indicates the directive is valid on an enum value SDL definition.")
.value("INPUT_OBJECT", DirectiveLocation.INPUT_OBJECT, "Indicates the directive is valid on an input object SDL definition.")
.value("INPUT_FIELD_DEFINITION", DirectiveLocation.INPUT_FIELD_DEFINITION, "Indicates the directive is valid on an input object field SDL definition.")
.build();
@SuppressWarnings("deprecation") // because graphql spec still has the deprecated fields
public static final GraphQLObjectType __Directive = newObject()
.name("__Directive")
.field(newFieldDefinition()
.name("name")
.type(GraphQLString))
.field(newFieldDefinition()
.name("description")
.type(GraphQLString))
.field(newFieldDefinition()
.name("locations")
.type(list(nonNull(__DirectiveLocation))))
.field(newFieldDefinition()
.name("args")
.type(nonNull(list(nonNull(__InputValue)))))
.field(newFieldDefinition()
.name("onOperation")
.type(GraphQLBoolean)
.deprecate("Use `locations`."))
.field(newFieldDefinition()
.name("onFragment")
.type(GraphQLBoolean)
.deprecate("Use `locations`."))
.field(newFieldDefinition()
.name("onField")
.type(GraphQLBoolean)
.deprecate("Use `locations`."))
.build();
static {
register(__Directive, "locations", environment -> {
GraphQLDirective directive = environment.getSource();
return new ArrayList<>(directive.validLocations());
});
register(__Directive, "args", environment -> {
GraphQLDirective directive = environment.getSource();
return directive.getArguments();
});
register(__Directive, "onOperation", environment -> {
GraphQLDirective directive = environment.getSource();
return directive.isOnOperation();
});
register(__Directive, "onFragment", environment -> {
GraphQLDirective directive = environment.getSource();
return directive.isOnFragment() ||
(directive.validLocations().contains(DirectiveLocation.INLINE_FRAGMENT)
&& directive.validLocations().contains(DirectiveLocation.FRAGMENT_SPREAD));
});
register(__Directive, "onField", environment -> {
GraphQLDirective directive = environment.getSource();
return directive.isOnField() ||
directive.validLocations().contains(DirectiveLocation.FIELD);
});
}
public static final GraphQLObjectType __Schema = newObject()
.name("__Schema")
.description("A GraphQL Introspection defines the capabilities" +
" of a GraphQL server. It exposes all available types and directives on " +
"the server, the entry points for query, mutation, and subscription operations.")
.field(newFieldDefinition()
.name("types")
.description("A list of all types supported by this server.")
.type(nonNull(list(nonNull(__Type)))))
.field(newFieldDefinition()
.name("queryType")
.description("The type that query operations will be rooted at.")
.type(nonNull(__Type)))
.field(newFieldDefinition()
.name("mutationType")
.description("If this server supports mutation, the type that mutation operations will be rooted at.")
.type(__Type))
.field(newFieldDefinition()
.name("directives")
.description("'A list of all directives supported by this server.")
.type(nonNull(list(nonNull(__Directive)))))
.field(newFieldDefinition()
.name("subscriptionType")
.description("'If this server support subscription, the type that subscription operations will be rooted at.")
.type(__Type))
.build();
static {
register(__Schema, "types", environment -> {
GraphQLSchema schema = environment.getSource();
return schema.getAllTypesAsList();
});
register(__Schema, "queryType", environment -> {
GraphQLSchema schema = environment.getSource();
return schema.getQueryType();
});
register(__Schema, "mutationType", environment -> {
GraphQLSchema schema = environment.getSource();
return schema.getMutationType();
});
register(__Schema, "directives", environment -> environment.getGraphQLSchema().getDirectives());
register(__Schema, "subscriptionType", environment -> {
GraphQLSchema schema = environment.getSource();
return schema.getSubscriptionType();
});
}
public static final DataFetcher
© 2015 - 2025 Weber Informatics LLC | Privacy Policy