PHP开发接口使用RSA进行加密解密方法

网络安全问题很重要,尤其是保证数据安全,遇到很多在写接口的程序员直接都是明文数据传输,在我看来这是很不专业的。本人提倡经过接口的数据都要进行加密解密之后进行使用。

这篇文章主要介绍使用PHP开发接口,数据实现RSA加密解密后使用,实例分析了PHP自定义RSA类实现加密与解密的技巧,非常具有实用价值,需要的朋友可以参考下。

简单介绍RSA:

RSA加密算法是最常用的非对称加密算法,CFCA在证书服务中离不了它。但是有不少新手对它不太了解。下面仅作简要介绍。RSA是第一个比较完善的公开密钥算法,它既能用于加密,也能用于数字签名。RSA以它的三个发明者Ron Rivest, Adi Shamir, Leonard Adleman的名字首字母命名,这个算法经受住了多年深入的密码分析,虽然密码分析者既不能证明也不能否定RSA的安全性,但这恰恰说明该算法有一定的可信性,目前它已经成为最流行的公开密钥算法。RSA的安全基于大数分解的难度。其公钥和私钥是一对大素数(100到200位十进制数或更大)的函数。从一个公钥和密文恢复出明文的难度,等价于分解两个大素数之积(这是公认的数学难题)。

下面为具体类、实例:

    <?php

    /**
     * RSA算法类
     * 签名及密文编码:base64字符串/十六进制字符串/二进制字符串流
     * 填充方式: PKCS1Padding(加解密)/NOPadding(解密)
     *
     * Notice:Only accepts a single block. Block size is equal to the RSA key size!
     * 如密钥长度为1024 bit,则加密时数据需小于128字节,加上PKCS1Padding本身的11字节信息,所以明文需小于117字节
     *
     * @author: ZHIHUA_WEI
     * @version: 1.0.0
     * @date: 2017/06/30
     */
    class RSA
    {
        private $pubKey = null;
        private $priKey = null;

        /**
         * 构造函数
         *
         * @param string 公钥文件(验签和加密时传入)
         * @param string 私钥文件(签名和解密时传入)
         */
        public function __construct($public_key_file = ‘‘, $private_key_file = ‘‘)
        {
            if ($public_key_file) {
                $this->_getPublicKey($public_key_file);
            }
            if ($private_key_file) {
                $this->_getPrivateKey($private_key_file);
            }
        }

        // 私有方法
        /**
         * 自定义错误处理
         */
        private function _error($msg)
        {
            die(‘RSA Error:‘ . $msg); //TODO
        }

        /**
         * 检测填充类型
         * 加密只支持PKCS1_PADDING
         * 解密支持PKCS1_PADDING和NO_PADDING
         *
         * @param int 填充模式
         * @param string 加密en/解密de
         * @return bool
         */
        private function _checkPadding($padding, $type)
        {
            if ($type == ‘en‘) {
                switch ($padding) {
                    case OPENSSL_PKCS1_PADDING:
                        $ret = true;
                        break;
                    default:
                        $ret = false;
                }
            } else {
                switch ($padding) {
                    case OPENSSL_PKCS1_PADDING:
                    case OPENSSL_NO_PADDING:
                        $ret = true;
                        break;
                    default:
                        $ret = false;
                }
            }
            return $ret;
        }

        private function _encode($data, $code)
        {
            switch (strtolower($code)) {
                case ‘base64‘:
                    $data = base64_encode(‘‘ . $data);
                    break;
                case ‘hex‘:
                    $data = bin2hex($data);
                    break;
                case ‘bin‘:
                default:
            }
            return $data;
        }

        private function _decode($data, $code)
        {
            switch (strtolower($code)) {
                case ‘base64‘:
                    $data = base64_decode($data);
                    break;
                case ‘hex‘:
                    $data = $this->_hex2bin($data);
                    break;
                case ‘bin‘:
                default:
            }
            return $data;
        }

        private function _getPublicKey($file)
        {
            $key_content = $this->_readFile($file);
            if ($key_content) {
                $this->pubKey = openssl_get_publickey($key_content);
            }
        }

        private function _getPrivateKey($file)
        {
            $key_content = $this->_readFile($file);
            if ($key_content) {
                $this->priKey = openssl_get_privatekey($key_content);
            }
        }

        private function _readFile($file)
        {
            $ret = false;
            if (!file_exists($file)) {
                $this->_error("The file {$file} is not exists");
            } else {
                $ret = file_get_contents($file);
            }
            return $ret;
        }

        private function _hex2bin($hex = false)
        {
            $ret = $hex !== false && preg_match(‘/^[0-9a-fA-F]+$/i‘, $hex) ? pack("H*", $hex) : false;
            return $ret;
        }

        /**
         * 生成签名
         *
         * @param string 签名材料
         * @param string 签名编码(base64/hex/bin)
         * @return 签名值
         */
        public function sign($data, $code = ‘base64‘)
        {
            $ret = false;
            if (openssl_sign($data, $ret, $this->priKey)) {
                $ret = $this->_encode($ret, $code);
            }
            return $ret;
        }

        /**
         * 验证签名
         *
         * @param string 签名材料
         * @param string 签名值
         * @param string 签名编码(base64/hex/bin)
         * @return bool
         */
        public function verify($data, $sign, $code = ‘base64‘)
        {
            $ret = false;
            $sign = $this->_decode($sign, $code);
            if ($sign !== false) {
                switch (openssl_verify($data, $sign, $this->pubKey)) {
                    case 1:
                        $ret = true;
                        break;
                    case 0:
                    case -1:
                    default:
                        $ret = false;
                }
            }
            return $ret;
        }

        /**
         * 加密
         *
         * @param string 明文
         * @param string 密文编码(base64/hex/bin)
         * @param int 填充方式(貌似php有bug,所以目前仅支持OPENSSL_PKCS1_PADDING)
         * @return string 密文
         */
        public function encrypt($data, $code = ‘base64‘, $padding = OPENSSL_PKCS1_PADDING)
        {
            $ret = false;
            if (!$this->_checkPadding($padding, ‘en‘)) $this->_error(‘padding error‘);
            if (openssl_public_encrypt($data, $result, $this->pubKey, $padding)) {
                $ret = $this->_encode($result, $code);
            }
            return $ret;
        }

        /**
         * 解密
         *
         * @param string 密文
         * @param string 密文编码(base64/hex/bin)
         * @param int 填充方式(OPENSSL_PKCS1_PADDING / OPENSSL_NO_PADDING)
         * @param bool 是否翻转明文(When passing Microsoft CryptoAPI-generated RSA cyphertext, revert the bytes in the block)
         * @return string 明文
         */
        public function decrypt($data, $code = ‘base64‘, $padding = OPENSSL_PKCS1_PADDING, $rev = false)
        {
            $ret = false;
            $data = $this->_decode($data, $code);
            if (!$this->_checkPadding($padding, ‘de‘)) $this->_error(‘padding error‘);
            if ($data !== false) {
                if (openssl_private_decrypt($data, $result, $this->priKey, $padding)) {
                    $ret = $rev ? rtrim(strrev($result), "\0") : ‘‘ . $result;
                }
            }
            return $ret;
        }
    }
    

