【问题标题】:How to test Java Spring Boot application without @SpringBootApplication using JUnit?如何使用 JUnit 在没有 @SpringBootApplication 的情况下测试 Java Spring Boot 应用程序?
【发布时间】:2020-04-03 05:03:34
【问题描述】:

我创建了一个 Spring Boot Java 库,其中包含一组执行某些功能的实用程序。然后可以在需要其实用程序的多个应用程序中使用该库。该库被设计为 Spring Boot 应用程序,以允许将这些实用程序注入应用程序上下文中。

现在我希望对其中一个实用程序执行 JUnit 测试,以确保它正常工作。但是,由于此应用程序基本上是实用程序的集合,而不是独立的应用程序,因此它没有主类或使用@SpringBootApplication 注释的主方法。现在,当我运行 JUnit 测试时,它出现了一个错误。

java.lang.IllegalStateException: Unable to find a @SpringBootConfiguration, you need to use @ContextConfiguration or @SpringBootTest(classes=...)

是否可以测试这个 Java 库,或者我们应该只在将使用这个库的应用程序中编写测试用例?

【问题讨论】:

  • 我不这么认为。如果它只是一个库,则只能运行单元测试。集成测试是应用程序的责任。或者您可以创建一个使用您的代码的演示应用程序,它是一个特定于您的代码的 Spring Boot 应用程序,并在那里编写集成测试。
  • 看看 Bramnik 提到的 spring.factories。看到这个答案->stackoverflow.com/questions/32107610/… 和这个博客blog.pchudzik.com/201903/spring-factories

标签: java spring spring-boot junit junit5


【解决方案1】:

我觉得你说的有些矛盾:

  1. 创建了一个库...设计为 Spring Boot 应用程序。
  2. 库可用于需要其实用程序的多个应用程序。

如果你实现“1”,那么有一个带有 spring boot maven/gradle 插件的模块被配置为创建应用程序的 JAR,它是一个库。

但是如果你有,比如说,想要使用你的库的模块 X,它不可能在这个模块中添加对你的库的依赖,spring boot JAR 工件不是 Java Sense 中的 JAR...所以这不会'不能在 IDE 和 maven 中工作(我的意思是从技术上讲你会遇到编译错误)。

然后你写了一些完全有道理的东西:你说库本身没有主类/@SpringBootApplication带注释的类。由此我得出结论,它不是一个 Spring Boot 应用程序,而是一个 Spring Boot 启动模块。

在这种情况下你不应该使用@SpringBootTest注解,因为它模仿了启动spring boot应用程序的方式(查找主类,根据包结构扫描包,加载配置等等)。你不需要这一切。好吧,也许从技术上讲,您仍然可以创建一个带有 @SpringBootApplication 注释的主类,并将其放入相关包中的 src/test/java/.../ 中,但这并没有真正意义。

所以基本上你有两个选择:

  1. 您完全可以在没有 spring 的情况下测试实用程序,就好像该实用程序只是一个 Java 类一样,使用 Mockito 模拟依赖关系就可以了。由于这些测试速度很快,因此您是最佳选择。

  2. 您可以通过使用应用程序创建的所有必需 bean 加载 spring 上下文来运行集成测试。


@ExtendWith(SpringExtension.class)
@ContextConfiguration(classes = {MyLibraryConfiguration.class})
public class SampleLibraryTest {

    @Autowired 
    private SomeBeanFromTheLibrary theBean;

    @Test
    public void testFoo() {
      ....
    }
}

现在虽然您可以使用组件扫描(在这种情况下您需要稍微不同的注解),但在示例中我假设您使用的是 java config,在那里注册库的所有 bean 并创建一个 @ 987654327@ 使用此 Java 配置文件创建自动配置(您在模块 X 中添加对库的依赖,它会自动加载库中定义的 bean)。

这个@ExtendsWith (@RunWith for junit 4) 与 Spring Boot 无关,它表现为一个“普通”的 spring,你可以自动装配 beans,创建 mock beans,缓存配置等等。

【讨论】:

  • 感谢您的详细解释。在我的示例中,我实际上使用的是组件扫描。您能否解释一下哪些注释会因此而改变?
  • 但是由于您的库无论如何都要与 Spring boot 集成,我认为 Java Configuration 将是一个更好的选择。 Spring Boot 应用程序(模块 X)可能无法对您的包启用组件扫描,根据 Spring Boot 约定,您应该编写一个真正的自动配置......
猜你喜欢
  • 2018-05-15
  • 2020-10-21
  • 2019-03-17
  • 2019-10-29
  • 1970-01-01
  • 2018-07-12
  • 2018-04-01
  • 2020-12-20
  • 2023-04-04
相关资源
最近更新 更多