php 支付宝回调验证失败,支付宝回调验证签名失败怎么解决?

支付宝回调验证签名失败的解决方法:1、确保使用的验证签名是正确的;2、确保传入的参数是正确的;3、要在支付宝中给你的回调域名授权;3、确保加密解密类型为RSA2。

19dbb769f19eeebe9a53701b61c3e6db.png

回调接口是支持扫码支付方式的回调的,最近业务需要又需要支持移动app的支付方式,回调时却签名验证失败。

支付宝回调验证签名失败怎么解决?

签名验证错误的检查顺序(这里是基于使用官方给的demo,自己封装的请绕道):

1:检查一下你使用的验证签名的方法是否正确?

bool signVerified = AlipaySignature.RSACheckV1(dic, alipay_public_key, config.charset);

2:检查一下你传入的参数是否正确?

参数1:dic,把回调的参数保存到key,value集合中Dictionary dic = new Dictionary();

var form = HttpContext.Current.Request.Form;

string str = "异步通知:\r\n";

foreach (var key in form)

{

dic[key.ToString()] = HttpContext.Current.Request.Form[key.ToString()];

var value = HttpContext.Current.Request.Form[key.ToString()];

//记录日志使用

str += $"{key.ToString()}:{value}\r\n";

}

参数2:alipay_public_key

这个参数是 支付宝公钥!! 很多小伙伴都写成了应用公钥,瞎几把写。

参数3:编码格式,UTF-8,这个一般没人会错。

3:检查一下你的环境

沙盒环境还是线上环境,沙盒环境会出错,具体为什么我不知道,百度来的。要在支付宝中给你的回调域名授权,不授权人家懒得回调给你。

4:检查一下你的加密解密类型

我从官网下载下来的demo里面的解密类型默认是RSA,但是官方文档已经明确说明现在都要用RSA2了,所以记得检查demo的源码public static bool RSACheckV1(IDictionary parameters, string publicKeyPem, string charset)

{

string sign = parameters["sign"];

string sign_type = parameters["sign_type"];

parameters.Remove("sign");

parameters.Remove("sign_type");

string signContent = GetSignContent(parameters);

return RSACheckContent(signContent, sign, publicKeyPem, charset, sign_type);

}

sign_type,这个就是解码类型,demo写的好像“RSA”,我这里改成动态获取了,我们在前期配置的地方也会配置加密类型,从哪获取都可以,别弄错了就行。

5:这里不检查了,回忆一下你的支付宝公钥,是直接存在文本中的,还是写在代码里的(区别:公钥.txt,string 公钥 = “巴拉巴拉巴拉一大堆”),一个是文件,一个是直接代码(我就是代码,所以我一直到最后才解决)(下面的解决方案只针对代码保存支付宝公钥的骚年)string alipay_public_key = "-----BEGIN PUBLIC KEY-----\r\n" + config.alipay_public_key + "-----END PUBLIC KEY-----\r\n\r\n";

bool signVerified = AlipaySignature.RSACheckV1(dic, alipay_public_key, config.charset);

如果是直接写在代码中的,要给支付宝公钥的头跟尾加上标识,具体标识看我贴出来的代码,如果是文件,请自动忽略

还没结束,官方给的demo也是默认找的文件,可是我用的代码存的,哪有文件,所以找不到文件是会报错的,报错直接返回false了,在修改一下源码(自己到AlipaySignature这个类里面去找)public static bool RSACheckContent(string signContent, string sign, string publicKeyPem, string charset, string signType)

{

try

{

if (string.IsNullOrEmpty(charset))

{

charset = DEFAULT_CHARSET;

}

if ("RSA2".Equals(signType))

{

//这里就是要改的地方

//从参数获取

string sPublicKeyPEM = publicKeyPem;

//从文件获取

//string sPublicKeyPEM = File.ReadAllText(publicKeyPem);

RSACryptoServiceProvider rsa = new RSACryptoServiceProvider();

rsa.PersistKeyInCsp = false;

RSACryptoServiceProviderExtension.LoadPublicKeyPEM(rsa, sPublicKeyPEM);

bool bVerifyResultOriginal = rsa.VerifyData(Encoding.GetEncoding(charset).GetBytes(signContent), "SHA256", Convert.FromBase64String(sign));

return bVerifyResultOriginal;

}

else

{

//这里就是要改的地方

//从参数获取

string sPublicKeyPEM = publicKeyPem;

//从文件获取

//string sPublicKeyPEM = File.ReadAllText(publicKeyPem);

RSACryptoServiceProvider rsa = new RSACryptoServiceProvider();

rsa.PersistKeyInCsp = false;

RSACryptoServiceProviderExtension.LoadPublicKeyPEM(rsa, sPublicKeyPEM);

SHA1CryptoServiceProvider sha1 = new SHA1CryptoServiceProvider();

bool bVerifyResultOriginal = rsa.VerifyData(Encoding.GetEncoding(charset).GetBytes(signContent), sha1, Convert.FromBase64String(sign));

return bVerifyResultOriginal;

}

}

catch (Exception e)

{

NLogGetter.NLog.ErrorLog(e);

return false;

}

}

好了,差不多就总结了这么多,基本上可以让你签名验证成功了。

更多相关知识,请访问 PHP中文网!!

本文原创发布php中文网,转载请注明出处,感谢您的尊重!

处理支付宝退款回调PHP 代码可以参考以下示例: ```php <?php // 获取支付宝退款回调数据 $refundData = $_POST; // 验证回调签名 if (!verifySign($refundData)) { exit('fail'); } // 处理退款回调业务逻辑 if ($refundData['trade_status'] == 'SUCCESS') { // 退款成功,更新订单状态等操作 // ... exit('success'); } else { // 退款失败,记录日志等操作 // ... exit('fail'); } /** * 验证回调签名 * * @param array $data 回调数据 * @return bool */ function verifySign($data) { // 获取支付宝公钥 $alipayPublicKey = file_get_contents('https://openapi.alipay.com/gateway.do?service=publicKey&charset=utf-8'); // 创建支付宝公钥对象 $publicKey = openssl_get_publickey($alipayPublicKey); // 处理回调数据,去除 signsign_type 参数 unset($data['sign']); unset($data['sign_type']); ksort($data); $queryString = http_build_query($data); // 验证签名 $result = openssl_verify($queryString, base64_decode($data['sign']), $publicKey, OPENSSL_ALGO_SHA256); // 释放公钥资源 openssl_free_key($publicKey); return $result === 1; } ``` 在上述代码中,`verifySign` 函数用于验证回调签名。首先获取支付宝公钥,然后将回调数据按照参数名进行排序,并将 signsign_type 参数去除,将剩余参数拼接成字符串。最后使用 OpenSSL 库的 `openssl_verify` 函数进行签名验证验证通过返回 1,否则返回 0 或 -1。 在处理退款回调业务逻辑时,首先判断退款状态是否为 SUCCESS,如果是则表示退款成功,可以更新订单状态等操作。如果退款状态不是 SUCCESS,则表示退款失败,可以记录日志等操作。最后需要输出 success 或 fail 字符串,表示处理结果,以告知支付宝服务器是否接收到了回调信息。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值