.net core 使用阿裡雲簡訊發送SMS

来源:https://www.cnblogs.com/kulend/archive/2018/04/12/8807716.html
-Advertisement-
Play Games

阿裡雲官方的skd(aliyun-net-sdk-core,aliyun-net-sdk-dysmsapi)在dnc中發送簡訊會出錯,nuget上的包貌似也一樣不管用。直接改下sdk當然也可以,但就發個簡訊,官方的sdk實在是有點繁雜,其實可以簡單化,一個類就搞定。 SmsObject.cs 調用方 ...


阿裡雲官方的skd(aliyun-net-sdk-core,aliyun-net-sdk-dysmsapi)在dnc中發送簡訊會出錯,nuget上的包貌似也一樣不管用。直接改下sdk當然也可以,但就發個簡訊,官方的sdk實在是有點繁雜,其實可以簡單化,一個類就搞定。

  1 using Newtonsoft.Json;
  2 using System;
  3 using System.Collections.Generic;
  4 using System.Globalization;
  5 using System.Net;
  6 using System.Net.Http;
  7 using System.Security.Cryptography;
  8 using System.Text;
  9 using System.Threading.Tasks;
 10 using System.Web;
 11 
 12 namespace Vino.Core.Communication.SMS
 13 {
 14     public class AliyunSmsSender : ISmsSender
 15     {
 16         private string RegionId = "cn-hangzhou";
 17         private string Version = "2017-05-25";
 18         private string Action = "SendSms";
 19         private string Format = "JSON";
 20         private string Domain = "dysmsapi.aliyuncs.com";
 21 
 22         private int MaxRetryNumber = 3;
 23         private bool AutoRetry = true;
 24         private const string SEPARATOR = "&";
 25         private int TimeoutInMilliSeconds = 100000;
 26 
 27         private string AccessKeyId;
 28         private string AccessKeySecret;
 29 
 30         public AliyunSmsSender(string accessKeyId, string accessKeySecret)
 31         {
 32             this.AccessKeyId = accessKeyId;
 33             this.AccessKeySecret = accessKeySecret;
 34         }
 35 
 36         /// <summary>
 37         /// 發送簡訊
 38         /// </summary>
 39         public async Task<(bool success, string response)> Send(SmsObject sms)
 40         {
 41             var paramers = new Dictionary<string, string>();
 42             paramers.Add("PhoneNumbers", sms.Mobile);
 43             paramers.Add("SignName", sms.Signature);
 44             paramers.Add("TemplateCode", sms.TempletKey);
 45             paramers.Add("TemplateParam", JsonConvert.SerializeObject(sms.Data));
 46             paramers.Add("OutId", sms.OutId);
 47             paramers.Add("AccessKeyId", AccessKeyId);
 48 
 49             try
 50             {
 51                 string url = GetSignUrl(paramers, AccessKeySecret);
 52 
 53                 int retryTimes = 1;
 54                 var reply = await HttpGetAsync(url);
 55                 while (500 <= reply.StatusCode && AutoRetry && retryTimes < MaxRetryNumber)
 56                 {
 57                     url = GetSignUrl(paramers, AccessKeySecret);
 58                     reply = await HttpGetAsync(url);
 59                     retryTimes++;
 60                 }
 61 
 62                 if (!string.IsNullOrEmpty(reply.response))
 63                 {
 64                     var res = JsonConvert.DeserializeObject<Dictionary<string, string>>(reply.response);
 65                     if (res != null && res.ContainsKey("Code") && "OK".Equals(res["Code"]))
 66                     {
 67                         return (true, response: reply.response);
 68                     }
 69                 }
 70 
 71                 return (false, response: reply.response);
 72             }
 73             catch (Exception ex)
 74             {
 75                 return (false, response: ex.Message);
 76             }
 77         }
 78 
 79         private string GetSignUrl(Dictionary<string, string> parameters, string accessSecret)
 80         {
 81             var imutableMap = new Dictionary<string, string>(parameters);
 82             imutableMap.Add("Timestamp", FormatIso8601Date(DateTime.Now));
 83             imutableMap.Add("SignatureMethod", "HMAC-SHA1");
 84             imutableMap.Add("SignatureVersion", "1.0");
 85             imutableMap.Add("SignatureNonce", Guid.NewGuid().ToString());
 86             imutableMap.Add("Action", Action);
 87             imutableMap.Add("Version", Version);
 88             imutableMap.Add("Format", Format);
 89             imutableMap.Add("RegionId", RegionId);
 90 
 91             IDictionary<string, string> sortedDictionary = new SortedDictionary<string, string>(imutableMap, StringComparer.Ordinal);
 92             StringBuilder canonicalizedQueryString = new StringBuilder();
 93             foreach (var p in sortedDictionary)
 94             {
 95                 canonicalizedQueryString.Append("&")
 96                 .Append(PercentEncode(p.Key)).Append("=")
 97                 .Append(PercentEncode(p.Value));
 98             }
 99 
100             StringBuilder stringToSign = new StringBuilder();
101             stringToSign.Append("GET");
102             stringToSign.Append(SEPARATOR);
103             stringToSign.Append(PercentEncode("/"));
104             stringToSign.Append(SEPARATOR);
105             stringToSign.Append(PercentEncode(canonicalizedQueryString.ToString().Substring(1)));
106 
107             string signature = SignString(stringToSign.ToString(), accessSecret + "&");
108 
109             imutableMap.Add("Signature", signature);
110 
111             return ComposeUrl(Domain, imutableMap);
112         }
113 
114         private static string FormatIso8601Date(DateTime date)
115         {
116             return date.ToUniversalTime().ToString("yyyy-MM-dd'T'HH:mm:ss'Z'", CultureInfo.CreateSpecificCulture("en-US"));
117         }
118 
119         /// <summary>
120         /// 簽名
121         /// </summary>
122         public static string SignString(string source, string accessSecret)
123         {
124             using (var algorithm = new HMACSHA1(Encoding.UTF8.GetBytes(accessSecret.ToCharArray())))
125             {
126                 return Convert.ToBase64String(algorithm.ComputeHash(Encoding.UTF8.GetBytes(source.ToCharArray())));
127             }
128         }
129 
130         private static string ComposeUrl(string endpoint, Dictionary<String, String> parameters)
131         {
132             StringBuilder urlBuilder = new StringBuilder("");
133             urlBuilder.Append("http://").Append(endpoint);
134             if (-1 == urlBuilder.ToString().IndexOf("?"))
135             {
136                 urlBuilder.Append("/?");
137             }
138             string query = ConcatQueryString(parameters);
139             return urlBuilder.Append(query).ToString();
140         }
141 
142         private static string ConcatQueryString(Dictionary<string, string> parameters)
143         {
144             if (null == parameters)
145             {
146                 return null;
147             }
148             StringBuilder sb = new StringBuilder();
149 
150             foreach (var entry in parameters)
151             {
152                 String key = entry.Key;
153                 String val = entry.Value;
154 
155                 sb.Append(HttpUtility.UrlEncode(key, Encoding.UTF8));
156                 if (val != null)
157                 {
158                     sb.Append("=").Append(HttpUtility.UrlEncode(val, Encoding.UTF8));
159                 }
160                 sb.Append("&");
161             }
162 
163             int strIndex = sb.Length;
164             if (parameters.Count > 0)
165                 sb.Remove(strIndex - 1, 1);
166 
167             return sb.ToString();
168         }
169 
170         public static string PercentEncode(string value)
171         {
172             StringBuilder stringBuilder = new StringBuilder();
173             string text = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789-_.~";
174             byte[] bytes = Encoding.GetEncoding("UTF-8").GetBytes(value);
175             foreach (char c in bytes)
176             {
177                 if (text.IndexOf(c) >= 0)
178                 {
179                     stringBuilder.Append(c);
180                 }
181                 else
182                 {
183                     stringBuilder.Append("%").Append(
184                         string.Format(CultureInfo.InvariantCulture, "{0:X2}", (int)c));
185                 }
186             }
187             return stringBuilder.ToString();
188         }
189 
190         private async Task<(int StatusCode, string response)> HttpGetAsync(string url)
191         {
192             HttpClientHandler handler = new HttpClientHandler();
193             handler.Proxy = null;
194             handler.AutomaticDecompression = DecompressionMethods.GZip;
195 
196             using (var http = new HttpClient(handler))
197             {
198                 http.Timeout = new TimeSpan(TimeSpan.TicksPerMillisecond * TimeoutInMilliSeconds);
199                 HttpResponseMessage response = await http.GetAsync(url);
200                 return ((int)response.StatusCode, await response.Content.ReadAsStringAsync());
201             }
202         }
203     }
204 }

 

