使用Spring Data Embedded Mongo在Mongo数据库中导入JSON文件

时间:2016-11-02 11:28:39

标签: java mongodb spring-boot spring-data

我正在尝试编写一些与需要从MongoDB中提取数据的方法相关的集成测试。详细地说,我使用的是Spring Data项目提供的Embedded Mongo。嵌入式mongo显然由Flapdoodle提供。

我需要将一些json文件导入Embedded Mongo。我查看了 flapdoodle 提供的测试,但我无法理解它们是如何与Spring Data + Spring Boot提供的 magic 集成的。

任何人都可以发布一些澄清片段吗?

3 个答案:

答案 0 :(得分:1)

您可以查看以下测试课程,由" flapdoodle"提供。该测试显示如何导入包含集合数据集的JSON文件: MongoImportExecutableTest.java

理论上,您还可以导入数据库的整个转储。 (使用MongoDB恢复): MongoRestoreExecutableTest.java

答案 1 :(得分:1)

您可以创建一个junit规则(ExternalResource),该规则在每次测试之前和之后运行。检查MongoEmbeddedRule类以了解实现细节。

集成测试:

@RunWith(SpringRunner.class)
@SpringBootTest(webEnvironment = RANDOM_PORT)
public abstract class TestRunner {

    @Autowired
    protected MongoTemplate mongoTemplate;

    @Rule
    public MongoEmbeddedRule mongoEmbeddedRule = new MongoEmbeddedRule(this);

外部资源规则:

public class MongoEmbeddedRule extends ExternalResource {

    private final Object testClassInstance;
    private final Map<String, Path> mongoCollectionDataPaths;
    private final String fieldName;
    private final String getterName;

    public MongoEmbeddedRule(final Object testClassInstance) {
        this(testClassInstance, "mongoTemplate", "getMongoTemplate");
    }

    protected MongoEmbeddedRule(final Object testClassInstance, final String fieldName, final String getterName) {
        this.fieldName = fieldName;
        this.getterName = getterName;
        this.testClassInstance = testClassInstance;
        this.mongoCollectionDataPaths = mongoExtendedJsonFilesLookup();
    }

    @Override
    protected void before() {
        dropCollections();
        createAndPopulateCollections();
    }

    @Override
    protected void after() {
    }

    protected Set<String> getMongoCollectionNames() {
        return mongoCollectionDataPaths.keySet();
    }

    public void dropCollections() {
        getMongoCollectionNames().forEach(collectionName -> getMongoTemplate().dropCollection(collectionName));
    }

    protected void createAndPopulateCollections() {
        mongoCollectionDataPaths.forEach((key, value) -> insertDocumentsFromMongoExtendedJsonFile(value, key));
    }

    protected MongoTemplate getMongoTemplate() {
        try {
            Object value = ReflectionTestUtils.getField(testClassInstance, fieldName);
            if (value instanceof MongoTemplate) {
                return (MongoTemplate) value;
            }
            value = ReflectionTestUtils.invokeGetterMethod(testClassInstance, getterName);
            if (value instanceof MongoTemplate) {
                return (MongoTemplate) value;
            }
        } catch (final IllegalArgumentException e) {
            // throw exception with dedicated message at the end
        }
        throw new IllegalArgumentException(
                String.format(
                        "%s expects either field '%s' or method '%s' in order to access the required MongoTemmplate",
                        this.getClass().getSimpleName(), fieldName, getterName));
    }

    private Map<String, Path> mongoExtendedJsonFilesLookup() {
        Map<String, Path> collections = new HashMap<>();
        try {
            Files.walk(Paths.get("src","test","resources","mongo"))
                    .filter(Files::isRegularFile)
                    .forEach(filePath -> collections.put(
                            filePath.getFileName().toString().replace(".json", ""),
                            filePath));
        } catch (IOException e) {
            e.printStackTrace();
        }
        return collections;
    }

    private void insertDocumentsFromMongoExtendedJsonFile(Path path, String collectionName) {
        try {
            List<Document> documents = new ArrayList<>();
            Files.readAllLines(path).forEach(l -> documents.add(Document.parse(l)));
            getMongoTemplate().getCollection(collectionName).insertMany(documents);
            System.out.println(documents.size() + " documents loaded for " + collectionName + " collection.");
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
}

json文件(names.json),带有MongoDB Extended JSON,其中每个文档都在一行中,集合名称是不带扩展名的文件名。

{ "_id" : ObjectId("594d324d5b49b78da8ce2f28"), "someId" : NumberLong(1), "name" : "Some Name 1", "lastModified" : ISODate("1970-01-01T00:00:00Z")}
{ "_id" : ObjectId("594d324d5b49b78da8ce2f29"), "someId" : NumberLong(2), "name" : "Some Name 2", "lastModified" : ISODate("1970-01-01T00:00:00Z")}

答案 2 :(得分:0)

您可以创建一个抽象类,并具有启动mongod和mongoimport进程的设置逻辑。

AbstractMongoDBTest.java

eval "$(jq -r '
  .rights[] |
  [ "psql", "-d", "template1", "-U", "postgres",
    "-c", "GRANT \"\(.grant)\" ON DATABASE \"\(.database)\" TO \"\(.username)\";"
  ] | @sh' \
  <"$config_path"
)"

}

YourTestClass.java

public abstract class AbstractMongoDBTest {

private MongodProcess mongodProcess;
private MongoImportProcess mongoImportProcess;
private MongoTemplate mongoTemplate;

void setup(String dbName, String collection, String jsonFile) throws Exception {
    String ip = "localhost";
    int port = 12345;

    IMongodConfig mongodConfig = new MongodConfigBuilder().version(Version.Main.PRODUCTION)
            .net(new Net(ip, port, Network.localhostIsIPv6()))
            .build();

    MongodStarter starter = MongodStarter.getDefaultInstance();
    MongodExecutable mongodExecutable = starter.prepare(mongodConfig);

    File dataFile = new File(Thread.currentThread().getContextClassLoader().getResource(jsonFile).getFile());
    MongoImportExecutable mongoImportExecutable = mongoImportExecutable(port, dbName,
            collection, dataFile.getAbsolutePath()
            , true, true, true);

    mongodProcess = mongodExecutable.start();
    mongoImportProcess = mongoImportExecutable.start();

    mongoTemplate = new MongoTemplate(new MongoClient(ip, port), dbName);
}

private MongoImportExecutable mongoImportExecutable(int port, String dbName, String collection, String jsonFile,
                                                    Boolean jsonArray, Boolean upsert, Boolean drop) throws
        IOException {
    IMongoImportConfig mongoImportConfig = new MongoImportConfigBuilder()
            .version(Version.Main.PRODUCTION)
            .net(new Net(port, Network.localhostIsIPv6()))
            .db(dbName)
            .collection(collection)
            .upsert(upsert)
            .dropCollection(drop)
            .jsonArray(jsonArray)
            .importFile(jsonFile)
            .build();

    return MongoImportStarter.getDefaultInstance().prepare(mongoImportConfig);
}

@AfterEach
void clean() {
    mongoImportProcess.stop();
    mongodProcess.stop();
}

public MongoTemplate getMongoTemplate(){
    return mongoTemplate;
}

}