0.背景

在实际项目当中,我们采用的是 Abp 框架,但是 Abp 框架官方并没有针对 Grpc 进行模块封装。基于此我结合 Abp 与 MagicOnion 封装了一个 Abp.Grpc 模块,它包括服务端和调用端两部分的包。通过这两个包,你可以很方便地在 Abp 框架当中集成 Grpc 实现服务内部通讯。

但是在实际使用当中会出现一个问题,当 A 服务调用 B 服务的时候,A 服务当前登录用户为 admin,调用 B 服务的 IAbpSession 的值仍然为空,这个时候当 B 服务内部实现使用了 IAbpSession 时会出现问题。

这是因为通过 Grpc 接口调用时,并没有传递诸如 Token 之类的东西,而在 B 服务内部的 IAbpSession 本身附加的数据是从 HttpContext 里面获取的,所以 B 服务当前是没有用户状态的。

1.解决

所幸 IAbpSession 提供了一个 Use 方法,通过这个方法我们可以临时地改变 IAbpSession 内部的值,当 。定义如下:

IDisposable Use(int? tenantId, long? userId);

使用方法如下:

public class TestAppService : ITransientDependency
{
private readonly IAbpSession _abpSession; public TestAppService(IAbpSession abpSession)
{
_abpSession = abpSession;
} public void TestMethod()
{
using(_abpSession.Use(10,20))
{
// 其他操作
} // 出去 using 语句之后会自动释放之前的值
}
}

2.Grpc 接口改造

这里 Abp.Grpc 库使用的是 MagicOnion 库实现 Grpc 接口的,底层序列化使用的是 MessagePack,速度也不比 Protocol Buffer 差。

2.1 服务定义

服务定义接口时,必须附加一个 GrpcSession 参数,这个参数用于调用方传递其 IAbpSession 值所使用。例如我有一个接口方法如下,用于返回服务方接收到的用户 Id 值。

public interface ITestGrpcService : IService<ITestGrpcService>
{
// 普通的 Grpc 接口定义
UnaryResult<int> Sum(int x, int y); // 带有 GrpcSession 的接口定义
UnaryResult<long?> TestGrpcSession(GrpcSession session);
}

2.2 服务提供方

服务提供方在实现 ITestGrpcService 的时候,需要在代码起始点就开始使用 using 语句包裹代码。

public class TestGrpcService : ServiceBase<ITestGrpcService>, ITestGrpcService
{
private readonly IAbpSession _abpSession; public TestGrpcService()
{
_abpSession = IocManager.Instance.Resolve<IIocManager>().Resolve<IAbpSession>();
} public UnaryResult<int> Sum(int x, int y)
{
return UnaryResult(x + y);
} public UnaryResult<long?> TestGrpcSession(GrpcSession session)
{
// 赋值前 Session 的值
Console.WriteLine(_abpSession.UserId); // 临时改变 Session 值
using (_abpSession.Use(session.TenantId, session.UserId))
{
Console.WriteLine(_abpSession.UserId);
} // 离开 using 语句时 Session 的值
Console.WriteLine(_abpSession.UserId); return new UnaryResult<long?>(1000);
}
}

2.3 服务调用方

服务调用方则直接在调用 Grpc 接口的时候,传递给接口当前服务的 Session 状态。

public class TestApplicationService : ApplicationService
{
private readonly IGrpcConnectionUtility _utility; public TestApplicationService(IGrpcConnectionUtility utility)
{
_utility = utility;
} public void TestAction()
{
// 获得指定的 Grpc 服务
var service = _utility.GetRemoteService<ITestGrpcService>("Grpc 服务名称");
// 调用测试方法,传递当前调用方的 Session 值
var userId = service.TestGrpcSession(AbpSession as AbpSessionBase).GetAwaiter().GetResult(); Console.WriteLine("TestGrpcSession 方法结果:" + userId);
}
}

2.4 最后的效果

当客户端调用 GRPC 接口时,会将自身的 Session 状态通过 GrpcSession 传递到服务端,这样服务端就能够共享客户端的绘画状态。

3.Abp.Grpc 项目地址

Abp.Grpc 库地址:https://github.com/GameBelial/Abp.Grpc

4.实现的 DEMO 地址

服务端:https://github.com/GameBelial/Abp.Grpc.Server.Demo

客户端:https://github.com/GameBelial/Abp.Grpc.Client.Demo

