$val) { $param .= "$key=" .urlencode($val). "&"; $sign .= "$key=$val&"; } $result = array( 'param' => substr($param, 0, -1), 'value' => substr($sign, 0, -1), ); if(!empty($config['key'])) { $result['sign'] = $result['value'].$config['key']; } return $result; } /** * [GetAlipayUserInfo 支付宝小程序获取用户信息] * @author Devil * @blog http://gong.gg/ * @version 1.0.0 * @datetime 2017-09-23T22:30:43+0800 * @param [string] $authcode [用户授权码] * @param [string] $app_id [应用appid] * @return [array|boolean] [成功返回用户数据, 则false] */ public function GetAlipayUserInfo($authcode, $app_id) { // 从缓存获取用户信息 $key = 'alipay_userinfo_'.$authcode; $result = GS($key); if($result !== false) { return ['status'=>0, 'msg'=>'success', 'data'=>$result]; } // 获取授权信息并且获取用户信息 $auth = $this->GetAuthAccessToken($authcode, $app_id); if($auth['status'] == 0) { // 请求参数 $param = [ 'app_id' => $app_id, 'method' => 'alipay.user.info.share', 'charset' => 'utf-8', 'format' => 'JSON', 'sign_type' => 'RSA2', 'timestamp' => date('Y-m-d H:i:s'), 'version' => '1.0', 'auth_token' => $auth['data']['access_token'], ]; // 生成签名参数+签名 $p = $this->GetParamSign($param); $param['sign'] = $this->MyRsaSign($p['value']); // 执行请求 $result = $this->HttpRequest('https://openapi.alipay.com/gateway.do', $param); if(!empty($result['alipay_user_info_share_response']['code']) && $result['alipay_user_info_share_response']['code'] == 10000) { // 验证签名正确则存储缓存返回数据 if(!$this->SyncRsaVerify($result, 'alipay_user_info_share_response')) { return ['status'=>-1, 'msg'=>'签名验证失败']; } // 存储缓存 SS($key, $result['alipay_user_info_share_response']); // 返回用户数据 return ['status'=>0, 'msg'=>'success', 'data'=>$result['alipay_user_info_share_response']]; } $msg = empty($result['error_response']['sub_msg']) ? '授权失败' : $result['error_response']['sub_msg']; return ['status'=>-1, 'msg'=>$msg]; } else { return $auth; } } /** * [GetAuthAccessToken 根据用户授权换取授权访问令牌] * @author Devil * @blog http://gong.gg/ * @version 1.0.0 * @datetime 2017-09-23T22:36:26+0800 * @param [string] $authcode [用户授权码] * @param [string] $app_id [应用appid] * @return [array|boolean] [失败false, 用户授权信息] */ public function GetAuthAccessToken($authcode, $app_id) { // 获取用户授权信息 $key = 'alipay_authcode_'.$authcode; $result = GS($key, 0, true); // 过期判断 if($result == false || $result['filemtime']+$result['re_expires_in'] < time()) { $result = $this->GetAuthCode($app_id, $key, $authcode); } else { if($result['filemtime']+$result['expires_in'] < time()) { $result = $this->GetAuthCode($app_id, $key, '', $result['refresh_token']); } } return $result; } /** * [GetAuthCode 获取用户授权信息] * @author Devil * @blog http://gong.gg/ * @version 1.0.0 * @datetime 2017-09-24T21:55:45+0800 * @param [string] $app_id [应用appid] * @param [string] $key [缓存key] * @param [string] $authcode [用户授权码] * @param [string] $refresh_token [刷新授权token] * @return [array|boolean] [失败false, 用户授权信息] */ private function GetAuthCode($app_id, $key, $authcode = '', $refresh_token = '') { if(empty($app_id) || empty($key) || (empty($authcode) && empty($refresh_token))) { return ['status'=>-1, 'msg'=>'参数有误']; } // 请求参数 $param = [ 'app_id' => $app_id, 'method' => 'alipay.system.oauth.token', 'charset' => 'utf-8', 'format' => 'JSON', 'sign_type' => 'RSA2', 'timestamp' => date('Y-m-d H:i:s'), 'version' => '1.0', 'biz_content' => 'mini-authcode', ]; if(!empty($authcode)) { $param['code'] = $authcode; $param['grant_type'] = 'authorization_code'; } if(!empty($refresh_token)) { $param['refresh_token'] = $refresh_token; $param['grant_type'] = 'refresh_token'; } // 生成签名参数+签名 $p = $this->GetParamSign($param); $param['sign'] = $this->MyRsaSign($p['value']); // 执行请求 $result = $this->HttpRequest('https://openapi.alipay.com/gateway.do', $param); // 结果正确则验证签名 并且 存储缓存返回access_token if(!empty($result['alipay_system_oauth_token_response']['user_id'])) { // 验证签名正确则存储缓存返回数据 if(!$this->SyncRsaVerify($result, 'alipay_system_oauth_token_response')) { return ['status'=>-1, 'msg'=>'签名验证失败']; } // 存储缓存 SS($key, $result['alipay_system_oauth_token_response']); return ['status'=>0, 'msg'=>'success', 'data'=>$result['alipay_system_oauth_token_response']]; } $msg = empty($result['error_response']['sub_msg']) ? '授权失败' : $result['error_response']['sub_msg']; return ['status'=>-1, 'msg'=>$msg]; } /** * [MyRsaSign 签名字符串] * @author Devil * @blog http://gong.gg/ * @version 1.0.0 * @datetime 2017-09-24T08:38:28+0800 * @param [string] $prestr [需要签名的字符串] * @return [string] [签名结果] */ private function MyRsaSign($prestr) { $res = "-----BEGIN RSA PRIVATE KEY-----\n"; $res .= wordwrap(MyC('common_app_mini_alipay_rsa_private'), 64, "\n", true); $res .= "\n-----END RSA PRIVATE KEY-----"; return openssl_sign($prestr, $sign, $res, OPENSSL_ALGO_SHA256) ? base64_encode($sign) : null; } /** * [MyRsaDecrypt RSA解密] * @author Devil * @blog http://gong.gg/ * @version 1.0.0 * @datetime 2017-09-24T09:12:06+0800 * @param [string] $content [需要解密的内容,密文] * @return [string] [解密后内容,明文] */ private function MyRsaDecrypt($content) { $res = "-----BEGIN PUBLIC KEY-----\n"; $res .= wordwrap(MyC('common_app_mini_alipay_rsa_public'), 64, "\n", true); $res .= "\n-----END PUBLIC KEY-----"; $res = openssl_get_privatekey($res); $content = base64_decode($content); $result = ''; for($i=0; $iOutRsaVerify($string, $data['sign']); } /** * [HttpRequest 网络请求] * @author Devil * @blog http://gong.gg/ * @version 1.0.0 * @datetime 2017-09-25T09:10:46+0800 * @param [string] $url [请求url] * @param [array] $data [发送数据] * @return [mixed] [请求返回数据] */ private function HttpRequest($url, $data) { $ch = curl_init(); curl_setopt($ch, CURLOPT_URL, $url); curl_setopt($ch, CURLOPT_FAILONERROR, false); curl_setopt($ch, CURLOPT_RETURNTRANSFER, true); curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, false); $body_string = ''; if(is_array($data) && 0 < count($data)) { foreach($data as $k => $v) { $body_string .= $k.'='.urlencode($v).'&'; } curl_setopt($ch, CURLOPT_POST, true); curl_setopt($ch, CURLOPT_POSTFIELDS, $body_string); } $headers = array('content-type: application/x-www-form-urlencoded;charset=UTF-8'); curl_setopt($ch, CURLOPT_HTTPHEADER, $headers); $reponse = curl_exec($ch); if(curl_errno($ch)) { return false; } else { $httpStatusCode = curl_getinfo($ch, CURLINFO_HTTP_CODE); if(200 !== $httpStatusCode) { return false; } } curl_close($ch); return json_decode($reponse, true); } /** * [AlipayQrcodeCreate 小程序二维码创建] * @author Devil * @blog http://gong.gg/ * @version 1.0.0 * @datetime 2017-10-28T21:31:41+0800 * @param [string] $query [生成小程序启动参数(如:type=page&page=shop&value=5)] * @param [string] $describe [二维码描述(默认:ShopXO)] */ public function AlipayQrcodeCreate($query, $describe = 'ShopXO') { // 请求参数 $params = [ 'app_id' => MyC('common_app_mini_alipay_appid'), 'method' => 'alipay.open.app.qrcode.create', 'charset' => 'utf-8', 'format' => 'JSON', 'sign_type' => 'RSA2', 'timestamp' => date('Y-m-d H:i:s'), 'version' => '1.0', ]; $biz_content = [ 'url_param' => C('alipay_mini_default_page'), 'query_param' => $query, 'describe' => $describe, ]; $params['biz_content'] = json_encode($biz_content, JSON_UNESCAPED_UNICODE); // 生成签名参数+签名 $p = $this->GetParamSign($params); $params['sign'] = $this->MyRsaSign($p['value']); // 执行请求 $result = $this->HttpRequest('https://openapi.alipay.com/gateway.do', $params); // 结果正确则验证签名 并且 存储缓存返回access_token if(!empty($result['alipay_open_app_qrcode_create_response']['code']) && $result['alipay_open_app_qrcode_create_response']['code'] == 10000) { // 验证签名正确则存储缓存返回数据 if(!$this->SyncRsaVerify($result, 'alipay_open_app_qrcode_create_response')) { return false; } return $result['alipay_open_app_qrcode_create_response']; } return false; } } ?>