using FastJSON;
using System;
using System.Collections;
using System.Collections.Generic;
using System.Xml;
using Top.Api.Parser;
using Top.Api.Util;
using Top.Api;
using Top.Api.DingTalk;
namespace DingTalk.Api
{
///
/// 基于REST的TOP客户端。
///
public class DefaultDingTalkClient : IDingTalkClient
{
internal string serverUrl;
internal string format = Constants.FORMAT_XML;
internal DateTime dt1970 = new DateTime(1970, 1, 1, 0, 0, 0, 0);
internal WebUtils webUtils;
internal ITopLogger topLogger;
internal bool disableParser = false; // 禁用响应结果解释
internal bool disableTrace = false; // 禁用日志调试功能
internal bool useSimplifyJson = false; // 是否采用精简化的JSON返回
internal bool useGzipEncoding = true; // 是否启用响应GZIP压缩
internal bool useJsonString = false; // 是否采用JsonString生成
internal IDictionary systemParameters; // 设置所有请求共享的系统级参数
#region DefaultDingTalkClient Constructors
public DefaultDingTalkClient(string serverUrl)
{
this.serverUrl = serverUrl;
this.webUtils = new WebUtils();
this.topLogger = Top.Api.Log.Instance;
}
public DefaultDingTalkClient(string serverUrl,string format)
: this(serverUrl)
{
this.format = format;
}
#endregion
public void SetTimeout(int timeout)
{
this.webUtils.Timeout = timeout;
}
public void SetReadWriteTimeout(int readWriteTimeout)
{
this.webUtils.ReadWriteTimeout = readWriteTimeout;
}
public void SetDisableParser(bool disableParser)
{
this.disableParser = disableParser;
}
public void SetDisableTrace(bool disableTrace)
{
this.disableTrace = disableTrace;
}
public void SetUseSimplifyJson(bool useSimplifyJson)
{
this.useSimplifyJson = useSimplifyJson;
}
public void SetUseGzipEncoding(bool useGzipEncoding)
{
this.useGzipEncoding = useGzipEncoding;
}
public void SetIgnoreSSLCheck(bool ignore)
{
this.webUtils.IgnoreSSLCheck = ignore;
}
public void SetSystemParameters(IDictionary systemParameters)
{
this.systemParameters = systemParameters;
}
#region IDingTalkClient Members
public virtual T Execute(IDingTalkRequest request) where T : DingTalkResponse
{
return DoExecute(request, null, DateTime.Now);
}
public virtual T Execute(IDingTalkRequest request, string session) where T : DingTalkResponse
{
return DoExecute(request, session, DateTime.Now);
}
public virtual T Execute(IDingTalkRequest request, string session, DateTime timestamp) where T : DingTalkResponse
{
return DoExecute(request, session, timestamp);
}
public T Execute(IDingTalkRequest request, string accessKey, string accessSecret) where T : DingTalkResponse
{
return Execute(request, accessKey, accessSecret, null, null);
}
public T Execute(IDingTalkRequest request, string accessKey, string accessSecret, string suiteTicket) where T : DingTalkResponse
{
return Execute(request, accessKey, accessSecret, suiteTicket, null);
}
public T Execute(IDingTalkRequest request, string accessKey, string accessSecret, string suiteTicket, string corpId) where T : DingTalkResponse
{
if (request.GetApiCallType() == null || request.GetApiCallType().Equals(DingTalkConstants.CALL_TYPE_TOP))
{
return DoExecuteTop(request, null, DateTime.Now);
}
else
{
return DoExecuteOApi(request, null, accessKey, accessSecret, suiteTicket, corpId, DateTime.Now);
}
}
#endregion
private T DoExecute(IDingTalkRequest request, string session, DateTime timestamp) where T : DingTalkResponse
{
if(request.GetApiCallType() == null || request.GetApiCallType().Equals(DingTalkConstants.CALL_TYPE_TOP)) {
return DoExecuteTop(request, session, timestamp);
} else {
return DoExecuteOApi(request, session, null, null, null, null, timestamp);
}
}
private T DoExecuteTop(IDingTalkRequest request, string session, DateTime timestamp) where T : DingTalkResponse
{
long start = DateTime.Now.Ticks;
// 提前检查业务参数
try
{
request.Validate();
}
catch (TopException e)
{
return CreateErrorResponse(e.ErrorCode, e.ErrorMsg);
}
// 添加协议级请求参数
TopDictionary txtParams = new TopDictionary(request.GetParameters());
txtParams.Add(Constants.METHOD, request.GetApiName());
txtParams.Add(Constants.VERSION, "2.0");
txtParams.Add(Constants.FORMAT, format);
txtParams.Add(Constants.PARTNER_ID, GetSdkVersion());
txtParams.Add(Constants.TIMESTAMP, timestamp);
txtParams.Add(Constants.SESSION, session);
txtParams.AddAll(this.systemParameters);
if (this.useSimplifyJson)
{
txtParams.Add(Constants.SIMPLIFY, "true");
}
// 添加头部参数
if (this.useGzipEncoding)
{
request.GetHeaderParameters()[Constants.ACCEPT_ENCODING] = Constants.CONTENT_ENCODING_GZIP;
}
string realServerUrl = GetServerUrl(this.serverUrl, request.GetApiName(), session);
string reqUrl = WebUtils.BuildRequestUrl(realServerUrl, txtParams);
try
{
string body;
if (request is IDingTalkUploadRequest) // 是否需要上传文件
{
IDingTalkUploadRequest uRequest = (IDingTalkUploadRequest)request;
IDictionary fileParams = TopUtils.CleanupDictionary(uRequest.GetFileParameters());
body = webUtils.DoPost(realServerUrl, txtParams, fileParams, request.GetHeaderParameters());
}
else
{
body = webUtils.DoPost(realServerUrl, txtParams, request.GetHeaderParameters());
}
// 解释响应结果
T rsp;
if (disableParser)
{
rsp = Activator.CreateInstance();
rsp.Body = body;
}
else
{
if (Constants.FORMAT_XML.Equals(format))
{
ITopParser tp = new TopXmlParser();
rsp = tp.Parse(body);
}
else
{
ITopParser tp;
if (useSimplifyJson)
{
tp = new TopSimplifyJsonParser();
}
else
{
tp = new TopJsonParser();
}
rsp = tp.Parse(body);
}
}
// 追踪错误的请求
if (rsp.IsError)
{
TimeSpan latency = new TimeSpan(DateTime.Now.Ticks - start);
TraceApiError(request.GetApiName(), serverUrl, txtParams, latency.TotalMilliseconds, rsp.Body);
}
return rsp;
}
catch (Exception e)
{
TimeSpan latency = new TimeSpan(DateTime.Now.Ticks - start);
TraceApiError(request.GetApiName(), serverUrl, txtParams, latency.TotalMilliseconds, e.GetType() + ": " + e.Message);
throw e;
}
}
private T DoExecuteOApi(IDingTalkRequest request, string session, string accessKey, string accessSecret, string suiteTicket, string corpId, DateTime timestamp) where T : DingTalkResponse
{
long start = DateTime.Now.Ticks;
// 提前检查业务参数
try
{
request.Validate();
}
catch (TopException e)
{
return CreateErrorResponse(e.ErrorCode, e.ErrorMsg);
}
this.format = Constants.FORMAT_JSON;
// 添加协议级请求参数
TopDictionary txtParams = new TopDictionary(request.GetParameters());
txtParams.Add(DingTalkConstants.ACCESS_TOKEN, session);
// 添加头部参数
if (this.useGzipEncoding)
{
request.GetHeaderParameters()[Constants.ACCEPT_ENCODING] = Constants.CONTENT_ENCODING_GZIP;
}
string realServerUrl = null;
// 签名优先
if (accessKey != null)
{
long dingTimestamp = GetTimestamp(DateTime.UtcNow);
// 验证签名有效性
String canonicalString = DingTalkSignatureUtil.GetCanonicalStringForIsv(dingTimestamp, suiteTicket);
String signature = DingTalkSignatureUtil.ComputeSignature(accessSecret, canonicalString);
IDictionary ps = new Dictionary();
ps.Add("accessKey", accessKey);
ps.Add("signature", signature);
ps.Add("timestamp", dingTimestamp + "");
if (suiteTicket != null)
{
ps.Add("suiteTicket", suiteTicket);
}
if (corpId != null)
{
ps.Add("corpId", corpId);
}
String queryStr = DingTalkSignatureUtil.ParamToQueryString(ps, "utf-8");
if (this.serverUrl.IndexOf("?") > 0)
{
realServerUrl = this.serverUrl + "&" + queryStr;
}
else
{
realServerUrl = this.serverUrl + "?" + queryStr;
}
}
else
{
if (this.serverUrl.IndexOf("?") > 0)
{
realServerUrl = this.serverUrl + (session != null && session != "" ? ("&access_token=" + session) : "");
}
else
{
realServerUrl = this.serverUrl + (session != null && session != "" ? ("?access_token=" + session) : "");
}
}
try
{
string body;
if(request.GetHttpMethod() == "POST")
{
if (request is IDingTalkUploadRequest) // 是否需要上传文件
{
IDingTalkUploadRequest uRequest = (IDingTalkUploadRequest)request;
IDictionary fileParams = TopUtils.CleanupDictionary(uRequest.GetFileParameters());
body = webUtils.DoPost(realServerUrl, null, fileParams, request.GetHeaderParameters());
}
else
{
IDictionary jsonParams = new Dictionary();
foreach (string key in request.GetParameters().Keys)
{
string value = request.GetParameters()[key];
if(useJsonString)
{
jsonParams.Add(key, value);
}
else
{
if (value.StartsWith("[") && value.EndsWith("]"))
{
IList childMap = (IList)TopUtils.JsonToObject(value);
jsonParams.Add(key, childMap);
}
else if (value.StartsWith("{") && value.EndsWith("}"))
{
IDictionary childMap = (IDictionary)TopUtils.JsonToObject(value);
jsonParams.Add(key, childMap);
}
else
{
jsonParams.Add(key, value);
}
}
}
body = webUtils.DoPostWithJson(realServerUrl, jsonParams, request.GetHeaderParameters());
}
} else
{
body = webUtils.DoGet(realServerUrl, request.GetParameters());
}
// 解释响应结果
T rsp;
if (disableParser)
{
rsp = Activator.CreateInstance();
rsp.Body = body;
}
else
{
ITopParser tp = new DingTalkJsonParser();
rsp = tp.Parse(body);
}
// 追踪错误的请求
if (rsp.IsError)
{
TimeSpan latency = new TimeSpan(DateTime.Now.Ticks - start);
TraceApiError(request.GetApiName(), serverUrl, txtParams, latency.TotalMilliseconds, rsp.Body);
}
return rsp;
}
catch (Exception e)
{
TimeSpan latency = new TimeSpan(DateTime.Now.Ticks - start);
TraceApiError(request.GetApiName(), serverUrl, txtParams, latency.TotalMilliseconds, e.GetType() + ": " + e.Message);
throw e;
}
}
internal virtual string GetServerUrl(string serverUrl, string apiName, string session)
{
return serverUrl;
}
internal virtual string GetSdkVersion()
{
return Constants.SDK_VERSION;
}
internal T CreateErrorResponse(string errCode, string errMsg) where T : DingTalkResponse
{
T rsp = Activator.CreateInstance();
rsp.ErrCode = errCode;
rsp.ErrMsg = errMsg;
if (Constants.FORMAT_XML.Equals(format))
{
XmlDocument root = new XmlDocument();
XmlElement bodyE = root.CreateElement(Constants.ERROR_RESPONSE);
XmlElement codeE = root.CreateElement(Constants.ERROR_CODE);
codeE.InnerText = errCode;
bodyE.AppendChild(codeE);
XmlElement msgE = root.CreateElement(Constants.ERROR_MSG);
msgE.InnerText = errMsg;
bodyE.AppendChild(msgE);
root.AppendChild(bodyE);
rsp.Body = root.OuterXml;
}
else
{
IDictionary errObj = new Dictionary();
errObj.Add(Constants.ERROR_CODE, errCode);
errObj.Add(Constants.ERROR_MSG, errMsg);
IDictionary root = new Dictionary();
root.Add(Constants.ERROR_RESPONSE, errObj);
string body = JSON.ToJSON(root);
rsp.Body = body;
}
return rsp;
}
internal void TraceApiError(string apiName, string url, Dictionary parameters, double latency, string errorMessage)
{
if (!disableTrace)
{
this.topLogger.TraceApiError("dingtalk",apiName, url, parameters, latency, errorMessage);
}
}
private long GetTimestamp(DateTime dateTime)
{
return (dateTime.Ticks - dt1970.Ticks) / 10000;
}
}
}