dengzedong
2024-10-14 558ffc4bcaf7aa5c683e7c9ce01e971feb9e4d95
提交 | 用户 | 时间
e7c126 1 package com.iailab.module.system.framework.sms.core.client.impl;
H 2
3 import cn.hutool.core.lang.Assert;
4 import cn.hutool.core.util.StrUtil;
5 import com.iailab.framework.common.core.KeyValue;
6 import com.iailab.framework.common.util.collection.ArrayUtils;
7 import com.iailab.framework.common.util.json.JsonUtils;
8 import com.iailab.module.system.framework.sms.core.client.dto.SmsReceiveRespDTO;
9 import com.iailab.module.system.framework.sms.core.client.dto.SmsSendRespDTO;
10 import com.iailab.module.system.framework.sms.core.client.dto.SmsTemplateRespDTO;
11 import com.iailab.module.system.framework.sms.core.client.impl.AbstractSmsClient;
12 import com.iailab.module.system.framework.sms.core.enums.SmsTemplateAuditStatusEnum;
13 import com.iailab.module.system.framework.sms.core.property.SmsChannelProperties;
14 import com.fasterxml.jackson.annotation.JsonFormat;
15 import com.fasterxml.jackson.annotation.JsonProperty;
16 import com.google.common.annotations.VisibleForTesting;
17 import com.tencentcloudapi.common.Credential;
18 import com.tencentcloudapi.sms.v20210111.SmsClient;
19 import com.tencentcloudapi.sms.v20210111.models.*;
20 import lombok.Data;
21
22 import java.time.LocalDateTime;
23 import java.util.List;
24 import java.util.Objects;
25
26 import static com.iailab.framework.common.util.collection.CollectionUtils.convertList;
27 import static com.iailab.framework.common.util.date.DateUtils.FORMAT_YEAR_MONTH_DAY_HOUR_MINUTE_SECOND;
28 import static com.iailab.framework.common.util.date.DateUtils.TIME_ZONE_DEFAULT;
29
30 /**
31  * 腾讯云短信功能实现
32  *
33  * 参见 <a href="https://cloud.tencent.com/document/product/382/52077">文档</a>
34  *
35  * @author shiwp
36  */
37 public class TencentSmsClient extends AbstractSmsClient {
38
39     /**
40      * 调用成功 code
41      */
42     public static final String API_CODE_SUCCESS = "Ok";
43
44     /**
45      * REGION,使用南京
46      */
47     private static final String ENDPOINT = "ap-nanjing";
48
49     /**
50      * 是否国际/港澳台短信:
51      *
52      * 0:表示国内短信。
53      * 1:表示国际/港澳台短信。
54      */
55     private static final long INTERNATIONAL_CHINA = 0L;
56
57     private SmsClient client;
58
59     public TencentSmsClient(SmsChannelProperties properties) {
60         super(properties);
61         Assert.notEmpty(properties.getApiSecret(), "apiSecret 不能为空");
62         validateSdkAppId(properties);
63     }
64
65     @Override
66     protected void doInit() {
67         // 实例化一个认证对象,入参需要传入腾讯云账户密钥对 secretId,secretKey
68         Credential credential = new Credential(getApiKey(), properties.getApiSecret());
69         client = new SmsClient(credential, ENDPOINT);
70     }
71
72     /**
73      * 参数校验腾讯云的 SDK AppId
74      *
75      * 原因是:腾讯云发放短信的时候,需要额外的参数 sdkAppId
76      *
77      * 解决方案:考虑到不破坏原有的 apiKey + apiSecret 的结构,所以将 secretId 拼接到 apiKey 字段中,格式为 "secretId sdkAppId"。
78      *
79      * @param properties 配置
80      */
81     private static void validateSdkAppId(SmsChannelProperties properties) {
82         String combineKey = properties.getApiKey();
83         Assert.notEmpty(combineKey, "apiKey 不能为空");
84         String[] keys = combineKey.trim().split(" ");
85         Assert.isTrue(keys.length == 2, "腾讯云短信 apiKey 配置格式错误,请配置 为[secretId sdkAppId]");
86     }
87
88     private String getSdkAppId() {
89         return StrUtil.subAfter(properties.getApiKey(), " ", true);
90     }
91
92     private String getApiKey() {
93         return StrUtil.subBefore(properties.getApiKey(), " ", true);
94     }
95
96     @Override
97     public SmsSendRespDTO sendSms(Long sendLogId, String mobile,
98                                   String apiTemplateId, List<KeyValue<String, Object>> templateParams) throws Throwable {
99         // 构建请求
100         SendSmsRequest request = new SendSmsRequest();
101         request.setSmsSdkAppId(getSdkAppId());
102         request.setPhoneNumberSet(new String[]{mobile});
103         request.setSignName(properties.getSignature());
104         request.setTemplateId(apiTemplateId);
105         request.setTemplateParamSet(ArrayUtils.toArray(templateParams, e -> String.valueOf(e.getValue())));
106         request.setSessionContext(JsonUtils.toJsonString(new SessionContext().setLogId(sendLogId)));
107         // 执行请求
108         SendSmsResponse response = client.SendSms(request);
109         SendStatus status = response.getSendStatusSet()[0];
110         return new SmsSendRespDTO().setSuccess(Objects.equals(status.getCode(), API_CODE_SUCCESS)).setSerialNo(status.getSerialNo())
111                 .setApiRequestId(response.getRequestId()).setApiCode(status.getCode()).setApiMsg(status.getMessage());
112     }
113
114     @Override
115     public List<SmsReceiveRespDTO> parseSmsReceiveStatus(String text) {
116         List<SmsReceiveStatus> callback = JsonUtils.parseArray(text, SmsReceiveStatus.class);
117         return convertList(callback, status -> new SmsReceiveRespDTO()
118                 .setSuccess(SmsReceiveStatus.SUCCESS_CODE.equalsIgnoreCase(status.getStatus()))
119                 .setErrorCode(status.getErrCode()).setErrorMsg(status.getDescription())
120                 .setMobile(status.getMobile()).setReceiveTime(status.getReceiveTime())
121                 .setSerialNo(status.getSerialNo()).setLogId(status.getSessionContext().getLogId()));
122     }
123
124     @Override
125     public SmsTemplateRespDTO getSmsTemplate(String apiTemplateId) throws Throwable {
126         // 构建请求
127         DescribeSmsTemplateListRequest request = new DescribeSmsTemplateListRequest();
128         request.setTemplateIdSet(new Long[]{Long.parseLong(apiTemplateId)});
129         request.setInternational(INTERNATIONAL_CHINA);
130         // 执行请求
131         DescribeSmsTemplateListResponse response = client.DescribeSmsTemplateList(request);
132         DescribeTemplateListStatus status = response.getDescribeTemplateStatusSet()[0];
133         if (status == null || status.getStatusCode() == null) {
134             return null;
135         }
136         return new SmsTemplateRespDTO().setId(status.getTemplateId().toString()).setContent(status.getTemplateContent())
137                 .setAuditStatus(convertSmsTemplateAuditStatus(status.getStatusCode().intValue())).setAuditReason(status.getReviewReply());
138     }
139
140     @VisibleForTesting
141     Integer convertSmsTemplateAuditStatus(int templateStatus) {
142         switch (templateStatus) {
143             case 1: return SmsTemplateAuditStatusEnum.CHECKING.getStatus();
144             case 0: return SmsTemplateAuditStatusEnum.SUCCESS.getStatus();
145             case -1: return SmsTemplateAuditStatusEnum.FAIL.getStatus();
146             default: throw new IllegalArgumentException(String.format("未知审核状态(%d)", templateStatus));
147         }
148     }
149
150     @Data
151     private static class SmsReceiveStatus {
152
153         /**
154          * 短信接受成功 code
155          */
156         public static final String SUCCESS_CODE = "SUCCESS";
157
158         /**
159          * 用户实际接收到短信的时间
160          */
161         @JsonProperty("user_receive_time")
162         @JsonFormat(pattern = FORMAT_YEAR_MONTH_DAY_HOUR_MINUTE_SECOND, timezone = TIME_ZONE_DEFAULT)
163         private LocalDateTime receiveTime;
164
165         /**
166          * 国家(或地区)码
167          */
168         @JsonProperty("nationcode")
169         private String nationCode;
170
171         /**
172          * 手机号码
173          */
174         private String mobile;
175
176         /**
177          * 实际是否收到短信接收状态,SUCCESS(成功)、FAIL(失败)
178          */
179         @JsonProperty("report_status")
180         private String status;
181
182         /**
183          * 用户接收短信状态码错误信息
184          */
185         @JsonProperty("errmsg")
186         private String errCode;
187
188         /**
189          * 用户接收短信状态描述
190          */
191         @JsonProperty("description")
192         private String description;
193
194         /**
195          * 本次发送标识 ID(与发送接口返回的SerialNo对应)
196          */
197         @JsonProperty("sid")
198         private String serialNo;
199
200         /**
201          * 用户的 session 内容(与发送接口的请求参数 SessionContext 一致)
202          */
203         @JsonProperty("ext")
204         private SessionContext sessionContext;
205
206     }
207
208     @VisibleForTesting
209     @Data
210     static class SessionContext {
211
212         /**
213          * 发送短信记录id
214          */
215         private Long logId;
216
217     }
218
219 }