热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

Feign远程调用请求头丢失问题分析与解决方案

本文详细探讨了在微服务架构中,使用Feign进行远程调用时出现的请求头丢失问题,并提供了具体的解决方案。重点讨论了单线程和异步调用两种场景下的处理方法。

Feign远程调用请求头丢失问题分析


在微服务架构中,Feign是一个常用的声明式HTTP客户端,用于简化服务间的调用。然而,在实际开发过程中,经常会遇到Feign远程调用时请求头丢失的问题,尤其是在涉及用户认证和授权的场景中。本文将详细探讨这一问题,并提供解决方案。



单线程中Feign远程调用丢失请求头的情况


在单线程环境中,Feign远程调用可能会导致请求头丢失,从而影响后续的服务调用。具体表现为,当用户在购物车中选好商品并点击结算时,订单服务通过Feign调用购物车服务获取购物车详情,但由于请求头丢失,购物车服务无法识别用户是否已登录,从而无法正确返回购物车信息。



代码案例


订单服务 - Controller

@GetMapping("/toTrade")
public String toTrade(Model model) {
OrderConfirmVo orderCOnfirmVo= orderService.confirmOrder();
model.addAttribute("orderConfirmData", orderConfirmVo);
return "confirm";
}


拦截器

@Component
public class OrderLoginIntercepted implements HandlerInterceptor {
public static ThreadLocal threadLocal = new ThreadLocal<>();

@Override
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
MemberResponseVo attribute = (MemberResponseVo) request.getSession().getAttribute(AuthConstant.LOGIN_USER);
if (attribute != null) {
threadLocal.set(attribute);
return true;
} else {
request.getSession().setAttribute("msg", "请先进行登录");
response.sendRedirect("http://auth.grapesmail.com/login.html");
return false;
}
}
}


ServiceImpl

@Override
public OrderConfirmVo confirmOrder() {
OrderConfirmVo orderCOnfirmVo= new OrderConfirmVo();
MemberResponseVo memberRespOnseVo= OrderLoginIntercepted.threadLocal.get();
List address = memberFeignService.getAddress(memberResponseVo.getId());
orderConfirmVo.setMemberAddressVos(address);

// 远程调用购物车服务查询购物车所有购物项
List currentUserCarItems = cartFeignService.getCurrentUserCarItems();
orderConfirmVo.setItems(currentUserCarItems);

Integer integration = memberResponseVo.getIntegration();
orderConfirmVo.setIntegration(integration);

// TODO 接口幂等性,防重令牌
return orderConfirmVo;
}


购物车服务 - 拦截器部分代码

@Override
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
UserInfoTo userInfoTo = new UserInfoTo();
HttpSession session = request.getSession();
MemberResponseVo user = (MemberResponseVo) session.getAttribute(AuthConstant.LOGIN_USER);
if (user != null) {
userInfoTo.setUserId(user.getId());
}
}


源码解析丢失请求头原因


Feign在进行远程调用时,默认会构造一个新的请求对象,这个新的请求对象不会携带原有的请求头信息。因此,如果依赖于请求头中的某些信息(如Session、COOKIE等),这些信息将会丢失,导致被调用的服务无法正确识别用户身份。



解决方法


为了确保请求头信息在Feign远程调用中不丢失,可以在订单服务中添加一个自定义的RequestInterceptor拦截器,该拦截器会在每次远程调用前将请求头信息复制到新的请求中。



添加RequestInterceptor拦截器

@Configuration
public class MailFeignConfig {
@Bean(name = "requestInterceptor")
public RequestInterceptor requestInterceptor() {
return new RequestInterceptor() {
@Override
public void apply(RequestTemplate requestTemplate) {
ServletRequestAttributes requestAttributes = (ServletRequestAttributes) RequestContextHolder.getRequestAttributes();
HttpServletRequest request = requestAttributes.getRequest();
if (request != null) {
String COOKIE = request.getHeader("COOKIE");
requestTemplate.header("COOKIE", COOKIE);
}
}
};
}
}


异步调用中Feign远程调用丢失请求头的情况


在异步调用中,由于多线程环境的特性,请求头信息更容易丢失。为了解决这一问题,可以通过手动设置请求上下文,确保每个线程都能获取到正确的请求头信息。



代码案例


@Override
public OrderConfirmVo confirmOrder() throws ExecutionException, InterruptedException {
OrderConfirmVo orderCOnfirmVo= new OrderConfirmVo();
MemberResponseVo memberRespOnseVo= OrderLoginIntercepted.threadLocal.get();

RequestAttributes requestAttributes = RequestContextHolder.getRequestAttributes();

CompletableFuture getAddress = CompletableFuture.runAsync(() -> {
RequestContextHolder.setRequestAttributes(requestAttributes);
List address = memberFeignService.getAddress(memberResponseVo.getId());
orderConfirmVo.setMemberAddressVos(address);
}, threadPoolExecutor);

CompletableFuture getCurrentUserCartItems = CompletableFuture.runAsync(() -> {
RequestContextHolder.setRequestAttributes(requestAttributes);
List currentUserCarItems = cartFeignService.getCurrentUserCarItems();
orderConfirmVo.setItems(currentUserCarItems);
}, threadPoolExecutor);

Integer integration = memberResponseVo.getIntegration();
orderConfirmVo.setIntegration(integration);

// 其他数据自动计算
// TODO 接口幂等性,防重令牌
CompletableFuture.allOf(getAddress, getCurrentUserCartItems).get();
return orderConfirmVo;
}


