package com.xnx3.net; //如果JDK版本低于1.8,请使用三方库提供Base64类 //import org.apache.commons.codec.binary.Base64; import org.apache.commons.codec.binary.Hex; import org.apache.commons.codec.digest.DigestUtils; import org.apache.http.HttpHeaders; import org.apache.http.HttpResponse; import org.apache.http.NameValuePair; import org.apache.http.client.ClientProtocolException; import org.apache.http.client.methods.RequestBuilder; import org.apache.http.client.utils.URLEncodedUtils; import org.apache.http.conn.ssl.NoopHostnameVerifier; import org.apache.http.entity.StringEntity; import org.apache.http.impl.client.CloseableHttpClient; import org.apache.http.impl.client.HttpClients; import org.apache.http.message.BasicNameValuePair; import org.apache.http.ssl.SSLContextBuilder; import org.apache.http.util.EntityUtils; import com.xnx3.BaseVO; import net.sf.json.JSONObject; import java.io.IOException; import java.io.UnsupportedEncodingException; import java.nio.charset.Charset; import java.security.KeyManagementException; import java.security.KeyStoreException; import java.security.NoSuchAlgorithmException; import java.text.SimpleDateFormat; import java.util.ArrayList; //如果JDK版本是1.8,可使用原生Base64类 import java.util.Base64; import java.util.Date; import java.util.List; import java.util.UUID; /** * 华为云短信发送. * 短信开通: http://huawei.leimingyun.com/ * @author 管雷鸣 * */ public class HuaweiSMSUtil { private static final String WSSE_HEADER_FORMAT = "UsernameToken Username=\"%s\",PasswordDigest=\"%s\",Nonce=\"%s\",Created=\"%s\""; //无需修改,用于格式化鉴权头域,给"X-WSSE"参数赋值 private static final String AUTH_HEADER_VALUE = "WSSE realm=\"SDP\",profile=\"UsernameToken\",type=\"Appkey\""; //无需修改,用于格式化鉴权头域,给"Authorization"参数赋值 private static final String url = "https://rtcsms.cn-north-1.myhuaweicloud.com:10743/sms/batchSendSms/v1"; //APP接入地址+接口访问URI private String appKey; //短信应用的appKey private String appSecret; //短信应用的appSecret private String sender; //国内短信签名通道号或国际/港澳台短信通道号 private String signature; //签名 /** * 华为云短信发送 * 短信开通: http://huawei.leimingyun.com/ * @param appKey 短信应用的appKey * @param appSecret 短信应用的appSecret * @param sender 国内短信签名通道号或国际/港澳台短信通道号 * @param signature 短信签名 */ public HuaweiSMSUtil(String appKey, String appSecret, String sender, String signature) { this.appKey = appKey; this.appSecret = appSecret; this.sender = sender; this.signature = signature; } /** * 发送短信 * @param phone 要发送的手机号。可以传入 +8618788888888 ,也可以不带+86,接口里面会自动加上 * @param templateParas 模板变量。 * 选填,使用无变量模板时请赋空值 String templateParas = ""; * 单变量模板示例:模板内容为"您的验证码是${1}"时,templateParas可填写为"[\"369751\"]" * 双变量模板示例:模板内容为"您有${1}件快递请到${2}领取"时,templateParas可填写为"[\"3\",\"人民公园正门\"]" * 模板中的每个变量都必须赋值,且取值不能为空 * 查看更多模板和变量规范:产品介绍>模板和变量规范 * @param templateId 模板ID * @return {@link BaseVO} result = 1 则是发送成功, =0则是失败,用getInfo() 获取失败原因 */ public BaseVO send(String phone, String templateId, String templateParas){ if(phone.indexOf("+") != 0){ phone = "+86"+phone; } //选填,短信状态报告接收地址,推荐使用域名,为空或者不填表示不接收状态报告 String statusCallBack = ""; //请求Body,不携带签名名称时,signature请填null String body = buildRequestBody(sender, phone, templateId, templateParas, statusCallBack, signature); if (null == body || body.isEmpty()) { return BaseVO.failure("body is null."); } //请求Headers中的X-WSSE参数值 String wsseHeader = buildWsseHeader(appKey, appSecret); if (null == wsseHeader || wsseHeader.isEmpty()) { return BaseVO.failure("wsse header is null."); } //如果JDK版本低于1.8,可使用如下代码 //为防止因HTTPS证书认证失败造成API调用失败,需要先忽略证书信任问题 //CloseableHttpClient client = HttpClients.custom() // .setSSLContext(new SSLContextBuilder().loadTrustMaterial(null, new TrustStrategy() { // @Override // public boolean isTrusted(X509Certificate[] x509Certificates, String s) throws CertificateException { // return true; // } // }).build()).setSSLHostnameVerifier(NoopHostnameVerifier.INSTANCE).build(); //如果JDK版本是1.8,可使用如下代码 //为防止因HTTPS证书认证失败造成API调用失败,需要先忽略证书信任问题 CloseableHttpClient client = null; try { client = HttpClients.custom() .setSSLContext(new SSLContextBuilder().loadTrustMaterial(null, (x509CertChain, authType) -> true).build()) .setSSLHostnameVerifier(NoopHostnameVerifier.INSTANCE) .build(); } catch (KeyManagementException e) { e.printStackTrace(); return BaseVO.failure(e.getMessage()); } catch (NoSuchAlgorithmException e) { e.printStackTrace(); return BaseVO.failure(e.getMessage()); } catch (KeyStoreException e) { e.printStackTrace(); return BaseVO.failure(e.getMessage()); } HttpResponse response; try { response = client.execute(RequestBuilder.create("POST")//请求方法POST .setUri(url) .addHeader(HttpHeaders.CONTENT_TYPE, "application/x-www-form-urlencoded") .addHeader(HttpHeaders.AUTHORIZATION, AUTH_HEADER_VALUE) .addHeader("X-WSSE", wsseHeader) .setEntity(new StringEntity(body)).build()); String responseContent = EntityUtils.toString(response.getEntity()); if(responseContent != null){ JSONObject json = JSONObject.fromObject(responseContent); if(json.get("code") != null && json.getString("code").equals("000000")){ return BaseVO.success("success"); } } return BaseVO.failure(responseContent); } catch (ClientProtocolException e) { e.printStackTrace(); return BaseVO.failure(e.getMessage()); } catch (UnsupportedEncodingException e) { e.printStackTrace(); return BaseVO.failure(e.getMessage()); } catch (IOException e) { e.printStackTrace(); return BaseVO.failure(e.getMessage()); } } public static void main(String[] args) throws Exception{ String appKey = "3RwC21LkS2xxxxxxxxxx"; String appSecret = "IWLr67lg4TshegBTlxxxxxxxxxxxx"; String sender = "10690400999000000000"; //国内短信签名通道号或国际/港澳台短信通道号 String signature = "华为云短信测试"; //签名名称 HuaweiSMSUtil sms = new HuaweiSMSUtil(appKey, appSecret, sender, signature); sms.send("17000000000", "58972990fb1b4b16abf312d991a00e00", "[\"369751\"]"); } /** * 构造请求Body体 * @param sender * @param receiver * @param templateId * @param templateParas * @param statusCallbackUrl * @param signature | 签名名称,使用国内短信通用模板时填写 * @return */ static String buildRequestBody(String sender, String receiver, String templateId, String templateParas, String statusCallbackUrl, String signature) { if (null == sender || null == receiver || null == templateId || sender.isEmpty() || receiver.isEmpty() || templateId.isEmpty()) { System.out.println("buildRequestBody(): sender, receiver or templateId is null."); return null; } List<NameValuePair> keyValues = new ArrayList<NameValuePair>(); keyValues.add(new BasicNameValuePair("from", sender)); keyValues.add(new BasicNameValuePair("to", receiver)); keyValues.add(new BasicNameValuePair("templateId", templateId)); if (null != templateParas && !templateParas.isEmpty()) { keyValues.add(new BasicNameValuePair("templateParas", templateParas)); } if (null != statusCallbackUrl && !statusCallbackUrl.isEmpty()) { keyValues.add(new BasicNameValuePair("statusCallback", statusCallbackUrl)); } if (null != signature && !signature.isEmpty()) { keyValues.add(new BasicNameValuePair("signature", signature)); } return URLEncodedUtils.format(keyValues, Charset.forName("UTF-8")); } /** * 构造X-WSSE参数值 * @param appKey * @param appSecret * @return */ static String buildWsseHeader(String appKey, String appSecret) { if (null == appKey || null == appSecret || appKey.isEmpty() || appSecret.isEmpty()) { System.out.println("buildWsseHeader(): appKey or appSecret is null."); return null; } SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd'T'HH:mm:ss'Z'"); String time = sdf.format(new Date()); //Created String nonce = UUID.randomUUID().toString().replace("-", ""); //Nonce byte[] passwordDigest = DigestUtils.sha256(nonce + time + appSecret); String hexDigest = Hex.encodeHexString(passwordDigest); //如果JDK版本是1.8,请加载原生Base64类,并使用如下代码 String passwordDigestBase64Str = Base64.getEncoder().encodeToString(hexDigest.getBytes()); //PasswordDigest //如果JDK版本低于1.8,请加载三方库提供Base64类,并使用如下代码 //String passwordDigestBase64Str = Base64.encodeBase64String(hexDigest.getBytes(Charset.forName("utf-8"))); //PasswordDigest //若passwordDigestBase64Str中包含换行符,请执行如下代码进行修正 //passwordDigestBase64Str = passwordDigestBase64Str.replaceAll("[\\s*\t\n\r]", ""); return String.format(WSSE_HEADER_FORMAT, appKey, passwordDigestBase64Str, nonce, time); } }