search
Homephp教程php手册PHP支付宝接口RSA验证

这两天一直困扰的PHP RSA签名验证问题终于解决了,由于之前RSA接触的不多,再加上官方至今还未有PHP的SDK可供参考,因此走了一些弯路,写在这里和大家分享。

    虽然支付宝官方还未提供相关SDK,PHP确实可以实现RSA方式的签名,这点其实很重要,由于不熟悉,在遇到困难的时候,经常会不由自主地想到是否PHP不支持RSA签名,干脆用MD5得了,这样就没有了前进的动力。其实说穿了MD5和RSA签名,不同的只是签名方式的区别,其他的都一样,因此我这里主要说一下如何用RSA进行签名和验签。 

    

首先你需要准备下面的东西:

    php的openssl扩展里已经封装好了验签的方法openssl_verify。

    如果在Windows下的php.ini需要开启Openssl模块: extension=php_openssl.dll

 

    商户私钥:

    即RSA私钥,按照手册,按以下方式生成:

 

    openssl genrsa -out rsa_private_key.pem 1024 

 

    商户公钥:

    即RSA私钥,按照手册,按以下方式生成:

    openssl rsa -in rsa_private_key.pem -pubout -out rsa_public_key.pem

 

    生成之后,按照手册的说明,需要在签约平台上传公钥,需要注意的是,上传的时候需要把所有的注释和换行都去掉。 

 

    另外手册中还有如下命令:

    openssl pkcs8 -topk8 -inform PEM -in rsa_private_key.pem -outform PEM -nocrypt

 

    该命令将RSA私钥转换成PKCS8格式,对于PHP来说,不需要。 

 

    支付宝公钥:

    根据手册,在签约平台获得。

    如果你直接复制下来的话,会得到一个字符串,需要进行下面的转换;

    1)把空格变成换行

    2)添加注释

    比如你复制下来的公钥是:MIGfMA0GCSqGSIb3DQEBAQUAA4GNADCBiQKBgQDRBMjkaBznjXk06ddsL751KyYt

 

ztPFg0D3tu7jLqCacgqL+lbshIaItDGEXAMZmKa3DV6Wxy+l48YMo0RyS+dWze4M

UmuxHU/v6tiT0ZTXJN3EwrjCtCyyttdv/ROB3CkheXnTKB76reTkQqg57OWW+m9j

TCoccYMDXEIWYTs3CwIDAQAB,那转换之后为:

    -----BEGIN PUBLIC KEY-----

 

MIGfMA0GCSqGSIb3DQEBAQUAA4GNADCBiQKBgQDRBMjkaBznjXk06ddsL751KyYt

ztPFg0D3tu7jLqCacgqL+lbshIaItDGEXAMZmKa3DV6Wxy+l48YMo0RyS+dWze4M

UmuxHU/v6tiT0ZTXJN3EwrjCtCyyttdv/ROB3CkheXnTKB76reTkQqg57OWW+m9j

TCoccYMDXEIWYTs3CwIDAQAB

-----END PUBLIC KEY-----

    把公钥保存在文件里。 

 

注意这个是2048位的公钥应该是9行或者10行,不能为1行,不然PHP的openssl_pkey_get_public无法读取,pub_key_id的结果为false,如果没有-----BEGIN PUBLIC KEY----- 和 -----END PUBLIC KEY----- 可以自己加上,最后保存到一个rsa_public_key.pem文件中。

 

好了,现在已经有了所有的东西,先看签名函数:  

复制代码

 1

 2 /**

 3  * 签名字符串

 4  * @param $prestr 需要签名的字符串

 5  * return 签名结果

 6  */

 7 function rsaSign($prestr) {

 8     $public_key= file_get_contents('rsa_private_key.pem');

 9     $pkeyid = openssl_get_privatekey($public_key);

10     openssl_sign($prestr, $sign, $pkeyid);

11     openssl_free_key($pkeyid);

12     $sign = base64_encode($sign);

13     return $sign;

14 }

15 ?>

复制代码

注意点:

1.$prestr的内容和MD5一样(参见手册,但不包含最后的MD5密码)

2.签名用商户私钥

3.最后的签名,需要用base64编码

4.这个函数返回的值,就是这次请求的RSA签名。

 

