DoubanClient.class.php 8.38 KB
<?php

class DoubanClient {
    
    /**
     * @brief 豆瓣Oauth类词头
     */
    const PREFIX = 'Douban';

    /**
     * @brief authorizeCode请求链接
     */
    protected $authorizeUri = 'https://www.douban.com/service/auth2/auth';
    
    /**
     * @brief accessToken请求链接
     */
    protected $accessUri = 'https://www.douban.com/service/auth2/token';
    
    /**
     * @brief api请求链接
     */
    protected $apiUri = 'https://api.douban.com';
                
    /**
     * @brief 豆瓣应用public key
     */
    protected $clientId;
    
    /**
     * @brief 豆瓣应用secret key
     */
    protected $secret;

    /**
     * @brief callback链接
     */
    protected $redirectUri;

    /**
     * @brief Api权限
     */
    protected $scope;
    
    /**
     * @brief 返回类型,默认使用code
     */
    protected $responseType;
    
    /**
     * @brief 用户授权码
     */
    protected $authorizeCode;

    /**
     * @brief 储存返回的令牌(accessToken,refreshToken)
     */
    protected $tokens;

    /**
     * @brief 通过authorizeCode获得的访问令牌
     */
    protected $accessToken;

    /**
     * @brief 用于刷新accessToken
     */
    protected $refreshToken;

    /**
     * @var 默认请求头信息 
     */
    protected $defaultHeader = array(
                'Content_type: application/x-www-form-urlencoded'
               );

    /**
     * @var 需授权的请求头
     */
    protected $authorizeHeader;
    
    /**
     * @var curl默认设置  
     */
    protected $CURL_OPTS = array(
                CURLOPT_CONNECTTIMEOUT => 10,
                CURLOPT_RETURNTRANSFER => true,
                CURLOPT_SSL_VERIFYPEER => false,
                CURLOPT_TIMEOUT        => 60,
                CURLOPT_USERAGENT      => 'simple-douban-oauth2-0.4',
                );

    /**
     * @brief 初始化豆瓣OAUTH,设置相关参数
     *
     * @param string $client_id
     * @param string $secret
     * @param string $redirect_uri
     * @param string $scope
     * @param string $responseType
     *
     * @return void
     */
    public function __construct($clientId, $secret, $redirectUri, $scope ='douban_basic_common', $responseType = 'code')
    {
        $this->clientId = $clientId;
        $this->secret = $secret;
        $this->redirectUri = $redirectUri;
        $this->scope = $scope;
        $this->responseType = $responseType;

//         // API基类路径
//         $basePath = dirname(__FILE__).'/api/DoubanBase.php';

//         // 载入API基类
//         try {
//             $this->fileLoader($basePath);
//         } catch(Exception $e) {
//             echo 'Baseloader error:'.$e->getMessage();
//         }
    }

    /**
     * @brief 跳转到豆瓣用户授权页面,获取AuthorizeCode
     *
     * @return redirect
     */
    public function requestAuthorizeCode()
    {
        // 获取AuthorizeCode请求链接
        $authorizeUrl = $this->getAuthorizeUrl();
        header('Location:'.$authorizeUrl);
    }
    
    /**
     * @brief 设置AuthorizeCode
     *
     * @param string $authorizeCode
     *
     * @return void
     */
    public function setAuthorizeCode($authorizeCode)
    {
        $this->authorizeCode = $authorizeCode;
    }

    /**
     * @brief 通过AuthorizeCode获取accessToken
     *
     * @return string
     */
    public function requestAccessToken()
    {
        // 获取accessToken请求链接
        $accessUrl = $this->getAccessUrl();
        $header = $this->defaultHeader;
        $result = $this->curl($accessUrl, 'POST', $header);
        $this->tokens = json_decode($result);
        $this->refreshToken = $this->tokens->refresh_token;
        $this->accessToken = $this->tokens->access_token;
    }
    
    /**
     * @brief 获取token
     *
     * @return string
     */
    public function getToken()
    {
        return $this->tokens;
    }
        
    /**
     * @brief 获取accessToken
     *
     * @return string
     */
    public function getAccessToken()
    {
        return $this->accessToken;
    }

