首页 > 开发 > PHP > 正文

详解PHP实现支付宝小程序用户授权的工具类

2024-05-04 22:41:41
字体:
来源:转载
供稿:网友

背景

最近项目需要上线支付宝小程序,同时需要走用户的授权流程完成用户信息的存储,以前做过微信小程序的开发,本以为实现授权的过程是很简单的事情,但是再实现的过程中还是遇到了不少的坑,因此记录一下实现的过程

学到的知识

支付宝开放接口的调用模式以及实现方式 支付宝小程序授权的流程 RSA加密方式

吐槽点

支付宝小程序的入口隐藏的很深,没有微信小程序那么直接了当
支付宝小程序的开发者工具比较难用,编译时候比较卡,性能有很大的问题
每提交一次代码,支付宝小程序的体验码都要进行更换,比较繁琐,而且localStorage的东西不知道要如何删除

事先准备

到支付宝开放平台注册一个开发者账号,并做好相应的认证等工作 创建一个小程序,并记录好相关的小程序信息,包括支付宝公钥,私钥,app公钥等,可以借鉴支付宝官方提供的相应的公钥生成工具来生成公钥和私钥,工具的下载地址:传送门 了解下支付宝小程序的签名机制,详细见https://docs.open.alipay.com/291/105974 熟悉下支付宝小程序获取用户信息的过程,详细见支付宝小程序用户授权指引

授权的步骤

授权时序图

实现流程

    客户端通过my.getAuthCode接口获取code,传给服务端 服务端通过code,调用获取token接口获取access_token,alipay.system.oauth.token(换取授权访问令牌) 通过token接口调用支付宝会员查询接口获取会员信息,alipay.user.info.share(支付宝会员授权信息查询接口) 将获取的用户信息保存到数据库

AmpHelper工具类