验签函数:

 

复制代码

 1

 2 /**

 3  * 验证签名

 4  * @param $prestr 需要签名的字符串

 5  * @param $sign 签名结果

 6  * return 签名结果

 7  */

 8 function rsaVerify($prestr, $sign) {

 9     $sign = base64_decode($sign);

10     $public_key= file_get_contents('rsa_public_key.pem');

11     $pkeyid = openssl_get_publickey($public_key);

12     if ($pkeyid) {

13         $verify = openssl_verify($prestr, $sign, $pkeyid);

14         openssl_free_key($pkeyid);

15     }

16     if($verify == 1){

17         return true;

18     }else{

19         return false;

20     }

21 }

22 ?>

复制代码

注意点:

1.$prestr的内容和MD5一样(参见手册)

2.$sign是支付宝接口返回的sign参数用base64_decode解码之后的二进制

3.验签用支付宝公钥

4.这个函数返回一个布尔值,直接告诉你,验签是否通过  

 

支付宝官方提供的PHP版SDK demo中只对MD5加密方式进行了处理,但android 端和ios端 请求支付宝加密方式只能用RSA加密算法,这时服务端PHP就无法验证签名了,所以需要对demo进行一些修改。

 

1、修改alipay_notify.class.php文件 

verifyNotify 函数第46行 

$isSign = $this->getSignVeryfy($_POST, $_POST["sign"]); 

改成

$isSign = $this->getSignVeryfy($_POST, $_POST["sign"], $_POST["sign_type"]); 

 

verifyReturn 函数第83行

$isSign = $this->getSignVeryfy($_GET, $_GET["sign"]); 

改成 

$isSign = $this->getSignVeryfy($_GET, $_GET["sign"], $_GET["sign_type"]);  

 

getSignVeryfy 函数 116行

