/graphql-builder

Builds a graphql schema from a model using reflection

Primary LanguageJavaApache License 2.0Apache-2.0

graphql-builder

Builds a graphql schema from a model using reflection. It reads parameter and method names of the java classes to build the schema. It requires java11 and -parameters compile argument. This allows method argument names to be read removing the need for an annotations per argument. This aproach means your method / argument names are limmited to valid java names. This library is also designed with fine grained security requirements in mind.

An example using this library can be found here

Getting Started

To build the object you pass the package name to the schema builder class

GraphQL build = SchemaBuilder.build("com.example.graph.schema.app").build();

Creating an Entity

type entity

@Entity
public class User {
  private String id;
  private String name;
  
  @Id
  public String getId() {
    return id;
  }
  
  public String getName() {
    return name;
  }
}

This defines a GraphQL output type that matches this schema

type User {
  id: ID!
  name: String!
}

Input entity

To create an input entity specify input on the @Entity annotaion you can also specify both the input entity in this case will sufixed with Input

@Entity(SchemaOption.INPUT)
public class UserInput {
  private String id;
  private String name;
  
  
  public void setId(@Id String id) {
    this.id = id;
  }
  
  public void setName(String name) {
    this.name = name;
  }
}

This defines a graphql input entity that matches

input User {
  id: ID!
  name: String!
}

Optional vs Required

by default using this library all fields are required. If you want something to be optional wrap it with Optional<type>. It is done this way since Optional is part of the JDK can has good 3rd party integration.

@Entity
public class User {
private String id;
private Optional<String> name;

@Id
public String getId() {
  return id;
}

public Optional<String> getName() {
  return name;
}
}

This defines a GraphQL output type that matches this schema

type User {
  id: ID!
  name: String
}

Context

any method may include the context as a parameter. The context class must include the @Context annotation so it knows not to treat it as an argument.

Defining context

@Context
public class ApiContext {
  private Database database;
  public Database getDatabase() {
    return database;
  }
}

Calling

public CompletableFuture<Address> getAddress(ApiContext context) {
  return context.getDatabase().getLink(this, Address.class);
}

DataFetchingEnvironment

To have access to the DataFetchingEnvironment object just add it as an argument and it will be passed in

Query

To perform a query you add the @Query annotation to a static method. It does not need to be on the matching type static method with the package will be scanned.

@Query
public static CompletableFuture<List<User>> users(ApiContext context, @Id String organisationId) {
  return context.getDatabase().query(User.class);
}

This will create the following schema

extend type Query {
	users(organisationId: ID!): [User!]!
}

Again if you want anything to be optional use that java Optional class

Mutation

Mutatations are similar queries @Mutation must be applied to a static method.

@Mutation
public static CompletableFuture<User> putUser(ApiContext context, @Id String organisationId, @Id Optional<String> userId, String name) {
  //insert logic
}

This will create the following schema

extend type Mutation {
  putUser(organisationId: ID!, userId: ID, name: String): User!
}

Subscriptions

very similar to query add @Subscription and method must return a reactive Publisher

@Subscription
public static Publisher<User> usersUpdated(ApiContext context, @Id String organisationId) {
  //subscription logic
}

This will create the following schema

extend type Subscription {
  usersUpdated(organisationId: ID!): User!
}

Inheritance

To create an inheritance type you can use interface or abstract class you need to add the @Entity annotation to the parent as well. Without that annotation inherited methods will be directly added to the type

@Entity
public abstract class Animal {
  String name;
  
  public String getName() {
    return name;
  }
}

@Entity
public class Cat extends Animal {
  String meow;
  
  public String getMeow() {
    return meow;
  }
}

This will create the following schema

interface Animal {
  name: String!
}

type Cat implements Animal {
  name: String!
  meow: String!
}

Ignore method

If there is a getter that you don't want exposed in the graphql schema add @GraphQLIgnore to the method

@Entity
public class User {
  String id;
  String dbId;
  
  @Id
  public String getId() {
    return id;
  }
  
  @GraphQLIgnore
  public String getDbId() {
    return dbId;
  }
}

