与期货一起玩2.2 EssentialAction

时间:2014-02-18 23:27:48

标签: scala playframework playframework-2.2 reactivemongo

我正在尝试实现类似于此示例的身份验证机制:

    def HasToken(action: String => EssentialAction): EssentialAction = EssentialAction { requestHeader =>
      val maybeToken = requestHeader.headers.get("X-SECRET-TOKEN")
      maybeToken map { token =>
        action(token)(requestHeader) // apply requestHeader to EssentialAction produces the Iteratee[Array[Byte], SimpleResult]
      } getOrElse {
        Done(Unauthorized("401 No Security Token\n")) // 'Done' means the Iteratee has completed its computations
      }
    }

但是,在我的情况下,我将随机令牌值映射到存储在Mongodb中的服务器端的会话。目标是让用户可以随意终止所有其他会话。

然而,我从ReactiveMongo获得的数据将包含在Future中。

我想要这样的事情:

    def HasToken(action: String => EssentialAction): EssentialAction = EssentialAction { requestHeader =>
      val maybeToken = requestHeader.headers.get("session")
      maybeToken map { token =>
        //This returns a future..
        Session.find(session).map { result => 
          result match
            case Some(session) => action(session)(requestHeader)
            case None => Done(Unauthorized())
        }
      } getOrElse {
        Done(Unauthorized("401 No Security Token\n")) // 'Done' means the Iteratee has completed its computations
      }
    }

EssentialAction可以实现吗?

2 个答案:

答案 0 :(得分:2)

Iteratee.flatten来自Future[Iteratee[A, E]] => Iteratee[A, E]所以你可以这样做:

def HasToken(action: String => EssentialAction): EssentialAction = EssentialAction { requestHeader =>
   val maybeToken = requestHeader.headers.get("session")

   val futureIteratee: Future[Iteratee[Array[Byte], SimpleResult]] = maybeToken map { token =>
     //This returns a future..
     Session.find(token).map {
       case Some(session) => action(session)(requestHeader)
       case None => Done[Array[Byte], SimpleResult](Unauthorized("Invalid token"))
     }
   } getOrElse {
     Future.successful(Done[Array[Byte], SimpleResult](Unauthorized("401 No Security Token\n")))
   }

   Iteratee.flatten(futureIteratee)
}

答案 1 :(得分:1)

您可以使用ActionBuilder作为invokeBlock方法返回Future [SimpleResult],这样您就可以将您的未来平面映射到对底层块的调用

这样的东西
object Authenticated extends ActionBuilder[AuthenticatedRequest] {
  def invokeBlock[A](request: Request[A], block: (AuthenticatedRequest[A]) => Future[SimpleResult]) = {
    Session.find(session).map { result => 
      result match
        case Some(session) => block(new AuthenticatedRequest(session))
        case None => Unauthorized()
    }        
}

}

其中AuthenticatedRequest是包装会话对象的请求类型