QQ互聯OAuth

/**
 * QQ互聯 oauth
 * @author dyllen
 *
 */
class Oauth
{
    //取Authorization Code Url
    const PC_CODE_URL = 'https://graph.qq.com/oauth2.0/authorize';
     
    //取Access Token Url
    const PC_ACCESS_TOKEN_URL = 'https://graph.qq.com/oauth2.0/token';
     
    //取用戶 Open Id Url
    const OPEN_ID_URL = 'https://graph.qq.com/oauth2.0/me';
     
    //用戶受權以後的回調地址
    public $redirectUri = null;
     
    // App Id
    public $appid = null;
     
    //App Key
    public $appKey = null;
     
    //受權列表
    //字符串,多個用逗號隔開
    public $scope = null;
     
    //受權code
    public $code = null;
     
    //續期access token的憑證
    public $refreshToken = null;
     
    //access token
    public $accessToken = null;
     
    //access token 有效期,單位秒
    public $expiresIn = null;
     
    //state
    public $state = null;
     
    public $openid = null;
     
    //construct
    public function __construct($config=[])
    {
        foreach($config as $key => $value) {
            $this->$key = $value;
        }
    }
     
    /**
     * 獲得獲取Code的url
     * @throws \InvalidArgumentException
     * @return string
     */
    public function codeUrl()
    {
        if (!$this->redirectUri) {
            throw new \Exception('parameter $redirectUri must be set.');
        }
        $query = [
                'response_type' => 'code',
                'client_id' => $this->appid,
                'redirect_uri' => $this->redirectUri,
                'state' => $this->getState(),
                'scope' => $this->scope,
        ];
     
        return self::PC_CODE_URL . '?' . http_build_query($query);
    }
     
    /**
     * 取access token
     * @throws Exception
     * @return boolean
     */
    public function getAccessToken()
    {
        $params = [
                'grant_type' => 'authorization_code',
                'client_id' => $this->appid,
                'client_secret' => $this->appKey,
                'code' => $this->code,
                'redirect_uri' => $this->redirectUri,
        ];
     
        $url = self::PC_ACCESS_TOKEN_URL . '?' . http_build_query($params);
        $content = $this->getUrl($url);
        parse_str($content, $res);
        if ( !isset($res['access_token']) ) {
            $this->thrwoError($content);
        }
     
        $this->accessToken = $res['access_token'];
        $this->expiresIn = $res['expires_in'];
        $this->refreshToken = $res['refresh_token'];
     
        return true;
    }
     
    /**
     * 刷新access token
     * @throws Exception
     * @return boolean
     */
    public function refreshToken()
    {
        $params = [
                'grant_type' => 'refresh_token',
                'client_id' => $this->appid,
                'client_secret' => $this->appKey,
                'refresh_token' => $this->refreshToken,
        ];
     
        $url = self::PC_ACCESS_TOKEN_URL . '?' . http_build_query($params);
        $content = $this->getUrl($url);
        parse_str($content, $res);
        if ( !isset($res['access_token']) ) {
            $this->thrwoError($content);
        }
     
        $this->accessToken = $res['access_token'];
        $this->expiresIn = $res['expires_in'];
        $this->refreshToken = $res['refresh_token'];
     
        return true;
    }
     
    /**
     * 取用戶open id
     * @return string
     */
    public function getOpenid()
    {
        $params = [
                'access_token' => $this->accessToken,
        ];
     
        $url = self::OPEN_ID_URL . '?' . http_build_query($params);
             
        $this->openid = $this->parseOpenid( $this->getUrl($url) );
         
        return $this->openid;
    }
     
    /**
     * get方式取url內容
     * @param string $url
     * @return mixed
     */
    public function getUrl($url)
    {
        $ch = curl_init();
        curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, FALSE);
        curl_setopt($ch, CURLOPT_RETURNTRANSFER, TRUE);
        curl_setopt($ch, CURLOPT_URL, $url);
        $response =  curl_exec($ch);
        curl_close($ch);
     
        return $response;
    }
     
    /**
     * post方式取url內容
     * @param string $url
     * @param array $keysArr
     * @param number $flag
     * @return mixed
     */
    public function postUrl($url, $keysArr, $flag = 0)
    {
        $ch = curl_init();
        if(! $flag) curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, FALSE);
        curl_setopt($ch, CURLOPT_RETURNTRANSFER, TRUE);
        curl_setopt($ch, CURLOPT_POST, TRUE);
        curl_setopt($ch, CURLOPT_POSTFIELDS, $keysArr);
        curl_setopt($ch, CURLOPT_URL, $url);
        $ret = curl_exec($ch);
     
        curl_close($ch);
        return $ret;
    }
     
     
    /**
     * 取state
     * @return string
     */
    protected function getState()
    {
        $this->state = md5(uniqid(rand(), true));
        //state暫存在緩存裏面
        //本身定義
                //。。。。。。。。。
     
        return $this->state;
    }
     
    /**
     * 驗證state
     * @return boolean
     */
    protected function verifyState()
    {
        //。。。。。。。
    }
     
    /**
     * 拋出異常
     * @param string $error
     * @throws \Exception
     */
    protected function thrwoError($error)
    {
        $subError = substr($error, strpos($error, "{"));
        $subError = strstr($subError, "}", true) . "}";
        $error = json_decode($subError, true);
         
        throw new \Exception($error['error_description'], (int)$error['error']);
    }
     
    /**
     * 從獲取openid接口的返回數據中解析出openid
     * @param string $str
     * @return string
     */
    protected function parseOpenid($str)
    {
        $subStr = substr($str, strpos($str, "{"));
        $subStr = strstr($subStr, "}", true) . "}";
        $strArr = json_decode($subStr, true);
        if(!isset($strArr['openid'])) {
            $this->thrwoError($str);
        }
         
        return $strArr['openid'];
    }
}
相關文章
相關標籤/搜索