回顾

大家好,我是老马。

最近 dubbo3.x 在公司内部分享,于是想系统梳理一下。

总体思路是官方文档入门+一些场景的问题思考+源码解析学习。

本文是Triple Rest的用户使用手册

注意

从 Dubbo 3.3 版本开始,原 Rest 协议已移至 Extensions 库,由 Triple 协议来对 Rest 提供更全面的支持,如需继续使用原 Rest 协议, 可引入对应 dubbo-spi-extensions 库依赖

前言

从 Dubbo 3.3 版本开始,Triple 协议重用已有的 HTTP 协议栈,实现了全面的 REST 风格服务导出能力。无需使用泛化或网关层协议转换,无需配置,用户即可通过 HTTP 协议去中心化直接访问后端的 Triple 协议服务。同时,针对高级 REST 用法,如路径定制、输出格式定制和异常处理,提供了丰富的注解和 SPI 扩展支持。其主要特性包括:

  • Triple协议融合
    重用Triple原有HTTP协议栈, 无需额外配置或新增端口,即可同时支持 HTTP/1、HTTP/2 和 HTTP/3 协议的访问。
  • 去中心化
    可直接对外暴露 Rest API,不再依赖网关应用进行流量转发,从而提升性能,并降低因网关引发的稳定性风险。安全问题可通过应用内部扩展解决,这一实践已在淘宝内部的 MTOP 中得到验证。
  • 支持已有servlet设施
    支持 Servlet API 和 Filter,用户可以重用现有基于 Servlet API 的安全组件。通过实现一个 Servlet Filter,即可集成 OAuth 和 Spring Security 等安全框架。
  • 多种方言
    考虑到大部分用户习惯使用 SpringMVC 或 JAX-RS 进行 REST API 开发,Triple Rest 允许继续沿用这些方式定义服务,并支持大部分扩展和异常处理机制(具备原框架 80% 以上的功能)。对于追求轻量级的用户,可使用 Basic 方言,Triple 的开箱即用 REST 访问能力即基于此方言导出服务。
  • 扩展能力强
    提供超过 20 个 扩展点,用户不仅可以轻松实现自定义方言,还能灵活定制参数获取、类型转换、错误处理等逻辑。
  • 开箱即用
    REST 能力开箱即用,只需启用 Triple 协议,即具备 REST 直接访问服务能力。
  • 高性能路由
    路由部分采用优化的 Radix Tree 和 Zero Copy 技术,提升路由性能。
  • OpenAPI无缝集成(TBD)
    即将完成 OpenAPI 集成,开箱即用支持导出 OpenAPI Schema, 引入 Swagger 依赖可直接使用 Web UI 来进行服务测试。有了 OpenAPI Schema 可使用 PostmanApifox 等API工具来管理和测试 API,利用 OpenAPI 生态可轻松实现跨语言调用。未来会进一步支持 Schema First 的方式,先和前端团队一起定义 OpenAPI, 前端基于 OpenAPI 来生成调用代码和 Mock,后端基于 OpenAPI 来生成 Stub 来开发服务,极大提升协同效率。

快速开始

让我们从一个简单的例子开始了解 Triple Rest。您可以直接下载已有的示例项目以快速上手,假设您已经安装好 Java、Maven 和 Git

下载并运行示例

  [shell]
1
2
3
4
5
6
7
8
# 获取示例代码 git clone --depth=1 [https://github.com/apache/dubbo-samples.git ](https://github.com/apache/dubbo-samples.git ) cd dubbo-samples/2-advanced/dubbo-samples-triple-rest/dubbo-samples-triple-rest-basic # 直接运行 mvn spring-boot:run # 或打包后运行 mvn clean package -DskipTests java -jar target/dubbo-samples-triple-rest-basic-1.0.0-SNAPSHOT.jar

当然,也可以直接用IDE导入工程后直接执行 org.apache.dubbo.rest.demo.BasicRestApplication#main 来运行,并通过下断点 debug 的方式来深入理解原理。

示例代码

  [java]
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
// 服务接口 package org.apache.dubbo.rest.demo; import org.apache.dubbo.remoting.http12.rest.Mapping; import org.apache.dubbo.remoting.http12.rest.Param; public interface DemoService { String hello(String name); @Mapping(path = "/hi", method = HttpMethods.POST) String hello(User user, @Param(value = "c", type = ParamType.Header) int count); } // 服务实现 @DubboService public class DemoServiceImpl implements DemoService { @Override public String hello(String name) { return "Hello " + name; } @Override public String hello(User user, int count) { return "Hello " + user.getTitle() + ". " + user.getName() + ", " + count; } } // 模型 @Data public class User { private String title; private String name; }

测试基本服务

  [shell]
1
2
3
4
5
6
7
8
9
10
11
12
13
14
curl -v "[http://127.0.0.1:8081/org.apache.dubbo.rest.demo.DemoService/hello ](http://127.0.0.1:8081/org.apache.dubbo.rest.demo.DemoService/hello )?name=world" # 输出如下 #> GET /org.apache.dubbo.rest.demo.DemoService/hello?name=world HTTP/1.1 #> Host: 127.0.0.1:8081 #> User-Agent: curl/8.7.1 #> Accept: */* #> #* Request completely sent off #< HTTP/1.1 200 OK #< content-type: application/json #< alt-svc: h2=":8081" #< content-length: 13 #< #"Hello world"

代码讲解:
可以看到输出了 “Hello world” ,有双引号是因为默认输出 content-type 为 application/json
通过这个例子可以了解 Triple 默认将服务导出到 /{serviceInterface}/{methodName}路径,并支持通过url方式传递参数

测试高级服务

  [shell]
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
curl -v -H "c: 3" -d 'name=Yang' "[http://127.0.0.1:8081/org.apache.dubbo.rest.demo.DemoService/hi.txt ](http://127.0.0.1:8081/org.apache.dubbo.rest.demo.DemoService/hi.txt )?title=Mr" # 输出如下 #> POST /org.apache.dubbo.rest.demo.DemoService/hi.txt?title=Mr HTTP/1.1 #> Host: 127.0.0.1:8081 #> User-Agent: curl/8.7.1 #> Accept: */* #> c: 3 #> Content-Length: 9 #> Content-Type: application/x-[www-form-urlencoded](https://www-form-urlencoded) #> #* upload completely sent off: 9 bytes #< HTTP/1.1 200 OK #< content-type: text/plain #< alt-svc: h2=":8081" #< content-length: 17 #< #Hello Mr. Yang, 3

代码讲解:
可以看到输出 Hello Mr. Yang, 3 ,没有双引号是因为通过指定后缀 txt 的方式要求用 text/plain 输出
通过这个例子可以了解如何通过 Mapping 注解来定制路径,通过 Param 注解来定制参数来源,并支持通过 post body 或 url方式传递参数,详细说明参见: Basic使用指南

观察日志

可以通过打开 debug 日志的方式来了解rest的启动和响应请求过程

  [yaml]
1
2
3
4
logging: level: "org.apache.dubbo.rpc.protocol.tri": debug "org.apache.dubbo.remoting": debug

打开后可以观察到 Rest 映射注册和请求响应过程

# 注册mapping
DEBUG o.a.d.r.p.t.TripleProtocol               :  [DUBBO] Register triple grpc mapping: 'org.apache.dubbo.rest.demo.DemoService' -> invoker[tri://192.168.2.216:8081/org.apache.dubbo.rest.demo.DemoService]
INFO .r.p.t.r.m.DefaultRequestMappingRegistry :  [DUBBO] BasicRequestMappingResolver resolving rest mappings for ServiceMeta{interface=org.apache.dubbo.rest.demo.DemoService, service=DemoServiceImpl@2a8f6e6} at url [tri://192.168.2.216:8081/org.apache.dubbo.rest.demo.DemoService]
DEBUG .r.p.t.r.m.DefaultRequestMappingRegistry :  [DUBBO] Register rest mapping: '/org.apache.dubbo.rest.demo.DemoService/hi' -> mapping=RequestMapping{name='DemoServiceImpl#hello', path=PathCondition{paths=[org.apache.dubbo.rest.demo.DemoService/hi]}, methods=MethodsCondition{methods=[POST]}}, method=MethodMeta{method=org.apache.dubbo.rest.demo.DemoService.hello(User, int), service=DemoServiceImpl@2a8f6e6}
DEBUG .r.p.t.r.m.DefaultRequestMappingRegistry :  [DUBBO] Register rest mapping: '/org.apache.dubbo.rest.demo.DemoService/hello' -> mapping=RequestMapping{name='DemoServiceImpl#hello~S', path=PathCondition{paths=[org.apache.dubbo.rest.demo.DemoService/hello]}}, method=MethodMeta{method=org.apache.dubbo.rest.demo.DemoService.hello(String), service=DemoServiceImpl@2a8f6e6}
INFO .r.p.t.r.m.DefaultRequestMappingRegistry :  [DUBBO] Registered 2 REST mappings for service [DemoServiceImpl@44627686] at url [tri://192.168.2.216:8081/org.apache.dubbo.rest.demo.DemoService] in 11ms

# 请求响应
DEBUG .a.d.r.p.t.r.m.RestRequestHandlerMapping :  [DUBBO] Received http request: DefaultHttpRequest{method='POST', uri='/org.apache.dubbo.rest.demo.DemoService/hi.txt?title=Mr', contentType='application/x-[www-form-urlencoded](https://www-form-urlencoded)'}
DEBUG .r.p.t.r.m.DefaultRequestMappingRegistry :  [DUBBO] Matched rest mapping=RequestMapping{name='DemoServiceImpl#hello', path=PathCondition{paths=[/org.apache.dubbo.rest.demo.DemoService/hi]}, methods=MethodsCondition{methods=[POST]}}, method=MethodMeta{method=org.apache.dubbo.rest.demo.DemoService.hello(User, int), service=DemoServiceImpl@2a8f6e6}
DEBUG .a.d.r.p.t.r.m.RestRequestHandlerMapping :  [DUBBO] Content-type negotiate result: request='application/x-[www-form-urlencoded](https://www-form-urlencoded)', response='text/plain'
DEBUG .d.r.h.AbstractServerHttpChannelObserver :  [DUBBO] Http response body is: '"Hello Mr. Yang, 3"'
DEBUG .d.r.h.AbstractServerHttpChannelObserver :  [DUBBO] Http response headers sent: {:status=[200], content-type=[text/plain], alt-svc=[h2=":8081"], content-length=[17]}

通用功能

路径映射

兼容 SpringMVC 和 JAX-RS 的映射方式,相关文档:

还支持通过实现 SPI org.apache.dubbo.rpc.protocol.tri.rest.mapping.RequestMappingResolver 来自定义路径映射

支持的模式

  1. books 字符串常量,最基本的类型,匹配一个固定的段
  2. ? 匹配一个字符
  3. * 匹配路径段中的零个或多个字符
  4. ** 匹配直到路径末尾的零个或多个路径段
  5. {spring} 匹配一个路径段并将其捕获为名为 “spring” 的变量
  6. {spring:[a-z]+} 使用正则表达式 [a-z]+ 匹配路径段,并将其捕获为名为 “spring” 的路径变量
  7. {*spring} 匹配直到路径末尾的零个或多个路径段,并将其捕获为名为 “spring” 的变量,如果写 {*} 表示不捕获

示例(来自Spring文档)

  • /pages/t?st.html — 匹配 /pages/test.html 以及 /pages/tXst.html,但不匹配 /pages/toast.html
  • /resources/*.png — 匹配 resources 目录中的所有 .png 文件
  • com/**/test.jsp — 匹配 com 路径下的所有 test.jsp 文件
  • org/springframework/**/*.jsp — 匹配 org/springframework 路径下的所有 .jsp 文件
  • /resources/** — 匹配 /resources/ 路径下的所有文件,包括 /resources/image.png/resources/css/spring.css
  • /resources/{*path} — 匹配 /resources/ 下的所有文件,以及 /resources,并将其相对路径捕获在名为 “path” 的变量中;例如, /resources/image.png 会匹配到 “path” → “/image.png”,/resources/css/spring.css 会匹配到 “path” → “/css/spring.css”
  • /resources/{filename:\\w+}.dat — 匹配 /resources/spring.dat 并将值 “spring” 分配给 filename 变量
  • /{name:[a-z-]+}-{version:\\d\\.\\d\\.\\d}{ext:\\.[a-z]+} — 匹配 /example-2.1.5.htmlnameexampleversion2.1.5ext.html

小技巧如果使用正则不希望跨段可以使用 {name:[^/]+} 来匹配

映射匹配完整流程

具体的匹配处理代码:DefaultRequestMappingRegistry.java RequestMapping.java

  1. 使用 PathUtils.normalize 对路径进行清洗,去掉诸如 /one/../ /one/./ 之类间接路径,保证一定已 / 开头
  2. 检查 http method 是否匹配
  3. 检查 path 是否匹配
  4. 检查 paramter 是否匹配(JAX-RS不支持)
  5. 检查 header 是否匹配
  6. 检查 content-type 是否匹配(Consumes)
  7. 检查 accept 是否匹配 (Produces)
  8. 检查 serviceGroupserviceVersion 是否匹配
  9. 检查 method 首字母签名是否匹配
  10. 未找到任何匹配项,如果尾 / 匹配开启并且路径 / 结尾则去掉尾 / 尝试从第2步开始匹配
  11. 未找到任何匹配项,如果扩展名匹配开启并且扩展名被支持,则去掉扩展名尝试从第2步开始匹配
  12. 如果最后一段路径包含 ~ 表示开启 method 首字母签名匹配,尝试从第2步开始匹配
  13. 如果候选项为0,匹配结束,返回null
  14. 如果候选项为0,匹配结束,返回命中项
  15. 如果不止一个候选项,对候选项进行排序
  16. 对第一项和第二项进行顺序比较
  17. 结果为0表示无法确认最终匹配项,抛异常失败
  18. 第一项胜出,匹配结束,返回命中项

路径重复问题

与 Spring 不同,Spring 在路径完全相同时会直接报错并阻止启动,而 Triple Rest 具备开箱即用的特性,为了避免影响现有服务,默认只会打印 WARN 日志。在运行时,如果最终无法确定最高优先级的映射,才会抛出错误。

入参类型

不同方言支持的入参类型不同,详情请参见各方言使用指南。

还支持通过实现 SPI org.apache.dubbo.rpc.protocol.tri.rest.argument.ArgumentResolver 来自定义入参解析

通用类型参数

以下是整理后的Markdown表格:

参数注解对照表

名称 说明 Basic 注解 SpringMVC注解 JAX-RS注解 数组或集合处理方式 Map处理方式
Param Query或Form的参数 @Param @RequestParam - 多值 所有参数的Map
Query url上带的参数 - - @QueryParam 多值 所有Query参数的Map
Form form表单带的参数 - - @FormParam 多值 所有Form参数的Map
Header HTTP头 @Param(type=Header) @RequestHeader @HeaderParam 多值 所有Header参数的Map
Cookie Cookie值 @Param(type=Cookie) @CookieValue @CookieParam 多值 所有Cookie参数的Map
Attribute Request属性 @Param(type=Attribute) @RequestAttribute - 多值 所有Attribute的Map
Part Multipart文件 @Param(type=Part) @RequestHeader @HeaderParam 多值 所有Part的Map
Body 请求body @Param(type=Body) @RequestBody @Body 尝试解析为数组或集合 尝试解析为目标类型
PathVariable path变量 @Param(type=PathVariable) @PathVariable @PathParam 单值数组或集合 单值Map
MatrixVariable matrix变量 @Param(type=MatrixVariable) @MatrixVariable @MatrixParam 多值 单值Map
Bean java bean 无需注解 @ModelAttribute @BeanParam 尝试解析为Bean数组或集合 -

特殊类型参数

类型 说明 激活条件
org.apache.dubbo.remoting.http12.HttpRequest HttpRequest对象 默认激活
org.apache.dubbo.remoting.http12.HttpResponse HttpResponse对象 默认激活
org.apache.dubbo.remoting.http12.HttpMethods 请求Http方法 默认激活
java.util.Locale 请求Locale 默认激活
java.io.InputStream 请求输入流 默认激活
java.io.OutputStream 响应输出流 默认激活
javax.servlet.http.HttpServletRequest Servlet HttpRequest对象 引入Servlet API jar
javax.servlet.http.HttpServletResponse Servlet HttpResponse对象 同上
javax.servlet.http.HttpSession Servlet HttpSession对象 同上
javax.servlet.http.Cookie Servlet Cookie对象 同上
java.io.Reader Servlet Request Reader对象 同上
java.io.Writer Servlet Response Writer对象 同上

无注解参数

不同方言处理方式不同,请参见各方言使用说明

无入参方式获取 HTTP 输入输出参数

可通过 RpcContext 来获取

  [java]
1
2
3
4
5
6
// Dubbo http req/resp HttpRequest request = RpcContext.getServiceContext().getRequest(HttpRequest.class); HttpResponse response = RpcContext.getServiceContext().getRequest(HttpResponse.class); // Servlet http req/resp HttpServletRequest request = RpcContext.getServiceContext().getRequest(HttpServletRequest.class); HttpServletResponse response = RpcContext.getServiceContext().getRequest(HttpServletResponse.class);

拿到request之后,通过 attribute 可以访问一些内置属性,参见:RestConstants.java

参数类型转换 默认支持大部分从 String 到目标类型的参数类型转换,主要包括以下几大类:

  • Jdk内置类型,包括基础类型和日期、Optional等

  • 数组类型

  • 集合类型

  • Map类型

同时也完整支持泛型类型,包括复杂嵌套,具体地实现代码参见: GeneralTypeConverter.java

还支持通过实现SPI org.apache.dubbo.rpc.protocol.tri.rest.argument.ArgumentConverter 来自定义参数类型转换

类型转换对照表

Source Type Target Type Description Default Value
String double Converts to a double 0.0d
String float Converts to a float 0.0f
String long Converts to a long 0L
String int Converts to an integer 0
String short Converts to a short 0
String char Converts to a character 0
String byte Converts to a byte 0
String boolean Converts to a boolean false
String BigInteger Converts to a BigInteger null
String BigDecimal Converts to a BigDecimal null
String Date Converts to a Date null
String Calendar Converts to a Calendar null
String Timestamp Converts to a Timestamp null
String Instant Converts to an Instant null
String ZonedDateTime Converts to a ZonedDateTime null
String LocalDate Converts to a LocalDate null
String LocalTime Converts to a LocalTime null
String LocalDateTime Converts to a LocalDateTime null
String ZoneId Converts to a ZoneId null
String TimeZone Converts to a TimeZone null
String File Converts to a File null
String Path Converts to a Path null
String Charset Converts to a Charset null
String InetAddress Converts to an InetAddress null
String URI Converts to a URI null
String URL Converts to a URL null
String UUID Converts to a UUID null
String Locale Converts to a Locale null
String Currency Converts to a Currency null
String Pattern Converts to a Pattern null
String Class Converts to a Class null
String byte[] Converts to a byte array null
String char[] Converts to a char array null
String OptionalInt Converts to an OptionalInt null
String OptionalLong Converts to an OptionalLong null
String OptionalDouble Converts to an OptionalDouble null
String Enum class Enum.valueOf null
String Array or Collection Split by comma null
String Specified class Try JSON String to Object null
String Specified class Try construct with single String null
String Specified class Try call static method valueOf null

支持的 Content-Type

默认支持以下 Content-Type,提供相应的编码和解码功能。
支持通过实现 SPI org.apache.dubbo.remoting.http12.message.(HttpMessageDecoderFactory|HttpMessageEncoderFactory) 扩展。

Media Type Description
application/json JSON format
application/xml XML format
application/yaml YAML format
application/octet-stream Binary data
application/grpc gRPC format
application/grpc+proto gRPC with Protocol Buffers
application/x-www-form-urlencoded URL-encoded form data
multipart/form-data Form data with file upload
text/json JSON format as text
text/xml XML format as text
text/yaml YAML format as text
text/css CSS format
text/javascript JavaScript format as text
text/html HTML format
text/plain Plain text

内容协商

支持完善的内容协商机制,可根据映射或输入来协商输出的 Content-Type,具体流程如下:

  • 尝试读取 Mapping 指定的 mediaType,获取 Produces指定的 mediaType 列表,并将通配符匹配到合适的 Media Type。例如Spring的: @RequestMapping(produces = - “application/json”)

  • 尝试通过 Accept 头查找 mediaType,解析请求的 Accept 头,并将通配符匹配到合适的 Media Type。例如: Accept: application/json

  • 尝试通过 format 参数查找 mediaType,读取 format 参数值,做为文件后缀匹配到合适的 Media Type。例如 /hello?format=yml

  • 尝试通过请求路径的扩展名查找 mediaType,扩展名做为文件后缀匹配到合适的 Media Type。例如 /hello.txt

  • 尝试读取请求的 Content-Type 头做为 Media Type(两种form类型除外)。例如 Content-Type: application/json

  • 使用 application/json 兜底

CORS支持

提供完整的CORS支持,通过配置全局参数即可启用,默认行为和SpringMVC一致,同时在SpringMVC方言中,也支持通过 @CrossOrigin 来做精细化配置。 支持的CORS 配置项参见:8.4CORS配置

自定义HTTP输出

很多场景需要对HTTP输出进行定制,比如做302跳转,写Http头,为此 Triple Rest提供以下通用方案,同时也支持各方言的特定写法,详情参见各方言使用指南

返回值设置为: org.apache.dubbo.remoting.http12.HttpResult 可通过 HttpResult#builder 来构建

抛出Payload异常: throws new org.apache.dubbo.remoting.http12.exception.HttpResultPayloadException(HttpResult) 示例代码:

  [java]
1
2
3
throw new HttpResult.found("https://a.com"). toPayload();

此异常已避免填充错误栈,对性能无太大影响,并且不用考虑返回值逻辑,推荐用这个方式来定制输出

获取 HttpResponse 后自定义,实例代码:

  [java]
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
HttpResponse response = RpcContext.getServiceContext().getRequest(HttpResponse.class); response. sendRedirect("https://a.com"); response. setStatus(404); response. outputStream(). write(data); // 写完输出建议 commit 来避免被其他扩展改写 response. commit();

如果只是增加 http header 推荐用这个方式

自定义JSON解析和输出

支持Jackson、fastjson2、fastjson和gson等多种JSON框架,使用前请确保对应jar依赖已被引入

指定使用的JSON框架

  [plaintext]
1
dubbo.protocol.triple.rest.json-framework=jackson

通过 JsonUtil SPI 定制

可通过实现 SPI org.apache.dubbo.common.json.JsonUtil 方式来自定义JSON处理,具体可以参考 org/apache/dubbo/common/json/impl 已有实现,建议继承现有实现并重写

异常处理

未被处理的异常最终被转换成 ErrorResponse 类编码后输出:

  [java]
1
2
3
4
5
6
7
8
9
10
11
12
@Data public class ErrorResponse { /** * http status code */ private String status; /** * exception message */ private String message; }

注意对于500及以上错误,为避免泄露服务端内部信息,默认只会输出 message “Internal Server Error”,如果需要自定义 message 可创建继承自 org.apache.dubbo.remoting.http12.exception.HttpStatusException 异常并重写 getDisplayMessage 方法。 提供了以下通用方法来定制异常处理:

参考 9.2自定义异常返回结果 使用SPI 自定义全局异常处理 使用 Dubbo的 Filter SPI 来加工转换异常,如果需要访问 Http 上下文,可继承 org.apache.dubbo.rpc.protocol.tri.rest.filter.RestFilterAdapter 使用 SPI org.apache.dubbo.rpc.protocol.tri.rest.filter.RestFilter 来转换异常,使用上更轻量并提供路径匹配配置能力

注意后2项只能拦截 invoke 链中出现的异常,如果在路径匹配阶段出现异常,只有有方法1能处理

Basic使用指南

示例参见:dubbo-samples-triple-rest/dubbo-samples-triple-rest-basic

路径映射

Basic做为开箱即用 Rest 映射,默认会将方法映射到: /{contextPath}/{serviceInterface}/{methodName} ,其中 /{contextPath} 如果协议没有配置会忽略,即为:/{serviceInterface}/{methodName}

映射的自定义通过注解 org.apache.dubbo.remoting.http12.rest.Mapping 来支持,属性说明如下:

请求映射配置说明

配置名 说明 默认行为
value 映射的 URL 路径,可以是一个或多个路径。 空数组([]
path 映射的 URL 路径,与 value 相同,可以是一个或多个路径。 空数组([]
method 支持的 HTTP 方法列表,例如 GETPOST 等。 空数组(支持所有方法)
params 请求必须包含的参数列表。 空数组([]
headers 请求必须包含的头部列表。 空数组([]
consumes 处理请求的内容类型(Content-Type),可以是一个或多个类型(如 application/json)。 空数组(不限制)
produces 生成响应的内容类型(Content-Type),可以是一个或多个类型(如 application/json)。 空数组(不限制)
enabled 是否启用该映射。 true(启用)

属性支持用占位符方式配置:@Mapping("${prefix}/hi")

如果不希望特定服务或方法被 rest 导出,可以通过设置 @Mapping(enabled = false) 解决

入参类型

通用入参见:3.2入参类型

无注解参数

Basic 的无注解参数由类:FallbackArgumentResolver.java 支持,具体处理流程如下:

无注解参数

SpringMVC & JAX-RS 使用指南

SpringMVC 使用指南

示例参见:dubbo-samples-triple-rest/dubbo-samples-triple-rest-springmvc

路径映射

直接参考 SpringMVC 文档 即可,支持绝大多数特性。

注意

  • 无需 @Controller@RestController 注解
  • 除了 @RequestMapping 还支持新的 @HttpExchange

入参类型

通用入参

参见:3.2 入参类型

注解类型参数

参见 3.2.1 通用类型参数

特殊类型参数

| 类型 | 说明 | 激活条件 | |——|——|———-| | org.springframework.web.context.request.WebRequest | WebRequest 对象 | 引入 SpringWeb 依赖 | | org.springframework.web.context.request.NativeWebRequest | NativeWebRequest 对象 | 同上 | | org.springframework.http.HttpEntity | Http 实体 | 同上 | | org.springframework.http.HttpHeaders | Http 头 | 同上 | | org.springframework.util.MultiValueMap | 多值 Map | 同上 |

无注解参数

  • 如果是基本类型(根据 TypeUtils#isSimpleProperty 判断),直接从 Parameter 中获取
  • 如果非基本类型,使用 @ModelAttribute 来绑定复杂 bean 类型参数

参数类型转换

  1. 优先使用 Spring 的 org.springframework.core.convert.ConversionService 来转换参数
    • Spring Boot 应用默认使用 mvcConversionService
    • 否则使用 org.springframework.core.convert.support.DefaultConversionService#getSharedInstance 获取共享 ConversionService
  2. 如果 ConversionService 不支持则会回退到通用类型转换:3.3 参数类型转换

异常处理

除了支持 3.8 异常处理中提到的方式,还支持 Spring @ExceptionHandler 注解方式,Exceptions

注意:通过这种方式仅能处理方法调用时抛出的异常,其他异常无法捕获

CORS 配置

除了支持 8.4 CORS 配置全局配置,还支持 Spring @CrossOrigin 来精细化配置,CORS

自定义 HTTP 输出

支持以下 Spring 自定义方式:

  • 使用 @ResponseStatus 注解
  • 返回 org.springframework.http.ResponseEntity 对象

支持的扩展

org.springframework.web.servlet.HandlerInterceptor
使用方式类似 7.1 使用 Filter 扩展


JAX-RS 使用指南

示例参见:dubbo-samples-triple-rest/dubbo-samples-triple-rest-jaxrs

路径映射

  • Service 需要显式添加注解 @Path
  • 方法需要添加 @GET@POST@HEAD 等请求方法注解
  • 直接参考 Resteasy 文档 即可,支持绝大多数特性

入参类型

通用入参

参见:3.2 入参类型

注解类型参数

| 注解 | 参数位置 | 说明 | |——|———-|——| | @QueryParam | querystring | ?a=a&b=b 对应的参数 | | @HeaderParam | header | | | @PathParam | path | | | @FormParam | form | body 为 key1=value2&key2=value2 格式 | | 无注解 | body | 不显式使用注解 |

特殊类型参数

| 类型 | 说明 | 激活条件 | |——|——|———-| | javax.ws.rs.core.Cookie | Cookie 对象 | 引入 Jax-rs 依赖 | | javax.ws.rs.core.Form | 表单对象 | 同上 | | javax.ws.rs.core.HttpHeaders | Http 头 | 同上 | | javax.ws.rs.core.MediaType | 媒体类型 | 同上 | | javax.ws.rs.core.MultivaluedMap | 多值 Map | 同上 | | javax.ws.rs.core.UriInfo | Uri 信息 | 同上 |

无注解参数

  • 如果是基本类型(根据 TypeUtils#isSimpleProperty 判断),直接从 Parameter 中获取
  • 如果非基本类型,将其视为请求体 (body) 来解码对象

参数类型转换

可通过扩展自定义参数转换,扩展接口:

  [java]
1
2
org.apache.dubbo.rpc.protocol.tri.rest.argument.ArgumentResolver javax.ws.rs.ext.ParamConverterProvider

异常处理

可通过扩展自定义异常处理,扩展接口:

  [java]
1
2
javax.ws.rs.ext.ExceptionMapper org.apache.dubbo.remoting.http12.ExceptionHandler

CORS 配置

支持 8.4 CORS 配置全局配置

自定义 HTTP 输出

支持以下 Jaxrs 自定义方式:

  • 返回 javax.ws.rs.core.Response 对象

支持的扩展

  [java]
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
javax.ws.rs.container.ContainerRequestFilter // 请求过滤器,允许在请求到达资源方法之前对请求进行预处理。 javax.ws.rs.container.ContainerResponseFilter // 响应过滤器,允许在响应离开资源方法之后对响应进行后处理。 javax.ws.rs.ext.ExceptionMapper // 异常映射器,将抛出的异常映射为 HTTP 响应。 javax.ws.rs.ext.ParamConverterProvider // 参数转换器,允许将请求参数转换为资源方法的参数类型。 javax.ws.rs.ext.ReaderInterceptor // 读取拦截器,允许在读取请求实体时进行拦截和处理。 javax.ws.rs.ext.WriterInterceptor // 写入拦截器,允许在写入响应实体时进行拦截和处理。

Servlet 使用指南

兼容性说明

同时支持低版本 javax 和高版本 jakarta servlet API,jakarta API 优先级更高。
只需引入相应 jar 即可使用 HttpServletRequestHttpServletResponse 作为参数。


使用 Filter 扩展

方法1:实现 Filter + RestExtension 接口

  [java]
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
import org.apache.dubbo.rpc.protocol.tri.rest.filter.RestExtension; import javax.servlet.Filter; import javax.servlet.ServletRequest; import javax.servlet.ServletResponse; public class DemoFilter implements Filter, RestExtension { @Override public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) { chain.doFilter(request, response); } @Override public String[] getPatterns() { return new String[]{"/demo/**", "!/demo/one"}; } @Override public int getPriority() { return -200; } }

方法2:实现 Supplier + RestExtension 接口

  [java]
1
2
3
4
5
6
7
8
public class DemoFilter implements Supplier<Filter>, RestExtension { private final Filter filter = new SsoFilter(); @Override public Filter get() { return filter; } }

从 Spring 容器获取 Filter 实例

  [java]
1
2
3
4
5
6
7
8
9
10
11
12
13
public class DemoFilter implements Supplier<Filter>, RestExtension { private final Filter filter; public DemoFilter(FrameworkModel frameworkModel) { SpringExtensionInjector injector = SpringExtensionInjector.get(frameworkModel.defaultApplication()); filter = injector.getInstance(SsoFilter.class, null); } @Override public Filter get() { return filter; } }

HttpSession 支持

实现 SPI:
org.apache.dubbo.rpc.protocol.tri.rest.support.servlet.HttpSessionFactory


当前限制

  1. Filter 包装限制
    • 不支持在 Filter 中 wrap request/response 对象(避免多类型过滤器嵌套导致的复杂性问题)
  2. 功能缺失
    • 不支持 request.getRequestDispatcher

安全配置

当 REST 服务直接暴露在公网时,需配置安全认证机制。

Basic 认证配置

Provider 端配置

  [yaml]
1
2
3
4
5
6
dubbo: provider: auth: true authenticator: basic username: admin password: admin

Consumer 端配置

  [yaml]
1
2
3
4
5
6
dubbo: consumer: auth: true authenticator: basic username: admin password: admin

安全建议

  • 生产环境使用强密码
  • 启用 HTTPS 加密传输

认证扩展方案

  1. 自定义 Authenticator
    实现 SPI:
    org.apache.dubbo.auth.spi.Authenticator
    通过 dubbo.provider.authenticator 配置启用

  2. HTTP 请求过滤
    实现以下任一 SPI:

    • org.apache.dubbo.rpc.HeaderFilter
    • org.apache.dubbo.rpc.protocol.tri.rest.filter.RestFilter

全局参数配置指南

路径匹配配置

配置名 类型 说明 默认值
dubbo.protocol.triple.rest.case-sensitive-match boolean 是否区分路径大小写(如 /users 不匹配 /Users true
dubbo.protocol.triple.rest.trailing-slash-match boolean 是否匹配尾部斜杠(如 /users 也匹配 /users/ true
dubbo.protocol.triple.rest.suffix-pattern-match boolean 是否启用后缀模式匹配(如 /users 也匹配 /users.json true

后缀匹配说明:启用后会自动支持内容协商,如 .json 对应 application/json

CORS 跨域配置

配置名 说明 默认值
dubbo.protocol.triple.rest.cors.allowed-origins 允许的跨域来源(* 表示所有) 未设置(不允许跨域)
dubbo.protocol.triple.rest.cors.allowed-methods 允许的HTTP方法(* 表示所有) 未设置(仅允许GET/HEAD)
dubbo.protocol.triple.rest.cors.allowed-headers 允许的请求头(* 表示所有) 未设置
dubbo.protocol.triple.rest.cors.exposed-headers 暴露给客户端的响应头 未设置
dubbo.protocol.triple.rest.cors.allow-credentials 是否允许用户凭证 未设置(false)
dubbo.protocol.triple.rest.cors.max-age 预检请求缓存时间(秒) 未设置

高级使用指南

支持的扩展点

Servlet/JAX-RS

  [java]
1
2
3
4
javax.servlet.Filter // Servlet过滤器 org.apache.dubbo.rpc.protocol.tri.rest.support.servlet.HttpSessionFactory // HttpSession支持 javax.ws.rs.container.ContainerRequestFilter // JAX-RS请求过滤器 javax.ws.rs.container.ContainerResponseFilter // JAX-RS响应过滤器

Spring MVC

  [java]
1
org.springframework.web.servlet.HandlerInterceptor // Spring拦截器

核心扩展点

  [java]
1
2
3
org.apache.dubbo.remoting.http12.ExceptionHandler // 异常处理 org.apache.dubbo.rpc.HeaderFilter // 头部过滤器 org.apache.dubbo.rpc.protocol.tri.rest.filter.RestFilter // REST过滤器

消息处理

  [java]
1
2
org.apache.dubbo.remoting.http12.message.HttpMessageDecoderFactory // 消息解码 org.apache.dubbo.remoting.http12.message.HttpMessageEncoderFactory // 消息编码

参数处理

  [java]
1
2
org.apache.dubbo.rpc.protocol.tri.rest.argument.ArgumentResolver // 参数解析 org.apache.dubbo.rpc.protocol.tri.rest.argument.ArgumentConverter // 参数转换

自定义异常处理

实现 ExceptionHandler 接口:

  [java]
1
2
3
4
5
6
7
8
9
10
11
public interface ExceptionHandler<E extends Throwable, T> { // 自定义日志级别 default Level resolveLogLevel(E throwable) { return null; // 返回null使用默认级别 } // 异常处理逻辑 default T handle(E throwable, RequestMetadata metadata, MethodDescriptor descriptor) { return null; // 返回非null结果将直接输出 } }

使用场景

  • 返回 HttpResult 可自定义HTTP状态码和响应头
  • 通过泛型指定要处理的异常类型

调试配置

开启DEBUG日志

  [yaml]
1
2
3
4
logging: level: "org.apache.dubbo.rpc.protocol.tri": debug "org.apache.dubbo.remoting": debug

输出详细请求/响应日志

开启VERBOSE模式

  [yaml]
1
2
3
4
dubbo: protocol: triple: verbose: true

输出错误堆栈和更详细的错误日志


最佳实践建议

  1. 生产环境应关闭verbose模式
  2. 跨域配置应明确指定允许的来源和方法
  3. 自定义异常处理时注意区分业务异常和系统异常

参考资料

https://cn.dubbo.apache.org/zh-cn/overview/mannual/java-sdk/reference-manual/protocol/tripe-rest-manual/