Skip to content
/ Ymir Public

Rpc framework based on Netty.基于Netty实现的RPC框架,具有服务发现与注册,泛化调用,负载均衡,提供者直连,隐式传参,异步调用,调用过滤,心跳检测等功能。已集成Nacos,Zookeeper注册中心,并保留扩展接口。

License

Notifications You must be signed in to change notification settings

Hegemol/Ymir

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

Ymir

背景

Ymir,出自动漫进击的巨人 中的角色,始祖巨人尤米尔。因为很喜欢这个动漫,所以以这个命名了这个项目。

Ymir是什么

一个Rpc框架的示例项目,大部分思路都来源于Dubbo的设计思想,主要的功能是基于Netty的服务之间的调用。大部分思想都是借鉴于Dubbo的一些设计理念,例如服务上传以及注册,包括SPI加载类等等。

Ymir的基础架构

  • 保持了和Dubbo一致的架构设计,区别在于去掉了Monitor的概念,关于服务监控,后面有时间会进行补充;
    • Provider启动,向Registry上传服务注册信息;
    • Consumer启动,先订阅所有需要的Provider信息,拉取所有的Provider信息落到本地缓存,之后注册Consumer节点;
    • Registry通知Provider,已经有Consumer接入;
    • Consumer向Provider发起请求;

Ymir有哪些功能

  • 启动时检查
  • 负载均衡
  • 泛化调用
  • SPI
  • 超时检测
  • 服务提供者直连
  • 服务分组(doing)
  • 多版本(doing)
  • 注册事件通知
  • 隐式传参
  • 心跳检测
  • 异步调用
  • 调用过滤

如何使用

添加依赖

  • 克隆代码到本地仓库 git clone https://github.com/KevinClair/Ymir.git
  • 打包到本地Maven仓库 mvn clean install -Dmaven.test.skip=true
  • 在需要使用的项目中添加Maven依赖
<dependency>
  <groupId>org.hegemol</groupId>
  <artifactId>ymir</artifactId>
  <version>1.0.0-SNAPSHOT</version>
</dependency>

配置信息

配置文件信息

  • Ymir共支持以下几种类型的配置信息
/**
 * 服务端口,不填默认为20777
 */
private Integer port = 20777;

/**
 * 服务序列化协议, 不写默认为protostuff
 */
private String serial = "protostuff";
  • Ymir对注册中心的支持;
    • 目前已支持Zookeeper和Nacos;
    • 预留接口ServiceDiscovery以及ServiceRegister;
    • 新增加的注册中心只需要实现上面两个接口就可以无缝对接;
ymir:
  register:
    // 目前可选注册类型zookeeper和nacos
    type: zookeeper 
    // 注册中心地址,集群用`,`分隔
    url: localhost:2181
    // 客户端连接参数,一般为连接超时时间等
    props:
      connectionTimeout: 6000

注解支持

  • @Service
public @interface Service {

    /**
     * 权重
     */
    int weight() default 0;

    /**
     * 是否需要注册
     */
    boolean register() default true;

    /**
     * 分组
     */
    String group() default "";

    /**
     * 版本
     */
    String version() default "";
}
  • @Reference
public @interface Reference {

    /**
     * 启动时检查
     */
    boolean check() default false;

    /**
     * 负载均衡
     */
    String loadBalance() default "random";

    /**
     * 超时时间
     */
    int timeout() default 3000;

    /**
     * 重试次数
     */
    int retries() default 2;

    /**
     * 服务直连url
     */
    String url() default "";

    /**
     * 过滤器
     */
    String filter() default "";
  
    /**
     * 是否开启异步调用
     */
    boolean async() default false;
}

开始使用

服务端

  • 编写接口
public interface TestService {
    String test(String name);
}
  • 在需要暴露的服务上添加注解
@Service
public class TestServiceImpl implements TestService {
    @Override
    public String test(String name) {
        return "Hello "+name;
    }
}

客户端

  • 添加接口对应的Maven依赖
<dependency>
  <groupId>org.hegemolorg.hegemol</groupId>
  <artifactId>ymir-example-common</artifactId>
  <version>1.0.0-SNAPSHOT</version>
</dependency>
  • 服务引用
@RestController
public class TestController {

    @Reference
    private TestService service;

    @PostMapping("/name")
    public String get(@RequestParam("name") String name) {
      return service.test(name);
    }
}
  • 发送请求,返回结果
curl --location --request POST 'http://localhost:port/name?name=11'

Ymir的一些设计理念

ymir协议

