Hibernate Envers:如何获取所有当前实体及其创建时间戳记

赫斯特

我使用Hibernate Envers审核名为Client的实体上的更改。现在,我想从持久层读取所有当前(未删除)的Client实体,包括它们的创建时间戳。Hibernate Envers将所有必要的信息存储在审核表中。对我来说,不清楚如何执行所需的查询。我尝试了以下内容:

//Get last Revision Number for Entity Client
private Number getLastRev() {
    AuditReader auditReader = AuditReaderFactory.get(getEm());
    AuditQuery query = auditReader.createQuery()
            .forRevisionsOfEntity(Client.class, true, true)
            .addProjection(AuditEntity.revisionNumber().max());
    Number n = (Number) query.getSingleResult();
    return n;
}
public List<Object[]> getAllClientsWithCreationTimestamp() {
    Number revision = getLastRev();
    AuditReader auditReader = AuditReaderFactory.get(getEm());
    AuditQuery query = auditReader.createQuery()
            .forEntitiesAtRevision(Client.class, revision)
            .addProjection(AuditEntity.property("id"))
            .addProjection(AuditEntity.property("firstName"))
            .addProjection(AuditEntity.property("lastName"))
            .addProjection(AuditEntity.property("email"))                
            .addProjection(AuditEntity.revisionProperty("timestamp"))
            .addOrder(AuditEntity.revisionProperty("timestamp").desc());
}

不幸的是,getAllClientsWithCreationTimestamp()列出了所有当前实体,其时间戳=“上次修改”而非“创建”时间戳。

希望任何人都可以帮助我创建正确的查询。

我实体的Java代码如下所示:

@Entity
@Table(name = "CLIENT")
@Audited
public class Client implements Serializable {

private static final long serialVersionUID = 1L;

@Id
@Column(name = "id", nullable = false)
@GeneratedValue(strategy = GenerationType.SEQUENCE, generator="CLIENT_SEQ")
@SequenceGenerator(name="CLIENT_SEQ", sequenceName="CLIENT_SEQ", allocationSize=1)
private Long Id;

@Column(name = "FIRST_NAME", length = 100)
private String firstName;

@Column(name = "LAST_NAME", nullable = false, length = 100)
private String lastName;

@Column(name = "EMAIL", nullable = false)
private String email;

public Long getId() {
    return Id;
}

public void setId(Long id) {
    Id = id;
}

public String getFirstName() {
    return firstName;
}

public void setFirstName(String firstName) {
    this.firstName = firstName;
}

public String getLastName() {
    return lastName;
}

public void setLastName(String lastName) {
    this.lastName = lastName;
}

public String getEmail() {
    return email;
}

public void setEmail(String email) {
    this.email = email;
}
}

我有以下数据库表:

  • 修订:启用主修订实体表
  • 客户:用于存储客户端实体的表
  • CLIENT_AUD:用于存储客户端修订的表

数据库表

优点

您的代码很接近,但是我相信您有一些错误。

为了获得创建时间戳,您将需要针对该用例使用其他查询,并将其与其他查询结合使用。以下查询将为您返回一个包含实体ID以及创建审计行时的时间戳的投影。您只需要考虑这一点并创建一个ID /时间戳对映射。

AuditQuery query = auditReader.createQuery()
  .forRevisionsOfEntity( Client.class, true, false )
  .addProjection( AuditEntity.id() )
  .addProjection( AuditEntity.revisionProperty( "timestamp" ) )
  .add( AuditEntity.revisionType().eq( RevisionType.ADD ) );

接下来,您将要获取每个实体ID的最新修订版号,然后将结果再次转换为ID /修订版号对的映射。

AuditQuery query = auditReader.createQuery()
  .forRevisionsOfEntity( Client.class, true, false )
  .addProjection( AuditEntity.id() )
  .addProjection( AuditEntity.revisionNumber().max() );

现在,只需要让Envers根据最大修订号返回您的数据投影查询,然后将其与上面的ID /时间戳映射结合即可。

for ( Map.Entry<YourEntityIdType,Number> entry : idRevisionPairs.entrySet() ) {
  AuditQuery query = auditReader.createQuery()
    .forEntitiesAtRevision( Client.class, entry.getValue() )
    .add( AuditEntity.id().eq( entry.getKey() )
    .addProjection( ... );

  // here take the results from query and the id-timestamp pairs and
  // marry them into some DTO you return.
}

本文收集自互联网,转载请注明来源。

如有侵权,请联系 [email protected] 删除。

编辑于
0

我来说两句

0 条评论
登录 后参与评论

相关文章