Spring-Cloud-Openfeign如何传递用户信息?

用户信息传递

微服务系统中,前端会携带登录生成的token访问后端接口,请求会首先到达网关,网关一般会做token解析,然后把解析出来的用户ID放到http的请求头中继续传递给后端的微服务,微服务中会有拦截器来做用户信息的拦截,把用户信息存放到ThreadLocal供接口使用。

feign远程调用的时候,有时候也需要把消费者端的用户信息传递到服务提供者端,否则可能因为没有用户信息而无法调用接口。feign调用并不会走网关,是由服务消费者直接去调用服务提供者,在消费者端已经有userId的前提下,只需要把userId放到Http的请求头中就可以继续复用微服务中HandlerInterceptor对userId的处理逻辑了。

feign拦截器

feign提供了一个拦截器的接口:

java 复制代码
public interface RequestInterceptor {
  /**
   * Called for every request. Add data using methods on the supplied {@link RequestTemplate}.
   */
  void apply(RequestTemplate template);
}

在这个接口中有一个RequestTemplate 对象,我们就可以使用这个对象来传递请求头。

1)需要定义一个拦截器的实现类

java 复制代码
public class FeignRelayUserInterceptor implements RequestInterceptor {
    @Override
    public void apply(RequestTemplate template) {
        // 从TL中获取用户id
        Long userId = UserContext.getUserId();
        if (userId == null) {
            return;
        }
        // 放入http的请求头中
        template.header("userId", userId.toString());
    }
}

2)在feign的配置类中注入这个拦截器

java 复制代码
public class FeignConfig {
    @Bean
    public FeignRelayUserInterceptor feignRelayUserInterceptor(){
        return new FeignRelayUserInterceptor();
    }
}

3)在feign接口中引用这个配置类

java 复制代码
@FeignClient(value = "user-service", 
	url = "http://localhost:8081", 
	configuration = FeignConfig.class)
public interface UserClient {
    @GetMapping("/user/{id}")
    public User queryById(@PathVariable("id") Long id);
}

测试

在服务提供者端从请求头中获取userId:

java 复制代码
@GetMapping("/{id}")
    public User queryById(@PathVariable("id") Long id,
     @RequestHeader("userId")Long userId) {
        log.info("============>userId:{}", userId);
        return userService.queryById(id);
    }

完整的源码下载:
https://github.com/xjs1919/enumdemo/tree/master/feign-interceptor-demo

相关推荐
prince051 小时前
用户积分系统怎么设计
java·大数据·数据库
96773 小时前
理解IOC控制反转和spring容器,@Autowired的参数的作用
java·sql·spring
SY_FC3 小时前
实现一个父组件引入了子组件,跳转到其他页面,其他页面返回回来重新加载子组件函数
java·前端·javascript
耀耀_很无聊4 小时前
09_Jenkins安装JDK环境
java·运维·jenkins
ノBye~4 小时前
Centos7.6 Docker安装redis(带密码 + 持久化)
java·redis·docker
黑臂麒麟4 小时前
openYuanrong:多语言运行时独立部署以库集成简化 Serverless 架构 & 拓扑感知调度:提升函数运行时性能
java·架构·serverless·openyuanrong
XiaoLeisj4 小时前
Android Jetpack 页面架构实战:从 LiveData、ViewModel 到 DataBinding 的生命周期管理与数据绑定
android·java·架构·android jetpack·livedata·viewmodel·databinding
⑩-4 小时前
为什么要用消息队列?使用场景?
java·rabbitmq
似水明俊德4 小时前
01-C#.Net-泛型-面试题
java·开发语言·面试·c#·.net
Allnadyy4 小时前
【C++项目】从零实现高并发内存池(一):核心原理与设计思路
java·开发语言·jvm