2

我正在评估 javers 以将其用于审计实体。我有一个带有 ValueObjects 嵌套集合的实体。我希望 valueobject 上的每个属性更改都会生成 Entity 的快照。仅当将 valueobject 添加到集合中时才会创建快照。在我的情况下,我向集合中添加了两个 valueobjects创建了实体的两个快照。第三次我只是更改了值对象的属性,而 javers 并没有将其识别为实体的更改,而是为内部值对象创建了快照。

我的问题是我的假设是否有效,或者跟踪集合中值对象的更改的最佳方法是什么

下面是我使用 spring boot 创建的一个简单测试的代码。

我正在使用 javers 3.2.0 版

我的实体如下

package com.example.javersdemo;

import com.fasterxml.jackson.annotation.JsonUnwrapped;
import lombok.Data;
import org.javers.core.metamodel.annotation.TypeName;

import javax.persistence.CollectionTable;
import javax.persistence.ElementCollection;
import javax.persistence.Entity;
import javax.persistence.Id;
import javax.persistence.JoinColumn;
import java.util.List;

@Data
@Entity
@TypeName("User")
class User{
    @Id
    private String name ;


    @JsonUnwrapped
    @ElementCollection(targetClass = Hobby.class)
    @CollectionTable(name = "USER_HOBBIES")
    @JoinColumn(name = "NAME")
    private List<Hobby> hobbies;

    private User(){

    }

    public User(String name, List<Hobby> hobbies) {
        this.name = name;
        this.hobbies = hobbies;
    }
}

值对象如下

package com.example.javersdemo;

import lombok.Data;

import javax.persistence.Embeddable;

@Data
@Embeddable
public class Hobby {

    private String hobby;

    private boolean active;

    private Hobby() {

    }

    public Hobby(String hobby, boolean active) {

        this.hobby = hobby;
        this.active = active;
    }
}

我的弹簧数据存储库如下

package com.example.javersdemo;

import org.javers.spring.annotation.JaversSpringDataAuditable;
import org.springframework.data.repository.CrudRepository;


@JaversSpringDataAuditable
interface TestUserRepository extends CrudRepository<User,String> {

}

下面是我创建的一个 spock 集成测试,用于验证对集合内对象属性的更改会创建一个新快照。

package com.example.javersdemo

import org.javers.core.Javers
import org.javers.repository.jql.QueryBuilder
import org.springframework.beans.factory.annotation.Autowired
import org.springframework.boot.test.context.SpringBootTest
import spock.lang.Specification

@SpringBootTest(webEnvironment = SpringBootTest.WebEnvironment.RANDOM_PORT)
class JaversInnerValueObjectsTest extends Specification {

    @Autowired
    TestUserRepository userRepository

    @Autowired
    Javers javers


    def 'should create 6 snapshots'() {

        given:
        def hobbies = [new Hobby('Reading books', true)]
        def user = new User('John', hobbies)

        when:
        userRepository.save(user)

        hobbies = [new Hobby('Reading books', true), new Hobby('Watching Soccer', true)]

        user.hobbies = hobbies

        userRepository.save(user)

        hobbies = [new Hobby('Reading books', true), new Hobby('Watching Soccer', false)]

        user.hobbies = hobbies

        userRepository.save(user)



        then:
        QueryBuilder jqlQuery = QueryBuilder.byInstanceId('John', User)
        def snapshots = javers.findSnapshots(jqlQuery.withChildValueObjects().build())
        snapshots.size() == 6


    }

}

并且测试失败并出现以下错误

不满足条件:

快照.size() == 6 | | | | 5 false [Snapshot{commit:3.0, id:User/John#hobbies/1, version:2, (hobby:Watching Soccer)}, Snapshot{commit:2.0, id:User/John#hobbies/1, version:1 , (active:true, hobby:Watching Soccer)}, Snapshot{commit:2.0, id:User/John, version:2, (hobbies:[User/John#hobbies/0, User/John#hobbies/1], name:John)}, Snapshot{commit:1.0, id:User/John#hobbies/0, version:1, (active:true, hobby:Reading books)}, Snapshot{commit:1.0, id:User/John,版本:1,(爱好:[用户/John#hobbies/0],姓名:John)}]

预期:6

实际:5

4

1 回答 1

0

在日志中有 JaVers 提交统计:

13:27:44.315 [main] INFO  org.javers.core.Javers - Commit(id:1.0, snapshots:2, author:author, changes - NewObject:2), done in 71 millis (diff:71, persist:0)
13:27:44.333 [main] INFO  org.javers.core.Javers - Commit(id:2.0, snapshots:2, author:author, changes - ListChange:1 NewObject:1), done in 17 millis (diff:17, persist:0)
13:27:44.336 [main] INFO  org.javers.core.Javers - Commit(id:3.0, snapshots:1, author:author, changes - ValueChange:1), done in 3 millis (diff:3, persist:0)

在第一次提交中,创建了 2 个快照,因为我们在这里有 2 个新对象。

在第二次提交中,创建了另外 2 个快照:

  • 用户(实体)快照,因为用户的字段hobbies已更改 - 新项目添加到列表中,因此 ListChange
  • 爱好(值对象)快照,因为它是一个新对象

在第三次提交中只创建了 1 个快照,因为用户的字段hobbies没有改变。JaVers 将值对象的集合和实体的集合视为引用的集合(GlobalId)。在这种情况下,引用不会更改。集合的状态捕获如下:

["User/John#hobbies/0",
 "User/John#hobbies/1"]

第三次提交的唯一变化是爱好的字段更改(true->false),因此 ValueObject 中的 ValueChange 与 GlobalIdUser/John#hobbies/1

换句话说,当您更改某个实体拥有的值对象时,实体状态没有变化,实体快照将不会创建,因为它与之前的快照完全相同。

于 2017-06-12T11:50:31.210 回答