Audit with JPA callbacks, Hibernate Envers and Spring Data JPA
This commit is contained in:
parent
50cd7009c5
commit
a19854cceb
|
@ -15,6 +15,16 @@
|
||||||
<artifactId>spring-context</artifactId>
|
<artifactId>spring-context</artifactId>
|
||||||
<version>${org.springframework.version}</version>
|
<version>${org.springframework.version}</version>
|
||||||
</dependency>
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.springframework</groupId>
|
||||||
|
<artifactId>spring-aspects</artifactId>
|
||||||
|
<version>${org.springframework.version}</version>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.springframework.security</groupId>
|
||||||
|
<artifactId>spring-security-core</artifactId>
|
||||||
|
<version>${org.springframework.security.version}</version>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
<!-- persistence -->
|
<!-- persistence -->
|
||||||
|
|
||||||
|
@ -23,21 +33,35 @@
|
||||||
<artifactId>spring-orm</artifactId>
|
<artifactId>spring-orm</artifactId>
|
||||||
<version>${org.springframework.version}</version>
|
<version>${org.springframework.version}</version>
|
||||||
</dependency>
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.springframework.data</groupId>
|
||||||
|
<artifactId>spring-data-jpa</artifactId>
|
||||||
|
<version>${org.springframework.data.version}</version>
|
||||||
|
</dependency>
|
||||||
<dependency>
|
<dependency>
|
||||||
<groupId>org.hibernate</groupId>
|
<groupId>org.hibernate</groupId>
|
||||||
<artifactId>hibernate-core</artifactId>
|
<artifactId>hibernate-core</artifactId>
|
||||||
<version>${hibernate.version}</version>
|
<version>${hibernate.version}</version>
|
||||||
</dependency>
|
</dependency>
|
||||||
<dependency>
|
<dependency>
|
||||||
<groupId>org.javassist</groupId>
|
<groupId>org.hibernate</groupId>
|
||||||
<artifactId>javassist</artifactId>
|
<artifactId>hibernate-envers</artifactId>
|
||||||
<version>${javassist.version}</version>
|
<version>${hibernate-envers.version}</version>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>javax.transaction</groupId>
|
||||||
|
<artifactId>jta</artifactId>
|
||||||
|
<version>${jta.version}</version>
|
||||||
|
</dependency>
|
||||||
|
<dependency>
|
||||||
|
<groupId>javax.el</groupId>
|
||||||
|
<artifactId>javax.el-api</artifactId>
|
||||||
|
<version>${el-api.version}</version>
|
||||||
</dependency>
|
</dependency>
|
||||||
<dependency>
|
<dependency>
|
||||||
<groupId>mysql</groupId>
|
<groupId>mysql</groupId>
|
||||||
<artifactId>mysql-connector-java</artifactId>
|
<artifactId>mysql-connector-java</artifactId>
|
||||||
<version>${mysql-connector-java.version}</version>
|
<version>${mysql-connector-java.version}</version>
|
||||||
<scope>runtime</scope>
|
|
||||||
</dependency>
|
</dependency>
|
||||||
|
|
||||||
<dependency>
|
<dependency>
|
||||||
|
@ -78,6 +102,13 @@
|
||||||
<scope>test</scope>
|
<scope>test</scope>
|
||||||
</dependency>
|
</dependency>
|
||||||
|
|
||||||
|
<dependency>
|
||||||
|
<groupId>org.springframework.security</groupId>
|
||||||
|
<artifactId>spring-security-test</artifactId>
|
||||||
|
<version>${org.springframework.security.version}</version>
|
||||||
|
<scope>test</scope>
|
||||||
|
</dependency>
|
||||||
|
|
||||||
<dependency>
|
<dependency>
|
||||||
<groupId>junit</groupId>
|
<groupId>junit</groupId>
|
||||||
<artifactId>junit-dep</artifactId>
|
<artifactId>junit-dep</artifactId>
|
||||||
|
@ -85,12 +116,6 @@
|
||||||
<scope>test</scope>
|
<scope>test</scope>
|
||||||
</dependency>
|
</dependency>
|
||||||
|
|
||||||
<dependency>
|
|
||||||
<groupId>org.hamcrest</groupId>
|
|
||||||
<artifactId>hamcrest-core</artifactId>
|
|
||||||
<version>${org.hamcrest.version}</version>
|
|
||||||
<scope>test</scope>
|
|
||||||
</dependency>
|
|
||||||
<dependency>
|
<dependency>
|
||||||
<groupId>org.hamcrest</groupId>
|
<groupId>org.hamcrest</groupId>
|
||||||
<artifactId>hamcrest-library</artifactId>
|
<artifactId>hamcrest-library</artifactId>
|
||||||
|
@ -162,7 +187,28 @@
|
||||||
</configuration>
|
</configuration>
|
||||||
</configuration>
|
</configuration>
|
||||||
</plugin>
|
</plugin>
|
||||||
|
<!--
|
||||||
|
<plugin>
|
||||||
|
<groupId>de.juplo</groupId>
|
||||||
|
<artifactId>hibernate4-maven-plugin</artifactId>
|
||||||
|
<version>${hibernate4-maven-plugin.version}</version>
|
||||||
|
<executions>
|
||||||
|
<execution>
|
||||||
|
<goals>
|
||||||
|
<goal>export</goal>
|
||||||
|
</goals>
|
||||||
|
</execution>
|
||||||
|
</executions>
|
||||||
|
<configuration>
|
||||||
|
<url>jdbc:mysql://localhost:3306/spring_hibernate4_01?createDatabaseIfNotExist=true</url>
|
||||||
|
<username>root</username>
|
||||||
|
<password>root</password>
|
||||||
|
<driverClassName>com.mysql.jdbc.Driver</driverClassName>
|
||||||
|
<hibernateDialect>org.hibernate.dialect.MySQL5Dialect</hibernateDialect>
|
||||||
|
<hibernateProperties>persistence-mysql.properties</hibernateProperties>
|
||||||
|
</configuration>
|
||||||
|
</plugin>
|
||||||
|
-->
|
||||||
</plugins>
|
</plugins>
|
||||||
|
|
||||||
</build>
|
</build>
|
||||||
|
@ -171,19 +217,22 @@
|
||||||
<!-- Spring -->
|
<!-- Spring -->
|
||||||
<org.springframework.version>4.2.2.RELEASE</org.springframework.version>
|
<org.springframework.version>4.2.2.RELEASE</org.springframework.version>
|
||||||
<org.springframework.security.version>4.0.2.RELEASE</org.springframework.security.version>
|
<org.springframework.security.version>4.0.2.RELEASE</org.springframework.security.version>
|
||||||
<javassist.version>3.20.0-GA</javassist.version>
|
<org.springframework.data.version>1.9.2.RELEASE</org.springframework.data.version>
|
||||||
|
|
||||||
<!-- persistence -->
|
<!-- persistence -->
|
||||||
<hibernate.version>4.3.11.Final</hibernate.version>
|
<hibernate.version>4.3.11.Final</hibernate.version>
|
||||||
|
<hibernate-envers.version>${hibernate.version}</hibernate-envers.version>
|
||||||
<mysql-connector-java.version>5.1.36</mysql-connector-java.version>
|
<mysql-connector-java.version>5.1.36</mysql-connector-java.version>
|
||||||
<tomcat-dbcp.version>7.0.42</tomcat-dbcp.version>
|
<tomcat-dbcp.version>7.0.42</tomcat-dbcp.version>
|
||||||
|
<jta.version>1.1</jta.version>
|
||||||
|
<el-api.version>2.2.4</el-api.version>
|
||||||
|
|
||||||
<!-- logging -->
|
<!-- logging -->
|
||||||
<org.slf4j.version>1.7.12</org.slf4j.version>
|
<org.slf4j.version>1.7.12</org.slf4j.version>
|
||||||
<logback.version>1.1.3</logback.version>
|
<logback.version>1.1.3</logback.version>
|
||||||
|
|
||||||
<!-- various -->
|
<!-- various -->
|
||||||
<hibernate-validator.version>5.2.1.Final</hibernate-validator.version>
|
<hibernate-validator.version>4.3.2.Final</hibernate-validator.version>
|
||||||
|
|
||||||
<!-- util -->
|
<!-- util -->
|
||||||
<guava.version>18.0</guava.version>
|
<guava.version>18.0</guava.version>
|
||||||
|
@ -204,6 +253,7 @@
|
||||||
<maven-surefire-plugin.version>2.18.1</maven-surefire-plugin.version>
|
<maven-surefire-plugin.version>2.18.1</maven-surefire-plugin.version>
|
||||||
<maven-resources-plugin.version>2.7</maven-resources-plugin.version>
|
<maven-resources-plugin.version>2.7</maven-resources-plugin.version>
|
||||||
<cargo-maven2-plugin.version>1.4.15</cargo-maven2-plugin.version>
|
<cargo-maven2-plugin.version>1.4.15</cargo-maven2-plugin.version>
|
||||||
|
<!-- <hibernate4-maven-plugin.version>1.1.0</hibernate4-maven-plugin.version> -->
|
||||||
|
|
||||||
</properties>
|
</properties>
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,8 @@
|
||||||
|
package org.baeldung.persistence.dao;
|
||||||
|
|
||||||
|
import org.baeldung.persistence.dao.common.IAuditOperations;
|
||||||
|
import org.baeldung.persistence.model.Bar;
|
||||||
|
|
||||||
|
public interface IBarAuditableDao extends IBarDao, IAuditOperations<Bar> {
|
||||||
|
//
|
||||||
|
}
|
|
@ -0,0 +1,10 @@
|
||||||
|
package org.baeldung.persistence.dao;
|
||||||
|
|
||||||
|
import java.io.Serializable;
|
||||||
|
|
||||||
|
import org.baeldung.persistence.model.Bar;
|
||||||
|
import org.springframework.data.repository.CrudRepository;
|
||||||
|
|
||||||
|
public interface IBarCrudRepository extends CrudRepository<Bar, Serializable> {
|
||||||
|
//
|
||||||
|
}
|
|
@ -0,0 +1,8 @@
|
||||||
|
package org.baeldung.persistence.dao;
|
||||||
|
|
||||||
|
import org.baeldung.persistence.dao.common.IOperations;
|
||||||
|
import org.baeldung.persistence.model.Bar;
|
||||||
|
|
||||||
|
public interface IBarDao extends IOperations<Bar> {
|
||||||
|
//
|
||||||
|
}
|
|
@ -0,0 +1,8 @@
|
||||||
|
package org.baeldung.persistence.dao;
|
||||||
|
|
||||||
|
import org.baeldung.persistence.dao.common.IAuditOperations;
|
||||||
|
import org.baeldung.persistence.model.Foo;
|
||||||
|
|
||||||
|
public interface IFooAuditableDao extends IFooDao, IAuditOperations<Foo> {
|
||||||
|
//
|
||||||
|
}
|
|
@ -0,0 +1,14 @@
|
||||||
|
package org.baeldung.persistence.dao.common;
|
||||||
|
|
||||||
|
import java.io.Serializable;
|
||||||
|
|
||||||
|
import com.google.common.base.Preconditions;
|
||||||
|
|
||||||
|
public abstract class AbstractDao<T extends Serializable> implements IOperations<T> {
|
||||||
|
|
||||||
|
protected Class<T> clazz;
|
||||||
|
|
||||||
|
protected final void setClazz(final Class<T> clazzToSet) {
|
||||||
|
clazz = Preconditions.checkNotNull(clazzToSet);
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,37 @@
|
||||||
|
package org.baeldung.persistence.dao.common;
|
||||||
|
|
||||||
|
import java.io.Serializable;
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
import org.hibernate.envers.AuditReader;
|
||||||
|
import org.hibernate.envers.AuditReaderFactory;
|
||||||
|
import org.hibernate.envers.query.AuditQuery;
|
||||||
|
|
||||||
|
@SuppressWarnings("unchecked")
|
||||||
|
public class AbstractHibernateAuditableDao<T extends Serializable> extends AbstractHibernateDao<T> implements IAuditOperations<T> {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public List<T> getEntitiesAtRevision(final Number revision) {
|
||||||
|
final AuditReader auditReader = AuditReaderFactory.get(getCurrentSession());
|
||||||
|
final AuditQuery query = auditReader.createQuery().forEntitiesAtRevision(clazz, revision);
|
||||||
|
final List<T> resultList = query.getResultList();
|
||||||
|
return resultList;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public List<T> getEntitiesModifiedAtRevision(final Number revision) {
|
||||||
|
final AuditReader auditReader = AuditReaderFactory.get(getCurrentSession());
|
||||||
|
final AuditQuery query = auditReader.createQuery().forEntitiesModifiedAtRevision(clazz, revision);
|
||||||
|
final List<T> resultList = query.getResultList();
|
||||||
|
return resultList;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public List<T> getRevisions() {
|
||||||
|
final AuditReader auditReader = AuditReaderFactory.get(getCurrentSession());
|
||||||
|
final AuditQuery query = auditReader.createQuery().forRevisionsOfEntity(clazz, true, true);
|
||||||
|
final List<T> resultList = query.getResultList();
|
||||||
|
return resultList;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -10,55 +10,49 @@ import org.springframework.beans.factory.annotation.Autowired;
|
||||||
import com.google.common.base.Preconditions;
|
import com.google.common.base.Preconditions;
|
||||||
|
|
||||||
@SuppressWarnings("unchecked")
|
@SuppressWarnings("unchecked")
|
||||||
public abstract class AbstractHibernateDao<T extends Serializable> implements IOperations<T> {
|
public abstract class AbstractHibernateDao<T extends Serializable> extends AbstractDao<T> implements IOperations<T> {
|
||||||
private Class<T> clazz;
|
|
||||||
|
|
||||||
@Autowired
|
@Autowired
|
||||||
private SessionFactory sessionFactory;
|
protected SessionFactory sessionFactory;
|
||||||
|
|
||||||
// API
|
// API
|
||||||
|
|
||||||
protected final void setClazz(final Class<T> clazzToSet) {
|
|
||||||
clazz = Preconditions.checkNotNull(clazzToSet);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public final T findOne(final long id) {
|
public T findOne(final long id) {
|
||||||
return (T) getCurrentSession().get(clazz, id);
|
return (T) getCurrentSession().get(clazz, id);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public final List<T> findAll() {
|
public List<T> findAll() {
|
||||||
return getCurrentSession().createQuery("from " + clazz.getName()).list();
|
return getCurrentSession().createQuery("from " + clazz.getName()).list();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public final void create(final T entity) {
|
public void create(final T entity) {
|
||||||
Preconditions.checkNotNull(entity);
|
Preconditions.checkNotNull(entity);
|
||||||
// getCurrentSession().persist(entity);
|
|
||||||
getCurrentSession().saveOrUpdate(entity);
|
getCurrentSession().saveOrUpdate(entity);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public final T update(final T entity) {
|
public T update(final T entity) {
|
||||||
Preconditions.checkNotNull(entity);
|
Preconditions.checkNotNull(entity);
|
||||||
return (T) getCurrentSession().merge(entity);
|
return (T) getCurrentSession().merge(entity);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public final void delete(final T entity) {
|
public void delete(final T entity) {
|
||||||
Preconditions.checkNotNull(entity);
|
Preconditions.checkNotNull(entity);
|
||||||
getCurrentSession().delete(entity);
|
getCurrentSession().delete(entity);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public final void deleteById(final long entityId) {
|
public void deleteById(final long entityId) {
|
||||||
final T entity = findOne(entityId);
|
final T entity = findOne(entityId);
|
||||||
Preconditions.checkState(entity != null);
|
Preconditions.checkState(entity != null);
|
||||||
delete(entity);
|
delete(entity);
|
||||||
}
|
}
|
||||||
|
|
||||||
protected final Session getCurrentSession() {
|
protected Session getCurrentSession() {
|
||||||
return sessionFactory.getCurrentSession();
|
return sessionFactory.getCurrentSession();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,56 @@
|
||||||
|
package org.baeldung.persistence.dao.common;
|
||||||
|
|
||||||
|
import java.io.Serializable;
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
import javax.persistence.EntityManager;
|
||||||
|
import javax.persistence.PersistenceContext;
|
||||||
|
import javax.persistence.TypedQuery;
|
||||||
|
import javax.persistence.criteria.CriteriaBuilder;
|
||||||
|
import javax.persistence.criteria.CriteriaQuery;
|
||||||
|
import javax.persistence.criteria.Root;
|
||||||
|
|
||||||
|
public class AbstractJpaDao<T extends Serializable> extends AbstractDao<T> implements IOperations<T> {
|
||||||
|
|
||||||
|
@PersistenceContext
|
||||||
|
private EntityManager em;
|
||||||
|
|
||||||
|
// API
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public T findOne(final long id) {
|
||||||
|
return em.find(clazz, Long.valueOf(id).intValue());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public List<T> findAll() {
|
||||||
|
final CriteriaBuilder cb = em.getCriteriaBuilder();
|
||||||
|
final CriteriaQuery<T> cq = cb.createQuery(clazz);
|
||||||
|
final Root<T> rootEntry = cq.from(clazz);
|
||||||
|
final CriteriaQuery<T> all = cq.select(rootEntry);
|
||||||
|
final TypedQuery<T> allQuery = em.createQuery(all);
|
||||||
|
return allQuery.getResultList();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void create(final T entity) {
|
||||||
|
em.persist(entity);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public T update(final T entity) {
|
||||||
|
em.merge(entity);
|
||||||
|
return entity;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void delete(final T entity) {
|
||||||
|
em.remove(entity);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void deleteById(final long entityId) {
|
||||||
|
delete(findOne(entityId));
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,14 @@
|
||||||
|
package org.baeldung.persistence.dao.common;
|
||||||
|
|
||||||
|
import java.io.Serializable;
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
public interface IAuditOperations<T extends Serializable> {
|
||||||
|
|
||||||
|
List<T> getEntitiesAtRevision(Number revision);
|
||||||
|
|
||||||
|
List<T> getEntitiesModifiedAtRevision(Number revision);
|
||||||
|
|
||||||
|
List<T> getRevisions();
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,28 @@
|
||||||
|
package org.baeldung.persistence.dao.impl;
|
||||||
|
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
import org.baeldung.persistence.dao.IBarAuditableDao;
|
||||||
|
import org.baeldung.persistence.dao.common.AbstractHibernateAuditableDao;
|
||||||
|
import org.baeldung.persistence.model.Bar;
|
||||||
|
|
||||||
|
public class BarAuditableDao extends AbstractHibernateAuditableDao<Bar> implements IBarAuditableDao {
|
||||||
|
|
||||||
|
public BarAuditableDao() {
|
||||||
|
super();
|
||||||
|
|
||||||
|
setClazz(Bar.class);
|
||||||
|
}
|
||||||
|
|
||||||
|
// API
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public List<Bar> getRevisions() {
|
||||||
|
final List<Bar> resultList = super.getRevisions();
|
||||||
|
for (final Bar bar : resultList) {
|
||||||
|
bar.getFooSet().size(); // force FooSet initialization
|
||||||
|
}
|
||||||
|
return resultList;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,19 @@
|
||||||
|
package org.baeldung.persistence.dao.impl;
|
||||||
|
|
||||||
|
import org.baeldung.persistence.dao.IBarDao;
|
||||||
|
import org.baeldung.persistence.dao.common.AbstractHibernateDao;
|
||||||
|
import org.baeldung.persistence.model.Bar;
|
||||||
|
import org.springframework.stereotype.Repository;
|
||||||
|
|
||||||
|
@Repository
|
||||||
|
public class BarDao extends AbstractHibernateDao<Bar> implements IBarDao {
|
||||||
|
|
||||||
|
public BarDao() {
|
||||||
|
super();
|
||||||
|
|
||||||
|
setClazz(Bar.class);
|
||||||
|
}
|
||||||
|
|
||||||
|
// API
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,19 @@
|
||||||
|
package org.baeldung.persistence.dao.impl;
|
||||||
|
|
||||||
|
import org.baeldung.persistence.dao.IBarDao;
|
||||||
|
import org.baeldung.persistence.dao.common.AbstractJpaDao;
|
||||||
|
import org.baeldung.persistence.model.Bar;
|
||||||
|
import org.springframework.stereotype.Repository;
|
||||||
|
|
||||||
|
@Repository
|
||||||
|
public class BarJpaDao extends AbstractJpaDao<Bar> implements IBarDao {
|
||||||
|
|
||||||
|
public BarJpaDao() {
|
||||||
|
super();
|
||||||
|
|
||||||
|
setClazz(Bar.class);
|
||||||
|
}
|
||||||
|
|
||||||
|
// API
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,17 @@
|
||||||
|
package org.baeldung.persistence.dao.impl;
|
||||||
|
|
||||||
|
import org.baeldung.persistence.dao.IFooAuditableDao;
|
||||||
|
import org.baeldung.persistence.dao.common.AbstractHibernateAuditableDao;
|
||||||
|
import org.baeldung.persistence.model.Foo;
|
||||||
|
|
||||||
|
public class FooAuditableDao extends AbstractHibernateAuditableDao<Foo> implements IFooAuditableDao {
|
||||||
|
|
||||||
|
public FooAuditableDao() {
|
||||||
|
super();
|
||||||
|
|
||||||
|
setClazz(Foo.class);
|
||||||
|
}
|
||||||
|
|
||||||
|
// API
|
||||||
|
|
||||||
|
}
|
|
@ -1,26 +1,66 @@
|
||||||
package org.baeldung.persistence.model;
|
package org.baeldung.persistence.model;
|
||||||
|
|
||||||
import java.io.Serializable;
|
import java.io.Serializable;
|
||||||
|
import java.util.Date;
|
||||||
import java.util.Set;
|
import java.util.Set;
|
||||||
|
|
||||||
import javax.persistence.CascadeType;
|
import javax.persistence.CascadeType;
|
||||||
import javax.persistence.Column;
|
import javax.persistence.Column;
|
||||||
import javax.persistence.Entity;
|
import javax.persistence.Entity;
|
||||||
|
import javax.persistence.EntityListeners;
|
||||||
import javax.persistence.FetchType;
|
import javax.persistence.FetchType;
|
||||||
import javax.persistence.GeneratedValue;
|
import javax.persistence.GeneratedValue;
|
||||||
import javax.persistence.GenerationType;
|
import javax.persistence.GenerationType;
|
||||||
import javax.persistence.Id;
|
import javax.persistence.Id;
|
||||||
import javax.persistence.NamedQuery;
|
import javax.persistence.NamedQuery;
|
||||||
import javax.persistence.OneToMany;
|
import javax.persistence.OneToMany;
|
||||||
|
import javax.persistence.PrePersist;
|
||||||
|
import javax.persistence.PreRemove;
|
||||||
|
import javax.persistence.PreUpdate;
|
||||||
|
|
||||||
import org.hibernate.annotations.OrderBy;
|
import org.hibernate.annotations.OrderBy;
|
||||||
|
import org.hibernate.envers.Audited;
|
||||||
|
import org.jboss.logging.Logger;
|
||||||
|
import org.springframework.data.annotation.CreatedBy;
|
||||||
|
import org.springframework.data.annotation.CreatedDate;
|
||||||
|
import org.springframework.data.annotation.LastModifiedBy;
|
||||||
|
import org.springframework.data.annotation.LastModifiedDate;
|
||||||
|
import org.springframework.data.jpa.domain.support.AuditingEntityListener;
|
||||||
|
|
||||||
import com.google.common.collect.Sets;
|
import com.google.common.collect.Sets;
|
||||||
|
|
||||||
@Entity
|
@Entity
|
||||||
@NamedQuery(name = "Bar.findAll", query = "SELECT b FROM Bar b")
|
@NamedQuery(name = "Bar.findAll", query = "SELECT b FROM Bar b")
|
||||||
|
@Audited
|
||||||
|
@EntityListeners(AuditingEntityListener.class)
|
||||||
public class Bar implements Serializable {
|
public class Bar implements Serializable {
|
||||||
|
|
||||||
|
private static Logger logger = Logger.getLogger(Bar.class);
|
||||||
|
|
||||||
|
public enum OPERATION {
|
||||||
|
INSERT, UPDATE, DELETE;
|
||||||
|
private String value;
|
||||||
|
|
||||||
|
OPERATION() {
|
||||||
|
value = toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getValue() {
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static OPERATION parse(final String value) {
|
||||||
|
OPERATION operation = null;
|
||||||
|
for (final OPERATION op : OPERATION.values()) {
|
||||||
|
if (op.getValue().equals(value)) {
|
||||||
|
operation = op;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return operation;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
@Id
|
@Id
|
||||||
@GeneratedValue(strategy = GenerationType.AUTO)
|
@GeneratedValue(strategy = GenerationType.AUTO)
|
||||||
@Column(name = "id")
|
@Column(name = "id")
|
||||||
|
@ -31,8 +71,31 @@ public class Bar implements Serializable {
|
||||||
|
|
||||||
@OneToMany(mappedBy = "bar", cascade = CascadeType.ALL, fetch = FetchType.LAZY)
|
@OneToMany(mappedBy = "bar", cascade = CascadeType.ALL, fetch = FetchType.LAZY)
|
||||||
@OrderBy(clause = "NAME DESC")
|
@OrderBy(clause = "NAME DESC")
|
||||||
|
// @NotAudited
|
||||||
private Set<Foo> fooSet = Sets.newHashSet();
|
private Set<Foo> fooSet = Sets.newHashSet();
|
||||||
|
|
||||||
|
@Column(name = "operation")
|
||||||
|
private String operation;
|
||||||
|
|
||||||
|
@Column(name = "timestamp")
|
||||||
|
private long timestamp;
|
||||||
|
|
||||||
|
@Column(name = "created_date")
|
||||||
|
@CreatedDate
|
||||||
|
private long createdDate;
|
||||||
|
|
||||||
|
@Column(name = "modified_date")
|
||||||
|
@LastModifiedDate
|
||||||
|
private long modifiedDate;
|
||||||
|
|
||||||
|
@Column(name = "created_by")
|
||||||
|
@CreatedBy
|
||||||
|
private String createdBy;
|
||||||
|
|
||||||
|
@Column(name = "modified_by")
|
||||||
|
@LastModifiedBy
|
||||||
|
private String modifiedBy;
|
||||||
|
|
||||||
public Bar() {
|
public Bar() {
|
||||||
super();
|
super();
|
||||||
}
|
}
|
||||||
|
@ -69,7 +132,57 @@ public class Bar implements Serializable {
|
||||||
this.name = name;
|
this.name = name;
|
||||||
}
|
}
|
||||||
|
|
||||||
//
|
public OPERATION getOperation() {
|
||||||
|
return OPERATION.parse(operation);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setOperation(final OPERATION operation) {
|
||||||
|
this.operation = operation.getValue();
|
||||||
|
}
|
||||||
|
|
||||||
|
public long getTimestamp() {
|
||||||
|
return timestamp;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setTimestamp(final long timestamp) {
|
||||||
|
this.timestamp = timestamp;
|
||||||
|
}
|
||||||
|
|
||||||
|
public long getCreatedDate() {
|
||||||
|
return createdDate;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setCreatedDate(final long createdDate) {
|
||||||
|
this.createdDate = createdDate;
|
||||||
|
}
|
||||||
|
|
||||||
|
public long getModifiedDate() {
|
||||||
|
return modifiedDate;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setModifiedDate(final long modifiedDate) {
|
||||||
|
this.modifiedDate = modifiedDate;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getCreatedBy() {
|
||||||
|
return createdBy;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setCreatedBy(final String createdBy) {
|
||||||
|
this.createdBy = createdBy;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getModifiedBy() {
|
||||||
|
return modifiedBy;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setModifiedBy(final String modifiedBy) {
|
||||||
|
this.modifiedBy = modifiedBy;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setOperation(final String operation) {
|
||||||
|
this.operation = operation;
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public int hashCode() {
|
public int hashCode() {
|
||||||
|
@ -103,4 +216,27 @@ public class Bar implements Serializable {
|
||||||
return builder.toString();
|
return builder.toString();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@PrePersist
|
||||||
|
public void onPrePersist() {
|
||||||
|
logger.info("@PrePersist");
|
||||||
|
audit(OPERATION.INSERT);
|
||||||
|
}
|
||||||
|
|
||||||
|
@PreUpdate
|
||||||
|
public void onPreUpdate() {
|
||||||
|
logger.info("@PreUpdate");
|
||||||
|
audit(OPERATION.UPDATE);
|
||||||
|
}
|
||||||
|
|
||||||
|
@PreRemove
|
||||||
|
public void onPreRemove() {
|
||||||
|
logger.info("@PreRemove");
|
||||||
|
audit(OPERATION.DELETE);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void audit(final OPERATION operation) {
|
||||||
|
setOperation(operation);
|
||||||
|
setTimestamp((new Date()).getTime());
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -12,7 +12,11 @@ import javax.persistence.Id;
|
||||||
import javax.persistence.JoinColumn;
|
import javax.persistence.JoinColumn;
|
||||||
import javax.persistence.ManyToOne;
|
import javax.persistence.ManyToOne;
|
||||||
|
|
||||||
|
import org.hibernate.envers.Audited;
|
||||||
|
|
||||||
@Entity
|
@Entity
|
||||||
|
@Audited
|
||||||
|
// @Proxy(lazy = false)
|
||||||
public class Foo implements Serializable {
|
public class Foo implements Serializable {
|
||||||
|
|
||||||
@Id
|
@Id
|
||||||
|
|
|
@ -0,0 +1,8 @@
|
||||||
|
package org.baeldung.persistence.service;
|
||||||
|
|
||||||
|
import org.baeldung.persistence.dao.common.IAuditOperations;
|
||||||
|
import org.baeldung.persistence.model.Bar;
|
||||||
|
|
||||||
|
public interface IBarAuditableService extends IBarService, IAuditOperations<Bar> {
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,8 @@
|
||||||
|
package org.baeldung.persistence.service;
|
||||||
|
|
||||||
|
import org.baeldung.persistence.dao.common.IOperations;
|
||||||
|
import org.baeldung.persistence.model.Bar;
|
||||||
|
|
||||||
|
public interface IBarService extends IOperations<Bar> {
|
||||||
|
//
|
||||||
|
}
|
|
@ -0,0 +1,8 @@
|
||||||
|
package org.baeldung.persistence.service;
|
||||||
|
|
||||||
|
import org.baeldung.persistence.dao.common.IAuditOperations;
|
||||||
|
import org.baeldung.persistence.model.Foo;
|
||||||
|
|
||||||
|
public interface IFooAuditableService extends IFooService, IAuditOperations<Foo> {
|
||||||
|
//
|
||||||
|
}
|
|
@ -0,0 +1,30 @@
|
||||||
|
package org.baeldung.persistence.service.common;
|
||||||
|
|
||||||
|
import java.io.Serializable;
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
import org.baeldung.persistence.dao.common.IAuditOperations;
|
||||||
|
import org.baeldung.persistence.dao.common.IOperations;
|
||||||
|
import org.springframework.transaction.annotation.Transactional;
|
||||||
|
|
||||||
|
@Transactional(value = "hibernateTransactionManager")
|
||||||
|
public abstract class AbstractHibernateAuditableService<T extends Serializable> extends AbstractHibernateService<T> implements IOperations<T>, IAuditOperations<T> {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public List<T> getEntitiesAtRevision(final Number revision) {
|
||||||
|
return getAuditableDao().getEntitiesAtRevision(revision);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public List<T> getEntitiesModifiedAtRevision(final Number revision) {
|
||||||
|
return getAuditableDao().getEntitiesModifiedAtRevision(revision);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public List<T> getRevisions() {
|
||||||
|
return getAuditableDao().getRevisions();
|
||||||
|
}
|
||||||
|
|
||||||
|
abstract protected IAuditOperations<T> getAuditableDao();
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,42 @@
|
||||||
|
package org.baeldung.persistence.service.common;
|
||||||
|
|
||||||
|
import java.io.Serializable;
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
import org.baeldung.persistence.dao.common.IOperations;
|
||||||
|
import org.springframework.transaction.annotation.Transactional;
|
||||||
|
|
||||||
|
@Transactional(value = "hibernateTransactionManager")
|
||||||
|
public abstract class AbstractHibernateService<T extends Serializable> extends AbstractService<T> implements IOperations<T> {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public T findOne(final long id) {
|
||||||
|
return super.findOne(id);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public List<T> findAll() {
|
||||||
|
return super.findAll();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void create(final T entity) {
|
||||||
|
super.create(entity);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public T update(final T entity) {
|
||||||
|
return super.update(entity);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void delete(final T entity) {
|
||||||
|
super.delete(entity);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void deleteById(final long entityId) {
|
||||||
|
super.deleteById(entityId);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,42 @@
|
||||||
|
package org.baeldung.persistence.service.common;
|
||||||
|
|
||||||
|
import java.io.Serializable;
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
import org.baeldung.persistence.dao.common.IOperations;
|
||||||
|
import org.springframework.transaction.annotation.Transactional;
|
||||||
|
|
||||||
|
@Transactional(value = "jpaTransactionManager")
|
||||||
|
public abstract class AbstractJpaService<T extends Serializable> extends AbstractService<T> implements IOperations<T> {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public T findOne(final long id) {
|
||||||
|
return super.findOne(id);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public List<T> findAll() {
|
||||||
|
return super.findAll();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void create(final T entity) {
|
||||||
|
super.create(entity);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public T update(final T entity) {
|
||||||
|
return super.update(entity);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void delete(final T entity) {
|
||||||
|
super.delete(entity);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void deleteById(final long entityId) {
|
||||||
|
super.deleteById(entityId);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -4,9 +4,7 @@ import java.io.Serializable;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
|
||||||
import org.baeldung.persistence.dao.common.IOperations;
|
import org.baeldung.persistence.dao.common.IOperations;
|
||||||
import org.springframework.transaction.annotation.Transactional;
|
|
||||||
|
|
||||||
@Transactional
|
|
||||||
public abstract class AbstractService<T extends Serializable> implements IOperations<T> {
|
public abstract class AbstractService<T extends Serializable> implements IOperations<T> {
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
|
|
@ -0,0 +1,46 @@
|
||||||
|
package org.baeldung.persistence.service.common;
|
||||||
|
|
||||||
|
import java.io.Serializable;
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
import org.baeldung.persistence.dao.common.IOperations;
|
||||||
|
import org.springframework.data.repository.CrudRepository;
|
||||||
|
import org.springframework.transaction.annotation.Transactional;
|
||||||
|
|
||||||
|
import com.google.common.collect.Lists;
|
||||||
|
|
||||||
|
@Transactional(value = "jpaTransactionManager")
|
||||||
|
public abstract class AbstractSpringDataJpaService<T extends Serializable> implements IOperations<T> {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public T findOne(final long id) {
|
||||||
|
return getDao().findOne(Long.valueOf(id));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public List<T> findAll() {
|
||||||
|
return Lists.newArrayList(getDao().findAll());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void create(final T entity) {
|
||||||
|
getDao().save(entity);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public T update(final T entity) {
|
||||||
|
return getDao().save(entity);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void delete(final T entity) {
|
||||||
|
getDao().delete(entity);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void deleteById(final long entityId) {
|
||||||
|
getDao().delete(Long.valueOf(entityId));
|
||||||
|
}
|
||||||
|
|
||||||
|
protected abstract CrudRepository<T, Serializable> getDao();
|
||||||
|
}
|
|
@ -0,0 +1,41 @@
|
||||||
|
package org.baeldung.persistence.service.impl;
|
||||||
|
|
||||||
|
import org.baeldung.persistence.dao.IBarAuditableDao;
|
||||||
|
import org.baeldung.persistence.dao.IBarDao;
|
||||||
|
import org.baeldung.persistence.dao.common.IAuditOperations;
|
||||||
|
import org.baeldung.persistence.dao.common.IOperations;
|
||||||
|
import org.baeldung.persistence.model.Bar;
|
||||||
|
import org.baeldung.persistence.service.IBarAuditableService;
|
||||||
|
import org.baeldung.persistence.service.common.AbstractHibernateAuditableService;
|
||||||
|
import org.springframework.beans.factory.annotation.Autowired;
|
||||||
|
import org.springframework.beans.factory.annotation.Qualifier;
|
||||||
|
import org.springframework.stereotype.Service;
|
||||||
|
|
||||||
|
@Service
|
||||||
|
public class BarAuditableService extends AbstractHibernateAuditableService<Bar> implements IBarAuditableService {
|
||||||
|
|
||||||
|
@Autowired
|
||||||
|
@Qualifier("barHibernateDao")
|
||||||
|
private IBarDao dao;
|
||||||
|
|
||||||
|
@Autowired
|
||||||
|
@Qualifier("barHibernateAuditableDao")
|
||||||
|
private IBarAuditableDao auditDao;
|
||||||
|
|
||||||
|
public BarAuditableService() {
|
||||||
|
super();
|
||||||
|
}
|
||||||
|
|
||||||
|
// API
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected IOperations<Bar> getDao() {
|
||||||
|
return dao;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected IAuditOperations<Bar> getAuditableDao() {
|
||||||
|
return auditDao;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,30 @@
|
||||||
|
package org.baeldung.persistence.service.impl;
|
||||||
|
|
||||||
|
import org.baeldung.persistence.dao.IBarDao;
|
||||||
|
import org.baeldung.persistence.dao.common.IOperations;
|
||||||
|
import org.baeldung.persistence.model.Bar;
|
||||||
|
import org.baeldung.persistence.service.IBarService;
|
||||||
|
import org.baeldung.persistence.service.common.AbstractJpaService;
|
||||||
|
import org.springframework.beans.factory.annotation.Autowired;
|
||||||
|
import org.springframework.beans.factory.annotation.Qualifier;
|
||||||
|
import org.springframework.stereotype.Service;
|
||||||
|
|
||||||
|
@Service
|
||||||
|
public class BarJpaService extends AbstractJpaService<Bar> implements IBarService {
|
||||||
|
|
||||||
|
@Autowired
|
||||||
|
@Qualifier("barJpaDao")
|
||||||
|
private IBarDao dao;
|
||||||
|
|
||||||
|
public BarJpaService() {
|
||||||
|
super();
|
||||||
|
}
|
||||||
|
|
||||||
|
// API
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected IOperations<Bar> getDao() {
|
||||||
|
return dao;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,30 @@
|
||||||
|
package org.baeldung.persistence.service.impl;
|
||||||
|
|
||||||
|
import org.baeldung.persistence.dao.IBarDao;
|
||||||
|
import org.baeldung.persistence.dao.common.IOperations;
|
||||||
|
import org.baeldung.persistence.model.Bar;
|
||||||
|
import org.baeldung.persistence.service.IBarService;
|
||||||
|
import org.baeldung.persistence.service.common.AbstractHibernateService;
|
||||||
|
import org.springframework.beans.factory.annotation.Autowired;
|
||||||
|
import org.springframework.beans.factory.annotation.Qualifier;
|
||||||
|
import org.springframework.stereotype.Service;
|
||||||
|
|
||||||
|
@Service
|
||||||
|
public class BarService extends AbstractHibernateService<Bar> implements IBarService {
|
||||||
|
|
||||||
|
@Autowired
|
||||||
|
@Qualifier("barHibernateDao")
|
||||||
|
private IBarDao dao;
|
||||||
|
|
||||||
|
public BarService() {
|
||||||
|
super();
|
||||||
|
}
|
||||||
|
|
||||||
|
// API
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected IOperations<Bar> getDao() {
|
||||||
|
return dao;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,26 @@
|
||||||
|
package org.baeldung.persistence.service.impl;
|
||||||
|
|
||||||
|
import java.io.Serializable;
|
||||||
|
|
||||||
|
import org.baeldung.persistence.dao.IBarCrudRepository;
|
||||||
|
import org.baeldung.persistence.model.Bar;
|
||||||
|
import org.baeldung.persistence.service.IBarService;
|
||||||
|
import org.baeldung.persistence.service.common.AbstractSpringDataJpaService;
|
||||||
|
import org.springframework.beans.factory.annotation.Autowired;
|
||||||
|
import org.springframework.data.repository.CrudRepository;
|
||||||
|
|
||||||
|
public class BarSpringDataJpaService extends AbstractSpringDataJpaService<Bar> implements IBarService {
|
||||||
|
|
||||||
|
@Autowired
|
||||||
|
private IBarCrudRepository dao;
|
||||||
|
|
||||||
|
public BarSpringDataJpaService() {
|
||||||
|
super();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected CrudRepository<Bar, Serializable> getDao() {
|
||||||
|
return dao;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -4,12 +4,12 @@ import org.baeldung.persistence.dao.IChildDao;
|
||||||
import org.baeldung.persistence.dao.common.IOperations;
|
import org.baeldung.persistence.dao.common.IOperations;
|
||||||
import org.baeldung.persistence.model.Child;
|
import org.baeldung.persistence.model.Child;
|
||||||
import org.baeldung.persistence.service.IChildService;
|
import org.baeldung.persistence.service.IChildService;
|
||||||
import org.baeldung.persistence.service.common.AbstractService;
|
import org.baeldung.persistence.service.common.AbstractHibernateService;
|
||||||
import org.springframework.beans.factory.annotation.Autowired;
|
import org.springframework.beans.factory.annotation.Autowired;
|
||||||
import org.springframework.stereotype.Service;
|
import org.springframework.stereotype.Service;
|
||||||
|
|
||||||
@Service
|
@Service
|
||||||
public class ChildService extends AbstractService<Child>implements IChildService {
|
public class ChildService extends AbstractHibernateService<Child>implements IChildService {
|
||||||
|
|
||||||
@Autowired
|
@Autowired
|
||||||
private IChildDao dao;
|
private IChildDao dao;
|
||||||
|
|
|
@ -0,0 +1,41 @@
|
||||||
|
package org.baeldung.persistence.service.impl;
|
||||||
|
|
||||||
|
import org.baeldung.persistence.dao.IFooAuditableDao;
|
||||||
|
import org.baeldung.persistence.dao.IFooDao;
|
||||||
|
import org.baeldung.persistence.dao.common.IAuditOperations;
|
||||||
|
import org.baeldung.persistence.dao.common.IOperations;
|
||||||
|
import org.baeldung.persistence.model.Foo;
|
||||||
|
import org.baeldung.persistence.service.IFooAuditableService;
|
||||||
|
import org.baeldung.persistence.service.common.AbstractHibernateAuditableService;
|
||||||
|
import org.springframework.beans.factory.annotation.Autowired;
|
||||||
|
import org.springframework.beans.factory.annotation.Qualifier;
|
||||||
|
import org.springframework.stereotype.Service;
|
||||||
|
|
||||||
|
@Service
|
||||||
|
public class FooAuditableService extends AbstractHibernateAuditableService<Foo> implements IFooAuditableService {
|
||||||
|
|
||||||
|
@Autowired
|
||||||
|
@Qualifier("fooHibernateDao")
|
||||||
|
private IFooDao dao;
|
||||||
|
|
||||||
|
@Autowired
|
||||||
|
@Qualifier("fooHibernateAuditableDao")
|
||||||
|
private IFooAuditableDao auditDao;
|
||||||
|
|
||||||
|
public FooAuditableService() {
|
||||||
|
super();
|
||||||
|
}
|
||||||
|
|
||||||
|
// API
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected IOperations<Foo> getDao() {
|
||||||
|
return dao;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected IAuditOperations<Foo> getAuditableDao() {
|
||||||
|
return auditDao;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -4,14 +4,16 @@ import org.baeldung.persistence.dao.IFooDao;
|
||||||
import org.baeldung.persistence.dao.common.IOperations;
|
import org.baeldung.persistence.dao.common.IOperations;
|
||||||
import org.baeldung.persistence.model.Foo;
|
import org.baeldung.persistence.model.Foo;
|
||||||
import org.baeldung.persistence.service.IFooService;
|
import org.baeldung.persistence.service.IFooService;
|
||||||
import org.baeldung.persistence.service.common.AbstractService;
|
import org.baeldung.persistence.service.common.AbstractHibernateService;
|
||||||
import org.springframework.beans.factory.annotation.Autowired;
|
import org.springframework.beans.factory.annotation.Autowired;
|
||||||
|
import org.springframework.beans.factory.annotation.Qualifier;
|
||||||
import org.springframework.stereotype.Service;
|
import org.springframework.stereotype.Service;
|
||||||
|
|
||||||
@Service
|
@Service
|
||||||
public class FooService extends AbstractService<Foo>implements IFooService {
|
public class FooService extends AbstractHibernateService<Foo>implements IFooService {
|
||||||
|
|
||||||
@Autowired
|
@Autowired
|
||||||
|
@Qualifier("fooHibernateDao")
|
||||||
private IFooDao dao;
|
private IFooDao dao;
|
||||||
|
|
||||||
public FooService() {
|
public FooService() {
|
||||||
|
|
|
@ -4,12 +4,12 @@ import org.baeldung.persistence.dao.IParentDao;
|
||||||
import org.baeldung.persistence.dao.common.IOperations;
|
import org.baeldung.persistence.dao.common.IOperations;
|
||||||
import org.baeldung.persistence.model.Parent;
|
import org.baeldung.persistence.model.Parent;
|
||||||
import org.baeldung.persistence.service.IParentService;
|
import org.baeldung.persistence.service.IParentService;
|
||||||
import org.baeldung.persistence.service.common.AbstractService;
|
import org.baeldung.persistence.service.common.AbstractHibernateService;
|
||||||
import org.springframework.beans.factory.annotation.Autowired;
|
import org.springframework.beans.factory.annotation.Autowired;
|
||||||
import org.springframework.stereotype.Service;
|
import org.springframework.stereotype.Service;
|
||||||
|
|
||||||
@Service
|
@Service
|
||||||
public class ParentService extends AbstractService<Parent>implements IParentService {
|
public class ParentService extends AbstractHibernateService<Parent>implements IParentService {
|
||||||
|
|
||||||
@Autowired
|
@Autowired
|
||||||
private IParentDao dao;
|
private IParentDao dao;
|
||||||
|
|
|
@ -5,7 +5,24 @@ import java.util.Properties;
|
||||||
import javax.sql.DataSource;
|
import javax.sql.DataSource;
|
||||||
|
|
||||||
import org.apache.tomcat.dbcp.dbcp.BasicDataSource;
|
import org.apache.tomcat.dbcp.dbcp.BasicDataSource;
|
||||||
import org.hibernate.SessionFactory;
|
import org.baeldung.persistence.dao.IBarAuditableDao;
|
||||||
|
import org.baeldung.persistence.dao.IBarDao;
|
||||||
|
import org.baeldung.persistence.dao.IFooAuditableDao;
|
||||||
|
import org.baeldung.persistence.dao.IFooDao;
|
||||||
|
import org.baeldung.persistence.dao.impl.BarAuditableDao;
|
||||||
|
import org.baeldung.persistence.dao.impl.BarDao;
|
||||||
|
import org.baeldung.persistence.dao.impl.BarJpaDao;
|
||||||
|
import org.baeldung.persistence.dao.impl.FooAuditableDao;
|
||||||
|
import org.baeldung.persistence.dao.impl.FooDao;
|
||||||
|
import org.baeldung.persistence.service.IBarAuditableService;
|
||||||
|
import org.baeldung.persistence.service.IBarService;
|
||||||
|
import org.baeldung.persistence.service.IFooAuditableService;
|
||||||
|
import org.baeldung.persistence.service.IFooService;
|
||||||
|
import org.baeldung.persistence.service.impl.BarAuditableService;
|
||||||
|
import org.baeldung.persistence.service.impl.BarJpaService;
|
||||||
|
import org.baeldung.persistence.service.impl.BarSpringDataJpaService;
|
||||||
|
import org.baeldung.persistence.service.impl.FooAuditableService;
|
||||||
|
import org.baeldung.persistence.service.impl.FooService;
|
||||||
import org.springframework.beans.factory.annotation.Autowired;
|
import org.springframework.beans.factory.annotation.Autowired;
|
||||||
import org.springframework.context.annotation.Bean;
|
import org.springframework.context.annotation.Bean;
|
||||||
import org.springframework.context.annotation.ComponentScan;
|
import org.springframework.context.annotation.ComponentScan;
|
||||||
|
@ -13,14 +30,23 @@ import org.springframework.context.annotation.Configuration;
|
||||||
import org.springframework.context.annotation.PropertySource;
|
import org.springframework.context.annotation.PropertySource;
|
||||||
import org.springframework.core.env.Environment;
|
import org.springframework.core.env.Environment;
|
||||||
import org.springframework.dao.annotation.PersistenceExceptionTranslationPostProcessor;
|
import org.springframework.dao.annotation.PersistenceExceptionTranslationPostProcessor;
|
||||||
|
import org.springframework.data.jpa.repository.config.EnableJpaAuditing;
|
||||||
|
import org.springframework.data.jpa.repository.config.EnableJpaRepositories;
|
||||||
import org.springframework.orm.hibernate4.HibernateTransactionManager;
|
import org.springframework.orm.hibernate4.HibernateTransactionManager;
|
||||||
import org.springframework.orm.hibernate4.LocalSessionFactoryBean;
|
import org.springframework.orm.hibernate4.LocalSessionFactoryBean;
|
||||||
|
import org.springframework.orm.jpa.JpaTransactionManager;
|
||||||
|
import org.springframework.orm.jpa.JpaVendorAdapter;
|
||||||
|
import org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean;
|
||||||
|
import org.springframework.orm.jpa.vendor.HibernateJpaVendorAdapter;
|
||||||
|
import org.springframework.transaction.PlatformTransactionManager;
|
||||||
import org.springframework.transaction.annotation.EnableTransactionManagement;
|
import org.springframework.transaction.annotation.EnableTransactionManagement;
|
||||||
|
|
||||||
import com.google.common.base.Preconditions;
|
import com.google.common.base.Preconditions;
|
||||||
|
|
||||||
@Configuration
|
@Configuration
|
||||||
@EnableTransactionManagement
|
@EnableTransactionManagement
|
||||||
|
@EnableJpaRepositories(basePackages = { "org.baeldung.persistence" }, transactionManagerRef = "jpaTransactionManager")
|
||||||
|
@EnableJpaAuditing
|
||||||
@PropertySource({ "classpath:persistence-mysql.properties" })
|
@PropertySource({ "classpath:persistence-mysql.properties" })
|
||||||
@ComponentScan({ "org.baeldung.persistence" })
|
@ComponentScan({ "org.baeldung.persistence" })
|
||||||
public class PersistenceConfig {
|
public class PersistenceConfig {
|
||||||
|
@ -42,6 +68,19 @@ public class PersistenceConfig {
|
||||||
return sessionFactory;
|
return sessionFactory;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Bean
|
||||||
|
public LocalContainerEntityManagerFactoryBean entityManagerFactory() {
|
||||||
|
final LocalContainerEntityManagerFactoryBean emf = new LocalContainerEntityManagerFactoryBean();
|
||||||
|
emf.setDataSource(restDataSource());
|
||||||
|
emf.setPackagesToScan(new String[] { "org.baeldung.persistence.model" });
|
||||||
|
|
||||||
|
final JpaVendorAdapter vendorAdapter = new HibernateJpaVendorAdapter();
|
||||||
|
emf.setJpaVendorAdapter(vendorAdapter);
|
||||||
|
emf.setJpaProperties(hibernateProperties());
|
||||||
|
|
||||||
|
return emf;
|
||||||
|
}
|
||||||
|
|
||||||
@Bean
|
@Bean
|
||||||
public DataSource restDataSource() {
|
public DataSource restDataSource() {
|
||||||
final BasicDataSource dataSource = new BasicDataSource();
|
final BasicDataSource dataSource = new BasicDataSource();
|
||||||
|
@ -54,12 +93,17 @@ public class PersistenceConfig {
|
||||||
}
|
}
|
||||||
|
|
||||||
@Bean
|
@Bean
|
||||||
@Autowired
|
public PlatformTransactionManager hibernateTransactionManager() {
|
||||||
public HibernateTransactionManager transactionManager(final SessionFactory sessionFactory) {
|
final HibernateTransactionManager transactionManager = new HibernateTransactionManager();
|
||||||
final HibernateTransactionManager txManager = new HibernateTransactionManager();
|
transactionManager.setSessionFactory(sessionFactory().getObject());
|
||||||
txManager.setSessionFactory(sessionFactory);
|
return transactionManager;
|
||||||
|
}
|
||||||
|
|
||||||
return txManager;
|
@Bean
|
||||||
|
public PlatformTransactionManager jpaTransactionManager() {
|
||||||
|
final JpaTransactionManager transactionManager = new JpaTransactionManager();
|
||||||
|
transactionManager.setEntityManagerFactory(entityManagerFactory().getObject());
|
||||||
|
return transactionManager;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Bean
|
@Bean
|
||||||
|
@ -67,7 +111,57 @@ public class PersistenceConfig {
|
||||||
return new PersistenceExceptionTranslationPostProcessor();
|
return new PersistenceExceptionTranslationPostProcessor();
|
||||||
}
|
}
|
||||||
|
|
||||||
final Properties hibernateProperties() {
|
@Bean
|
||||||
|
public IBarService barJpaService() {
|
||||||
|
return new BarJpaService();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Bean
|
||||||
|
public IBarService barSpringDataJpaService() {
|
||||||
|
return new BarSpringDataJpaService();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Bean
|
||||||
|
public IFooService fooHibernateService() {
|
||||||
|
return new FooService();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Bean
|
||||||
|
public IBarAuditableService barHibernateAuditableService() {
|
||||||
|
return new BarAuditableService();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Bean
|
||||||
|
public IFooAuditableService fooHibernateAuditableService() {
|
||||||
|
return new FooAuditableService();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Bean
|
||||||
|
public IBarDao barJpaDao() {
|
||||||
|
return new BarJpaDao();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Bean
|
||||||
|
public IBarDao barHibernateDao() {
|
||||||
|
return new BarDao();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Bean
|
||||||
|
public IBarAuditableDao barHibernateAuditableDao() {
|
||||||
|
return new BarAuditableDao();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Bean
|
||||||
|
public IFooDao fooHibernateDao() {
|
||||||
|
return new FooDao();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Bean
|
||||||
|
public IFooAuditableDao fooHibernateAuditableDao() {
|
||||||
|
return new FooAuditableDao();
|
||||||
|
}
|
||||||
|
|
||||||
|
private final Properties hibernateProperties() {
|
||||||
final Properties hibernateProperties = new Properties();
|
final Properties hibernateProperties = new Properties();
|
||||||
hibernateProperties.setProperty("hibernate.hbm2ddl.auto", env.getProperty("hibernate.hbm2ddl.auto"));
|
hibernateProperties.setProperty("hibernate.hbm2ddl.auto", env.getProperty("hibernate.hbm2ddl.auto"));
|
||||||
hibernateProperties.setProperty("hibernate.dialect", env.getProperty("hibernate.dialect"));
|
hibernateProperties.setProperty("hibernate.dialect", env.getProperty("hibernate.dialect"));
|
||||||
|
@ -76,6 +170,9 @@ public class PersistenceConfig {
|
||||||
// hibernateProperties.setProperty("hibernate.format_sql", "true");
|
// hibernateProperties.setProperty("hibernate.format_sql", "true");
|
||||||
// hibernateProperties.setProperty("hibernate.globally_quoted_identifiers", "true");
|
// hibernateProperties.setProperty("hibernate.globally_quoted_identifiers", "true");
|
||||||
|
|
||||||
|
// Envers properties
|
||||||
|
hibernateProperties.setProperty("org.hibernate.envers.audit_table_suffix", env.getProperty("envers.audit_table_suffix"));
|
||||||
|
|
||||||
return hibernateProperties;
|
return hibernateProperties;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -8,3 +8,6 @@ jdbc.pass=tutorialmy5ql
|
||||||
hibernate.dialect=org.hibernate.dialect.MySQL5Dialect
|
hibernate.dialect=org.hibernate.dialect.MySQL5Dialect
|
||||||
hibernate.show_sql=false
|
hibernate.show_sql=false
|
||||||
hibernate.hbm2ddl.auto=create-drop
|
hibernate.hbm2ddl.auto=create-drop
|
||||||
|
|
||||||
|
# envers.X
|
||||||
|
envers.audit_table_suffix=_audit_log
|
||||||
|
|
|
@ -5,7 +5,8 @@
|
||||||
http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-4.2.xsd"
|
http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-4.2.xsd"
|
||||||
>
|
>
|
||||||
|
|
||||||
<http access-denied-page="/access-denied.html" use-expressions="true">
|
<http use-expressions="true">
|
||||||
|
<access-denied-handler error-page="/access-denied.html"/>
|
||||||
<intercept-url pattern="/access-denied*" access="hasAnyRole('ROLE_LOCATION_WRITE','ROLE_POLYGON_WRITE')"/>
|
<intercept-url pattern="/access-denied*" access="hasAnyRole('ROLE_LOCATION_WRITE','ROLE_POLYGON_WRITE')"/>
|
||||||
<intercept-url pattern="/admin/**" access="hasAnyRole('ROLE_ADMIN')"/>
|
<intercept-url pattern="/admin/**" access="hasAnyRole('ROLE_ADMIN')"/>
|
||||||
<intercept-url pattern="/organization/**" access="hasAnyRole('ROLE_ORGANIZATION')"/>
|
<intercept-url pattern="/organization/**" access="hasAnyRole('ROLE_ORGANIZATION')"/>
|
||||||
|
|
|
@ -0,0 +1,14 @@
|
||||||
|
package org.baeldung.persistence.audit;
|
||||||
|
|
||||||
|
import org.junit.runner.RunWith;
|
||||||
|
import org.junit.runners.Suite;
|
||||||
|
|
||||||
|
@RunWith(Suite.class)
|
||||||
|
@Suite.SuiteClasses({ // @formatter:off
|
||||||
|
EnversFooBarAuditTest.class,
|
||||||
|
JPABarAuditTest.class,
|
||||||
|
SpringDataJPABarAuditTest.class
|
||||||
|
}) // @formatter:on
|
||||||
|
public class AuditTestSuite {
|
||||||
|
//
|
||||||
|
}
|
|
@ -0,0 +1,142 @@
|
||||||
|
package org.baeldung.persistence.audit;
|
||||||
|
|
||||||
|
import static org.junit.Assert.assertEquals;
|
||||||
|
import static org.junit.Assert.assertNotNull;
|
||||||
|
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
import org.baeldung.persistence.model.Bar;
|
||||||
|
import org.baeldung.persistence.model.Foo;
|
||||||
|
import org.baeldung.persistence.service.IBarAuditableService;
|
||||||
|
import org.baeldung.persistence.service.IFooAuditableService;
|
||||||
|
import org.baeldung.spring.PersistenceConfig;
|
||||||
|
import org.hibernate.Session;
|
||||||
|
import org.hibernate.SessionFactory;
|
||||||
|
import org.junit.After;
|
||||||
|
import org.junit.AfterClass;
|
||||||
|
import org.junit.Before;
|
||||||
|
import org.junit.BeforeClass;
|
||||||
|
import org.junit.Test;
|
||||||
|
import org.junit.runner.RunWith;
|
||||||
|
import org.slf4j.Logger;
|
||||||
|
import org.slf4j.LoggerFactory;
|
||||||
|
import org.springframework.beans.factory.annotation.Autowired;
|
||||||
|
import org.springframework.beans.factory.annotation.Qualifier;
|
||||||
|
import org.springframework.test.context.ContextConfiguration;
|
||||||
|
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
|
||||||
|
import org.springframework.test.context.support.AnnotationConfigContextLoader;
|
||||||
|
|
||||||
|
@RunWith(SpringJUnit4ClassRunner.class)
|
||||||
|
@ContextConfiguration(classes = { PersistenceConfig.class }, loader = AnnotationConfigContextLoader.class)
|
||||||
|
public class EnversFooBarAuditTest {
|
||||||
|
|
||||||
|
private static Logger logger = LoggerFactory.getLogger(EnversFooBarAuditTest.class);
|
||||||
|
|
||||||
|
@BeforeClass
|
||||||
|
public static void setUpBeforeClass() throws Exception {
|
||||||
|
logger.info("setUpBeforeClass()");
|
||||||
|
}
|
||||||
|
|
||||||
|
@AfterClass
|
||||||
|
public static void tearDownAfterClass() throws Exception {
|
||||||
|
logger.info("tearDownAfterClass()");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Autowired
|
||||||
|
@Qualifier("fooHibernateAuditableService")
|
||||||
|
private IFooAuditableService fooService;
|
||||||
|
|
||||||
|
@Autowired
|
||||||
|
@Qualifier("barHibernateAuditableService")
|
||||||
|
private IBarAuditableService barService;
|
||||||
|
|
||||||
|
@Autowired
|
||||||
|
private SessionFactory sessionFactory;
|
||||||
|
|
||||||
|
private Session session;
|
||||||
|
|
||||||
|
@Before
|
||||||
|
public void setUp() throws Exception {
|
||||||
|
logger.info("setUp()");
|
||||||
|
makeRevisions();
|
||||||
|
session = sessionFactory.openSession();
|
||||||
|
}
|
||||||
|
|
||||||
|
@After
|
||||||
|
public void tearDown() throws Exception {
|
||||||
|
logger.info("tearDown()");
|
||||||
|
session.close();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void makeRevisions() {
|
||||||
|
final Bar bar = rev1();
|
||||||
|
rev2(bar);
|
||||||
|
rev3(bar);
|
||||||
|
rev4(bar);
|
||||||
|
}
|
||||||
|
|
||||||
|
// REV #1: insert BAR & FOO1
|
||||||
|
private Bar rev1() {
|
||||||
|
final Bar bar = new Bar("BAR");
|
||||||
|
final Foo foo1 = new Foo("FOO1");
|
||||||
|
foo1.setBar(bar);
|
||||||
|
fooService.create(foo1);
|
||||||
|
return bar;
|
||||||
|
}
|
||||||
|
|
||||||
|
// REV #2: insert FOO2 & update BAR
|
||||||
|
private void rev2(final Bar bar) {
|
||||||
|
final Foo foo2 = new Foo("FOO2");
|
||||||
|
foo2.setBar(bar);
|
||||||
|
fooService.create(foo2);
|
||||||
|
}
|
||||||
|
|
||||||
|
// REV #3: update BAR
|
||||||
|
private void rev3(final Bar bar) {
|
||||||
|
|
||||||
|
bar.setName("BAR1");
|
||||||
|
barService.update(bar);
|
||||||
|
}
|
||||||
|
|
||||||
|
// REV #4: insert FOO3 & update BAR
|
||||||
|
private void rev4(final Bar bar) {
|
||||||
|
|
||||||
|
final Foo foo3 = new Foo("FOO3");
|
||||||
|
foo3.setBar(bar);
|
||||||
|
fooService.create(foo3);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public final void whenFooBarsModified_thenFooBarsAudited() {
|
||||||
|
|
||||||
|
List<Bar> barRevisionList;
|
||||||
|
List<Foo> fooRevisionList;
|
||||||
|
|
||||||
|
// test Bar revisions
|
||||||
|
|
||||||
|
barRevisionList = barService.getRevisions();
|
||||||
|
|
||||||
|
assertNotNull(barRevisionList);
|
||||||
|
assertEquals(4, barRevisionList.size());
|
||||||
|
|
||||||
|
assertEquals("BAR", barRevisionList.get(0).getName());
|
||||||
|
assertEquals("BAR", barRevisionList.get(1).getName());
|
||||||
|
assertEquals("BAR1", barRevisionList.get(2).getName());
|
||||||
|
assertEquals("BAR1", barRevisionList.get(3).getName());
|
||||||
|
|
||||||
|
assertEquals(1, barRevisionList.get(0).getFooSet().size());
|
||||||
|
assertEquals(2, barRevisionList.get(1).getFooSet().size());
|
||||||
|
assertEquals(2, barRevisionList.get(2).getFooSet().size());
|
||||||
|
assertEquals(3, barRevisionList.get(3).getFooSet().size());
|
||||||
|
|
||||||
|
// test Foo revisions
|
||||||
|
|
||||||
|
fooRevisionList = fooService.getRevisions();
|
||||||
|
assertNotNull(fooRevisionList);
|
||||||
|
assertEquals(3, fooRevisionList.size());
|
||||||
|
assertEquals("FOO1", fooRevisionList.get(0).getName());
|
||||||
|
assertEquals("FOO2", fooRevisionList.get(1).getName());
|
||||||
|
assertEquals("FOO3", fooRevisionList.get(2).getName());
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,106 @@
|
||||||
|
package org.baeldung.persistence.audit;
|
||||||
|
|
||||||
|
import static org.junit.Assert.assertEquals;
|
||||||
|
import static org.junit.Assert.assertNotNull;
|
||||||
|
import static org.junit.Assert.assertTrue;
|
||||||
|
|
||||||
|
import javax.persistence.EntityManager;
|
||||||
|
import javax.persistence.EntityManagerFactory;
|
||||||
|
|
||||||
|
import org.baeldung.persistence.model.Bar;
|
||||||
|
import org.baeldung.persistence.model.Bar.OPERATION;
|
||||||
|
import org.baeldung.persistence.service.IBarService;
|
||||||
|
import org.baeldung.spring.PersistenceConfig;
|
||||||
|
import org.junit.After;
|
||||||
|
import org.junit.AfterClass;
|
||||||
|
import org.junit.Before;
|
||||||
|
import org.junit.BeforeClass;
|
||||||
|
import org.junit.Test;
|
||||||
|
import org.junit.runner.RunWith;
|
||||||
|
import org.slf4j.Logger;
|
||||||
|
import org.slf4j.LoggerFactory;
|
||||||
|
import org.springframework.beans.factory.annotation.Autowired;
|
||||||
|
import org.springframework.beans.factory.annotation.Qualifier;
|
||||||
|
import org.springframework.test.context.ContextConfiguration;
|
||||||
|
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
|
||||||
|
import org.springframework.test.context.support.AnnotationConfigContextLoader;
|
||||||
|
|
||||||
|
@RunWith(SpringJUnit4ClassRunner.class)
|
||||||
|
@ContextConfiguration(classes = { PersistenceConfig.class }, loader = AnnotationConfigContextLoader.class)
|
||||||
|
public class JPABarAuditTest {
|
||||||
|
|
||||||
|
private static Logger logger = LoggerFactory.getLogger(JPABarAuditTest.class);
|
||||||
|
|
||||||
|
@BeforeClass
|
||||||
|
public static void setUpBeforeClass() throws Exception {
|
||||||
|
logger.info("setUpBeforeClass()");
|
||||||
|
}
|
||||||
|
|
||||||
|
@AfterClass
|
||||||
|
public static void tearDownAfterClass() throws Exception {
|
||||||
|
logger.info("tearDownAfterClass()");
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@Autowired
|
||||||
|
@Qualifier("barJpaService")
|
||||||
|
private IBarService barService;
|
||||||
|
|
||||||
|
@Autowired
|
||||||
|
private EntityManagerFactory entityManagerFactory;
|
||||||
|
|
||||||
|
private EntityManager em;
|
||||||
|
|
||||||
|
|
||||||
|
@Before
|
||||||
|
public void setUp() throws Exception {
|
||||||
|
logger.info("setUp()");
|
||||||
|
em = entityManagerFactory.createEntityManager();
|
||||||
|
}
|
||||||
|
|
||||||
|
@After
|
||||||
|
public void tearDown() throws Exception {
|
||||||
|
logger.info("tearDown()");
|
||||||
|
em.close();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public final void whenBarsModified_thenBarsAudited() {
|
||||||
|
|
||||||
|
// insert BAR1
|
||||||
|
Bar bar1 = new Bar("BAR1");
|
||||||
|
barService.create(bar1);
|
||||||
|
|
||||||
|
// update BAR1
|
||||||
|
bar1.setName("BAR1a");
|
||||||
|
barService.update(bar1);
|
||||||
|
|
||||||
|
// insert BAR2
|
||||||
|
Bar bar2 = new Bar("BAR2");
|
||||||
|
barService.create(bar2);
|
||||||
|
|
||||||
|
// update BAR1
|
||||||
|
bar1.setName("BAR1b");
|
||||||
|
barService.update(bar1);
|
||||||
|
|
||||||
|
|
||||||
|
// get BAR1 and BAR2 from the DB and check the audit values
|
||||||
|
// detach instances from persistence context to make sure we fire db
|
||||||
|
em.detach(bar1);
|
||||||
|
em.detach(bar2);
|
||||||
|
bar1 = barService.findOne(bar1.getId());
|
||||||
|
bar2 = barService.findOne(bar2.getId());
|
||||||
|
|
||||||
|
assertNotNull(bar1);
|
||||||
|
assertNotNull(bar2);
|
||||||
|
assertEquals(OPERATION.UPDATE, bar1.getOperation());
|
||||||
|
assertEquals(OPERATION.INSERT, bar2.getOperation());
|
||||||
|
assertTrue(bar1.getTimestamp() > bar2.getTimestamp());
|
||||||
|
|
||||||
|
barService.deleteById(bar1.getId());
|
||||||
|
barService.deleteById(bar2.getId());
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,76 @@
|
||||||
|
package org.baeldung.persistence.audit;
|
||||||
|
|
||||||
|
import static org.junit.Assert.assertEquals;
|
||||||
|
import static org.junit.Assert.assertTrue;
|
||||||
|
|
||||||
|
import javax.persistence.EntityManager;
|
||||||
|
import javax.persistence.EntityManagerFactory;
|
||||||
|
|
||||||
|
import org.baeldung.persistence.model.Bar;
|
||||||
|
import org.baeldung.persistence.service.IBarService;
|
||||||
|
import org.baeldung.spring.PersistenceConfig;
|
||||||
|
import org.junit.After;
|
||||||
|
import org.junit.AfterClass;
|
||||||
|
import org.junit.Before;
|
||||||
|
import org.junit.BeforeClass;
|
||||||
|
import org.junit.Test;
|
||||||
|
import org.junit.runner.RunWith;
|
||||||
|
import org.slf4j.Logger;
|
||||||
|
import org.slf4j.LoggerFactory;
|
||||||
|
import org.springframework.beans.factory.annotation.Autowired;
|
||||||
|
import org.springframework.beans.factory.annotation.Qualifier;
|
||||||
|
import org.springframework.security.test.context.support.WithMockUser;
|
||||||
|
import org.springframework.test.context.ContextConfiguration;
|
||||||
|
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
|
||||||
|
import org.springframework.test.context.support.AnnotationConfigContextLoader;
|
||||||
|
|
||||||
|
@RunWith(SpringJUnit4ClassRunner.class)
|
||||||
|
@ContextConfiguration(classes = { PersistenceConfig.class }, loader = AnnotationConfigContextLoader.class)
|
||||||
|
public class SpringDataJPABarAuditTest {
|
||||||
|
|
||||||
|
private static Logger logger = LoggerFactory.getLogger(SpringDataJPABarAuditTest.class);
|
||||||
|
|
||||||
|
@BeforeClass
|
||||||
|
public static void setUpBeforeClass() throws Exception {
|
||||||
|
logger.info("setUpBeforeClass()");
|
||||||
|
}
|
||||||
|
|
||||||
|
@AfterClass
|
||||||
|
public static void tearDownAfterClass() throws Exception {
|
||||||
|
logger.info("tearDownAfterClass()");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Autowired
|
||||||
|
@Qualifier("barSpringDataJpaService")
|
||||||
|
private IBarService barService;
|
||||||
|
|
||||||
|
@Autowired
|
||||||
|
private EntityManagerFactory entityManagerFactory;
|
||||||
|
|
||||||
|
private EntityManager em;
|
||||||
|
|
||||||
|
@Before
|
||||||
|
public void setUp() throws Exception {
|
||||||
|
logger.info("setUp()");
|
||||||
|
em = entityManagerFactory.createEntityManager();
|
||||||
|
}
|
||||||
|
|
||||||
|
@After
|
||||||
|
public void tearDown() throws Exception {
|
||||||
|
logger.info("tearDown()");
|
||||||
|
em.close();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
@WithMockUser(username = "tutorialuser")
|
||||||
|
public final void whenBarsModified_thenBarsAudited() {
|
||||||
|
Bar bar = new Bar("BAR1");
|
||||||
|
barService.create(bar);
|
||||||
|
assertEquals(bar.getCreatedDate(), bar.getModifiedDate());
|
||||||
|
assertEquals("tutorialuser", bar.getCreatedBy(), bar.getModifiedBy());
|
||||||
|
bar.setName("BAR2");
|
||||||
|
bar = barService.update(bar);
|
||||||
|
assertTrue(bar.getCreatedDate() < bar.getModifiedDate());
|
||||||
|
assertEquals("tutorialuser", bar.getCreatedBy(), bar.getModifiedBy());
|
||||||
|
}
|
||||||
|
}
|
|
@ -8,6 +8,7 @@ import org.junit.Ignore;
|
||||||
import org.junit.Test;
|
import org.junit.Test;
|
||||||
import org.junit.runner.RunWith;
|
import org.junit.runner.RunWith;
|
||||||
import org.springframework.beans.factory.annotation.Autowired;
|
import org.springframework.beans.factory.annotation.Autowired;
|
||||||
|
import org.springframework.beans.factory.annotation.Qualifier;
|
||||||
import org.springframework.dao.DataAccessException;
|
import org.springframework.dao.DataAccessException;
|
||||||
import org.springframework.dao.DataIntegrityViolationException;
|
import org.springframework.dao.DataIntegrityViolationException;
|
||||||
import org.springframework.dao.InvalidDataAccessApiUsageException;
|
import org.springframework.dao.InvalidDataAccessApiUsageException;
|
||||||
|
@ -20,6 +21,7 @@ import org.springframework.test.context.support.AnnotationConfigContextLoader;
|
||||||
public class FooServicePersistenceIntegrationTest {
|
public class FooServicePersistenceIntegrationTest {
|
||||||
|
|
||||||
@Autowired
|
@Autowired
|
||||||
|
@Qualifier("fooHibernateService")
|
||||||
private IFooService service;
|
private IFooService service;
|
||||||
|
|
||||||
// tests
|
// tests
|
||||||
|
|
Loading…
Reference in New Issue