    /**
     * @brief 设置token
     *
     * @return string
     */
    public function setAccessToken($token)
    {
        if (is_array($token) && isset($token['access_token'])) {
            $this->accessToken = $token['access_token'];
        }
    }
    
    /**
     * @brief 生成豆瓣用户授权页面完整地址
     *
     * @return string
     */
    public function getAuthorizeUrl()
    {
        $params = array(
                    'client_id' => $this->clientId,
                    'redirect_uri' => $this->redirectUri,
                    'response_type' => $this->responseType,
                    'scope' => $this->scope
                    );

        return $this->authorizeUri.'?'.http_build_query($params);
    }

    /**
     * @brief 生成豆瓣access_token完整获取链接
     *
     * @return string
     */
    public function getAccessUrl()
    {
        $params = array(
                    'client_id' => $this->clientId,
                    'client_secret' => $this->secret,
                    'redirect_uri' => $this->redirectUri,
                    'grant_type' => 'authorization_code',
                    'code' => $this->authorizeCode,
                    );

        return $this->accessUri.'?'.http_build_query($params);
    }
    
    /**
     * @brief 请求豆瓣API,返回包含相关数据的对象
     *
     * @param object $API
     * @param array $data
     * @param boolean 为true时会在header中发送accessToken
     *
     * @return object
     */
    public function api($api, $type = 'POST', $data = null, $authorization = false)
    {
        // API的完整URL
        $url = $this->apiUri.$api;
        $header = $authorization ? $this->getAuthorizeHeader() : $this->defaultHeader;
    
        $response = $this->curl($url, $type, $header, $data);
        $result = json_decode($response, true);
        
        return $result;
    }

    /**
     * @brief 请求豆瓣API,返回包含相关数据的对象
     *
     * @param object $API
     * @param array $data
     * @param boolean 为true时会在header中发送accessToken
     *
     * @return object
     */
    public function makeRequest($api, $data = null, $authorization = false)
    {
        // API的完整URL
        $url = $this->apiUri.$api->uri;
        $header = $authorization ? $this->getAuthorizeHeader() : $this->defaultHeader;
        $type = $api->type;

        return $this->curl($url, $type, $header, $data);
    }
    
    /**
     * @brief 豆瓣API实例注册函数
     *
     * @param string $api
     *
     * @return object
     */
    public function apiRegister($api)
    {
        $doubanApi = self::PREFIX.ucfirst(strtolower($api));
        // 需要注册的API路径
        $apiPath = dirname(__FILE__).'/api/'.$doubanApi.'.php';

        try {
            $this->fileLoader($apiPath);
        } catch(Exception $e) {
            echo 'Apiloader error:'.$e->getMessage();
        }

        return new $doubanApi($this->clientId);
    }

    /**
     * @brief 获取Authorization header
     *
     * @return array
     */
    protected function getAuthorizeHeader()
    {
        return $this->authorizeHeader = array('Authorization: Bearer '.$this->accessToken);
    }

    /**
     * @brief 使用CURL模拟请求,并返回取得的数据
     *
     * @param string $url
     * @param string $type
     * @param array $header
     * @param array $data
     *
     * @return object
     */
    protected function curl($url, $type, $header, $data = null)
    {
        $opts = $this->CURL_OPTS;
        $opts[CURLOPT_URL] = $url;
        $opts[CURLOPT_CUSTOMREQUEST] = $type;
        $header[] = 'Expect:'; 
        $opts[CURLOPT_HTTPHEADER] = $header;
        if ($type == 'POST' || $type =='PUT') {
            $opts[CURLOPT_POSTFIELDS] = $data;
        }

        $ch = curl_init();
        curl_setopt_array($ch, $opts);
        $result = curl_exec($ch);

//         if (curl_errno($ch)) {
//             die('CURL error: '.curl_error($ch));
//         }

        curl_close($ch);  
        return $result;
    }
    
    /**
     * @brief 文件加载类
     *
     * @param string $path
     *
     * @return void
     */
    protected function fileLoader($path)
    {
        // 文件路径错误时抛出异常
        if ( ! file_exists($path)) {
            throw new Exception('The file you wanted to load does not exists.');
        }

        require $path;
    }
}