2021-07-20 01:20:31 +08:00
using System ;
using System.Text ;
namespace SKIT.FlurlHttpClient.Wechat.TenpayV3
{
/// <summary>
/// 为 <see cref="WechatTenpayClient"/> 提供响应签名验证的扩展方法。
/// </summary>
public static class WechatTenpayClientResponseVerificationExtensions
{
/// <summary>
/// <para>验证响应签名。</para>
/// <para>REF: https://pay.weixin.qq.com/wiki/doc/apiv3/wechatpay/wechatpay4_1.shtml </para>
/// <para>REF: https://pay.weixin.qq.com/wiki/doc/apiv3_partner/wechatpay/wechatpay4_1.shtml </para>
/// </summary>
/// <typeparam name="TResponse"></typeparam>
/// <param name="client"></param>
/// <param name="response"></param>
/// <returns></returns>
public static bool VerifyResponseSignature < TResponse > ( this WechatTenpayClient client , TResponse response )
where TResponse : WechatTenpayResponse
2021-11-25 11:58:06 +08:00
{
return VerifyResponseSignature ( client , response , out _ ) ;
}
/// <summary>
/// <para>验证响应签名。</para>
/// <para>REF: https://pay.weixin.qq.com/wiki/doc/apiv3/wechatpay/wechatpay4_1.shtml </para>
/// <para>REF: https://pay.weixin.qq.com/wiki/doc/apiv3_partner/wechatpay/wechatpay4_1.shtml </para>
/// </summary>
/// <typeparam name="TResponse"></typeparam>
/// <param name="client"></param>
/// <param name="response"></param>
/// <param name="error"></param>
/// <returns></returns>
/// <exception cref="ArgumentNullException"></exception>
public static bool VerifyResponseSignature < TResponse > ( this WechatTenpayClient client , TResponse response , out Exception ? error )
where TResponse : WechatTenpayResponse
2021-07-20 01:20:31 +08:00
{
if ( client = = null ) throw new ArgumentNullException ( nameof ( client ) ) ;
if ( response = = null ) throw new ArgumentNullException ( nameof ( response ) ) ;
2022-01-21 14:41:40 +08:00
if ( client . PlatformCertificateManager ! = null )
2021-07-20 01:20:31 +08:00
{
try
{
2022-01-21 14:41:40 +08:00
var cert = client . PlatformCertificateManager . GetEntry ( response . WechatpayCertificateSerialNumber ) ! ;
2021-11-25 11:58:06 +08:00
if ( ! cert . HasValue )
{
error = new Exceptions . WechatTenpayResponseVerificationException ( "Verify signature of response failed, because there is no platform certificate matched the serial number." ) ;
return false ;
}
2021-07-20 01:20:31 +08:00
2021-11-25 11:58:06 +08:00
error = null ;
return Utilities . RSAUtility . VerifyWithSHA256ByCertificate (
certificate : cert . Value . Certificate ,
2021-07-20 01:20:31 +08:00
plainText : GetPlainTextForSignature ( response ) ,
signature : response . WechatpaySignature
) ;
}
2022-01-21 14:30:17 +08:00
catch ( Exception ex )
2021-11-25 11:58:06 +08:00
{
error = new Exceptions . WechatTenpayResponseVerificationException ( "Verify signature of response failed. Please see the `InnerException` for more details." , ex ) ;
return false ;
}
2021-07-20 01:20:31 +08:00
}
2021-08-02 22:06:22 +08:00
2021-11-25 11:58:06 +08:00
error = new Exceptions . WechatTenpayResponseVerificationException ( "Verify signature of response failed, because there is no platform certificate in the manager." ) ;
2021-08-02 22:06:22 +08:00
return false ;
2021-07-20 01:20:31 +08:00
}
private static string GetPlainTextForSignature ( WechatTenpayResponse response )
{
string timestamp = response . WechatpayTimestamp ;
string nonce = response . WechatpayNonce ;
string body = Encoding . UTF8 . GetString ( response . RawBytes ) ;
return $"{timestamp}\n{nonce}\n{body}\n" ;
}
}
}