Как настроить однонаправленное отображение один-ко-многим для приложения grails в GAE?

Я пытаюсь выполнить тестирование однонаправленного сопоставления «один-ко-многим» для приложения grails на движке приложений Google (GAE) с использованием JPA. Я пытаюсь определить однонаправленное сопоставление «один ко многим» между классами «Пользователь» и «Роль». К сожалению, я застрял. Просто любопытно, есть ли какой-нибудь разработчик, способный заставить его работать успешно.

Ниже представлена ​​моя среда разработки:

  • Windows XP Professional
  • Пакет SDK для App Engine 1.2.2
  • Grails 1.1.1
  • Используемые плагины: app-engine 0.8.3, gorm-jpa 0.5

Исходный код находится по адресу:

Полная трассировка стека ошибок находится здесь:

 [java] Sep 4, 2009 2:08:42 AM com.google.apphosting.utils.jetty.JettyLogger warn
 [java] WARNING: Failed startup of context com.google.apphosting.utils.jetty.DevAppEngineWebAppContext@1ebd75b{/,C:\Documents and Settings\limcheekin\.grails\1.1.1\projects\one2many\stage}
 [java] org.springframework.beans.factory.access.BootstrapException: Error executing bootstraps; nested exception is org.codehaus.groovy.runtime.InvokerInvocationException: org.springframework.orm.jpa.JpaSystemException: Class "com.vobject.grailsfuse.User" has collection field "roles" and this has no mapping in the table for the element class "com.vobject.grailsfuse.Role"; nested exception is javax.persistence.PersistenceException: Class "com.vobject.grailsfuse.User" has collection field "roles" and this has no mapping in the table for the element class "com.vobject.grailsfuse.Role"
 [java]     at org.codehaus.groovy.grails.web.context.GrailsContextLoader.createWebApplicationContext(GrailsContextLoader.java:74)
 [java]     at org.springframework.web.context.ContextLoader.initWebApplicationContext(ContextLoader.java:199)
 [java]     at org.springframework.web.context.ContextLoaderListener.contextInitialized(ContextLoaderListener.java:45)
 [java]     at org.mortbay.jetty.handler.ContextHandler.startContext(ContextHandler.java:530)
 [java]     at org.mortbay.jetty.servlet.Context.startContext(Context.java:135)
 [java]     at org.mortbay.jetty.webapp.WebAppContext.startContext(WebAppContext.java:1218)
 [java]     at org.mortbay.jetty.handler.ContextHandler.doStart(ContextHandler.java:500)
 [java]     at org.mortbay.jetty.webapp.WebAppContext.doStart(WebAppContext.java:448)
 [java]     at org.mortbay.component.AbstractLifeCycle.start(AbstractLifeCycle.java:40)
 [java]     at org.mortbay.jetty.handler.HandlerWrapper.doStart(HandlerWrapper.java:117)
 [java]     at org.mortbay.component.AbstractLifeCycle.start(AbstractLifeCycle.java:40)
 [java]     at org.mortbay.jetty.handler.HandlerWrapper.doStart(HandlerWrapper.java:117)
 [java]     at org.mortbay.jetty.Server.doStart(Server.java:217)
 [java]     at org.mortbay.component.AbstractLifeCycle.start(AbstractLifeCycle.java:40)
 [java]     at com.google.appengine.tools.development.JettyContainerService.startContainer(JettyContainerService.java:152)
 [java]     at com.google.appengine.tools.development.AbstractContainerService.startup(AbstractContainerService.java:116)
 [java]     at com.google.appengine.tools.development.DevAppServerImpl.start(DevAppServerImpl.java:218)
 [java]     at com.google.appengine.tools.development.DevAppServerMain$StartAction.apply(DevAppServerMain.java:162)
 [java]     at com.google.appengine.tools.util.Parser$ParseResult.applyArgs(Parser.java:48)
 [java]     at com.google.appengine.tools.development.DevAppServerMain.<init>(DevAppServerMain.java:113)
 [java]     at com.google.appengine.tools.development.DevAppServerMain.main(DevAppServerMain.java:89)
 [java] Caused by: org.codehaus.groovy.runtime.InvokerInvocationException: org.springframework.orm.jpa.JpaSystemException: Class "com.vobject.grailsfuse.User" has collection field "roles" and this has no mapping in the table for the element class "com.vobject.grailsfuse.Role"; nested exception is javax.persistence. PersistenceException: Class "com.vobject.grailsfuse.User" has collection field "roles" and this has no mapping in the table for the element class "com.vobject.grailsfuse.Role"
 [java]     ... 7 more
 [java] Caused by: org.springframework.orm.jpa.JpaSystemException: Class "com.vobject.grailsfuse.User" has collection field "roles" and this has no mapping in the table for the element class "com.vobject.grailsfuse.Role"; nested exception is javax.persistence.PersistenceException: Class "com.vobject.grailsfuse.User" has collection field "roles" and this has no mapping in the table for the element class "com.vobject.grailsfuse.Role"
 [java]     at org.grails.jpa.JpaPluginSupport$__clinit__closure3_closure6_closure11_closure38.doCall(JpaPluginSupport.groovy:452)
 [java]     at BootStrap$_closure1.doCall(BootStrap.groovy:13)
 [java]     ... 7 more
 [java] Caused by: javax.persistence.PersistenceException: Class "com.vobject.grailsfuse.User" has collection field "roles" and this has no mapping in the table for the element class "com.vobject.grailsfuse.Role"
 [java]     at org.datanucleus.jpa.NucleusJPAHelper.getJPAExceptionForNucleusException(NucleusJPAHelper.java:264)
 [java]     at org.datanucleus.jpa.EntityTransactionImpl.commit(EntityTransactionImpl.java:122)
 [java]     ... 9 more
 [java] Caused by: org.datanucleus.exceptions.NucleusUserException: Class "com.vobject.grailsfuse.User" has collection field "roles" and this has no mapping in the table for the element class "com.vobject.grailsfuse.Role"
 [java]     at org.datanucleus.store.mapped.scostore.FKSetStore.<init>(FKSetStore.java:184)
 [java]     at org.datanucleus.store.appengine.DatastoreFKSetStore.<init>(DatastoreFKSetStore.java:38)
 [java]     at org.datanucleus.store.appengine.DatastoreManager.newFKSetStore(DatastoreManager.java:353)
 [java]     at org.datanucleus.store.mapped.MappedStoreManager.getBackingStoreForCollection(MappedStoreManager.java:734)
 [java]     at org.datanucleus.store.mapped.MappedStoreManager.getBackingStoreForField(MappedStoreManager.java:646)
 [java]     at org.datanucleus.sco.backed.HashSet.<init>(HashSet.java:102)
 [java]     at org.datanucleus.util.ClassUtils.newInstance(ClassUtils.java:94)
 [java]     at org.datanucleus.sco.SCOUtils.newSCOInstance(SCOUtils.java:164)
 [java]     at org.datanucleus.state.JDOStateManagerImpl.wrapSCOField(JDOStateManagerImpl.java:3040)
 [java]     at org.datanucleus.store.fieldmanager.LoadFieldManager.internalFetchObjectField(LoadFieldManager.java:92)
 [java]     at org.datanucleus.store.fieldmanager.AbstractFetchFieldManager.fetchObjectField(AbstractFetchFieldManager.java:104)
 [java]     at org.datanucleus.state.AbstractStateManager.replacingObjectField(AbstractStateManager.java:1197)
 [java]     at com.vobject.grailsfuse.User.jdoReplaceField(User.groovy)
 [java]     at com.vobject.grailsfuse.User.jdoReplaceFields(User.groovy)
 [java]     at org.datanucleus.state.JDOStateManagerImpl.replaceFields(JDOStateManagerImpl.java:2772)
 [java]     at org.datanucleus.state.JDOStateManagerImpl.replaceFields(JDOStateManagerImpl.java:2791)
 [java]     at org.datanucleus.state.JDOStateManagerImpl.loadFieldsInFetchPlan(JDOStateManagerImpl.java:1610)
 [java]     at org.datanucleus.ObjectManagerImpl.performDetachAllOnCommitPreparation(ObjectManagerImpl.java:3192)
 [java]     at org.datanucleus.ObjectManagerImpl.preCommit(ObjectManagerImpl.java:2931)
 [java]     at org.datanucleus.TransactionImpl.internalPreCommit(TransactionImpl.java:369)
 [java]     at org.datanucleus.TransactionImpl.commit(TransactionImpl.java:256)
 [java]     at org.datanucleus.jpa.EntityTransactionImpl.commit(EntityTransactionImpl.java:104)
 [java]     ... 9 more
 [java] Sep 4, 2009 2:08:42 AM com.google.apphosting.utils.jetty.JettyLogger warn
 [java] WARNING: Nested in org.springframework.beans.factory.access.BootstrapException: Error executing bootstraps; nested exception is org.codehaus.groovy.runtime.InvokerInvocationException: org.springframework.orm.jpa.JpaSystemException: Class "com.vobject.grailsfuse.User" has collection field "roles" and this has no mapping in the table for the element class "com.vobject.grailsfuse.Role"; nested exception is javax.persistence.PersistenceException: Class "com.vobject.grailsfuse.User" has collection field "roles" and this has no mapping in the table for the element class "com.vobject.grailsfuse.Role":
 [java] Class "com.vobject.grailsfuse.User" has collection field "roles" and this has no mapping in the table for the element class "com.vobject.grailsfuse.Role"
 [java] org.datanucleus.exceptions.NucleusUserException: Class "com.vobject.grailsfuse.User" has collection field "roles" and this has no mapping in the table for the element class "com.vobject.grailsfuse.Role"
 [java]     at org.datanucleus.store.mapped.scostore.FKSetStore.<init>(FKSetStore.java:184)
 [java]     at org.datanucleus.store.appengine.DatastoreFKSetStore.<init>(DatastoreFKSetStore.java:38)
 [java]     at org.datanucleus.store.appengine.DatastoreManager.newFKSetStore(DatastoreManager.java:353)
 [java]     at org.datanucleus.store.mapped.MappedStoreManager.getBackingStoreForCollection(MappedStoreManager.java:734)
 [java]     at org.datanucleus.store.mapped.MappedStoreManager.getBackingStoreForField(MappedStoreManager.java:646)
 [java]     at org.datanucleus.sco.backed.HashSet.<init>(HashSet.java:102)
 [java]     at org.datanucleus.util.ClassUtils.newInstance(ClassUtils.java:94)
 [java]     at org.datanucleus.sco.SCOUtils.newSCOInstance(SCOUtils.java:164)
 [java]     at org.datanucleus.state.JDOStateManagerImpl.wrapSCOField(JDOStateManagerImpl.java:3040)
 [java]     at org.datanucleus.store.fieldmanager.LoadFieldManager.internalFetchObjectField(LoadFieldManager.java:92)
 [java]     at org.datanucleus.store.fieldmanager.AbstractFetchFieldManager.fetchObjectField(AbstractFetchFieldManager.java:104)
 [java]     at org.datanucleus.state.AbstractStateManager.replacingObjectField(AbstractStateManager.java:1197)
 [java]     at com.vobject.grailsfuse.User.jdoReplaceField(User.groovy)
 [java]     at com.vobject.grailsfuse.User.jdoReplaceFields(User.groovy)
 [java]     at org.datanucleus.state.JDOStateManagerImpl.replaceFields(JDOStateManagerImpl.java:2772)
 [java]     at org.datanucleus.state.JDOStateManagerImpl.replaceFields(JDOStateManagerImpl.java:2791)
 [java]     at org.datanucleus.state.JDOStateManagerImpl.loadFieldsInFetchPlan(JDOStateManagerImpl.java:1610)
 [java]     at org.datanucleus.ObjectManagerImpl.performDetachAllOnCommitPreparation(ObjectManagerImpl.java:3192)
 [java]     at org.datanucleus.ObjectManagerImpl.preCommit(ObjectManagerImpl.java:2931)
 [java]     at org.datanucleus.TransactionImpl.internalPreCommit(TransactionImpl.java:369)
 [java]     at org.datanucleus.TransactionImpl.commit(TransactionImpl.java:256)
 [java]     at org.datanucleus.jpa.EntityTransactionImpl.commit(EntityTransactionImpl.java:104)
 [java]     at org.grails.jpa.JpaPluginSupport$__clinit__closure3_closure6_closure11_closure38.doCall(JpaPluginSupport.groovy:452)
 [java]     at BootStrap$_closure1.doCall(BootStrap.groovy:13)
 [java]     at com.google.appengine.tools.development.JettyContainerService.startContainer(JettyContainerService.java:152)
 [java]     at com.google.appengine.tools.development.AbstractContainerService.startup(AbstractContainerService.java:116)
 [java]     at com.google.appengine.tools.development.DevAppServerImpl.start(DevAppServerImpl.java:218)
 [java]     at com.google.appengine.tools.development.DevAppServerMain$StartAction.apply(DevAppServerMain.java:162)
 [java]     at com.google.appengine.tools.util.Parser$ParseResult.applyArgs(Parser.java:48)
 [java]     at com.google.appengine.tools.development.DevAppServerMain.<init>(DevAppServerMain.java:113)
 [java]     at com.google.appengine.tools.development.DevAppServerMain.main(DevAppServerMain.java:89)

