Akka HTTP:自定义指令(Directive)
自定义指令
有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 。
低调大师中文资讯倾力打造互联网数据资讯、行业资源、电子商务、移动互联网、网络营销平台。
持续更新报道IT业界、互联网、市场资讯、驱动更新,是最及时权威的产业资讯及硬件资讯报道平台。
转载内容版权归作者及来源网站所有,本站原创内容转载请注明来源。
- 上一篇
Go 垃圾回收原理
1. 前言 所谓垃圾就是不再需要的内存块,这些垃圾如果不清理就没办法再次被分配使用,在不支持垃圾回收的编程语言里,这些垃圾内存就是泄露的内存。 Golang的垃圾回收(GC)也是内存管理的一部分,了解垃圾回收最好先了解前面介绍的内存分配原理。 2. 垃圾回收算法 业界常见的垃圾回收算法有以下几种: 引用计数:对每个对象维护一个引用计数,当引用该对象的对象被销毁时,引用计数减1,当引用计数器为0是回收该对象。 优点:对象可以很快的被回收,不会出现内存耗尽或达到某个阀值时才回收。 缺点:不能很好的处理循环引用,而且实时维护引用计数,有也一定的代价。 代表语言:Python、PHP、Swift 标记-清除:从根变量开始遍历所有引用的对象,引用的对象标记为"被引用",没有被标记的进行回收。 优点:解决了引用计数的缺点。 缺点:需要STW,即要暂时停掉程序运行。 代表语言:Golang(其采用三色标记法) 分代收集:按照对象生命周期长短划分不同的代空间,生命周期长的放入老年代,而短的放入新生代,不同代有不能的回收算法和回收频率。 优点:回收性能好 缺点:算法复杂 代表语言: JAVA 3. Go...
- 下一篇
移植facebook flashcache模块支持 linux kernel 4.13以上
环境 准备在ubuntu 18.04 lts下搭建kvm+libvirt+glusterfs的虚拟化环境。 4台服务器是2块SSD,4块SATA(5400RPM)。 尝试了gluster tier,效果不好。 对于虚拟化环境下大文件的存储,tier的promote和demote的策略很难设置, 在大文件发生大量数据交换时会出现tier层"no space left"的错误,文件无法写入。 在没有更好的解决方案之前,facebook的flashcache(https://github.com/facebookarchive/flashcache )对于提速SATA盘的作用还是非常显著的。 但ubuntu18.04的apt源还没有。 https://launchpad.net/flashcache/+packages 错误 于是下载flashcache源码编译dkms模块的时候,报错: DKMS make.log for flashcache-3.1.3+git20150701 for kernel 4.15.0-36-generic (x86_64) Fri Oct 12 15:06:...
相关文章
文章评论
共有0条评论来说两句吧...