两者区别


单线程和异步调用中Feign远程调用丢失请求头的主要区别在于线程管理。单线程环境下,通过自定义RequestInterceptor即可解决问题;而在异步调用中,需要手动设置请求上下文,确保每个线程都能获取到正确的请求头信息。



详细视频教学:Feign远程调用丢失


推荐阅读
  • Explore a common issue encountered when implementing an OAuth 1.0a API, specifically the inability to encode null objects and how to resolve it. ... [详细]
  • 深入解析Spring Cloud Ribbon负载均衡机制
    本文详细介绍了Spring Cloud中的Ribbon组件如何实现服务调用的负载均衡。通过分析其工作原理、源码结构及配置方式,帮助读者理解Ribbon在分布式系统中的重要作用。 ... [详细]
  • 本文介绍如何在Spring Boot项目中集成Redis,并通过具体案例展示其配置和使用方法。包括添加依赖、配置连接信息、自定义序列化方式以及实现仓储接口。 ... [详细]
  • 本文将介绍如何编写一些有趣的VBScript脚本,这些脚本可以在朋友之间进行无害的恶作剧。通过简单的代码示例,帮助您了解VBScript的基本语法和功能。 ... [详细]
  • 本文详细介绍了如何使用 Yii2 的 GridView 组件在列表页面实现数据的直接编辑功能。通过具体的代码示例和步骤,帮助开发者快速掌握这一实用技巧。 ... [详细]
  • 本文深入探讨 MyBatis 中动态 SQL 的使用方法,包括 if/where、trim 自定义字符串截取规则、choose 分支选择、封装查询和修改条件的 where/set 标签、批量处理的 foreach 标签以及内置参数和 bind 的用法。 ... [详细]
  • 本文详细介绍了Java中org.eclipse.ui.forms.widgets.ExpandableComposite类的addExpansionListener()方法,并提供了多个实际代码示例,帮助开发者更好地理解和使用该方法。这些示例来源于多个知名开源项目,具有很高的参考价值。 ... [详细]
  • 本文详细介绍了如何构建一个高效的UI管理系统,集中处理UI页面的打开、关闭、层级管理和页面跳转等问题。通过UIManager统一管理外部切换逻辑,实现功能逻辑分散化和代码复用,支持多人协作开发。 ... [详细]
  • 深入解析 Apache Shiro 安全框架架构
    本文详细介绍了 Apache Shiro,一个强大且灵活的开源安全框架。Shiro 专注于简化身份验证、授权、会话管理和加密等复杂的安全操作,使开发者能够更轻松地保护应用程序。其核心目标是提供易于使用和理解的API,同时确保高度的安全性和灵活性。 ... [详细]
  • 深入解析 Spring Security 用户认证机制
    本文将详细介绍 Spring Security 中用户登录认证的核心流程,重点分析 AbstractAuthenticationProcessingFilter 和 AuthenticationManager 的工作原理。通过理解这些组件的实现,读者可以更好地掌握 Spring Security 的认证机制。 ... [详细]
  • 本文将详细探讨Linux pinctrl子系统的各个关键数据结构,帮助读者深入了解其内部机制。通过分析这些数据结构及其相互关系,我们将进一步理解pinctrl子系统的工作原理和设计思路。 ... [详细]
  • Kubernetes 持久化存储与数据卷详解
    本文深入探讨 Kubernetes 中持久化存储的使用场景、PV/PVC/StorageClass 的基本操作及其实现原理,旨在帮助读者理解如何高效管理容器化应用的数据持久化需求。 ... [详细]
  • 深入解析SpringMVC核心组件:DispatcherServlet的工作原理
    本文详细探讨了SpringMVC的核心组件——DispatcherServlet的运作机制,旨在帮助有一定Java和Spring基础的开发人员理解HTTP请求是如何被映射到Controller并执行的。文章将解答以下问题:1. HTTP请求如何映射到Controller;2. Controller是如何被执行的。 ... [详细]
  • 深入解析ESFramework中的AgileTcp组件
    本文详细介绍了ESFramework框架中AgileTcp组件的设计与实现。AgileTcp是ESFramework提供的ITcp接口的高效实现,旨在优化TCP通信的性能和结构清晰度。 ... [详细]
  • 本文探讨了在Django项目中,如何在对象详情页面添加前后导航链接,以提升用户体验。文章详细描述了遇到的问题及解决方案。 ... [详细]
author-avatar
msf6688
PHP小白,请大神 们多多关照!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有