如果我必须测试使用可变实体的服务,我将构建我需要的最小对象(真实对象)并将其传递给我的服务。例子:
User joe = new User();
joe.setEmail("joe@example.com");
resetPasswordService.resetPassword(joe);
verif(emailServiceMock).sendEmail("joe@example.com", "Your password has been reset!");
显然用户有很多字段,但我没有设置它们,因为resetPasswordService
不需要它们。这对重构非常友好,因为如果我重命名不是电子邮件的用户字段,则不会更改此测试。
当我尝试对Immutables对象执行相同操作时,就会出现问题。我将坚持使用相同的示例,并将 User 从实体转变为不可变的。
@Value.Immutable
public abstract class User {
public abstract String getEmail();
public abstract PostalAddress getPostalAddress();
//more fields
}
User joe = new ImmutableUserBuilder().email("joe@example.com").build();
resetPasswordService.resetPassword(joe);
verif(emailServiceMock).sendEmail("joe@example.com", "Your password has been reset!");
java.lang.IllegalStateException:无法构建用户,一些必需的属性没有设置 [postalAddress, signupDate, city, ....]
当构建器尝试构建对象时,它会失败。所以我该怎么做?
- 为用户使用模拟并让它返回模拟,即使每次模拟返回模拟时,仙女都会死去
- 创建一个测试 DSL 并拥有某种工厂来构建包含所有我不需要的字段的整个用户树结构?看起来很重,而且对重构不太友好。这使得测试的要求不那么透明。
- 使用户中
@Nullable
的所有字段并让构建器不验证对象?这会使我面临在生产中拥有不完整对象的风险,对吧? - 我错过了其他选择?
我知道用户应该是实体而不是不可变的值对象。我在这个例子中使用了 User,因为它很容易理解。