function getSignVeryfy($para_temp, $sign) {

改成

function getSignVeryfy($para_temp, $sign, $sign_type) { 

 

getSignVeryfy 函数 127行

switch (strtoupper(trim($this->alipay_config['sign_type']))) {

    case "MD5" :

        $isSgin = md5Verify($prestr, $sign, $this->alipay_config['key']);

break;

    default :

        $isSgin = false;

改成

switch (strtoupper(trim($sign_type))) {

    case "MD5" :

        $isSgin = md5Verify($prestr, $sign, $this->alipay_config['key']);

break;

    case "RSA" :

        $isSgin = rsaVerify($prestr, $sign);

        break; 

    default :

        $isSgin = false;

}  

2、新建一个alipay_rsa.function.php文件

复制代码

 1

 2 /* *

 3  * RSA

 4  * 详细:RSA加密

 5  * 版本:3.3

 6  * 日期:2014-02-20

 7  * 说明:

 8  * 以下代码只是为了方便商户测试而提供的样例代码,商户可以根据自己网站的需要,按照技术文档编写,并非一定要使用该代码。

 9  * 该代码仅供学习和研究支付宝接口使用,只是提供一个参考。

10  */

11 /**

12  * 签名字符串

13  * @param $prestr 需要签名的字符串

14  * return 签名结果

15  */

16 function rsaSign($prestr) {

17     $public_key= file_get_contents('rsa_private_key.pem');

18     $pkeyid = openssl_get_privatekey($public_key);

19     openssl_sign($prestr, $sign, $pkeyid);

20     openssl_free_key($pkeyid);

21     $sign = base64_encode($sign);

22     return $sign;

23 }

24 /**

25  * 验证签名

26  * @param $prestr 需要签名的字符串

27  * @param $sign 签名结果

28  * return 签名结果

29  */

30 function rsaVerify($prestr, $sign) {

31     $sign = base64_decode($sign);

32     $public_key= file_get_contents('rsa_public_key.pem');

33     $pkeyid = openssl_get_publickey($public_key);

34     if ($pkeyid) {

35         $verify = openssl_verify($prestr, $sign, $pkeyid);

36         openssl_free_key($pkeyid);

37     }

38     if($verify == 1){

39         return true;

40     }else{

41         return false;

42     }

43 }

44 ?>

复制代码

最后要说的是官方提供的手册上说的基本上都是正确的,只是有些地方没有说的很详细,开发的时候一定要多参考,大致就是这样,祝大家好运。

Statement
The content of this article is voluntarily contributed by netizens, and the copyright belongs to the original author. This site does not assume corresponding legal responsibility. If you find any content suspected of plagiarism or infringement, please contact admin@php.cn
php怎么把负数转为正整数php怎么把负数转为正整数Apr 19, 2022 pm 08:59 PM

php把负数转为正整数的方法:1、使用abs()函数将负数转为正数,使用intval()函数对正数取整,转为正整数,语法“intval(abs($number))”;2、利用“~”位运算符将负数取反加一,语法“~$number + 1”。

php怎么实现几秒后执行一个函数php怎么实现几秒后执行一个函数Apr 24, 2022 pm 01:12 PM

实现方法:1、使用“sleep(延迟秒数)”语句,可延迟执行函数若干秒;2、使用“time_nanosleep(延迟秒数,延迟纳秒数)”语句,可延迟执行函数若干秒和纳秒;3、使用“time_sleep_until(time()+7)”语句。

php怎么除以100保留两位小数php怎么除以100保留两位小数Apr 22, 2022 pm 06:23 PM

php除以100保留两位小数的方法:1、利用“/”运算符进行除法运算,语法“数值 / 100”;2、使用“number_format(除法结果, 2)”或“sprintf("%.2f",除法结果)”语句进行四舍五入的处理值,并保留两位小数。

php字符串有没有下标php字符串有没有下标Apr 24, 2022 am 11:49 AM

php字符串有下标。在PHP中,下标不仅可以应用于数组和对象,还可应用于字符串,利用字符串的下标和中括号“[]”可以访问指定索引位置的字符,并对该字符进行读写,语法“字符串名[下标值]”;字符串的下标值(索引值)只能是整数类型,起始值为0。

php怎么根据年月日判断是一年的第几天php怎么根据年月日判断是一年的第几天Apr 22, 2022 pm 05:02 PM

判断方法:1、使用“strtotime("年-月-日")”语句将给定的年月日转换为时间戳格式;2、用“date("z",时间戳)+1”语句计算指定时间戳是一年的第几天。date()返回的天数是从0开始计算的,因此真实天数需要在此基础上加1。

php怎么读取字符串后几个字符php怎么读取字符串后几个字符Apr 22, 2022 pm 08:31 PM

在php中,可以使用substr()函数来读取字符串后几个字符,只需要将该函数的第二个参数设置为负值,第三个参数省略即可;语法为“substr(字符串,-n)”,表示读取从字符串结尾处向前数第n个字符开始,直到字符串结尾的全部字符。

php怎么替换nbsp空格符php怎么替换nbsp空格符Apr 24, 2022 pm 02:55 PM

方法:1、用“str_replace(" ","其他字符",$str)”语句,可将nbsp符替换为其他字符;2、用“preg_replace("/(\s|\&nbsp\;||\xc2\xa0)/","其他字符",$str)”语句。

php怎么判断有没有小数点php怎么判断有没有小数点Apr 20, 2022 pm 08:12 PM

php判断有没有小数点的方法:1、使用“strpos(数字字符串,'.')”语法,如果返回小数点在字符串中第一次出现的位置,则有小数点;2、使用“strrpos(数字字符串,'.')”语句,如果返回小数点在字符串中最后一次出现的位置,则有。

See all articles

Hot AI Tools

Undresser.AI Undress

Undresser.AI Undress

AI-powered app for creating realistic nude photos

AI Clothes Remover

AI Clothes Remover

Online AI tool for removing clothes from photos.

Undress AI Tool

Undress AI Tool

Undress images for free

Clothoff.io

Clothoff.io

AI clothes remover

AI Hentai Generator

AI Hentai Generator

Generate AI Hentai for free.

Hot Tools

SublimeText3 Mac version

SublimeText3 Mac version

God-level code editing software (SublimeText3)

Dreamweaver Mac version

Dreamweaver Mac version

Visual web development tools

SublimeText3 Chinese version

SublimeText3 Chinese version

Chinese version, very easy to use

VSCode Windows 64-bit Download

VSCode Windows 64-bit Download

A free and powerful IDE editor launched by Microsoft

SublimeText3 Linux new version

SublimeText3 Linux new version

SublimeText3 Linux latest version