全部产品
存储与CDN 数据库 安全 应用服务 数加·人工智能 数加·大数据基础服务 互联网中间件 视频服务 开发者工具 解决方案 物联网 钉钉智能硬件
短信服务

短信消息API---.NET

更新时间:2017-11-29 15:31:07

简介

当您使用短信的API接口发送短信后,可以通过使用MNS的Queue模型来接收短信的回执消息,假如服务出现异常情况时(如网络问题),导致消息回执未成功获取,还可以通过短信发送记录查询API接口进行一定的补偿(目前支持30天内发送记录的查询,可查询一天的发送数据)。

消息的订阅

  • 云通信的所有业务消息都用过MNS消息服务向外发送。用户每订阅一个类别的消息(比如上行短信消息SmsUp),系统都会为用户分配一个独立的消息队列。
  • 用户可以通过阿里云账号拿到一个临时的token用于获取队列中的消息。用户可以下载demo,编写简单的消息处理类即可完成消息处理的任务。
  • 在页面上订阅消息,订阅完消息后,能拿到消息队列名称(queueName)。比如:Alicom-Queue-xxxxxx-SmsReport 。队列名字每个用户都不同。

路径:短信产品进入控制台→短信服务概览→云通信消息接收→短信状态报告接收→向右滑动(MNS消息队列消费模式。接收消息队列信息,请下载消息服务SDK进行程序编写,详见API文档)位置如下图所有示; 点击进入控制台配图

消息类型

短信提供2种消息类型SmsReport(短信回执报告消息) 和 SmsUp(上行短信消息)

  • 通过订阅SmsReport短信状态报告,可以获知每条短信的发送情况,了解短信是否达到终端用户的状态与相关信息
  • 通过订阅SmsUp上行短信消息,可以获知终端用户回复短信的内容
短信回执消息SmsReport消息体格式
名称 类型 描述 示例
phone_number String 短信接收号码 13000000000
success Boolean 发送是否成功 true
biz_id String 发送回执ID 1234^345
out_id String 调用发送短信接口时传的outId 123456
send_time String 转发给运营商的时间 2017-06-01 10:00:00
report_time String 收到运营商回执的时间 2017-06-01 10:00:05
err_code String 错误码 UNKNOW
err_msg String 错误信息 未知异常
sms_size String 140字节算一条短信,短信长度超过140字节时会拆分成多条短信发送 1,2,3
上行短信消息SmsUp
名称 类型 描述 示例
phone_number String 短信接收号码 13000000000
content String 短信内容 true
send_time String 时间 20150101120000
dest_code String 扩展码 123456
sequence_id Double 消息序列ID 123456

技术对接步骤

下载消息SDK

下载对应语音的消息DEMO工程,工程所需要的所有依赖包都放在DEMO工程的lib目录下,将对应的包引入到您的工程当中既可按照DEMO样例编写接收消息的程序。

SDK&DEMO【下载地址】