此为具体的RSA类。

<?php
/**
 * Author: Wei ZhiHua
 * Date: 2017/6/30 0030
 * Time: 上午 10:15
 */
header(‘Content-Type:text/html;Charset=utf-8;‘);
include "RSA.php";
echo ‘<pre>‘;  

$pubfile = ‘D:\WWW\test\rsa_public_key.pem‘;
$prifile = ‘D:\WWW\test\rsa_private_key.pem‘;
$rsa = new RSA($pubfile, $prifile);
$rst = array(
    ‘ret‘ => 200,
    ‘code‘ => 1,
    ‘data‘ => array(1, 2, 3, 4, 5, 6),
    ‘msg‘ => "success",
);
$ex = json_encode($rst);
//加密
$ret_e = $rsa->encrypt($ex);
//解密
$ret_d = $rsa->decrypt($ret_e);
echo $ret_e;
echo ‘<pre>‘;
echo $ret_d;  

echo ‘<pre>‘;  

$a = ‘test‘;
//签名
$x = $rsa->sign($a);
//验证
$y = $rsa->verify($a, $x);
var_dump($x, $y);
exit;  

使用方法。

时间: 2024-10-06 19:07:37

PHP开发接口使用RSA进行加密解密方法的相关文章

基于RSA的加密/解密示例C#代码

在C#程序中,大家可能比较熟悉的方式是md5加密解密方式,对RSA可能并不是很熟悉, 下面就说一下RSA加密和解密的算法: using System;using System.Security.Cryptography;using System.Text;class RSACSPSample{    static void Main()    {        try        {            string str_Plain_Text = "How are you?How are

