All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.hibernate.envers.strategy.internal.ValidityAuditStrategy Maven / Gradle / Ivy

The newest version!
/*
 * Hibernate, Relational Persistence for Idiomatic Java
 *
 * License: GNU Lesser General Public License (LGPL), version 2.1 or later.
 * See the lgpl.txt file in the root directory or .
 */
package org.hibernate.envers.strategy.internal;

import static org.hibernate.envers.internal.entities.mapper.relation.query.QueryConstants.MIDDLE_ENTITY_ALIAS;
import static org.hibernate.envers.internal.entities.mapper.relation.query.QueryConstants.REVISION_PARAMETER;

import java.io.Serializable;
import java.sql.Connection;
import java.sql.PreparedStatement;
import java.sql.SQLException;
import java.util.Date;
import java.util.List;
import java.util.Map;

import org.dom4j.Element;
import org.hibernate.LockOptions;
import org.hibernate.Session;
import org.hibernate.action.spi.BeforeTransactionCompletionProcess;
import org.hibernate.engine.spi.SessionImplementor;
import org.hibernate.engine.spi.SharedSessionContractImplementor;
import org.hibernate.envers.RevisionType;
import org.hibernate.envers.configuration.internal.AuditEntitiesConfiguration;
import org.hibernate.envers.configuration.internal.GlobalConfiguration;
import org.hibernate.envers.configuration.internal.metadata.MetadataTools;
import org.hibernate.envers.internal.entities.PropertyData;
import org.hibernate.envers.internal.entities.mapper.PersistentCollectionChangeData;
import org.hibernate.envers.internal.entities.mapper.relation.MiddleComponentData;
import org.hibernate.envers.internal.entities.mapper.relation.MiddleIdData;
import org.hibernate.envers.internal.synchronization.SessionCacheCleaner;
import org.hibernate.envers.internal.tools.ReflectionTools;
import org.hibernate.envers.internal.tools.query.Parameters;
import org.hibernate.envers.internal.tools.query.QueryBuilder;
import org.hibernate.envers.strategy.AuditStrategy;
import org.hibernate.envers.strategy.spi.MappingContext;
import org.hibernate.event.spi.EventSource;
import org.hibernate.jdbc.ReturningWork;
import org.hibernate.persister.entity.Queryable;
import org.hibernate.persister.entity.UnionSubclassEntityPersister;
import org.hibernate.property.access.spi.Getter;
import org.hibernate.service.ServiceRegistry;
import org.hibernate.sql.Update;
import org.hibernate.type.CollectionType;
import org.hibernate.type.ComponentType;
import org.hibernate.type.MapType;
import org.hibernate.type.MaterializedClobType;
import org.hibernate.type.MaterializedNClobType;
import org.hibernate.type.TimestampType;
import org.hibernate.type.Type;

