php签名认证

一、概述
        开年第一篇,该篇主要讲述了接口开发中,如何安全认证、如何用php签名认证。

二、说说历史
        签名认证是什么?为什么要做签名认证?签名认证哪里会用到?no、no、no…是不是,是不是,一下子疑问就这么多了!没事儿,通过追溯历史,我们来明白这些。

1、签名认证是什么?
        数字签名是一种类似写在纸上的普通的物理签名,但是使用了公钥加密领域的技术实现,用于鉴别数字信息的方法。一套数字签名通常定义两种互补的运算,一个用于签名,另一个用于验证。
数字签名,就是只有信息的发送者才能产生的别人无法伪造的一段数字串,这段数字串同时也是对信息的发送者发送信息真实性的一个有效证明。
        在这个以“数据为生命”的时代,每一个开发商都尽可能的收集客户的数据建立自己的BI库,各系统、各平台间数据的传输和调用变得非常普遍且非常重要;那么作为开发人员,我们不但要防止系统被攻击被入侵,我们还要确保数据的安全和完整
     2、为什么要做签名认证?
        在使用http或者soap传输数据的时候,签名作为其中一个参数,可以起到关键作用:1、鉴权(通过客户的密钥,服务端的密钥匹配);2、数据防篡改(参数是明文传输,将参数及密钥加密作为签名与服务器匹配);下面来分析下具体的方式:

将请求参数中的各个键值对按照key的字符串顺序升序排列(大小写敏感),把key和value拼成一串之后最后加上密钥,组成key1value1key2value2PRIVATEKEY的格式,转成utf-8编码的字节序列后计算md5,作为请求的签名。计算出来的签名串应当全为小写形式。如果某个参数的值为空,则此参数不参与签名。

3、签名认证哪里会用到?
        最常见的使我们在开发接口的时候,为了不被非法访问,往往我们会做签名认证,比如支付接口…

然后就是第三方平台的开发,比如微信公众平台…

三、流程分析
      客户端:首先我们为提供给用户一份接口文档,文档里面我们给用户提供api地址、签名算法解析过程、数据说明。

服务端:进行客户端检验,通常就是参数个数、格式验证,app_key验证(这个是博主这里使用的,这个app_key会交给用户,用户访问接口的时候必须带上该app_key),签名结果验证。

以我这次写的为例子来描述一下整个流程:

用户参照我们提供的接口文档,请求我们的接口,用户按照我们接口说明的签名算法生成签名串作为参数,然后带上必要的参数(GET、POST),如果服务端验证成功,则返回真实数据。

在服务端:我们首要验证提交参数的个数、格式是否正确,然后我们通过提交的参数用签名算法生成一个签名串,最后服务端生成的签名串和客户端提交过来的签名串进行比较,成功返回真实数据。

四、上代码

<?php 
namespace ceshi;

/**
 * 签名认证算法
 * HMAC-SHA256加密方式
 * 登录认证加入access_token:access-token通过登录接口去获取,通过刷新接口去刷新,需要注意返回的过期时间,要在过期时间之前刷新重新获取access-token。目前约定所有接口都必须传access-token,也就是用户必须先登录才可以看到相关内容
 * 随机函数可换更好的方法-本实例随机函数比较简单,随机性不够
 * ApiSign类作为服务端,类以外的代码作为客户端示例代码
 * author jiechengyang https://www.cnblogs.com/YangJieCheng/
 */
class ApiSign 
{
    CONST DELAY_TIME = 2000;
    CONST ACCESS_TOCEN_PATH = './access_token';

    private $_config = [];
    protected $AppKey = 'voBVVQxfMxDmhuxV70';
    protected $AppSecret = 'QJF5P8qWFJakF9Ve89ZcIstHKbkt5fVA';
    protected $timeout = 300;
    protected $algo = 'sha256';
    public $loginCheck = false;

    public function __construct($config, $loginCheck)
    {
        $this->_config = $config;
        empty($this->AppKey) && $this->AppKey = $this->generateRandomString();
        empty($this->AppSecret) && $this->AppSecret = $this->generateRangeNum();
        $this->loginCheck = $loginCheck;
        if ($this->init()) {
            echo '<span style="color:red" id="sign_success">恭喜,签名认证成功</span><script type="text/javascript"></script>';
            echo <<<JS
                <script>
                    var colors = ['#ffff00', '#ff66ff', '#99cc33', '#66ff33', '#000000', '#FF83FA', '#CAE1FF'];
                    var tmp = 0;
                    var timer = setInterval(colorChanage, 1000);
                    function colorChanage()
                    {
                        if(tmp == colors.length) {
                            tmp = 0;
                        }
                        document.getElementById('sign_success').style.color = colors[tmp++];
                    }
                </script>
JS;
        }
    }

