修改zuul过滤器中的请求头

use*_*r09 4 httprequest netflix-zuul spring-cloud-netflix

我有一个后端服务,它接受授权标头来验证访问。我使用 Spring cloud zuul 创建了一个网关服务,并将请求路由到后端服务。

网关服务本身受 OAuth2 保护,并接受授权标头来验证访问。现在,一旦请求授权网关服务,网关服务将充当客户端,使用其 clientid、secret 和后端服务资源 ID 在标头中发送更新的访问令牌。由于它必须发送授权标头,因此我尝试更新请求中的授权标头,但它采用以下两种方式。

ctx = RequestContext.getCurrentContext();
ctx.addZuulRequestHeader("Authorization", accessToken);
Run Code Online (Sandbox Code Playgroud)

使用上面的代码,它添加了授权标头,但将其添加到后端服务无法识别的 zuul 标头中。

我创建了一个包装器来修改请求对象,但它不起作用

public class RequestWrapper extends HttpServletRequestWrapper
{
    private final Map<String, String[]> modifiableParameters;
    private Map<String, String[]> allParameters = null;

    public RequestWrapper(final HttpServletRequest request, 
                                                    final Map<String, String[]> additionalParams)
    {
        super(request);
        modifiableParameters = new TreeMap<String, String[]>();
        modifiableParameters.putAll(additionalParams);
    }

    @Override
    public String getParameter(final String name)
    {
        String[] strings = getParameterMap().get(name);
        if (strings != null)
        {
            return strings[0];
        }
        return super.getParameter(name);
    }

    @Override
    public Map<String, String[]> getParameterMap()
    {
        if (allParameters == null)
        {
            allParameters = new TreeMap<String, String[]>();
            allParameters.putAll(super.getParameterMap());
            allParameters.putAll(modifiableParameters);
        }

        return Collections.unmodifiableMap(allParameters);
    }

    @Override
    public Enumeration<String> getParameterNames()
    {
        return Collections.enumeration(getParameterMap().keySet());
    }

    @Override
    public String[] getParameterValues(final String name)
    {
        return getParameterMap().get(name);
    }
}
Run Code Online (Sandbox Code Playgroud)

上面的包装器是从http://www.ocpsoft.org/opensource/how-to-safely-add-modify-servlet-request-parameter-values/找到的 ,并在zuul过滤器中调用它

    authToken = getAuthToken();     
    String accessToken = "Bearer " + authToken;
    Map<String,String[]> additionalParams = new HashMap<>();

    additionalParams.put("Authorization", new String[] {accessToken});     
   ctx.setRequest(new RequestWrapper(request, additionalParams));
Run Code Online (Sandbox Code Playgroud)

我做错了什么或者有其他修改请求标头的方法吗?

red*_*off 5

如果您询问如何向 Zuul 请求添加标头,您的代码是正确的,但您应该验证过滤器类型(Pre、Route、Post,...)和过滤器的顺序。

检查此线程:重定向时向 Zuul 添加标头

也许这也有帮助:How to selectroute based on header in Zuul

如果需要通过Zuul将授权传递给后端服务,可以查看sensitiveHeadersproperty。就像是 :

zuul.routes.YOURSERVICE.sensitiveHeaders=Cookie,Set-Cookie