3

我目前正在尝试编写一个使用 Spring Data Mongo 存储库的集成测试类。我使用de.flapdoodle.embed.mongo依赖项提供的嵌入式 Mongo 实例。Spring Data 文档指定我们只需将此依赖项放入项目EmbedMongoAutoConfiguration中,其余部分由其负责。

现在,没关系,将端口设置为0使自动配置过程找到一个空闲端口来启动 mongo 实例。

这个特性对我来说是必要的,以避免与其他测试发生冲突(这些测试与我公司的其他项目一起在 Jenkins CI 服务器上运行)。

现在问题来了,我希望能够在我的每个测试方法运行之前从一些外部文件中注入一些测试数据。我发现 NoSQL Unit 可以通过一个简单的方法注释和一个 JUnit 来做到这一点@Rule

这是一个例子:

@Value("${local.mongo.port}")
private int mongoPort; // <- still 0 a the time the Rule below is created.

@Rule
public MongoDbRule managedMongoDb = new MongoDbRule(MongoDbConfigurationBuilder.mongoDb().databaseName("myAwesomeDb").port(mongoPort).build());

@Test
@UsingDataSet(locations = "testdata.json", loadStrategy = LoadStrategyEnum.CLEAN_INSERT)
public void testMyData() {
   // ...
}

我的问题是,@Rule需要 Mongo 端口在其构建器中实例化底层的 MongoClient,但是在实例化 @Rule 时,Spring 上下文尚未完全初始化并且EmbeddedMongoAutoConfiguration尚未发布端口。

所以我的问题是,有没有人曾经在 NoSQL 单元中使用过 Embedded Mongo 功能,有没有办法,例如在 Spring 上下文初始化@Rule 后创建?

我想知道是否自己找到空闲端口(以静态方式),将其设置为@Rule然后告诉EmbeddedMongoAutoConfiguration通过覆盖IMongodConfigbean 来使用它是一个好主意吗?还是有“更简单”的方式?

注意:我刚刚看到 fladdoodle 库提供了一个类和一个静态方法来查找一个空闲的服务器端口,并且它被 Spring 使用,如下所示:

Network.getFreeServerPort(getHost()), Network.localhostIsIPv6()))

提前谢谢大家!

编辑:我尝试了我刚才谈到的解决方案,它似乎有效,但我仍然认为它有点“冗长”和肮脏。

private static final Logger log = LoggerFactory.getLogger(MyAwesomeIT.class);
private static int mongoPort;
static {
    try {
        mongoPort = Network.getFreeServerPort();
    } catch (IOException e) {
        log.error("Error while trying to find a free port for Mongo", e);
        mongoPort = -1; // test should then not work
    }
}

@Rule
public MongoDbRule managedMongoDb = new MongoDbRule(MongoDbConfigurationBuilder.mongoDb().databaseName("myAwesomeDb").port(mongoPort).build());

然后在关联的配置类中:

@Configuration
@EnableAutoConfiguration
@EnableMongoRepositories
@EnableConfigurationProperties(MongoProperties.class)
static class ContextConfiguration {
    @Autowired
    private MongoProperties mongoProperties;

    @PostConstruct
    public void init() {
        // Here, I override the port property
        mongoProperties.setPort(mongoPort);
    }
}
4

3 回答 3

6

细化@user6599111 给出的解决方案,可以得到portFlapdoodle Embedded Mongo 随机选择的,只需注入一个 type 的对象IMongodConfig

Spring Boot 会自动为您构建此对象,如此处所述。

那么,配置类就会变成下面这样。

@Configuration
@EnableAutoConfiguration(exclude = { EmbeddedMongoAutoConfiguration.class })
public class MongoConfiguration {
    @Autowired
    private Environment environment;

    @Autowired
    private MongoProperties properties;

    @Autowired(required = false)
    private MongoClientOptions options;

    @Autowired
    private IMongodConfig mongoConfig;

    @Bean
    public MongoClient mongo() throws Exception {
        properties.setPort(mongoConfig.net().getPort());
        return properties.createMongoClient(this.options, this.environment);
    }
}
于 2016-11-29T15:23:30.880 回答
3

我有同样的问题,这是我的解决方案

@配置
公共类 EmbeddedMongoConfig 扩展 AbstractMongoConfiguration {

    @自动连线
    私有环境环境;

    @自动连线
    私有 MongoProperties 属性;

    @Autowired(必需 = 假)
    私有 MongoClientOptions 选项;

    @覆盖
    受保护的字符串 getDatabaseName() {
        返回属性.getDatabase();
    }

    @覆盖
    @Bean(destroyMethod = "关闭")
    公共 Mongo mongo() 抛出异常 {
        properties.setPort(Network.getFreeServerPort());
        返回 properties.createMongoClient(this.options, this.environment);
    }

}

@RunWith(SpringJUnit4ClassRunner.class)
@SpringApplicationConfiguration(classes = { AppRunner.class, EmbeddedMongoConfig.class })
公共类 BaseTest {

}

公共类 CategoryServiceTest 扩展 BaseTest{

    @自动连线
    私有 CategoryService 类别服务;

    @测试
    公共无效 someTest(){
        fail("测试未实现");
    }

}

于 2016-07-17T04:34:34.107 回答
0

我试过这个:

int mongoPort = SocketUtils.findAvailableTcpPort();

来源: https ://docs.spring.io/spring/docs/4.0.5.RELEASE/javadoc-api/org/springframework/util/SocketUtils.html

这对我有用。

我们嵌入了运行单元测试的 Mongo,并且在 Jenkins 上构建了多个应用程序,其中一些应用程序失败了,因为每个人的端口都是相同的。也尝试过手动更改端口,但由于有许多应用程序,其中一些使用通用基类,因此失败了。

我不确定@Rule 部分,但你可以试试这个。

于 2020-05-27T09:30:11.463 回答