Abp + gRpc 如何实现用户会话状态传递的更多相关文章

  1. abp中多种登陆用户的设计

    项目地址:https://gitee.com/bxjg1987/abp 场景 在<学校管理系统>中,学生.家长.教师.教务都可能登陆,做一些属于他们自己的操作.这些用户需要的属性各不相同, ...

  2. Abp Vnext3 vue-admin-template(二用户退出)

    先修改用户退出,中午有点困先改简单的 退出代码在src\layout\components\Navbar.vue代码如下,讲流程这里不需要修改 methods: { toggleSideBar() { ...

  3. Abp Vnext3 vue-admin-template(一用户登录)

    Git地址https://github.com/PanJiaChen/vue-admin-template/blob/master/README-zh.md 官方文档https://panjiache ...

  4. 重识linux-linux主机上的用户信息传递

    1 查询用户 w,who,last,lastlog 1)在线用户查询  w ,who 2)账号最近的登录时间  last lastlog 2 用户对谈 write,mesg,wall 1)write ...

  5. 用户登录拦截器查询到登录用户后如何将用户信息传递到后面的Controller

    taotao创建订单代码中之前忘了加入用户信息,那么加上呢? 分析:用户创建订单的时候,我们会强制要求用户先登录,也就是说,创建订单的Controller执行时,一定是用户已经登录了的,而用户只要登录 ...

  6. Linux学习-Linux 主机上的用户讯息传递

    查询使用者: w, who, last, lastlog 如果你想要知道目前已登入在系统上面的用户呢?可以透过 w 或 who 来查询喔!如下范例所示: [root@study ~]# w 01:49 ...

  7. [Abp 源码分析]十二、多租户体系与权限验证

    0.简介 承接上篇文章我们会在这篇文章详细解说一下 Abp 是如何结合 IPermissionChecker 与 IFeatureChecker 来实现一个完整的多租户系统的权限校验的. 1.多租户的 ...

  8. [Abp 源码分析]十一、权限验证

    0.简介 Abp 本身集成了一套权限验证体系,通过 ASP.NET Core 的过滤器与 Castle 的拦截器进行拦截请求,并进行权限验证.在 Abp 框架内部,权限分为两块,一个是功能(Featu ...

  9. Vue Abp vNext获取当前登录用户

    系统默认提供了获取当前用户的api方法 https://localhost:44364/api/identity/my-profile 手工实现方法:abp后台获取当前用户需要在AppService应 ...

随机推荐

  1. sqlserver 并行度

    转载地址:http://www.cnblogs.com/zhijianliutang/p/4148540.html

  2. 在Java的Condition接口【唤醒全部线程】

    在Java的Condition接口中,存在的几个方法跟Synchronized中的wait(),waitall(),wait(time ^),这个几个方法一一对应起来,但是在Lock.newCondi ...

  3. ----改写superheros的json以及上传到github----

    以下为js代码: var header = document.querySelector('header'); var section = document.querySelector('sectio ...

  4. LinkedList - 好一个双向链表

    LinkedList是常用的集合结构之一,数据存储结构为链式存储,每个节点都有元素.前指针和后指针,指针指向了前节点和后节点的位置.同是LinkedList也是一个队列,实现了Deque接口,Dequ ...

  5. CentOS7安装特定版本的Docker

    查询可用版本 [root@bogon ~]# yum list docker-ce --showduplicates | sort -r 查询结果 * updates: centos.ustc.edu ...

  6. spring boot 注解

    一级注解:(写在类名前面的)@RestController: 等价于在函数前面写@ResponseBody ,会直接返回要显示的内容 @ControllerString返回的是模板文件的名称. 二级注 ...

  7. CodeForces - 939A,解题报告

    题意:给出一个n个点有向图,问是否存在三个点,这三个点构成一个回路.n<=5000 模拟即可. 注意是必须三个点 多了居然不行. import java.util.*; public class ...

  8. java程序员随笔

    之前坚持过一段时间的博客,不过后来因为一些琐事,也因为自己的懒惰,没坚持下来.一晃本科毕业到现在已经快9年了,本科毕业的时候经常想,自己十年之后会是什么样子,那时候筹措满志,心里的每一个答案,都离现在 ...

  9. nlp L1

    前向最大匹配: 最大匹配出的词必须保证下一个扫描不是词表中的词或词的前缀才可以结束. 正向最大匹配算法:从左到右将待分词文本中的几个连续字符与词表匹配,如果匹配上,则切分出一个词.但这里有一个问题:要 ...

  10. media 标签解释

    一:常用标签这句话是自动设置缩放,然而,它并不能完全适应所有的手机,并且你在用浏览器手机模式调试的时候可能正常,但是换到真实的手机端其实是不正常的.所以我们还要进行改动. <meta name= ...