0

我正在研究这个 Scala play-authenticate-usage-scala项目,我在 Java 框架play-authenticate之上构建(我知道,生活并不完美......)。在运行应用程序时,我看到下面包含的异常。

仔细检查堆栈跟踪导致我看到views._providerPartial.scala.html正在使用forProviders.scala.html模板的视图,该模板又使用依赖于 Java 的 Play 的类play.mvc.Http.{ Session, Context, etc},因此出现错误,因为我的示例应用程序带来了 Scalaplay.api.mvc._的。

我知道我可以在 Scala 中做到:

import play.core.j.JavaHelpers

val jContext : play.mvc.Http.Context = JavaHelpers.createJavaContext(request)
val jSession : play.mvc.Http.Session = context.session() 

问题是我如何才能使上述这些隐含地适用于该视图。

[error] application - 

! @72ef8b5j8 - Internal server error, for (GET) [/login] ->

play.api.http.HttpErrorHandlerExceptions$$anon$1: Execution exception[[RuntimeException: There is no HTTP Context available from here.]]
    at play.api.http.HttpErrorHandlerExceptions$.throwableToUsefulException(HttpErrorHandler.scala:293)
    at play.api.http.DefaultHttpErrorHandler.onServerError(HttpErrorHandler.scala:220)
    at play.api.GlobalSettings$class.onError(GlobalSettings.scala:160)
    at play.api.DefaultGlobal$.onError(GlobalSettings.scala:188)
    at play.api.http.GlobalSettingsHttpErrorHandler.onServerError(HttpErrorHandler.scala:100)
    at play.core.server.netty.PlayRequestHandler$$anonfun$2$$anonfun$apply$1.applyOrElse(PlayRequestHandler.scala:100)
    at play.core.server.netty.PlayRequestHandler$$anonfun$2$$anonfun$apply$1.applyOrElse(PlayRequestHandler.scala:99)
    at scala.concurrent.Future$$anonfun$recoverWith$1.apply(Future.scala:346)
    at scala.concurrent.Future$$anonfun$recoverWith$1.apply(Future.scala:345)
    at scala.concurrent.impl.CallbackRunnable.run(Promise.scala:32)
Caused by: java.lang.RuntimeException: There is no HTTP Context available from here.
    at play.mvc.Http$Context.current(Http.java:62)
    at play.mvc.Http$Context$Implicit.session(Http.java:330)
    at com.feth.play.module.pa.views.html.forProviders_Scope0$forProviders$$anonfun$apply$1.apply(forProviders.template.scala:37)
    at com.feth.play.module.pa.views.html.forProviders_Scope0$forProviders$$anonfun$apply$1.apply(forProviders.template.scala:35)
    at play.twirl.api.TemplateMagic$.defining(TemplateMagic.scala:13)
    at com.feth.play.module.pa.views.html.forProviders_Scope0$forProviders.apply(forProviders.template.scala:35)
    at views.html._providerPartial_Scope0$_providerPartial.apply(_providerPartial.template.scala:38)
    at views.html.login_Scope0$login_Scope1$login.apply(login.template.scala:74)
    at controllers.Application$$anonfun$4$$anonfun$apply$4.apply(Application.scala:57)
    at controllers.Application$$anonfun$4$$anonfun$apply$4.apply(Application.scala:57)

更新我尝试了以下但没有奏效。基本上,Java 方面期望所有这些都在ThreadLocal. 具体ThreadLocal实例公开位于play.mvc.Http.Context.current. 因此,我准备了一个模板助手,_adaptScalaToJava.scala.html如下所示:

@import be.objectify.deadbolt.scala._
@import play.core.j.JavaHelpers

@()(implicit request: AuthenticatedRequest[Any])

@play.mvc.Http.Context.current.set(JavaHelpers.createJavaContext(request))

它直接在当前线程中设置 jContext 并像这样使用:

@_adaptScalaToJava
@forProviders(playAuth, skipCurrent) { p =>
    <li>
    @if(p.getKey() == "openid") {
        <a href="javascript:void(0);" onclick="askOpenID('@p.getUrl()');">
    } else {
        <a href="@p.getUrl()">
    }
    @_providerIcon(p.getKey())</a>
    </li>
}

但即使编译发现它仍然会触发相同的异常......

4

1 回答 1

0

在麻烦的代码解决问题之前只放置这一行:

@play.mvc.Http.Context.current.set(play.core.j.JavaHelpers.createJavaContext(request))
@forProviders(playAuth, skipCurrent) { p =>

试图让它更干净一点,即它不会一直重置,这对我来说仍然是个谜,为什么这不起作用:

@if (play.mvc.Http.Context.current.get() == null) { 
    @play.mvc.Http.Context.current.set(play.core.j.JavaHelpers.
         createJavaContext(request)) 
} 

似乎play.mvc.Http.Context.current不是null......底线是我需要连接IDE并调试代码。

更新实际上这是官方的方法:

@play.core.j.JavaHelpers.withContext(request) { jContext =>
    @forProviders(playAuth, skipCurrent) { p =>
        ...
    }
}
于 2016-12-22T19:04:03.423 回答