详情

  • ymir基于本身的数据传递规则,设计了自己的消息协议,具体规则为
    • magic code(魔法值),占用4个字节;
    • full length(body长度),代表整个消息体的长度数据;
    • type,代表本次的消息类型,具体请查看枚举MessageTypeEnum
    • serial,代表本次的消息序列化类型,具体请查看枚举SerializationTypeEnum
    • requestId,代表本次请求的请求id,由客户端生成;
    • 请求body;
 *   0     1     2     3     4     5     6     7     8     9     10     11    12    13    14
 *   +-----+-----+-----+-----+----—+-----+-----+-----+-----+------+-----+-----+-----+-----+
 *   |   magic   code        |      full length      | type|serial|       requestId       |
 *   +-----------------------+-----------------------+-----+------+-----------------------+
 *   |                                                                                    |
 *   |                                       body                                         |
 *   |                                                                                    |
 *   |                                                                                    |
 *   +------------------------------------------------------------------------------------+
 * 4B  magic code(魔法数)   4B requestId(请求的Id)    1B type(消息类型)
 * 1B serial(序列化类型)    4B  full length(消息长度)
 * body(object类型数据)

泛化调用

如何使用

  • Ymir的泛化调用允许客户端不依赖服务端的依赖就可以调用服务。在需要使用的地方添加GenericService 的引入即可;
@RestController
public class TestController {

  @Reference
  private GenericService service;

  @PostMapping("/name")
    public String get(@RequestParam("name") String name){
        return service.invoke("org.hegemol.ymir.example.client.controller.TestInterface", "test", new String[]{"java.lang.String"}, new Object[]{name});
    }
}
  • 在invoke的方法中填入参数就可以通过泛化调用请求服务;

隐式传参

如何使用

  • Ymir的服务允许provider和consumer通过RpcContext进行跨端之间的参数传递;
  • consumer在使用时,只需要通过RpcContext进行参数设置即可;
import org.hegemol.ymir.core.context.RpcContext;

@RestController
public class TestController {

  @Reference
  private TestService service;

  @PostMapping("/name")
  public String get(@RequestParam("name") String name) {
    RpcContext.getContext().setAttachments("testKey", "testValue");
    return service.test(name);
  }
}
  • provider就可以通过RpcContext进行获取传递的参数;
import org.hegemol.ymir.core.context.RpcContext;

@Service
public class TestServiceImpl implements TestService {
  @Override
  public String test(String name) {
    RpcContext.getContext().getAttachments().get("testKey");
    return "Hello " + name;
  }
}

异步调用

如何使用

  • Ymir允许客户端异步调用服务端的接口,有以下两种方式开启异步调用;
    • Reference 注解中将async设置为true即可;
    • 在方法调用前通过隐式传参RpcContext.getContext().setAttachments("async","true")即可;
  • 设置了异步调用后,客户端调用接口的返回会返回Null,客户端需要通过RpcContext.getFuture()返回的CompletableFuture对象完成后续的操作,其中CompletableFuture中的值就是接口定义的返回值;
import org.hegemol.ymir.core.context.RpcContext;

@RestController
public class TestController {

  @Reference(async = true)
  private TestService service;

  @PostMapping("/name")
  public String get(@RequestParam("name") String name) {
    // 此时这个接口会返回Null
    String response = service.test(name);
    CompletableFuture<Object> future = RpcContext.getFuture();
    future.whenComplete(object -> {
      System.out.println("接口调用");
    });
    return response;
  }
}

SPI

Java SPI

  • Java的SPI允许我们在对应的位置添加实现,就可以通过ServiceLoader来加载对应的接口实现,但是缺点在于会一次性加载所有的扩展点,例如:
  • 新增一个接口
public interface SpiInterface {
    String hello();
}
  • 增加两个实现
public class SpiInterfaceImplOne implements SpiInterface{
    @Override
    public String hello() {
        return "哈哈哈";
    }
}

public class SpiInterfaceImplSecond implements SpiInterface{
  @Override
  public String hello() {
    return "嘿嘿嘿";
  }
}
  • 在resources目录下的META-INF/services/新建文件名org.hegemol.ymir.spi.SpiInterface,填充值
org.hegemol.ymir.spi.SpiInterfaceImplOne
org.hegemol.ymir.spi.SpiInterfaceImplSecond
  • 运行测试用例
public class SpiTest {
  @Test
  public void testJavaSpi() {
    Iterator<SpiInterface> iterator = ServiceLoader.load(SpiInterface.class).iterator();
    while (iterator.hasNext()) {
      System.out.println(iterator.next().hello());
    }
  }
}
  • 查看返回结果
哈哈哈
嘿嘿嘿
  • 可以看到通过ServiceLoader去加载实现时,一次性加载了它的所有实现,而Ymir想要的是在需要的时候去加载对应的实现,所以通过Java SPI是不满足的。

