Pedal-tx
Pedal-tx, a member of the Pedal family (pedal-dialect, pedal-loader), is a Java 8 based idiomatic JPA DAO framework.
Feature Highlights
- Transaction management
- Lambda based transaction boundary
- Transaction attached objects
- Transaction attached jobs
- DAO shell with predefined functions
- Fluent interface for HQL/JQL queries
- Simple integration with QueryDSL
- Hooks to set insert/update date/time automatically
- DAO registry function for dynamic CRUD operations
Getting started
Download the Pedal jar from Maven central:
<groupId>com.eclecticlogic</groupId>
<artifactId>pedal</artifactId>
<version>1.5.0</version>
Minimum dependencies that you need to provide in your application:
- slf4j (over logback or log4j) v1.7.7 or higher
- spring-tx, spring-context and spring-orm v4.0 or higher
- hibernate-core and hibernate-entitymanager 4.3 or higher.
- JDBC4 compliant driver and connection pool manager (BoneCP recommended).
Configuration
Setting up Spring beans
Pedal’s transaction object requires an implementation of PlatformTransactionManager which Spring’s JPATransactionManager (and JTATransactionManager) provide. However, to enable advanced features, we recommend using Pedal’s JPATransactionWrapper (which derives from JPATransactionManager). After wiring up your usual suspect JPA/Hibernate Spring beans, create and instance of Pedal’s TransactionImpl and set the platformTransactionManager property to the JPATransactionWrapper bean reference (example below in the next section on DAO wiring).
Wiring up DAOs
Pedal DAO classes should derive from the AbstractDAO base class. It is recommended that you introduce an application specific parent class that dervices from AbstractDAO and have your DAOs derive from the application specific classes. The minimum requirement for a DAO class is to provide the implementation of the abstract getEntityClass() method:
public Class<E> getEntityClass();
This is usually as simple as (for the Entity class Student.java; the code is necessitated by the limitations of generics in Java):
public class StudentDAO extends AbstractDAO<Student> {
@override
public Class<Student> getEntityClass() {
return Student.class;
}
}
The DAO should also be provided with a reference to an EntityManager. This can be done as part of Spring wiring or in code as:
@Override
@PersistenceContext
public void setEntityManager(EntityManager entityManager) {
super.setEntityManager(entityManager);
}
The StudentDAO now supports basic CRUD operations and provides a fluent API for selects.
The typical Spring based wiring of a DAO (with an application specific parent DAO called AppDAO) looks like this:
<bean id="ptf" class="com.eclecticlogic.pedal.impl.JPATransactionWrapper">
<property name="entityManagerFactory" ref="emf" />
</bean>
<bean id="tx" class="com.eclecticlogic.pedal.impl.TransactionImpl">
<property name="platformTransactionManager" ref="ptf" />
</bean>
<bean id="abstractDAO" abstract="true" class="com.myapp.AppDAO">
<property name="transaction" ref="tx" />
<property name="providerAccess" ref="hibernateProviderAccess" />
</bean>
<bean parent="abstractDAO" class="com.eclecticlogic.pedal.test.dm.dao.StudentDAO" />
When using Spring Boot, setup your beans as shown below (it is important to override the default PlatformTransactionManager):
@Bean
public PlatformTransactionManager transactionManager(EntityManagerFactory factory) {
JPATransactionWrapper wrapper = new JPATransactionWrapper();
wrapper.setEntityManagerFactory(factory);
return wrapper;
}
@Bean
public Transaction transaction(PlatformTransactionManager manager) {
TransactionImpl transaction = new TransactionImpl();
transaction.setPlatformTransactionManager(manager);
return transaction;
}
Usage
Create, Update, Delete
To create an entity, simply call the DAO’s create method.
Student student = new Student();
// set various attributes
studentDAO.create(student);
// also studentDAO.create(student1, student2 ...);
Similarly, to update an entity call the update(entity) method and for deleting the delete(entity) method.
Queries
We strongly recommend that all queries be contained within the DAO class. To write a select query, you can use CritieriaQuery objects, HQL, JPA-QL or native SQL. Here is a simple HQL query implementation in the StudentDAO:
public List<Student> findByGradeAndGPARange(int grade, float gpaLow, float gpaHigh) {
return select("from Student where grade = :grade and gpa between :gpaLow and :gpaHigh") //
.bind("grade", grade) //
.bind("gpaLow", gpaLow) //
.bind("gpaHigh", gpaHigh) //
.list();
}
You can use the get() method to return just one result (it returns Optional
Update Queries
The update(query) api is similar to the select and can be used to execute updates and deletes. Here is an example:
public int graduateAboveGPA(float gpa) {
update("update Student set grade = grade + 1 where gpa >= :gpa") //
.bind("gpa", gpa) //
.update(); // returns rows updated.
}
Inserted/Updated date-time setting
Pedal-tx’s AbstractDAO
can automatically set inserted/updated date time on your objects (i.e., on the client-side for use-cases where this is appropriate or preferred to database side operation). To enable this functionality, set the DateTimeProvider
property of AbstractDAO
by calling the setter with a DateTimeProvider
or derivative. The default implementation com.eclecticlogic.pedal.dm.DateTimeProvider
looks for properties named insertedOn
or updatedOn
. It is recommended that your application specific base derivative of AbstractDAO
set the provider and call AbstractDAO.init()
.
Transaction Management
Transactions
The Pedal Transaction object allows programmatic transaction delineation alongside Spring’s @Transactional annotation. To execute a block of code in a transaction, simply get a reference to an injected transaction reference (you can supply the transaction reference to the AbstractDAO) and call either the run method (no return value) or exec method. The run has variants that takes a Consumer or Runnable. The consumer gets a reference to a “Context” instance. The exec can take a Supplier (no Context reference) or a Function that takes a Context and returns a value. The following snipper shows the transaction in action:
public void createStudentAndScore() {
getTransaction().run(() -> {
Student student = new Student();
student...
getStudentDAO.create(student);
Score score = new Score();
score.setStudent(student);
...
getScoreDAO().create(score);
});
}
Transaction attached objects
Pedal allows you to attach data and jobs to the current transaction. To enable this feature, you must use Pedal’s JPATransactionWrapper instead of Spring’s JPATransactionManager. Here is an example of setting and retrieving transaction attached data:
public void setupData() {
getTransaction().run(context -> {
context.put("myKey", someObject);
verifyData();
});
}
public void verifyData() {
getTransaction().run(context -> {
assert context.get("myKey") == someObject;
});
}
Transaction attached jobs
Transaction attached jobs allow you to fire code either just before and just after the transaction commits. The after-commit is only called if the transaction successfully commits.
public void jobFun() {
getTransaction().run(context -> {
context.beforeCommit((DataContext dc) -> {
// This is called before the commit.
}
context.afterCommit((DataContext dc) -> {
// The data is now in the database and visible to other transactions.
}
});
}
Release notes
1.5.1
- Made all dependencies ‘provided’ in scope.
1.5.0
- Major refactoring. Pedal is now pedal-tx, pedal-dialect and pedal-loader; three separate projects. Dialect and loader specific functionality have been separated to allow for more granular use.