在Play和Scala中向case类获取请求参数

问题描述 投票:1回答:1

我继承了2个控制器方法(对于GET请求),它接受相同的10个请求参数,如下所示:

class Application @Inject() (cc: ControllerComponents) extends AbstractController(cc) {

  def func1(param1: String,
            param2: String,
            param3: String
              ...
            param10: String
           ) = Action {
      ...
  }

  def func2(param1: String,
            param2: String,
            param3: String
              ...
            param10: String
           ) = Action {
      ...
  }

}

这些映射如下:

GET           /f1                      blah.blah.Application.func1(p1: String, p2: String...p10: String)
GET           /f2                      blah.blah.Application.func2(p1: String, p2: String...p10: String)

我喜欢避免重复。我想知道是否可以定义一个案例类,其中包含以请求参数命名的10个字段,控制器方法是否接受case-class-type的一个参数,并将Play匹配请求参数名称添加到字段名称并绑定值?

如果在POST请求正文中提交了相同的值,则可以轻松实现此目的。但这不是一个选择,因为这个终点已经暴露给客户。

scala playframework playframework-2.6
1个回答
2
投票

Query string binders用于此。基本上,您告诉Play如何解析参数,将它们分组到一个类并反转(将它们转回String表示)。假设你想要一个Page抽象:

case class Page(from: Int, to: Int)

implicit def pageQSB(implicit intBinder: QueryStringBindable[Int]) = new QueryStringBindable[Page] {
  override def bind(key: String, params: Map[String, Seq[String]]): Option[Either[String, Page]] = {
    for {
      from <- intBinder.bind("from", params)
      to <- intBinder.bind("to", params)
    } yield {
      (from, to) match {
        case (Right(from), Right(to)) => Right(Page(from, to))
        case _ => Left("Unable to bind a Page")
      }
    }
  }
  override def unbind(key: String, page: Page): String = {
    intBinder.unbind("from", page.from) + "&" + intBinder.unbind("to", page.to)
  }
}

请注意,您必须将这些含义导入到路径范围(在build.sbt中),例如:

routesImport += "utils.MyBinders._" 
© www.soinside.com 2019 - 2024. All rights reserved.