背景:项目重构,生产线上url需要按照通过网关转发给不同的service,之前用过自定义路由,在yml文件配置route的方式去做转发时,遇到一个问题:
zuul:
routes:
<自定义一个serviceid>:
path = /account/**
serviceId=account
上面的方式不能保证请求的URL在/account/后面的url路径与account服务里面的路径一致,从而这样会有一个问题。原系统中调用方中的C接口的URL不变,而新的服务提供方没有适配的URL,只有C1接口。此处提供一种解决方案:Filter转发。
1. 网关代码
@Component
public class CommonServicePathFilter extends ZuulFilter {
private final static String GETWAY_FOWARD_PREFIX = "getway_forward_";
private final static String GETWAY_COMPAY_CONFIG_KEY = "getway_company";
@Autowired
private RedisTemplate redisTemplate;
@Override
public String filterType() {
//这里重要,必须是route。过滤类型有四种:pre/route/post/error
return "route";
}
@Override
public int filterOrder() {
return 1;
}
@Override
public Object run() throws ZuulException {
RequestContext ctx = RequestContext.getCurrentContext();
String url = ctx.getRequest().getRequestURI();
Map<String,String> forwardMap = getForwardMap(url);
if(forwardMap != null){
String fowardUrl = forwardMap.get(url);
String serviceId = getServiceId(fowardUrl);
String requestUrl = getRequestUrl(fowardUrl,serviceId);
//1.设置目标service的Controller的路径
ctx.put(FilterConstants.REQUEST_URI_KEY,requestUrl);
//2.设置目标service的serviceId
ctx.put(FilterConstants.SERVICE_ID_KEY,serviceId);
}
return null;
}
private String getServiceId(String url){
if(url.startsWith("/")){
String temp = url.substring(1);
return temp.split("/")[0];
}else{
return null;
}
}
private String getRequestUrl(String url,String serviceId){
return url.substring(serviceId.length() +1);
}
@Override
public boolean shouldFilter() {
return true;
}
private Map<String,String> getForwardMap(String originalUrl){
//TODO:这里是返回一个map,传入一个originUrl,返回一个要转发的url
}
}
2. 注意点
ctx.put(FilterConstants.REQUEST_URI_KEY,requestUrl);
但是我使用的时候,一直报404,后来跟踪服务之后,发现其实ctx里面还有一个serviceId的属性,它是跟当前请求的这个原始url映射的serviceId保持一致的。如果你要转发的serviceId并不是这个的话,那就会报404,所以必须要在这里重新定义serviceId,ctx.put(FilterConstants.SERVICE_ID_KEY,serviceId);