1. Hibernate
Hibernate ORM module.
1.1. Usage
1) Add the dependencies (hikari + hibernate):
<!-- DataSource via HikariCP-->
<dependency>
<groupId>io.jooby</groupId>
<artifactId>jooby-hikari</artifactId>
<version>3.5.3</version>
</dependency>
<!-- Hibernate Module-->
<dependency>
<groupId>io.jooby</groupId>
<artifactId>jooby-hibernate</artifactId>
<version>3.5.3</version>
</dependency>
2) Add database driver (mySQL here):
<dependency>
<groupId>mysql</groupId>
<artifactId>mysql-connector-java</artifactId>
<version>${mysql-connector-java.version}</version>
</dependency>
3) Set database properties
db.url = "jdbc:mysql://localhost/mydb"
db.user = myuser
db.password = mypass
4) Install and use Hibernate
import io.jooby.hikari.HikariModule;
import io.jooby.hibernate.HibernateModule;
{
install(new HikariModule()); (1)
install(new HibernateModule()); (2)
get("/", ctx -> {
EntityManager em = require(EntityManager.class); (3)
Transaction trx = em.getTransaction(); (4)
try {
trx.begin(); (5)
// work with EntityManager compute a result (6)
trx.commit(); (7)
return result;
} catch(Exception x) {
trx.rollback(); (8)
throw x;
} finally {
em.close(); (9)
}
});
}
1 | Install and creates a DataSource |
2 | Install and initializes Hibernate. Entities are automatically detected |
3 | Get a new EntityManager |
4 | Creates a new transaction |
5 | Being the transaction |
6 | Work with EntityManager (read, write to database) |
7 | Commit the transaction |
8 | Rollback transaction in case of error |
9 | Close the EntityManager |
1.2. Entity Discovering
By default the HibernateModule module detects all the persistent entities under base/root package. The module provides two options for more explicit control:
-
List persistent classes at creation time:
install(new HibernateModule(MyEntity1.class, MyEntity2.class));
-
Explicit package scanning
install(new HibernateModule().scan("mypackage"));
1.3. Transactional Request
The TransactionalRequest filter takes care of a lifecycle of an EntityManager
/StatelessSession
per HTTP request.
The filter creates, bind, begin/commit/rollback transaction and finally close it, so route handler
doesn’t have to deal with that boring lines of code.
import io.jooby.hikari.HikariModule;
import io.jooby.hibernate.HibernateModule;
import io.jooby.hibernate.TransactionalRequest;
{
install(new HikariModule());
install(new HibernateModule());
use(new TransactionalRequest());
post("/create", ctx -> {
EntityManager em = require(EntityManager.class);
MyEntity e = ...;
em.persist(e);
return e;
});
}
import io.jooby.hikari.HikariModule;
import io.jooby.hibernate.HibernateModule;
import io.jooby.hibernate.TransactionalRequest;
{
install(new HikariModule());
install(new HibernateModule());
use(new TransactionalRequest().useStatelessSession());
post("/create", ctx -> {
StatelessSession session = require(StatelessSession.class);
MyEntity e = ...;
session.insert(e);
return e;
});
}
The EntityManager
/StatelessSession
is tied to the current HTTP request. Multiple require
/injection
calls produce
the same EntityManager
/StatelessSession
. It is a simple way of managed simple read/write operations.
The TransactionalRequest doesn’t extend session to the rendering phase (json, html, etc.).
The route handler needs to make sure all the information required by the rendering phase is available.
Otherwise, you are going to see |
There is a SessionRequest filter that works identically but leaves transaction management to you, so no transaction is started/committed or rollback during a HTTP request.
1.3.1. @Transactional
If you simply install the decorator it becomes enabled by default, this means that each route in its scope become transactional. You can exclude an MVC route by annotating it with the Transactional annotation:
import io.jooby.annotation.Transactional;
@Transactional(false)
@GET("/")
public void get(Context ctx) {
// no automatic transaction management here
}
You also have the option to invert this logic by disabling the decorator by default:
import io.jooby.ebean.TransactionalRequest;
{
...
use(new TransactionalRequest().enabledByDefault(false));
...
}
Then you can enable it for the selected routes using @Transactional(true)
:
import io.jooby.annotation.Transactional;
@Inject
private EntityManager entityManager;
@Transactional(true)
@GET("/")
public void get(Context ctx) {
// work with EntityManager
}
This feature is not limited to MVC routes. For script routes use the constant Transactional.ATTRIBUTE
:
{
get("/", ctx -> {
...
}).attribute(Transactional.ATTRIBUTE, false);
}
The behavior of the SessionRequest
decorator is not affected by @Transactional
.
1.4. UnitOfWork
Another way of managing the lifecycle of an EntityManager and transactions is working with an instance
of UnitOfWork
. You may acquire one from the service registry or inject it via DI.
Usage:
{
get("/pets", ctx -> require(UnitOfWork.class)
.apply(em -> em.createQuery("from Pet", Pet.class).getResultList()));
}
UnitOfWork
automatically begins a transaction. After the code block passed to apply(…)
or accept(…)
returns the transaction is being committed and the EntityManager
closed.
If the code block throws an exception, the transaction is rolled back, and the EntityManager
is released as well.
You may access a UnitOfWork.TransactionHandler
instance to be able to work with multiple transactions:
{
get("/update", ctx -> require(UnitOfWork.class)
.apply((em, txh) -> {
em.createQuery("from Pet", Pet.class).getResultList().forEach(pet -> {
pet.setName(pet.getName() + " Updated");
txh.commit(); // update each entity in a separate transaction
});
return "ok";
}));
}
A call to UnitOfWork.TransactionHandler.commit()
commits the current transaction and automatically begins a new one.
Similarly, you can issue a rollback using UnitOfWork.TransactionHandler.rollback()
which also begins a new transaction
after rolling back the current one.
UnitOfWork
does not allow nesting:
{
get("/nope", ctx -> require(UnitOfWork.class)
.apply(em -> {
// will lead to exception
require(UnitOfWork.class).accept(...);
return "ok";
}));
}
Neither can it be used together with SessionRequest
or TransactionalRequest
:
{
use(new TransactionalRequest());
// will lead to exception
get("/nope", ctx -> require(UnitOfWork.class)
.apply(em -> em.createQuery("from Pet", Pet.class).getResultList()));
}
1.5. Schema Creation
Schema creation is controlled by the hibernate.hbm2ddl.auto
property. The Hibernate module configure this property using the following rules:
-
When the FlywayModule module is present, the value of
hibernate.hbm2ddl.auto
is set tonone
-
When
application.env
property is set todev
ortest
, the value ofhibernate.hbm2ddl.auto
is set toupdate
-
Otherwise is set to
none
1.6. Advanced Options
Advanced Hibernate configuration is supported from application configuration properties.
hibernate.hbm2ddl.auto = create
Or by providing a custom HibernateConfigurer instance. The HibernateConfigurer let you hook and customize Hibernate bootstrap process:
import io.jooby.hibernate.HibernateConfigurer;
public class MyConfigurer extends HibernateConfigurer {
/**
* Hook into bootstrap registry and customize it.
*
* @param builder Builder.
* @param config Configuration.
*/
public void configure(BootstrapServiceRegistryBuilder builder, Config config) {
}
/**
* Hook into service registry and customize it.
*
* @param builder Builder.
* @param config Configuration.
*/
public void configure(StandardServiceRegistryBuilder builder, Config config) {
}
/**
* Hook into metadata sources and customize it.
*
* @param sources Sources.
* @param config Configuration.
*/
public void configure(MetadataSources sources, Config config) {
}
/**
* Hook into metadata builder and customize it.
*
* @param builder Builder.
* @param config Configuration.
*/
public void configure(MetadataBuilder builder, Config config) {
}
/**
* Hook into SessionFactory creation and customize it.
*
* @param builder Builder.
* @param config Configuration.
*/
public void configure(SessionFactoryBuilder builder, Config config) {
}
}
{
install(new HibernateModule().with(new MyConfigurer()));
}
Something similar is possible for custom Session using a SessionProvider:
{
install(new HibernateModule().with(builder -> {
return builder
.flushMode(AUTO)
.openSession();
}));
}