<?php /* ! * HybridAuth * http://hybridauth.sourceforge.net | http://github.com/hybridauth/hybridauth * (c) 2009-2012, HybridAuth authors | http://hybridauth.sourceforge.net/licenses.html */ /** * Hybrid_Providers_Twitter provider adapter based on OAuth1 protocol */ class Hybrid_Providers_Twitter extends Hybrid_Provider_Model_OAuth1 { /** * {@inheritdoc} */ function initialize() { parent::initialize(); // Provider api end-points $this->api->api_base_url = "https://api.twitter.com/1.1/"; $this->api->authorize_url = "https://api.twitter.com/oauth/authenticate"; $this->api->request_token_url = "https://api.twitter.com/oauth/request_token"; $this->api->access_token_url = "https://api.twitter.com/oauth/access_token"; if (isset($this->config['api_version']) && $this->config['api_version']) { $this->api->api_base_url = "https://api.twitter.com/{$this->config['api_version']}/"; } if (isset($this->config['authorize']) && $this->config['authorize']) { $this->api->authorize_url = "https://api.twitter.com/oauth/authorize"; } $this->api->curl_auth_header = false; } /** * {@inheritdoc} */ function loginBegin() { // Initiate the Reverse Auth flow; cf. https://dev.twitter.com/docs/ios/using-reverse-auth if (isset($_REQUEST['reverse_auth']) && ($_REQUEST['reverse_auth'] == 'yes')) { $stage1 = $this->api->signedRequest($this->api->request_token_url, 'POST', array('x_auth_mode' => 'reverse_auth')); if ($this->api->http_code != 200) { throw new Exception("Authentication failed! {$this->providerId} returned an error. " . $this->errorMessageByStatus($this->api->http_code), 5); } $responseObj = array('x_reverse_auth_parameters' => $stage1, 'x_reverse_auth_target' => $this->config["keys"]["key"]); $response = json_encode($responseObj); header("Content-Type: application/json", true, 200); echo $response; die(); } $tokens = $this->api->requestToken($this->endpoint); // request tokens as received from provider $this->request_tokens_raw = $tokens; // check the last HTTP status code returned if ($this->api->http_code != 200) { throw new Exception("Authentication failed! {$this->providerId} returned an error. " . $this->errorMessageByStatus($this->api->http_code), 5); } if (!isset($tokens["oauth_token"])) { throw new Exception("Authentication failed! {$this->providerId} returned an invalid oauth token.", 5); } $this->token("request_token", $tokens["oauth_token"]); $this->token("request_token_secret", $tokens["oauth_token_secret"]); // redirect the user to the provider authentication url with force_login if (( isset($this->config['force_login']) && $this->config['force_login'] ) || ( isset($this->config['force']) && $this->config['force'] === true )) { Hybrid_Auth::redirect($this->api->authorizeUrl($tokens, array('force_login' => true))); } // else, redirect the user to the provider authentication url Hybrid_Auth::redirect($this->api->authorizeUrl($tokens)); } /** * {@inheritdoc} */ function loginFinish() { // in case we are completing a Reverse Auth flow; cf. https://dev.twitter.com/docs/ios/using-reverse-auth if (isset($_REQUEST['oauth_token_secret'])) { $tokens = $_REQUEST; $this->access_tokens_raw = $tokens; // we should have an access_token unless something has gone wrong if (!isset($tokens["oauth_token"])) { throw new Exception("Authentication failed! {$this->providerId} returned an invalid access token.", 5); } // Get rid of tokens we don't need $this->deleteToken("request_token"); $this->deleteToken("request_token_secret"); // Store access_token and secret for later use $this->token("access_token", $tokens['oauth_token']); $this->token("access_token_secret", $tokens['oauth_token_secret']); // set user as logged in to the current provider $this->setUserConnected(); return; } parent::loginFinish(); } /** * {@inheritdoc} */ function getUserProfile() { $includeEmail = isset($this->config['includeEmail']) ? (bool) $this->config['includeEmail'] : false; $response = $this->api->get('account/verify_credentials.json'. ($includeEmail ? '?include_email=true' : '')); // check the last HTTP status code returned if ($this->api->http_code != 200) { throw new Exception("User profile request failed! {$this->providerId} returned an error. " . $this->errorMessageByStatus($this->api->http_code), 6); } if (!is_object($response) || !isset($response->id)) { throw new Exception("User profile request failed! {$this->providerId} api returned an invalid response: " . Hybrid_Logger::dumpData( $response ), 6); } # store the user profile. $this->user->profile->identifier = (property_exists($response, 'id')) ? $response->id : ""; $this->user->profile->displayName = (property_exists($response, 'screen_name')) ? $response->screen_name : ""; $this->user->profile->description = (property_exists($response, 'description')) ? $response->description : ""; $this->user->profile->firstName = (property_exists($response, 'name')) ? $response->name : ""; $this->user->profile->photoURL = (property_exists($response, 'profile_image_url')) ? (str_replace('_normal', '', $response->profile_image_url)) : ""; $this->user->profile->profileURL = (property_exists($response, 'screen_name')) ? ("http://twitter.com/" . $response->screen_name) : ""; $this->user->profile->webSiteURL = (property_exists($response, 'url')) ? $response->url : ""; $this->user->profile->region = (property_exists($response, 'location')) ? $response->location : ""; if($includeEmail) $this->user->profile->email = (property_exists($response, 'email')) ? $response->email : ""; return $this->user->profile; } /** * {@inheritdoc} */ function getUserContacts() { $parameters = array('cursor' => '-1'); $response = $this->api->get('friends/ids.json', $parameters); // check the last HTTP status code returned if ($this->api->http_code != 200) { throw new Exception("User contacts request failed! {$this->providerId} returned an error. " . $this->errorMessageByStatus($this->api->http_code)); } if (!$response || !count($response->ids)) { return array(); } // 75 id per time should be okey $contactsids = array_chunk($response->ids, 75); $contacts = array(); foreach ($contactsids as $chunk) { $parameters = array('user_id' => implode(",", $chunk)); $response = $this->api->get('users/lookup.json', $parameters); // check the last HTTP status code returned if ($this->api->http_code != 200) { throw new Exception("User contacts request failed! {$this->providerId} returned an error. " . $this->errorMessageByStatus($this->api->http_code)); } if ($response && count($response)) { foreach ($response as $item) { $uc = new Hybrid_User_Contact(); $uc->identifier = (property_exists($item, 'id')) ? $item->id : ""; $uc->displayName = (property_exists($item, 'name')) ? $item->name : ""; $uc->profileURL = (property_exists($item, 'screen_name')) ? ("http://twitter.com/" . $item->screen_name) : ""; $uc->photoURL = (property_exists($item, 'profile_image_url')) ? $item->profile_image_url : ""; $uc->description = (property_exists($item, 'description')) ? $item->description : ""; $contacts[] = $uc; } } } return $contacts; } /** * {@inheritdoc} */ function setUserStatus($status) { if (is_array($status) && isset($status['message']) && isset($status['picture'])) { $response = $this->api->post('statuses/update_with_media.json', array('status' => $status['message'], 'media[]' => file_get_contents($status['picture'])), null, null, true); } else { $response = $this->api->post('statuses/update.json', array('status' => $status)); } // check the last HTTP status code returned if ($this->api->http_code != 200) { throw new Exception("Update user status failed! {$this->providerId} returned an error. " . $this->errorMessageByStatus($this->api->http_code)); } return $response; } /** * {@inheritdoc} */ function getUserStatus($tweetid) { $info = $this->api->get('statuses/show.json?id=' . $tweetid . '&include_entities=true'); // check the last HTTP status code returned if ($this->api->http_code != 200 || !isset($info->id)) { throw new Exception("Cannot retrieve user status! {$this->providerId} returned an error. " . $this->errorMessageByStatus($this->api->http_code)); } return $info; } /** * load the user latest activity * - timeline : all the stream * - me : the user activity only * * by default return the timeline * {@inheritdoc} */ function getUserActivity($stream) { if ($stream == "me") { $response = $this->api->get('statuses/user_timeline.json'); } else { $response = $this->api->get('statuses/home_timeline.json'); } // check the last HTTP status code returned if ($this->api->http_code != 200) { throw new Exception("User activity stream request failed! {$this->providerId} returned an error. " . $this->errorMessageByStatus($this->api->http_code)); } if (!$response) { return array(); } $activities = array(); foreach ($response as $item) { $ua = new Hybrid_User_Activity(); $ua->id = (property_exists($item, 'id')) ? $item->id : ""; $ua->date = (property_exists($item, 'created_at')) ? strtotime($item->created_at) : ""; $ua->text = (property_exists($item, 'text')) ? $item->text : ""; $ua->user->identifier = (property_exists($item->user, 'id')) ? $item->user->id : ""; $ua->user->displayName = (property_exists($item->user, 'name')) ? $item->user->name : ""; $ua->user->profileURL = (property_exists($item->user, 'screen_name')) ? ("http://twitter.com/" . $item->user->screen_name) : ""; $ua->user->photoURL = (property_exists($item->user, 'profile_image_url')) ? $item->user->profile_image_url : ""; $activities[] = $ua; } return $activities; } }