|
@ -0,0 +1,323 @@
|
|
|
package camel.central.gateway.processor;
|
|
|
|
|
|
import com.fasterxml.jackson.databind.JsonNode;
|
|
|
import com.fasterxml.jackson.databind.ObjectMapper;
|
|
|
import com.yihu.hos.core.datatype.StringUtil;
|
|
|
import com.yihu.hos.core.http.HTTPResponse;
|
|
|
import com.yihu.hos.core.http.HttpClientKit;
|
|
|
import org.apache.camel.CamelContext;
|
|
|
import org.apache.camel.Exchange;
|
|
|
import org.apache.camel.Processor;
|
|
|
import org.apache.camel.ProducerTemplate;
|
|
|
import org.springframework.beans.factory.annotation.Autowired;
|
|
|
import org.springframework.cloud.client.ServiceInstance;
|
|
|
import org.springframework.cloud.client.discovery.DiscoveryClient;
|
|
|
import org.springframework.stereotype.Component;
|
|
|
|
|
|
import java.io.IOException;
|
|
|
import java.io.UnsupportedEncodingException;
|
|
|
import java.net.URLEncoder;
|
|
|
import java.text.ParseException;
|
|
|
import java.text.SimpleDateFormat;
|
|
|
import java.util.Calendar;
|
|
|
import java.util.HashMap;
|
|
|
import java.util.List;
|
|
|
import java.util.Map;
|
|
|
|
|
|
|
|
|
@Component
|
|
|
public class GatewayProcessor implements Processor {
|
|
|
|
|
|
static final String agUrl = "http://192.168.220.84:10000";
|
|
|
static final String brokerUrl = "http://192.168.200.62:8099";
|
|
|
// static final String agUrl = "http://172.19.103.73:10000";
|
|
|
// static final String brokerUrl = "http://10.176.97.14:8099";
|
|
|
static Map<String, AppApi> apiMap = new HashMap<>();
|
|
|
@Autowired
|
|
|
private DiscoveryClient discoveryClient;
|
|
|
private ErrorHandle errorHandle = new ErrorHandle();
|
|
|
|
|
|
public void process(Exchange exchange) throws Exception {
|
|
|
}
|
|
|
|
|
|
|
|
|
public String route(Exchange exchange) throws Exception {
|
|
|
|
|
|
System.out.println("进入同一网关route流程!");
|
|
|
// body = URLDecoder.decode(body, "UTF-8");
|
|
|
Map<String, Object> params = exchange.getIn().getHeaders();
|
|
|
if (params.get("appKey") == null) {
|
|
|
exchange.getOut().setHeader(Exchange.HTTP_URI, "paramError");
|
|
|
System.out.println("进入route:appKey is null ");
|
|
|
return errorHandle.paramError();
|
|
|
}
|
|
|
String appKey = params.get("appKey").toString();
|
|
|
String secret = getSecret(appKey);
|
|
|
|
|
|
if (secret.equals("")) {
|
|
|
exchange.getOut().setHeader(Exchange.HTTP_URI, "paramError");
|
|
|
System.out.println("appKey 获取到的secret是空的 ");
|
|
|
return errorHandle.paramError();
|
|
|
}
|
|
|
|
|
|
boolean pass = checkParams(params);
|
|
|
if (!pass) {
|
|
|
exchange.getOut().setHeader(Exchange.HTTP_URI, "paramError");
|
|
|
System.out.println("a参数校验失败 "+ params);
|
|
|
return errorHandle.paramError();
|
|
|
}
|
|
|
|
|
|
pass = checkTimeStamp(params);
|
|
|
if (!pass) {
|
|
|
exchange.getOut().setHeader(Exchange.HTTP_URI, "outdateError");
|
|
|
return errorHandle.outdateError();
|
|
|
}
|
|
|
|
|
|
//获取secret接口,不做sign验证
|
|
|
if (!"admin.apps.get".equals(params.get("api"))) {
|
|
|
pass = checkSign(params, secret);
|
|
|
if (!pass) {
|
|
|
exchange.getOut().setHeader(Exchange.HTTP_URI, "signValidError");
|
|
|
return errorHandle.signValidError();
|
|
|
}
|
|
|
}
|
|
|
|
|
|
pass = checkAuthorized(params);
|
|
|
if (!pass) {
|
|
|
exchange.getOut().setHeader(Exchange.HTTP_URI, "unauthorizedError");
|
|
|
return errorHandle.unauthorizedError();
|
|
|
}
|
|
|
|
|
|
return handle(params, exchange);
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* 通过服务名获取微服务地址
|
|
|
*
|
|
|
* @param serviceName 微服务名
|
|
|
* @return
|
|
|
*/
|
|
|
public String serviceUrl(String serviceName) {
|
|
|
List<ServiceInstance> instances = discoveryClient.getInstances(serviceName);
|
|
|
if (instances != null && !instances.isEmpty()) {
|
|
|
int index = (int) (Math.random() * instances.size()); //随机获取其中一个服务
|
|
|
ServiceInstance serviceInstance = instances.get(index);
|
|
|
return serviceInstance.getUri().toString();
|
|
|
} else {
|
|
|
return null;
|
|
|
}
|
|
|
}
|
|
|
|
|
|
public String nodeValue(JsonNode node) {
|
|
|
Object value = null;
|
|
|
try {
|
|
|
switch (node.getNodeType().name()) {
|
|
|
case "NUMBER":
|
|
|
value = node.asInt();
|
|
|
break;
|
|
|
case "STRING":
|
|
|
value = node.asText();
|
|
|
break;
|
|
|
case "BOOLEAN":
|
|
|
value = node.asBoolean();
|
|
|
break;
|
|
|
case "OBJECT":
|
|
|
value = node.toString();
|
|
|
break;
|
|
|
default:
|
|
|
value = "";
|
|
|
break;
|
|
|
}
|
|
|
return URLEncoder.encode(value.toString(), "UTF-8");
|
|
|
} catch (UnsupportedEncodingException e) {
|
|
|
e.printStackTrace();
|
|
|
}
|
|
|
return "";
|
|
|
}
|
|
|
|
|
|
private boolean checkParams(Map<String, Object> params) {
|
|
|
Object api = params.get("api"); // API接口名称
|
|
|
Object sign = params.get("sign"); // 簽名
|
|
|
Object timestamp = params.get("timestamp"); // 时间戳
|
|
|
Object appKey = params.get("appKey"); // HOP分配给应用的AppKey ,创建应用时可获得
|
|
|
Object version = params.get("v"); // API协议版本
|
|
|
StringBuilder sb = new StringBuilder();
|
|
|
sb.append("api:").append(api)
|
|
|
.append(",sign:").append(sign)
|
|
|
.append(",timestamp:").append(timestamp)
|
|
|
.append(",appKey:").append(appKey)
|
|
|
.append(",v").append(version);
|
|
|
System.out.println("检查参数,"+ sb.toString());
|
|
|
return !(StringUtil.isEmpty(api) || StringUtil.isEmpty(timestamp)
|
|
|
|| StringUtil.isEmpty(appKey) || StringUtil.isEmpty(version)
|
|
|
|| StringUtil.isEmpty(sign));
|
|
|
}
|
|
|
|
|
|
private boolean checkTimeStamp(Map<String, Object> params) throws ParseException {
|
|
|
final long ONE_MIN = 60000; //millisecond
|
|
|
|
|
|
String timestamp = params.get("timestamp").toString();
|
|
|
SimpleDateFormat format = new SimpleDateFormat("yyyy-MM-dd'T'HH:mm:ssZ");
|
|
|
|
|
|
java.util.Date ts = format.parse(timestamp);
|
|
|
if (ts == null) {
|
|
|
return false; //时间格式不正确
|
|
|
}
|
|
|
|
|
|
Calendar date = Calendar.getInstance();
|
|
|
long timeInMillis = date.getTimeInMillis();
|
|
|
java.util.Date min = new java.util.Date(timeInMillis - 30 * ONE_MIN);
|
|
|
java.util.Date max = new java.util.Date(timeInMillis + 30 * ONE_MIN);
|
|
|
|
|
|
return ts.after(min) && ts.before(max);
|
|
|
|
|
|
}
|
|
|
|
|
|
private boolean checkSign(Map<String, Object> params, String secret) {
|
|
|
try {
|
|
|
String sign = params.get("sign").toString(); // 簽名
|
|
|
|
|
|
ParamVerifyBean paramSign = new ParamVerifyBean();
|
|
|
paramSign.addParam(params);
|
|
|
// paramSign.genParam();
|
|
|
//TODO 获取app secret传入验证
|
|
|
String md5Sign = paramSign.signParam(secret);
|
|
|
if (!md5Sign.equals(sign)) {
|
|
|
System.out.println("传递的签名:" + sign);
|
|
|
System.out.println("生成的签名:" + md5Sign);
|
|
|
return false;
|
|
|
}
|
|
|
|
|
|
return true;
|
|
|
} catch (Exception e) {
|
|
|
e.printStackTrace();
|
|
|
return false;
|
|
|
}
|
|
|
}
|
|
|
|
|
|
private boolean checkAuthorized(Map<String, Object> params) {
|
|
|
ObjectMapper objectMapper = new ObjectMapper();
|
|
|
//TODO 设置固定的验证入口地址
|
|
|
HTTPResponse response = HttpClientKit.get(agUrl + "/api/v1.0/admin/appApiAuth?appId=" + params.get("appKey") + "&apiName=" + params.get("api"));
|
|
|
if (response.getStatusCode() != 200) {
|
|
|
System.out.println("验证API权限,请求失败!");
|
|
|
return false;
|
|
|
}
|
|
|
|
|
|
try {
|
|
|
Map map = objectMapper.readValue(response.getBody(), Map.class);
|
|
|
if ((Boolean) map.get("successFlg")) {
|
|
|
return true;
|
|
|
} else {
|
|
|
System.out.println("验证失败:" + map.get("errorMsg"));
|
|
|
return false;
|
|
|
}
|
|
|
} catch (IOException e) {
|
|
|
e.printStackTrace();
|
|
|
return false;
|
|
|
}
|
|
|
}
|
|
|
|
|
|
private String handle(Map<String, Object> params, Exchange exchange) {
|
|
|
String api = params.get("api").toString(); // API接口名称
|
|
|
String param = params.get("param").toString();
|
|
|
|
|
|
AppApi appApi = getApiInfo(api);
|
|
|
ObjectMapper objectMapper = new ObjectMapper();
|
|
|
try {
|
|
|
if (appApi == null) {
|
|
|
return "appApi is null";
|
|
|
}
|
|
|
|
|
|
Map<String, String> methodMap = new HashMap<>();
|
|
|
methodMap.put("0", "get");
|
|
|
methodMap.put("1", "post");
|
|
|
methodMap.put("2", "delete");
|
|
|
methodMap.put("3", "put");
|
|
|
|
|
|
JsonNode jsonNode = objectMapper.readValue(param, JsonNode.class);
|
|
|
String url = appApi.getMicroServiceUri();
|
|
|
final String[] endPoint = {"restlet:" + url + appApi.getMsMethodName() + "?socketTimeout=60000&connectionTimeout=60000&restletMethod=" + methodMap.get(appApi.getMethod())};
|
|
|
final String[] body = {""};
|
|
|
|
|
|
appApi.getParameters().forEach(p -> {
|
|
|
System.out.println(p.getName());
|
|
|
JsonNode paramNode = jsonNode.get(p.getName());
|
|
|
System.out.println(paramNode);
|
|
|
if (paramNode == null) {
|
|
|
return;
|
|
|
}
|
|
|
|
|
|
Object value = nodeValue(paramNode);
|
|
|
if (p.getType().equals("0")) { //path param
|
|
|
endPoint[0] = endPoint[0].replaceAll("\\{[^}]*\\}", value.toString());
|
|
|
} else if (p.getType().equals("2")) { //head param
|
|
|
exchange.getOut().setHeader(p.getName(), value);
|
|
|
} else {
|
|
|
body[0] += ("&" + p.getName() + "=" + value);
|
|
|
}
|
|
|
|
|
|
});
|
|
|
|
|
|
CamelContext context = exchange.getContext();
|
|
|
ProducerTemplate producerTemplate = context.createProducerTemplate();
|
|
|
return producerTemplate.requestBody(endPoint[0], body[0], String.class);
|
|
|
} catch (IOException e) {
|
|
|
e.printStackTrace();
|
|
|
}
|
|
|
return "";
|
|
|
}
|
|
|
|
|
|
private AppApi getApiInfo(String api) {
|
|
|
AppApi appApi = apiMap.get(api);
|
|
|
if (appApi != null) {
|
|
|
return appApi;
|
|
|
}
|
|
|
|
|
|
HTTPResponse response = HttpClientKit.get(agUrl + "/api/v1.0/admin/appApi/search?filters=methodName%3D" + api + "&size=15&page=1");
|
|
|
if (response.getStatusCode() != 200) {
|
|
|
return null;
|
|
|
}
|
|
|
|
|
|
try {
|
|
|
appApi = AppApi.parse(response.getBody());
|
|
|
apiMap.put(api, appApi);
|
|
|
} catch (IOException e) {
|
|
|
e.printStackTrace();
|
|
|
|
|
|
return null;
|
|
|
}
|
|
|
|
|
|
return appApi;
|
|
|
}
|
|
|
|
|
|
private String getSecret(String appKey) {
|
|
|
ObjectMapper objectMapper = new ObjectMapper();
|
|
|
//TODO 设置固定的验证入口地址
|
|
|
HTTPResponse response = HttpClientKit.get(agUrl + "/api/v1.0/admin/apps/" + appKey);
|
|
|
if (response.getStatusCode() != 200) {
|
|
|
System.out.println("获取app的secret请求失败,status: " + response.getStatusCode() + "error: " + response.getBody());
|
|
|
return "";
|
|
|
}
|
|
|
|
|
|
try {
|
|
|
Map map = objectMapper.readValue(response.getBody(), Map.class);
|
|
|
if ((Boolean) map.get("successFlg")) {
|
|
|
Map<String, Object> obj = (Map) map.get("obj");
|
|
|
if (obj == null) {
|
|
|
return "";
|
|
|
}
|
|
|
String secret = obj.get("secret").toString();
|
|
|
return secret;
|
|
|
} else {
|
|
|
System.out.println("验证失败:" + map.get("errorMsg"));
|
|
|
return "";
|
|
|
}
|
|
|
} catch (IOException e) {
|
|
|
e.printStackTrace();
|
|
|
return "";
|
|
|
}
|
|
|
}
|
|
|
|
|
|
|
|
|
}
|