parent
6654ab9b76
commit
c25e59a991
@ -0,0 +1,116 @@ |
||||
<?xml version="1.0" encoding="UTF-8"?> |
||||
<project xmlns="http://maven.apache.org/POM/4.0.0" |
||||
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" |
||||
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"> |
||||
<modelVersion>4.0.0</modelVersion> |
||||
|
||||
<groupId>com.tsl3060.open.extend</groupId> |
||||
<artifactId>OpenExtendDemo</artifactId> |
||||
<version>0.3.6-wanshun</version> |
||||
|
||||
<properties> |
||||
<maven.compiler.source>8</maven.compiler.source> |
||||
<maven.compiler.target>8</maven.compiler.target> |
||||
|
||||
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding> |
||||
<hutool.version>5.8.19</hutool.version> |
||||
<!-- 日志--> |
||||
<logback.version>1.4.7</logback.version> |
||||
</properties> |
||||
|
||||
<dependencies> |
||||
|
||||
<dependency> |
||||
<groupId>cn.hutool</groupId> |
||||
<artifactId>hutool-log</artifactId> |
||||
</dependency> |
||||
|
||||
<!-- 日志库--> |
||||
<dependency> |
||||
<groupId>ch.qos.logback</groupId> |
||||
<artifactId>logback-core</artifactId> |
||||
<version>${logback.version}</version> |
||||
</dependency> |
||||
<dependency> |
||||
<groupId>ch.qos.logback</groupId> |
||||
<artifactId>logback-classic</artifactId> |
||||
<version>${logback.version}</version> |
||||
</dependency> |
||||
<dependency> |
||||
<groupId>org.bouncycastle</groupId> |
||||
<artifactId>bcprov-jdk15to18</artifactId> |
||||
<version>1.69</version> |
||||
</dependency> |
||||
<dependency> |
||||
<groupId>org.junit.jupiter</groupId> |
||||
<artifactId>junit-jupiter</artifactId> |
||||
<version>RELEASE</version> |
||||
<scope>test</scope> |
||||
</dependency> |
||||
<dependency> |
||||
<groupId>com.squareup.okhttp3</groupId> |
||||
<artifactId>okhttp</artifactId> |
||||
<version>4.10.0</version> |
||||
</dependency> |
||||
<dependency> |
||||
<groupId>com.alibaba.fastjson2</groupId> |
||||
<artifactId>fastjson2</artifactId> |
||||
<version>2.0.33</version> |
||||
</dependency> |
||||
|
||||
|
||||
|
||||
|
||||
</dependencies> |
||||
|
||||
<dependencyManagement> |
||||
<dependencies> |
||||
<dependency> |
||||
<groupId>cn.hutool</groupId> |
||||
<artifactId>hutool-bom</artifactId> |
||||
<version>${hutool.version}</version> |
||||
<type>pom</type> |
||||
<!-- 注意这里是import --> |
||||
<scope>import</scope> |
||||
</dependency> |
||||
</dependencies> |
||||
</dependencyManagement> |
||||
|
||||
<build> |
||||
<plugins> |
||||
<plugin> |
||||
<groupId>org.apache.maven.plugins</groupId> |
||||
<artifactId>maven-compiler-plugin</artifactId> |
||||
<version>3.11.0</version> |
||||
<configuration> |
||||
<source>8</source> |
||||
<target>8</target> |
||||
<encoding>UTF-8</encoding> |
||||
</configuration> |
||||
</plugin> |
||||
|
||||
<!-- <plugin>--> |
||||
<!-- <artifactId>maven-assembly-plugin</artifactId>--> |
||||
<!-- <configuration>--> |
||||
<!-- <descriptorRefs>--> |
||||
<!-- <descriptorRef>jar-with-dependencies</descriptorRef>--> |
||||
<!-- </descriptorRefs>--> |
||||
<!-- <archive>--> |
||||
<!-- <manifest>--> |
||||
<!-- <mainClass></mainClass>--> |
||||
<!-- </manifest>--> |
||||
<!-- </archive>--> |
||||
<!-- </configuration>--> |
||||
<!-- <executions>--> |
||||
<!-- <execution>--> |
||||
<!-- <id>make-assembly</id>--> |
||||
<!-- <phase>package</phase>--> |
||||
<!-- <goals>--> |
||||
<!-- <goal>single</goal>--> |
||||
<!-- </goals>--> |
||||
<!-- </execution>--> |
||||
<!-- </executions>--> |
||||
<!-- </plugin>--> |
||||
</plugins> |
||||
</build> |
||||
</project> |
@ -0,0 +1,265 @@ |
||||
package com.tsl3060.open.extend.core; |
||||
|
||||
import cn.hutool.core.date.DateTime; |
||||
import cn.hutool.core.util.StrUtil; |
||||
import cn.hutool.core.util.XmlUtil; |
||||
import cn.hutool.log.Log; |
||||
import cn.hutool.log.LogFactory; |
||||
import com.alibaba.fastjson2.JSON; |
||||
import com.alibaba.fastjson2.JSONObject; |
||||
import com.tsl3060.open.extend.core.exception.ApiException; |
||||
import com.tsl3060.open.extend.core.exception.BadResourceException; |
||||
import com.tsl3060.open.extend.core.notify.IAnswer; |
||||
import com.tsl3060.open.extend.core.router.INotifyRouter; |
||||
import com.tsl3060.open.extend.core.router.NotifyMapRouter; |
||||
import com.tsl3060.open.extend.core.secure.ISecure; |
||||
import com.tsl3060.open.extend.core.secure.SecureTool; |
||||
import okhttp3.*; |
||||
import org.w3c.dom.Document; |
||||
|
||||
import java.io.IOException; |
||||
import java.text.DateFormat; |
||||
import java.text.SimpleDateFormat; |
||||
import java.util.*; |
||||
|
||||
public class ApiClient { |
||||
|
||||
private static final int ERROR_OK = 1000; |
||||
|
||||
private static final int SUB_OK = 2000; |
||||
|
||||
/** |
||||
* 请求内容格式 |
||||
*/ |
||||
private final MediaType contentType = MediaType.parse("application/json;charset=utf-8"); |
||||
|
||||
/** |
||||
* 返回内容格式 |
||||
*/ |
||||
private final MediaType acceptType = MediaType.parse("application/json"); |
||||
|
||||
|
||||
private SecureTool secureTool; |
||||
|
||||
private Config config; |
||||
|
||||
|
||||
private final Log log = LogFactory.get(); |
||||
|
||||
private SecureTool getSecureTool() { |
||||
if (secureTool == null) { |
||||
secureTool = new SecureTool(getConfig()); |
||||
} |
||||
return secureTool; |
||||
} |
||||
|
||||
public Config getConfig() { |
||||
return config; |
||||
} |
||||
|
||||
public void setConfig(Config config) { |
||||
this.config = config; |
||||
} |
||||
|
||||
public <T> T request(IApiRequest request, Class<T> tClass) throws Exception { |
||||
ApiResponse apiResponse = this.request(request); |
||||
if (apiResponse == null) { |
||||
return null; |
||||
} |
||||
if (apiResponse.getErrCode() != ERROR_OK) { |
||||
//系统级错误
|
||||
throw new ApiException(apiResponse.getDescription() + "(" + apiResponse.getErrCode() + ")"); |
||||
} |
||||
if (apiResponse.getSubErr() != SUB_OK) { |
||||
//业务级错误
|
||||
throw new ApiException(apiResponse.getDescription() + "(" + apiResponse.getSubErr() + ")"); |
||||
} |
||||
Object o = apiResponse.getPayload(); |
||||
if (o != null) { |
||||
return JSON.to(tClass, o); |
||||
} |
||||
return null; |
||||
} |
||||
|
||||
/** |
||||
* 接口请求 |
||||
* |
||||
* @param request |
||||
* @return |
||||
*/ |
||||
public ApiResponse request(IApiRequest request) throws Exception { |
||||
ApiRequest apiRequest = pack(request.path(), request); |
||||
ApiResponse apiResponse = this.request(apiRequest); |
||||
if (apiResponse == null) { |
||||
throw new Exception("返回结果为空"); |
||||
} |
||||
return apiResponse; |
||||
} |
||||
|
||||
/** |
||||
* 打包请求 |
||||
* |
||||
* @param path |
||||
* @param data |
||||
* @return |
||||
*/ |
||||
private ApiRequest pack(String path, Object data) { |
||||
ApiRequest apiRequest = new ApiRequest(); |
||||
apiRequest.setPath(path); |
||||
apiRequest.setCharset(this.config.getCharset()); |
||||
DateFormat dateFormat = new SimpleDateFormat(this.config.getDataFormat()); |
||||
apiRequest.setTime(dateFormat.format(new Date())); |
||||
apiRequest.setSignType(this.config.getSignType()); |
||||
apiRequest.setPayload(data); |
||||
apiRequest.setAccessToken(""); |
||||
apiRequest.setAppId(this.config.getAppid()); |
||||
return apiRequest; |
||||
} |
||||
|
||||
/** |
||||
* 执行请求 |
||||
* |
||||
* @param apiRequest |
||||
* @return |
||||
*/ |
||||
public ApiResponse request(ApiRequest apiRequest) { |
||||
ISecure iSecure = getSecureTool().getSecure(apiRequest.getSignType()); |
||||
String signStr = iSecure.requestSign(apiRequest); |
||||
apiRequest.setSign(signStr); |
||||
String fBody = JSON.toJSONString(apiRequest); |
||||
log.debug(">>> {}", fBody); |
||||
RequestBody requestBody = RequestBody.create(fBody, contentType); |
||||
/** |
||||
* 通讯主机地址 |
||||
*/ |
||||
|
||||
String host = this.config.getHost(); |
||||
Request request = new Request.Builder() |
||||
.url(String.format("%s%s", host, apiRequest.getPath())) |
||||
.addHeader("ACCEPT", acceptType.toString()) |
||||
.post(requestBody) |
||||
.build(); |
||||
OkHttpClient.Builder builder = new OkHttpClient.Builder(); |
||||
OkHttpClient okHttpClient = builder.build(); |
||||
try { |
||||
Call call = okHttpClient.newCall(request); |
||||
try (Response response = call.execute()) { |
||||
ResponseBody responseBody = response.body(); |
||||
ApiResponse apiResponse = null; |
||||
if (responseBody != null) { |
||||
String bStr = responseBody.string(); |
||||
log.debug("<<< {}", bStr); |
||||
apiResponse = JSON.parseObject(bStr, ApiResponse.class); |
||||
} |
||||
if (apiResponse == null) { |
||||
return null; |
||||
} |
||||
//验证签名
|
||||
if (this.verifyResponse(apiResponse)) { |
||||
return apiResponse; |
||||
} |
||||
return null; |
||||
|
||||
} |
||||
} catch (IOException e) { |
||||
log.error(e); |
||||
} |
||||
return null; |
||||
} |
||||
|
||||
/** |
||||
* 验证反馈数据 |
||||
* |
||||
* @param response |
||||
* @return |
||||
*/ |
||||
public boolean verifyResponse(ApiResponse response) { |
||||
ISecure iSecure = this.secureTool.getSecure(response.getSignType()); |
||||
if (iSecure == null) { |
||||
log.debug("没有找到签名类型 %s", response.getSignType()); |
||||
return false; |
||||
} |
||||
return iSecure.verifyResponse(response); |
||||
} |
||||
|
||||
|
||||
private INotifyListener notifyListener; |
||||
|
||||
public INotifyListener getNotifyListener() { |
||||
return notifyListener; |
||||
} |
||||
|
||||
public void setNotifyListener(INotifyListener notifyListener) { |
||||
this.notifyListener = notifyListener; |
||||
} |
||||
|
||||
|
||||
private NotifyMapRouter notifyMapRouter; |
||||
|
||||
/** |
||||
* 验证回调数据 |
||||
* |
||||
* @return |
||||
*/ |
||||
public String notifyRun(String raw, String contentType, String accept) throws BadResourceException { |
||||
if (notifyMapRouter == null) { |
||||
notifyMapRouter = new NotifyMapRouter(this.notifyListener); |
||||
} |
||||
if (StrUtil.isEmpty(raw)) { |
||||
throw new BadResourceException("通知内容为空"); |
||||
} |
||||
|
||||
NotifyRequest notifyRequest; |
||||
if (contentType.contains("application/json")) { |
||||
notifyRequest = JSON.parseObject(raw, NotifyRequest.class); |
||||
} else if (contentType.contains("application/xml")) { |
||||
//XML格式
|
||||
Document document = XmlUtil.parseXml(raw); |
||||
//TODO 对XML解析
|
||||
throw new BadResourceException("暂不支持的格式"); |
||||
} else { |
||||
throw new BadResourceException("不支持的数据格式"); |
||||
} |
||||
//验证通知
|
||||
ISecure iSecure = getSecureTool().getSecure(notifyRequest.getSignType()); |
||||
if (!iSecure.verifyNotify(notifyRequest)) { |
||||
throw new BadResourceException("验签未通过"); |
||||
} |
||||
INotifyRouter notifyRouter = this.notifyMapRouter.getRouter(notifyRequest.getModule()); |
||||
if (notifyRouter == null) { |
||||
throw new BadResourceException("未知的通知"); |
||||
} |
||||
|
||||
//组装
|
||||
NotifyAnswerResponse notifyAnswerResponse = new NotifyAnswerResponse(); |
||||
notifyAnswerResponse.setAnswerId(notifyRequest.getNotifyId()); |
||||
notifyAnswerResponse.setAppId(this.config.getAppid()); |
||||
notifyAnswerResponse.setTime(DateTime.now().toString(this.config.getDataFormat())); |
||||
notifyAnswerResponse.setSignType(this.config.getSignType()); |
||||
notifyAnswerResponse.setCharset(this.config.getCharset()); |
||||
//通知
|
||||
try { |
||||
IAnswer o = notifyRouter.makeBody(notifyRequest); |
||||
notifyAnswerResponse.setPayload(o); |
||||
notifyAnswerResponse.setResult("ok"); |
||||
} catch (Exception e) { |
||||
e.printStackTrace(); |
||||
notifyAnswerResponse.setResult("fail"); |
||||
} |
||||
//生成签名
|
||||
ISecure makeSecure = getSecureTool().getSecure(notifyAnswerResponse.getSignType()); |
||||
String signStr = makeSecure.answerSign(notifyAnswerResponse); |
||||
notifyAnswerResponse.setSign(signStr); |
||||
//解析完成
|
||||
if (accept.contains("application/json")) { |
||||
|
||||
return JSON.toJSONString(notifyAnswerResponse); |
||||
} else if (accept.contains("application/xml")) { |
||||
//返回XML格式
|
||||
return ""; |
||||
} else { |
||||
//未知的格式
|
||||
return ""; |
||||
} |
||||
} |
||||
} |
@ -0,0 +1,107 @@ |
||||
package com.tsl3060.open.extend.core; |
||||
|
||||
import com.alibaba.fastjson2.annotation.JSONField; |
||||
|
||||
public class ApiRequest { |
||||
@JSONField(name = "app_id") |
||||
private String appId; |
||||
|
||||
/** |
||||
* 请求路径 |
||||
*/ |
||||
private String path; |
||||
|
||||
/** |
||||
* 访问TOKEN |
||||
*/ |
||||
@JSONField(name = "access_token") |
||||
private String accessToken; |
||||
|
||||
/** |
||||
* 签名数据 |
||||
*/ |
||||
private String sign; |
||||
|
||||
/** |
||||
* 签名类型 |
||||
*/ |
||||
@JSONField(name = "sign_type") |
||||
private String signType; |
||||
/** |
||||
* 数据编码 |
||||
*/ |
||||
private String charset; |
||||
/** |
||||
* 时间 |
||||
*/ |
||||
private String time; |
||||
/** |
||||
* 载体 |
||||
*/ |
||||
private Object payload; |
||||
|
||||
|
||||
public String getAppId() { |
||||
return appId; |
||||
} |
||||
|
||||
public void setAppId(String appId) { |
||||
this.appId = appId; |
||||
} |
||||
|
||||
public String getPath() { |
||||
return path; |
||||
} |
||||
|
||||
public void setPath(String path) { |
||||
this.path = path; |
||||
} |
||||
|
||||
public String getAccessToken() { |
||||
return accessToken; |
||||
} |
||||
|
||||
public void setAccessToken(String accessToken) { |
||||
this.accessToken = accessToken; |
||||
} |
||||
|
||||
public String getSign() { |
||||
return sign; |
||||
} |
||||
|
||||
public void setSign(String sign) { |
||||
this.sign = sign; |
||||
} |
||||
|
||||
public String getSignType() { |
||||
return signType; |
||||
} |
||||
|
||||
public void setSignType(String signType) { |
||||
this.signType = signType; |
||||
} |
||||
|
||||
public String getCharset() { |
||||
return charset; |
||||
} |
||||
|
||||
public void setCharset(String charset) { |
||||
this.charset = charset; |
||||
} |
||||
|
||||
public String getTime() { |
||||
return time; |
||||
} |
||||
|
||||
public void setTime(String time) { |
||||
this.time = time; |
||||
} |
||||
|
||||
public Object getPayload() { |
||||
return payload; |
||||
} |
||||
|
||||
public void setPayload(Object payload) { |
||||
this.payload = payload; |
||||
} |
||||
} |
@ -0,0 +1,125 @@ |
||||
package com.tsl3060.open.extend.core; |
||||
|
||||
import com.alibaba.fastjson2.annotation.JSONField; |
||||
|
||||
public class ApiResponse { |
||||
private String time = ""; |
||||
private String sign = ""; |
||||
private String charset = "UTF-8"; |
||||
@JSONField(name = "response_id") |
||||
private String responseId = ""; |
||||
@JSONField(name = "err_code") |
||||
private int errCode; |
||||
@JSONField(name = "err_msg") |
||||
private String errMsg = ""; |
||||
@JSONField(name = "sub_err") |
||||
private int subErr; |
||||
@JSONField(name = "sub_msg") |
||||
private String subMsg = ""; |
||||
|
||||
@JSONField(name = "sign_type") |
||||
private String signType = ""; |
||||
|
||||
private Object payload; |
||||
|
||||
@JSONField(name = "open_id") |
||||
private String openId = ""; |
||||
|
||||
private String description=""; |
||||
public String getOpenId() { |
||||
return openId; |
||||
} |
||||
|
||||
public void setOpenId(String openId) { |
||||
this.openId = openId; |
||||
} |
||||
|
||||
public String getTime() { |
||||
return time; |
||||
} |
||||
|
||||
|
||||
public void setTime(String time) { |
||||
this.time = time; |
||||
} |
||||
|
||||
public String getSign() { |
||||
return sign; |
||||
} |
||||
|
||||
public void setSign(String sign) { |
||||
this.sign = sign; |
||||
} |
||||
|
||||
public String getCharset() { |
||||
return charset; |
||||
} |
||||
|
||||
public void setCharset(String charset) { |
||||
this.charset = charset; |
||||
} |
||||
|
||||
public String getResponseId() { |
||||
return responseId; |
||||
} |
||||
|
||||
public void setResponseId(String responseId) { |
||||
this.responseId = responseId; |
||||
} |
||||
|
||||
public int getErrCode() { |
||||
return errCode; |
||||
} |
||||
|
||||
public void setErrCode(int errCode) { |
||||
this.errCode = errCode; |
||||
} |
||||
|
||||
public String getErrMsg() { |
||||
return errMsg; |
||||
} |
||||
|
||||
public void setErrMsg(String errMsg) { |
||||
this.errMsg = errMsg; |
||||
} |
||||
|
||||
public int getSubErr() { |
||||
return subErr; |
||||
} |
||||
|
||||
public void setSubErr(int subErr) { |
||||
this.subErr = subErr; |
||||
} |
||||
|
||||
public String getSubMsg() { |
||||
return subMsg; |
||||
} |
||||
|
||||
public void setSubMsg(String subMsg) { |
||||
this.subMsg = subMsg; |
||||
} |
||||
|
||||
public String getSignType() { |
||||
return signType; |
||||
} |
||||
|
||||
public void setSignType(String signType) { |
||||
this.signType = signType; |
||||
} |
||||
|
||||
public Object getPayload() { |
||||
return payload; |
||||
} |
||||
|
||||
public void setPayload(Object payload) { |
||||
this.payload = payload; |
||||
} |
||||
|
||||
public String getDescription() { |
||||
return description; |
||||
} |
||||
|
||||
public void setDescription(String description) { |
||||
this.description = description; |
||||
} |
||||
} |
@ -0,0 +1,113 @@ |
||||
package com.tsl3060.open.extend.core; |
||||
|
||||
import com.tsl3060.open.extend.core.secure.RSASecure; |
||||
|
||||
public class Config { |
||||
private String privateKey = ""; |
||||
private String publicKey = ""; |
||||
/** |
||||
* API 平台公钥 |
||||
*/ |
||||
private String apiPublicKey = ""; |
||||
|
||||
private String appid = ""; |
||||
|
||||
private String signType = RSASecure.NAME; |
||||
|
||||
private String charset = "utf-8"; |
||||
|
||||
private String host = "https://open.tsl3060.com"; |
||||
private String dataFormat = "yyyy-MM-dd HH:mm:ss"; |
||||
|
||||
|
||||
public String getPrivateKey() { |
||||
return privateKey; |
||||
} |
||||
|
||||
public void setPrivateKey(String privateKey) { |
||||
this.privateKey = privateKey; |
||||
if (this.privateKey != null) { |
||||
this.privateKey = this.privateKey |
||||
.replace("\r", "") |
||||
.replace("\n", "") |
||||
.replace("-----BEGIN PRIVATE KEY-----", "") |
||||
.replace("-----END PRIVATE KEY-----", ""); |
||||
} |
||||
} |
||||
|
||||
public String getPublicKey() { |
||||
return publicKey; |
||||
} |
||||
|
||||
public void setPublicKey(String publicKey) { |
||||
this.publicKey = publicKey; |
||||
if (this.publicKey != null) { |
||||
this.publicKey = this.publicKey |
||||
.replace("\r", "") |
||||
.replace("\n", "") |
||||
.replace("-----BEGIN PUBLIC KEY-----", "") |
||||
.replace("-----END PUBLIC KEY-----", ""); |
||||
} |
||||
} |
||||
|
||||
public String getApiPublicKey() { |
||||
return apiPublicKey; |
||||
} |
||||
|
||||
public void setApiPublicKey(String apiPublicKey) { |
||||
this.apiPublicKey = apiPublicKey; |
||||
if (this.apiPublicKey != null) { |
||||
this.apiPublicKey = this.apiPublicKey |
||||
.replace("\r", "") |
||||
.replace("\n", "") |
||||
.replace("-----BEGIN PUBLIC KEY-----", "") |
||||
.replace("-----END PUBLIC KEY-----", ""); |
||||
|
||||
} |
||||
} |
||||
|
||||
public String getAppid() { |
||||
return appid; |
||||
} |
||||
|
||||
public void setAppid(String appid) { |
||||
this.appid = appid; |
||||
} |
||||
|
||||
public String getSignType() { |
||||
return signType; |
||||
} |
||||
|
||||
public void setSignType(String signType) { |
||||
this.signType = signType; |
||||
} |
||||
|
||||
public String getCharset() { |
||||
return charset; |
||||
} |
||||
|
||||
public void setCharset(String charset) { |
||||
this.charset = charset; |
||||
} |
||||
|
||||
public String getHost() { |
||||
return host; |
||||
} |
||||
|
||||
public void setHost(String host) { |
||||
this.host = host; |
||||
} |
||||
|
||||
public String getDataFormat() { |
||||
return dataFormat; |
||||
} |
||||
|
||||
public void setDataFormat(String dataFormat) { |
||||
this.dataFormat = dataFormat; |
||||
} |
||||
|
||||
|
||||
public void useSandBox() { |
||||
this.setHost("https://opendev.tsl3060.com"); |
||||
} |
||||
} |
@ -0,0 +1,11 @@ |
||||
package com.tsl3060.open.extend.core; |
||||
|
||||
public interface IApiRequest { |
||||
/** |
||||
* 请求路径 |
||||
* |
||||
* @return |
||||
*/ |
||||
String path(); |
||||
|
||||
} |
@ -0,0 +1,14 @@ |
||||
package com.tsl3060.open.extend.core; |
||||
|
||||
import com.tsl3060.open.extend.core.notify.CarbonOrderNotify; |
||||
import com.tsl3060.open.extend.core.notify.CarbonOrderNotifyAnswer; |
||||
|
||||
public interface INotifyListener { |
||||
/** |
||||
* 低碳积分通知 |
||||
* |
||||
* @param notify |
||||
* @return |
||||
*/ |
||||
CarbonOrderNotifyAnswer carbon(CarbonOrderNotify notify); |
||||
} |
@ -0,0 +1,84 @@ |
||||
package com.tsl3060.open.extend.core; |
||||
|
||||
import com.alibaba.fastjson2.annotation.JSONField; |
||||
|
||||
public class NotifyAnswerResponse { |
||||
@JSONField(name = "answer_id") |
||||
private String answerId; |
||||
@JSONField(name = "app_id") |
||||
private String appId; |
||||
private String time; |
||||
private Object payload; |
||||
|
||||
private String sign; |
||||
@JSONField(name = "sign_type") |
||||
private String signType; |
||||
|
||||
private String charset; |
||||
|
||||
private String result; |
||||
|
||||
public String getResult() { |
||||
return result; |
||||
} |
||||
|
||||
public void setResult(String result) { |
||||
this.result = result; |
||||
} |
||||
|
||||
public String getCharset() { |
||||
return charset; |
||||
} |
||||
|
||||
public void setCharset(String charset) { |
||||
this.charset = charset; |
||||
} |
||||
|
||||
public String getAnswerId() { |
||||
return answerId; |
||||
} |
||||
|
||||
public void setAnswerId(String answerId) { |
||||
this.answerId = answerId; |
||||
} |
||||
|
||||
public String getAppId() { |
||||
return appId; |
||||
} |
||||
|
||||
public void setAppId(String appId) { |
||||
this.appId = appId; |
||||
} |
||||
|
||||
public String getTime() { |
||||
return time; |
||||
} |
||||
|
||||
public void setTime(String time) { |
||||
this.time = time; |
||||
} |
||||
|
||||
public Object getPayload() { |
||||
return payload; |
||||
} |
||||
|
||||
public void setPayload(Object payload) { |
||||
this.payload = payload; |
||||
} |
||||
|
||||
public String getSign() { |
||||
return sign; |
||||
} |
||||
|
||||
public void setSign(String sign) { |
||||
this.sign = sign; |
||||
} |
||||
|
||||
public String getSignType() { |
||||
return signType; |
||||
} |
||||
|
||||
public void setSignType(String signType) { |
||||
this.signType = signType; |
||||
} |
||||
} |
@ -0,0 +1,150 @@ |
||||
package com.tsl3060.open.extend.core; |
||||
|
||||
import com.alibaba.fastjson2.annotation.JSONField; |
||||
|
||||
public class NotifyRequest { |
||||
@JSONField(name = "notify_id") |
||||
private String notifyId; |
||||
@JSONField(name = "app_id") |
||||
private String appId; |
||||
private String module = ""; |
||||
@JSONField(name = "source_id") |
||||
private String sourceId = ""; |
||||
@JSONField(name = "err_code") |
||||
private int errCode; |
||||
@JSONField(name = "err_msg") |
||||
private String errMsg; |
||||
@JSONField(name = "sub_err") |
||||
private int subErr; |
||||
@JSONField(name = "sub_msg") |
||||
private String subMsg; |
||||
@JSONField(name = "sign_type") |
||||
private String signType; |
||||
private Object payload; |
||||
@JSONField(name = "open_id") |
||||
private String openId; |
||||
private String description; |
||||
private String charset = "UTF-8"; |
||||
private String time; |
||||
private String sign; |
||||
|
||||
public String getNotifyId() { |
||||
return notifyId; |
||||
} |
||||
|
||||
public void setNotifyId(String notifyId) { |
||||
this.notifyId = notifyId; |
||||
} |
||||
|
||||
public String getAppId() { |
||||
return appId; |
||||
} |
||||
|
||||
public void setAppId(String appId) { |
||||
this.appId = appId; |
||||
} |
||||
|
||||
public String getModule() { |
||||
return module; |
||||
} |
||||
|
||||
public void setModule(String module) { |
||||
this.module = module; |
||||
} |
||||
|
||||
public String getSourceId() { |
||||
return sourceId; |
||||
} |
||||
|
||||
public void setSourceId(String sourceId) { |
||||
this.sourceId = sourceId; |
||||
} |
||||
|
||||
public int getErrCode() { |
||||
return errCode; |
||||
} |
||||
|
||||
public void setErrCode(int errCode) { |
||||
this.errCode = errCode; |
||||
} |
||||
|
||||
public String getErrMsg() { |
||||
return errMsg; |
||||
} |
||||
|
||||
public void setErrMsg(String errMsg) { |
||||
this.errMsg = errMsg; |
||||
} |
||||
|
||||
public int getSubErr() { |
||||
return subErr; |
||||
} |
||||
|
||||
public void setSubErr(int subErr) { |
||||
this.subErr = subErr; |
||||
} |
||||
|
||||
public String getSubMsg() { |
||||
return subMsg; |
||||
} |
||||
|
||||
public void setSubMsg(String subMsg) { |
||||
this.subMsg = subMsg; |
||||
} |
||||
|
||||
public String getSignType() { |
||||
return signType; |
||||
} |
||||
|
||||
public void setSignType(String signType) { |
||||
this.signType = signType; |
||||
} |
||||
|
||||
public Object getPayload() { |
||||
return payload; |
||||
} |
||||
|
||||
public void setPayload(Object payload) { |
||||
this.payload = payload; |
||||
} |
||||
|
||||
public String getOpenId() { |
||||
return openId; |
||||
} |
||||
|
||||
public void setOpenId(String openId) { |
||||
this.openId = openId; |
||||
} |
||||
|
||||
public String getDescription() { |
||||
return description; |
||||
} |
||||
|
||||
public void setDescription(String description) { |
||||
this.description = description; |
||||
} |
||||
|
||||
public String getCharset() { |
||||
return charset; |
||||
} |
||||
|
||||
public void setCharset(String charset) { |
||||
this.charset = charset; |
||||
} |
||||
|
||||
public String getTime() { |
||||
return time; |
||||
} |
||||
|
||||
public void setTime(String time) { |
||||
this.time = time; |
||||
} |
||||
|
||||
public String getSign() { |
||||
return sign; |
||||
} |
||||
|
||||
public void setSign(String sign) { |
||||
this.sign = sign; |
||||
} |
||||
} |
@ -0,0 +1,11 @@ |
||||
package com.tsl3060.open.extend.core.exception; |
||||
|
||||
public class ApiException extends Exception { |
||||
|
||||
public ApiException() { |
||||
} |
||||
|
||||
public ApiException(String message) { |
||||
super(message); |
||||
} |
||||
} |
@ -0,0 +1,21 @@ |
||||
package com.tsl3060.open.extend.core.exception; |
||||
|
||||
import java.io.IOException; |
||||
|
||||
public class BadResourceException extends IOException { |
||||
public BadResourceException() { |
||||
super(); |
||||
} |
||||
|
||||
public BadResourceException(String message) { |
||||
super(message); |
||||
} |
||||
|
||||
public BadResourceException(String message, Throwable cause) { |
||||
super(message, cause); |
||||
} |
||||
|
||||
public BadResourceException(Throwable cause) { |
||||
super(cause); |
||||
} |
||||
} |
@ -0,0 +1,110 @@ |
||||
package com.tsl3060.open.extend.core.notify; |
||||
|
||||
import com.alibaba.fastjson2.annotation.JSONField; |
||||
|
||||
public class CarbonOrderNotify { |
||||
/** |
||||
* 用户OpenID |
||||
*/ |
||||
private String openid; |
||||
/** |
||||
* 订单号 |
||||
*/ |
||||
@JSONField(name = "order_no") |
||||
private String orderNo; |
||||
/** |
||||
* 低碳订单号 |
||||
*/ |
||||
@JSONField(name = "carbon_no") |
||||
private String carbonNo; |
||||
|
||||
/** |
||||
* 用户低碳积分 |
||||
*/ |
||||
@JSONField(name = "carbon") |
||||
private double carbon; |
||||
|
||||
/** |
||||
* 本次订单新增积分 |
||||
*/ |
||||
@JSONField(name = "amount") |
||||
private double amount; |
||||
/** |
||||
* 订单时间 |
||||
*/ |
||||
@JSONField(name = "order_time") |
||||
private String orderTime; |
||||
/** |
||||
* 低碳积分订单完成时间 |
||||
*/ |
||||
@JSONField(name = "complete_time") |
||||
private String completeTime; |
||||
/** |
||||
* 订单类型 |
||||
*/ |
||||
private String type; |
||||
|
||||
public String getOpenid() { |
||||
return openid; |
||||
} |
||||
|
||||
public void setOpenid(String openid) { |
||||
this.openid = openid; |
||||
} |
||||
|
||||
public String getOrderNo() { |
||||
return orderNo; |
||||
} |
||||
|
||||
public void setOrderNo(String orderNo) { |
||||
this.orderNo = orderNo; |
||||
} |
||||
|
||||
public String getCarbonNo() { |
||||
return carbonNo; |
||||
} |
||||
|
||||
public void setCarbonNo(String carbonNo) { |
||||
this.carbonNo = carbonNo; |
||||
} |
||||
|
||||
public double getCarbon() { |
||||
return carbon; |
||||
} |
||||
|
||||
public void setCarbon(double carbon) { |
||||
this.carbon = carbon; |
||||
} |
||||
|
||||
public double getAmount() { |
||||
return amount; |
||||
} |
||||
|
||||
public void setAmount(double amount) { |
||||
this.amount = amount; |
||||
} |
||||
|
||||
public String getOrderTime() { |
||||
return orderTime; |
||||
} |
||||
|
||||
public void setOrderTime(String orderTime) { |
||||
this.orderTime = orderTime; |
||||
} |
||||
|
||||
public String getCompleteTime() { |
||||
return completeTime; |
||||
} |
||||
|
||||
public void setCompleteTime(String completeTime) { |
||||
this.completeTime = completeTime; |
||||
} |
||||
|
||||
public String getType() { |
||||
return type; |
||||
} |
||||
|
||||
public void setType(String type) { |
||||
this.type = type; |
||||
} |
||||
} |
@ -0,0 +1,20 @@ |
||||
package com.tsl3060.open.extend.core.notify; |
||||
|
||||
public class CarbonOrderNotifyAnswer implements IAnswer { |
||||
private boolean answer; |
||||
|
||||
public CarbonOrderNotifyAnswer() { |
||||
} |
||||
|
||||
public CarbonOrderNotifyAnswer(boolean answer) { |
||||
this.answer = answer; |
||||
} |
||||
|
||||
public boolean isAnswer() { |
||||
return answer; |
||||
} |
||||
|
||||
public void setAnswer(boolean answer) { |
||||
this.answer = answer; |
||||
} |
||||
} |
@ -0,0 +1,4 @@ |
||||
package com.tsl3060.open.extend.core.notify; |
||||
|
||||
public interface IAnswer { |
||||
} |
@ -0,0 +1,113 @@ |
||||
package com.tsl3060.open.extend.core.payload; |
||||
|
||||
import com.alibaba.fastjson2.annotation.JSONField; |
||||
import com.tsl3060.open.extend.core.IApiRequest; |
||||
|
||||
public class CarbonOrderPayload extends RequestPayload implements IApiRequest { |
||||
@Override |
||||
public String path() { |
||||
return "/v1/wanshun/wallet/carbon"; |
||||
} |
||||
|
||||
private String openid; |
||||
@JSONField(name = "order_no") |
||||
private String orderNo; |
||||
|
||||
private double mileage; |
||||
@JSONField(name = "order_time") |
||||
private String orderTime; |
||||
@JSONField(name = "order_state") |
||||
private String orderState; |
||||
@JSONField(name = "vehicle_model") |
||||
private String vehicleModel = ""; |
||||
@JSONField(name = "new_energy") |
||||
private boolean newEnergy = false; |
||||
@JSONField(name = "complete_time") |
||||
private String completeTime = ""; |
||||
|
||||
@JSONField(name = "order_pay") |
||||
private double orderPay = 0; |
||||
private String behavior = ""; |
||||
|
||||
|
||||
public String getOpenid() { |
||||
return openid; |
||||
} |
||||
|
||||
public void setOpenid(String openid) { |
||||
this.openid = openid; |
||||
} |
||||
|
||||
public String getOrderNo() { |
||||
return orderNo; |
||||
} |
||||
|
||||
public void setOrderNo(String orderNo) { |
||||
this.orderNo = orderNo; |
||||
} |
||||
|
||||
public double getMileage() { |
||||
return mileage; |
||||
} |
||||
|
||||
public void setMileage(double mileage) { |
||||
this.mileage = mileage; |
||||
} |
||||
|
||||
public String getOrderTime() { |
||||
return orderTime; |
||||
} |
||||
|
||||
public void setOrderTime(String orderTime) { |
||||
this.orderTime = orderTime; |
||||
} |
||||
|
||||
public boolean isNewEnergy() { |
||||
return newEnergy; |
||||
} |
||||
|
||||
public void setNewEnergy(boolean newEnergy) { |
||||
this.newEnergy = newEnergy; |
||||
} |
||||
|
||||
public String getOrderState() { |
||||
return orderState; |
||||
} |
||||
|
||||
public void setOrderState(String orderState) { |
||||
this.orderState = orderState; |
||||
} |
||||
|
||||
public String getVehicleModel() { |
||||
return vehicleModel; |
||||
} |
||||
|
||||
public void setVehicleModel(String vehicleModel) { |
||||
this.vehicleModel = vehicleModel; |
||||
} |
||||
|
||||
|
||||
public String getCompleteTime() { |
||||
return completeTime; |
||||
} |
||||
|
||||
public void setCompleteTime(String completeTime) { |
||||
this.completeTime = completeTime; |
||||
} |
||||
|
||||
public double getOrderPay() { |
||||
return orderPay; |
||||
} |
||||
|
||||
public void setOrderPay(double orderPay) { |
||||
this.orderPay = orderPay; |
||||
} |
||||
|
||||
public String getBehavior() { |
||||
return behavior; |
||||
} |
||||
|
||||
public void setBehavior(String behavior) { |
||||
this.behavior = behavior; |
||||
} |
||||
} |
@ -0,0 +1,13 @@ |
||||
package com.tsl3060.open.extend.core.payload; |
||||
|
||||
public abstract class RequestPayload { |
||||
private String source; |
||||
|
||||
public String getSource() { |
||||
return source; |
||||
} |
||||
|
||||
public void setSource(String source) { |
||||
this.source = source; |
||||
} |
||||
} |
@ -0,0 +1,35 @@ |
||||
package com.tsl3060.open.extend.core.payload; |
||||
|
||||
import com.alibaba.fastjson2.annotation.JSONField; |
||||
import com.tsl3060.open.extend.core.IApiRequest; |
||||
|
||||
/** |
||||
* 用户登录载体 |
||||
*/ |
||||
public class UserLoginPayload extends RequestPayload implements IApiRequest { |
||||
private String device; |
||||
|
||||
private String openid; |
||||
|
||||
|
||||
public String getDevice() { |
||||
return device; |
||||
} |
||||
|
||||
public void setDevice(String device) { |
||||
this.device = device; |
||||
} |
||||
|
||||
public String getOpenid() { |
||||
return openid; |
||||
} |
||||
|
||||
public void setOpenid(String openid) { |
||||
this.openid = openid; |
||||
} |
||||
|
||||
@Override |
||||
public String path() { |
||||
return "/v1/wanshun/account/login"; |
||||
} |
||||
} |
@ -0,0 +1,40 @@ |
||||
package com.tsl3060.open.extend.core.payload; |
||||
|
||||
import com.tsl3060.open.extend.core.IApiRequest; |
||||
import com.tsl3060.open.extend.core.payload.RequestPayload; |
||||
|
||||
public class UserRegisterRequestPayload extends RequestPayload implements IApiRequest { |
||||
@Override |
||||
public String path() { |
||||
return "/v1/wanshun/account/register"; |
||||
} |
||||
|
||||
private String phone; |
||||
private String uid; |
||||
private String type; |
||||
|
||||
|
||||
public String getPhone() { |
||||
return phone; |
||||
} |
||||
|
||||
public void setPhone(String phone) { |
||||
this.phone = phone; |
||||
} |
||||
|
||||
public String getUid() { |
||||
return uid; |
||||
} |
||||
|
||||
public void setUid(String uid) { |
||||
this.uid = uid; |
||||
} |
||||
|
||||
public String getType() { |
||||
return type; |
||||
} |
||||
|
||||
public void setType(String type) { |
||||
this.type = type; |
||||
} |
||||
} |
@ -0,0 +1,22 @@ |
||||
package com.tsl3060.open.extend.core.payload; |
||||
|
||||
import com.tsl3060.open.extend.core.IApiRequest; |
||||
|
||||
public class WalletQueryPayload extends RequestPayload implements IApiRequest { |
||||
|
||||
private String openid; |
||||
|
||||
|
||||
public String getOpenid() { |
||||
return openid; |
||||
} |
||||
|
||||
public void setOpenid(String openid) { |
||||
this.openid = openid; |
||||
} |
||||
|
||||
@Override |
||||
public String path() { |
||||
return "/v1/wanshun/wallet/query"; |
||||
} |
||||
} |
@ -0,0 +1,31 @@ |
||||
package com.tsl3060.open.extend.core.response; |
||||
|
||||
/** |
||||
* 验证码载体 |
||||
*/ |
||||
public class CaptchaResponse { |
||||
/** |
||||
* 验证码内容 |
||||
*/ |
||||
private String image; |
||||
/** |
||||
* 验证码ID |
||||
*/ |
||||
private String id; |
||||
|
||||
public String getImage() { |
||||
return image; |
||||
} |
||||
|
||||
public void setImage(String image) { |
||||
this.image = image; |
||||
} |
||||
|
||||
public String getId() { |
||||
return id; |
||||
} |
||||
|
||||
public void setId(String id) { |
||||
this.id = id; |
||||
} |
||||
} |
@ -0,0 +1,52 @@ |
||||
package com.tsl3060.open.extend.core.response; |
||||
|
||||
import com.alibaba.fastjson2.annotation.JSONField; |
||||
|
||||
public class CarbonOrderResponse { |
||||
private String openid; |
||||
private String carbon; |
||||
private double total; |
||||
private String record; |
||||
@JSONField(name = "record_time") |
||||
private String recordTime; |
||||
|
||||
public String getOpenid() { |
||||
return openid; |
||||
} |
||||
|
||||
public void setOpenid(String openid) { |
||||
this.openid = openid; |
||||
} |
||||
|
||||
public String getCarbon() { |
||||
return carbon; |
||||
} |
||||
|
||||
public void setCarbon(String carbon) { |
||||
this.carbon = carbon; |
||||
} |
||||
|
||||
public double getTotal() { |
||||
return total; |
||||
} |
||||
|
||||
public void setTotal(double total) { |
||||
this.total = total; |
||||
} |
||||
|
||||
public String getRecord() { |
||||
return record; |
||||
} |
||||
|
||||
public void setRecord(String record) { |
||||
this.record = record; |
||||
} |
||||
|
||||
public String getRecordTime() { |
||||
return recordTime; |
||||
} |
||||
|
||||
public void setRecordTime(String recordTime) { |
||||
this.recordTime = recordTime; |
||||
} |
||||
} |
@ -0,0 +1,44 @@ |
||||
package com.tsl3060.open.extend.core.response; |
||||
|
||||
import com.alibaba.fastjson2.annotation.JSONField; |
||||
|
||||
public class SmsSendResponse { |
||||
private String phone; |
||||
private String send; |
||||
private int time; |
||||
@JSONField(name = "send_time") |
||||
private String sendTime; |
||||
|
||||
public String getPhone() { |
||||
return phone; |
||||
} |
||||
|
||||
public void setPhone(String phone) { |
||||
this.phone = phone; |
||||
} |
||||
|
||||
public String getSend() { |
||||
return send; |
||||
} |
||||
|
||||
public void setSend(String send) { |
||||
this.send = send; |
||||
} |
||||
|
||||
public int getTime() { |
||||
return time; |
||||
} |
||||
|
||||
public void setTime(int time) { |
||||
this.time = time; |
||||
} |
||||
|
||||
public String getSendTime() { |
||||
return sendTime; |
||||
} |
||||
|
||||
public void setSendTime(String sendTime) { |
||||
this.sendTime = sendTime; |
||||
} |
||||
} |
||||
|
@ -0,0 +1,35 @@ |
||||
package com.tsl3060.open.extend.core.response; |
||||
|
||||
import com.alibaba.fastjson2.annotation.JSONField; |
||||
|
||||
public class UserLoginResponse { |
||||
@JSONField(name = "expire_at") |
||||
private String expireAt; |
||||
private String openid; |
||||
private String token; |
||||
|
||||
|
||||
public String getOpenid() { |
||||
return openid; |
||||
} |
||||
|
||||
public void setOpenid(String openid) { |
||||
this.openid = openid; |
||||
} |
||||
|
||||
public String getToken() { |
||||
return token; |
||||
} |
||||
|
||||
public void setToken(String token) { |
||||
this.token = token; |
||||
} |
||||
|
||||
public String getExpireAt() { |
||||
return expireAt; |
||||
} |
||||
|
||||
public void setExpireAt(String expireAt) { |
||||
this.expireAt = expireAt; |
||||
} |
||||
} |
@ -0,0 +1,42 @@ |
||||
package com.tsl3060.open.extend.core.response; |
||||
|
||||
public class UserRegisterResponse { |
||||
private String nickname; |
||||
private String avatar; |
||||
private String openid; |
||||
|
||||
private String appid; |
||||
|
||||
public String getNickname() { |
||||
return nickname; |
||||
} |
||||
|
||||
public void setNickname(String nickname) { |
||||
this.nickname = nickname; |
||||
} |
||||
|
||||
public String getAvatar() { |
||||
return avatar; |
||||
} |
||||
|
||||
public void setAvatar(String avatar) { |
||||
this.avatar = avatar; |
||||
} |
||||
|
||||
public String getOpenid() { |
||||
return openid; |
||||
} |
||||
|
||||
public void setOpenid(String openid) { |
||||
this.openid = openid; |
||||
} |
||||
|
||||
|
||||
public String getAppid() { |
||||
return appid; |
||||
} |
||||
|
||||
public void setAppid(String appid) { |
||||
this.appid = appid; |
||||
} |
||||
} |
@ -0,0 +1,32 @@ |
||||
package com.tsl3060.open.extend.core.response; |
||||
|
||||
public class WalletQueryResponse { |
||||
private String openid; |
||||
private String carbon; |
||||
|
||||
private String query; |
||||
|
||||
public String getOpenid() { |
||||
return openid; |
||||
} |
||||
|
||||
public void setOpenid(String openid) { |
||||
this.openid = openid; |
||||
} |
||||
|
||||
public String getCarbon() { |
||||
return carbon; |
||||
} |
||||
|
||||
public void setCarbon(String carbon) { |
||||
this.carbon = carbon; |
||||
} |
||||
|
||||
public String getQuery() { |
||||
return query; |
||||
} |
||||
|
||||
public void setQuery(String query) { |
||||
this.query = query; |
||||
} |
||||
} |
@ -0,0 +1,10 @@ |
||||
package com.tsl3060.open.extend.core.router; |
||||
|
||||
import com.tsl3060.open.extend.core.NotifyRequest; |
||||
import com.tsl3060.open.extend.core.notify.IAnswer; |
||||
|
||||
public interface INotifyRouter { |
||||
String path(); |
||||
|
||||
IAnswer makeBody(NotifyRequest notifyRequest) throws Exception; |
||||
} |
@ -0,0 +1,30 @@ |
||||
package com.tsl3060.open.extend.core.router; |
||||
|
||||
import cn.hutool.core.util.ClassUtil; |
||||
import cn.hutool.core.util.ReflectUtil; |
||||
import com.tsl3060.open.extend.core.INotifyListener; |
||||
|
||||
import java.util.HashMap; |
||||
import java.util.Map; |
||||
import java.util.Set; |
||||
|
||||
public class NotifyMapRouter { |
||||
private Map<String, INotifyRouter> map = new HashMap<>(); |
||||
|
||||
public NotifyMapRouter(INotifyListener notifyListener) { |
||||
Set<Class<?>> classes = ClassUtil.scanPackage("com.tsl3060.open.extend.core.router.notify"); |
||||
|
||||
for (Class<?> c : classes) { |
||||
INotifyRouter notifyRouter = (INotifyRouter) ReflectUtil.newInstance(c,notifyListener); |
||||
map.put(notifyRouter.path(), notifyRouter); |
||||
} |
||||
|
||||
} |
||||
|
||||
|
||||
public INotifyRouter getRouter(String module) { |
||||
return map.get(module); |
||||
} |
||||
|
||||
|
||||
} |
@ -0,0 +1,29 @@ |
||||
package com.tsl3060.open.extend.core.router.notify; |
||||
|
||||
import com.alibaba.fastjson2.JSON; |
||||
import com.tsl3060.open.extend.core.INotifyListener; |
||||
import com.tsl3060.open.extend.core.NotifyRequest; |
||||
import com.tsl3060.open.extend.core.notify.CarbonOrderNotify; |
||||
import com.tsl3060.open.extend.core.notify.IAnswer; |
||||
import com.tsl3060.open.extend.core.router.INotifyRouter; |
||||
|
||||
public class CarbonOrderNotifyRouter implements INotifyRouter { |
||||
|
||||
|
||||
private final INotifyListener notifyListener; |
||||
|
||||
public CarbonOrderNotifyRouter(INotifyListener notifyListener) { |
||||
this.notifyListener = notifyListener; |
||||
} |
||||
|
||||
@Override |
||||
public String path() { |
||||
return "/v1/wanshun/notify/carbon"; |
||||
} |
||||
|
||||
@Override |
||||
public IAnswer makeBody(NotifyRequest notifyRequest) throws Exception { |
||||
CarbonOrderNotify carbonOrderNotify = JSON.to(CarbonOrderNotify.class, notifyRequest.getPayload()); |
||||
return this.notifyListener.carbon(carbonOrderNotify); |
||||
} |
||||
} |
@ -0,0 +1,47 @@ |
||||
package com.tsl3060.open.extend.core.secure; |
||||
|
||||
import com.tsl3060.open.extend.core.*; |
||||
|
||||
public interface ISecure { |
||||
/** |
||||
* 参数设置 |
||||
* |
||||
* @param config |
||||
*/ |
||||
void setConfig(Config config); |
||||
|
||||
/** |
||||
* 安全模块名称 |
||||
* |
||||
* @return 名称 |
||||
*/ |
||||
String name(); |
||||
|
||||
/** |
||||
* 对请求进行签名 |
||||
* |
||||
* @param apiRequest 请求 |
||||
* @return 签名字符串 |
||||
*/ |
||||
String requestSign(ApiRequest apiRequest); |
||||
|
||||
/** |
||||
* 同步返回验签 |
||||
* |
||||
* @param apiResponse 同步返回 |
||||
* @return 是否正确 |
||||
*/ |
||||
boolean verifyResponse(ApiResponse apiResponse); |
||||
|
||||
/** |
||||
* 异步通知验签 |
||||
* |
||||
* @param notifyRequest 异步通知 |
||||
* @return 是否正确 |
||||
*/ |
||||
boolean verifyNotify(NotifyRequest notifyRequest); |
||||
|
||||
|
||||
String answerSign(NotifyAnswerResponse response); |
||||
|
||||
} |
@ -0,0 +1,220 @@ |
||||
package com.tsl3060.open.extend.core.secure; |
||||
|
||||
import cn.hutool.core.collection.CollectionUtil; |
||||
import cn.hutool.core.util.HexUtil; |
||||
import cn.hutool.log.Log; |
||||
import cn.hutool.log.LogFactory; |
||||
import com.alibaba.fastjson2.JSONObject; |
||||
import com.tsl3060.open.extend.core.*; |
||||
|
||||
import java.io.UnsupportedEncodingException; |
||||
import java.math.BigDecimal; |
||||
import java.nio.charset.Charset; |
||||
import java.security.*; |
||||
import java.security.spec.InvalidKeySpecException; |
||||
import java.security.spec.PKCS8EncodedKeySpec; |
||||
import java.security.spec.X509EncodedKeySpec; |
||||
import java.util.*; |
||||
import java.util.stream.Collectors; |
||||
|
||||
public class RSASecure implements ISecure { |
||||
|
||||
private final Log log = LogFactory.get(); |
||||
private static final String SIGNATURE_ALGORITHM = "SHA256withRSA"; |
||||
private static final String DIGEST_ALGORITHM = "SHA-256"; |
||||
public static final String NAME = "RSA"; |
||||
private Config config; |
||||
|
||||
@Override |
||||
public void setConfig(Config config) { |
||||
this.config = config; |
||||
} |
||||
|
||||
@Override |
||||
public String name() { |
||||
return NAME; |
||||
} |
||||
|
||||
private String object2LinkStr(Object ob) { |
||||
List<String> payloadStr = new ArrayList<>(); |
||||
if (ob != null) { |
||||
JSONObject j = JSONObject.from(ob); |
||||
//排序
|
||||
Set<String> _keys = j.keySet(); |
||||
List<String> keys = _keys.stream().sorted(Comparator.naturalOrder()).collect(Collectors.toList()); |
||||
|
||||
for (String key : keys) { |
||||
if (j.get(key) != null) { |
||||
Object v = j.get(key); |
||||
String vx; |
||||
if (v instanceof Double) { |
||||
vx = String.format("%.2f", v); |
||||
} else if (v instanceof Float) { |
||||
vx = String.format("%.2f", v); |
||||
} else if (v instanceof Boolean) { |
||||
vx = String.format("%s", v); |
||||
} else if (v instanceof BigDecimal) { |
||||
vx = String.format("%.2f", v); |
||||
} else { |
||||
vx = String.valueOf(v); |
||||
} |
||||
payloadStr.add(String.format("%s=%s", key, vx)); |
||||
} |
||||
} |
||||
} |
||||
return CollectionUtil.join(payloadStr, "&"); |
||||
} |
||||
|
||||
public String sign(String content, String charset) { |
||||
|
||||
log.debug("req: {}", content); |
||||
try { |
||||
byte[] bytesContent = content.getBytes(charset); |
||||
MessageDigest messageDigest = MessageDigest.getInstance(DIGEST_ALGORITHM); |
||||
byte[] digestData = messageDigest.digest(bytesContent); |
||||
//签名工具
|
||||
byte[] pKeyData = Base64.getDecoder().decode(this.config.getPrivateKey()); |
||||
PKCS8EncodedKeySpec spec = new PKCS8EncodedKeySpec(pKeyData); |
||||
KeyFactory keyFactory = KeyFactory.getInstance(NAME); |
||||
PrivateKey pkey = keyFactory.generatePrivate(spec); |
||||
|
||||
Signature signature = Signature.getInstance(SIGNATURE_ALGORITHM); |
||||
signature.initSign(pkey); |
||||
signature.update(digestData); |
||||
byte[] s = signature.sign(); |
||||
return HexUtil.encodeHexStr(s); |
||||
|
||||
} catch (UnsupportedEncodingException | NoSuchAlgorithmException | InvalidKeySpecException | |
||||
SignatureException | |
||||
InvalidKeyException e) { |
||||
throw new RuntimeException(e); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* 对请求生成签名 |
||||
* |
||||
* @param request 请求 |
||||
* @return |
||||
*/ |
||||
@Override |
||||
public String requestSign(ApiRequest request) { |
||||
Object ob = request.getPayload(); |
||||
String waitStr = this.object2LinkStr(ob); |
||||
String content = String.format("%s&%s&%s&%s&%s&%s&%s", |
||||
request.getPath(), |
||||
request.getAppId(), |
||||
request.getAccessToken(), |
||||
request.getSignType(), |
||||
request.getCharset(), |
||||
request.getTime(), |
||||
waitStr |
||||
); |
||||
log.debug("req: {}", content); |
||||
|
||||
return this.sign(content, request.getCharset()); |
||||
|
||||
} |
||||
|
||||
|
||||
private boolean responseServerVerify(byte[] content, String sign) { |
||||
try { |
||||
MessageDigest messageDigest = MessageDigest.getInstance(DIGEST_ALGORITHM); |
||||
byte[] digestData = messageDigest.digest(content); |
||||
byte[] pubKeyData = Base64.getDecoder().decode(this.config.getApiPublicKey()); |
||||
X509EncodedKeySpec spec = new X509EncodedKeySpec(pubKeyData); |
||||
KeyFactory keyFactory = KeyFactory.getInstance(NAME); |
||||
PublicKey pubKey = keyFactory.generatePublic(spec); |
||||
Signature signature = Signature.getInstance(SIGNATURE_ALGORITHM); |
||||
signature.initVerify(pubKey); |
||||
signature.update(digestData); |
||||
|
||||
byte[] bodySign = HexUtil.decodeHex(sign); |
||||
return signature.verify(bodySign); |
||||
} catch (NoSuchAlgorithmException | InvalidKeySpecException | SignatureException | InvalidKeyException e) { |
||||
log.error(e); |
||||
return false; |
||||
} |
||||
} |
||||
|
||||
@Override |
||||
public boolean verifyResponse(ApiResponse response) { |
||||
String waitStr = this.object2LinkStr(response.getPayload()); |
||||
String formatStr = String.format( |
||||
"%s&%s&%s&%s&%s&%s&%s&%s&%s&%s&%s", |
||||
response.getResponseId(), |
||||
response.getErrCode(), |
||||
response.getErrMsg(), |
||||
response.getSubErr(), |
||||
response.getSubMsg(), |
||||
response.getTime(), |
||||
response.getOpenId(), |
||||
response.getSignType(), |
||||
response.getCharset(), |
||||
response.getDescription(), |
||||
waitStr |
||||
); |
||||
log.debug("反馈内容 {}", formatStr); |
||||
log.debug("签名内容 {}", response.getSign()); |
||||
Charset bodyCharset = Charset.forName(response.getCharset()); |
||||
byte[] bytesContent = formatStr.getBytes(bodyCharset); |
||||
return this.responseServerVerify(bytesContent, response.getSign()); |
||||
|
||||
} |
||||
|
||||
/** |
||||
* 验证通知 |
||||
* |
||||
* @param response 异步通知 |
||||
* @return |
||||
*/ |
||||
@Override |
||||
public boolean verifyNotify(NotifyRequest response) { |
||||
String waitStr = this.object2LinkStr(response.getPayload()); |
||||
String formatStr = String.format( |
||||
"%s&%s&%s&%s&%s&%s&%s&%s&%s&%s&%s&%s&%s", |
||||
response.getNotifyId(), |
||||
response.getSourceId(), |
||||
response.getAppId(), |
||||
response.getErrCode(), |
||||
response.getErrMsg(), |
||||
response.getSubErr(), |
||||
response.getSubMsg(), |
||||
response.getTime(), |
||||
response.getOpenId(), |
||||
response.getSignType(), |
||||
response.getCharset(), |
||||
response.getDescription(), |
||||
waitStr |
||||
); |
||||
log.debug("通知内容 {}", formatStr); |
||||
log.debug("通知签名内容 {}", response.getSign()); |
||||
Charset bodyCharset = Charset.forName(response.getCharset()); |
||||
byte[] bytesContent = formatStr.getBytes(bodyCharset); |
||||
|
||||
return this.responseServerVerify(bytesContent, response.getSign()); |
||||
|
||||
} |
||||
|
||||
/** |
||||
* 通知反馈签名 |
||||
* |
||||
* @param response |
||||
* @return |
||||
*/ |
||||
@Override |
||||
public String answerSign(NotifyAnswerResponse response) { |
||||
Object ob = response.getPayload(); |
||||
String waitStr = this.object2LinkStr(ob); |
||||
String content = String.format("%s&%s&%s&%s&%s&%s&%s", |
||||
response.getAnswerId(), |
||||
response.getAppId(), |
||||
response.getResult(), |
||||
response.getSignType(), |
||||
response.getCharset(), |
||||
response.getTime(), |
||||
waitStr |
||||
); |
||||
return this.sign(content, response.getCharset()); |
||||
} |
||||
} |
@ -0,0 +1,24 @@ |
||||
package com.tsl3060.open.extend.core.secure; |
||||
|
||||
import com.tsl3060.open.extend.core.Config; |
||||
|
||||
import java.util.HashMap; |
||||
|
||||
/** |
||||
* |
||||
*/ |
||||
public class SecureTool { |
||||
|
||||
private final HashMap<String, ISecure> secureHashMap = new HashMap<>(); |
||||
|
||||
public SecureTool(Config config) { |
||||
RSASecure rsaSecure = new RSASecure(); |
||||
rsaSecure.setConfig(config); |
||||
secureHashMap.put(rsaSecure.name(), rsaSecure); |
||||
} |
||||
|
||||
public ISecure getSecure(String name) { |
||||
return secureHashMap.get(name); |
||||
} |
||||
|
||||
} |
@ -0,0 +1,76 @@ |
||||
<?xml version="1.0" encoding="utf-8" ?> |
||||
<!-- 从高到地低 OFF 、 FATAL 、 ERROR 、 WARN 、 INFO 、 DEBUG 、 TRACE 、 ALL --> |
||||
<!-- 日志输出规则 根据当前ROOT 级别,日志输出时,级别高于root默认的级别时 会输出 --> |
||||
<!-- 以下 每个配置的 filter 是过滤掉输出文件里面,会出现高级别文件,依然出现低级别的日志信息,通过filter 过滤只记录本级别的日志--> |
||||
|
||||
<!-- 属性描述 scan:性设置为true时,配置文件如果发生改变,将会被重新加载,默认值为true scanPeriod:设置监测配置文件是否有修改的时间间隔,如果没有给出时间单位, |
||||
默认单位是毫秒。当scan为true时,此属性生效。默认的时间间隔为1分钟。debug:当此属性设置为true时,将打印出logback内部日志信息,实时查看logback运行状态。默认值为false。 --> |
||||
<configuration scan="true" scanPeriod="60 seconds" debug="false"> |
||||
<!-- 定义日志文件 输入位置 --> |
||||
<property name="logPath" value="logs"/> |
||||
<!-- 日志最大的历史 30天 --> |
||||
<property name="maxHistory" value="3"/> |
||||
|
||||
<!-- 配置项, 通过此节点配置日志输出位置(控制台、文件、数据库)、输出格式等--> |
||||
<!-- ConsoleAppender代表输出到控制台 --> |
||||
<appender name="consoleLog" class="ch.qos.logback.core.ConsoleAppender"> |
||||
<!-- layout代表输出格式 --> |
||||
<layout class="ch.qos.logback.classic.PatternLayout"> |
||||
<pattern>%d{yyyy-MM-dd HH:mm:ss} [%thread] %-5level %logger -%msg%n</pattern> |
||||
</layout> |
||||
</appender> |
||||
<!-- 日志输出文件 --> |
||||
<appender name="fileInfoLog" class="ch.qos.logback.core.rolling.RollingFileAppender"> |
||||
<encoder> |
||||
<pattern>%d{yyyy-MM-dd HH:mm:ss} [%thread] %-5level %logger -%msg%n</pattern> |
||||
</encoder> |
||||
<!-- 滚动记录文件,先将日志记录到指定文件,当符合某个条件时,将日志记录到其他文件 RollingFileAppender--> |
||||
<!-- 滚动策略,它根据时间来制定滚动策略.既负责滚动也负责触发滚动 --> |
||||
<rollingPolicy class="ch.qos.logback.core.rolling.TimeBasedRollingPolicy"> |
||||
<!-- 输出路径 --> |
||||
<fileNamePattern>${logPath}/info/%d.log</fileNamePattern> |
||||
<!-- 可选节点,控制保留的归档文件的最大数量,超出数量就删除旧文件假设设置每个月滚动,且<maxHistory>是6, |
||||
则只保存最近6个月的文件,删除之前的旧文件。注意,删除旧文件是,那些为了归档而创建的目录也会被删除--> |
||||
<maxHistory>${maxHistory}</maxHistory> |
||||
</rollingPolicy> |
||||
<!-- 按照固定窗口模式生成日志文件,当文件大于20MB时,生成新的日志文件。窗口大小是1到3,当保存了3个归档文件后,将覆盖最早的日志。 |
||||
<rollingPolicy class="ch.qos.logback.core.rolling.FixedWindowRollingPolicy"> |
||||
<fileNamePattern>${logPath}/%d{yyyy-MM-dd}/.log.zip</fileNamePattern> |
||||
<minIndex>1</minIndex> |
||||
<maxIndex>3</maxIndex> |
||||
</rollingPolicy> --> |
||||
<!-- 查看当前活动文件的大小,如果超过指定大小会告知RollingFileAppender 触发当前活动文件滚动 |
||||
<triggeringPolicy class="ch.qos.logback.core.rolling.SizeBasedTriggeringPolicy"> |
||||
<maxFileSize>5MB</maxFileSize> |
||||
</triggeringPolicy> --> |
||||
</appender> |
||||
<!-- 特殊记录Error日志 --> |
||||
<appender name="fileErrorLog" class="ch.qos.logback.core.rolling.RollingFileAppender"> |
||||
<!-- 只记录ERROR级别日志,添加范围过滤,可以将该类型的日志特殊记录到某个位置 --> |
||||
<filter class="ch.qos.logback.classic.filter.ThresholdFilter"> |
||||
<level>ERROR</level> |
||||
</filter> |
||||
<encoder> |
||||
<pattern>%d{yyyy-MM-dd HH:mm:ss} [%thread] %-5level %logger -%msg%n</pattern> |
||||
</encoder> |
||||
<rollingPolicy class="ch.qos.logback.core.rolling.TimeBasedRollingPolicy"> |
||||
<fileNamePattern>${logPath}/error/%d.log</fileNamePattern> |
||||
<!-- 日志最大的历史 60天 --> |
||||
<maxHistory>3</maxHistory> |
||||
</rollingPolicy> |
||||
</appender> |
||||
|
||||
<logger name="io.lettuce" level="OFF"/> |
||||
<logger name="io.undertow.request" level="OFF"/> |
||||
<logger name="com.zaxxer" level="ERROR"/> |
||||
<logger name="io.netty" level="OFF"/> |
||||
<logger name="org.tio.core.task" level="OFF"/> |
||||
<!-- 根节点,表名基本的日志级别,里面可以由多个appender规则 --> |
||||
<!-- level="info"代表基础日志级别为info --> |
||||
<root level="debug"> |
||||
<!-- 引入控制台输出规则 --> |
||||
<appender-ref ref="consoleLog"/> |
||||
<!-- <appender-ref ref="fileInfoLog"/>--> |
||||
<!-- <appender-ref ref="fileErrorLog"/>--> |
||||
</root> |
||||
</configuration> |
Loading…
Reference in new issue