Skip to main content

Simple Queries

Just like other Spring Data implementations, users can define abstract methods in the Repository interface. As long as the names, parameters and return values of these methods follow conventions, Jimmer implements them automatically. For example:

BookRepository.java
package com.example.repository;

import com.example.model.Book;

import org.babyfish.jimmer.spring.repository.JRepository;
import org.jetbrains.annotations.Nullable;

public interface BookRepository extends JRepository<Book, Long> {

List<Book> findByNameOrderByEditionDesc(
@Nullable String name
);

List<Book> findByPriceBetweenOrderByName(
@Nullable BigDecimal minPrice,
@Nullable BigDecimal maxPrice
);

long countByName(String name);
}
note

There are various conventions for method names, but basic usage is similar to Spring Data JPA.

So please refer to https://docs.spring.io/spring-data/jpa/docs/current/reference/html/#repository-query-keywords, this article will not repeat it.

Dynamic WHERE

You may have noticed that in the above examples, many parameters can be null.

In Jimmer, the automatically implemented abstract methods are inherently dynamic queries, that is, any query parameter can be null.

Let's look at another more representative example, where the abstract method is defined as:

BookRepository.java
package com.example.repository;

import com.example.model.Book;

import org.babyfish.jimmer.spring.repository.JRepository;
import org.jetbrains.annotations.Nullable;

public interface BookRepository extends JRepository<Book, Long> {

List<Book> findByNameLikeIgnoreCaseAndPriceBetween(
@Nullable String name,
@Nullable BigDecimal minPrice,
@Nullable BigDecimal maxPrice
);
}

Each parameter of this method can be null, look at the 6 possible invocations:

  • Not specify any parameters

    List<Book> books = bookRepository
    .findByNameLikeIgnoreCaseAndPriceBetween(
    null,
    null,
    null
    );

    The generated SQL is (formatted for readability):

    select tb_1_.ID, tb_1_.NAME, tb_1_.EDITION, tb_1_.PRICE, tb_1_.STORE_ID
    from BOOK as tb_1_
  • Specify name

    List<Book> books = bookRepository
    .findByNameLikeIgnoreCaseAndPriceBetween(
    "G",
    null,
    null
    );

    The generated SQL is (formatted for readability):

    select tb_1_.ID, tb_1_.NAME, tb_1_.EDITION, tb_1_.PRICE, tb_1_.STORE_ID
    from BOOK as tb_1_
    where lower(tb_1_.NAME) like ? /* %g% */
  • Specify minPrice

    List<Book> books = bookRepository
    .findByNameLikeIgnoreCaseAndPriceBetween(
    null,
    new BigDecimal(40),
    null
    );

    The generated SQL is (formatted for readability):

    select tb_1_.ID, tb_1_.NAME, tb_1_.EDITION, tb_1_.PRICE, tb_1_.STORE_ID
    from BOOK as tb_1_
    where tb_1_.PRICE >= ? /* 40 */
  • Specify maxPrice

    List<Book> books = bookRepository
    .findByNameLikeIgnoreCaseAndPriceBetween(
    null,
    null,
    new BigDecimal(60)
    );

    The generated SQL is (formatted for readability):

    select tb_1_.ID, tb_1_.NAME, tb_1_.EDITION, tb_1_.PRICE, tb_1_.STORE_ID
    from BOOK as tb_1_
    where tb_1_.PRICE <= ? /* 60 */
  • Specify both minPrice and maxPrice

    List<Book> books = bookRepository
    .findByNameLikeIgnoreCaseAndPriceBetween(
    null,
    new BigDecimal(40),
    new BigDecimal(60)
    );

    The generated SQL is (formatted for readability):

    select tb_1_.ID, tb_1_.NAME, tb_1_.EDITION, tb_1_.PRICE, tb_1_.STORE_ID
    from BOOK as tb_1_
    where
    (tb_1_.PRICE between ? /* 40 */ and ? /* 60 */)
  • Specify all parameters

    List<Book> books = bookRepository
    .findByNameLikeIgnoreCaseAndPriceBetween(
    "G",
    new BigDecimal(40),
    new BigDecimal(60)
    );

    The generated SQL is (formatted for readability):

    select tb_1_.ID, tb_1_.NAME, tb_1_.EDITION, tb_1_.PRICE, tb_1_.STORE_ID
    from BOOK as tb_1_
    where
    lower(tb_1_.NAME) like ? /* %g% */
    and
    (tb_1_.PRICE between ? /* 40 */ and ? /* 60 */)

