.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
  • 示例項目結構 在 Visual Studio 中創建一個 WinForms 應用程式後,項目結構如下所示: MyWinFormsApp/ │ ├───Properties/ │ └───Settings.settings │ ├───bin/ │ ├───Debug/ │ └───Release/ ...
  • [STAThread] 特性用於需要與 COM 組件交互的應用程式,尤其是依賴單線程模型(如 Windows Forms 應用程式)的組件。在 STA 模式下,線程擁有自己的消息迴圈,這對於處理用戶界面和某些 COM 組件是必要的。 [STAThread] static void Main(stri ...
  • 在WinForm中使用全局異常捕獲處理 在WinForm應用程式中,全局異常捕獲是確保程式穩定性的關鍵。通過在Program類的Main方法中設置全局異常處理,可以有效地捕獲並處理未預見的異常,從而避免程式崩潰。 註冊全局異常事件 [STAThread] static void Main() { / ...
  • 前言 給大家推薦一款開源的 Winform 控制項庫,可以幫助我們開發更加美觀、漂亮的 WinForm 界面。 項目介紹 SunnyUI.NET 是一個基於 .NET Framework 4.0+、.NET 6、.NET 7 和 .NET 8 的 WinForm 開源控制項庫,同時也提供了工具類庫、擴展 ...
  • 說明 該文章是屬於OverallAuth2.0系列文章,每周更新一篇該系列文章(從0到1完成系統開發)。 該系統文章,我會儘量說的非常詳細,做到不管新手、老手都能看懂。 說明:OverallAuth2.0 是一個簡單、易懂、功能強大的許可權+可視化流程管理系統。 有興趣的朋友,請關註我吧(*^▽^*) ...
  • 一、下載安裝 1.下載git 必須先下載並安裝git,再TortoiseGit下載安裝 git安裝參考教程:https://blog.csdn.net/mukes/article/details/115693833 2.TortoiseGit下載與安裝 TortoiseGit,Git客戶端,32/6 ...
  • 前言 在項目開發過程中,理解數據結構和演算法如同掌握蓋房子的秘訣。演算法不僅能幫助我們編寫高效、優質的代碼,還能解決項目中遇到的各種難題。 給大家推薦一個支持C#的開源免費、新手友好的數據結構與演算法入門教程:Hello演算法。 項目介紹 《Hello Algo》是一本開源免費、新手友好的數據結構與演算法入門 ...
  • 1.生成單個Proto.bat內容 @rem Copyright 2016, Google Inc. @rem All rights reserved. @rem @rem Redistribution and use in source and binary forms, with or with ...
  • 一:背景 1. 講故事 前段時間有位朋友找到我,說他的窗體程式在客戶這邊出現了卡死,讓我幫忙看下怎麼回事?dump也生成了,既然有dump了那就上 windbg 分析吧。 二:WinDbg 分析 1. 為什麼會卡死 窗體程式的卡死,入口門檻很低,後續往下分析就不一定了,不管怎麼說先用 !clrsta ...
  • 前言 人工智慧時代,人臉識別技術已成為安全驗證、身份識別和用戶交互的關鍵工具。 給大家推薦一款.NET 開源提供了強大的人臉識別 API,工具不僅易於集成,還具備高效處理能力。 本文將介紹一款如何利用這些API,為我們的項目添加智能識別的亮點。 項目介紹 GitHub 上擁有 1.2k 星標的 C# ...