在测试中找不到保存的实体
can't find saved entity in test
我正在测试我的 spring 应用程序,但存在有线问题。我不知道为什么会这样。
在我的 spring 代码中,我使用 crudRepository 将 requestBody 保存到数据库中。我可以用 crudRepository 找到保存的项目。
但是当我 运行 单元测试并使用 TestRestTemplate 调用相同的方法时,我找不到使用 crudRepository 保存的项目。
服务器
@PostMapping("")
public Question saveQuestion(@RequestBody Question question) {
questionRepository.save(question);
for(Question _p:questionRepository.findAll()) {
Application.LOG.info("[saveQuestion] "+_p.toString());
}
// I can find saved Question
return question;
}
单元测试
@RunWith(SpringRunner.class)
@SpringBootTest(webEnvironment = SpringBootTest.WebEnvironment.RANDOM_PORT)
@Transactional
public class QuestionControllerTest {
@Autowired TestRestTemplate restTemplate;
@Autowired QuestionRepository questionRepository;
@Test
public void addNewQuestionTest() throws InterruptedException {
String json= "{\"uniqueId\":1}";
HttpEntity<String> req = new HttpEntity<>(json, headers);
ResponseEntity<Question> response = restTemplate.postForEntity("/question", req, Question.class);
for(Question _p:questionRepository.findAll()) {
Application.LOG.info("[addNewQuestion]"+_p.toString());
}
// I can't find saved question.
}
}
日志看起来像这样。
[saveQuestion] Question(0)
[saveQuestion] Question(1)
[addNewQuestionTest] Question(0)
但我除外
[saveQuestion] Question(0)
[saveQuestion] Question(1)
[addNewQuestionTest] Question(0)
[addNewQuestionTest] Question(1)
我检查了 db 有数据并且 saveQuestion 方法打印了 2 个项目。
但是 addNewQuestionTest 方法中只有一项。
我是不是漏掉了什么?
这只是一个假设,但可能会发生如下情况:
1)您在某个交易下将整个测试标记为运行:
@Transactional
public class QuestionControllerTest {
2) 当你点击 post 方法时,它会保存实体,但在它自己的/新事务中
ResponseEntity<Question> response = restTemplate.postForEntity("/question", req, Question.class);
来自 1) 的交易在这里仍然有效,因此您有 2 笔交易。
3) 由于保存已在单独的事务中执行,而第一个事务仍处于活动状态,因此第一个事务尚未访问事务 2) 保存的数据.
解决方案
由于 Spring JPA 存储库本质上是事务性的,我将从测试配置中删除 @Transactional
。
现在,当您点击 questionRepository.findAll()
时,它会触发一个新事务,该事务可以访问第一个保留问题的事务保存的数据。
我正在测试我的 spring 应用程序,但存在有线问题。我不知道为什么会这样。
在我的 spring 代码中,我使用 crudRepository 将 requestBody 保存到数据库中。我可以用 crudRepository 找到保存的项目。 但是当我 运行 单元测试并使用 TestRestTemplate 调用相同的方法时,我找不到使用 crudRepository 保存的项目。
服务器
@PostMapping("")
public Question saveQuestion(@RequestBody Question question) {
questionRepository.save(question);
for(Question _p:questionRepository.findAll()) {
Application.LOG.info("[saveQuestion] "+_p.toString());
}
// I can find saved Question
return question;
}
单元测试
@RunWith(SpringRunner.class)
@SpringBootTest(webEnvironment = SpringBootTest.WebEnvironment.RANDOM_PORT)
@Transactional
public class QuestionControllerTest {
@Autowired TestRestTemplate restTemplate;
@Autowired QuestionRepository questionRepository;
@Test
public void addNewQuestionTest() throws InterruptedException {
String json= "{\"uniqueId\":1}";
HttpEntity<String> req = new HttpEntity<>(json, headers);
ResponseEntity<Question> response = restTemplate.postForEntity("/question", req, Question.class);
for(Question _p:questionRepository.findAll()) {
Application.LOG.info("[addNewQuestion]"+_p.toString());
}
// I can't find saved question.
}
}
日志看起来像这样。
[saveQuestion] Question(0)
[saveQuestion] Question(1)
[addNewQuestionTest] Question(0)
但我除外
[saveQuestion] Question(0)
[saveQuestion] Question(1)
[addNewQuestionTest] Question(0)
[addNewQuestionTest] Question(1)
我检查了 db 有数据并且 saveQuestion 方法打印了 2 个项目。 但是 addNewQuestionTest 方法中只有一项。
我是不是漏掉了什么?
这只是一个假设,但可能会发生如下情况:
1)您在某个交易下将整个测试标记为运行:
@Transactional
public class QuestionControllerTest {
2) 当你点击 post 方法时,它会保存实体,但在它自己的/新事务中
ResponseEntity<Question> response = restTemplate.postForEntity("/question", req, Question.class);
来自 1) 的交易在这里仍然有效,因此您有 2 笔交易。
3) 由于保存已在单独的事务中执行,而第一个事务仍处于活动状态,因此第一个事务尚未访问事务 2) 保存的数据.
解决方案
由于 Spring JPA 存储库本质上是事务性的,我将从测试配置中删除 @Transactional
。
现在,当您点击 questionRepository.findAll()
时,它会触发一个新事务,该事务可以访问第一个保留问题的事务保存的数据。