简体   繁体   中英

Morphia can't map List of List of Integer

I have a simplified example of a problem I am working on. I am trying to load a document that contains an array of an array of numbers.

{
    "_id" : ObjectId("570cf0167640ed9f8bcff8e7"),
    "matrix" : [ 
        [ 
            42
        ]
    ]
}

In order to create all indexes I call org.mongodb.morphia.Datastore#ensureIndexes() . As I understand, from what is documented, I need to call org.mongodb.morphia.Morphia#map(Class ...) to tell Morphia on which classes to look for @Indexes annotation. Without Morphia#map(...) the app runs fine (and as expected no indexes will be created). If I add Morphia#map(...) I get an Exception.

Exception in thread "main" java.lang.RuntimeException: java.lang.IllegalArgumentException: BasicBSONList can only work with numeric keys, not: [elementData]
    at org.mongodb.morphia.mapping.EmbeddedMapper.fromDBObject(EmbeddedMapper.java:74)
    at org.mongodb.morphia.mapping.Mapper.readMappedField(Mapper.java:772)
    at org.mongodb.morphia.mapping.Mapper.fromDb(Mapper.java:230)
    at org.mongodb.morphia.mapping.Mapper.fromDBObject(Mapper.java:191)
    at org.mongodb.morphia.query.MorphiaIterator.convertItem(MorphiaIterator.java:134)
    at org.mongodb.morphia.query.MorphiaIterator.processItem(MorphiaIterator.java:146)
    at org.mongodb.morphia.query.MorphiaIterator.next(MorphiaIterator.java:117)
    at org.mongodb.morphia.query.QueryImpl.asList(QueryImpl.java:150)
    at it.test.Main.fails(Main.java:41)
    at it.test.Main.main(Main.java:24)
Caused by: java.lang.IllegalArgumentException: BasicBSONList can only work with numeric keys, not: [elementData]
    at org.bson.types.BasicBSONList._getInt(BasicBSONList.java:168)
    at org.bson.types.BasicBSONList._getInt(BasicBSONList.java:160)
    at org.bson.types.BasicBSONList.get(BasicBSONList.java:105)
    at org.mongodb.morphia.mapping.MappedField.getDbObjectValue(MappedField.java:190)
    at org.mongodb.morphia.converters.Converters.fromDBObject(Converters.java:121)
    at org.mongodb.morphia.mapping.ValueMapper.fromDBObject(ValueMapper.java:20)
    at org.mongodb.morphia.mapping.Mapper.readMappedField(Mapper.java:766)
    at org.mongodb.morphia.mapping.Mapper.fromDb(Mapper.java:230)
    at org.mongodb.morphia.mapping.EmbeddedMapper.readMapOrCollectionOrEntity(EmbeddedMapper.java:206)
    at org.mongodb.morphia.mapping.EmbeddedMapper.readCollection(EmbeddedMapper.java:142)
    at org.mongodb.morphia.mapping.EmbeddedMapper.fromDBObject(EmbeddedMapper.java:45)
    ... 9 more

Can someone explain why explicitly calling map() breaks Morphia?

The following exmaple reproduces the problem (just add org.mongodb.morphia:morphia:1.1.1 as dependency).

package it.test;

import java.util.List;

import org.bson.types.ObjectId;
import org.mongodb.morphia.Datastore;
import org.mongodb.morphia.Morphia;
import org.mongodb.morphia.dao.BasicDAO;
import org.mongodb.morphia.query.QueryResults;

import com.mongodb.MongoClient;
import com.mongodb.MongoClientURI;

import it.test.model.Doc;

public class Main {
    private static final String URI = "mongodb://localhost:27017";
    private static final String NAME = "test";
    private static final MongoClientURI CLIENT_URI = new MongoClientURI(URI + "/" + NAME);

    public static void main(String[] args) {
        Main main = new Main();

        main.works();
        main.fails();
    }

    private void fails() {
        try (MongoClient client = new MongoClient(CLIENT_URI)) {
            Morphia morphia = new Morphia();
            morphia.getMapper().getOptions().setStoreEmpties(true);
            morphia.mapPackage("it.test.model");

            find(morphia, client, CLIENT_URI.getDatabase());
        }
    }

    private void works() {
        try (MongoClient client = new MongoClient(CLIENT_URI)) {
            Morphia morphia = new Morphia();
            morphia.getMapper().getOptions().setStoreEmpties(true);
            // morphia.mapPackage("it.test.model"); // bad call?

            find(morphia, client, CLIENT_URI.getDatabase());
        }
    }

    private void find(Morphia morphia, MongoClient client, String dbName) {
        Datastore datastore = morphia.createDatastore(client, dbName);

        BasicDAO<Doc, ObjectId> dao = new BasicDAO<>(Doc.class, datastore);
        QueryResults<Doc> result = dao.find();
        List<Doc> rootEntities = result.asList();

        System.out.println("Found " + rootEntities.size() + " RootEntity documents.");
    }
}

package it.test.model;

import java.util.List;

import org.bson.types.ObjectId;
import org.mongodb.morphia.annotations.Entity;
import org.mongodb.morphia.annotations.Id;

@Entity
public class Doc {

    @Id
    private ObjectId id;

    public ObjectId getId() {
        return id;
    }

    public void setId(ObjectId id) {
        this.id = id;
    }

    private List<List<Integer>> matrix;

    public List<List<Integer>> getMatrix() {
        return matrix;
    }

    public void setMatrix(List<List<Integer>> matrix) {
        this.matrix = matrix;
    }

}

If you look at this test you can see List<List<Integer>> works just fine. Looking at your example, I don't see anything obvious that would lead to the errors you're getting but I can verify at least that List<List<Integer>> works. What version of the Java driver are you using? It should be in the 3.x line at the least.

The technical post webpages of this site follow the CC BY-SA 4.0 protocol. If you need to reprint, please indicate the site URL or the original address.Any question please contact:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM