同步操作将从 stylefeng/Roses 强制同步,此操作会覆盖自 Fork 仓库以来所做的任何修改,且无法恢复!!!
确定后同步将在后台操作,完成时将刷新页面,请耐心等待。
Roses基于Spring Boot, 是开源项目Guns(https://gitee.com/naan1993/guns)的升级版本,致力做更简洁的分布式和服务化解决方案,Roses提供基于Spring Cloud的分布式框架,整合了springmvc + mybatis-plus + eureka + zuul + feign + ribbon + hystrix等等,提供Roses独有的高效率的开发体验,提供可靠消息最终一致性分布式事务解决方案,提供基于调用链的服务治理,提供可靠的服务异常定位方案(Log + Trace),一个分布式框架不仅需要构建高效稳定的底层开发框架,更需要解决分布式带来的种种挑战。
模块名称 | 说明 | 端口 | 备注 |
---|---|---|---|
roses-api | 服务接口和model | 无 | 封装所有服务的接口,model,枚举等 |
roses-core | 项目骨架 | 无 | 封装框架所需的基础配置,工具类和运行机制等 |
roses-scanner | 资源扫描器 | 无 | 接口资源无须录入,启动即可录入系统 |
roses-register | 注册中心 | 8761 | eureka注册中心 |
roses-gateway | 网关 | 8000 | 转发,资源权限校验,请求号生成等 |
roses-auth | 鉴权服务 | 8001 | 提供用户,资源,权限等接口 |
roses-config | 配置中心服务器 | 8002 | 配置集中管理 |
roses-logger | 日志服务 | 8003 | 消费并存储日志 |
roses-monitor | 监控中心 | 9000 | 监控服务运行状况 |
roses-message-service | 消息服务 | 9001 | 可靠消息最终一致性(柔性事务解决方案) |
roses-message-checker | 消息恢复和消息状态确认子系统 | 9002 | 可靠消息最终一致性(柔性事务解决方案) |
roses-example-order | 订单服务 | 9003 | 演示如何解决分布式事务 |
roses-example-account | 账户服务 | 9004 | 演示如何解决分布式事务 |
- JDK 1.8
- Redis最新版
- ActiveMQ 5.9.1(可以根据
MessageSender
接口自行拓展为自己公司需要的队列)- Mysql 5.7 +,项目运行前请初始化数据库,脚本在各自项目的sql文件夹
- 启动Redis,启动ActiveMQ,初始化好myslq数据库
- 启动配置中心roses-config
- 启动注册中心roses-register
- 启动监控中心roses-monitor
- 启动鉴权服务roses-auth
- 启动网关roses-gateway
- 剩下的消息服务message-service和消息恢复服务message-checker根据使用情况启动
微服务不一定适用于所有的系统构建,需要根据各自公司业务情况来评估。以下列举一些微服务的优缺点,仅供参考。
Roses中,所有业务请求经过网关,网关做统一的鉴权,权限过滤,数据签名校验,这样做的好处就是业务系统中不用集成鉴权模块,只需要关心各自服务的业务编写。鉴权方面Roses依旧使用jwt,如果想对某个用户作某些资源访问的限制,需要开启PathMatchFilter
,该过滤器会对用户当前访问的资源进行权限校验,资源的收集方式也做了大大的简化,Roses中通过资源搜集器roses-scanner
收集资源(详见特点6),所有的资源只需要用@ApiResource
注解标识,项目运行后会把所有资源发送到roses-auth
服务中,对比以往的每新增一个资源都需要在后台管理系统中新增一条记录,省了不少工作量。
Roses中,所有业务请求经过网关,网关做统一的鉴权,权限过滤,数据签名校验,并为每个请求生成该次请求的唯一请求号。唯一请求号的作用如下:请求在业务流转过程中可能经过多个微服务,查看这次请求的info日志信息,或者error日志信息等,需要从多个微服务的日志记录里去查找,效率非常低,那么,有了唯一请求号标识之后,可以用唯一请求号把请求经过的所有业务流转串起来,并存储起来,当请求遇到问题后,可以通过唯一请求号快速把这次请求的所有日志搜集并展示起来,从而方便排查问题。
Roses中,请求号在中转过程中填充到请求的Request-Header中,与之对应,响应时也会在Response-Hedaer中把本次的请求号输出。过滤器中的写法如下:
RequestContext currentContext = RequestContext.getCurrentContext();
HttpServletResponse response = currentContext.getResponse();
String requestNo = IdWorker.getIdStr();
currentContext.addZuulRequestHeader(RosesConstants.REQUEST_NO_HEADER_NAME, requestNo);
response.addHeader(RosesConstants.REQUEST_NO_HEADER_NAME, requestNo);
为了让Feign调用中,自动填充网关生成的唯一号,Roses增加了RosesFeignHeaderProcessInterceptor拦截器,实现如下:
public class RosesFeignHeaderProcessInterceptor implements RequestInterceptor {
@Override
public void apply(RequestTemplate requestTemplate) {
//当前feign远程调用环境不是由http接口发起,例如test单元测试中的feign调用或者项目启动后的feign调用
HttpServletRequest request = null;
try {
request = HttpContext.getRequest();
} catch (NullPointerException e) {
//被调环境中不存在request对象,则不往header里添加当前请求环境的header
return;
}
if (request != null) {
Enumeration<String> headerNames = request.getHeaderNames();
if (headerNames != null) {
while (headerNames.hasMoreElements()) {
String name = headerNames.nextElement();
String values = request.getHeader(name);
requestTemplate.header(name, values);
}
}
}
}
}
首先,分布式事务在不同业务场景下,解决方案是不一样的,时效性要求较高的场景下,例如订单支付成功后,更改订单状态,给用户账户加款,给积分账户加积分,三个操作在三个不同的服务下,这个时候可以用TCC方式解决事务问题;在时效性要求较为不严格下,例如订单支付成功后,需要异步录入会计凭证(不严格要求时效性),这个时候可以用可靠消息最终一致性解决。
Roses中实现了可靠消息最终一致性的解决方案(如上所说第二个例子),TCC方案目前未集成到系统。
单纯依靠消息队列无法实现消息的可靠投递和消费,所以借助预发送待确认消息,业务执行成功后再发送确认消息来保证消息的可靠投递,并且通过中间服务(消息服务)来控制统一的消息预存储和确认发送,统一执行发送到消息队列的操作,并且消息服务有单独的消息表来记录消息是否已经投递成功。
Roses中roses-message-service为消息服务,为可靠消息最终一致性实现的核心,roses-message-checker为定时任务执行器,每隔一定时间来轮训消息表中是否有消息不一致的数据,若消息不一致则从业务系统中调用接口来查询具体业务的执行状态,从而来更新消息表中的消息。
roses-example-order和roses-example-account两个模块,模拟了分布式事务的场景,首先通过order模块下一个单(/place接口),再执行完成此订单(/finish接口),在完成订单过程中,先调用了预发送消息接口(preSaveMessage),之后执行完业务后调用确认并发送消息(confirmAndSendMessage)。在account模块中有消息的监听器,监听到消息后存储账号交易流水记录(recordFlow)。account和order模块为了演示业务流程用,数据库设计比较简单,不合理处请见谅。
//创建预发送消息
ReliableMessage reliableMessage = createMessage(order);
//预发送消息
messageServiceConsumer.preSaveMessage(reliableMessage);
//更新订单为成功状态(百分之50几率失败,模拟错误数据)(此处错误已添加到消息表的数据会被roses-message-checker轮询时删除掉)
updateToSuccess(order);
//确认消息
messageServiceConsumer.confirmAndSendMessage(reliableMessage.getMessageId());
消息的投递有重试机制,所以在消息的消费端需要加上幂等性校验,使得多次消费消息也可以让业务实际只执行一次,在account模块中幂等性的判断通过订单号来标识操作的唯一性。
//幂等判断
EntityWrapper<FlowRecord> wrapper = new EntityWrapper<>();
wrapper.eq("order_id", goodsFlowParam.getId());
List<FlowRecord> flowRecords = this.selectList(wrapper);
if (flowRecords != null && !flowRecords.isEmpty()) {
return;
}
在Roses中,正如您所见,所有的模块都是在一个大工程下,但是实际日常开发中,模块往往分隔在多个项目中,每个项目有单独的小组来维护,小组与小组之间甚至代码都不是可见。当项目中的一些通用配置变动时,例如数据库地址,账号密码等,要么你通知各个小组修改他们的配置,要么你自己打开所有项目修改一遍。如果使用了分布式配置中心,可以把所有项目的配置收集起来,集中配置,那么你只需要打开配置中心的git仓库来修改配置,从而简化配置的维护工作。
除此之外,如果spring boot应用开启了actuator,配置仓库中的配置更改后,应用还可以通过/refresh动态刷新项目的配置,这在网关动态增加路由等配置上非常方便。
当然,开启分布式配置之后也有不完美的地方,在本地开发调试中,常常需要修改配置,直接修改远程仓库的配置又会影响到别人的本地开发,关于这种情况,Roses在每个项目的src/test/resources文件夹中,都保留了一个application-local.yml
,这个配置文件是local环境下的应用的配置文件,您可以把它拷贝到src/main/resources中,并配置bootstrap.yml
中的spring.cloud.config.enable
配置为false,即可关闭配置中心的配置,使用本地的配置。
为了方便日常接口开发,Roses对控制器层的请求参数和响应进行了统一封装。所有post方式的请求,并且带有json请求body的都可以用RequestData类来作为参数接收请求数据,所有的响应都可以用ResponseBody来作响应的结果。
RequestData类中封装了对请求参数获取的常用方法,例如getString(),getInteger(),parse()等等,可以很方便的获取请求中包含的字符串数据,整型数据,以及解析请求为某个类。而ResponseData类中包含了对常用成功或者失败响应的封装,可以通过静态方法ResponseData.success()或者ResponseData.error()来响应成功的结果或者失败的响应。
例如,请求的数据是一段json:
{name:"order001",count:20}
控制器中可以用如下写法获取参数,并响应成功的返回结果:
/**
* 测试RequestData
*/
@RequestMapping("/test")
public Object test(RequestData requestData) {
String name = requestData.getString("name");
System.out.println(name);
Integer count = requestData.getInteger("count");
System.out.println(count);
MyOrder order = requestData.parse(MyOrder.class);
System.out.println(order);
return ResponseData.success(order);
}
并不是所有请求和响应都必须用RequestData和ResponseData来接收和响应,这只是Roses提供的一种拓展功能,一种便利,可以选择使用或否。
所有接口的参数和响应都封装在两个对象里,开发时候不用构建许多的vo(view object)和qo(query object),但是也有不利的地方,由于不知道请求和响应包含哪些参数,维护时候可能会增加难度,所以,在使用这种机制的时候,配合类似于RAP这样的接口管理工具会更加的好用。
Roses中设立了@ApiResource注解,用来标注控制器里的接口,在Roses框架中,用户对应了角色,角色又对应着资源,菜单关联资源。资源扫描器的作用一方面是便于管理所有的接口资源的权限控制,另一方面是简化了资源录入系统的方式 (不用手写),当程序启动时,会自动扫描带有@ApiResource注解的方法,扫描之后会对资源进行包装,写入到数据库。所以当使用了资源扫描器之后可以很方便的搜集所有服务上面的接口资源,并通过roses-auth的接口,统一的汇报到roses-auth服务上面去,从而实现了资源的集中管理。
Roses继承了Guns框架的业务编写方式,在适当的的业务错误场景,如果不能再继续执行下去业务,可以抛出ServiceException,让DefualtExceptionHandler拦截到异常,直接返回给前端业务上的错误提示信息。那么,在分布式的场景下,如何利用ServiceException的抛出来返回前端提示呢,当A服务调用B服务,B服务又调用C服务过程中,如何把异常信息逐级返回给上个服务呢,Roses中利ErrorDecoder并覆盖其中的decode方法,当response中的信息为Roses中的错误编码格式的话,会在Feign错误解析过程中直接抛出ServiceException,从而直接在调用方服务中抛出被调用方同样的服务异常,实现逐级响应Service服务异常的功能,如果担心服务异常带来的性能问题,可以通过 override 掉异常类的 fillInStackTrace() 方法为空方法,使其不拷贝栈信息。
为了方便业务异常以及分布式调用链中调用异常排查,Roses编写了LogUtil和TraceUtil两个类来记录业务中的调试,提示,错误日志和调用链调用过程中的信息日志,LogUtil中包含LogUtil.info(),LogUtil.debug(),LogUtil.error()等静态方法,TraceUtil中包含TraceUtil.trace()等静态方法,这两个类都采用线程池异步记录日志的方式来记录,目前日志是通过Redis的List放到队列在通过roses-logger模块监听队列来消费记录日志,当然,Roses提供了拓展,如果这种记录方式或是存储介质不符合您的业务需求,您可以通过继承com.stylefeng.roses.core.log.LogProducerService接口,实现您自己的日志记录方式,而不必修改LogUtil中的日志记录方法。总之,Log + Trace的日志记录方法,为多服务异常排查,和调用链服务治理提供了很好的保障。
Roses采用logback来记录日志,并且每个模块都有统一的规范的日志记录格式,具体可见每个模块下的logback-spring.xml
配置文件,在这个配置中,定义了两种profile。
第一种profile是在spring.profiles.active
为local时激活,也就是,在本地开发中,日志只打印到控制台中,不会输出到文件中,默认日志输出级别是info,但是com.stylefeng.roses
包中的类的日志,以debug级别输出,为了方便开发人员本地调试。
<springProfile name="local">
<appender name="STDOUT" class="ch.qos.logback.core.ConsoleAppender">
<encoder>
<pattern>===%d{yyyy-MM-dd HH:mm:ss.SSS} %-5level %logger Line:%-3L - %msg%n</pattern>
<charset>utf-8</charset>
</encoder>
</appender>
<root level="info">
<appender-ref ref="STDOUT"/>
</root>
<logger name="com.stylefeng.roses" level="debug" additivity="false">
<appender-ref ref="STDOUT"/>
</logger>
</springProfile>
第二种profile是在spring.profiles.active
不是local时激活,也就是,不管在正式环境或者测试环境的linux服务器中,都不会在控制台打印logback记录的日志,都会把日志输出到文件中,在这种profile下,记录的日志文件分为两类,第一类日志文件是只记录ERROR级别的日志,可以定期查看这个文件的错误日志,排查服务问题,第二类则是记录所有级别的日志。在两类日志记录器中,日志的切割都是以日期和文件大小(默认2M)切分。
<springProfile name="!local">
<appender name="FILE_ERROR" class="ch.qos.logback.core.rolling.RollingFileAppender">
...
</appender>
<appender name="FILE_ALL" class="ch.qos.logback.core.rolling.RollingFileAppender">
...
</appender>
<root level="info">
<appender-ref ref="FILE_ERROR"/>
<appender-ref ref="FILE_ALL"/>
</root>
</springProfile>
在roses-core模块的com.stylefeng.roses.core.config
包下整合了大量开发常用到的配置,其中包含默认异常拦截,登陆用户的上下文获取,默认缓存配置,默认fastjson的配置,默认mybatis-plus的配置,默认的swagger的配置,默认的web配置等等等等,使得在新业务开发中,只要pom引入roses-core这个模块,即可很方便的注入这些特性,直接上手开编写业务,大大减少了新业务,新模块的配置,调试,各种框架集成拼接的时间,因为这些在Roses中已经为您提供好了,利用Spring Boot的自动配置机制,同样的,这些配置在项目启动的时候会默认加载,因为在roses-core模块下的META-INF/spring.factories中配有这些类,当然,如果您不需要某些特性(自动配置类)您可以在@SpringBootApplication注解上增加exclude参数来排除这些自动配置。
有了roses-core模块,开发别的模块时,您可以把百分之90的精力花在编写业务上,百分之10的精力花在搭建项目和配置项目上。
更多有趣,实用,高效的功能尽在Roses
此处可能存在不合适展示的内容,页面不予展示。您可通过相关编辑功能自查并修改。
如您确认内容无涉及 不当用语 / 纯广告导流 / 暴力 / 低俗色情 / 侵权 / 盗版 / 虚假 / 无价值内容或违法国家有关法律法规的内容,可点击提交进行申诉,我们将尽快为您处理。