javaweb使用华为云短信通知公共类调用

17 篇文章 0 订阅
10 篇文章 1 订阅

情景:公司业务需求,短信从阿里云切换到华为云,参照华为云短信调用的相关文档遇到不少坑,在此记录一下。

开发环境:JDK1.8  系统环境:SpringBoot

1、华为云短信配置信息在application.yml中配置

sms:
  huawei:
    url: https://rtcsms.cn-north-1.myhuaweicloud.com:10743/sms/batchSendSms/v1
    appKey: ******
    appSecret: ******

2、创建短信公共类:HwSmsSender

package com.seeker.common.utils.hwsms;

import com.alibaba.fastjson.JSONObject;
import com.seeker.common.utils.Constants;
import com.seeker.common.utils.LoggerUtils;
import com.seeker.common.utils.SmsConstants;
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.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 org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Component;

import java.nio.charset.Charset;
import java.text.SimpleDateFormat;
import java.util.*;


@Component
public class HwSmsSender {

    protected Logger logger = LoggerFactory.getLogger(getClass());

    /**
     * 无需修改,用于格式化鉴权头域,给"X-WSSE"参数赋值
     */
    private static final String WSSE_HEADER_FORMAT = "UsernameToken Username=\"%s\",PasswordDigest=\"%s\",Nonce=\"%s\",Created=\"%s\"";
    /**
     * 无需修改,用于格式化鉴权头域,给"Authorization"参数赋值
     */
    private static final String AUTH_HEADER_VALUE = "WSSE realm=\"SDP\",profile=\"UsernameToken\",type=\"Appkey\"";

    @Value("${sms.huawei.url}")
    private String url;

    @Value("${sms.huawei.appKey}")
    private String appKey;

    @Value("${sms.huawei.appSecret}")
    private String appSecret;

    public String sendNotifyMsg(String mobile, String templateId, String templateParas) throws Exception {
        // 默认通知类
        return sendMsg(mobile, SmsConstants.ResultMessage.SIGN_NOTIFY_ID, templateId, templateParas);
    }

