0

我正在尝试按照他们的文档为 OVal 实施激活规则,但似乎在找到我用于比较的变量时遇到了问题。不幸的是,除了他们文档中的一小部分之外,关于该主题的在线内容并不多。

我试图解决的问题的另一部分是使用注解也让这个工作用于构造函数验证。如果没有我对这个问题@Guarded的回答中描述的约束规则,这可以正常工作,但当我在JavaScriptGroovy中添加激活规则时就不行。

3.4. 声明约束的激活规则

public class BusinessObject
{
    private String fieldA;

    @NotNull(when = "groovy:_this.fieldA != null")
    private String fieldB;
}

我已经尝试过 JS 和 groovy 并尝试过使用和不使用_this. 删除它会导致:ReferenceError: "someString" is not defined所以我认为他们在文档中列出的方式是正确的,但我遗漏了一些东西。

字段验证代码:

public class BusinessObject {
    private String fieldA;

    //@NotNull(when = "groovy:_this.fieldA != null") //works for public & private
    @NotNull(when = "javascript:_this.fieldA != null") //only works when fieldA is public
    private String fieldB;

    public BusinessObject(){}

    public BusinessObject(String fieldA, String fieldB) {
        this.fieldA = fieldA;
        this.fieldB = fieldB;
    }
}

构造函数验证代码:

@Guarded
public class BusinessObjectConstructorValidation {
    private String fieldA;
    private String fieldB;

    public BusinessObjectConstructorValidation(
            String fieldA,
            @NotNull(when = "groovy:_this.fieldA != null") String fieldB) {

        this.fieldA = fieldA;
        this.fieldB = fieldB;
    }
}

我如何测试对象:

public class BusinessObjectTest {

    @Test
    public void fieldANullFieldBNotValidatedNoViolations() {
        BusinessObject businessObject = new BusinessObject(null, null);
        Validator validator = new Validator();
        validator.validate(businessObject);
    }

    //This test will fail if the fields are private and using javascript
    //If it's public or using groovy it passes
    @Test
    public void fieldANotNullFieldBValidatedViolationsSizeIsOne() {
        BusinessObject businessObject = new BusinessObject("A", null);
        Validator validator = new Validator();
        List<ConstraintViolation> errors = validator.validate(businessObject);
        System.out.println(errors.size());
        assertThat(errors.size(), is(1));
    }

    @Test
    public void fieldANullFieldBNotNullNoViolations() {
        BusinessObject businessObject = new BusinessObject(null, "B");
        Validator validator = new Validator();
        validator.validate(businessObject);
    }
}

我不确定为什么 JavaScript 版本的行为与 groovy 版本不同,我尝试更改我能想到的所有组合,包括:_this.fieldA、、、和__this.fieldAwindow.fieldAfieldA__fieldA

更新 JavaScript 似乎适用于私有字段,只要它有一个公共 getter。

4

1 回答 1

0

我通过执行以下操作并按照开发人员的建议切换到 groovy 而不是 JavaScript 解决了这个问题

JavaScript Engine Rhino 不能直接访问私有字段。例如,“javascript:_this.myPrivateField != null && _this.myPrivateField.length > 10”之类的内容将始终返回 false,无论私有字段的值如何。如果 Rhino 无法访问私有字段,它显然不会引发异常。

我建议您使用 groovy 而不是 javascript 来激活约束。对于大多数脚本运行时(包括 JavaScript),语句 _this.someVariable 将导致调用 _this.getSomeVariable() getter,而不是直接访问同名的私有字段。

在我的代码示例的第一行中,(checkInvariants = false)如果您要验证的对象具有任何公共 getter 方法,则它是必需的,否则将导致StackOverflowError

不幸的是,正如我在我的问题中发布的那样,在构造函数中添加验证不起作用。所以为了解决这个问题,我需要将验证添加到字段并将@PostValidateThis注释添加到构造函数。

调用构造函数后使用验证的示例 Pojo。

@Guarded(checkInvariants = false)// removing this results in StackOverflowError
public class User {
    private final String firstName;
    private final String lastName;
    @NotNull(when = "groovy:_this.lastName != null")
    private final Integer age;

    @PostValidateThis
    public User(String firstName, String lastName, Integer age) {
        this.firstName = firstName;
        this.lastName = lastName;
        this.age = age;
    }

    public String getFirstName() {
        return firstName;
    }

    public String getLastName() {
        return lastName;
    }

    public Integer getAge() {
        return age;
    }
}

上述 pojo 的基本单元测试。

public class UserTest {
    @Test
    public void userValidParamsNoException() throws Exception {
        User user = new User("foo","bar",123);
        assertThat(user, is(not(nullValue())));
        assertThat(user.getFirstName(), is("foo"));
        assertThat(user.getLastName(), is("bar"));
        assertThat(user.getAge(), is(123));
    }

    @Test
    public void userLastNameNullNoException() throws Exception {
        User user = new User("foo",null, null);
        assertThat(user, is(not(nullValue())));
        assertThat(user.getFirstName(), is("foo"));
    }

    @Test(expected = ConstraintsViolatedException.class)
    public void userLastNameNotNullAgeNullThrowsException() throws Exception {
        User user = new User("foo","bar", null);
    }
}
于 2017-05-21T00:01:30.910 回答