This will create the following schema

type User {
  id: ID!
}

Enum

to create a GraphQL enum add the @Entity annotation to a java enum

public enum Animal {
  CAT,
  DOG
}

This will create the following schema

enum Animal {
  CAT
  DOG
}

Package Authorizer

The base package requires an Authorizer. This is a call that will determine if an endpoint is accessable. This will also be used by child packages unless they have also defined an Authorizer.

This is designed for things like organisation access

This class needs to implement a method called allow, that could look like something like the following.

public class UserAuthorizer implements Authorizer {
  public CompletableFuture<Boolean> allow(DataFetchingEnvironment env) {
  ApiContext context = env.getContext();
  context.setOrganisationId(env.getArgument("organisationId"));
  if(context.getUser() == null) {
    return Promise.done(false);
  }
  return context.getUser().getMembership(context, context.getOrganisationId()).thenApply(membership -> {
  if(membership == null) {
    return false;
  }
  context.setOrganisationMembership(membership);
    return true;
  });
}

Entity type restrictions

If you have a permissions matrix that needs implemented this makes this easy. It will validate all entries before returning them from the query. Any that do not pass will be removed from the array or replaced with null. This can lead to an error if the type is not optional.

Using this approach it allows you to write your data access layer without worrying about permissions. Return all matching entities from the method then have them automatically filter from everywhere in the application.

To implement this you need to add an annotation to the class and implement the restriction factory

@Entity
@Restrict(AnimalRestriction.class)
public class Animal {
  ...
}


public class AnimalRestriction implements RestrictTypeFactory<Animal> {

  @Override
  public CompletableFuture<RestrictType<Animal>> create(DataFetchingEnvironment env) {
    ...
  }
}

public class AssetRestrict implements RestrictType<Animal> {

  @Override
  public CompletableFuture<Boolean> allow(Animal animal) {
    ...
  }
}

Directives

These are similar to GraphQL directives but just implemented on the java model You define a custom annotation and add the @Directive to it. The directive annotation must contain an array of DirectiveLocations which will be used in the GraphQL definition. Any function defined in the annotation will be placed on the schema definition as an argument.

@Retention(RUNTIME)
@Directive( { Introspection.DirectiveLocation.FIELD_DEFINITION } )
public @interface CustomDirective {
    String input();
}

This directive can now be placed where set:

@Query
@CustomDirective(input = "Custom Directive Contents")
public static String sayHello() {
  return "Hello world";
}

Which will then end up on the schema like so

directive @CustomDirective(input: String!) on FIELD_DEFINITION

type Query {
  sayHello: String! @CustomDirective(input: "Custom Directive Contents")
}

DataFetcherWrapper

Similar to the setup of a Directive the DataFetcherWrapper is created as an annotation. This annotation is then passed into the DirectiveCaller allowing you to add options to the annotation if need be

@Retention(RUNTIME)
@DataFetcherWrapper(AdminOnly.AdminOnlyDirective.class)
public @interface AdminOnly {
  ...
}

public class AdminOnlyDirective implements DirectiveCaller<AdminOnly> {
  
  @Override
  public Object process(AdminOnly annotation, DataFetchingEnvironment env, DataFetcher<?> fetcher) throws Exception {
    ...
  }
}

The annotation can then be used on any method

@Query
@AdminOnly
public static CompletableFuture<List<User>> users(ApiContext context, @Id String organisationId) {
  return context.getDatabase().query(User.class);
}

Scalar

To add a scalar you add the @Scalar Annotation this requires defining Coercing class

@Scalar(Animal.CoercingImpl.class)
public class Animal {
  public static class CoercingImpl implements Coercing<Animal, Object> {

    @Override
    public Object serialize(Object dataFetcherResult) throws CoercingSerializeException {
      return dataFetcherResult;
    }

    @Override
     public Animal parseValue(Object input) throws CoercingParseValueException {
      return null;
    }

    @Override
    public Animal parseLiteral(Object input) throws CoercingParseLiteralException {
      return null;
    }

  }
}