123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637 |
- <?php
- /**
- * StatusNet, the distributed open-source microblogging tool
- *
- * Superclass for plugins that do instant messaging
- *
- * PHP version 5
- *
- * LICENCE: 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 Craig Andrews <candrews@integralblue.com>
- * @license http://www.fsf.org/licensing/licenses/agpl-3.0.html GNU Affero General Public License version 3.0
- * @link http://status.net/
- */
- if (!defined('STATUSNET') && !defined('LACONICA')) {
- exit(1);
- }
- /**
- * Superclass for plugins that do IM transport
- *
- * Implementations will likely want to override onStartIoManagerClasses() so that their
- * IO manager is used
- *
- * @category Module
- * @package GNUsocial
- * @author Craig Andrews <candrews@integralblue.com>
- * @license http://www.fsf.org/licensing/licenses/agpl-3.0.html GNU Affero General Public License version 3.0
- * @link http://status.net/
- */
- abstract class ImPlugin extends Plugin
- {
- public $widgetOpts;
- public $scoped;
- //name of this IM transport
- public $transport = null;
- //list of screennames that should get all public notices
- public $public = array();
- protected $requires_cli = true;
- /**
- * normalize a screenname for comparison
- *
- * @param string $screenname screenname to normalize
- *
- * @return string an equivalent screenname in normalized form
- */
- abstract function normalize($screenname);
- /**
- * validate (ensure the validity of) a screenname
- *
- * @param string $screenname screenname to validate
- *
- * @return boolean
- */
- abstract function validate($screenname);
- /**
- * get the internationalized/translated display name of this IM service
- *
- * @return string
- */
- abstract function getDisplayName();
- /**
- * send a single notice to a given screenname
- * The implementation should put raw data, ready to send, into the outgoing
- * queue using enqueueOutgoingRaw()
- *
- * @param string $screenname screenname to send to
- * @param Notice $notice notice to send
- *
- * @return boolean success value
- */
- function sendNotice($screenname, Notice $notice)
- {
- return $this->sendMessage($screenname, $this->formatNotice($notice));
- }
- /**
- * send a message (text) to a given screenname
- * The implementation should put raw data, ready to send, into the outgoing
- * queue using enqueueOutgoingRaw()
- *
- * @param string $screenname screenname to send to
- * @param Notice $body text to send
- *
- * @return boolean success value
- */
- abstract function sendMessage($screenname, $body);
- /**
- * receive a raw message
- * Raw IM data is taken from the incoming queue, and passed to this function.
- * It should parse the raw message and call handleIncoming()
- *
- * Returning false may CAUSE REPROCESSING OF THE QUEUE ITEM, and should
- * be used for temporary failures only. For permanent failures such as
- * unrecognized addresses, return true to indicate your processing has
- * completed.
- *
- * @param object $data raw IM data
- *
- * @return boolean true if processing completed, false for temporary failures
- */
- abstract function receiveRawMessage($data);
- /**
- * get the screenname of the daemon that sends and receives message for this service
- *
- * @return string screenname of this plugin
- */
- abstract function daemonScreenname();
- //========================UTILITY FUNCTIONS USEFUL TO IMPLEMENTATIONS - MISC ========================\
- /**
- * Put raw message data (ready to send) into the outgoing queue
- *
- * @param object $data
- */
- function enqueueOutgoingRaw($data)
- {
- $qm = QueueManager::get();
- $qm->enqueue($data, $this->transport . '-out');
- }
- /**
- * Put raw message data (received, ready to be processed) into the incoming queue
- *
- * @param object $data
- */
- function enqueueIncomingRaw($data)
- {
- $qm = QueueManager::get();
- $qm->enqueue($data, $this->transport . '-in');
- }
- /**
- * given a screenname, get the corresponding user
- *
- * @param string $screenname
- *
- * @return User user
- */
- function getUser($screenname)
- {
- $user_im_prefs = $this->getUserImPrefsFromScreenname($screenname);
- if($user_im_prefs){
- $user = User::getKV('id', $user_im_prefs->user_id);
- $user_im_prefs->free();
- return $user;
- }else{
- return false;
- }
- }
- /**
- * given a screenname, get the User_im_prefs object for this transport
- *
- * @param string $screenname
- *
- * @return User_im_prefs user_im_prefs
- */
- function getUserImPrefsFromScreenname($screenname)
- {
- $user_im_prefs = User_im_prefs::pkeyGet(
- array('transport' => $this->transport,
- 'screenname' => $this->normalize($screenname)));
- if ($user_im_prefs) {
- return $user_im_prefs;
- } else {
- return false;
- }
- }
- /**
- * given a User, get their screenname
- *
- * @param User $user
- *
- * @return string screenname of that user
- */
- function getScreenname($user)
- {
- $user_im_prefs = $this->getUserImPrefsFromUser($user);
- if ($user_im_prefs) {
- return $user_im_prefs->screenname;
- } else {
- return false;
- }
- }
- /**
- * given a User, get their User_im_prefs
- *
- * @param User $user
- *
- * @return User_im_prefs user_im_prefs of that user
- */
- function getUserImPrefsFromUser($user)
- {
- $user_im_prefs = User_im_prefs::pkeyGet(
- array('transport' => $this->transport,
- 'user_id' => $user->id));
- if ($user_im_prefs){
- return $user_im_prefs;
- } else {
- return false;
- }
- }
- //========================UTILITY FUNCTIONS USEFUL TO IMPLEMENTATIONS - SENDING ========================\
- /**
- * Send a message to a given screenname from the site
- *
- * @param string $screenname screenname to send the message to
- * @param string $msg message contents to send
- *
- * @param boolean success
- */
- protected function sendFromSite($screenname, $msg)
- {
- $text = '['.common_config('site', 'name') . '] ' . $msg;
- $this->sendMessage($screenname, $text);
- }
- /**
- * Send a confirmation code to a user
- *
- * @param string $screenname screenname sending to
- * @param string $code the confirmation code
- * @param Profile $target For whom the code is valid for
- *
- * @return boolean success value
- */
- function sendConfirmationCode($screenname, $code, Profile $target)
- {
- // TRANS: Body text for confirmation code e-mail.
- // TRANS: %1$s is a user nickname, %2$s is the StatusNet sitename,
- // TRANS: %3$s is the display name of an IM plugin.
- $body = sprintf(_('User "%1$s" on %2$s has said that your %3$s screenname belongs to them. ' .
- 'If that is true, you can confirm by clicking on this URL: ' .
- '%4$s' .
- ' . (If you cannot click it, copy-and-paste it into the ' .
- 'address bar of your browser). If that user is not you, ' .
- 'or if you did not request this confirmation, just ignore this message.'),
- $target->getNickname(), common_config('site', 'name'), $this->getDisplayName(), common_local_url('confirmaddress', null, array('code' => $code)));
- return $this->sendMessage($screenname, $body);
- }
- /**
- * send a notice to all public listeners
- *
- * For notices that are generated on the local system (by users), we can optionally
- * forward them to remote listeners by XMPP.
- *
- * @param Notice $notice notice to broadcast
- *
- * @return boolean success flag
- */
- function publicNotice($notice)
- {
- // Now, users who want everything
- // FIXME PRIV don't send out private messages here
- // XXX: should we send out non-local messages if public,localonly
- // = false? I think not
- foreach ($this->public as $screenname) {
- common_log(LOG_INFO,
- 'Sending notice ' . $notice->id .
- ' to public listener ' . $screenname,
- __FILE__);
- $this->sendNotice($screenname, $notice);
- }
- return true;
- }
- /**
- * broadcast a notice to all subscribers and reply recipients
- *
- * This function will send a notice to all subscribers on the local server
- * who have IM addresses, and have IM notification enabled, and
- * have this subscription enabled for IM. It also sends the notice to
- * all recipients of @-replies who have IM addresses and IM notification
- * enabled. This is really the heart of IM distribution in StatusNet.
- *
- * @param Notice $notice The notice to broadcast
- *
- * @return boolean success flag
- */
- function broadcastNotice($notice)
- {
- $ni = $notice->whoGets();
- foreach ($ni as $user_id => $reason) {
- $user = User::getKV($user_id);
- if (empty($user)) {
- // either not a local user, or just not found
- continue;
- }
- $user_im_prefs = $this->getUserImPrefsFromUser($user);
- if(!$user_im_prefs || !$user_im_prefs->notify){
- continue;
- }
- switch ($reason) {
- case NOTICE_INBOX_SOURCE_REPLY:
- if (!$user_im_prefs->replies) {
- continue 2;
- }
- break;
- case NOTICE_INBOX_SOURCE_SUB:
- $sub = Subscription::pkeyGet(array('subscriber' => $user->id,
- 'subscribed' => $notice->profile_id));
- if (empty($sub) || !$sub->jabber) {
- continue 2;
- }
- break;
- case NOTICE_INBOX_SOURCE_GROUP:
- break;
- default:
- // TRANS: Exception thrown when trying to deliver a notice to an unknown inbox.
- // TRANS: %d is the unknown inbox ID (number).
- throw new Exception(sprintf(_('Unknown inbox source %d.'), $reason));
- }
- common_log(LOG_INFO,
- 'Sending notice ' . $notice->id . ' to ' . $user_im_prefs->screenname,
- __FILE__);
- $this->sendNotice($user_im_prefs->screenname, $notice);
- $user_im_prefs->free();
- }
- return true;
- }
- /**
- * makes a plain-text formatted version of a notice, suitable for IM distribution
- *
- * @param Notice $notice notice being sent
- *
- * @return string plain-text version of the notice, with user nickname prefixed
- */
- protected function formatNotice(Notice $notice)
- {
- $profile = $notice->getProfile();
- $nicknames = $profile->getNickname();
- try {
- $parent = $notice->getParent();
- $orig_profile = $parent->getProfile();
- $nicknames = sprintf('%1$s => %2$s', $profile->getNickname(), $orig_profile->getNickname());
- } catch (NoParentNoticeException $e) {
- // Not a reply, no parent notice stored
- } catch (NoResultException $e) {
- // Parent notice was probably deleted
- }
- return sprintf('%1$s: %2$s [%3$u]', $nicknames, $notice->content, $notice->id);
- }
- //========================UTILITY FUNCTIONS USEFUL TO IMPLEMENTATIONS - RECEIVING ========================\
- /**
- * Attempt to handle a message as a command
- * @param User $user user the message is from
- * @param string $body message text
- * @return boolean true if the message was a command and was executed, false if it was not a command
- */
- protected function handleCommand($user, $body)
- {
- $inter = new CommandInterpreter();
- $cmd = $inter->handle_command($user, $body);
- if ($cmd) {
- $chan = new IMChannel($this);
- $cmd->execute($chan);
- return true;
- }
- return false;
- }
- /**
- * Is some text an autoreply message?
- * @param string $txt message text
- * @return boolean true if autoreply
- */
- protected function isAutoreply($txt)
- {
- if (preg_match('/[\[\(]?[Aa]uto[-\s]?[Rr]e(ply|sponse)[\]\)]/', $txt)) {
- return true;
- } else if (preg_match('/^System: Message wasn\'t delivered. Offline storage size was exceeded.$/', $txt)) {
- return true;
- } else {
- return false;
- }
- }
- /**
- * Is some text an OTR message?
- * @param string $txt message text
- * @return boolean true if OTR
- */
- protected function isOtr($txt)
- {
- if (preg_match('/^\?OTR/', $txt)) {
- return true;
- } else {
- return false;
- }
- }
- /**
- * Helper for handling incoming messages
- * Your incoming message handler will probably want to call this function
- *
- * @param string $from screenname the message was sent from
- * @param string $message message contents
- *
- * @param boolean success
- */
- protected function handleIncoming($from, $notice_text)
- {
- $user = $this->getUser($from);
- // For common_current_user to work
- global $_cur;
- $_cur = $user;
- if (!$user) {
- $this->sendFromSite($from, 'Unknown user; go to ' .
- common_local_url('imsettings') .
- ' to add your address to your account');
- common_log(LOG_WARNING, 'Message from unknown user ' . $from);
- return;
- }
- if ($this->handleCommand($user, $notice_text)) {
- common_log(LOG_INFO, "Command message by $from handled.");
- return;
- } else if ($this->isAutoreply($notice_text)) {
- common_log(LOG_INFO, 'Ignoring auto reply from ' . $from);
- return;
- } else if ($this->isOtr($notice_text)) {
- common_log(LOG_INFO, 'Ignoring OTR from ' . $from);
- return;
- } else {
- common_log(LOG_INFO, 'Posting a notice from ' . $user->nickname);
- $this->addNotice($from, $user, $notice_text);
- }
- $user->free();
- unset($user);
- unset($_cur);
- unset($message);
- }
- /**
- * Helper for handling incoming messages
- * Your incoming message handler will probably want to call this function
- *
- * @param string $from screenname the message was sent from
- * @param string $message message contents
- *
- * @param boolean success
- */
- protected function addNotice($screenname, $user, $body)
- {
- $body = trim(strip_tags($body));
- $content_shortened = common_shorten_links($body);
- if (Notice::contentTooLong($content_shortened)) {
- $this->sendFromSite($screenname,
- // TRANS: Message given when a status is too long. %1$s is the maximum number of characters,
- // TRANS: %2$s is the number of characters sent (used for plural).
- sprintf(_m('Message too long - maximum is %1$d character, you sent %2$d.',
- 'Message too long - maximum is %1$d characters, you sent %2$d.',
- Notice::maxContent()),
- Notice::maxContent(),
- mb_strlen($content_shortened)));
- return;
- }
- try {
- $notice = Notice::saveNew($user->id, $content_shortened, $this->transport);
- } catch (Exception $e) {
- common_log(LOG_ERR, $e->getMessage());
- $this->sendFromSite($from, $e->getMessage());
- return;
- }
- common_log(LOG_INFO,
- 'Added notice ' . $notice->id . ' from user ' . $user->nickname);
- $notice->free();
- unset($notice);
- }
- //========================EVENT HANDLERS========================\
- /**
- * Register notice queue handler
- *
- * @param QueueManager $manager
- *
- * @return boolean hook return
- */
- function onEndInitializeQueueManager($manager)
- {
- // If we don't require CLI mode, or if we do and GNUSOCIAL_CLI _is_ set, then connect the transports
- // This check is made mostly because some IM plugins can't deliver to transports unless they
- // have continuously running daemons (such as XMPP) and we can't have that over HTTP requests.
- if (!$this->requires_cli || defined('GNUSOCIAL_CLI')) {
- $manager->connect($this->transport . '-in', new ImReceiverQueueHandler($this), 'im');
- $manager->connect($this->transport, new ImQueueHandler($this));
- $manager->connect($this->transport . '-out', new ImSenderQueueHandler($this), 'im');
- }
- return true;
- }
- function onStartImDaemonIoManagers(&$classes)
- {
- //$classes[] = new ImManager($this); // handles sending/receiving/pings/reconnects
- return true;
- }
- function onStartEnqueueNotice($notice, &$transports)
- {
- $profile = Profile::getKV($notice->profile_id);
- if (!$profile) {
- common_log(LOG_WARNING, 'Refusing to broadcast notice with ' .
- 'unknown profile ' . common_log_objstring($notice),
- __FILE__);
- }else{
- $transports[] = $this->transport;
- }
- return true;
- }
- function onEndShowHeadElements(Action $action)
- {
- if ($action instanceof ShownoticeAction) {
- $user_im_prefs = new User_im_prefs();
- $user_im_prefs->user_id = $action->notice->getProfile()->getID();
- $user_im_prefs->transport = $this->transport;
- } elseif ($action instanceof ShowstreamAction) {
- $user_im_prefs = new User_im_prefs();
- $user_im_prefs->user_id = $action->getTarget()->getID();
- $user_im_prefs->transport = $this->transport;
- }
- }
- function onNormalizeImScreenname($transport, &$screenname)
- {
- if($transport == $this->transport)
- {
- $screenname = $this->normalize($screenname);
- return false;
- }
- }
- function onValidateImScreenname($transport, $screenname, &$valid)
- {
- if($transport == $this->transport)
- {
- $valid = $this->validate($screenname);
- return false;
- }
- }
- function onGetImTransports(&$transports)
- {
- $transports[$this->transport] = array(
- 'display' => $this->getDisplayName(),
- 'daemonScreenname' => $this->daemonScreenname());
- }
- function onSendImConfirmationCode($transport, $screenname, $code, Profile $target)
- {
- if($transport == $this->transport)
- {
- $this->sendConfirmationCode($screenname, $code, $target);
- return false;
- }
- }
- function onUserDeleteRelated($user, &$tables)
- {
- $tables[] = 'User_im_prefs';
- return true;
- }
- function onHaveImPlugin(&$haveImPlugin) {
- $haveImPlugin = true; // set flag true (we're loaded, after all!)
- return false; // stop looking
- }
- function initialize()
- {
- if( ! common_config('queue', 'enabled'))
- {
- // TRANS: Server exception thrown trying to initialise an IM plugin without meeting all prerequisites.
- throw new ServerException(_('Queueing must be enabled to use IM plugins.'));
- }
- if(is_null($this->transport)){
- // TRANS: Server exception thrown trying to initialise an IM plugin without a transport method.
- throw new ServerException(_('Transport cannot be null.'));
- }
- }
- }
|