Поймать IllegalStateException с Objectify - PullRequest
1 голос
/ 05 января 2020

Я использую Objectify 6.0.5, Ktor 1.2.6, com.google.appengine: appengine: 1.9.60

Я настраиваю сеть. xml, загрузчик по учебнику https://github.com/objectify/objectify/wiki/Setup

web. xml

    ...
    <listener>
        <listener-class>com.group.Bootstrapper</listener-class>
    </listener>

    <filter>
        <filter-name>ObjectifyFilter</filter-name>
        <filter-class>com.googlecode.objectify.ObjectifyFilter</filter-class>
    </filter>
    <filter-mapping>
        <filter-name>ObjectifyFilter</filter-name>
        <url-pattern>/*</url-pattern>
    </filter-mapping>
    ...

Boostrapper

class Bootstrapper : ServletContextListener {
    override fun contextInitialized(sce: ServletContextEvent?) {
        println("         contextInitialized")
        ObjectifyService.init(
            ObjectifyFactory()
        )

        ObjectifyService.register(User::class.java)
    }

    override fun contextDestroyed(sce: ServletContextEvent?) {}
}

Когда я вызываю этот метод

    fun save(entity: T) {
        ofy().save().entity(entity)
    }

Я улавливаю ошибку

2020-01-05 17:55:09 ERROR Application:104 - Unhandled: GET - /test
java.lang.IllegalStateException: You must call ObjectifyService.init() before using Objectify
    at com.google.common.base.Preconditions.checkState(Preconditions.java:511) ~[guava-28.1-android.jar:?]
    at com.googlecode.objectify.ObjectifyService.factory(ObjectifyService.java:34) ~[objectify-6.0.5.jar:?]
    at com.googlecode.objectify.ObjectifyService.ofy(ObjectifyService.java:51) ~[objectify-6.0.5.jar:?]
    at com.group.dao.BaseDao.listAll(BaseDao.kt:15) ~[classes/:?]
    at com.group.ApplicationKt$module$2$2.invokeSuspend(Application.kt:50) ~[classes/:?]
    at com.group.ApplicationKt$module$2$2.invoke(Application.kt) ~[classes/:?]
    at io.ktor.util.pipeline.SuspendFunctionGun.loop(PipelineContext.kt:268) ~[ktor-utils-jvm-1.2.6.jar:1.2.6]
    at io.ktor.util.pipeline.SuspendFunctionGun.proceed(PipelineContext.kt:141) ~[ktor-utils-jvm-1.2.6.jar:1.2.6]
    at io.ktor.util.pipeline.SuspendFunctionGun.execute(PipelineContext.kt:161) ~[ktor-utils-jvm-1.2.6.jar:1.2.6]
    at io.ktor.util.pipeline.Pipeline.execute(Pipeline.kt:27) ~[ktor-utils-jvm-1.2.6.jar:1.2.6]
    at io.ktor.routing.Routing.executeResult(Routing.kt:147) ~[ktor-server-core-1.2.6.jar:1.2.6]
...

Я запускаю хранилище данных, используя «запуск хранилища бета-эмуляторов gcloud». Это определенно работает. Может быть, не подключен к нему?

Когда я запускаю приложение, использую appengineRun, также запускается хранилище данных?

1 Ответ

1 голос
/ 26 апреля 2020

Проблема в том, что ktor использует сопрограммы. Сопрограммы запускаются в пуле потоков по умолчанию, и ObjectifyFilter активен только для потока, получающего запрос. Я использую этот обходной путь, пока не найду что-то лучшее:

get(path) {
    withContext(Dispatchers.Unconfined) {
        val content = page.doGet()
        call.respond(content)
    }
}

Диспетчер Unconfined запускает сопрограмму в том же потоке, который ее запустил, по крайней мере, до первого приостановления.

...