Dynamic JOIN

Users can not only apply filters to properties of the currently queried object, but also to properties of associated objects. For example:

BookRepository.java
package com.example.repository;

import com.example.model.Book;

import org.babyfish.jimmer.spring.repository.JRepository;
import org.jetbrains.annotations.Nullable;

public interface BookRepository extends JRepository<Book, Long> {

// name -> `Book.name`
// storeName -> `Book.store.name`
List<Book> findByNameStartsWithAndStoreName(
@Nullable String name,
@Nullable String storeName
);
}

Here, storeName in findByNameStartWithAndStoreName actually refers to store.name.

It means a join is performed through Book.store to BookStore first, then a condition is applied on BookStore.name.

note
  • JOIN will only be generated in the SQL if the storeName parameter is specified

  • The association that can be used by the convention method must be a non-collection association (one-to-one, many-to-one)

Let's see how to use it:

  • Specify name parameter, no JOIN

    List<Book> books = bookRepository
    .findByNameStartsWithAndStoreName("G", null);

    The generated SQL is (formatted for readability):

    select tb_1_.ID, tb_1_.NAME, tb_1_.EDITION, tb_1_.PRICE, tb_1_.STORE_ID
    from BOOK as tb_1_
    where tb_1_.NAME like ? /* G% */
  • Specify storeName parameter, JOIN generated

    List<Book> books = bookRepository
    .findByNameStartsWithAndStoreName(null, "MANNING");

    The generated SQL is (formatted for readability):

    select tb_1_.ID, tb_1_.NAME, tb_1_.EDITION, tb_1_.PRICE, tb_1_.STORE_ID
    from BOOK as tb_1_
    inner join BOOK_STORE as tb_2_
    on tb_1_.STORE_ID = tb_2_.ID
    where tb_2_.NAME = ? /* MANNING */

Dynamic ORDER BY

As long as the abstract method has a parameter of type org.springframework.data.domain.Sort, dynamic ordering can be achieved. For example:

BookRepository.java
package com.example.repository;

import com.example.model.Book;

import org.babyfish.jimmer.spring.repository.JRepository;
import org.jetbrains.annotations.Nullable;
import org.springframework.data.domain.Sort;

public interface BookRepository extends JRepository<Book, Long> {

List<Book> findByNameLikeIgnoreCase(

// This parameter is not used in subsequent examples, always null.
// Reason:
// If a query does not need any parameters, the method from base
// interface is enough, no need to define this method. The value
// of this parameter in this example is solely to make the current
// custom abstract method look reasonable.
@Nullable String name,

@Nullable Sort sort
);
}

To facilitate the top-level code to accept sorting strings from the frontend, Jimmer provides the utility class org.babyfish.jimmer.spring.model.SortUtils to convert the sorting string passed from the client to org.springframework.data.domain.Sort.

It can be used like:

Sort sort = SortUtils.toSort(
"store.name asc", "name asc", "edition desc"
);

or

Sort sort = SortUtils.toSort( 
"store.name asc, name asc, edition desc"
);
  • ORDER BY without JOIN

    List<Book> books = bookRepository
    .findByName(
    null,
    SortUtils.toSort("name, edition desc")
    );

    The generated SQL is (formatted for readability):

    select tb_1_.ID, tb_1_.NAME, tb_1_.EDITION, tb_1_.PRICE, tb_1_.STORE_ID
    from BOOK as tb_1_
    order by
    tb_1_.NAME asc,
    tb_1_.EDITION desc
  • ORDER BY requiring JOIN

    List<Book> books = bookRepository
    .findByName(
    null,
    SortUtils.toSort("store.name, name, edition desc")
    );

    The generated SQL is (formatted for readability):

    select tb_1_.ID, tb_1_.NAME, tb_1_.EDITION, tb_1_.PRICE, tb_1_.STORE_ID
    from BOOK as tb_1_
    left join BOOK_STORE as tb_2_
    on tb_1_.STORE_ID = tb_2_.ID
    order by
    tb_2_.NAME asc,
    tb_1_.NAME asc,
    tb_1_.EDITION desc

