目录
前言:想实现一个轻量级的接口反向代理和转发的一个接口服务,可以通过这个服务做一些需要认证才能访问的接口给到前端使用,这样就实现了一种认证可以调用多种第三方系统的服务。
基本逻辑就是将请求的请求方式、请求头、请求体提取出来,将这些信息转发到另外一个接口
- 1.1 配置类
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.context.annotation.Configuration;
import org.springframework.stereotype.Component;
import java.util.List;
/**
* 路由代理配置
*/
@Configuration
@ConfigurationProperties(prefix = "delegate.config.api", ignoreUnknownFields = false)
public class RouterDelegateProperties {
/**
* 网关地址
*/
String rootPath;
/**
* 服务名称, 服务名称和服务网关和服务拓展器一一对应, 服务网关和服务拓展器没有则用default代替
*/
List<String> serviceName;
/**
* 服务网关, 服务名称和服务网关和服务拓展器一一对应, 服务网关和服务拓展器没有则用default代替
*/
List<String> serviceRoot;
/**
* 服务拓展器, 服务名称和服务网关和服务拓展器一一对应, 服务网关和服务拓展器没有则用default代替
*/
List<String> serviceExtractor;
public String getRootPath() {
return rootPath;
}
public void setRootPath(String rootPath) {
this.rootPath = rootPath;
}
public List<String> getServiceName() {
return serviceName;
}
public void setServiceName(List<String> serviceName) {
this.serviceName = serviceName;
}
public List<String> getServiceRoot() {
return serviceRoot;
}
public void setServiceRoot(List<String> serviceRoot) {
this.serviceRoot = serviceRoot;
}
public List<String> getServiceExtractor() {
return serviceExtractor;
}
public void setServiceExtractor(List<String> serviceExtractor) {
this.serviceExtractor = serviceExtractor;
}
}
- 1.2 实体DTO
import java.io.Serializable;
/**
* 代理路由配置 DTO
*/
public class RouterDelegateConfigDTO implements Serializable {
private static final long serialVersionUID = 1L;
/**
* 网关地址
*/
private String rootPath;
/**
* 服务名称
*/
private String serviceName;
/**
* 服务名称地址
*/
private String serviceRoot;
/**
* 服务名称处理器
*/
private String serviceExtractor;
public String getRootPath() {
return rootPath;
}
public void setRootPath(String rootPath) {
this.rootPath = rootPath;
}
public String getServiceName() {
return serviceName;
}
public void setServiceName(String serviceName) {
this.serviceName = serviceName;
}
public String getServiceRoot() {
return serviceRoot;
}
public void setServiceRoot(String serviceRoot) {
this.serviceRoot = serviceRoot;
}
public String getServiceExtractor() {
return serviceExtractor;
}
public void setServiceExtractor(String serviceExtractor) {
this.serviceExtractor = serviceExtractor;
}
}
- 1.3 路由代理拓展器
import org.springframework.http.HttpHeaders;
import zsoft.gov.datacenter.biztable.common.dto.router.RouterDelegateConfigDTO;
import javax.servlet.http.HttpServletRequest;
/**
* 路由代理拓展器
*/
public interface RouterDelegateExtractor {
/**
* 处理请求url, 返回null则使用通用处理逻辑
*
* @param request 请求体对象
* @param configDTO 服务配置对象
* @param prefix 代理前缀
* @return
*/
String getRequestRootUrl(HttpServletRequest request, RouterDelegateConfigDTO configDTO, String prefix);
/**
* 处理请求头
*
* @param request 请求体对象
* @param headers 请求头
*/
void parseRequestHeader(HttpServletRequest request, HttpHeaders headers);
/**
* 处理请求体, 返回null则使用通用处理逻辑
*
* @param request 请求体对象
* @return
*/
byte[] parseRequestBody(HttpServletRequest request);
}
- 1.4 请求对象 RestTemplate
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Qualifier;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.http.client.ClientHttpRequestFactory;
import org.springframework.http.client.ClientHttpResponse;
import org.springframework.http.client.SimpleClientHttpRequestFactory;
import org.springframework.http.converter.HttpMessageConverter;
import org.springframework.http.converter.StringHttpMessageConverter;
import org.springframework.web.client.DefaultResponseErrorHandler;
import org.springframework.web.client.RestTemplate;
import java.io.IOException;
import java.nio.charset.Charset;
import java.util.List;
@Configuration
public class FetchApiRestTemplateConfig {
@Bean({"fetchApiRestTemplate"})
@Autowired
public RestTemplate restTemplate(@Qualifier("fetchApiClientHttpRequestFactory") ClientHttpRequestFactory factory) {
RestTemplate restTemplate = new RestTemplate(factory);
restTemplate.setErrorHandler(new DefaultResponseErrorHandler() {
@Override
public void handleError(ClientHttpResponse response) throws IOException {
// if (response.getRawStatusCode() != 401 && response.getRawStatusCode() != 404) {
// super.handleError(response);
// }
// 处理返回 4xx 的状态码时不抛出异常
if (!response.getStatusCode().is4xxClientError()) {
super.handleError(response);
}
}
});
// 中文乱码问题
List<HttpMessageConverter<?>> httpMessageConverters = restTemplate.getMessageConverters();
httpMessageConverters.stream().forEach(httpMessageConverter -> {
if (httpMessageConverter instanceof StringHttpMessageConverter) {
StringHttpMessageConverter messageConverter = (StringHttpMessageConverter) httpMessageConverter;
messageConverter.setDefaultCharset(Charset.forName("UTF-8"));
}
});
return restTemplate;
}
@Bean({"fetchApiClientHttpRequestFactory"})
public ClientHttpRequestFactory simpleClientHttpRequestFactory() {
SimpleClientHttpRequestFactory factory = new SimpleClientHttpRequestFactory();
factory.setReadTimeout(1000 * 50); // 读取超时(毫秒)
factory.setConnectTimeout(1000 * 10); // 连接超时(毫秒)
return factory;
}
}
- 2、核心转发代码
import org.apache.commons.collections.CollectionUtils;
import org.apache.commons.lang3.StringUtils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Qualifier;
import org.springframework.boot.ApplicationArguments;
import org.springframework.boot.ApplicationRunner;
import org.springframework.http.*;
import org.springframework.stereotype.Service;
import org.springframework.util.StreamUtils;
import org.springframework.web.client.RestTemplate;
import zsoft.gov.datacenter.biztable.common.config.RouterDelegateProperties;
import zsoft.gov.datacenter.biztable.common.dto.router.RouterDelegateConfigDTO;
import zsoft.gov.datacenter.biztable.common.response.Result;
import javax.annotation.Resource;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.io.InputStream;
import java.net.URI;
import java.net.URISyntaxException;
import java.util.Collections;
import java.util.List;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
/**
* 路由代理
*/
@Service
public class RouterDelegate implements ApplicationRunner {
protected Logger logger = LoggerFactory.getLogger(getClass());
private static Map<String, RouterDelegateConfigDTO> configMap;
@Resource
@Qualifier("fetchApiRestTemplate")
private RestTemplate restTemplate;
@Resource
private RouterDelegateProperties properties;
@Resource
private Map<String, RouterDelegateExtractor> stringRouterDelegateExtractorMap;
/**
* 初始化配置类
*
* @param args
* @throws Exception
*/
@Override
public void run(ApplicationArguments args) throws Exception {
boolean intiFlag = false;
logger.info(">>> -----开始初始化路由代理配置类!");
/**
* 最终configMap效果
* {
* 服务名称: {
* rootPath: "系统网关地址",
* serviceName: "服务名称",
* serviceRoot: "服务网关",
* serviceExtractor: "服务拓展器",
* }
* }
*/
String rootPath = properties.getRootPath();
List<String> serviceName = properties.getServiceName();
List<String> serviceRoot = properties.getServiceRoot();
List<String> serviceExtractor = properties.getServiceExtractor();
// 服务名称, 服务名称和服务网关和服务处理器一一对应, 如果没有对应的服务网关和服务处理器, 则用英文逗号隔开
if (StringUtils.isNotBlank(rootPath)
&& CollectionUtils.isNotEmpty(serviceName)
&& CollectionUtils.isNotEmpty(serviceExtractor)
&& CollectionUtils.isNotEmpty(serviceRoot)
&& serviceName.size() == serviceRoot.size()
&& serviceName.size() == serviceExtractor.size()) {
intiFlag = true;
// 初始化大小避免扩容
int initialCapacity = (int) (serviceName.size() / 0.75) + 1;
configMap = new ConcurrentHashMap<>(initialCapacity);
for (int i = 0; i < serviceName.size(); i++) {
RouterDelegateConfigDTO dto = new RouterDelegateConfigDTO();
String serName = serviceName.get(i);
dto.setRootPath(rootPath);
dto.setServiceName(serName);
// default 是占位符, 配置成default相当于没有配置
dto.setServiceRoot("default".equals(serviceRoot.get(i)) ? null : serviceRoot.get(i));
dto.setServiceExtractor("default".equals(serviceExtractor.get(i)) ? null : serviceExtractor.get(i));
configMap.put(serName, dto);
}
}
if (intiFlag) logger.info(">>> 初始化路由代理配置类成功!");
else logger.error(">>> 初始化路由代理配置类失败!");
}
public ResponseEntity<byte[]> redirect(HttpServletRequest request, HttpServletResponse response, String prefix, String serviceName) {
String requestURI = request.getRequestURI();
RouterDelegateConfigDTO currentConfig = getCurrentServiceConfig(serviceName);
if (currentConfig == null) {
return buildErrorResponseEntity("SERVICE ERROR! 服务不存在!", HttpStatus.NOT_FOUND);
}
RouterDelegateExtractor extractorCallBack = getRouterDelegateExtractor(serviceName);
try {
// 创建url
String redirectUrl = createRequestUrl(request, currentConfig, prefix, extractorCallBack);
logger.info(">>> redirectUrl代理后的完整地址: [{}]", redirectUrl);
RequestEntity requestEntity = createRequestEntity(request, redirectUrl, extractorCallBack);
// return route(request, redirectUrl, extractorCallBack);
ResponseEntity<byte[]> result = route(requestEntity);
if (result.getHeaders() != null && result.getHeaders().containsKey(HttpHeaders.TRANSFER_ENCODING)) {
// 移除响应头 Transfer-Encoding, 因为高版本的nginx会自动添加该响应头, 多个响应值nginx会报错
// 多个响应值nginx报错: *6889957 upstream sent duplicate header line: "Transfer-Encoding: chunked", previous value: "Transfer-Encoding: chunked" while reading response header from upstream
HttpHeaders headers = HttpHeaders.writableHttpHeaders(result.getHeaders());
headers.remove(HttpHeaders.TRANSFER_ENCODING);
}
//logger.info(">>> [{}] 代理成功, 请求耗时: [{}]", requestURI, System.currentTimeMillis() - l1);
return result;
} catch (Exception e) {
logger.error("REDIRECT ERROR", e);
//logger.error(">>> [{}] 代理失败, 请求耗时: [{}]", requestURI, System.currentTimeMillis() - l1);
return buildErrorResponseEntity("REDIRECT ERROR! " + e.getMessage(), HttpStatus.INTERNAL_SERVER_ERROR);
}
}
private ResponseEntity buildErrorResponseEntity(String msg, HttpStatus httpStatus) {
HttpHeaders headers = new HttpHeaders();
headers.setContentType(MediaType.APPLICATION_JSON);
Result body = Result.build(httpStatus.value(), msg);
return new ResponseEntity(body, headers, httpStatus);
}
/**
* 获取当前服务配置
*
* @param serviceName
* @return
*/
public RouterDelegateConfigDTO getCurrentServiceConfig(String serviceName) {
if (configMap == null || !configMap.containsKey(serviceName)) {
return null;
}
return configMap.get(serviceName);
}
/**
* 获取当前路由服务拓展器
*
* @param serviceName
* @return
*/
private RouterDelegateExtractor getRouterDelegateExtractor(String serviceName) {
RouterDelegateConfigDTO currentConfig = getCurrentServiceConfig(serviceName);
if (currentConfig == null) {
return null;
}
String serviceExtractor = currentConfig.getServiceExtractor();
if (StringUtils.isBlank(serviceExtractor)) {
return null;
}
RouterDelegateExtractor extractor = stringRouterDelegateExtractorMap.get(serviceExtractor + "RouterDelegateExtractor");
return extractor;
}
/**
* 创建请求地址
*
* @param request
* @param configDTO
* @param prefix
* @param extractorCallback
* @return
*/
private String createRequestUrl(HttpServletRequest request, RouterDelegateConfigDTO configDTO, String prefix, RouterDelegateExtractor extractorCallback) {
String routeUrl = configDTO.getRootPath();
// 拓展器不为null, 并且有返回结果才使用
if (extractorCallback != null) {
String hostUrl = extractorCallback.getRequestRootUrl(request, configDTO, prefix);
if (hostUrl != null) routeUrl = hostUrl;
}
String queryString = request.getQueryString();
// return routeUrl + request.getRequestURI().replace(prefix, "") +
// (queryString != null ? "?" + queryString : "");
// request.getRequestURI() 包括 server.servlet.context-path
// request.getServletPath() 不包括 server.servlet.context-path
// http://127.0.0.1/databook-api/graphdb/sj/tianda/openapi/v1/applets?name=ts
// request.getRequestURI() = /databook-api/graphdb/sj/tianda/openapi/v1/applets
// request.getServletPath() = /graphdb/sj/tianda/openapi/v1/applets
String serviceName = configDTO.getServiceName();
return routeUrl + request.getServletPath().replaceFirst(prefix + "/" + serviceName, "") +
(queryString != null ? "?" + queryString : "");
}
private RequestEntity createRequestEntity(HttpServletRequest request, String url, RouterDelegateExtractor extractorCallBack) throws URISyntaxException, IOException {
String method = request.getMethod();
HttpMethod httpMethod = HttpMethod.resolve(method);
HttpHeaders headers = parseRequestHeader(request, extractorCallBack);
byte[] body = parseRequestBody(request, extractorCallBack);
return new RequestEntity<>(body, headers, httpMethod, new URI(url));
}
private ResponseEntity<byte[]> route(HttpServletRequest request, String url, RouterDelegateExtractor extractorCallBack) throws IOException, URISyntaxException {
String method = request.getMethod();
HttpMethod httpMethod = HttpMethod.resolve(method);
HttpHeaders headers = parseRequestHeader(request, extractorCallBack);
byte[] body = parseRequestBody(request, extractorCallBack);
// 设置请求实体
HttpEntity<byte[]> httpEntity = new HttpEntity<>(body, headers);
URI uri = new URI(url);
return restTemplate.exchange(uri, httpMethod, httpEntity, byte[].class);
}
private ResponseEntity<byte[]> route(RequestEntity requestEntity) {
return restTemplate.exchange(requestEntity, byte[].class);
}
/**
* 处理请求头
*
* @param request
* @param extractorCallBack
* @return
*/
private HttpHeaders parseRequestHeader(HttpServletRequest request, RouterDelegateExtractor extractorCallBack) {
List<String> headerNames = Collections.list(request.getHeaderNames());
HttpHeaders headers = new HttpHeaders();
for (String headerName : headerNames) {
List<String> headerValues = Collections.list(request.getHeaders(headerName));
for (String headerValue : headerValues) {
headers.add(headerName, headerValue);
}
}
if (extractorCallBack != null) {
extractorCallBack.parseRequestHeader(request, headers);
}
// 移除请求头accept-encoding, 不移除会导致响应体转成String时会乱码
headers.remove("accept-encoding");
return headers;
}
/**
* 处理请求体
*
* @param request
* @param extractorCallBack
* @return
* @throws IOException
*/
private byte[] parseRequestBody(HttpServletRequest request, RouterDelegateExtractor extractorCallBack) throws IOException {
// 拓展器不为null, 并且返回的结果也不为null才使用返回结果, 否则使用通用处理逻辑
if (extractorCallBack != null) {
byte[] body = extractorCallBack.parseRequestBody(request);
if (body != null) return body;
}
InputStream inputStream = request.getInputStream();
return StreamUtils.copyToByteArray(inputStream);
}
}
- 3、暴露接口
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.http.ResponseEntity;
import org.springframework.web.bind.annotation.PathVariable;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;
import org.springframework.web.bind.annotation.RestController;
import zsoft.gov.datacenter.biztable.common.router.RouterDelegate;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
/**
* 路由代理接口
*/
@RestController
@RequestMapping
public class RouterDelegateController {
public final static String DELEGATE_PREFIX = "/delegate";
@Autowired
private RouterDelegate routerDelegate;
/**
* 路由代理接口
*
* @param serviceName
* @param request
* @param response
* @return
*/
@RequestMapping(value = DELEGATE_PREFIX + "/{serviceName}/**", method = {RequestMethod.GET, RequestMethod.POST, RequestMethod.PUT, RequestMethod.DELETE})
public ResponseEntity redirect(@PathVariable("serviceName") String serviceName, HttpServletRequest request, HttpServletResponse response) {
return routerDelegate.redirect(request, response, DELEGATE_PREFIX, serviceName);
}
}
- 4、基础配置
#路由代理配置-网关地址
delegate.config.api.rootPath=http://192.168.50.43:7612
#路由代理配置-服务名称, 服务名称和服务网关和服务拓展器一一对应, 服务网关和服务拓展器没有则用default代替
delegate.config.api.serviceName=common,csdn
#路由代理配置-服务网关, 服务名称和服务网关和服务拓展器一一对应, 服务网关和服务拓展器没有则用default代替
delegate.config.api.serviceRoot=default,https://csdn.net
#路由代理配置-服务拓展器, 服务名称和服务网关和服务拓展器一一对应, 服务网关和服务拓展器没有则用default代替
delegate.config.api.serviceExtractor=default,csdnBlog