Пожалуйста посоветуй. Посмотрите, знаете ли вы, что пошло не так…

Спасибо.


person limcheekin    schedule 04.09.2009    source источник


Ответы (2)


Для обновления я выпустил плагин appengine-gorm для репозитория плагинов grails, поэтому я написал этот плагин для поддержки операции пакетной вставки. Поскольку после стольких попыток я не могу установить связь «один-ко-многим» с помощью аннотации (@OneToMany), работающей в Google App Engine, я решил управлять отношениями вручную, а не использовать сопоставление. Однако, хотя я пытаюсь сохранить объекты во многих аспектах отношения «один-ко-многим», я сталкиваюсь с тайм-аутом запроса движка приложений Google. Затем я обнаружил, что низкоуровневый API-интерфейс поддержки хранилища данных сохраняет множество объектов в одном вызове API, так что это порождение этого плагина. Надеюсь, это принесет пользу вам и другим.

Наконец, мне удалось заставить отношения «один ко многим» работать вручную и без сопоставления. Пожалуйста, посмотрите живую демонстрацию приложения под названием GrailsFuse по адресу http://grailsfuse.vobject.com/.

Жду ваших отзывов и комментариев. Пожалуйста, напишите мне письмо на адрес limcheekin на vobject dot com.

Спасибо.