Paged Queries

To perform paged queries, the method needs:

  • A parameter of type org.springframework.data.domain.Pageable
  • Return org.springframework.data.domain.Page<Current Entity>
BookRepository.java
package com.example.repository;

import com.example.model.Book;

import org.babyfish.jimmer.spring.repository.JRepository;
import org.jetbrains.annotations.Nullable;
import org.springframework.data.domain.Pageable;
import org.springframework.data.domain.Page;

public interface BookRepository extends JRepository<Book, Long> {

Page<Book> findByName(

// This parameter is not used in subsequent examples, always null.
// Reason:
// If a query does not need any parameters, the method from base
// interface is enough, no need to define this method. The value
// of this parameter in this example is solely to make the current
// custom abstract method look reasonable.
@Nullable String name,

Pageable pageable
);
}

It can be used like:

Page<Book> page = bookRepository
.findByName(
null,
PageRequest.of(
1, // zero based, 1 means second page
5,
SortUtils.toSort("name, edition desc")
)
);

The returned Page object is like:

{
"content":[
{
"id":10,
"name":"GraphQL in Action",
"edition":1,
"price":80,
"store":{
"id":2
}
},
{
"id":3,
"name":"Learning GraphQL",
"edition":3,
"price":51,
"store":{
"id":1
}
},
{
"id":2,
"name":"Learning GraphQL",
"edition":2,
"price":55,
"store":{
"id":1
}
},
{
"id":1,
"name":"Learning GraphQL",
"edition":1,
"price":45,
"store":{
"id":1
}
},
{
"id":9,
"name":"Programming TypeScript",
"edition":3,
"price":48,
"store":{
"id":1
}
}
],
"pageable":{
"sort":{
"unsorted":false,
"sorted":true,
"empty":false
},
"pageNumber":1,
"pageSize":5,
"offset":5,
"paged":true,
"unpaged":false
},
"totalPages":3,
"totalElements":12,
"last":false,
"numberOfElements":5,
"first":false,
"sort":{
"unsorted":false,
"sorted":true,
"empty":false
},
"number":1,
"size":5,
"empty":false
}

The generated SQL is (formatted for readability):

/* Step 1: Query total rows before paging */
select count(tb_1_.ID) from BOOK as tb_1_

/* Step 2: Query data within one page */
select tb_1_.ID, tb_1_.NAME, tb_1_.EDITION, tb_1_.PRICE, tb_1_.STORE_ID
from BOOK as tb_1_
order by
tb_1_.NAME asc,
tb_1_.EDITION desc
/* MySQL paging */
limit ?, /* 5 (offset) */ ? /* 5 (limit) */

Object Fetchers

Object fetchers are one of Jimmer's signature features, allowing querying of arbitrary complex data structures instead of just simple entity objects.

Adding a parameter of type org.babyfish.jimmer.sql.fetcher.Fetcher<Current Entity> makes the abstract method capable of this.

BookRepository.java
package com.example.repository;

import com.example.model.Book;

import org.babyfish.jimmer.spring.repository.JRepository;
import org.babyfish.jimmer.sql.fetcher.Fetcher;
import org.jetbrains.annotations.Nullable;
import org.springframework.data.domain.Pageable;
import org.springframework.data.domain.Page;

public interface BookRepository extends JRepository<Book, Long> {

Page<Book> findByName(

// This parameter is not used in subsequent examples, always null.
// Reason:
// If a query does not need any parameters, the method from base
// interface is enough, no need to define this method. The value
// of this parameter in this example is solely to make the current
// custom abstract method look reasonable.
@Nullable String name,

Pageable pageable,

@Nullable Fetcher<Book> fetcher
);
}

If no Fetcher is passed or a shape of simple objects is passed, the result will be necessarily similar to previous examples, no need to repeat.

So let's just demonstrate querying a complex data structure:

