3

Google App Engine を使用して、Clientクラスのエンドポイントを作成しました。単純な GET ステートメントを使用してこれにアクセスすると、エラーが発生します。これは、単純なデータを挿入しただけでも発生します。

クラスのスケルトン コードを最後にいくつか置きます。private List<Assessment> assessmentクラス内の行にある種の派手な注釈を付ける必要があると思いますがClient、何をすべきかわかりません。

誰かがこれを修正する方法を知っていますか?可能であれば、評価フィールドにデータがないため、このエラーがセットアップに関連する理由を知りたいです.

挿入データ

{
  "firstName" : "Jon",
  "lastName" : "Doe"
}

これを挿入した後、次の単純なステートメントを実行して、すべてのfirstNameフィールドとlastNameフィールドを選択します。

GET http://localhost:8888/_ah/api/clientendpoint/v1/client?fields=items(firstName%2Cid%2ClastName)

ここでエラーが返されます

エラー

com.google.appengine.repackaged.org.codehaus.jackson.map.JsonMappingException: フィールド \"assessment\" にアクセスしようとしましたが、オブジェクトをデタッチしたときに、このフィールドはデタッチされませんでした。このフィールドにアクセスしないか、オブジェクトをデタッチするときにデタッチしてください。(参照チェーン経由: com.google.api.server.spi.response.CollectionResponse[\"items\"]->com.google.appengine.datanucleus.query.StreamingQueryResult[0]->com.my.app.client .Client[\"評価\"])

更新: Eclipse Google API プラグインを介してエンドポイントを生成しました。これを最後に配置しました。これは GET ステートメントに対応します。

もう少し追跡しましたが、返されたオブジェクトがどのように作成されるかについての理解が不足している可能性があります。

エラーは

com.my.app.client.Client.jdoGetassessment(Client.java) com.my.app.client.Client.getAssessment(Client.java:247)

しかしjdoGetAssessment()、私のコードには存在しないので、これは Google App Engine 内で生成されていると推測しています。getAssessment()によって定義されます。

public List<Assessment> getAssessment() {
     return assessment;
}

これによりエラー メッセージが表示されるようですが、データがない状態でエラー メッセージが表示される理由はまだわかりません。


クライアント

@Entity
public class Client {
    @Id
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    private Key id;
    private String firstName;
    private String lastName;

    @ElementCollection
    private List<Assessment> assessment;

    public Key getId() {
        return id;
    }

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

    public String getFirstName() {
        return firstName;
    }

    public void setFirstName(String firstName) {
        this.firstName = firstName;
    }

    public String getLastName() {
        return lastName;
    }

    public void setLastName(String lastName) {
        this.lastName = lastName;
    }

    public List<Assessment> getAssessment() {
        return assessment;
    }

    public void setAssessment(List<Assessment> assessment) {
        this.assessment = assessment;
    }
}

評価

@Embeddable
public class Assessment {
    private Date date;

    public Date getDate() {
        return date;
    }

    public void setDate(Date date) {
        this.date = date;
    }
}

生成された ClientEndpoint.java

@Api(name = "clientendpoint", namespace = @ApiNamespace(ownerDomain = "myapp.com", ownerName = "myapp.com", packagePath = "mypackage.client"))
public class ClientEndpoint {

    /**
     * This method lists all the entities inserted in datastore.
     * It uses HTTP GET method and paging support.
     *`
     * @return A CollectionResponse class containing the list of all entities
     * persisted and a cursor to the next page.
     */
    @SuppressWarnings({ "unchecked", "unused" })
    @ApiMethod(name = "listClient")
    public CollectionResponse<Client> listClient(
            @Nullable @Named("cursor") String cursorString,
            @Nullable @Named("limit") Integer limit) {

        EntityManager mgr = null;
        Cursor cursor = null;
        List<Client> execute = null;

        try {
            mgr = getEntityManager();
            Query query = mgr.createQuery("select from Client as Client");
            if (cursorString != null && cursorString != "") {
                cursor = Cursor.fromWebSafeString(cursorString);
                query.setHint(JPACursorHelper.CURSOR_HINT, cursor);
            }

            if (limit != null) {
                query.setFirstResult(0);
                query.setMaxResults(limit);
            }

            execute = (List<Client>) query.getResultList();
            cursor = JPACursorHelper.getCursor(execute);
            if (cursor != null)
                cursorString = cursor.toWebSafeString();

            // Tight loop for fetching all entities from datastore and accomodate
            // for lazy fetch.
            for (Client obj : execute)
                ;
        } finally {
            mgr.close();
        }

        return CollectionResponse.<Client> builder().setItems(execute)
                .setNextPageToken(cursorString).build();
    }