编写样例程序

  1. using System;
  2. using System.Collections.Generic;
  3. using System.Linq;
  4. using System.Text;
  5. using Aliyun.Acs.Core;
  6. using Aliyun.Acs.Core.Exceptions;
  7. using Aliyun.Acs.Core.Profile;
  8. using Aliyun.Acs.Dybaseapi.Model.V20170525;
  9. //using Aliyun.Acs.Dybaseapi.Model.V20170525.QueryTokenForMnsQueueResponse.MessageTokenDTO_;
  10. using Aliyun.MNS;
  11. using Aliyun.MNS.Model;
  12. using System.Threading;
  13. using System.Threading.Tasks;
  14. namespace ConsoleApplication7
  15. {
  16. class Program
  17. {
  18. private static int maxThread = 2;
  19. static String accessId = ""; //你的accessKeyId,参考本文档步骤2
  20. static String accessSecret = "";//你的accessKeySecret,参考本文档步骤2
  21. static String messageType = "SmsReport";//消息类型目前有4种. 短信回执:SmsReport; 短信上行:SmsUp; 语音回执:VoiceReport; 流量回执:FlowReport;
  22. static String queueName = "Alicom-Queue-17888888794453-SmsReport";// 短信上行的队列名称. 格式是 "前缀(Alicom-Queue-)+阿里云uid+消息类型"
  23. static String domainForPop = "dybaseapi.aliyuncs.com";
  24. static String regionIdForPop = "cn-hangzhou";//(请勿修改)
  25. static String productName = "Dybaseapi";
  26. static IAcsClient acsClient = null;
  27. public static IAcsClient InitAcsClient(String regionIdForPop, String accessId, String accessSecret, String productName, String domainForPop)
  28. {
  29. IClientProfile profile = DefaultProfile.GetProfile(regionIdForPop, accessId, accessSecret);
  30. DefaultProfile.AddEndpoint(regionIdForPop, regionIdForPop, productName, domainForPop);
  31. IAcsClient acsClient = new DefaultAcsClient(profile);
  32. return acsClient;
  33. }
  34. // 初始化环境
  35. private static void InitData()
  36. {
  37. acsClient = InitAcsClient(regionIdForPop, accessId, accessSecret, productName, domainForPop);
  38. }
  39. static void Main(string[] args)
  40. {
  41. InitData();
  42. for (int i = 0; i < maxThread; i++)
  43. {
  44. TestTask testTask = new TestTask("PullMessageTask-thread-" + i, messageType, queueName, acsClient);
  45. Thread t = new Thread(new ThreadStart(testTask.Handle));
  46. //启动线程
  47. t.Start();
  48. }
  49. Console.ReadKey();
  50. }
  51. }
  52. class TestTask
  53. {
  54. private object o = new object();
  55. private int sleepTime = 50;
  56. public String name { get; private set; }
  57. public String messageType { get; private set; }
  58. public String queueName { get; private set; }
  59. public int TaskID { get; private set; }
  60. public IAcsClient acsClient { get; private set; }
  61. public TestTask(String name, String messageType, String queueName, IAcsClient acsClient)
  62. {
  63. this.name = name;
  64. this.messageType = messageType;
  65. this.queueName = queueName;
  66. this.acsClient = acsClient;
  67. }
  68. long bufferTime = 60 * 2;//过期时间小于2分钟则重新获取,防止服务器时间误差
  69. String mnsAccountEndpoint = "https://1943695596114318.mns.cn-hangzhou.aliyuncs.com/";//阿里通信消息的endpoint,固定
  70. Dictionary<string, QueryTokenForMnsQueueResponse.MessageTokenDTO_> tokenMap = new Dictionary<string, QueryTokenForMnsQueueResponse.MessageTokenDTO_>();
  71. Dictionary<string, Queue> queueMap = new Dictionary<string, Queue>();
  72. public QueryTokenForMnsQueueResponse.MessageTokenDTO_ GetTokenByMessageType(IAcsClient acsClient, String messageType)
  73. {
  74. QueryTokenForMnsQueueRequest request = new QueryTokenForMnsQueueRequest();
  75. request.MessageType = messageType;
  76. QueryTokenForMnsQueueResponse queryTokenForMnsQueueResponse = acsClient.GetAcsResponse(request);
  77. QueryTokenForMnsQueueResponse.MessageTokenDTO_ token = queryTokenForMnsQueueResponse.MessageTokenDTO;
  78. return token;
  79. }
  80. /// 处理消息
  81. public void Handle()
  82. {
  83. while (true)
  84. {
  85. try
  86. {
  87. QueryTokenForMnsQueueResponse.MessageTokenDTO_ token = null;
  88. Queue queue = null;
  89. lock (o)
  90. {
  91. if (tokenMap.ContainsKey(messageType))
  92. {
  93. token = tokenMap[messageType];
  94. }
  95. if (queueMap.ContainsKey(queueName))
  96. {
  97. queue = queueMap[queueName];
  98. }
  99. TimeSpan ts = new TimeSpan(0);
  100. if (token != null)
  101. {
  102. DateTime b = Convert.ToDateTime(token.ExpireTime);
  103. DateTime c = Convert.ToDateTime(DateTime.Now);
  104. ts = b - c;
  105. }
  106. if (token == null || ts.TotalSeconds < bufferTime || queue == null)
  107. {
  108. token = GetTokenByMessageType(acsClient, messageType);
  109. IMNS client = new Aliyun.MNS.MNSClient(token.AccessKeyId, token.AccessKeySecret, mnsAccountEndpoint, token.SecurityToken);
  110. queue = client.GetNativeQueue(queueName);
  111. if (tokenMap.ContainsKey(messageType))
  112. {
  113. tokenMap.Remove(messageType);
  114. }
  115. if (queueMap.ContainsKey(queueName))
  116. {
  117. queueMap.Remove(queueName);
  118. }
  119. tokenMap.Add(messageType, token);
  120. queueMap.Add(queueName, queue);
  121. }
  122. }
  123. BatchReceiveMessageResponse batchReceiveMessageResponse = queue.BatchReceiveMessage(16);
  124. List<Message> messages = batchReceiveMessageResponse.Messages;
  125. for (int i = 0; i <= messages.Count - 1; i++)
  126. {
  127. try
  128. {
  129. byte[] outputb = Convert.FromBase64String(messages[i].Body);
  130. string orgStr = Encoding.UTF8.GetString(outputb);
  131. System.Console.WriteLine(orgStr);
  132. //TODO 具体消费逻辑,待客户自己实现.
  133. //消费成功的前提下删除消息
  134. queue.DeleteMessage(messages[i].ReceiptHandle);
  135. }
  136. catch (Exception e)
  137. {
  138. System.Console.WriteLine("Hello World!", e);
  139. }
  140. }
  141. }
  142. catch (Exception e)
  143. {
  144. System.Console.WriteLine("Handle exception", e);
  145. }
  146. Thread.Sleep(sleepTime);
  147. }
  148. }
  149. }
  150. }
本文导读目录