学更好的别人,
做更好的自己。
——《微卡智享》
本文长度为3821字,预计阅读8分钟
前言
前阵子因为项目中要对接腾讯的云支付,腾讯云里有C#相关的SDK,所以就直接从官网下载下来方便开发,本篇就是主要记录一下使用SDK调用腾讯云支付时遇到的问题及解决方法。
腾讯云SDK调用遇到的问题
![](https://i-blog.csdnimg.cn/blog_migrate/3c49773a66a660b04cbe2b89c3b9f137.png)
微卡智享
在腾讯云开发接口文档中找到了下载地址
解压后我们直接用VS2019打开
由于我做的主要就是收银系统,所以接口只对接刷卡付这一项就可以了,选择MicroPayPage.aspx然后点击运行调试。
来到了支付页面,我们先随便输个授权码,点击提交试试
然后就出来了远程主机强迫关闭了一个现在的连接错误。
01
排查问题
遇到了上面的情况,首先给我的感觉就是什么鬼,官方的SDK竟然都调用不成功,录了个上面的视频也发给了腾讯云的官方,官方人员后来反馈说一直没遇到过这个问题。
于是就在网上找有关c#的这个报错的解决方案,基本上大部分都是说用HttpWebRequest访问Https需要加上支持tls1.2
//设置只支持tls1.2
ServicePointManager.SecurityProtocol = SecurityProtocolType.Tls12;
OK,那我们就跟着代码一步一步地找,结果在HttpService类中POST请求里用到了HttpWebRequest进行通讯
结果在代码里发现其实有这段代码,怎么别人的问题这样解决,我这里面有这段还是不行,看来还要是自己找一下原因。
原HttpService类中Post函数方法
public static string Post(string req_data, string url, bool isUseCert, int timeout)
{
System.GC.Collect();//垃圾回收,回收没有正常关闭的http连接
string result = "";//返回结果
HttpWebRequest request = null;
HttpWebResponse response = null;
Stream reqStream = null;
try
{
//设置最大连接数
ServicePointManager.DefaultConnectionLimit = 200;
//设置https验证方式
if (url.StartsWith("https", StringComparison.OrdinalIgnoreCase))
{
ServicePointManager.ServerCertificateValidationCallback =
new RemoteCertificateValidationCallback(CheckValidationResult);
}
/***************************************************************
* 下面设置HttpWebRequest的相关属性
* ************************************************************/
request = (HttpWebRequest)WebRequest.Create(url);
request.Method = "POST";
request.Timeout = timeout * 1000;
//设置代理服务器
//WebProxy proxy = new WebProxy(); //定义一个网关对象
//proxy.Address = new Uri(WxPayConfig.PROXY_URL); //网关服务器端口:端口
//request.Proxy = proxy;
//设置POST的数据类型和长度
request.ContentType = "application/json";
byte[] data = System.Text.Encoding.UTF8.GetBytes(req_data);
request.ContentLength = data.Length;
//是否使用证书
if (isUseCert)
{
string path = HttpContext.Current.Request.PhysicalApplicationPath;
X509Certificate2 cert = new X509Certificate2(path + CloudPayConfig.ROOTCA_PATH);
request.ClientCertificates.Add(cert);
Log.Debug("CloudPayPayApi", "Post used cert");
}
//设置只支持tls1.2
ServicePointManager.SecurityProtocol = SecurityProtocolType.Tls12;
//往服务器写入数据
reqStream = request.GetRequestStream();
reqStream.Write(data, 0, data.Length);
reqStream.Close();
//获取服务端返回
response = (HttpWebResponse)request.GetResponse();
//获取服务端返回数据
StreamReader sr = new StreamReader(response.GetResponseStream(), Encoding.UTF8);
result = sr.ReadToEnd().Trim();
sr.Close();
}
catch (System.Threading.ThreadAbortException e)
{
Log.Error("HttpService", "Thread - caught ThreadAbortException - resetting.");
Log.Error("Exception message: {0}", e.Message);
System.Threading.Thread.ResetAbort();
}
catch (WebException e)
{
Log.Error("HttpService", e.ToString());
if (e.Status == WebExceptionStatus.ProtocolError)
{
Log.Error("HttpService", "StatusCode : " + ((HttpWebResponse)e.Response).StatusCode);
Log.Error("HttpService", "StatusDescription : " + ((HttpWebResponse)e.Response).StatusDescription);
}
throw new CloudPayException(e.ToString());
}
catch (Exception e)
{
Log.Error("HttpService", e.ToString());
throw new CloudPayException(e.ToString());
}
finally
{
//关闭连接和流
if (response != null)
{
response.Close();
}
if (request != null)
{
request.Abort();
}
}
return result;
}
02
调试代码
遇到上面的问题后,只能自己一步步的调试看看问题出在哪里,经过反复几次调试后发现一个规律:
每次刚开始运行第一次时当运行到request.GetRequestStream()时会抛出异常:状态为WebExceptionStatus.SendFailure,然后提示远程主机强制关闭了一个连接。
如果再继续请求的话,通讯就一切都正常了。
为了验证一直是我自己机器问题还是都有,又换了一台机器,也出现了这个情况,规律也是一样的。于是这就好办了,我们可以改造一个这个POST的函数,加一个输入参数来设置重新调用就可以解决这个问题了。
# | 需要注意的问题 |
---|---|
1 | 输入的参数有默认值,这样原来调用的方式可以不用改变 |
2 | 出现发送请求错误后重新调用,我们还要判断出是否已经是重新调用过了,防止请求无限循环 |
按照上面两个需要注意的问题,我们可以在POST函数最后加上一个bool的变量,默认传入的false,当默认调用时,如果出现发送请求错误时,我们重新调用这个POST函数,并把最后bool的变量改为true调用,如果再请求错误并且判断这个bool值已经是重复请求的话,那我们就正常返回错误信息,防止无限循环。
03
修改POST代码
输入参数的最后加入了一个bool iscall的参数,并且直接默认值为false
在catch里面加入status为WebExceptionStatus.SendFailure的判断,当默认的iscall是false时,我们就再重新调用Post函数,并且iscall的值改为true,说明我们重新调用了POST。
如果iscall为true时,说明我们已经重复调用过了,如果还是遇到这个报错,就直接抛出异常即可。
最终按上面的思路后我们修改完的代码
public static string Post(string req_data, string url, bool isUseCert, int timeout, bool iscall = false)
{
System.GC.Collect();//垃圾回收,回收没有正常关闭的http连接
string result = "";//返回结果
HttpWebRequest request = null;
HttpWebResponse response = null;
Stream reqStream = null;
try
{
//设置最大连接数
ServicePointManager.DefaultConnectionLimit = 200;
//设置https验证方式
if (url.StartsWith("https", StringComparison.OrdinalIgnoreCase))
{
ServicePointManager.ServerCertificateValidationCallback =
new RemoteCertificateValidationCallback(CheckValidationResult);
}
/***************************************************************
* 下面设置HttpWebRequest的相关属性
* ************************************************************/
request = (HttpWebRequest)WebRequest.Create(url);
request.Method = "POST";
request.Timeout = timeout * 1000;
//设置代理服务器
//WebProxy proxy = new WebProxy(); //定义一个网关对象
//proxy.Address = new Uri(WxPayConfig.PROXY_URL); //网关服务器端口:端口
//request.Proxy = proxy;
//设置POST的数据类型和长度
request.ContentType = "application/json";
byte[] data = System.Text.Encoding.UTF8.GetBytes(req_data);
request.ContentLength = data.Length;
//是否使用证书
if (isUseCert)
{
string path = HttpContext.Current.Request.PhysicalApplicationPath;
X509Certificate2 cert = new X509Certificate2(path + CloudPayConfig.ROOTCA_PATH);
request.ClientCertificates.Add(cert);
Log.Debug("CloudPayPayApi", "Post used cert");
}
//设置只支持tls1.2
ServicePointManager.SecurityProtocol = SecurityProtocolType.Tls12;
//往服务器写入数据
reqStream = request.GetRequestStream();
reqStream.Write(data, 0, data.Length);
reqStream.Close();
//获取服务端返回
response = (HttpWebResponse)request.GetResponse();
//获取服务端返回数据
StreamReader sr = new StreamReader(response.GetResponseStream(), Encoding.UTF8);
result = sr.ReadToEnd().Trim();
sr.Close();
}
catch (System.Threading.ThreadAbortException e)
{
Log.Error("HttpService", "Thread - caught ThreadAbortException - resetting.");
Log.Error("Exception message: {0}", e.Message);
System.Threading.Thread.ResetAbort();
}
catch (WebException e)
{
Log.Error("HttpService", e.ToString());
if (e.Status == WebExceptionStatus.ProtocolError)
{
Log.Error("HttpService", "StatusCode : " + ((HttpWebResponse)e.Response).StatusCode);
Log.Error("HttpService", "StatusDescription : " + ((HttpWebResponse)e.Response).StatusDescription);
}
if (e.Status == WebExceptionStatus.SendFailure)
{
if (!iscall)
{
return Post(req_data, url, isUseCert, timeout, true);
}
else
{
throw new CloudPayException(e.ToString());
}
}
else
{
throw new CloudPayException(e.ToString());
}
}
catch (Exception e)
{
Log.Error("HttpService", e.ToString());
throw new CloudPayException(e.ToString());
}
finally
{
//关闭连接和流
if (response != null)
{
response.Close();
}
if (request != null)
{
request.Abort();
}
}
return result;
}
简单的改了几行代码,然后完美的解决了问题。
完
扫描二维码
获取更多精彩
微卡智享
「 往期文章 」