lib/plugin.php now has a parent onAutoload function that finds most common
files that are used in plugins (actions, dataobjects, forms, libs etc.) if
they are put in the standardised directories ('actions', 'classes', 'forms',
'lib' and perhaps some others in the future).
		
	
		
			
				
	
	
		
			631 lines
		
	
	
		
			17 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			631 lines
		
	
	
		
			17 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | |
| /**
 | |
|  * StatusNet - the distributed open-source microblogging tool
 | |
|  * Copyright (C) 2010-2011, StatusNet, Inc.
 | |
|  *
 | |
|  * A plugin for integrating Facebook with StatusNet. Includes single-sign-on
 | |
|  * and publishing notices to Facebook using Facebook's Graph API.
 | |
|  *
 | |
|  * PHP version 5
 | |
|  *
 | |
|  * 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
 | |
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | |
|  * GNU Affero General Public License for more details.
 | |
|  *
 | |
|  * You should have received a copy of the GNU Affero General Public License
 | |
|  * along with this program.  If not, see <http://www.gnu.org/licenses/>.
 | |
|  *
 | |
|  * @category  Plugin
 | |
|  * @package   StatusNet
 | |
|  * @author    Zach Copley <zach@status.net>
 | |
|  * @copyright 2011 StatusNet, Inc.
 | |
|  * @license   http://www.fsf.org/licensing/licenses/agpl-3.0.html AGPL 3.0
 | |
|  * @link      http://status.net/
 | |
|  */
 | |
| 
 | |
| if (!defined('STATUSNET')) {
 | |
|     exit(1);
 | |
| }
 | |
| 
 | |
| define("FACEBOOK_SERVICE", 2);
 | |
| 
 | |
| /**
 | |
|  * Main class for Facebook Bridge plugin
 | |
|  *
 | |
|  * @category  Plugin
 | |
|  * @package   StatusNet
 | |
|  * @author    Zach Copley <zach@status.net>
 | |
|  * @copyright 2010-2011 StatusNet, Inc.
 | |
|  * @license   http://www.fsf.org/licensing/licenses/agpl-3.0.html AGPL 3.0
 | |
|  * @link      http://status.net/
 | |
|  */
 | |
| class FacebookBridgePlugin extends Plugin
 | |