SmsObject.cs

 1 using System;
 2 using System.Collections.Generic;
 3 using System.Text;
 4 
 5 namespace Vino.Core.Communication.SMS
 6 {
 7     public class SmsObject
 8     {
 9         /// <summary>
10         /// 手機號
11         /// </summary>
12         public string Mobile { set; get; }
13 
14         /// <summary>
15         /// 簽名
16         /// </summary>
17         public string Signature { get; set; }
18 
19         /// <summary>
20         /// 模板Key
21         /// </summary>
22         public string TempletKey { set; get; }
23 
24         /// <summary>
25         /// 簡訊數據
26         /// </summary>
27         public IDictionary<string, string> Data { set; get; }
28 
29         /// <summary>
30         /// 業務ID
31         /// </summary>
32         public string OutId { set; get; }
33     }
34 }

 

調用方法

 1                         IDictionary<string, string> data = new Dictionary<string, string>();
 2                         data.Add("code", "111111");
 3                         var sms = new SmsObject
 4                         {
 5                             Mobile = "18812345678",
 6                             Signature = "我的簽名",
 7                             TempletKey = "模板ID",
 8                             Data = data,
 9                             OutId = "OutId"
10                         };
11 
12                         var res = await new AliyunSmsSender(accessKeyId, accessKeySecret).Send(sms);
13                         Debug.WriteLine($"發送結果:{res.success}");
14                         Debug.WriteLine($"Response:{res.response}");

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 1 /// <summary> 2 /// 智能查詢類型 3 /// </summary> 4 5 public enum QueryType : byte 6 { 7 /// <summary> 8 /// 發貨方聯繫人信息 9 /// </summary> 10 [DataMappingAttr ...
  • 1 <html> 2 <head> 3 </head> 4 <body> 5 <form id="recordform" name="recordform" autocomplete="off"> 6 <table cellpadding="0" cellspacing="0" class="tab ...
  • 【C#】 使用Gsof.Native 動態調用 C動態庫 一、背景 使用C# 開發客戶端時候,我們經常會調用一些標準的動態庫或是C的類庫。雖然C# 提供的PInvoke的方式,但因為使用的場景的多變,有些時候可能沒辦法,固定位置的調用,或是需要按需求調用不同的庫。 二、說明 支持根據路徑動態載入DL ...
  • 不知大家有沒有遇到過需要一個函數返回多個值的情況。 當寫代碼要返回多個值的時候,當然可以返回一個數組來實現,但如果遇到需要返回的多個值的類型不同呢?這個時候怎麼辦? c#中,out關鍵字和ref關鍵字就起作用了。(且out更適合用到多返回值的地方) 一個函數中,使用out或ref作為形參前的關鍵字, ...
  • 演算法描述 1.假定未排序序列中第一位為數組最小值,通過與後面的數值進行比較,找到未排序序列中最小值,與未排序序列第一位交換位置; 2.重覆步驟一,對剩餘未排序序列進行比較找出最小值,與未排序序列中第一位數值交換位置,直到數組所有數值排序完成; 代碼實現 完整代碼 ...
  • 推薦 C#參考教程 http://www.csref.cn ...
  • 以下為一個合併學生分數的示例數據,依據主鍵統計各個學生的總分數 拿到這種數據,如果要將數據合併,常規的做法就是通過遍歷實現,代碼簡單也容易理解 不考慮代碼行數為工作績效的情況下,可以通過Linq或擴展方法一行優雅的搞定 通過GroupBy過濾主鍵去掉重覆項,避免ToDictionary時重覆主鍵異常 ...
  • 有時候我們需要對數據表進行篩選,微軟為我們封裝了一個公共方法, DataTable.Select(),其用法如下: Select() Select(string filterExpression) Select(string filterExpression, string sort) Select ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...