graphql-filter-java

A java library to transform graphql filter into database filter

License

License

Categories

Categories

Java Languages
GroupId

GroupId

com.intuit.graphql
ArtifactId

ArtifactId

graphql-filter-java
Last Version

Last Version

1.0.0
Release Date

Release Date

Type

Type

jar
Description

Description

graphql-filter-java
A java library to transform graphql filter into database filter
Project URL

Project URL

https://github.com/intuit/graphql-filter-java
Source Code Management

Source Code Management

https://github.com/intuit/graphql-filter-java

Download graphql-filter-java

How to add to project

<!-- https://jarcasting.com/artifacts/com.intuit.graphql/graphql-filter-java/ -->
<dependency>
    <groupId>com.intuit.graphql</groupId>
    <artifactId>graphql-filter-java</artifactId>
    <version>1.0.0</version>
</dependency>
// https://jarcasting.com/artifacts/com.intuit.graphql/graphql-filter-java/
implementation 'com.intuit.graphql:graphql-filter-java:1.0.0'
// https://jarcasting.com/artifacts/com.intuit.graphql/graphql-filter-java/
implementation ("com.intuit.graphql:graphql-filter-java:1.0.0")
'com.intuit.graphql:graphql-filter-java:jar:1.0.0'
<dependency org="com.intuit.graphql" name="graphql-filter-java" rev="1.0.0">
  <artifact name="graphql-filter-java" type="jar" />
</dependency>
@Grapes(
@Grab(group='com.intuit.graphql', module='graphql-filter-java', version='1.0.0')
)
libraryDependencies += "com.intuit.graphql" % "graphql-filter-java" % "1.0.0"
[com.intuit.graphql/graphql-filter-java "1.0.0"]

Dependencies

compile (4)

Group / Artifact Type Version
com.graphql-java : graphql-java jar 14.0
com.graphql-java : graphql-java-extended-scalars jar 1.0
org.springframework.data : spring-data-jpa jar 2.0.9.RELEASE
javax.persistence : javax.persistence-api jar 2.2

test (1)

Group / Artifact Type Version
junit : junit jar 4.11

Project Modules

There are no modules declared in this project.

GraphQL-Filters

Build Status Apache 2

Overview

This library helps GraphQL developers build awesome APIs with fine grain filtering support.

Requirements

  • grahql-java (v13.x)
  • Java 8.x & Above

Features

This library will help create filter conditions which are dynamically created by combining any supported filter criteria field along with any of the supported logical operations including AND, OR and NOT. Consider the examples below.

Queries

OR

{
  searchEmployees (filter : {
      or : [{ firstName : {contains : "Saurabh"}},{ lastName : {equals : "Jaiswal"}}]
    }) {
      firstName
      lastName
      age
    }
}

AND

{
  searchEmployees (filter : {
      and : [{ firtName : {contains : "Saurabh"}},{ age : {gte : 25}}]
    }) {
      firstName
      lastName
      age
    }
}

AND with OR

{
  searchEmployees (filter : {
      and : [
        { firstName : {contains : "Saurabh"}},
        { or : [{ lastName: {equals : "Jaiswal"}},{ age: {gte: 25}}
        ]}]
    }) {
      firstName
      lastName
      age
    }
}

Usage

Dependency

<dependency>
    <groupId>com.intuit.graphql</groupId>
    <artifactId>graphql-filter-java</artifactId>
    <version>1.0.0</version>
</dependency>

Schema

# Define the query type
type Query {
   searchEmployees(filter: Filter): [Employee!]
 }
 
# Define the types
type Employee {
   firstName: String!
   lastName: Int!
   age: Int!
 }
  
# Define filter input
input Filter {
   firstName: StringExpression
   lastName: StringExpression
   age: IntExpression
    
   and: [Filter!]
   or: [Filter!]
   not: Filter
}
 
# Define String expression
input StringExpression {
   equals: String
   contains: String
}
 
# Define Int Expression
input IntExpression {
   eq: Int
   gt: Int
   gte: Int
   lt: Int
   lte: Int
}

JPA Specification

Generates filter expression using JPA Specification for any SQL database.

@Repository
public interface EmployeeRepository extends JpaRepository<EmployeeEntity, String>, JpaSpecificationExecutor<EmployeeEntity> {
}
@Getter
@Setter
@Entity
@Table(name = "employee")
public class EmployeeEntity {
    private String firstName;
    private String lastName;
    private Integer age;
}
@Component
@Transactional
public class EmployeeService {
    @Autowired
    private EmployeeRepository employeeRepository;

    /**
     * Searched employees based on filter criteria.
     */ 
    public  List<EmployeeEntity> searchEmployees(DataFetchingEnvironment env) {
        List<EmployeeEntity> employees = null;
        Specification<EmployeeEntity> specification = getSpecification(env);
        if (specification != null) {
           employees = employeeRepository.findAll(specification);
        } else {
            employees = employeeRepository.findAll();
        }
        return employees;
    }
    
    /**
     * Generates the filter JPA specification
     */
    private Specification<EmployeeEntity> getSpecification(DataFetchingEnvironment env) {
        FilterExpression.FilterExpressionBuilder builder = FilterExpression.newFilterExpressionBuilder();
        FilterExpression filterExpression = builder.field(env.getField())
                .args(env.getArguments())
                .build();
        Specification<EmployeeEntity> specification = filterExpression.getExpression(ExpressionFormat.JPA);
        return specification;
    }
}

SQL WHERE

Generates SQL WHERE clause which can then be directly applied to any SQL database.

private String getExpression(DataFetchingEnvironment env) {
    FilterExpression.FilterExpressionBuilder builder = FilterExpression.newFilterExpressionBuilder();
    FilterExpression filterExpression = builder.field(env.getField())
        .args(env.getArguments())
        .build();
    String expression = filterExpression.getExpression(ExpressionFormat.SQL);
    return expression;   
}

Expression output

WHERE ((lastName = 'Jaiswal') OR (firstName LIKE '%Saurabh%'))

How it works?

When graphql-java receives and parses the source filter expression, it creates an AST in memory which contains all the fields, operators and values supplied in the source filter. The problem is the generated AST does not know about the valid rules of a correct logical expression with multiple filter criteria. In order to get a meaningful expression out of the source filter input, filter library parses the GraphQL generated AST and generates a new expression AST with correct syntax and semantics. After this step, the generated AST looks as shown below in memory.

(firstName contains Saurabh) and ((lastName equals Jaiswal) or (age gte 25))

Supported Formats

  • Infix String
  • SQL WHERE clause
  • JPA Specification

Supported Operators

Relational

  • String (EQUALS, CONTAINS, STARTS, ENDS)
  • Numeric (EQ, LT, GT, LTE, GTE)
  • Range (IN, BETWEEN)

Logical

  • AND
  • OR
  • NOT

Supported Database

  • MySQL

Complete GraphQL JPA Example

GraphQL Java Filtering With JPA Specification

com.intuit.graphql

Intuit

Powering prosperity around the world.

Versions

Version
1.0.0