首頁 >後端開發 >php教程 >用PHP實作SHA1withRSA簽章、加密、驗證

用PHP實作SHA1withRSA簽章、加密、驗證

little bottle
little bottle轉載
2019-04-18 11:19:179574瀏覽

網路上看到有PHP大牛分享的PHP的簽章加密代碼,轉來和大家一起分享一下,有興趣的小夥伴來了解一下吧。

概念解釋:

SHA1安全雜湊演算法主要適用於數位簽章標準裡面定義的數位簽章演算法。

RSA是目前最具影響力的公鑰加密演算法,它能夠抵抗到目前為止已知的絕大多數密碼攻擊,已被ISO推薦為公鑰資料加密標準。

SHA1WithRSA:用SHA演算法進行簽名,用RSA演算法加密。

業務要求:

每個簽章組裝的內容是依欄位名稱的字典順序升序排序連接的

先組裝需要簽章的內容:

/**
     * 拼接需要签名的内容
     * Author: Tao.
     *
     * @param array $data 需签名的字段内容
     * 
     * @return string
     */     
    public static function getSign($data)
    {
        foreach ($data as $k => $v) {
            $Parameters[$k] = $v;
        }
        //按字典序排序参数
        ksort($Parameters);
        $sign = '';
        foreach ($Parameters as $k => $v) {
            $sign .= $k . "=" . $v . "&";
        }
        $sign = '&' . rtrim($sign, '&');
        return $sign;
    }

簽章字串如下範例:
&amount=amount 值&ccy=ccy 值&merchantId=merchantId 值¬ifyUrl=notifyUrl 值&orderId=orderId 值&payeeAcctNo=payeeAcctNo 值(明文)。

要注意的是,根據業務需求選擇,是否在簽名內容前拼接 &符。

然後產生秘鑰簽章:

/**
     * 秘钥加密
     * Author: Tao.
     *
     * @param string $data 之前生成好的需加密内容
     * @param $key 私钥证书位置(.pfx文件)
     * @param string $pwd 证书密码
     *
     * @return string
     */
    public static function SHA1withRSA($data, $key,$pwd)
    {
        openssl_pkcs12_read(file_get_contents($key), $certs, $pwd); 
        if (!$certs) return;
        $signature = '';
        openssl_sign($data, $signature, $certs['pkey']);
        return bin2hex($signature);  
    }

由於第三方公司要求轉換使用16進制,可依需求選擇bin2hex()或base64_encode()。

這裡要注意的是,根據業務需要,簽名後的內容是否要求大小寫敏感。

簽名後的內容應該是小寫的,可以使用strtoupper()轉換成大寫。

以上就是給大家整理好的私鑰加密方法。

 

但此業務中另要求將銀行卡號需要進行RSA公鑰加密
以下是取得公鑰的方法:
這裡是取得對方平台憑證的公鑰(.cer檔案)


/**
     * 获取公钥
     * Author: Tao.
     *
     * @param $path //公钥证书位置 (.cer文件)
     *
     * @return mixed
     * @throws \Exception
     */
    public static function loadCert($path)
    {
        $file = file_get_contents($path);
        if (!$file) {
            throw new \Exception('loadx509Cert::file_get_contents ERROR');
        }

        $cert = chunk_split(base64_encode($file), 64, "\n");
        $cert = "-----BEGIN CERTIFICATE-----\n" . $cert . "-----END CERTIFICATE-----\n";

        $res = openssl_pkey_get_public($cert);
        $detail = openssl_pkey_get_details($res);
        openssl_free_key($res);

        if (!$detail) {
            throw new \Exception('loadX509Cert::openssl_pkey_get_details ERROR');
        }
        return $detail['key'];
    }

    /**
     * 公钥加密
     * Author: Tao.
     * 
     * @param $pubPath  //公钥证书位置 (.cer文件)
     * @param string $bankCode //银行卡号
     * 
     * @return string
     */
    public static function rsa_encode($bankCode,$pubPath)
    {
        $pubkey = self::loadCert($pubPath);
        $encrypt_data = '';
        openssl_public_encrypt($bankCode, $encrypt_data, $pubkey);
        $encrypt_data = base64_encode($encrypt_data);
        return $encrypt_data;
    }

你要問我為什麼私鑰是bin2hex(),公鑰換了base64_encode()。我也不知道為什麼,問過說是16位,但請求簽名一直失敗,換了64成功了。對方說文檔太老了,忘了。 。依需求選擇吧

最後回呼結果驗簽

先將回呼資料中組裝簽章欄位的內容取出來,並依照上面的getSign()方法排序。
接著進行驗證:

/**
     * 验证返回的签名是否正确
     *
     * @param string $data 要验证的签名原文
     * @param string $signature 签名内容
      *@param $pubPath  公钥证书位置 (.cer文件)
     *
     * @return bool
     */
    public static function verifyRespondSign($data, $signature,$pubPath)
    {
        $keys = self::loadCert($pubPath);
        $signature = hex2bin($signature);
        $ok = openssl_verify($data, $signature, $keys);
        if ($ok == 1) {
            return true;
        }
        return false;
    }

 希望對各位有幫助,有不對的地方請指出,及時修改,互相學習。相關教學:PHP影片教學

#

以上是用PHP實作SHA1withRSA簽章、加密、驗證的詳細內容。更多資訊請關注PHP中文網其他相關文章!

陳述:
本文轉載於:cnblogs.com。如有侵權,請聯絡admin@php.cn刪除