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;
}
}
}