首页 腾讯云动态正文

笔记|C#腾讯云支付SDK遇到的问题及解决

云返利网 腾讯云动态 2020-08-21 21:38:31 8 0 腾讯云服务


学更好的别人,

做更好的自己。

——《微卡智享》




本文长度为3821,预计阅读8分钟




前言

前阵子因为项目中要对接腾讯的云支付,腾讯云里有C#相关的SDK,所以就直接从官网下载下来方便开发,本篇就是主要记录一下使用SDK调用腾讯云支付时遇到的问题及解决方法。


腾讯云SDK调用遇到的问题

微卡智享


腾讯 云开发 接口文档中找到了下载地址


解压后我们直接用VS2019打开



由于我做的主要就是 收银系统 ,所以接口只对接刷卡付这一项就可以了, 选择 Micro Pay Pag e.aspx 然后点击运行 调试。

来到了支付页面,我们先随便输个授权码,点击提交试试

然后就出来了远程主机强迫关闭了一个现在的连接错误。


01

排查问题


遇到了上面的情况,首先给我的感觉就是什么鬼,官方的SDK竟然都调用不成功,录了个上面的视频也发给了腾讯云的官方,官方人员后来反馈说一直没遇到过这个问题。

于是就在网上找有关c#的这个报错的解决方案,基本上大部分都是说用HttpWebRequest访问Https需要加上支持tls1.2
//设置只支持tls1.2ServicePointManager.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; }

简单的改了几行代码,然后完美的解决了问题。




扫描二维码

获取更多精彩

微卡智享



「 往期文章 」









本文分享自微信公众号 - 微卡智享(VaccaeShare)。
如有侵权,请联系 support@oschina.cn 删除。
本文参与“”,欢迎正在阅读的你也加入,一起分享。

【关于云返利网】

云返利网是阿里云、腾讯云、华为云产品推广返利平台,在各个品牌云产品官网优惠活动之外,云返利网还提供返利。您可以无门槛获得阿里云、华为云、腾讯云所有产品返利,在官网下单后就可以领取,无论是自己用、公司用还是帮客户采购,您个人都可以获得返利。云返利网的目标是让返利更多、更快、更简单!详情咨询13121395187(微信同号)