如何通过当前请求的 Feign 拦截器将动态 header 值添加到 feign-client?

How to add dynamic header values to feign-client through Feign interceptor from current request?

我是 Spring-cloud-openfeign 的新手。我正在尝试构建一个基于 micro-service 的系统。为了使事情看起来更简单,只需要其中的 3 个服务,它们是 Gateway、Service-A、Service-B。当我通过网关从前端向 Service-A 发出请求时,网关将验证请求中的 JWT 令牌,并从令牌中提取用户信息 (userId) 并将请求作为 header 并将其转发给 Service-A。现在 Service-A 将通过 feign-client 调用 service-B。现在,在通过 feign-client 的 inter-service 调用期间,我试图通过 Feign RequestIterceptor 将 userId 从 Serice-A 中的当前请求转发到 service-B 的传出请求。但是我无法在拦截器中检索当前请求。我尝试了这个 , but it doesn't seem to work. I think I face the same problem as of this github_issue 中的解决方案。我可以看到一些博客和文章建议使用 RequestContextListner 或 RequestContextFilter 从 Dispatcher-Servlet 中获取当前请求,但我找不到如何使用或实现它。以下是我目前使用的代码

@Component
public class ProxyInterceptor implements RequestInterceptor {
        
        
        private final static Logger LOGGER = LoggerFactory.getLogger(ProxyInterceptor.class);
        
        @Override
        public void apply(RequestTemplate requestTemplate) {
                
                HttpServletRequest request = ((ServletRequestAttributes) RequestContextHolder.getRequestAttributes())
                        .getRequest();
                
                String userId = request.getHeader("userId");
                LOGGER.info("userId {}", userId);
                
                if (Optional.ofNullable(userId).isPresent()) {
                        requestTemplate.header(USER_ID_REQUEST_HEADER_VARIABLE, userId);
                }
        }
}

依赖关系

<dependency>
            <groupId>org.springframework.cloud</groupId>
            <artifactId>spring-cloud-starter-openfeign</artifactId>
</dependency>
<dependency>
            <groupId>org.springframework.cloud</groupId>                     
            <artifactId>spring-cloud-starter-circuitbreaker-resilience4j</artifactId>
</dependency>

<spring-cloud.version>Hoxton.SR8</spring-cloud.version>

此代码抛出 NullPointer 异常作为 RequestContextHolder.getRequestAttributes() return null。知道如何解决这个问题吗?

我有类似的用例,我需要在拦截器中获取 header。代码类似于下面。

circuitBreakerFactory
                    .create("createUserProfile").run(
                            () -> {
                                    String resp = callerClient.callExternalService();
                                    return resp;
                                    
                            }, exception -> {
                                    logger.info(exception.getMessage());
                            }
                    );

问题是 运行 方法中的第一个参数是 lambda 函数,因此会创建一个新线程。这个新线程没有处理请求的前一个线程的请求上下文信息。

将下面的bean添加到配置中,然后新线程将继承请求上下文。因此 RequestContextHolder.getRequestAttributes() 不会 return null.

   @Bean
   DispatcherServlet dispatcherServlet() {
       DispatcherServlet servlet = new DispatcherServlet();
       servlet.setThreadContextInheritable(true);
       return servlet;
   }

可以参考this回答,对我帮助很大