<?php/** * Created by PhpStorm. * User: My * Date: 2018/8/16 * Time: 17:45 */namespace App/Http/Helper;use App/Http/Helper/Sys/BusinessHelper;use Illuminate/Support/Facades/Log;class AmpHelper{  const API_DOMAIN = "https://openapi.alipay.com/gateway.do?";  const API_METHOD_GENERATE_QR = 'alipay.open.app.qrcode.create';  const API_METHOD_AUTH_TOKEN = 'alipay.system.oauth.token';  const API_METHOD_GET_USER_INFO = 'alipay.user.info.share';  const SIGN_TYPE_RSA2 = 'RSA2';  const VERSION = '1.0';  const FILE_CHARSET_UTF8 = "UTF-8";  const FILE_CHARSET_GBK = "GBK";  const RESPONSE_OUTER_NODE_QR = 'alipay_open_app_qrcode_create_response';  const RESPONSE_OUTER_NODE_AUTH_TOKEN = 'alipay_system_oauth_token_response';  const RESPONSE_OUTER_NODE_USER_INFO = 'alipay_user_info_share_response';  const RESPONSE_OUTER_NODE_ERROR_RESPONSE = 'error_response';  const STATUS_CODE_SUCCESS = 10000;  const STATUS_CODE_EXCEPT = 20000;  /**   * 获取用户信息接口,根据token   * @param $code 授权码   * 通过授权码获取用户的信息   */  public static function getAmpUserInfoByAuthCode($code){    $aliUserInfo = [];    $tokenData = AmpHelper::getAmpToken($code);    //如果token不存在,这种主要是为了处理支付宝的异常记录    if(isset($tokenData['code'])){      return $tokenData;    }    $token = formatArrValue($tokenData,'access_token');    if($token){      $userBusiParam = self::getAmpUserBaseParam($token);      $url = self::buildRequestUrl($userBusiParam);      $resonse = self::getResponse($url,self::RESPONSE_OUTER_NODE_USER_INFO);      if($resonse['code'] == self::STATUS_CODE_SUCCESS){        //有效的字段列        $userInfoColumn = ['user_id','avatar','province','city','nick_name','is_student_certified','user_type','user_status','is_certified','gender'];        foreach ($userInfoColumn as $column){          $aliUserInfo[$column] = formatArrValue($resonse,$column,'');        }      }else{        $exceptColumns = ['code','msg','sub_code','sub_msg'];        foreach ($exceptColumns as $column){          $aliUserInfo[$column] = formatArrValue($resonse,$column,'');        }      }    }    return $aliUserInfo;  }  /**   * 获取小程序token接口   */  public static function getAmpToken($code){    $param = self::getAuthBaseParam($code);    $url = self::buildRequestUrl($param);    $response = self::getResponse($url,self::RESPONSE_OUTER_NODE_AUTH_TOKEN);    $tokenResult = [];    if(isset($response['code']) && $response['code'] != self::STATUS_CODE_SUCCESS){      $exceptColumns = ['code','msg','sub_code','sub_msg'];      foreach ($exceptColumns as $column){        $tokenResult[$column] = formatArrValue($response,$column,'');      }    }else{      $tokenResult = $response;    }    return $tokenResult;  }  /**   * 获取二维码链接接口   * 433ac5ea4c044378826afe1532bcVX78   * https://openapi.alipay.com/gateway.do?timestamp=2013-01-01 08:08:08&method=alipay.open.app.qrcode.create&app_id=2893&sign_type=RSA2&sign=ERITJKEIJKJHKKKKKKKHJEREEEEEEEEEEE&version=1.0&biz_content=  {"url_param":"/index.html?name=ali&loc=hz", "query_param":"name=1&age=2", "describe":"二维码描述"}  */  public static function generateQrCode($mpPage = 'pages/index',$queryParam = [],$describe){    $param = self::getQrcodeBaseParam($mpPage,$queryParam,$describe );    $url = self::buildRequestUrl($param);    $response = self::getResponse($url,self::RESPONSE_OUTER_NODE_QR);    return $response;  }  /**   * 获取返回的数据,对返回的结果做进一步的封装和解析,因为支付宝的每个接口的返回都是由一个特定的     * key组成的,因此这里直接封装了而一个通用的方法,对于不同的接口只需要更改相应的node节点就可以了   */  public static function getResponse($url,$responseNode){    $json = curlRequest($url);    $response = json_decode($json,true);    $responseContent = formatArrValue($response,$responseNode,[]);    $errResponse = formatArrValue($response,self::RESPONSE_OUTER_NODE_ERROR_RESPONSE,[]);    if($errResponse){      return $errResponse;    }    return $responseContent;  }  /**   * 获取请求的链接   */  public static function buildQrRequestUrl($mpPage = 'pages/index',$queryParam = []){    $paramStr = http_build_query(self::getQrBaseParam($mpPage,$queryParam));    return self::API_DOMAIN . $paramStr;  }  /**   * 构建请求链接   */  public static function buildRequestUrl($param){    $paramStr = http_build_query($param);    return self::API_DOMAIN . $paramStr;  }  /**   * 获取用户的基础信息接口   */  public static function getAmpUserBaseParam($token){    $busiParam = [      'auth_token' => $token,    ];    $param = self::buildApiBuisinessParam($busiParam,self::API_METHOD_GET_USER_INFO);    return $param;  }  /**   *获取二维码的基础参数   */  public static function getQrcodeBaseParam($page= 'pages/index/index',$queryParam = [],$describe = ''){    $busiParam = [      'biz_content' => self::getQrBizContent($page,$queryParam,$describe)    ];    $param = self::buildApiBuisinessParam($busiParam,self::API_METHOD_GENERATE_QR);    return $param;  }  /**   *获取授权的基础参数   */  public static function getAuthBaseParam($code,$refreshToken = ''){    $busiParam = [      'grant_type' => 'authorization_code',      'code' => $code,      'refresh_token' => $refreshToken,    ];    $param = self::buildApiBuisinessParam($busiParam,self::API_METHOD_AUTH_TOKEN);    return $param;  }  /**   * 构建业务参数   */  public static function buildApiBuisinessParam($businessParam,$apiMethod){    $pubParam = self::getApiPubParam($apiMethod);    $businessParam = array_merge($pubParam,$businessParam);    $signContent = self::getSignContent($businessParam);    error_log('sign_content ===========>'.$signContent);    $rsaHelper = new RsaHelper();    $sign = $rsaHelper->createSign($signContent);    error_log('sign ===========>'.$sign);    $businessParam['sign'] = $sign;    return $businessParam;  }  /**   * 公共参数   *   */  public static function getApiPubParam($apiMethod){    $ampBaseInfo = BusinessHelper::getAmpBaseInfo();    $param = [      'timestamp' => date('Y-m-d H:i:s') ,      'method' => $apiMethod,      'app_id' => formatArrValue($ampBaseInfo,'appid',config('param.amp.appid')),      'sign_type' =>self::SIGN_TYPE_RSA2,      'charset' =>self::FILE_CHARSET_UTF8,      'version' =>self::VERSION,    ];    return $param;  }  /**   * 获取签名的内容   */  public static function getSignContent($params) {    ksort($params);    $stringToBeSigned = "";    $i = 0;    foreach ($params as $k => $v) {      if (!empty($v) && "@" != substr($v, 0, 1)) {        if ($i == 0) {          $stringToBeSigned .= "$k" . "=" . "$v";        } else {          $stringToBeSigned .= "&" . "$k" . "=" . "$v";        }        $i++;      }    }    unset ($k, $v);    return $stringToBeSigned;  }  public static function convertArrToQueryParam($param){    $queryParam = [];    foreach ($param as $key => $val){      $obj = $key.'='.$val;      array_push($queryParam,$obj);    }    $queryStr = implode('&',$queryParam);    return $queryStr;  }  /**   * 转换字符集编码   * @param $data   * @param $targetCharset   * @return string   */  public static function characet($data, $targetCharset) {    if (!empty($data)) {      $fileType = self::FILE_CHARSET_UTF8;      if (strcasecmp($fileType, $targetCharset) != 0) {        $data = mb_convert_encoding($data, $targetCharset, $fileType);      }    }    return $data;  }  /**   * 获取业务参数内容   */  public static function getQrBizContent($page, $queryParam = [],$describe = ''){    if(is_array($queryParam)){      $queryParam = http_build_query($queryParam);    }    $obj = [      'url_param' => $page,      'query_param' => $queryParam,      'describe' => $describe    ];    $bizContent = json_encode($obj,JSON_UNESCAPED_UNICODE);    return $bizContent;  }}            
发表评论 共有条评论
用户名: 密码:
验证码: 匿名发表