Ymir SPI

  • Dubbo SPI ,在需要的时候去加载对应的实现,避免资源浪费
  • Dubbo SPI 在原有的Java SPI基础上扩展了很多,包括自适应加载等非常强大的功能
  • Ymir借鉴了Dubbo SPI以及Shenyu SPI 的设计,基本做到了在需要的时候去加载对应实现的功能,通过ExtensionLoader来加载实现
  • 新增一个接口
// @SPI注解来标注这是一个可以被SPI管理的类,value代表它的默认实现
@SPI("one")
public interface SpiInterface {
    String hello();
}
  • 增加两个实现
public class SpiInterfaceImplOne implements SpiInterface{
    @Override
    public String hello() {
        return "哈哈哈";
    }
}

public class SpiInterfaceImplSecond implements SpiInterface{
  @Override
  public String hello() {
    return "嘿嘿嘿";
  }
}
  • 在resources目录下的META-INF/ymir/新建文件名org.hegemol.ymir.spi.SpiInterface,文件名为接口的全路径名;

这里的内容为key=value的样式,其中key为对应的扩展实现,通过ExtensionLoader进行加载时需要用到

one=org.hegemol.ymir.spi.SpiInterfaceImplOne
two=org.hegemol.ymir.spi.SpiInterfaceImplSecond
  • 运行测试用例
public class SpiTest {
    @Test
    public void testYmirSpi() {
        // 当getLoader里为空时,会读取@SPI注解内的默认值的实现
        SpiInterface loader = ExtensionLoader.getExtensionLoader(SpiInterface.class).getLoader("");
        System.out.println(loader.hello());

        // 加载key为one的实现类
        SpiInterface loaderOne = ExtensionLoader.getExtensionLoader(SpiInterface.class).getLoader("one");
        System.out.println(loaderOne.hello());

        // 加载key为two的实现类
        SpiInterface loaderTwo = ExtensionLoader.getExtensionLoader(SpiInterface.class).getLoader("two");
        System.out.println(loaderTwo.hello());
    }
}
  • 查看返回结果
哈哈哈
哈哈哈
嘿嘿嘿
  • 在Ymir的很多地方都用到了SPI的,例如负载均衡和序列化

服务注册

  • 服务注册信息会落到本地缓存内,防止注册中心挂掉后,服务端无法处理请求;

服务发现

  • 服务发现主要是在客户端操作,客户端来发现自己需要注入的服务信息,之后存储在本地,用来后续的服务调用发起

Netty请求处理器

心跳检测处理器

  • 基于Netty的IdleStateHandler
    • 客户端监听写时间,超时时间为30s;
    • 服务端监听读事件,超时时间为2min;
  • 具体做法
    • 客户端监听写事件,如果在30s内,客户端没有写事件发生,触发IdleStateEvent
      • 第一次发送心跳请求,客户端心跳请求次数+1,服务端收到心跳请求,做出响应,客户端收到心跳响应,心跳请求次数重新置为0,本次心跳结束,等待下一次心跳;
      • 第一次发送心跳请求,客户端心跳请求次数+1,服务端收到心跳请求,但是未响应心跳结果。客户端未收到心跳响应,等待30s后,继续发送心跳请求,如果心跳请求超过3次后,仍未获取到服务端响应心跳结果,客户端主动关闭通道,断开连接,清除缓存;
    • 服务端监听读事件,如果在2min内,服务端没有读事件,触发IdleStateEvent
      • 服务端监听到读事件,删除当前客户端连接地址缓存,服务端关闭连接;

编码/解码处理器

序列化

Gson序列化

  • 使用GsonUtils工具类来对请求参数以及返回参数进行序列化,反序列化操作
  • 需要在配置文件中设置序列化方式为Gson序列化

Protostuff序列化

  • 采用Protostuff 的序列化方式,是Ymir默认的序列化方式

Kryo序列化

  • 采用Kryo 的序列化方式

序列化性能对比

负载均衡

随机

  • 随机获取存在的服务列表中的某一个Service,计算时加权重

轮询

  • 轮询请求服务列表中的Service

有问题反馈

在使用中有任何问题,欢迎反馈给我,可以用以下联系方式跟我交流

致谢

感谢以下的项目,在个人学习过程中,给我起到了很大的帮助

最近计划

  • @Service属性扩展,增加多版本,分组等属性;
  • @Reference属性扩展,多版本,分组等;
  • 负载均衡算法优化;
  • 增加provider和consumer的线程池配置参数;

About

Rpc framework based on Netty.基于Netty实现的RPC框架,具有服务发现与注册,泛化调用,负载均衡,提供者直连,隐式传参,异步调用,调用过滤,心跳检测等功能。已集成Nacos,Zookeeper注册中心,并保留扩展接口。

Topics

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Languages