Page<Book> page = bookRepository
.findByName(
null,
PageRequest.of(
1, // zero based, 1 means second page
5,
SortUtils.toSort("name, edition desc")
),
Fetchers.BOOK_FETCHER
.allScalarFields()
.store(
Fetchers.BOOK_FETCHER
.name() // associated object only queries id (implicit+mandatory) and name
)
.authors(
Fetchers.AUTHOR_FETCHER
// associated object only queries id (implicit+mandatory), firstName and lastName
.firstName().lastName()
)
);

The returned Page object is:

{
"content":[
{
"id":10,
"name":"GraphQL in Action",
"edition":1,
"price":80,
"store":{
"id":2,
"name":"MANNING"
},
"authors":[
{
"id":5,
"firstName":"Samer",
"lastName":"Buna"
}
]
},
{
"id":3,
"name":"Learning GraphQL",
"edition":3,
"price":51,
"store":{
"id":1,
"name":"O'REILLY"
},
"authors":[
{
"id":2,
"firstName":"Alex",
"lastName":"Banks"
},
{
"id":1,
"firstName":"Eve",
"lastName":"Procello"
}
]
},
{
"id":2,
"name":"Learning GraphQL",
"edition":2,
"price":55,
"store":{
"id":1,
"name":"O'REILLY"
},
"authors":[
{
"id":2,
"firstName":"Alex",
"lastName":"Banks"
},
{
"id":1,
"firstName":"Eve",
"lastName":"Procello"
}
]
},
{
"id":1,
"name":"Learning GraphQL",
"edition":1,
"price":45,
"store":{
"id":1,
"name":"O'REILLY"
},
"authors":[
{
"id":2,
"firstName":"Alex",
"lastName":"Banks"
},
{
"id":1,
"firstName":"Eve",
"lastName":"Procello"
}
]
},
{
"id":9,
"name":"Programming TypeScript",
"edition":3,
"price":48,
"store":{
"id":1,
"name":"O'REILLY"
},
"authors":[
{
"id":4,
"firstName":"Boris",
"lastName":"Cherny"
}
]
}
],
"pageable":{
"sort":{
"unsorted":false,
"sorted":true,
"empty":false
},
"pageNumber":1,
"pageSize":5,
"offset":5,
"paged":true,
"unpaged":false
},
"totalPages":3,
"totalElements":12,
"last":false,
"sort":{
"unsorted":false,
"sorted":true,
"empty":false
},
"numberOfElements":5,
"number":1,
"first":false,
"size":5,
"empty":false
}

The generated SQL is (formatted for readability):

/* Step 1: Query total rows before paging */ 
select count(tb_1_.ID) from BOOK as tb_1_

/* Step 2: Query aggregate root objects within one page */
select tb_1_.ID, tb_1_.NAME, tb_1_.EDITION, tb_1_.PRICE, tb_1_.STORE_ID
from BOOK as tb_1_
order by
tb_1_.NAME asc,
tb_1_.EDITION desc
/* MySQL paging */
limit ?, /* 5 (offset) */ ? /* 5 (limit) */

/*
* Step 3: For the 5 paged data (not 12 rows before paging),
* query the associated objects of property `Book.store`
*
* Note:
* In current case, the foreign key `STORE_ID` will be queried, so finding parent
* objects directly via foreign key. Although there are 5 rows of data,
* the foreign keys only have two distinct values, so only two SQL params.
*/
select tb_1_.ID, tb_1_.NAME
from BOOK_STORE as tb_1_
where tb_1_.ID in (
?, ?
/* actual param list: 2, 1 */
)

/*
* Step 4: For the 5 paged data (not 12 rows before paging),
* query the associated objects of property `Book.authors`
*/
select tb_2_.BOOK_ID, tb_1_.ID, tb_1_.FIRST_NAME, tb_1_.LAST_NAME
from AUTHOR as tb_1_
inner join BOOK_AUTHOR_MAPPING as tb_2_
on tb_1_.ID = tb_2_.AUTHOR_ID
where tb_2_.BOOK_ID in (
?, ?, ?, ?, ?
/* actual param list: 10, 3, 2, 1, 9 */
)
tip

Whether it is simple queries discussed in this article, or complex queries to be discussed in next article, as long as the query returns entity objects or their collections instead of simple column tuples, it is highly recommended to add a Fetcher parameter to make all object queries as powerful in data structure shaping as GraphQL.

This brings great convenience to higher level business code.