person limcheekin    schedule 30.11.2009

Похоже, проблема в том, что у вас нет пользователя в роли, так как вам нужно указать mappedBy в коллекции, я думаю.

@OneToMany(mappedBy="user",fetch=FetchType.EAGER)
Set<Role> roles 

Кроме того, вам действительно не нужны материалы с таблицами соединения, поскольку вы на самом деле не работаете с таблицами в движке приложения. По сути, роль будет существовать под пользователем.

Надеюсь, это поможет :)

person Community    schedule 04.09.2009
comment
о, и не забудьте добавить пользовательское поле в класс Role - person ; 04.09.2009
comment
Спасибо за быстрый ответ и полезные советы. Я попробую ваш совет, обратившись к java2s.com/Code/Java/JPA/ Но я столкнулся с ошибкой ниже при доступе к странице ролей / списков. Причина ошибки: org.datanucleus.exceptions.NucleusUserException: Поле com.vobject.grailsfuse.Role.user должно иметь возможность предоставлять ссылку на своего родителя, но у объекта нет родителя. Возможно, вы пытались создать экземпляр com.vobject.grailsfuse.Role как дочерний для экземпляра com.vobject.grailsfuse.User после того, как дочерний элемент уже был сохранен? - person limcheekin; 05.09.2009
comment
Последний исходный код находится по адресу: rapidshare.com/files/275893205/one2many.zip.html - person limcheekin; 05.09.2009