![](/img/trans.png)
[英]How to update embedded mongo document using spring mongo data api
[英]Use Spring Data random (embedded) Mongo port with NoSQL JUnit @Rule
我目前正在尝试编写一个使用 Spring Data Mongo 存储库的集成测试类。 我使用de.flapdoodle.embed.mongo
依赖项提供的嵌入式 Mongo 实例。 Spring Data 文档指定我们只需将这个依赖项放在项目中,其余的由EmbedMongoAutoConfiguration
处理。
现在,没关系,将端口设置为0
会使自动配置过程找到一个空闲端口来启动 mongo 实例。
此功能对我来说是必要的,以避免与其他测试(这些测试与我公司的其他项目一起在 Jenkins CI 服务器上运行)发生冲突。
现在问题来了,我希望能够在我的每个测试方法运行之前从某个外部文件注入一些测试数据。 我发现 NoSQL Unit 可以通过一个简单的方法注释和一个 JUnit @Rule
来做到这@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 单元中使用过嵌入式 Mongo 功能,有没有办法,例如在 Spring 上下文初始化后创建@Rule
?
我想知道自己(以静态方式)找到空闲端口,将其设置为@Rule
然后告诉EmbeddedMongoAutoConfiguration
通过覆盖IMongodConfig
bean 使用它IMongodConfig
是个好主意? 还是有“更简单”的方法?
注意:我刚刚看到 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);
}
}
细化@user6599111给出的方案,可以得到Flapdoodle Embedded Mongo随机选择的port
,只需注入一个IMongodConfig
类型的对象IMongodConfig
。
春天开机自动地建立此对象为你,为说明这里。
那么,配置类就会变成下面这样。
@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);
}
}
我有同样的问题,这是我的解决方案
@Configuration public class EmbeddedMongoConfig extends AbstractMongoConfiguration { @Autowired private Environment environment; @Autowired private MongoProperties properties; @Autowired(required = false) private MongoClientOptions options; @Override protected String getDatabaseName() { return properties.getDatabase(); } @Override @Bean(destroyMethod = "close") public Mongo mongo() throws Exception { properties.setPort(Network.getFreeServerPort()); return properties.createMongoClient(this.options, this.environment); } }
@RunWith(SpringJUnit4ClassRunner.class) @SpringApplicationConfiguration(classes = { AppRunner.class, EmbeddedMongoConfig.class }) public class BaseTest { }
public class CategoryServiceTest extends BaseTest{ @Autowired private CategoryService categoryService; @Test public void someTest(){ fail("Test not implemented"); } }
我试过这个:
int mongoPort = SocketUtils.findAvailableTcpPort();
来源: https : //docs.spring.io/spring/docs/4.0.5.RELEASE/javadoc-api/org/springframework/util/SocketUtils.html
这对我有用。
我们嵌入了运行单元测试的 Mongo,并且在 Jenkins 上构建了多个应用程序,其中一些失败了,因为每个人的端口都是相同的。 也尝试过手动更改端口,但由于有许多应用程序并且其中一些使用了公共基类,因此失败了。
我不确定@Rule 部分,但你可以试试这个。
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.