内容简介:有3种创建自定义指令的基本方法:创建自定义指令最简便的方法就是将一个或多个已有指令通过配置的方式分配一个新的名字来定义。事实上Akka HTTP预定义的大多数指令都由以较低级别指令命名配置的方式来定义的。如:第二种方式是通过“转换方法”来转换现有指令,这是在
自定义指令
有3种创建自定义指令的基本方法:
- 将已有指令通过命名配置(比如通过组合的方式)的方式来定义新的指令
- 转换已存在的指令
- 从头开始实现一个指令
命名配置
创建自定义指令最简便的方法就是将一个或多个已有指令通过配置的方式分配一个新的名字来定义。事实上Akka HTTP预定义的大多数指令都由以较低级别指令命名配置的方式来定义的。如:
val getPut = get & put def postEntity[T](um: FromRequestUnmarshaller[T]): Directive1[T] = post & entity(um) def completeOk: Route = complete(HttpEntity.Empty) def completeNotImplemented: Route = complete(StatusCodes.NotImplemented)
转换已存在的指令
第二种方式是通过“转换方法”来转换现有指令,这是在 Directive
类上定义的方法:
map/tmap flatMap/tflatMap require/trequire recover/recoverPF
map、tmap
map、tmap
就和Scala集合库上的 map
转换类似,它可以将值映射转换成另一个值。 map
用于 Directive1
类型的指令(单值指令),而 tmap
用于值为其它元组的情况,它的签名如下:
def tmap[R](f: L => R): Directive[Out]
tmap
可以用来将提取的元组转换成另一个元组,提取的数量和类型都可以改变,而 map
只用改变变换后的类型。如下是一个虚构的例子:
val twoIntParameters: Directive[(Int, Int)] = parameters(("a".as[Int], "b".as[Int])) val myDirective: Directive1[String] = twoIntParameters.tmap { case (a, b) => (a + b).toString } // tests: Get("/?a=2&b=5") ~> myDirective(x => complete(x)) ~> check { responseAs[String] mustBe "7" }
flatMap、tflatMap
通过 map、tmap
可以将指令抽取的值转换成其它值,但不能改变其“抽取”的性质。当需要抽取一个对它做一些转换操作,并将结果交给一个嵌套的指令使用时, map、tmap
就无能为力了。同 map、tmap
类似, flatMap
也是用于单值指令,而 tflatMap
用于其它元组值。 tflatMap
的函数签名如下:
def tflatMap[R: Tuple](f: L => Directive[R]): Directive[R]
可以看一个例子,预定义的 method
指令,它的定义如下:
def method(httpMethod: HttpMethod): Directive0 = extractMethod.flatMap[Unit] { case `httpMethod` => pass case _ => reject(MethodRejection(httpMethod)) } & cancelRejections(classOf[MethodRejection]) val get: Directive0 = method(HttpMethods.GET) val post: Directive0 = method(HttpMethods.POST)
-
通过调用
extractMethod
指令获取请求的HTTP方法,再通过flatMap[Unit]
转换方法对它进行处理。因为extractMethod
是一个单值指令且转换后值为Unit
(也是个单值),这里调用flatMap
方法。 -
当请求的实际HTTP方法与传入参数
httpMethod
匹配时,调用pass
指令使其通过,否则调用reject(MethodRejection(httpMethod))
拒绝。
require、trequire
require方法将单个指令转换为没有抽取值的指令,该指令根据谓词函数过滤请求,所有谓词函数调用后为false的请求都被拒绝,其它请求保持不变。它的定义如下:
def require(predicate: T => Boolean, rejections: Rejection*): Directive0 = underlying.filter(predicate, rejections: _*).tflatMap(_ => Empty)
从定义可以看出,它实际上是先通过谓词函数调用 filter
方法对请求进行过滤,然后再调用 tflatMap
函数将指令抽取的值去掉。
recover、recoverPF
recover方法允许“捕获”由底层指令向上冒泡产生的rejections,并生成且有相同抽取类型的替代指令。这样就可以恢复指令来通过而不是拒绝它。它们的定义分别如下:
def recover[R >: L: Tuple](recovery: immutable.Seq[Rejection] => Directive[R]): Directive[R] = Directive[R] { inner => ctx => import ctx.executionContext @volatile var rejectedFromInnerRoute = false tapply({ list => c => rejectedFromInnerRoute = true; inner(list)(c) })(ctx).fast.flatMap { case RouteResult.Rejected(rejections) if !rejectedFromInnerRoute => recovery(rejections).tapply(inner)(ctx) case x => FastFuture.successful(x) } } def recoverPF[R >: L: Tuple](recovery: PartialFunction[immutable.Seq[Rejection], Directive[R]]): Directive[R] = recover { rejections => recovery.applyOrElse(rejections, (rejs: Seq[Rejection]) => RouteDirectives.reject(rejs: _*)) }
从头开始实现一个指令
可以通过调用 Directive.apply
或它的子类型来从头开始定义一个指令, Directive
的简化定义看起来像下面这样:
abstract class Directive[L](implicit val ev: Tuple[L]) { def tapply(f: L => Route): Route } object Directive { /** * Constructs a directive from a function literal. */ def apply[T: Tuple](f: (T => Route) => Route): Directive[T] = new Directive[T] { def tapply(inner: T => Route) = f(inner) } }
Directive
类型有一个抽象方法 tapply
,参数 f
是一个函数类型,将类型 L
传入并返回 Route
。 Directive
的伴身对象提供了 apply
来实现自定义指令。它的参数是一个高阶函数 (T => Route) => Route
,就像小括号那样,我们应把 (T => Route)
看成一个整体,它是函数参数,返回类型为 Route
。
f
为我们自定义指令用于从 RequestContext
里抽取值(值的类型为 Tuple[L]
),而 inner
就是 f
抽取值后调用的嵌套路由,在调用 inner
时将抽取出的值作为参数传入。
对于一个抽取访问host和port的指令,可以这样实现:
def hostnameAndPort: Directive[(String, Int)] = Directive[(String, Int)] { inner => ctx => // inner: (String, Int) => Route // ctx: RequestContext val authority: Uri.Authority = ctx.request.uri.authority val tupleValue: (String, Int) = (authority.host.address(), authority.port) val route: Route = inner(tupleValue) route(ctx) // Future[RouteResult] }
让我们来分析下这个例子:
-
首先是
hostnameAndPort
指令的类型Directive[(String, Int)]
,它从请求上下文(RequestContext
)中抽取出的值是Tuple2[String, Int]
。 -
apply
方法执行的代码参数是:inner => ctx => ....
其实可以看成:inner => ((ctx: RequestContext) => Future[RouteResult])
,inner
就是f
函数参数(T => Route)
部分。 -
inner(tupleValue)
执行后结果route
的类型是Route
,这时这段代码为的类型就为inner => ctx => Route
,而实际上Directive.apply
需要的参数类型为inner => Route
。之前我们知道,Route
是一个类型别名RequestContext => Future[RouteResult]
,所以我们需要将ctx => Route
转换为Route
。而将tupleValue
作为参数调用route
后将获取结果类型Future[RouteResult]
,这段代码的类型就是inner => ctx => Future[RouteResult]
->inner => Route
。
本文节选自《Scala Web开发》,更多内容请访问: https://www.yangbajing.me/scala-web-development/server-api/routing-dsl/custom-directive.html 。
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持 码农网
猜你喜欢:- angularjs自定义指令Directive
- 使用Angular自定义字段校验指令
- vue2.0自定义指令
- 分享 8 个非常实用的 Vue 自定义指令
- Vue 自定义指令上报 Google Analytics 事件统计
- [Vue 2.x Todo 教程] 自定义指令实现自动聚焦
本站部分资源来源于网络,本站转载出于传递更多信息之目的,版权归原作者或者来源机构所有,如转载稿涉及版权问题,请联系我们。