/**
 * An audit strategy implementation that persists and fetches audit information using a validity
 * algorithm, based on the start-revision and end-revision of a row in the audit table schema.
 * 

* This algorithm works as follows: *

    *
  • For a new row, only the start-revision column is set in the row.
  • *
  • Concurrently, the end-revision of the prior audit row is set to the current revision
  • *
  • Queries using a between start and end revision predicate rather than using subqueries.
  • *
*

* This has a few important consequences which must be considered: *

    *
  • Persisting audit information is sightly slower due to an extra update required
  • *
  • Retreiving audit information is considerably faster
  • *
* * @author Stephanie Pau * @author Adam Warski (adam at warski dot org) * @author Lukasz Antoniak (lukasz dot antoniak at gmail dot com) * @author Chris Cranford */ public class ValidityAuditStrategy implements AuditStrategy { /** * getter for the revision entity field annotated with @RevisionTimestamp */ private Getter revisionTimestampGetter; private final SessionCacheCleaner sessionCacheCleaner; public ValidityAuditStrategy() { sessionCacheCleaner = new SessionCacheCleaner(); } @Override public void postInitialize( Class revisionInfoClass, PropertyData revisionInfoTimestampData, ServiceRegistry serviceRegistry) { // further initialization required final Getter revisionTimestampGetter = ReflectionTools.getGetter( revisionInfoClass, revisionInfoTimestampData, serviceRegistry ); setRevisionTimestampGetter( revisionTimestampGetter ); } @Override public void addAdditionalColumns(MappingContext mappingContext) { // Add the end-revision field, if the appropriate strategy is used. Element endRevMapping = (Element) mappingContext.getRevisionEntityMapping().clone(); endRevMapping.setName( "many-to-one" ); endRevMapping.addAttribute( "name", mappingContext.getAuditEntityConfiguration().getRevisionEndFieldName() ); MetadataTools.addOrModifyColumn( endRevMapping, mappingContext.getAuditEntityConfiguration().getRevisionEndFieldName() ); mappingContext.getAuditEntityMapping().add( endRevMapping ); if ( mappingContext.getAuditEntityConfiguration().isRevisionEndTimestampEnabled() ) { // add a column for the timestamp of the end revision final String revisionInfoTimestampSqlType = TimestampType.INSTANCE.getName(); final Element timestampProperty = MetadataTools.addProperty( mappingContext.getAuditEntityMapping(), mappingContext.getAuditEntityConfiguration().getRevisionEndTimestampFieldName(), revisionInfoTimestampSqlType, true, true, false ); MetadataTools.addColumn( timestampProperty, mappingContext.getAuditEntityConfiguration().getRevisionEndTimestampFieldName(), null, null, null, null, null, null ); } } @Override public void perform( final Session session, final String entityName, final AuditEntitiesConfiguration audEntitiesCfg, final Serializable id, final Object data, final Object revision) { final String auditedEntityName = audEntitiesCfg.getAuditEntityName( entityName ); final String revisionInfoEntityName = audEntitiesCfg.getRevisionInfoEntityName(); // Save the audit data session.save( auditedEntityName, data ); // Update the end date of the previous row. // // When application reuses identifiers of previously removed entities: // The UPDATE statement will no-op if an entity with a given identifier has been // inserted for the first time. But in case a deleted primary key value was // reused, this guarantees correct strategy behavior: exactly one row with // null end date exists for each identifier. final boolean reuseEntityIdentifier = audEntitiesCfg.getEnversService().getGlobalConfiguration().isAllowIdentifierReuse(); if ( reuseEntityIdentifier || getRevisionType( audEntitiesCfg, data ) != RevisionType.ADD ) { // Register transaction completion process to guarantee execution of UPDATE statement after INSERT. ( (EventSource) session ).getActionQueue().registerProcess( new BeforeTransactionCompletionProcess() { @Override public void doBeforeTransactionCompletion(final SessionImplementor sessionImplementor) { final Queryable productionEntityQueryable = getQueryable( entityName, sessionImplementor ); final Queryable rootProductionEntityQueryable = getQueryable( productionEntityQueryable.getRootEntityName(), sessionImplementor ); final Queryable auditedEntityQueryable = getQueryable( auditedEntityName, sessionImplementor ); final Queryable rootAuditedEntityQueryable = getQueryable( auditedEntityQueryable.getRootEntityName(), sessionImplementor ); final String updateTableName; if ( UnionSubclassEntityPersister.class.isInstance( rootProductionEntityQueryable ) ) { // this is the condition causing all the problems in terms of the generated SQL UPDATE // the problem being that we currently try to update the in-line view made up of the union query // // this is extremely hacky means to get the root table name for the union subclass style entities. // hacky because it relies on internal behavior of UnionSubclassEntityPersister // !!!!!! NOTICE - using subclass persister, not root !!!!!! updateTableName = auditedEntityQueryable.getSubclassTableName( 0 ); } else { updateTableName = rootAuditedEntityQueryable.getTableName(); } final Type revisionInfoIdType = sessionImplementor.getFactory().getMetamodel().entityPersister( revisionInfoEntityName ).getIdentifierType(); final String revEndColumnName = rootAuditedEntityQueryable.toColumns( audEntitiesCfg.getRevisionEndFieldName() )[0]; final boolean isRevisionEndTimestampEnabled = audEntitiesCfg.isRevisionEndTimestampEnabled(); // update audit_ent set REVEND = ? [, REVEND_TSTMP = ?] where (prod_ent_id) = ? and REV <> ? and REVEND is null final Update update = new Update( sessionImplementor.getFactory().getJdbcServices().getDialect() ).setTableName( updateTableName ); // set REVEND = ? update.addColumn( revEndColumnName ); // set [, REVEND_TSTMP = ?] if ( isRevisionEndTimestampEnabled ) { update.addColumn( rootAuditedEntityQueryable.toColumns( audEntitiesCfg.getRevisionEndTimestampFieldName() )[0] ); } // where (prod_ent_id) = ? update.addPrimaryKeyColumns( rootProductionEntityQueryable.getIdentifierColumnNames() ); // where REV <> ? update.addWhereColumn( rootAuditedEntityQueryable.toColumns( audEntitiesCfg.getRevisionNumberPath() )[0], "<> ?" ); // where REVEND is null update.addWhereColumn( revEndColumnName, " is null" ); // Now lets execute the sql... final String updateSql = update.toStatementString(); int rowCount = sessionImplementor.doReturningWork( new ReturningWork() { @Override public Integer execute(Connection connection) throws SQLException { PreparedStatement preparedStatement = sessionImplementor .getJdbcCoordinator().getStatementPreparer().prepareStatement( updateSql ); try { int index = 1; // set REVEND = ? final Number revisionNumber = audEntitiesCfg.getEnversService() .getRevisionInfoNumberReader() .getRevisionNumber( revision ); revisionInfoIdType.nullSafeSet( preparedStatement, revisionNumber, index, sessionImplementor ); index += revisionInfoIdType.getColumnSpan( sessionImplementor.getFactory() ); // set [, REVEND_TSTMP = ?] if ( isRevisionEndTimestampEnabled ) { final Object revEndTimestampObj = revisionTimestampGetter.get( revision ); final Date revisionEndTimestamp = convertRevEndTimestampToDate( revEndTimestampObj ); final Type revEndTsType = rootAuditedEntityQueryable.getPropertyType( audEntitiesCfg.getRevisionEndTimestampFieldName() ); revEndTsType.nullSafeSet( preparedStatement, revisionEndTimestamp, index, sessionImplementor ); index += revEndTsType.getColumnSpan( sessionImplementor.getFactory() ); } // where (prod_ent_id) = ? final Type idType = rootProductionEntityQueryable.getIdentifierType(); idType.nullSafeSet( preparedStatement, id, index, sessionImplementor ); index += idType.getColumnSpan( sessionImplementor.getFactory() ); // where REV <> ? final Type revType = rootAuditedEntityQueryable.getPropertyType( audEntitiesCfg.getRevisionNumberPath() ); revType.nullSafeSet( preparedStatement, revisionNumber, index, sessionImplementor ); // where REVEND is null // nothing to bind.... return sessionImplementor .getJdbcCoordinator().getResultSetReturn().executeUpdate( preparedStatement ); } finally { sessionImplementor.getJdbcCoordinator().getLogicalConnection().getResourceRegistry().release( preparedStatement ); sessionImplementor.getJdbcCoordinator().afterStatementExecution(); } } } ); if ( rowCount != 1 && ( !reuseEntityIdentifier || ( getRevisionType( audEntitiesCfg, data ) != RevisionType.ADD ) ) ) { throw new RuntimeException( "Cannot update previous revision for entity " + auditedEntityName + " and id " + id ); } } } ); } sessionCacheCleaner.scheduleAuditDataRemoval( session, data ); } @Override @SuppressWarnings({"unchecked"}) public void performCollectionChange( Session session, String entityName, String propertyName, AuditEntitiesConfiguration auditEntitiesConfiguration, PersistentCollectionChangeData persistentCollectionChangeData, Object revision) { final QueryBuilder qb = new QueryBuilder( persistentCollectionChangeData.getEntityName(), MIDDLE_ENTITY_ALIAS, ( (SharedSessionContractImplementor) session ).getFactory() ); final String originalIdPropName = auditEntitiesConfiguration.getOriginalIdPropName(); final Map originalId = (Map) persistentCollectionChangeData.getData().get( originalIdPropName ); final String revisionFieldName = auditEntitiesConfiguration.getRevisionFieldName(); final String revisionTypePropName = auditEntitiesConfiguration.getRevisionTypePropName(); final String ordinalPropName = auditEntitiesConfiguration.getEmbeddableSetOrdinalPropertyName(); // Adding a parameter for each id component, except the rev number and type. for ( Map.Entry originalIdEntry : originalId.entrySet() ) { if ( !revisionFieldName.equals( originalIdEntry.getKey() ) && !revisionTypePropName.equals( originalIdEntry.getKey() ) && !ordinalPropName.equals( originalIdEntry.getKey() ) ) { qb.getRootParameters().addWhereWithParam( originalIdPropName + "." + originalIdEntry.getKey(), true, "=", originalIdEntry.getValue() ); } } if ( isNonIdentifierWhereConditionsRequired( entityName, propertyName, (SessionImplementor) session ) ) { addNonIdentifierWhereConditions( qb, persistentCollectionChangeData.getData(), originalIdPropName ); } addEndRevisionNullRestriction( auditEntitiesConfiguration, qb.getRootParameters() ); final List l = qb.toQuery( session ).setLockOptions( LockOptions.UPGRADE ).list(); // Update the last revision if one exists. // HHH-5967: with collections, the same element can be added and removed multiple times. So even if it's an // ADD, we may need to update the last revision. if ( l.size() > 0 ) { updateLastRevision( session, auditEntitiesConfiguration, l, originalId, persistentCollectionChangeData.getEntityName(), revision ); } // Save the audit data session.save( persistentCollectionChangeData.getEntityName(), persistentCollectionChangeData.getData() ); sessionCacheCleaner.scheduleAuditDataRemoval( session, persistentCollectionChangeData.getData() ); } /** * {@inheritDoc} * * For this implmenetation, the revision-end column is used *

* {@code e.revision <= :revision and (e.endRevision > :revision or e.endRevision is null} */ @Override public void addEntityAtRevisionRestriction( GlobalConfiguration globalCfg, QueryBuilder rootQueryBuilder, Parameters parameters, String revisionProperty, String revisionEndProperty, boolean addAlias, MiddleIdData idData, String revisionPropertyPath, String originalIdPropertyName, String alias1, String alias2, boolean inclusive) { addRevisionRestriction( parameters, revisionProperty, revisionEndProperty, addAlias, inclusive ); } /** * {@inheritDoc} * * For this implmenetation, the revision-end column is used *

* {@code e.revision <= :revision and (e.endRevision > :revision or e.endRevision is null} */ @Override public void addAssociationAtRevisionRestriction( QueryBuilder rootQueryBuilder, Parameters parameters, String revisionProperty, String revisionEndProperty, boolean addAlias, MiddleIdData referencingIdData, String versionsMiddleEntityName, String eeOriginalIdPropertyPath, String revisionPropertyPath, String originalIdPropertyName, String alias1, boolean inclusive, MiddleComponentData... componentDatas) { addRevisionRestriction( parameters, revisionProperty, revisionEndProperty, addAlias, inclusive ); } /** * @deprecated since 5.4 with no replacement. */ @Deprecated public void setRevisionTimestampGetter(Getter revisionTimestampGetter) { this.revisionTimestampGetter = revisionTimestampGetter; } private void addRevisionRestriction( Parameters rootParameters, String revisionProperty, String revisionEndProperty, boolean addAlias, boolean inclusive) { // e.revision <= _revision and (e.endRevision > _revision or e.endRevision is null) Parameters subParm = rootParameters.addSubParameters( "or" ); rootParameters.addWhereWithNamedParam( revisionProperty, addAlias, inclusive ? "<=" : "<", REVISION_PARAMETER ); subParm.addWhereWithNamedParam( revisionEndProperty + ".id", addAlias, inclusive ? ">" : ">=", REVISION_PARAMETER ); subParm.addWhere( revisionEndProperty, addAlias, "is", "null", false ); } @SuppressWarnings({"unchecked"}) private RevisionType getRevisionType(AuditEntitiesConfiguration auditEntitiesConfiguration, Object data) { return (RevisionType) ( (Map) data ).get( auditEntitiesConfiguration.getRevisionTypePropName() ); } @SuppressWarnings({"unchecked"}) private void updateLastRevision( Session session, AuditEntitiesConfiguration auditEntitiesConfiguration, List l, Object id, String auditedEntityName, Object revision) { // There should be one entry if ( l.size() == 1 ) { // Setting the end revision to be the current rev Object previousData = l.get( 0 ); String revisionEndFieldName = auditEntitiesConfiguration.getRevisionEndFieldName(); ( (Map) previousData ).put( revisionEndFieldName, revision ); if ( auditEntitiesConfiguration.isRevisionEndTimestampEnabled() ) { // Determine the value of the revision property annotated with @RevisionTimestamp String revEndTimestampFieldName = auditEntitiesConfiguration.getRevisionEndTimestampFieldName(); Object revEndTimestampObj = this.revisionTimestampGetter.get( revision ); Date revisionEndTimestamp = convertRevEndTimestampToDate( revEndTimestampObj ); // Setting the end revision timestamp ( (Map) previousData ).put( revEndTimestampFieldName, revisionEndTimestamp ); } // Saving the previous version session.save( auditedEntityName, previousData ); sessionCacheCleaner.scheduleAuditDataRemoval( session, previousData ); } else { throw new RuntimeException( "Cannot find previous revision for entity " + auditedEntityName + " and id " + id ); } } private Date convertRevEndTimestampToDate(Object revEndTimestampObj) { // convert to a java.util.Date if ( revEndTimestampObj instanceof Date ) { return (Date) revEndTimestampObj; } return new Date( (Long) revEndTimestampObj ); } private Queryable getQueryable(String entityName, SessionImplementor sessionImplementor) { return (Queryable) sessionImplementor.getFactory().getMetamodel().entityPersister( entityName ); } private void addEndRevisionNullRestriction(AuditEntitiesConfiguration auditEntitiesConfiguration, Parameters rootParameters) { rootParameters.addWhere( auditEntitiesConfiguration.getRevisionEndFieldName(), true, "is", "null", false ); } private void addNonIdentifierWhereConditions(QueryBuilder qb, Map data, String originalIdPropertyName) { final Parameters parameters = qb.getRootParameters(); for ( Map.Entry entry : data.entrySet() ) { if ( !originalIdPropertyName.equals( entry.getKey() ) ) { if ( entry.getValue() != null ) { parameters.addWhereWithParam( entry.getKey(), true, "=", entry.getValue() ); } else { parameters.addNullRestriction( entry.getKey(), true ); } } } } private boolean isNonIdentifierWhereConditionsRequired(String entityName, String propertyName, SessionImplementor session) { final Type propertyType = session.getSessionFactory().getMetamodel().entityPersister( entityName ).getPropertyType( propertyName ); if ( propertyType.isCollectionType() ) { final CollectionType collectionType = (CollectionType) propertyType; final Type collectionElementType = collectionType.getElementType( session.getSessionFactory() ); if ( collectionElementType instanceof ComponentType ) { // required for Embeddables return true; } else if ( collectionElementType instanceof MaterializedClobType || collectionElementType instanceof MaterializedNClobType ) { // for Map<> using @Lob annotations return collectionType instanceof MapType; } } return false; } }