20,
CURLOPT_SSL_VERIFYPEER => 0,
CURLOPT_HTTPHEADER => array('Expect:'),
CURLOPT_USERAGENT => 'Twitter for PHP',
);
/** @var Twitter_OAuthSignatureMethod */
private $signatureMethod;
/** @var Twitter_OAuthConsumer */
private $consumer;
/** @var Twitter_OAuthConsumer */
private $token;
/**
* Creates object using consumer and access keys.
* @param string consumer key
* @param string app secret
* @param string optional access token
* @param string optinal access token secret
* @throws TwitterException when CURL extension is not loaded
*/
public function __construct($consumerKey, $consumerSecret, $accessToken = NULL, $accessTokenSecret = NULL)
{
if (!extension_loaded('curl')) {
throw new TwitterException('PHP extension CURL is not loaded.');
}
$this->signatureMethod = new Twitter_OAuthSignatureMethod_HMAC_SHA1();
$this->consumer = new Twitter_OAuthConsumer($consumerKey, $consumerSecret);
$this->token = new Twitter_OAuthConsumer($accessToken, $accessTokenSecret);
}
/**
* Tests if user credentials are valid.
* @return boolean
* @throws TwitterException
*/
public function authenticate()
{
try {
$res = $this->request('account/verify_credentials', 'GET');
return !empty($res->id);
} catch (TwitterException $e) {
if ($e->getCode() === 401) {
return FALSE;
}
throw $e;
}
}
/**
* Sends message to the Twitter.
* @param string message encoded in UTF-8
* @return object
* @throws TwitterException
*/
public function send($message)
{
return $this->request('statuses/update', 'POST', array('status' => $message));
}
/**
* Returns the most recent statuses.
* @param int timeline (ME | ME_AND_FRIENDS | REPLIES) and optional (RETWEETS)
* @param int number of statuses to retrieve
* @param int page of results to retrieve
* @return mixed
* @throws TwitterException
*/
public function load($flags = self::ME, $count = 20, array $data = NULL)
{
static $timelines = array(self::ME => 'user_timeline', self::ME_AND_FRIENDS => 'home_timeline', self::REPLIES => 'mentions_timeline');
if (!isset($timelines[$flags & 3])) {
throw new InvalidArgumentException;
}
return $this->cachedRequest('statuses/' . $timelines[$flags & 3], (array) $data + array(
'count' => $count,
'include_rts' => $flags & self::RETWEETS ? 1 : 0,
));
}
/**
* Returns information of a given user.
* @param string name
* @return mixed
* @throws TwitterException
*/
public function loadUserInfo($user)
{
return $this->cachedRequest('users/show', array('screen_name' => $user));
}
/**
* Returns information of a given user by id.
* @param string name
* @return mixed
* @throws TwitterException
*/
public function loadUserInfoById($id)
{
return $this->cachedRequest('users/show', array('user_id' => $id));
}
/**
* Returns followers of a given user.
* @param string name
* @return mixed
* @throws TwitterException
*/
public function loadUserFollowers($user, $count = 5000, $cursor = -1, $cacheExpiry = null)
{
return $this->cachedRequest('followers/ids', array('screen_name' => $user, 'count' => $count, 'cursor' => $cursor), $cacheExpiry);
}
/**
* Destroys status.
* @param int id of status to be destroyed
* @return mixed
* @throws TwitterException
*/
public function destroy($id)
{
$res = $this->request("statuses/destroy/$id", 'POST');
return $res->id ? $res->id : FALSE;
}
/**
* Returns tweets that match a specified query.
* @param string|array query
* @return mixed
* @throws TwitterException
*/
public function search($query)
{
return $this->request('search/tweets', 'GET', is_array($query) ? $query : array('q' => $query))->statuses;
}
/**
* Process HTTP request.
* @param string URL or twitter command
* @param string HTTP method GET or POST
* @param array data
* @return mixed
* @throws TwitterException
*/
public function request($resource, $method, array $data = NULL)
{
if (!strpos($resource, '://')) {
if (!strpos($resource, '.')) {
$resource .= '.json';
}
$resource = self::API_URL . $resource;
}
foreach (array_keys((array) $data, NULL, TRUE) as $key) {
unset($data[$key]);
}
$request = Twitter_OAuthRequest::from_consumer_and_token($this->consumer, $this->token, $method, $resource, $data);
$request->sign_request($this->signatureMethod, $this->consumer, $this->token);
$options = array(
CURLOPT_HEADER => FALSE,
CURLOPT_RETURNTRANSFER => TRUE,
) + ($method === 'POST' ? array(
CURLOPT_POST => TRUE,
CURLOPT_POSTFIELDS => $request->to_postdata(),
CURLOPT_URL => $request->get_normalized_http_url(),
) : array(
CURLOPT_URL => $request->to_url(),
)) + $this->httpOptions;
$curl = curl_init();
curl_setopt_array($curl, $options);
$result = curl_exec($curl);
if (curl_errno($curl)) {
throw new TwitterException('Server error: ' . curl_error($curl));
}
$payload = version_compare(PHP_VERSION, '5.4.0') >= 0 ?
@json_decode($result, FALSE, 128, JSON_BIGINT_AS_STRING) : @json_decode($result); // intentionally @
if ($payload === FALSE) {
throw new TwitterException('Invalid server response');
}
$code = curl_getinfo($curl, CURLINFO_HTTP_CODE);
if ($code >= 400) {
throw new TwitterException(isset($payload->errors[0]->message) ? $payload->errors[0]->message : "Server error #$code", $code);
}
return $payload;
}
/**
* Cached HTTP request.
* @param string URL or twitter command
* @param array
* @param int
* @return mixed
*/
public function cachedRequest($resource, array $data = NULL, $cacheExpire = NULL)
{
if (!self::$cacheDir) {
return $this->request($resource, 'GET', $data);
}
if ($cacheExpire === NULL) {
$cacheExpire = self::$cacheExpire;
}
$cacheFile = self::$cacheDir . '/twitter.' . md5($resource . json_encode($data) . serialize(array($this->consumer, $this->token)));
$cache = @json_decode(@file_get_contents($cacheFile)); // intentionally @
if ($cache && @filemtime($cacheFile) + $cacheExpire > time()) { // intentionally @
return $cache;
}
try {
$payload = $this->request($resource, 'GET', $data);
file_put_contents($cacheFile, json_encode($payload));
return $payload;
} catch (TwitterException $e) {
if ($cache) {
return $cache;
}
throw $e;
}
}
/**
* Makes twitter links, @usernames and #hashtags clickable.
* @param stdClass|string status
* @return string
*/
public static function clickable($status)
{
if (!is_object($status)) { // back compatibility
trigger_error(__METHOD__ . '() has been changed; pass as parameter status object, not just text.', E_USER_WARNING);
return preg_replace_callback(
'~(?&]~u',
array(__CLASS__, 'clickableCallback'),
html_entity_decode($status, ENT_QUOTES, 'UTF-8')
);
}
$all = array();
foreach ($status->entities->hashtags as $item) {
$all[$item->indices[0]] = array("http://twitter.com/search?q=%23$item->text", "#$item->text", $item->indices[1]);
}
foreach ($status->entities->urls as $item) {
if (!isset($item->expanded_url)) {
$all[$item->indices[0]] = array($item->url, $item->url, $item->indices[1]);
} else {
$all[$item->indices[0]] = array($item->expanded_url, $item->display_url, $item->indices[1]);
}
}
foreach ($status->entities->user_mentions as $item) {
$all[$item->indices[0]] = array("http://twitter.com/$item->screen_name", "@$item->screen_name", $item->indices[1]);
}
if (isset($status->entities->media)) {
foreach ($status->entities->media as $item) {
$all[$item->indices[0]] = array($item->url, $item->display_url, $item->indices[1]);
}
}
krsort($all);
$s = $status->text;
foreach ($all as $pos => $item) {
$s = iconv_substr($s, 0, $pos, 'UTF-8')
. '' . htmlspecialchars($item[1]) . ''
. iconv_substr($s, $item[2], iconv_strlen($s, 'UTF-8'), 'UTF-8');
}
return $s;
}
private static function clickableCallback($m)
{
$m = htmlspecialchars($m[0]);
if ($m[0] === '#') {
$m = substr($m, 1);
return "#$m";
} elseif ($m[0] === '@') {
$m = substr($m, 1);
return "@$m";
} elseif ($m[0] === 'w') {
return "$m";
} elseif ($m[0] === 'h') {
return "$m";
} else {
return $m;
}
}
}
/**
* An exception generated by Twitter.
*/
class TwitterException extends Exception
{
}