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

学更好的别人,

做更好的自己。

——《微卡智享》

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

前言

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

腾讯云SDK调用遇到的问题

微卡智享

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

解压后我们直接用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;
        }


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

扫描二维码

获取更多精彩

微卡智享

「 往期文章 」

分享|自己平时常用的几款效率工具

项目实战|C#Socket通讯方式改造(二)--利用Quartz实现定时任务处理

项目实战|C#Socket通讯方式改造(一)--Socket实现Ftp的上传和下载

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Vaccae

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值