    /**
     * This method gets the entity having primary key id. It uses HTTP GET method.
     *
     * @param id the primary key of the java bean.
     * @return The entity with primary key id.
     */
    @ApiMethod(name = "getClient")
    public Client getClient(@Named("id") Long id) {
        EntityManager mgr = getEntityManager();
        Client client = null;
        try {
            client = mgr.find(Client.class, id);
        } finally {
            mgr.close();
        }
        return client;
    }

    /**
     * This inserts a new entity into App Engine datastore. If the entity already
     * exists in the datastore, an exception is thrown.
     * It uses HTTP POST method.
     *
     * @param client the entity to be inserted.
     * @return The inserted entity.
     */
    @ApiMethod(name = "insertClient")
    public Client insertClient(Client client) {
        EntityManager mgr = getEntityManager();
        try {
            if (containsClient(client)) {
                throw new EntityExistsException("Object already exists");
            }
            mgr.persist(client);
        } finally {
            mgr.close();
        }
        return client;
    }

    /**
     * This method is used for updating an existing entity. If the entity does not
     * exist in the datastore, an exception is thrown.
     * It uses HTTP PUT method.
     *
     * @param client the entity to be updated.
     * @return The updated entity.
     */
    @ApiMethod(name = "updateClient")
    public Client updateClient(Client client) {
        EntityManager mgr = getEntityManager();
        try {
            if (!containsClient(client)) {
                throw new EntityNotFoundException("Object does not exist");
            }
            mgr.persist(client);
        } finally {
            mgr.close();
        }
        return client;
    }

    /**
     * This method removes the entity with primary key id.
     * It uses HTTP DELETE method.
     *
     * @param id the primary key of the entity to be deleted.
     * @return The deleted entity.
     */
    @ApiMethod(name = "removeClient")
    public Client removeClient(@Named("id") Long id) {
        EntityManager mgr = getEntityManager();
        Client client = null;
        try {
            client = mgr.find(Client.class, id);
            mgr.remove(client);
        } finally {
            mgr.close();
        }
        return client;
    }

    private boolean containsClient(Client client) {
        if (client.getId() == null)
            return false;

        EntityManager mgr = getEntityManager();
        boolean contains = true;
        try {

            Client item = mgr.find(Client.class, client.getId());
            if (item == null) {
                contains = false;
            }
        } finally {
            mgr.close();
        }
        return contains;
    }

    private static EntityManager getEntityManager() {
        return EMF.get().createEntityManager();
    }

}
4

1 に答える 1

3

問題は JDO の遅延読み込みにある可能性があります。オブジェクトを要求していますが、JDO はオブジェクトを使用しようとするまでその属性をフェッチしません。その時点で、データストアへの接続はすでに失われています。自動生成された例で、遅延読み込みを無効にするコードの例を確認できます。

// Tight loop for fetching all entities from datastore and accomodate
// for lazy fetch.
for (Client obj : execute)
  ;

これにより、リスト内のすべてのオブジェクトがロードClientされてから返されます。ただし、ListプロパティClientも遅延ロードされるため、Clientオブジェクトのassessment属性ごとにこのプロセスを繰り返す必要があります。

編集FetchType.EAGER: App Engine では機能しないため、使用する提案を削除しました。

于 2013-05-31T22:02:03.653 に答える