    public String sendMsg(String mobile, String sender, String templateId, String templateParas) throws Exception {
        //条件必填,国内短信关注,当templateId指定的模板类型为通用模板时生效且必填,必须是已审核通过的,与模板类型一致的签名名称
        //国际/港澳台短信不用关注该参数
        //签名名称
        String signature = "美团外卖";
        // String sender = "10690400999304584";

        String receiver = "+86" + mobile;

        //选填,短信状态报告接收地址,推荐使用域名,为空或者不填表示不接收状态报告
        String statusCallBack = "";

        /**
         * 选填,使用无变量模板时请赋空值 String templateParas = "";
         * 单变量模板示例:模板内容为"您的验证码是${1}"时,templateParas可填写为"[\"369751\"]"
         * 双变量模板示例:模板内容为"您有${1}件快递请到${2}领取"时,templateParas可填写为"[\"3\",\"人民公园正门\"]"
         * 模板中的每个变量都必须赋值,且取值不能为空
         * 查看更多模板和变量规范:产品介绍>模板和变量规范
         */
        //模板变量,此处以单变量验证码短信为例,请客户自行生成6位验证码,并定义为字符串类型,以杜绝首位0丢失的问题(例如:002569变成了2569)。
          //templateParas = "[\"369751\"]";

        //请求Body,不携带签名名称时,signature请填null
        String body = buildRequestBody(sender, receiver, templateId, templateParas, statusCallBack, signature);
        if (null == body || body.isEmpty()) {
            LoggerUtils.info(logger, "body is null.");
            return "1";
        }

        //请求Headers中的X-WSSE参数值
        String wsseHeader = buildWsseHeader(appKey, appSecret);
        if (null == wsseHeader || wsseHeader.isEmpty()) {
            LoggerUtils.info(logger, "wsse header is null.");
            return "1";
        }

        //如果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 = HttpClients.custom()
                .setSSLContext(new SSLContextBuilder().loadTrustMaterial(null,
                        (x509CertChain, authType) -> true).build())
                .setSSLHostnameVerifier(NoopHostnameVerifier.INSTANCE)
                .build();

        //请求方法POST
        HttpResponse response = client.execute(RequestBuilder.create("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());

        //打印响应头域信息
        LoggerUtils.info(logger, (response.toString()));
        //打印响应消息实体
        String entity = EntityUtils.toString(response.getEntity());
        LoggerUtils.info(logger, "消息实体" + entity);
        HwSmsRoot hwSmsRoot = JSONObject.parseObject(entity, HwSmsRoot.class);

        return "0";
    }

    /**
     * 构造请求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'");
        //Created
        String time = sdf.format(new Date());
        //Nonce
        String nonce = UUID.randomUUID().toString().replace("-", "");

        byte[] passwordDigest = DigestUtils.sha256(nonce + time + appSecret);
        String hexDigest = Hex.encodeHexString(passwordDigest);

        //如果JDK版本是1.8,请加载原生Base64类,并使用如下代码
        //PasswordDigest
        String passwordDigestBase64Str = Base64.getEncoder().encodeToString(hexDigest.getBytes());
        //如果JDK版本低于1.8,请加载三方库提供Base64类,并使用如下代码
        //PasswordDigest
        //String passwordDigestBase64Str = Base64.encodeBase64String(hexDigest.getBytes(Charset.forName("utf-8")));
        //若passwordDigestBase64Str中包含换行符,请执行如下代码进行修正
        //passwordDigestBase64Str = passwordDigestBase64Str.replaceAll("[\\s*\t\n\r]", "");

        return String.format(WSSE_HEADER_FORMAT, appKey, passwordDigestBase64Str, nonce, time);
    }

    //本地调试方法
    public static void main(String[] args) throws Exception {
        //必填,请参考"开发准备"获取如下数据,替换为实际值
        //APP接入地址+接口访问URI
        String url = "https://rtcsms.cn-north-1.myhuaweicloud.com:10743/sms/batchSendSms/v1";
        //APP_Key 输入自己的
        String appKey = "******";
        //APP_Secret 输入自己的
        String appSecret = "******";
        //国内短信签名通道号或国际/港澳台短信通道号
        String sender = "8820032023657";
        //模板ID
        String templateId = "d1e8f2c7ab964c6998bda5638238bb7d";

        //条件必填,国内短信关注,当templateId指定的模板类型为通用模板时生效且必填,必须是已审核通过的,与模板类型一致的签名名称
        //国际/港澳台短信不用关注该参数
        //签名名称
        String signature = "美团外卖";

        //必填,全局号码格式(包含国家码),示例:+8615123**6789,多个号码之间用英文逗号分隔
        //String receiver = "+8615123**6789,+8615234**7890"; //短信接收人号码
        String receiver = "+8617520**2687";

        //选填,短信状态报告接收地址,推荐使用域名,为空或者不填表示不接收状态报告
        String statusCallBack = "";

        /**
         * 选填,使用无变量模板时请赋空值 String templateParas = "";
         * 单变量模板示例:模板内容为"您的验证码是${1}"时,templateParas可填写为"[\"369751\"]"
         * 双变量模板示例:模板内容为"您有${1}件快递请到${2}领取"时,templateParas可填写为"[\"3\",\"人民公园正门\"]"
         * 模板中的每个变量都必须赋值,且取值不能为空
         * 查看更多模板和变量规范:产品介绍>模板和变量规范
         */
        //模板变量,此处以单变量验证码短信为例,请客户自行生成6位验证码,并定义为字符串类型,以杜绝首位0丢失的问题(例如:002569变成了2569)。
        String templateParas = JSONObject.toJSONString(new String[]{"598745", "1"});
        //请求Body,不携带签名名称时,signature请填null
        String body = buildRequestBody(sender, receiver, templateId, templateParas, statusCallBack, signature);
        if (null == body || body.isEmpty()) {
            System.out.println("body is null.");
            return;
        }

        //请求Headers中的X-WSSE参数值
        String wsseHeader = buildWsseHeader(appKey, appSecret);
        if (null == wsseHeader || wsseHeader.isEmpty()) {
            System.out.println("wsse header is null.");
            return;
        }

        //如果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 = HttpClients.custom()
                .setSSLContext(new SSLContextBuilder().loadTrustMaterial(null,
                        (x509CertChain, authType) -> true).build())
                .setSSLHostnameVerifier(NoopHostnameVerifier.INSTANCE)
                .build();

        //请求方法POST
        HttpResponse response = client.execute(RequestBuilder.create("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());

        //打印响应头域信息
        System.out.println(response.toString());
        //打印响应消息实体
        String entity = EntityUtils.toString(response.getEntity());
        System.out.println(entity);

        HwSmsRoot hwSmsRoot = JSONObject.parseObject(entity, HwSmsRoot.class);

        System.out.println(hwSmsRoot.getCode());
    }

}

 3、上面公共类还用到以下类:

HwSmsResult
import lombok.Data;

@Data
public class HwSmsResult {

    private String originTo;

    private String createTime;

    private String from;

    private String smsMsgId;

    private String status;

}

 HwSmsRoot

@Data
public class HwSmsRoot {

    private List<HwSmsResult> result;

    private String code;

    private String description;

}
SmsConstants
public class SmsConstants {

    public static class ResultMessage {
        /**
         * 通道 美团外卖 验证码类
         */
        public static String SIGN_NOTIFY_ID = "8821041535507";
        public static String SMS_SEND_ID = "8fed7b85f85d4a248fe983a321";
      
    }

}
LoggerUtils
import org.slf4j.Logger;

public class LoggerUtils {

    public static void debug(Logger logger, Object message) {
        if (logger.isDebugEnabled()) {
            logger.debug("Debug日志信息{}", message);
        }
    }

    public static void info(Logger logger, Object message) {
        if (logger.isInfoEnabled()) {
            logger.info("Info日志信息{}", message);
        }
    }

    public static void error(Logger logger, Object message) {
        if (logger.isErrorEnabled()) {
            logger.error("Error日志信息{}", message);
        }
    }
}

 4、实战调用如下:

    /**
     * 获取验证码接口
     */
    @PostMapping("/sendMessageWxUser")
    public AjaxResult sendMessageWxUser(@RequestBody WxUser user, HttpServletRequest request, HttpServletResponse response) throws Exception {
        String ipAddr = IPUtils.getIpAddr(request);
        AjaxResult ajaxResult = new AjaxResult(0, "短信发送成功!请等待查收!");
        if (StringUtils.isEmpty(user.getLoginName())) {
            return AjaxResult.error(500, "手机号不能为空");
        }
        //生成6位随机数,有效期1分钟
        String randNum = VerificatieCodeUtils.getRandNum(6);
        String[] templateParas = new String[]{randNum, "1"};
        String templateId = SmsConstants.ResultMessage.SMS_SEND_ID;
        String code = smsSendUtils.sendRegisterSMS(user.getLoginName(), templateId, JSONObject.toJSONString(templateParas));
        logger.info("--打印验证码:" + randNum);
        if ("0".equals(code)) {
            //缓存按ip+varcode 作为key保存
            redisCache.setCacheObject(ipAddr + "varcode", randNum, 60, TimeUnit.SECONDS);
            return ajaxResult;
        } else {
            return new AjaxResult(500, "短信发送失败!");
        }
    }

VerificatieCodeUtils :验证码随机数生成

import java.util.Random;

public class VerificatieCodeUtils {

    public static String getRandNum(int charCount) {
        String charValue = "";
        for (int i = 0; i < charCount; i++) {
            char c = (char) (randomInt(0, 10) + '0');
            charValue += String.valueOf(c);
        }
        return charValue;

    }

    public static int randomInt(int from, int to) {
        Random r = new Random();
        return from + r.nextInt(to - from);
    }

}

最后:需要注意的是,根据自己项目开发环境版本调整一些相关方法,以及华为云一些密钥什么需要填写正确。

OK 这次就到这里了啦 ^_^

 

 

 

  • 1
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
华为云是华为公司推出的云计算服务平台,提供了丰富的云服务和解决方案,包括云服务器、云数据库、云存储等。下面是在华为云上部署JavaWeb项目的一般步骤: 1. 注册华为云账号并登录控制台:首先,您需要在华为云官网注册一个账号,并登录到华为云控制台。 2. 创建云服务器:在控制台中,选择云服务器ECS服务,点击创建实例,根据您的需求选择合适的配置和镜像,创建一台云服务器。 3. 配置安全组规则:在创建实例时,需要配置安全组规则,确保能够访问您的JavaWeb项目所需的端口(如80端口用于HTTP访问)。 4. 安装Java环境:登录到您的云服务器后,根据您的需求安装适合的Java环境(如OpenJDK或Oracle JDK)。 5. 部署Web容器:选择适合的Web容器(如Tomcat或Jetty),并将其安装到您的云服务器上。 6. 部署JavaWeb项目:将您的JavaWeb项目打包成war文件,并将其部署到Web容器中。具体的部署方式可以根据您使用的Web容器而有所不同,一般来说,您可以将war文件放置在Web容器的指定目录下,然后启动Web容器。 7. 配置域名和DNS解析:如果您需要通过域名访问您的JavaWeb项目,您可以在华为云控制台中配置域名和DNS解析,将域名指向您的云服务器的公网IP地址。 8. 测试访问:完成上述步骤后,您可以通过浏览器访问您的JavaWeb项目,输入域名或公网IP地址加上相应的端口号,即可访问您的项目。 希望以上步骤对您有所帮助!

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值