using Aliyun.Api.Parser; using Aliyun.Api.Util; using FastJSON; using System; using System.Collections; using System.Collections.Generic; using System.Globalization; using System.Security.Cryptography; using System.Text; using System.Xml; using Top.Api; using Top.Api.Util; namespace Aliyun.Api { /// /// 基于REST的阿里云客户端。 /// public class DefaultAliyunClient : IAliyunClient { public const string FORMAT_XML = "xml"; public const string FORMAT_JSON = "json"; public const string HTTP_METHOD_POST = "POST"; public const string SDK_VERSION = "top-sdk-net-dynamicVersionNo"; // SDK自动生成会替换成真实的版本 private AliyunWebUtils webUtils; private ITopLogger topLogger; private bool disableParser = false; // 禁用响应结果解释 private bool disableTrace = false; // 禁用日志调试功能 private IDictionary systemParameters; // 设置所有请求共享的系统级参数 private const string ISO8601_DATE_FORMAT = "yyyy-MM-dd'T'HH:mm:ss'Z'"; private const string ENCODING_UTF8 = "UTF-8"; private string httpMethod = HTTP_METHOD_POST; private string format = FORMAT_XML; private string serverUrl; private string accessKeyId; private string accessKeySecret; #region DefaultTopClient Constructors public DefaultAliyunClient(string serverUrl, string accessKeyId, string accessKeySecret) { this.accessKeyId = accessKeyId; this.accessKeySecret = accessKeySecret; this.serverUrl = serverUrl; this.webUtils = new AliyunWebUtils(); this.topLogger = Top.Api.Log.Instance; } public DefaultAliyunClient(string serverUrl, string accessKeyId, string accessKeySecret, string format) : this(serverUrl, accessKeyId, accessKeySecret) { this.format = format; } #endregion public void SetTopLogger(ITopLogger topLogger) { this.topLogger = topLogger; } public void SetTimeout(int timeout) { this.webUtils.Timeout = timeout; } public void SetDisableParser(bool disableParser) { this.disableParser = disableParser; } public void SetDisableTrace(bool disableTrace) { this.disableTrace = disableTrace; } public void SetSystemParameters(IDictionary systemParameters) { this.systemParameters = systemParameters; } #region ITopClient Members public T Execute(IAliyunRequest request) where T : AliyunResponse { return Execute(request, null); } public T Execute(IAliyunRequest request, string session) where T : AliyunResponse { return Execute(request, session, DateTime.Now); } public T Execute(IAliyunRequest request, string session, DateTime timestamp) where T : AliyunResponse { return DoExecute(request, session, timestamp); } #endregion private T DoExecute(IAliyunRequest request, string session, DateTime timestamp) where T : AliyunResponse { // 提前检查业务参数 try { request.Validate(); } catch (TopException e) { return CreateErrorResponse(e.ErrorCode, e.ErrorMsg); } // 添加协议级请求参数 TopDictionary txtParams = new TopDictionary(request.GetParameters()); txtParams.AddAll(this.systemParameters); AddCommonParams(request, txtParams); string reqUrl = webUtils.BuildGetUrl(this.serverUrl, txtParams); try { string body; if (request is IAliyunUploadRequest) // 是否需要上传文件 { IAliyunUploadRequest uRequest = (IAliyunUploadRequest)request; IDictionary fileParams = TopUtils.CleanupDictionary(uRequest.GetFileParameters()); body = webUtils.DoPost(this.serverUrl.TrimEnd('/'), txtParams, fileParams); } else { body = webUtils.DoPost(this.serverUrl.TrimEnd('/'), txtParams); } // 解释响应结果 T rsp; if (disableParser) { rsp = Activator.CreateInstance(); rsp.Body = body; } else { if (FORMAT_XML.Equals(format)) { IAliyunParser tp = new AliyunXmlParser(); rsp = tp.Parse(body); } else { IAliyunParser tp = new AliyunJsonParser(); rsp = tp.Parse(body); } } // 追踪错误的请求 if (!disableTrace && rsp.IsError) { StringBuilder sb = new StringBuilder(reqUrl).Append(" response error!\r\n").Append(rsp.Body); topLogger.Warn(sb.ToString()); } return rsp; } catch (Exception e) { if (!disableTrace) { StringBuilder sb = new StringBuilder(reqUrl).Append(" request error!\r\n").Append(e.StackTrace); topLogger.Error(sb.ToString()); } throw e; } } private T CreateErrorResponse(string errCode, string errMsg) where T : AliyunResponse { T rsp = Activator.CreateInstance(); rsp.Code = errCode; rsp.Message = errMsg; if (FORMAT_XML.Equals(format)) { XmlDocument root = new XmlDocument(); XmlElement bodyE = root.CreateElement("Error"); XmlElement codeE = root.CreateElement("Code"); codeE.InnerText = errCode; bodyE.AppendChild(codeE); XmlElement msgE = root.CreateElement("Message"); msgE.InnerText = errMsg; bodyE.AppendChild(msgE); root.AppendChild(bodyE); rsp.Body = root.OuterXml; } else { IDictionary errObj = new Dictionary(); errObj.Add("Code", errCode); errObj.Add("Message", errMsg); string body = JSON.ToJSON(errObj); rsp.Body = body; } return rsp; } private void AddCommonParams(IAliyunRequest request, TopDictionary parameters) where T : AliyunResponse { String[] strArray = request.GetApiName().Split('.'); if (strArray.Length < 5) { throw new TopException("Wrong api name."); } String action = strArray[3]; parameters.Add("Action", action); String version = strArray[4]; parameters.Add("Version", version); parameters.Add("AccessKeyId", accessKeyId); parameters.Add("Timestamp", FormatIso8601Date(DateTime.Now)); parameters.Add("SignatureMethod", "HMAC-SHA1"); parameters.Add("SignatureVersion", "1.0"); parameters.Add("SignatureNonce", Guid.NewGuid().ToString()); // 可以使用GUID作为SignatureNonce parameters.Add("Format", format); // 计算签名,并将签名结果加入请求参数中 parameters.Add("Signature", ComputeSignature(parameters)); } private String ComputeSignature(TopDictionary parameters) { const String SEPARATOR = "&"; // 生成规范化请求字符串 StringBuilder canonicalizedQueryString = new StringBuilder(); var orderedParameters = SortDictionary(parameters); foreach (var p in orderedParameters) { canonicalizedQueryString.Append("&") .Append(PercentEncode(p.Key)).Append("=") .Append(PercentEncode(p.Value)); } // 生成用于计算签名的字符串 stringToSign StringBuilder stringToSign = new StringBuilder(); stringToSign.Append(httpMethod).Append(SEPARATOR); stringToSign.Append(PercentEncode("/")).Append(SEPARATOR); stringToSign.Append(PercentEncode( canonicalizedQueryString.ToString().Substring(1))); // 注意accessKeySecret后面要加入一个字符"&" String signature = CalculateSignature(accessKeySecret + "&", stringToSign.ToString()); return signature; } private static String FormatIso8601Date(DateTime date) { // 注意使用UTC时间 return date.ToUniversalTime().ToString(ISO8601_DATE_FORMAT, CultureInfo.CreateSpecificCulture("en-US")); } private static String CalculateSignature(String key, String stringToSign) { // 使用HmacSHA1算法计算HMAC值 using (var algorithm = KeyedHashAlgorithm.Create("HMACSHA1")) { algorithm.Key = Encoding.GetEncoding(ENCODING_UTF8).GetBytes(key.ToCharArray()); return Convert.ToBase64String( algorithm.ComputeHash( Encoding.GetEncoding(ENCODING_UTF8).GetBytes(stringToSign.ToCharArray()))); } } private static string PercentEncode(String value) { StringBuilder stringBuilder = new StringBuilder(); string text = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789-_.~"; byte[] bytes = Encoding.GetEncoding(ENCODING_UTF8).GetBytes(value); foreach (char c in bytes) { if (text.IndexOf(c) >= 0) { stringBuilder.Append(c); } else { stringBuilder.Append("%").Append( string.Format(CultureInfo.InvariantCulture, "{0:X2}", (int)c)); } } return stringBuilder.ToString(); } private static Dictionary SortDictionary(Dictionary dic) { ArrayList arrayList = new ArrayList(dic.Keys); arrayList.Sort(StringComparer.Ordinal); Dictionary sortedDictionary = new Dictionary(); foreach (string key in arrayList) { sortedDictionary.Add(key, dic[key]); } return sortedDictionary; } } }