RSA算法加密解密,数据传输,前台与后台数据交互

1.RSA算法加密解密思路. java后台随机生成公钥.私钥.存储于session中,告知前台js获取公钥.通过jsencrypt.min.js进行加密.传输回后台,后台通过私钥解密. 2.RSA常见异常分解. 问题一 Cannot find any provider supporting RSA 出现此问题,属于JDK版本bug问题,笔者1.8.0_171遇到此问题,更换1.8.0_211之后即可解决. 问题二 DER input, Integer tag error 出现此问题,主要是公钥,

ios常见加密解密方法

在其他平台中经常会计算MD5值,在iOS平台中也提供了该方法,首先需要导入头文件 [cpp] view plain copy #import <CommonCrypto/CommonDigest.h> 方法CC_MD5可以获取MD5的16个字符的数组,再通过%02X的形式输出即可获取32位MD5值. [cpp] view plain copy @implementation NSString (CCCryptUtil) -(NSString*) md5 { const char * cStrV

C#/IOS/Android通用加密解密方法

原文:C#/IOS/Android通用加密解密方法 公司在做移动端ios/android,服务器提供接口使用的.net,用到加密解密这一块,也在网上找了一些方法,有些是.net加密了android解密不了,或者反之.下面的是三个平台都可以加密解密的方法.加密解密中用到的key="1234578";在调取方法时传值即可. C#代码 #region 跨平台加解密(c#) /// <summary> /// 对字符串进行DES加密 /// </summary> ///

mcrypt本身就提供了强大的加密解密方法

由于项目的需要,要写一个能生成“授权码”的类(授权码主要包含项目使用的到期时间),生成的授权码将会写入到一个文件当中,每当项目运行的时候,会自动读取出文件中的密文,然后使用唯一的“密钥”来调用某个函数,对密文进行解密,从中解读出大都会娱乐城项目的使用到期时间. 之前,自己有先试着写了下,主要是base64+md5+反转字符串.算法太过简单,很容易被破解,而且也没有能过做到“密钥”在加解密中的重要性,故而舍之. 后来,查找了相关资料,发现,原来PHP中内置了一个功能强大的函数库,即Mcrypt.

(转)C#实现RSA非对称加密解密

转自:http://blog.csdn.net/u010678947/article/details/48652875 一.RSA简介 RSA公钥加密算法是1977年由Ron Rivest.Adi Shamirh和LenAdleman在(美国麻省理工学院)开发的.RSA取名来自开发他们三者的名字.RSA是目前最有影响力的公钥加密算法,它能够抵抗到目前为止已知的所有密码攻击,已被ISO推荐为公钥数据加密标准.RSA算法基于一个十分简单的数论事实:将两个大素数相乘十分容易,但那时想要对其乘积进行因式

C#开发中常用加密解密方法解析

一.MD5加密算法 我想这是大家都常听过的算法,可能也用的比较多.那么什么是MD5算法呢?MD5全称是message-digest algorithm 5,简单的说就是单向的加密,即是说无法根据密文推导出明文. MD5主要用途: 1.对一段信息生成信息摘要,该摘要对该信息具有唯一性,可以作为数字签名. 2.用于验证文件的有效性(是否有丢失或损坏的数据), 3.对用户密码的加密, 4.在哈希函数中计算散列值 从上边的主要用途中我们看到,由于算法的某些不可逆特征,在加密应用上有较好的安全性.通过使用

java&amp;Ios rsa 进行加密解密

最近公司要准备做一个传输数据加密准备用RSA 首先说一下自己开发中越到到坑 开始后台做给了我字符串格式的公钥.私钥做调试 后台自己加密.解密.加签.验签,都没有问题, 我这边同样也是加密.解密.加签.验签都没有问题 加密每次都是不同的,但是加签只要铭文固定,结果是一样 问题出在,同样的铭文,前端和后台加签出来不结果 最后和后台讨论用我们前端生成的文件进行加密解密 一.首先说一下生成公钥.私钥.证书的过程 openssl genrsa -out private_key.pem 1024 opens

RSA算法加密解密

该工具类中用到了BASE64,需要借助第三方类库:javabase64-1.3.1. jar 注意:RSA加密明文最大长度117字节,解密要求密文最大长度为128字节,所以在加密和解密的过程中需要分块进行. RSA加密对明文的长度是有限制的,如果加密数据过大会抛出如下异常: Exception in thread "main" javax.crypto.IllegalBlockSizeException: Data must not be longer than 117 bytes a