123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623 |
- <?php
- // This file is part of GNU social - https://www.gnu.org/software/social
- //
- // GNU social 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.
- //
- // GNU social 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 GNU social. If not, see <http://www.gnu.org/licenses/>.
- /**
- * Plugin for sending and importing Twitter statuses
- *
- * @category Plugin
- * @package GNUsocial
- * @author Zach Copley <zach@status.net>
- * @author Julien C <chaumond@gmail.com>
- * @copyright 2009-2010 StatusNet, Inc.
- * @license https://www.gnu.org/licenses/agpl.html GNU AGPL v3 or later
- */
- defined('GNUSOCIAL') || die();
- require_once __DIR__ . '/twitter.php';
- /**
- * Plugin for sending and importing Twitter statuses
- *
- * This class allows users to link their Twitter accounts
- *
- * Depends on Favorite plugin.
- *
- * @category Plugin
- * @package GNUsocial
- * @author Zach Copley <zach@status.net>
- * @author Julien C <chaumond@gmail.com>
- * @license https://www.gnu.org/licenses/agpl.html GNU AGPL v3 or later
- */
- class TwitterBridgePlugin extends Plugin
- {
- const PLUGIN_VERSION = '2.0.0';
- public $adminImportControl = false; // Should the 'import' checkbox be exposed in the admin panel?
- /**
- * Initializer for the plugin.
- */
- public function initialize()
- {
- // Allow the key and secret to be passed in
- // Control panel will override
- if (isset($this->consumer_key)) {
- $key = common_config('twitter', 'consumer_key');
- if (empty($key)) {
- Config::save('twitter', 'consumer_key', $this->consumer_key);
- }
- }
- if (isset($this->consumer_secret)) {
- $secret = common_config('twitter', 'consumer_secret');
- if (empty($secret)) {
- Config::save(
- 'twitter',
- 'consumer_secret',
- $this->consumer_secret
- );
- }
- }
- }
- /**
- * Check to see if there is a consumer key and secret defined
- * for Twitter integration.
- *
- * @return boolean result
- */
- public static function hasKeys()
- {
- $ckey = common_config('twitter', 'consumer_key');
- $csecret = common_config('twitter', 'consumer_secret');
- if (empty($ckey) && empty($csecret)) {
- $ckey = common_config('twitter', 'global_consumer_key');
- $csecret = common_config('twitter', 'global_consumer_secret');
- }
- if (!empty($ckey) && !empty($csecret)) {
- return true;
- }
- return false;
- }
- /**
- * Add Twitter-related paths to the router table
- *
- * Hook for RouterInitialized event.
- *
- * @param URLMapper $m path-to-action mapper
- *
- * @return boolean hook return
- */
- public function onRouterInitialized(URLMapper $m)
- {
- $m->connect('panel/twitter', ['action' => 'twitteradminpanel']);
- if (self::hasKeys()) {
- $m->connect(
- 'twitter/authorization',
- ['action' => 'twitterauthorization']
- );
- $m->connect(
- 'settings/twitter',
- ['action' => 'twittersettings']
- );
- if (common_config('twitter', 'signin')) {
- $m->connect(
- 'main/twitterlogin',
- ['action' => 'twitterlogin']
- );
- }
- }
- return true;
- }
- /*
- * Add a login tab for 'Sign in with Twitter'
- *
- * @param Action $action the current action
- *
- * @return void
- */
- public function onEndLoginGroupNav($action)
- {
- $action_name = $action->trimmed('action');
- if (self::hasKeys() && common_config('twitter', 'signin')) {
- $action->menuItem(
- common_local_url('twitterlogin'),
- // TRANS: Menu item in login navigation.
- _m('MENU', 'Twitter'),
- // TRANS: Title for menu item in login navigation.
- _m('Login or register using Twitter.'),
- 'twitterlogin' === $action_name
- );
- }
- return true;
- }
- /**
- * Add the Twitter Settings page to the Connect Settings menu
- *
- * @param Action $action The calling page
- *
- * @return boolean hook return
- */
- public function onEndConnectSettingsNav($action)
- {
- if (self::hasKeys()) {
- $action_name = $action->trimmed('action');
- $action->menuItem(
- common_local_url('twittersettings'),
- // TRANS: Menu item in connection settings navigation.
- _m('MENU', 'Twitter'),
- // TRANS: Title for menu item in connection settings navigation.
- _m('Twitter integration options'),
- $action_name === 'twittersettings'
- );
- }
- return true;
- }
- /**
- * Add a Twitter queue item for each notice
- *
- * @param Notice $notice the notice
- * @param array &$transports the list of transports (queues)
- *
- * @return boolean hook return
- */
- public function onStartEnqueueNotice($notice, &$transports)
- {
- if (self::hasKeys() && $notice->isLocal() && $notice->inScope(null)) {
- // Avoid a possible loop
- if ($notice->source != 'twitter') {
- array_push($transports, 'twitter');
- }
- }
- return true;
- }
- /**
- * Add Twitter bridge daemons to the list of daemons to start
- *
- * @param array $daemons the list fo daemons to run
- *
- * @return boolean hook return
- */
- public function onGetValidDaemons(&$daemons)
- {
- if (self::hasKeys()) {
- array_push(
- $daemons,
- INSTALLDIR
- . '/plugins/TwitterBridge/daemons/synctwitterfriends.php'
- );
- if (common_config('twitterimport', 'enabled')) {
- array_push(
- $daemons,
- INSTALLDIR
- . '/plugins/TwitterBridge/daemons/twitterstatusfetcher.php'
- );
- }
- }
- return true;
- }
- /**
- * Register Twitter notice queue handler
- *
- * @param QueueManager $manager
- *
- * @return boolean hook return
- */
- public function onEndInitializeQueueManager($manager)
- {
- if (self::hasKeys()) {
- // Outgoing notices -> twitter
- $manager->connect('twitter', 'TwitterQueueHandler');
- // Incoming statuses <- twitter
- $manager->connect('tweetin', 'TweetInQueueHandler');
- }
- return true;
- }
- /**
- * If the plugin's installed, this should be accessible to admins
- */
- public function onAdminPanelCheck($name, &$isOK)
- {
- if ($name == 'twitter') {
- $isOK = true;
- return false;
- }
- return true;
- }
- /**
- * Add a Twitter tab to the admin panel
- *
- * @param Widget $nav Admin panel nav
- *
- * @return boolean hook value
- */
- public function onEndAdminPanelNav($nav)
- {
- if (AdminPanelAction::canAdmin('twitter')) {
- $action_name = $nav->action->trimmed('action');
- $nav->out->menuItem(
- common_local_url('twitteradminpanel'),
- // TRANS: Menu item in administrative panel that leads to the Twitter bridge configuration.
- _m('Twitter'),
- // TRANS: Menu item title in administrative panel that leads to the Twitter bridge configuration.
- _m('Twitter bridge configuration page.'),
- $action_name == 'twitteradminpanel',
- 'nav_twitter_admin_panel'
- );
- }
- return true;
- }
- /**
- * Plugin version data
- *
- * @param array &$versions array of version blocks
- *
- * @return boolean hook value
- */
- public function onPluginVersion(array &$versions): bool
- {
- $versions[] = array(
- 'name' => 'TwitterBridge',
- 'version' => self::PLUGIN_VERSION,
- 'author' => 'Zach Copley, Julien C, Jean Baptiste Favre',
- 'homepage' => GNUSOCIAL_ENGINE_REPO_URL . 'tree/master/plugins/TwitterBridge',
- // TRANS: Plugin description.
- 'rawdescription' => _m(
- 'The Twitter "bridge" plugin allows integration ' .
- 'of a StatusNet instance with ' .
- '<a href="http://twitter.com/">Twitter</a>.'
- )
- );
- return true;
- }
- /**
- * Expose the adminImportControl setting to the administration panel code.
- * This allows us to disable the import bridge enabling checkbox for administrators,
- * since on a bulk farm site we can't yet automate the import daemon setup.
- *
- * @return boolean hook value;
- */
- public function onTwitterBridgeAdminImportControl()
- {
- return (bool)$this->adminImportControl;
- }
- /**
- * Database schema setup
- *
- * We maintain a table mapping StatusNet notices to Twitter statuses
- *
- * @see Schema
- * @see ColumnDef
- *
- * @return boolean hook value; true means continue processing, false means stop.
- */
- public function onCheckSchema()
- {
- $schema = Schema::get();
- // For saving the last-synched status of various timelines
- // home_timeline, messages (in), messages (out), ...
- $schema->ensureTable('twitter_synch_status', Twitter_synch_status::schemaDef());
- // For storing user-submitted flags on profiles
- $schema->ensureTable('notice_to_status', Notice_to_status::schemaDef());
- return true;
- }
- /**
- * If a notice gets deleted, remove the Notice_to_status mapping and
- * delete the status on Twitter.
- *
- * @param User $user The user doing the deleting
- * @param Notice $notice The notice getting deleted
- *
- * @return boolean hook value
- */
- public function onStartDeleteOwnNotice(User $user, Notice $notice)
- {
- $n2s = Notice_to_status::getKV('notice_id', $notice->id);
- if ($n2s instanceof Notice_to_status) {
- try {
- $flink = Foreign_link::getByUserID($notice->profile_id, TWITTER_SERVICE); // twitter service
- } catch (NoResultException $e) {
- return true;
- }
- if (!TwitterOAuthClient::isPackedToken($flink->credentials)) {
- $this->log(LOG_INFO, "Skipping deleting notice for {$notice->id} since link is not OAuth.");
- return true;
- }
- try {
- $token = TwitterOAuthClient::unpackToken($flink->credentials);
- $client = new TwitterOAuthClient($token->key, $token->secret);
- $client->statusesDestroy($n2s->status_id);
- } catch (Exception $e) {
- common_log(LOG_ERR, "Error attempting to delete bridged notice from Twitter: " . $e->getMessage());
- }
- $n2s->delete();
- }
- 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
- */
- public function onEndFavorNotice(Profile $profile, Notice $notice)
- {
- try {
- $flink = Foreign_link::getByUserID($profile->getID(), TWITTER_SERVICE); // twitter service
- } catch (NoResultException $e) {
- return true;
- }
- if (!TwitterOAuthClient::isPackedToken($flink->credentials)) {
- $this->log(LOG_INFO, "Skipping fave processing for {$profile->getID()} since link is not OAuth.");
- return true;
- }
- $status_id = twitter_status_id($notice);
- if (empty($status_id)) {
- return true;
- }
- try {
- $token = TwitterOAuthClient::unpackToken($flink->credentials);
- $client = new TwitterOAuthClient($token->key, $token->secret);
- $client->favoritesCreate($status_id);
- } catch (Exception $e) {
- common_log(LOG_ERR, "Error attempting to favorite bridged notice on Twitter: " . $e->getMessage());
- }
- 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
- */
- public function onEndDisfavorNotice(Profile $profile, Notice $notice)
- {
- try {
- $flink = Foreign_link::getByUserID($profile->getID(), TWITTER_SERVICE); // twitter service
- } catch (NoResultException $e) {
- return true;
- }
- if (!TwitterOAuthClient::isPackedToken($flink->credentials)) {
- $this->log(LOG_INFO, "Skipping fave processing for {$profile->id} since link is not OAuth.");
- return true;
- }
- $status_id = twitter_status_id($notice);
- if (empty($status_id)) {
- return true;
- }
- try {
- $token = TwitterOAuthClient::unpackToken($flink->credentials);
- $client = new TwitterOAuthClient($token->key, $token->secret);
- $client->favoritesDestroy($status_id);
- } catch (Exception $e) {
- common_log(LOG_ERR, "Error attempting to unfavorite bridged notice on Twitter: " . $e->getMessage());
- }
- return true;
- }
- public function onStartGetProfileUri($profile, &$uri)
- {
- if (preg_match('!^https?://twitter.com/!', $profile->profileurl)) {
- $uri = $profile->profileurl;
- return false;
- }
- return true;
- }
- /**
- * Add links in the user's profile block to their Twitter profile URL.
- *
- * @param Profile $profile The profile being shown
- * @param Array &$links Writeable array of arrays (href, text, image).
- *
- * @return boolean hook value (true)
- */
- public function onOtherAccountProfiles($profile, &$links)
- {
- $fuser = null;
- try {
- $flink = Foreign_link::getByUserID($profile->id, TWITTER_SERVICE);
- $fuser = $flink->getForeignUser();
- $links[] = array("href" => $fuser->uri,
- "text" => sprintf(_("@%s on Twitter"), $fuser->nickname),
- "image" => $this->path("icons/twitter-bird-white-on-blue.png"));
- } catch (NoResultException $e) {
- // no foreign link and/or user for Twitter on this profile ID
- }
- return true;
- }
- public function onEndShowHeadElements(Action $action)
- {
- // Showing a notice
- if ($action instanceof ShowNoticeAction) {
- $notice = Notice::getKV('id', $action->arg('notice'));
- if (is_null($notice)) {
- return true;
- }
- try {
- $flink = Foreign_link::getByUserID($notice->profile_id, TWITTER_SERVICE);
- $fuser = Foreign_user::getForeignUser($flink->foreign_id, TWITTER_SERVICE);
- } catch (NoResultException $e) {
- return true;
- }
- $statusId = twitter_status_id($notice);
- if ($notice instanceof Notice && $notice->isLocal() && $statusId) {
- $tweetUrl = 'https://twitter.com/' . $fuser->nickname . '/status/' . $statusId;
- $action->element('link', array('rel' => 'syndication', 'href' => $tweetUrl));
- }
- }
- if (!($action instanceof AttachmentAction)) {
- return true;
- }
- /* Twitter card support. See https://dev.twitter.com/docs/cards */
- /* @fixme: should we display twitter cards only for attachments posted
- * by local users ? Seems mandatory to display twitter:creator
- *
- * Author: jbfavre
- */
- switch ($action->attachment->mimetype) {
- case 'image/pjpeg':
- case 'image/jpeg':
- case 'image/jpg':
- case 'image/png':
- case 'image/gif':
- $action->element(
- 'meta',
- [
- 'name' => 'twitter:card',
- 'content' => 'photo',
- ],
- null
- );
- $action->element(
- 'meta',
- [
- 'name' => 'twitter:url',
- 'content' => common_local_url(
- 'attachment',
- ['attachment' => $action->attachment->id]
- )
- ],
- null
- );
- $action->element(
- 'meta',
- [
- 'name' => 'twitter:image',
- 'content' => $action->attachment->url,
- ]
- );
- $action->element(
- 'meta',
- [
- 'name' => 'twitter:title',
- 'content' => $action->attachment->title,
- ]
- );
- $ns = new AttachmentNoticeSection($action);
- $notices = $ns->getNotices();
- $noticeArray = $notices->fetchAll();
- // Should not have more than 1 notice for this attachment.
- if (count($noticeArray) != 1) {
- break;
- }
- $post = $noticeArray[0];
- try {
- $flink = Foreign_link::getByUserID($post->profile_id, TWITTER_SERVICE);
- $fuser = Foreign_user::getForeignUser($flink->foreign_id, TWITTER_SERVICE);
- $action->element('meta', array('name' => 'twitter:creator',
- 'content' => '@'.$fuser->nickname));
- } catch (NoResultException $e) {
- // no foreign link and/or user for Twitter on this profile ID
- }
- break;
- default:
- break;
- }
- return true;
- }
-
- /**
- * Set the object_type field of previously imported Twitter notices to
- * ActivityObject::NOTE if they are unset. Null object_type caused a notice
- * not to show on the timeline.
- */
- public function onEndUpgrade()
- {
- printfnq('Ensuring all Twitter notices have an object_type...');
- $notice = new Notice();
- $notice->whereAdd("source = 'twitter'");
- $notice->whereAdd('object_type IS NULL');
- if ($notice->find()) {
- while ($notice->fetch()) {
- $orig = Notice::getKV('id', $notice->id);
- $notice->object_type = ActivityObject::NOTE;
- $notice->update($orig);
- }
- }
- printfnq("DONE.\n");
- }
- }
|