From debb7b1ef8bb4e96e09b2d878ad821b4b2f3a110 Mon Sep 17 00:00:00 2001
From: xin <1099200748@qq.com>
Date: Fri, 07 Nov 2025 22:00:17 +0800
Subject: [PATCH] 短信api更新维护(未完成)
---
oying-system/src/main/resources/config/application-dev.yml | 8
oying-system/src/main/java/com/oying/modules/winnerlook/util/Md5Util.java | 78 +++++++++++
oying-system/src/main/java/com/oying/modules/security/utils/WinnerLookEnum.java | 2
oying-system/src/main/java/com/oying/modules/winnerlook/client/OkHttpService.java | 201 ++++++++++++++++++++++++++++
oying-system/src/main/java/com/oying/modules/security/utils/WinnerLookProperties.java | 3
oying-system/src/main/resources/config/application-prod.yml | 8
oying-system/src/main/java/com/oying/modules/winnerlook/client/BaseApi.java | 48 ++++++
oying-system/src/main/java/com/oying/modules/security/rest/VerificationController.java | 4
oying-system/src/main/resources/config/application.yml | 2
oying-system/src/main/java/com/oying/modules/security/utils/SendMessageUtils.java | 24 +++
10 files changed, 363 insertions(+), 15 deletions(-)
diff --git a/oying-system/src/main/java/com/oying/modules/security/rest/VerificationController.java b/oying-system/src/main/java/com/oying/modules/security/rest/VerificationController.java
index c7bc1c7..d127e75 100644
--- a/oying-system/src/main/java/com/oying/modules/security/rest/VerificationController.java
+++ b/oying-system/src/main/java/com/oying/modules/security/rest/VerificationController.java
@@ -4,8 +4,8 @@
import com.oying.annotation.rest.AnonymousGetMapping;
import com.oying.utils.R;
import com.oying.utils.RedisUtils;
-import com.oying.utils.SendMessageUtils;
-import com.oying.utils.WinnerLookProperties;
+import com.oying.modules.security.utils.SendMessageUtils;
+import com.oying.modules.security.utils.WinnerLookProperties;
import io.swagger.annotations.Api;
import io.swagger.annotations.ApiOperation;
import lombok.RequiredArgsConstructor;
diff --git a/oying-tools/src/main/java/com/oying/utils/SendMessageUtils.java b/oying-system/src/main/java/com/oying/modules/security/utils/SendMessageUtils.java
similarity index 64%
rename from oying-tools/src/main/java/com/oying/utils/SendMessageUtils.java
rename to oying-system/src/main/java/com/oying/modules/security/utils/SendMessageUtils.java
index 8be1331..ce8d382 100644
--- a/oying-tools/src/main/java/com/oying/utils/SendMessageUtils.java
+++ b/oying-system/src/main/java/com/oying/modules/security/utils/SendMessageUtils.java
@@ -1,6 +1,8 @@
-package com.oying.utils;
+package com.oying.modules.security.utils;
+import cn.hutool.core.util.IdUtil;
import com.oying.exception.BadRequestException;
+import com.oying.utils.HttpRequest;
import lombok.Data;
import lombok.extern.slf4j.Slf4j;
import org.springframework.http.HttpMethod;
@@ -24,8 +26,26 @@
@Data
public class SendMessageUtils {
- public static final String SIGN = "【立研】";
+ public static final String SIGN = "【哦应科技】";
public static final String MESSAGE = "您的验证码为:{code},请勿泄露于他人!";
+ public static void main(String[] args) {
+ System.out.println(IdUtil.simpleUUID());
+ Map<String, String> params = new HashMap<>();
+ params.put("userCode", "SHOYYJ");
+ params.put("userPass", "Oy250928");
+ params.put("DesNo", "15213186640");
+ params.put("Msg", "【哦应科技】您的验证码为:784121,请勿泄露于他人!");
+ params.put("smsType", "101");
+ String str = extractWithRegex(HttpRequest.exchangeMsg(HttpMethod.POST, "https://118.178.116.15:8443/winnerrxd/api/trigger/SendMsg", convert(params)));
+ if (ObjectUtils.isEmpty(str)) {
+ log.error("短信调用异常 {}", str);
+ throw new BadRequestException("短信调用异常");
+ }
+ long i = Long.parseLong(str);
+ if (i < 0) {
+ throw new BadRequestException(WinnerLookEnum.find(str));
+ }
+ }
public static void sendMsg(String url, String phone, String sign, String message, WinnerLookProperties properties) {
Map<String, String> params = new HashMap<>();
diff --git a/oying-tools/src/main/java/com/oying/utils/WinnerLookEnum.java b/oying-system/src/main/java/com/oying/modules/security/utils/WinnerLookEnum.java
similarity index 97%
rename from oying-tools/src/main/java/com/oying/utils/WinnerLookEnum.java
rename to oying-system/src/main/java/com/oying/modules/security/utils/WinnerLookEnum.java
index 73429de..5c74142 100644
--- a/oying-tools/src/main/java/com/oying/utils/WinnerLookEnum.java
+++ b/oying-system/src/main/java/com/oying/modules/security/utils/WinnerLookEnum.java
@@ -1,4 +1,4 @@
-package com.oying.utils;
+package com.oying.modules.security.utils;
import lombok.AllArgsConstructor;
import lombok.Getter;
diff --git a/oying-tools/src/main/java/com/oying/utils/WinnerLookProperties.java b/oying-system/src/main/java/com/oying/modules/security/utils/WinnerLookProperties.java
similarity index 90%
rename from oying-tools/src/main/java/com/oying/utils/WinnerLookProperties.java
rename to oying-system/src/main/java/com/oying/modules/security/utils/WinnerLookProperties.java
index 7d7a0f3..df2e4b9 100644
--- a/oying-tools/src/main/java/com/oying/utils/WinnerLookProperties.java
+++ b/oying-system/src/main/java/com/oying/modules/security/utils/WinnerLookProperties.java
@@ -1,4 +1,4 @@
-package com.oying.utils;
+package com.oying.modules.security.utils;
import lombok.Data;
import org.springframework.boot.context.properties.ConfigurationProperties;
@@ -22,6 +22,7 @@
* 每次不要超过50个号码
*/
private String urlSendMsgBatch;
+ private String token;
private String userCode;
private String userPass;
}
diff --git a/oying-system/src/main/java/com/oying/modules/winnerlook/client/BaseApi.java b/oying-system/src/main/java/com/oying/modules/winnerlook/client/BaseApi.java
new file mode 100644
index 0000000..7e86dd6
--- /dev/null
+++ b/oying-system/src/main/java/com/oying/modules/winnerlook/client/BaseApi.java
@@ -0,0 +1,48 @@
+package com.oying.modules.winnerlook.client;
+
+import lombok.extern.slf4j.Slf4j;
+import org.apache.commons.lang3.StringUtils;
+
+import java.util.HashMap;
+import java.util.Map;
+
+/**
+ * @author MinaWu
+ * @description
+ * @date 2025/5/14 16:41
+ */
+@Slf4j
+public class BaseApi {
+ private static final String API_V1_TRIGGER_SEND_MSG = "/api/v1/trigger/SendMsg";
+ private static final String API_V1_TRIGGER_SEND_BATCH_MSG = "/api/v1/trigger/SendBatchMsg";
+
+ public static String TRIGGER = "101";
+
+ /**
+ * 发送短信
+ *
+ * @param userCode 登录名称
+ * @param token token
+ * @param desNo 手机号
+ * @param msg 内容
+ * @param autograph 签名编号
+ * @param customerUuid 用户标识字段
+ * @return String
+ */
+ public static String sendMsgByPost(String userCode, String token, String desNo, String msg, String autograph,
+ String customerUuid, String baseUrl) {
+ Map<String, Object> params = new HashMap<>();
+ params.put("userCode", userCode);
+ params.put("DesNo", desNo);
+ params.put("Msg", msg);
+ params.put("smsType", TRIGGER);
+ if (StringUtils.isNotBlank(autograph)) {
+ params.put("autograph", autograph);
+ }
+ if (StringUtils.isNotBlank(customerUuid)) {
+ params.put("customerUuid", customerUuid);
+ }
+ String xmlResponse = OkHttpService.doPost(params, baseUrl, userCode, token);
+ return OkHttpService.parseXmlResponse(xmlResponse);
+ }
+}
diff --git a/oying-system/src/main/java/com/oying/modules/winnerlook/client/OkHttpService.java b/oying-system/src/main/java/com/oying/modules/winnerlook/client/OkHttpService.java
new file mode 100644
index 0000000..fdf7ff5
--- /dev/null
+++ b/oying-system/src/main/java/com/oying/modules/winnerlook/client/OkHttpService.java
@@ -0,0 +1,201 @@
+package com.oying.modules.winnerlook.client;
+
+import cn.hutool.core.util.IdUtil;
+import com.alibaba.fastjson2.JSONArray;
+import com.alibaba.fastjson2.JSONObject;
+import com.oying.exception.BadRequestException;
+import com.oying.modules.winnerlook.util.Md5Util;
+import lombok.extern.slf4j.Slf4j;
+import org.w3c.dom.Document;
+import org.w3c.dom.Element;
+import org.w3c.dom.NodeList;
+import shade.okhttp3.*;
+
+import javax.xml.parsers.DocumentBuilder;
+import javax.xml.parsers.DocumentBuilderFactory;
+import java.io.ByteArrayInputStream;
+import java.nio.charset.StandardCharsets;
+import java.util.Map;
+import java.util.Set;
+import java.util.TreeMap;
+import java.util.UUID;
+
+/**
+ * @author MinaWu
+ * @description
+ * @date 2025/5/14 16:16
+ */
+
+@Slf4j
+public class OkHttpService {
+ private static final String SECRET_KEY = "X-Winner-Secret-Key";
+ private static final String TIMESTAMP_HEADER = "X-Winner-Timestamp";
+ private static final String SIGNATURE_HEADER = "X-Winner-Signature";
+ private static final String REQUEST_ID = "X-Winner-Request-Id";
+ private static final String TOKEN = "X-Winner-token";
+ private static final String PARAMS = "params";
+
+ /**
+ * 生成 requestId
+ *
+ * @param timestamp 时间戳
+ * @return requestId
+ */
+ private static String generateRequestId(String userName, long timestamp) {
+ return userName.replace("_", "") + "_" + IdUtil.simpleUUID() + "_" + timestamp;
+ }
+
+ public static String doPost(Map<String, Object> params, String baseUrl, String userName, String token) {
+ String secretKey = UUID.randomUUID().toString().replace("-", "");
+ String boundary = UUID.randomUUID().toString().replace("-", "");
+
+ // 生成时间戳
+ long timestamp = System.currentTimeMillis();
+ String timestampStr = String.valueOf(timestamp);
+ // 生成 requestId
+ String requestId = generateRequestId(userName, timestamp);
+
+ // 生成签名
+ String signature = generateSignature(params, timestampStr, secretKey, requestId, token);
+
+ // 创建请求体
+ MultipartBody.Builder bodyBuilder = new MultipartBody.Builder(boundary)
+ .setType(MultipartBody.FORM);
+ for (Map.Entry<String, Object> entry : params.entrySet()) {
+ bodyBuilder.addFormDataPart(entry.getKey(), entry.getValue().toString());
+ }
+ // 创建请求
+ Request request = new Request.Builder()
+ .url(baseUrl)
+ .header(TIMESTAMP_HEADER, timestampStr)
+ .header(SIGNATURE_HEADER, signature)
+ .header(REQUEST_ID, requestId)
+ .header(SECRET_KEY, secretKey)
+ .header("Content-Type", "multipart/form-data;charset=UTF-8; boundary=" + boundary)
+ .post(bodyBuilder.build())
+ .build();
+ // 发送请求
+ try {
+ return "response";
+ } catch (Exception e) {
+ log.info("doPost requestId:{},body:{},headers:{} ", requestId, request.body().toString(), request.headers().toString());
+ throw new BadRequestException("短信POST:" + e.getMessage());
+ }
+ }
+
+ /**
+ * 对 treeMap 的键进行排序
+ */
+ private static void sortParams(Map<String, Object> treeMap) {
+ for (String key : treeMap.keySet()) {
+ Object value = treeMap.get(key);
+ if (value instanceof JSONObject) {
+ // 如果是 JSONObject,递归排序
+ treeMap.put(key, sortJsonObject((JSONObject) value));
+ } else if (value instanceof JSONArray) {
+ // 如果是 JSONArray,对数组中的每个对象排序
+ treeMap.put(key, sortJsonArray((JSONArray) value));
+ } else {
+ // 其他类型直接放入
+ treeMap.put(key, value);
+ }
+ }
+ }
+
+ /**
+ * 对 JSONObject 的键进行排序
+ */
+ private static JSONObject sortJsonObject(JSONObject jsonObject) {
+ Map<String, Object> treeMap = new TreeMap<>();
+ Set<String> keys = jsonObject.keySet();
+ for (String key : keys) {
+ Object value = jsonObject.get(key);
+ if (value instanceof JSONObject) {
+ // 如果是 JSONObject,递归排序
+ treeMap.put(key, sortJsonObject((JSONObject) value));
+ } else if (value instanceof JSONArray) {
+ // 如果是 JSONArray,对数组中的每个对象排序
+ treeMap.put(key, sortJsonArray((JSONArray) value));
+ } else {
+ // 其他类型直接放入
+ treeMap.put(key, value);
+ }
+ }
+ return new JSONObject(treeMap);
+ }
+
+ /**
+ * 对 JSONArray 中的每个 JSONObject 进行排序
+ */
+ private static JSONArray sortJsonArray(JSONArray jsonArray) {
+ JSONArray sortedJsonArray = new JSONArray();
+ for (Object item : jsonArray) {
+ if (item instanceof JSONObject) {
+ // 如果是 JSONObject,递归排序
+ sortedJsonArray.add(sortJsonObject((JSONObject) item));
+ } else {
+ // 其他类型直接放入
+ sortedJsonArray.add(item);
+ }
+ }
+ return sortedJsonArray;
+ }
+
+
+ /**
+ * 生成签名
+ *
+ * @param map 请求参数
+ * @param timestamp 时间戳
+ * @param secretKey 密钥
+ * @return 签名
+ */
+ private static String generateSignature(Map<String, Object> map, String timestamp, String secretKey, String requestId, String token) {
+ try {
+ // 对参数进行排序 将params转成字符串再转成map是为了将其中的对象和集合转成jsonObject和jsonArray 方便排序处理
+// Map<String, Object> map = JSONObject.parseObject(JSONObject.toJSONString(map), Map.class);
+ Map<String, Object> sortedParams = new TreeMap<>(map);
+ sortParams(sortedParams);
+ // 将参数转换为 JSON 字符串
+ String jsonString = JSONObject.toJSONString(sortedParams);
+ // 计算参数的 MD5
+ String paramsMD5 = Md5Util.getMD5ByCharset(jsonString, StandardCharsets.UTF_8);
+ log.info("generateSignature requestId:{} jsonString:{} paramsMD5:{}", requestId, jsonString, paramsMD5);
+ // 生成待签名的数据
+ String dataToSign = PARAMS + "=" + paramsMD5 +
+ "&" + TIMESTAMP_HEADER + "=" + timestamp +
+ "&" + REQUEST_ID + "=" + requestId +
+ "&" + TOKEN + "=" + token;
+
+ // 使用 HMAC-SHA256 生成签名
+ return Md5Util.hmacSha256(dataToSign, secretKey);
+ } catch (Exception e) {
+ throw new BadRequestException("短信生成签名:" + e.getMessage());
+ }
+ }
+
+ /**
+ * 解析 XML 响应并提取 <string> 标签中的值
+ *
+ * @param xmlResponse XML 响应字符串
+ * @return <string> 标签中的值
+ */
+ public static String parseXmlResponse(String xmlResponse) {
+ try {
+ // 创建 DocumentBuilder
+ DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance();
+ DocumentBuilder builder = factory.newDocumentBuilder();
+
+ // 将 XML 字符串转换为 Document 对象
+ Document document = builder.parse(new ByteArrayInputStream(xmlResponse.getBytes(StandardCharsets.UTF_8)));
+
+ // 获取 <string> 标签
+ NodeList nodeList = document.getElementsByTagName("string");
+ Element stringElement = (Element) nodeList.item(0);
+ // 提取标签中的文本内容
+ return stringElement.getTextContent();
+ } catch (Exception e) {
+ throw new BadRequestException("短信解析XML:" + e.getMessage());
+ }
+ }
+}
diff --git a/oying-system/src/main/java/com/oying/modules/winnerlook/util/Md5Util.java b/oying-system/src/main/java/com/oying/modules/winnerlook/util/Md5Util.java
new file mode 100644
index 0000000..83046f4
--- /dev/null
+++ b/oying-system/src/main/java/com/oying/modules/winnerlook/util/Md5Util.java
@@ -0,0 +1,78 @@
+package com.oying.modules.winnerlook.util;
+
+import com.oying.exception.BadRequestException;
+import org.apache.commons.codec.binary.Base64;
+
+import javax.crypto.Mac;
+import javax.crypto.spec.SecretKeySpec;
+import java.nio.charset.Charset;
+import java.nio.charset.StandardCharsets;
+import java.security.MessageDigest;
+import java.security.NoSuchAlgorithmException;
+
+/**
+ * @author MinaWu
+ * @description 加密 HMAC_SHA256、MD5
+ * @date 2025/3/31 14:41
+ */
+public class Md5Util {
+
+ // 用来将字节转换成 16 进制表示的字符
+ static char[] hexDigits = {'0', '1', '2', '3', '4', '5', '6', '7', '8',
+ '9', 'a', 'b', 'c', 'd', 'e', 'f'};
+
+ private static final String HMAC_SHA256 = "HmacSHA256";
+
+ /**
+ * HMAC_SHA256加密
+ * @param data 待加密数据
+ * @param key 密钥
+ * @return String
+ */
+ public static String hmacSha256(String data, String key) throws Exception {
+ Mac mac = Mac.getInstance(HMAC_SHA256);
+ SecretKeySpec secretKey = new SecretKeySpec(key.getBytes(StandardCharsets.UTF_8), HMAC_SHA256);
+ mac.init(secretKey);
+ byte[] hmacBytes = mac.doFinal(data.getBytes(StandardCharsets.UTF_8));
+ return Base64.encodeBase64String(hmacBytes);
+ }
+
+ /**
+ * 对一段String生成MD5加密信息
+ *
+ * @param message 要加密的String
+ * @return 生成的MD5信息
+ */
+ public static String getMD5ByCharset(String message, Charset charset) {
+ try {
+ MessageDigest md = MessageDigest.getInstance("MD5");
+ byte[] b = md.digest(message.getBytes(charset));
+ return byteToHexString(b);
+ } catch (NoSuchAlgorithmException e) {
+ throw new BadRequestException("短信MD5加密:" + e.getMessage());
+ }
+ }
+
+ /**
+ * 把byte[]数组转换成十六进制字符串表示形式
+ *
+ * @param tmp 要转换的byte[]
+ * @return 十六进制字符串表示形式
+ */
+ private static String byteToHexString(byte[] tmp) {
+ String s;
+ // 用字节表示就是 16 个字节
+ char[] str = new char[16 * 2]; // 每个字节用 16 进制表示的话,使用两个字符,
+ // 所以表示成 16 进制需要 32 个字符
+ int k = 0; // 表示转换结果中对应的字符位置
+ for (int i = 0; i < 16; i++) { // 从第一个字节开始,对 MD5 的每一个字节
+ // 转换成 16 进制字符的转换
+ byte byte0 = tmp[i]; // 取第 i 个字节
+ str[k++] = hexDigits[byte0 >>> 4 & 0xf]; // 取字节中高 4 位的数字转换,
+ // >>> 为逻辑右移,将符号位一起右移
+ str[k++] = hexDigits[byte0 & 0xf]; // 取字节中低 4 位的数字转换
+ }
+ s = new String(str); // 换后的结果转换为字符串
+ return s;
+ }
+}
diff --git a/oying-system/src/main/resources/config/application-dev.yml b/oying-system/src/main/resources/config/application-dev.yml
index 394d288..c2ab1c6 100644
--- a/oying-system/src/main/resources/config/application-dev.yml
+++ b/oying-system/src/main/resources/config/application-dev.yml
@@ -178,10 +178,10 @@
endpoint: https://obs.cn-southwest-2.myhuaweicloud.com
winner-look:
- url-send-msg: https://118.178.116.15:8443/winnerrxd/api/trigger/SendMsg
- url-send-msg-batch: https://118.178.116.15:8443/winnerrxd/api/trigger/SendBatchMsg
- user-code: CQLYSXYJ
- user-pass: lych1205!
+ url-send-msg: https://118.178.116.15/winnerrxd
+ token: e2318b05713e473890f79fe83d18e235
+ user-code: SHOYYJ
+ user-pass: Oy250928
# 支付类型
pay:
diff --git a/oying-system/src/main/resources/config/application-prod.yml b/oying-system/src/main/resources/config/application-prod.yml
index 7872280..dcd9b87 100644
--- a/oying-system/src/main/resources/config/application-prod.yml
+++ b/oying-system/src/main/resources/config/application-prod.yml
@@ -193,10 +193,10 @@
endpoint: https://obs.cn-southwest-2.myhuaweicloud.com
winner-look:
- url-send-msg: https://118.178.116.15:8443/winnerrxd/api/trigger/SendMsg
- url-send-msg-batch: https://118.178.116.15:8443/winnerrxd/api/trigger/SendBatchMsg
- user-code: CQLYSXYJ
- user-pass: lych1205!
+ url-send-msg: https://118.178.116.15/winnerrxd
+ token: e2318b05713e473890f79fe83d18e235
+ user-code: SHOYYJ
+ user-pass: Oy250928
# 支付类型
pay:
diff --git a/oying-system/src/main/resources/config/application.yml b/oying-system/src/main/resources/config/application.yml
index b916632..cb3222b 100644
--- a/oying-system/src/main/resources/config/application.yml
+++ b/oying-system/src/main/resources/config/application.yml
@@ -23,7 +23,7 @@
check-template-location: false
profiles:
# 激活的环境,如果需要 quartz 分布式支持,需要修改 active: dev,quartz
- active: dev
+ active: prod
data:
redis:
repositories:
--
Gitblit v1.9.3