| 
									
										
										
										
											2008-11-18 20:11:28 -05:00
										 |  |  | #!/usr/bin/env php
 | 
					
						
							|  |  |  | <?php | 
					
						
							|  |  |  | /* | 
					
						
							| 
									
										
										
										
											2009-08-25 18:14:12 -04:00
										 |  |  |  * StatusNet - the distributed open-source microblogging tool | 
					
						
							| 
									
										
										
										
											2009-08-25 18:12:20 -04:00
										 |  |  |  * Copyright (C) 2008, 2009, StatusNet, Inc. | 
					
						
							| 
									
										
										
										
											2008-11-18 20:11:28 -05:00
										 |  |  |  * | 
					
						
							|  |  |  |  * This program is free software: you can redistribute it and/or modify | 
					
						
							|  |  |  |  * it under the terms of the GNU Affero General Public License as published by | 
					
						
							|  |  |  |  * the Free Software Foundation, either version 3 of the License, or | 
					
						
							|  |  |  |  * (at your option) any later version. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This program is distributed in the hope that it will be useful, | 
					
						
							|  |  |  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							| 
									
										
										
										
											2008-12-23 14:19:07 -05:00
										 |  |  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.     See the | 
					
						
							| 
									
										
										
										
											2008-11-18 20:11:28 -05:00
										 |  |  |  * GNU Affero General Public License for more details. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * You should have received a copy of the GNU Affero General Public License | 
					
						
							| 
									
										
										
										
											2008-12-23 14:19:07 -05:00
										 |  |  |  * along with this program.     If not, see <http://www.gnu.org/licenses/>. | 
					
						
							| 
									
										
										
										
											2008-11-18 20:11:28 -05:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-26 00:59:06 +00:00
										 |  |  | define('INSTALLDIR', realpath(dirname(__FILE__) . '/../../..')); | 
					
						
							| 
									
										
										
										
											2008-11-18 20:11:28 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-06 07:03:05 +00:00
										 |  |  | $shortoptions = 'di::'; | 
					
						
							|  |  |  | $longoptions = array('id::', 'debug'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | $helptext = <<<END_OF_TRIM_HELP | 
					
						
							|  |  |  | Batch script for synching local friends with Twitter friends. | 
					
						
							|  |  |  |   -i --id              Identity (default 'generic') | 
					
						
							|  |  |  |   -d --debug           Debug (lots of log output) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | END_OF_TRIM_HELP; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | require_once INSTALLDIR . '/scripts/commandline.inc'; | 
					
						
							|  |  |  | require_once INSTALLDIR . '/lib/parallelizingdaemon.php'; | 
					
						
							| 
									
										
										
										
											2009-08-26 00:59:06 +00:00
										 |  |  | require_once INSTALLDIR . '/plugins/TwitterBridge/twitter.php'; | 
					
						
							| 
									
										
										
										
											2009-10-14 04:50:16 +00:00
										 |  |  | require_once INSTALLDIR . '/plugins/TwitterBridge/twitteroauthclient.php'; | 
					
						
							| 
									
										
										
										
											2009-08-06 07:03:05 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * Daemon to sync local friends with Twitter friends | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @category Twitter | 
					
						
							| 
									
										
										
										
											2009-08-25 18:12:20 -04:00
										 |  |  |  * @package  StatusNet | 
					
						
							| 
									
										
										
										
											2009-08-25 18:19:04 -04:00
										 |  |  |  * @author   Zach Copley <zach@status.net> | 
					
						
							|  |  |  |  * @author   Evan Prodromou <evan@status.net> | 
					
						
							| 
									
										
										
										
											2009-08-06 07:03:05 +00:00
										 |  |  |  * @license  http://www.fsf.org/licensing/licenses/agpl-3.0.html GNU Affero General Public License version 3.0 | 
					
						
							| 
									
										
										
										
											2009-08-25 18:16:46 -04:00
										 |  |  |  * @link     http://status.net/ | 
					
						
							| 
									
										
										
										
											2009-08-06 07:03:05 +00:00
										 |  |  |  */ | 
					
						
							|  |  |  | class SyncTwitterFriendsDaemon extends ParallelizingDaemon | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-08-06 22:52:58 +00:00
										 |  |  |     /** | 
					
						
							|  |  |  |      *  Constructor | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @param string  $id           the name/id of this daemon | 
					
						
							|  |  |  |      * @param int     $interval     sleep this long before doing everything again | 
					
						
							|  |  |  |      * @param int     $max_children maximum number of child processes at a time | 
					
						
							|  |  |  |      * @param boolean $debug        debug output flag | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return void | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      **/ | 
					
						
							| 
									
										
										
										
											2009-08-06 07:03:05 +00:00
										 |  |  |     function __construct($id = null, $interval = 60, | 
					
						
							|  |  |  |                          $max_children = 2, $debug = null) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         parent::__construct($id, $interval, $max_children, $debug); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-05-07 18:22:14 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-06 07:03:05 +00:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Name of this daemon | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return string Name of the daemon. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     function name() | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2009-08-10 07:18:09 +00:00
										 |  |  |         return ('synctwitterfriends.' . $this->_id); | 
					
						
							| 
									
										
										
										
											2009-08-06 07:03:05 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2008-11-18 20:11:28 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-06 22:52:58 +00:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Find all the Twitter foreign links for users who have requested | 
					
						
							|  |  |  |      * automatically subscribing to their Twitter friends locally. | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return array flinks an array of Foreign_link objects | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2009-08-06 07:03:05 +00:00
										 |  |  |     function getObjects() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $flinks = array(); | 
					
						
							|  |  |  |         $flink = new Foreign_link(); | 
					
						
							| 
									
										
										
										
											2008-11-18 20:11:28 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-06 07:03:05 +00:00
										 |  |  |         $conn = &$flink->getDatabaseConnection(); | 
					
						
							| 
									
										
										
										
											2008-11-18 20:11:28 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-06 07:03:05 +00:00
										 |  |  |         $flink->service = TWITTER_SERVICE; | 
					
						
							|  |  |  |         $flink->orderBy('last_friendsync'); | 
					
						
							|  |  |  |         $flink->limit(25);  // sync this many users during this run
 | 
					
						
							|  |  |  |         $flink->find(); | 
					
						
							| 
									
										
										
										
											2008-11-18 20:11:28 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-06 07:03:05 +00:00
										 |  |  |         while ($flink->fetch()) { | 
					
						
							|  |  |  |             if (($flink->friendsync & FOREIGN_FRIEND_RECV) == FOREIGN_FRIEND_RECV) { | 
					
						
							|  |  |  |                 $flinks[] = clone($flink); | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2009-04-07 18:29:10 -07:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-06 07:03:05 +00:00
										 |  |  |         $conn->disconnect(); | 
					
						
							| 
									
										
										
										
											2008-11-18 20:11:28 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-06 07:03:05 +00:00
										 |  |  |         global $_DB_DATAOBJECT; | 
					
						
							|  |  |  |         unset($_DB_DATAOBJECT['CONNECTIONS']); | 
					
						
							| 
									
										
										
										
											2008-11-19 15:45:01 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-06 07:03:05 +00:00
										 |  |  |         return $flinks; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     function childTask($flink) { | 
					
						
							|  |  |  |         // Each child ps needs its own DB connection
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Note: DataObject::getDatabaseConnection() creates
 | 
					
						
							| 
									
										
										
										
											2009-11-09 20:01:46 +01:00
										 |  |  |         // a new connection if there isn't one already
 | 
					
						
							| 
									
										
										
										
											2009-08-06 07:03:05 +00:00
										 |  |  |         $conn = &$flink->getDatabaseConnection(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $this->subscribeTwitterFriends($flink); | 
					
						
							| 
									
										
										
										
											2009-05-07 18:22:14 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         $flink->last_friendsync = common_sql_now(); | 
					
						
							|  |  |  |         $flink->update(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-06 07:03:05 +00:00
										 |  |  |         $conn->disconnect(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-08 23:28:51 +01:00
										 |  |  |         // XXX: Couldn't find a less brutal way to blow
 | 
					
						
							| 
									
										
										
										
											2009-08-06 07:03:05 +00:00
										 |  |  |         // away a cached connection
 | 
					
						
							|  |  |  |         global $_DB_DATAOBJECT; | 
					
						
							|  |  |  |         unset($_DB_DATAOBJECT['CONNECTIONS']); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     function fetchTwitterFriends($flink) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $friends = array(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-28 07:02:27 +00:00
										 |  |  |         $client = null; | 
					
						
							| 
									
										
										
										
											2009-08-10 07:00:59 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-28 07:02:27 +00:00
										 |  |  |         if (TwitterOAuthClient::isPackedToken($flink->credentials)) { | 
					
						
							|  |  |  |             $token = TwitterOAuthClient::unpackToken($flink->credentials); | 
					
						
							|  |  |  |             $client = new TwitterOAuthClient($token->key, $token->secret); | 
					
						
							|  |  |  |             common_debug($this->name() . '- Grabbing friends IDs with OAuth.'); | 
					
						
							|  |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2010-09-04 12:58:20 -04:00
										 |  |  |             common_debug("Skipping Twitter friends for {$flink->user_id} since not OAuth."); | 
					
						
							|  |  |  |             return $friends; | 
					
						
							| 
									
										
										
										
											2009-08-28 07:02:27 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-08-06 07:03:05 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         try { | 
					
						
							| 
									
										
										
										
											2009-08-10 06:05:43 +00:00
										 |  |  |             $friends_ids = $client->friendsIds(); | 
					
						
							| 
									
										
										
										
											2009-08-28 07:02:27 +00:00
										 |  |  |         } catch (Exception $e) { | 
					
						
							| 
									
										
										
										
											2009-08-06 07:03:05 +00:00
										 |  |  |             common_log(LOG_WARNING, $this->name() . | 
					
						
							| 
									
										
										
										
											2009-10-28 15:29:20 -04:00
										 |  |  |                        ' - error getting friend ids: ' . | 
					
						
							|  |  |  |                        $e->getMessage()); | 
					
						
							| 
									
										
										
										
											2009-08-06 07:03:05 +00:00
										 |  |  |             return $friends; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (empty($friends_ids)) { | 
					
						
							|  |  |  |             common_debug($this->name() . | 
					
						
							|  |  |  |                          " - Twitter user $flink->foreign_id " . | 
					
						
							|  |  |  |                          'doesn\'t have any friends!'); | 
					
						
							|  |  |  |             return $friends; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         common_debug($this->name() . ' - Twitter\'s API says Twitter user id ' . | 
					
						
							|  |  |  |                      "$flink->foreign_id has " . | 
					
						
							|  |  |  |                      count($friends_ids) . ' friends.'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Calculate how many pages to get...
 | 
					
						
							|  |  |  |         $pages = ceil(count($friends_ids) / 100); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if ($pages == 0) { | 
					
						
							|  |  |  |             common_debug($this->name() . " - $user seems to have no friends."); | 
					
						
							| 
									
										
										
										
											2008-12-23 14:19:07 -05:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-08-06 07:03:05 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         for ($i = 1; $i <= $pages; $i++) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         try { | 
					
						
							| 
									
										
										
										
											2009-08-10 06:05:43 +00:00
										 |  |  |             $more_friends = $client->statusesFriends(null, null, null, $i); | 
					
						
							| 
									
										
										
										
											2009-08-28 07:02:27 +00:00
										 |  |  |         } catch (Exception $e) { | 
					
						
							| 
									
										
										
										
											2009-08-06 07:03:05 +00:00
										 |  |  |             common_log(LOG_WARNING, $this->name() . | 
					
						
							|  |  |  |                        ' - cURL error getting Twitter statuses/friends ' . | 
					
						
							|  |  |  |                        "page $i - " . $e->getCode() . ' - ' . | 
					
						
							|  |  |  |                        $e->getMessage()); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (empty($more_friends)) { | 
					
						
							|  |  |  |                 common_log(LOG_WARNING, $this->name() . | 
					
						
							| 
									
										
										
										
											2009-11-08 23:28:51 +01:00
										 |  |  |                            " - Couldn't retrieve page $i " . | 
					
						
							| 
									
										
										
										
											2009-08-06 07:03:05 +00:00
										 |  |  |                            "of Twitter user $flink->foreign_id friends."); | 
					
						
							|  |  |  |                 continue; | 
					
						
							|  |  |  |             } else { | 
					
						
							|  |  |  |                 $friends = array_merge($friends, $more_friends); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return $friends; | 
					
						
							| 
									
										
										
										
											2008-12-23 14:19:07 -05:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2008-11-18 20:11:28 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-06 07:03:05 +00:00
										 |  |  |     function subscribeTwitterFriends($flink) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $friends = $this->fetchTwitterFriends($flink); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (empty($friends)) { | 
					
						
							|  |  |  |             common_debug($this->name() . | 
					
						
							|  |  |  |                          ' - Couldn\'t get friends from Twitter for ' . | 
					
						
							|  |  |  |                          "Twitter user $flink->foreign_id."); | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $user = $flink->getUser(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         foreach ($friends as $friend) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             $friend_name = $friend->screen_name; | 
					
						
							|  |  |  |             $friend_id = (int) $friend->id; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             // Update or create the Foreign_user record for each
 | 
					
						
							|  |  |  |             // Twitter friend
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (!save_twitter_user($friend_id, $friend_name)) { | 
					
						
							| 
									
										
										
										
											2010-03-02 22:09:52 -08:00
										 |  |  |                 common_log(LOG_WARNING, $this->name() . | 
					
						
							| 
									
										
										
										
											2009-11-08 23:28:51 +01:00
										 |  |  |                            " - Couldn't save $screen_name's friend, $friend_name."); | 
					
						
							| 
									
										
										
										
											2009-08-06 07:03:05 +00:00
										 |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-08 23:28:51 +01:00
										 |  |  |             // Check to see if there's a related local user
 | 
					
						
							| 
									
										
										
										
											2009-08-06 07:03:05 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |             $friend_flink = Foreign_link::getByForeignID($friend_id, | 
					
						
							|  |  |  |                                                          TWITTER_SERVICE); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (!empty($friend_flink)) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 // Get associated user and subscribe her
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 $friend_user = User::staticGet('id', $friend_flink->user_id); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 if (!empty($friend_user)) { | 
					
						
							|  |  |  |                     $result = subs_subscribe_to($user, $friend_user); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     if ($result === true) { | 
					
						
							|  |  |  |                         common_log(LOG_INFO, | 
					
						
							|  |  |  |                                    $this->name() . ' - Subscribed ' . | 
					
						
							|  |  |  |                                    "$friend_user->nickname to $user->nickname."); | 
					
						
							|  |  |  |                     } else { | 
					
						
							|  |  |  |                         common_debug($this->name() . | 
					
						
							|  |  |  |                                      ' - Tried subscribing ' . | 
					
						
							|  |  |  |                                      "$friend_user->nickname to $user->nickname - " . | 
					
						
							|  |  |  |                                      $result); | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return true; | 
					
						
							| 
									
										
										
										
											2009-05-07 18:22:14 -07:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-06 07:03:05 +00:00
										 |  |  | $id    = null; | 
					
						
							|  |  |  | $debug = null; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | if (have_option('i')) { | 
					
						
							|  |  |  |     $id = get_option_value('i'); | 
					
						
							|  |  |  | } else if (have_option('--id')) { | 
					
						
							|  |  |  |     $id = get_option_value('--id'); | 
					
						
							|  |  |  | } else if (count($args) > 0) { | 
					
						
							|  |  |  |     $id = $args[0]; | 
					
						
							|  |  |  | } else { | 
					
						
							|  |  |  |     $id = null; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | if (have_option('d') || have_option('debug')) { | 
					
						
							|  |  |  |     $debug = true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | $syncer = new SyncTwitterFriendsDaemon($id, 60, 2, $debug); | 
					
						
							|  |  |  | $syncer->runOnce(); |