| {
 | |
|     public $appId;  // Facebook application ID
 | |
|     public $secret; // Facebook application secret
 | |
| 
 | |
|     public $facebook = null; // Facebook application instance
 | |
|     public $dir      = null; // Facebook plugin dir
 | |
| 
 | |
|     /**
 | |
|      * Initializer for this plugin
 | |
|      *
 | |
|      * Gets an instance of the Facebook API client object
 | |
|      *
 | |
|      * @return boolean hook value; true means continue processing, false means stop.
 | |
|      */
 | |
|     function initialize()
 | |
|     {
 | |
| 
 | |
|         // Allow the id and key to be passed in
 | |
|         // Control panel will override
 | |
| 
 | |
|         if (isset($this->appId)) {
 | |
|             $appId = common_config('facebook', 'appid');
 | |
|             if (empty($appId)) {
 | |
|                 Config::save(
 | |
|                     'facebook',
 | |
|                     'appid',
 | |
|                     $this->appId
 | |
|                 );
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         if (isset($this->secret)) {
 | |
|             $secret = common_config('facebook', 'secret');
 | |
|             if (empty($secret)) {
 | |
|                 Config::save('facebook', 'secret', $this->secret);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         $this->facebook = Facebookclient::getFacebook(
 | |
|             $this->appId,
 | |
|             $this->secret
 | |
|         );
 | |
| 
 | |
|         return true;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Load related modules when needed
 | |
|      *
 | |
|      * @param string $cls Name of the class to be loaded
 | |
|      *
 | |
|      * @return boolean hook value; true means continue processing, false means stop.
 | |
|      */
 | |
|     function onAutoload($cls)
 | |
|     {
 | |
|         $dir = dirname(__FILE__);
 | |
| 
 | |
|         switch ($cls)
 | |
|         {
 | |
|         case 'Facebook': // Facebook PHP SDK
 | |
|             include_once $dir . '/extlib/base_facebook.php';
 | |
|             include_once $dir . '/extlib/facebook.php';
 | |
|             return false;
 | |
|         }
 | |
| 
 | |
|         return parent::onAutoload($cls);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Database schema setup
 | |
|      *
 | |
|      * We maintain a table mapping StatusNet notices to Facebook items
 | |
|      *
 | |
|      * @see Schema
 | |
|      * @see ColumnDef
 | |
|      *
 | |
|      * @return boolean hook value; true means continue processing, false means stop.
 | |
|      */
 | |
|     function onCheckSchema()
 | |
|     {
 | |
|         $schema = Schema::get();
 | |
|         $schema->ensureTable('notice_to_item', Notice_to_item::schemaDef());
 | |
|         return true;
 | |
|     }
 | |
| 
 | |
|     /*
 | |
|      * Does this $action need the Facebook JavaScripts?
 | |
|      */
 | |
|     function needsScripts($action)
 | |
|     {
 | |
|         static $needy = array(
 | |
|             'FacebookloginAction',
 | |
|             'FacebookfinishloginAction',
 | |
|             'FacebookadminpanelAction',
 | |
|             'FacebooksettingsAction'
 | |
|         );
 | |
| 
 | |
|         if (in_array(get_class($action), $needy)) {
 | |
|             return true;
 | |
|         } else {
 | |
|             return false;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Map URLs to actions
 | |
|      *
 | |
|      * @param Net_URL_Mapper $m path-to-action mapper
 | |
|      *
 | |
|      * @return boolean hook value; true means continue processing, false means stop.
 | |
|      */
 | |
|     function onRouterInitialized($m)
 | |
|     {
 | |
|         // Always add the admin panel route
 | |
|         $m->connect('panel/facebook', array('action' => 'facebookadminpanel'));
 | |
| 
 | |
|         $m->connect(
 | |
|             'main/facebooklogin',
 | |
|             array('action' => 'facebooklogin')
 | |
|         );
 | |
|         $m->connect(
 | |
|             'main/facebookfinishlogin',
 | |
|             array('action' => 'facebookfinishlogin')
 | |
|         );
 | |
|         $m->connect(
 | |
|             'settings/facebook',
 | |
|             array('action' => 'facebooksettings')
 | |
|         );
 | |
|         $m->connect(
 | |
|             'facebook/deauthorize',
 | |
|             array('action' => 'facebookdeauthorize')
 | |
|         );
 | |
| 
 | |
|         return true;
 | |
|     }
 | |
| 
 | |
|     /*
 | |
|      * Add a login tab for Facebook, but only if there's a Facebook
 | |
|      * application defined for the plugin to use.
 | |
|      *
 | |
|      * @param Action $action the current action
 | |
|      *
 | |
|      * @return void
 | |
|      */
 | |
|     function onEndLoginGroupNav($action)
 | |
|     {
 | |
|         $action_name = $action->trimmed('action');
 | |
| 
 | |
|         if ($this->hasApplication()) {
 | |
| 
 | |
|             $action->menuItem(
 | |
|                 // TRANS: Menu item for "Facebook" login.
 | |
|                 common_local_url('facebooklogin'),
 | |
|                 _m('MENU', 'Facebook'),
 | |
|                 // TRANS: Menu title for "Facebook" login.
 | |
|                 _m('Login or register using Facebook.'),
 | |
|                'facebooklogin' === $action_name
 | |
|             );
 | |
|         }
 | |
| 
 | |
|         return true;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * If the plugin's installed, this should be accessible to admins
 | |
|      */
 | |
|     function onAdminPanelCheck($name, &$isOK)
 | |
|     {
 | |
|         if ($name == 'facebook') {
 | |
|             $isOK = true;
 | |
|             return false;
 | |
|         }
 | |
| 
 | |
|         return true;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Add a Facebook tab to the admin panels
 | |
|      *
 | |
|      * @param Widget $nav Admin panel nav
 | |
|      *
 | |
|      * @return boolean hook value
 | |
|      */
 | |
|     function onEndAdminPanelNav($nav)
 | |
|     {
 | |
|         if (AdminPanelAction::canAdmin('facebook')) {
 | |
| 
 | |
|             $action_name = $nav->action->trimmed('action');
 | |
| 
 | |
|             $nav->out->menuItem(
 | |
|                 common_local_url('facebookadminpanel'),
 | |
|                 // TRANS: Menu item for "Facebook" in administration panel.
 | |
|                 _m('MENU','Facebook'),
 | |
|                 // TRANS: Menu title for "Facebook" in administration panel.
 | |
|                 _m('Facebook integration configuration.'),
 | |
|                 $action_name == 'facebookadminpanel',
 | |
|                 'nav_facebook_admin_panel'
 | |
|             );
 | |
|         }
 | |
| 
 | |
|         return true;
 | |
|     }
 | |
| 
 | |
|     /*
 | |
|      * Add a tab for user-level Facebook settings if the user
 | |
|      * has a link to Facebook
 | |
|      *
 | |
|      * @param Action $action the current action
 | |
|      *
 | |
|      * @return void
 | |
|      */
 | |
|     function onEndConnectSettingsNav($action)
 | |
|     {
 | |
|         if ($this->hasApplication()) {
 | |
|             $action_name = $action->trimmed('action');
 | |
| 
 | |
|             $user = common_current_user();
 | |
| 
 | |
|             $flink = null;
 | |
| 
 | |
|             if (!empty($user)) {
 | |
|                 $flink = Foreign_link::getByUserID(
 | |
|                     $user->id,
 | |
|                     FACEBOOK_SERVICE
 | |
|                 );
 | |
|             }
 | |
| 
 | |
|             if (!empty($flink)) {
 | |
| 
 | |
|                 $action->menuItem(
 | |
|                     common_local_url('facebooksettings'),
 | |
|                     // TRANS: Menu item for "Facebook" in user settings.
 | |
|                     _m('MENU','Facebook'),
 | |
|                     // TRANS: Menu title for "Facebook" in user settings.
 | |
|                     _m('Facebook settings.'),
 | |
|                     $action_name === 'facebooksettings'
 | |
|                 );
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /*
 | |
|      * Is there a Facebook application for the plugin to use?
 | |
|      *
 | |
|      * Checks to see if a Facebook application ID and secret
 | |
|      * have been configured and a valid Facebook API client
 | |
|      * object exists.
 | |
|      *
 | |
|      */
 | |
|     function hasApplication()
 | |
|     {
 | |
|         if (!empty($this->facebook)) {
 | |
| 
 | |
|             $appId  = $this->facebook->getAppId();
 | |
|             $secret = $this->facebook->getApiSecret();
 | |
| 
 | |
|             if (!empty($appId) && !empty($secret)) {
 | |
|                 return true;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     /*
 | |
|      * Output a Facebook div for the Facebook JavaSsript SDK to use
 | |
|      *
 | |
|      * @param Action $action the current action
 | |
|      *
 | |
|      */
 | |
|     function onStartShowHeader($action)
 | |
|     {
 | |
|         // output <div id="fb-root"></div> as close to <body> as possible
 | |
|         $action->element('div', array('id' => 'fb-root'));
 | |
|         return true;
 | |
|     }
 | |
| 
 | |
|     /*
 | |
|      * Load the Facebook JavaScript SDK on pages that need them.
 | |
|      *
 | |
|      * @param Action $action the current action
 | |
|      *
 | |
|      */
 | |
|     function onEndShowScripts($action)
 | |
|     {
 | |
|         if ($this->needsScripts($action)) {
 | |
| 
 | |
|             $action->script('https://connect.facebook.net/en_US/all.js');
 | |
| 
 | |
|             $script = <<<ENDOFSCRIPT
 | |
| function setCookie(name, value) {
 | |
|     var date = new Date();
 | |
|     date.setTime(date.getTime() + (5 * 60 * 1000)); // 5 mins
 | |
|     var expires = "; expires=" + date.toGMTString();
 | |
|     document.cookie = name + "=" + value + expires + "; path=/";
 | |
| }
 | |
| 
 | |
| FB.init({appId: %1\$s, status: true, cookie: true, xfbml: true, oauth: true});
 | |
| 
 | |
| $('#facebook_button').bind('click', function(event) {
 | |
| 
 | |
|     event.preventDefault();
 | |
| 
 | |
|     FB.login(function(response) {
 | |
|         if (response.authResponse) {
 | |
|             // put the access token in a cookie for the next step
 | |
|             setCookie('fb_access_token', response.authResponse.accessToken);
 | |
|             window.location.href = '%2\$s';
 | |
|         } else {
 | |
|             // NOP (user cancelled login)
 | |
|         }
 | |
|     }, {scope:'read_stream,publish_stream,offline_access,user_status,user_location,user_website,email'});
 | |
| });
 | |
| ENDOFSCRIPT;
 | |
| 
 | |
|             $action->inlineScript(
 | |
|                 sprintf(
 | |
|                     $script,
 | |
|                     json_encode($this->facebook->getAppId()),
 | |
|                     common_local_url('facebookfinishlogin')
 | |
|                 )
 | |
|             );
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /*
 | |
|      * Log the user out of Facebook, per the Facebook authentication guide
 | |
|      *
 | |
|      * @param Action action the current action
 | |
|      */
 | |
|     function onStartLogout($action)
 | |
|     {
 | |
|         if ($this->hasApplication()) {
 | |
| 
 | |
|             $cur = common_current_user();
 | |
|             $flink = Foreign_link::getByUserID($cur->id, FACEBOOK_SERVICE);
 | |
| 
 | |
|             if (!empty($flink)) {
 | |
| 
 | |
|                 $this->facebook->setAccessToken($flink->credentials);
 | |
| 
 | |
|                 if (common_config('singleuser', 'enabled')) {
 | |
|                     $user = User::singleUser();
 | |
| 
 | |
|                     $destination = common_local_url(
 | |
|                         'showstream',
 | |
|                         array('nickname' => $user->nickname)
 | |
|                     );
 | |
|                 } else {
 | |
|                     $destination = common_local_url('public');
 | |
|                 }
 | |
| 
 | |
|                 $logoutUrl = $this->facebook->getLogoutUrl(
 | |
|                     array('next' => $destination)
 | |
|                 );
 | |
| 
 | |
|                 common_log(
 | |
|                     LOG_INFO,
 | |
|                     sprintf(
 | |
|                         "Logging user out of Facebook (fbuid = %s)",
 | |
|                         $fbuid
 | |
|                     ),
 | |
|                     __FILE__
 | |
|                 );
 | |
| 
 | |
|                 $action->logout();
 | |
| 
 | |
|                 common_redirect($logoutUrl, 303);
 | |
|                 return false; // probably never get here, but hey
 | |
|             }
 | |
| 
 | |
|             return true;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /*
 | |
|      * Add fbml namespace to our HTML, so Facebook's JavaScript SDK can parse
 | |
|      * and render XFBML tags
 | |
|      *
 | |
|      * @param Action    $action   the current action
 | |
|      * @param array     $attrs    array of attributes for the HTML tag
 | |
|      *
 | |
|      * @return nothing
 | |
|      */
 | |
|     function onStartHtmlElement($action, $attrs) {
 | |
| 
 | |
|         if ($this->needsScripts($action)) {
 | |
|             $attrs = array_merge(
 | |
|                 $attrs,
 | |
|                 array('xmlns:fb' => 'http://www.facebook.com/2008/fbml')
 | |
|             );
 | |
|         }
 | |
| 
 | |
|         return true;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Add a Facebook queue item for each notice
 | |
|      *
 | |
|      * @param Notice $notice      the notice
 | |
|      * @param array  &$transports the list of transports (queues)
 | |
|      *
 | |
|      * @return boolean hook return
 | |
|      */
 | |
|     function onStartEnqueueNotice($notice, &$transports)
 | |
|     {
 | |
|         if (self::hasApplication() && $notice->isLocal() && $notice->inScope(null)) {
 | |
|             array_push($transports, 'facebook');
 | |
|         }
 | |
|         return true;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Register Facebook notice queue handler
 | |
|      *
 | |
|      * @param QueueManager $manager
 | |
|      *
 | |
|      * @return boolean hook return
 | |
|      */
 | |
|     function onEndInitializeQueueManager($manager)
 | |
|     {
 | |
|         if (self::hasApplication()) {
 | |
|             $manager->connect('facebook', 'FacebookQueueHandler');
 | |
|         }
 | |
|         return true;
 | |
|     }
 | |
| 
 | |
|     /*
 | |
|      * Use SSL for Facebook stuff
 | |
|      *
 | |
|      * @param string $action name
 | |
|      * @param boolean $ssl outval to force SSL
 | |
|      * @return mixed hook return value
 | |
|      */
 | |
|     function onSensitiveAction($action, &$ssl)
 | |
|     {
 | |
|         $sensitive = array(
 | |
|             'facebookadminpanel',
 | |
|             'facebooksettings',
 | |
|             'facebooklogin',
 | |
|             'facebookfinishlogin'
 | |
|         );
 | |
| 
 | |
|         if (in_array($action, $sensitive)) {
 | |
|             $ssl = true;
 | |
|             return false;
 | |
|         } else {
 | |
|             return true;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * If a notice gets deleted, remove the Notice_to_item mapping and
 | |
|      * delete the item on Facebook
 | |
|      *
 | |
|      * @param User   $user   The user doing the deleting
 | |
|      * @param Notice $notice The notice getting deleted
 | |
|      *
 | |
|      * @return boolean hook value
 | |
|      */
 | |
|     function onStartDeleteOwnNotice(User $user, Notice $notice)
 | |
|     {
 | |
|         $client = new Facebookclient($notice);
 | |
|         $client->streamRemove();
 | |
| 
 | |
|         return true;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Notify remote users when their notices get favorited.
 | |
|      *
 | |
|      * @param Profile or User $profile of local user doing the faving
 | |
|      * @param Notice $notice being favored
 | |
|      * @return hook return value
 | |
|      */
 | |
|     function onEndFavorNotice(Profile $profile, Notice $notice)
 | |
|     {
 | |
|         $client = new Facebookclient($notice, $profile);
 | |
|         $client->like();
 | |
| 
 | |
|         return true;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Notify remote users when their notices get de-favorited.
 | |
|      *
 | |
|      * @param Profile $profile Profile person doing the de-faving
 | |
|      * @param Notice  $notice  Notice being favored
 | |
|      *
 | |
|      * @return hook return value
 | |
|      */
 | |
|     function onEndDisfavorNotice(Profile $profile, Notice $notice)
 | |
|     {
 | |
|         $client = new Facebookclient($notice, $profile);
 | |
|         $client->unLike();
 | |
| 
 | |
|         return true;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Add links in the user's profile block to their Facebook profile URL.
 | |
|      *
 | |
|      * @param Profile $profile The profile being shown
 | |
|      * @param Array   &$links  Writeable array of arrays (href, text, image).
 | |
|      *
 | |
|      * @return boolean hook value (true)
 | |
|      */
 | |
| 
 | |
|     function onOtherAccountProfiles($profile, &$links)
 | |
|     {
 | |
|         $fuser = null;
 | |
| 
 | |
|         $flink = Foreign_link::getByUserID($profile->id, FACEBOOK_SERVICE);
 | |
| 
 | |
|         if (!empty($flink)) {
 | |
| 
 | |
|             $fuser = $this->getFacebookUser($flink->foreign_id);
 | |
| 
 | |
|             if (!empty($fuser)) {
 | |
|                 $links[] = array("href" => $fuser->link,
 | |
|                                  "text" => sprintf(_("%s on Facebook"), $fuser->name),
 | |
|                                  "image" => $this->path("images/f_logo.png"));
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         return true;
 | |
|     }
 | |
| 
 | |
|     function getFacebookUser($id) {
 | |
| 
 | |
|         $key = Cache::key(sprintf("FacebookBridgePlugin:userdata:%s", $id));
 | |
| 
 | |
|         $c = Cache::instance();
 | |
| 
 | |
|         if ($c) {
 | |
|             $obj = $c->get($key);
 | |
|             if ($obj) {
 | |
|                 return $obj;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         $url = sprintf("https://graph.facebook.com/%s", $id);
 | |
|         $client = new HTTPClient();
 | |
|         $resp = $client->get($url);
 | |
| 
 | |
|         if (!$resp->isOK()) {
 | |
|             return null;
 | |
|         }
 | |
| 
 | |
|         $user = json_decode($resp->getBody());
 | |
| 
 | |
|         if ($user->error) {
 | |
|             return null;
 | |
|         }
 | |
| 
 | |
|         if ($c) {
 | |
|             $c->set($key, $user);
 | |
|         }
 | |
| 
 | |
|         return $user;
 | |
|     }
 | |
| 
 | |
|     /*
 | |
|      * Add version info for this plugin
 | |
|      *
 | |
|      * @param array &$versions    plugin version descriptions
 | |
|      */
 | |
|     function onPluginVersion(&$versions)
 | |
|     {
 | |
|         $versions[] = array(
 | |
|             'name' => 'Facebook Bridge',
 | |
|             'version' => STATUSNET_VERSION,
 | |
|             'author' => 'Craig Andrews, Zach Copley',
 | |
|             'homepage' => 'http://status.net/wiki/Plugin:FacebookBridge',
 | |
|             'rawdescription' =>
 | |
|              // TRANS: Plugin description.
 | |
|             _m('A plugin for integrating StatusNet with Facebook.')
 | |
|         );
 | |
| 
 | |
|         return true;
 | |
|     }
 | |
| }
 |