    protected function init()
    {
        if (!isset($this->_config['_key']) 
            || !isset($this->_config['_sign'])
            || !isset($this->_config['_time'])
            || !isset($this->_config['_nonce'])
            || strlen($this->_config['_nonce']) !== 32
            ||  !is_numeric($this->_config['_time'])) {
                $this->callback(['message' => '请求参数不全, 或参数不规范'], 'error');
            }

        if (!isset($this->_config['_time']) || $this->getIsTimeOut()) {
            $this->callback(['message' => '请求超时'], 'error');
        }

        // 客户端验证
        $requestSignature = $this->_config['_sign'];
        $requestSignature = str_replace(' ', '+', $requestSignature);
        if($this->_config['_key'] !== $this->AppKey) {
            $this->callback(['message' => '非法的app_key'], 'error');
        }

        $signature = $this->generateSign($this->_config);
        if ($requestSignature != $signature) {
            $this->callback(['message' => '签名错误'], 'error');
        }

          $accessToken = $this->getAccessToken();

          // 用户登录token验证
          if ($this->loginCheck && $accessToken != $this->_config['access_token']) {
            $this->callback(['message' => 'access_token错误'], 'error');
          }

        return true;
    }

    private function callback($json=[], $status='ok')
    {
        $config = [
            'ok' => [200, '操作成功'],
            'error' => [300, '操作失败'],
            'timeout' => [300, '操作超时'],
        ];
        $json['statusCode'] = $config[$status][0];
        !isset($json['message']) && $json['message'] = $config[$status][1];
        echo json_encode($json);
        exit;
    }

    private function getIsTimeOut()
    {
        if (abs($this->_config['_time'] - time()) > $this->timeout) {
            return true;
        }

        return false;
    }

   /**
     * 生成随便数
     */
    public  function generateRangeNum($length = 32, $isToLower = false)
    {
        $str = $this->generateRandomString($length);
        if ($isToLower) {
            $str = strtolower($str);
        }

        return $str;
    }

    private function generateRandomString($length = 10) { 
        $characters = '0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ'; 
        $randomString = ''; 
        for ($i = 0; $i < $length; $i++) { 
            $randomString .= $characters[rand(0, strlen($characters) - 1)]; 
        } 

        return $randomString; 
    }

    protected function generateSign($params)
    {
        if (isset($params['_sign'])) {
            unset($params['_sign']);
        }

        if($this->loginCheck && isset($params['access_token'])) {
            unset($params['access_token']);
        }

        ksort($params);
        $str = '';
        foreach ($params as $key => $value) {
            $str .= $key . '=' . $value . '&';
        }

        $str = rtrim($str, '&');

        return hash_hmac($this->algo, $str, $this->AppSecret, false);
    }

    public  function getAccessToken()
    {

        if (!file_exists(self::ACCESS_TOCEN_PATH)) {
            $json = ['value' => $this->_config['access_token'], 'expires_in' => 7200, 'time' => time()];
            file_put_contents(self::ACCESS_TOCEN_PATH, json_encode($json));
            return $this->_config['access_token'];
        }

        $accessToken = json_decode(file_get_contents(self::ACCESS_TOCEN_PATH), true);
        if (time() - $accessToken['time'] > $accessToken['expires_in'] - self::DELAY_TIME) {
            $json = ['value' => $this->_config['access_token'], 'expires_in' => 7200, 'time' => time()];
            file_put_contents(self::ACCESS_TOCEN_PATH, json_encode($json));
            return $this->_config['access_token'];
        }    

        return $accessToken['value'];    

    }
}

/**
 * 生成随便数
 */
function generateRangeNum($length = 32, $isToLower = false)
{
    $str = generateRandomString($length);
    if ($isToLower) {
        $str = strtolower($str);
    }

    return $str;
}

function generateRandomString($length = 10) { 
    $characters = '0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ'; 
    $randomString = ''; 
    for ($i = 0; $i < $length; $i++) { 
        $randomString .= $characters[rand(0, strlen($characters) - 1)]; 
    } 

    return $randomString; 
}

function generateSign($algo, $params, $AppSecret)
{
    if (isset($params['_sign'])) {
        unset($params['_sign']);
    }

    ksort($params);
    $str = '';
    foreach ($params as $key => $value) {
        $str .= $key . '=' . $value . '&';
    }

    $str = rtrim($str, '&');

    return hash_hmac($algo, $str, $AppSecret, false);
}

header("content-type:text/html;charset:utf-8");
$algo = 'sha256';
$AppKey = 'voBVVQxfMxDmhuxV70';
$AppKey = 'Rd1bW719zRbCXOBx3L';//---用于公司项目测试
$AppSecret = 'QJF5P8qWFJakF9Ve89ZcIstHKbkt5fVA';
$AppSecret = '73ZBAnbwVPDu0dvdlYE0RMvzsbhehejd';//---用于公司项目测试
$nonce = generateRangeNum(32);
$nonce = 'kj4ESP2Qcngaj3eAjpnrhCQR9g4yKnTM';//---用于公司项目测试
$loginCheck = false;
$params = [
    '_key' => $AppKey,
    '_time' => time(),
    '_nonce' => $nonce,
];
$sign = generateSign($algo, $params, $AppSecret);
echo $params['_time'],'<hr/>';//---用于公司项目测试
echo $sign;exit;//---用于公司项目测试
$params['_sign'] = $sign;

$loginCheck && $params['access_token'] = generateRangeNum(16);
// echo '<pre>';print_r($params);
$apiSignModel = new ApiSign($params, $loginCheck);

猜你喜欢

转载自blog.csdn.net/weixin_43224306/article/details/87791847