如何解密平台证书
更新时间:2024.12.20为了保证安全性,微信支付在回调通知和平台证书下载接口中,对关键信息进行了AES-256-GCM加密,商户收到报文后,要解密出明文,APIv3密钥是解密时使用的对称密钥。本章节详细介绍了加密报文的格式,以及如何进行解密。

# 1. 加密报文格式
AES-GCM是一种NIST标准的认证加密算法, 是一种能够同时保证数据的保密性、 完整性和真实性的一种加密模式。
对于加密的数据,我们使用了一个独立的JSON对象来表示。为了方便阅读,示例做了Pretty格式化,并加入了注释。
1{2 "original_type": "transaction", // 加密前的对象类型3 "algorithm": "AEAD_AES_256_GCM", // 加密算法45 // Base64编码后的密文6 "ciphertext": "...",7 // 加密使用的随机串初始化向量)8 "nonce": "...",9 // 附加数据包(可能为空)10 "associated_data": ""11}
注意
加密的随机串,跟签名时使用的随机串没有任何关系,是不一样的。
# 2. 解密
算法接口的细节,可以参考rfc5116 (opens new window)。
大部分编程语言(较新版本)都支持了AEAD_AES_256_GCM。开发者可以参考下列的示例,了解如何使用您的编程语言实现解密。
示例代码
1import java.io.IOException;2import java.security.GeneralSecurityException;3import java.security.InvalidAlgorithmParameterException;4import java.security.InvalidKeyException;5import java.security.NoSuchAlgorithmException;6import java.util.Base64;7import javax.crypto.Cipher;8import javax.crypto.NoSuchPaddingException;9import javax.crypto.spec.GCMParameterSpec;10import javax.crypto.spec.SecretKeySpec;1112public class AesUtil {1314 static final int KEY_LENGTH_BYTE = 32;15 static final int TAG_LENGTH_BIT = 128;16 private final byte[] aesKey;1718 public AesUtil(byte[] key) {19 if (key.length != KEY_LENGTH_BYTE) {20 throw new IllegalArgumentException("无效的ApiV3Key,长度必须为32个字节");21 }22 this.aesKey = key;23 }2425 public String decryptToString(byte[] associatedData, byte[] nonce, String ciphertext)26 throws GeneralSecurityException, IOException {27 try {28 Cipher cipher = Cipher.getInstance("AES/GCM/NoPadding");2930 SecretKeySpec key = new SecretKeySpec(aesKey, "AES");31 GCMParameterSpec spec = new GCMParameterSpec(TAG_LENGTH_BIT, nonce);3233 cipher.init(Cipher.DECRYPT_MODE, key, spec);34 cipher.updateAAD(associatedData);3536 return new String(cipher.doFinal(Base64.getDecoder().decode(ciphertext)), "utf-8");37 } catch (NoSuchAlgorithmException | NoSuchPaddingException e) {38 throw new IllegalStateException(e);39 } catch (InvalidKeyException | InvalidAlgorithmParameterException e) {40 throw new IllegalArgumentException(e);41 }42 }43}
1class AesUtil {2 /**3 * AES key4 *5 * @var string6 */7 private $aesKey;89 const KEY_LENGTH_BYTE = 32;10 const AUTH_TAG_LENGTH_BYTE = 16;1112 /**13 * Constructor14 */15 public16 function __construct($aesKey) {17 if (strlen($aesKey) != self::KEY_LENGTH_BYTE) {18 throw new InvalidArgumentException('无效的ApiV3Key,长度应为32个字节');19 }20 $this - > aesKey = $aesKey;21 }2223 /**24 * Decrypt AEAD_AES_256_GCM ciphertext25 *26 * @param string $associatedData AES GCM additional authentication data27 * @param string $nonceStr AES GCM nonce28 * @param string $ciphertext AES GCM cipher text29 *30 * @return string|bool Decrypted string on success or FALSE on failure31 */32 public33 function decryptToString($associatedData, $nonceStr, $ciphertext) {34 $ciphertext = \base64_decode($ciphertext);35 if (strlen($ciphertext) <= self::AUTH_TAG_LENGTH_BYTE) {36 return false;37 }3839 // ext-sodium (default installed on >= PHP 7.2)40 if (function_exists('\sodium_crypto_aead_aes256gcm_is_available') && \sodium_crypto_aead_aes256gcm_is_available()) {41 return \sodium_crypto_aead_aes256gcm_decrypt($ciphertext, $associatedData, $nonceStr, $this - > aesKey);42 }4344 // ext-libsodium (need install libsodium-php 1.x via pecl)45 if (function_exists('\Sodium\crypto_aead_aes256gcm_is_available') && \Sodium\crypto_aead_aes256gcm_is_available()) {46 return \Sodium\crypto_aead_aes256gcm_decrypt($ciphertext, $associatedData, $nonceStr, $this - > aesKey);47 }4849 // openssl (PHP >= 7.1 support AEAD)50 if (PHP_VERSION_ID >= 70100 && in_array('aes-256-gcm', \openssl_get_cipher_methods())) {51 $ctext = substr($ciphertext, 0, -self::AUTH_TAG_LENGTH_BYTE);52 $authTag = substr($ciphertext, -self::AUTH_TAG_LENGTH_BYTE);5354 return \openssl_decrypt($ctext, 'aes-256-gcm', $this - > aesKey, \OPENSSL_RAW_DATA, $nonceStr,55 $authTag, $associatedData);56 }5758 throw new \RuntimeException('AEAD_AES_256_GCM需要PHP 7.1以上或者安装libsodium-php');59 }60}
1public class AesGcm2{3 private static string ALGORITHM = "AES/GCM/NoPadding";4 private static int TAG_LENGTH_BIT = 128;5 private static int NONCE_LENGTH_BYTE = 12;6 private static string AES_KEY = "yourkeyhere";78 public static string AesGcmDecrypt(string associatedData, string nonce, string ciphertext)9 {10 GcmBlockCipher gcmBlockCipher = new GcmBlockCipher(new AesEngine());11 AeadParameters aeadParameters = new AeadParameters(12 new KeyParameter(Encoding.UTF8.GetBytes(AES_KEY)), 13 128, 14 Encoding.UTF8.GetBytes(nonce), 15 Encoding.UTF8.GetBytes(associatedData));16 gcmBlockCipher.Init(false, aeadParameters);1718 byte[] data = Convert.FromBase64String(ciphertext);19 byte[] plaintext = new byte[gcmBlockCipher.GetOutputSize(data.Length)];20 int length = gcmBlockCipher.ProcessBytes(data, 0, data.Length, plaintext, 0);21 gcmBlockCipher.DoFinal(plaintext, length);22 return Encoding.UTF8.GetString(plaintext);23 }24}
1from cryptography.hazmat.primitives.ciphers.aead import AESGCM2import base6434def decrypt(nonce, ciphertext, associated_data):5 key = "Your32Apiv3Key"67 key_bytes = str.encode(key)8 nonce_bytes = str.encode(nonce)9 ad_bytes = str.encode(associated_data)10 data = base64.b64decode(ciphertext)1112 aesgcm = AESGCM(key_bytes)13 return aesgcm.decrypt(nonce_bytes, data, ad_bytes)
文档是否有帮助