move activity importing code to two different queuehandler classes
This commit is contained in:
		
							
								
								
									
										317
									
								
								lib/activityimporter.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										317
									
								
								lib/activityimporter.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,317 @@ | ||||
| <?php | ||||
| /** | ||||
|  * StatusNet - the distributed open-source microblogging tool | ||||
|  * Copyright (C) 2010, StatusNet, Inc. | ||||
|  * | ||||
|  * class to import activities as part of a user's timeline | ||||
|  *  | ||||
|  * 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  Cache | ||||
|  * @package   StatusNet | ||||
|  * @author    Evan Prodromou <evan@status.net> | ||||
|  * @copyright 2010 StatusNet, Inc. | ||||
|  * @license   http://www.fsf.org/licensing/licenses/agpl-3.0.html AGPL 3.0 | ||||
|  * @link      http://status.net/ | ||||
|  */ | ||||
|  | ||||
| if (!defined('STATUSNET')) { | ||||
|     // This check helps protect against security problems; | ||||
|     // your code file can't be executed directly from the web. | ||||
|     exit(1); | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Class comment | ||||
|  * | ||||
|  * @category  General | ||||
|  * @package   StatusNet | ||||
|  * @author    Evan Prodromou <evan@status.net> | ||||
|  * @copyright 2010 StatusNet, Inc. | ||||
|  * @license   http://www.fsf.org/licensing/licenses/agpl-3.0.html AGPL 3.0 | ||||
|  * @link      http://status.net/ | ||||
|  */ | ||||
|  | ||||
| class ActivityImporter extends QueueHandler | ||||
| { | ||||
|     private $trusted = false; | ||||
|  | ||||
|     /** | ||||
|      * Function comment | ||||
|      * | ||||
|      * @param | ||||
|      * | ||||
|      * @return | ||||
|      */ | ||||
|  | ||||
|     function handle($data) | ||||
|     { | ||||
|         list($user, $author, $activity, $trusted) = $data; | ||||
|  | ||||
|         $this->trusted = $trusted; | ||||
|  | ||||
|         try { | ||||
|             switch ($activity->verb) { | ||||
|             case ActivityVerb::FOLLOW: | ||||
|                 $this->subscribeProfile($user, $author, $activity); | ||||
|                 break; | ||||
|             case ActivityVerb::JOIN: | ||||
|                 $this->joinGroup($user, $activity); | ||||
|                 break; | ||||
|             case ActivityVerb::POST: | ||||
|                 $this->postNote($user, $activity); | ||||
|                 break; | ||||
|             default: | ||||
|                 throw new Exception("Unknown verb: {$activity->verb}"); | ||||
|             } | ||||
|         } catch (ClientException $ce) { | ||||
|             common_log(LOG_WARNING, $ce->getMessage()); | ||||
|             return true; | ||||
|         } catch (ServerException $se) { | ||||
|             common_log(LOG_ERR, $ce->getMessage()); | ||||
|             return false; | ||||
|         } catch (Exception $e) { | ||||
|             common_log(LOG_ERR, $ce->getMessage()); | ||||
|             return false; | ||||
|         } | ||||
|         return true; | ||||
|     } | ||||
|      | ||||
|     function subscribeProfile($user, $author, $activity) | ||||
|     { | ||||
|         $profile = $user->getProfile(); | ||||
|  | ||||
|         if ($activity->objects[0]->id == $author->id) { | ||||
|             $other = $activity->actor; | ||||
|             $otherUser = User::staticGet('uri', $other->id); | ||||
|              | ||||
|             if (!empty($otherUser)) { | ||||
|                 $otherProfile = $otherUser->getProfile(); | ||||
|             } else { | ||||
|                 throw new Exception("Can't force remote user to subscribe."); | ||||
|             } | ||||
|  | ||||
|             // XXX: don't do this for untrusted input! | ||||
|  | ||||
|             Subscription::start($otherProfile, $profile); | ||||
|  | ||||
|         } else if (empty($activity->actor)  | ||||
|                    || $activity->actor->id == $author->id) { | ||||
|  | ||||
|             $other = $activity->objects[0]; | ||||
|  | ||||
|             $otherProfile = Profile::fromUri($other->id); | ||||
|  | ||||
|             if (empty($otherProfile)) { | ||||
|                 throw new ClientException(_("Unknown profile.")); | ||||
|             } | ||||
|  | ||||
|             Subscription::start($profile, $otherProfile); | ||||
|         } else { | ||||
|             throw new Exception("This activity seems unrelated to our user."); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     function joinGroup($user, $activity) | ||||
|     { | ||||
|         // XXX: check that actor == subject | ||||
|  | ||||
|         $uri = $activity->objects[0]->id; | ||||
|  | ||||
|         $group = User_group::staticGet('uri', $uri); | ||||
|  | ||||
|         if (empty($group)) { | ||||
|             $oprofile = Ostatus_profile::ensureActivityObjectProfile($activity->objects[0]); | ||||
|             if (!$oprofile->isGroup()) { | ||||
|                 throw new Exception("Remote profile is not a group!"); | ||||
|             } | ||||
|             $group = $oprofile->localGroup(); | ||||
|         } | ||||
|  | ||||
|         assert(!empty($group)); | ||||
|  | ||||
|         if (Event::handle('StartJoinGroup', array($group, $user))) { | ||||
|             Group_member::join($group->id, $user->id); | ||||
|             Event::handle('EndJoinGroup', array($group, $user)); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     // XXX: largely cadged from Ostatus_profile::processNote() | ||||
|  | ||||
|     function postNote($user, $activity) | ||||
|     { | ||||
|         $note = $activity->objects[0]; | ||||
|  | ||||
|         $sourceUri = $note->id; | ||||
|  | ||||
|         $notice = Notice::staticGet('uri', $sourceUri); | ||||
|  | ||||
|         if (!empty($notice)) { | ||||
|             // This is weird. | ||||
|             $orig = clone($notice); | ||||
|             $notice->profile_id = $user->id; | ||||
|             $notice->update($orig); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         // Use summary as fallback for content | ||||
|  | ||||
|         if (!empty($note->content)) { | ||||
|             $sourceContent = $note->content; | ||||
|         } else if (!empty($note->summary)) { | ||||
|             $sourceContent = $note->summary; | ||||
|         } else if (!empty($note->title)) { | ||||
|             $sourceContent = $note->title; | ||||
|         } else { | ||||
|             // @fixme fetch from $sourceUrl? | ||||
|             // @todo i18n FIXME: use sprintf and add i18n. | ||||
|             throw new ClientException("No content for notice {$sourceUri}."); | ||||
|         } | ||||
|  | ||||
|         // Get (safe!) HTML and text versions of the content | ||||
|  | ||||
|         $rendered = $this->purify($sourceContent); | ||||
|         $content = html_entity_decode(strip_tags($rendered), ENT_QUOTES, 'UTF-8'); | ||||
|  | ||||
|         $shortened = $user->shortenLinks($content); | ||||
|  | ||||
|         $options = array('is_local' => Notice::LOCAL_PUBLIC, | ||||
|                          'uri' => $sourceUri, | ||||
|                          'rendered' => $rendered, | ||||
|                          'replies' => array(), | ||||
|                          'groups' => array(), | ||||
|                          'tags' => array(), | ||||
|                          'urls' => array()); | ||||
|  | ||||
|         // Check for optional attributes... | ||||
|  | ||||
|         if (!empty($activity->time)) { | ||||
|             $options['created'] = common_sql_date($activity->time); | ||||
|         } | ||||
|  | ||||
|         if ($activity->context) { | ||||
|             // Any individual or group attn: targets? | ||||
|  | ||||
|             list($options['groups'], $options['replies']) = $this->filterAttention($activity->context->attention); | ||||
|  | ||||
|             // Maintain direct reply associations | ||||
|             // @fixme what about conversation ID? | ||||
|             if (!empty($activity->context->replyToID)) { | ||||
|                 $orig = Notice::staticGet('uri', | ||||
|                                           $activity->context->replyToID); | ||||
|                 if (!empty($orig)) { | ||||
|                     $options['reply_to'] = $orig->id; | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             $location = $activity->context->location; | ||||
|  | ||||
|             if ($location) { | ||||
|                 $options['lat'] = $location->lat; | ||||
|                 $options['lon'] = $location->lon; | ||||
|                 if ($location->location_id) { | ||||
|                     $options['location_ns'] = $location->location_ns; | ||||
|                     $options['location_id'] = $location->location_id; | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         // Atom categories <-> hashtags | ||||
|  | ||||
|         foreach ($activity->categories as $cat) { | ||||
|             if ($cat->term) { | ||||
|                 $term = common_canonical_tag($cat->term); | ||||
|                 if ($term) { | ||||
|                     $options['tags'][] = $term; | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         // Atom enclosures -> attachment URLs | ||||
|         foreach ($activity->enclosures as $href) { | ||||
|             // @fixme save these locally or....? | ||||
|             $options['urls'][] = $href; | ||||
|         } | ||||
|  | ||||
|         $saved = Notice::saveNew($user->id, | ||||
|                                  $content, | ||||
|                                  'restore', // TODO: restore the actual source | ||||
|                                  $options); | ||||
|  | ||||
|         return $saved; | ||||
|     } | ||||
|  | ||||
|     function filterAttention($attn) | ||||
|     { | ||||
|         $groups = array(); | ||||
|         $replies = array(); | ||||
|  | ||||
|         foreach (array_unique($attn) as $recipient) { | ||||
|  | ||||
|             // Is the recipient a local user? | ||||
|  | ||||
|             $user = User::staticGet('uri', $recipient); | ||||
|  | ||||
|             if ($user) { | ||||
|                 // @fixme sender verification, spam etc? | ||||
|                 $replies[] = $recipient; | ||||
|                 continue; | ||||
|             } | ||||
|  | ||||
|             // Is the recipient a remote group? | ||||
|             $oprofile = Ostatus_profile::ensureProfileURI($recipient); | ||||
|  | ||||
|             if ($oprofile) { | ||||
|                 if (!$oprofile->isGroup()) { | ||||
|                     // may be canonicalized or something | ||||
|                     $replies[] = $oprofile->uri; | ||||
|                 } | ||||
|                 continue; | ||||
|             } | ||||
|  | ||||
|             // Is the recipient a local group? | ||||
|             // @fixme uri on user_group isn't reliable yet | ||||
|             // $group = User_group::staticGet('uri', $recipient); | ||||
|             $id = OStatusPlugin::localGroupFromUrl($recipient); | ||||
|  | ||||
|             if ($id) { | ||||
|                 $group = User_group::staticGet('id', $id); | ||||
|                 if ($group) { | ||||
|                     // Deliver to all members of this local group if allowed. | ||||
|                     $profile = $sender->localProfile(); | ||||
|                     if ($profile->isMember($group)) { | ||||
|                         $groups[] = $group->id; | ||||
|                     } else { | ||||
|                         common_log(LOG_INFO, "Skipping reply to local group {$group->nickname} as sender {$profile->id} is not a member"); | ||||
|                     } | ||||
|                     continue; | ||||
|                 } else { | ||||
|                     common_log(LOG_INFO, "Skipping reply to bogus group $recipient"); | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         return array($groups, $replies); | ||||
|     } | ||||
|   | ||||
|  | ||||
|     function purify($content) | ||||
|     { | ||||
|         $config = array('safe' => 1, | ||||
|                         'deny_attribute' => 'id,style,on*'); | ||||
|         return htmLawed($content, $config); | ||||
|     } | ||||
| } | ||||
| @@ -3,7 +3,7 @@ | ||||
|  * StatusNet - the distributed open-source microblogging tool | ||||
|  * Copyright (C) 2010, StatusNet, Inc. | ||||
|  * | ||||
|  * A class for restoring accounts | ||||
|  * Importer for feeds of activities | ||||
|  *  | ||||
|  * PHP version 5 | ||||
|  * | ||||
| @@ -35,12 +35,10 @@ if (!defined('STATUSNET')) { | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * A class for restoring accounts | ||||
|  * Importer for feeds of activities | ||||
|  * | ||||
|  * This is a clumsy objectification of the functions in restoreuser.php. | ||||
|  *  | ||||
|  * Note that it quite illegally uses the OStatus_profile class which may | ||||
|  * not even exist on this server. | ||||
|  * Takes an XML file representing a feed of activities and imports each | ||||
|  * activity to the user in question. | ||||
|  * | ||||
|  * @category  Account | ||||
|  * @package   StatusNet | ||||
| @@ -50,309 +48,82 @@ if (!defined('STATUSNET')) { | ||||
|  * @link      http://status.net/ | ||||
|  */ | ||||
|  | ||||
| class AccountRestorer | ||||
| class FeedImporter extends QueueHandler | ||||
| { | ||||
|     private $_trusted = false; | ||||
|     /** | ||||
|      * Transport identifier | ||||
|      * | ||||
|      * @return string identifier for this queue handler | ||||
|      */ | ||||
|  | ||||
|     function loadXML($xml) | ||||
|     public function transport() | ||||
|     { | ||||
|         $dom = DOMDocument::loadXML($xml); | ||||
|  | ||||
|         if ($dom->documentElement->namespaceURI != Activity::ATOM || | ||||
|             $dom->documentElement->localName != 'feed') { | ||||
|             throw new Exception("'$filename' is not an Atom feed."); | ||||
|         } | ||||
|  | ||||
|         return $dom; | ||||
|         return 'feedimp'; | ||||
|     } | ||||
|  | ||||
|     function importActivityStream($user, $doc) | ||||
|     function handle($data) | ||||
|     { | ||||
|         $feed = $doc->documentElement; | ||||
|         list($user, $xml, $trusted) = $data; | ||||
|  | ||||
|         $subjectEl = ActivityUtils::child($feed, Activity::SUBJECT, Activity::SPEC); | ||||
|         try { | ||||
|             $doc = DOMDocument::loadXML($xml); | ||||
|  | ||||
|         if (!empty($subjectEl)) { | ||||
|             $subject = new ActivityObject($subjectEl); | ||||
|         } else { | ||||
|             throw new Exception("Feed doesn't have an <activity:subject> element."); | ||||
|         } | ||||
|  | ||||
|         if (is_null($user)) { | ||||
|             $user = $this->userFromSubject($subject); | ||||
|         } | ||||
|  | ||||
|         $entries = $feed->getElementsByTagNameNS(Activity::ATOM, 'entry'); | ||||
|  | ||||
|         $activities = $this->entriesToActivities($entries, $feed); | ||||
|  | ||||
|         // XXX: sort entries here | ||||
|  | ||||
|         foreach ($activities as $activity) { | ||||
|             try { | ||||
|                 switch ($activity->verb) { | ||||
|                 case ActivityVerb::FOLLOW: | ||||
|                     $this->subscribeProfile($user, $subject, $activity); | ||||
|                     break; | ||||
|                 case ActivityVerb::JOIN: | ||||
|                     $this->joinGroup($user, $activity); | ||||
|                     break; | ||||
|                 case ActivityVerb::POST: | ||||
|                     $this->postNote($user, $activity); | ||||
|                     break; | ||||
|                 default: | ||||
|                     throw new Exception("Unknown verb: {$activity->verb}"); | ||||
|                 } | ||||
|             } catch (Exception $e) { | ||||
|                 common_log(LOG_WARNING, $e->getMessage()); | ||||
|                 continue; | ||||
|             if ($doc->documentElement->namespaceURI != Activity::ATOM || | ||||
|                 $doc->documentElement->localName != 'feed') { | ||||
|                 throw new ClientException(_("Not an atom feed.")); | ||||
|             } | ||||
|  | ||||
|             $feed = $doc->documentElement; | ||||
|  | ||||
|             $author = ActivityUtils::getFeedAuthor($feed); | ||||
|  | ||||
|             if (empty($author)) { | ||||
|                 throw new ClientException(_("No author in the feed.")); | ||||
|             } | ||||
|  | ||||
|             if (empty($user)) { | ||||
|                 if ($trusted) { | ||||
|                     $user = $this->userFromAuthor($author); | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             $entries = $feed->getElementsByTagNameNS(Activity::ATOM, 'entry'); | ||||
|  | ||||
|             $activities = $this->entriesToActivities($entries, $feed); | ||||
|  | ||||
|             $qm = QueueManager::get(); | ||||
|  | ||||
|             foreach ($activities as $activity) { | ||||
|                 $qm->enqueue(array($user, $author, $activity, $trusted), 'actimp'); | ||||
|             } | ||||
|         } catch (ClientException $ce) { | ||||
|             common_log(LOG_WARNING, $ce->getMessage()); | ||||
|             return true; | ||||
|         } catch (ServerException $se) { | ||||
|             common_log(LOG_ERR, $ce->getMessage()); | ||||
|             return false; | ||||
|         } catch (Exception $e) { | ||||
|             common_log(LOG_ERR, $ce->getMessage()); | ||||
|             return false; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     function subscribeProfile($user, $subject, $activity) | ||||
|     function userFromAuthor($author) | ||||
|     { | ||||
|         $profile = $user->getProfile(); | ||||
|  | ||||
|         if ($activity->objects[0]->id == $subject->id) { | ||||
|             if (!$this->_trusted) { | ||||
|                     throw new Exception("Skipping a pushed subscription."); | ||||
|             } else { | ||||
|                 $other = $activity->actor; | ||||
|                 $otherUser = User::staticGet('uri', $other->id); | ||||
|  | ||||
|                 if (!empty($otherUser)) { | ||||
|                     $otherProfile = $otherUser->getProfile(); | ||||
|                 } else { | ||||
|                     throw new Exception("Can't force remote user to subscribe."); | ||||
|                 } | ||||
|                 // XXX: don't do this for untrusted input! | ||||
|                 Subscription::start($otherProfile, $profile); | ||||
|             } | ||||
|         } else if (empty($activity->actor)  | ||||
|                    || $activity->actor->id == $subject->id) { | ||||
|  | ||||
|             $other = $activity->objects[0]; | ||||
|             $otherUser = User::staticGet('uri', $other->id); | ||||
|  | ||||
|             if (!empty($otherUser)) { | ||||
|                 $otherProfile = $otherUser->getProfile(); | ||||
|             } else { | ||||
|                 $oprofile = Ostatus_profile::ensureActivityObjectProfile($other); | ||||
|                 $otherProfile = $oprofile->localProfile(); | ||||
|             } | ||||
|  | ||||
|             Subscription::start($profile, $otherProfile); | ||||
|         } else { | ||||
|             throw new Exception("This activity seems unrelated to our user."); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     function joinGroup($user, $activity) | ||||
|     { | ||||
|         // XXX: check that actor == subject | ||||
|  | ||||
|         $uri = $activity->objects[0]->id; | ||||
|  | ||||
|         $group = User_group::staticGet('uri', $uri); | ||||
|  | ||||
|         if (empty($group)) { | ||||
|             $oprofile = Ostatus_profile::ensureActivityObjectProfile($activity->objects[0]); | ||||
|             if (!$oprofile->isGroup()) { | ||||
|                 throw new Exception("Remote profile is not a group!"); | ||||
|             } | ||||
|             $group = $oprofile->localGroup(); | ||||
|         } | ||||
|  | ||||
|         assert(!empty($group)); | ||||
|  | ||||
|         if (Event::handle('StartJoinGroup', array($group, $user))) { | ||||
|             Group_member::join($group->id, $user->id); | ||||
|             Event::handle('EndJoinGroup', array($group, $user)); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     // XXX: largely cadged from Ostatus_profile::processNote() | ||||
|  | ||||
|     function postNote($user, $activity) | ||||
|     { | ||||
|         $note = $activity->objects[0]; | ||||
|  | ||||
|         $sourceUri = $note->id; | ||||
|  | ||||
|         $notice = Notice::staticGet('uri', $sourceUri); | ||||
|  | ||||
|         if (!empty($notice)) { | ||||
|             // This is weird. | ||||
|             $orig = clone($notice); | ||||
|             $notice->profile_id = $user->id; | ||||
|             $notice->update($orig); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         // Use summary as fallback for content | ||||
|  | ||||
|         if (!empty($note->content)) { | ||||
|             $sourceContent = $note->content; | ||||
|         } else if (!empty($note->summary)) { | ||||
|             $sourceContent = $note->summary; | ||||
|         } else if (!empty($note->title)) { | ||||
|             $sourceContent = $note->title; | ||||
|         } else { | ||||
|             // @fixme fetch from $sourceUrl? | ||||
|             // @todo i18n FIXME: use sprintf and add i18n. | ||||
|             throw new ClientException("No content for notice {$sourceUri}."); | ||||
|         } | ||||
|  | ||||
|         // Get (safe!) HTML and text versions of the content | ||||
|  | ||||
|         $rendered = $this->purify($sourceContent); | ||||
|         $content = html_entity_decode(strip_tags($rendered), ENT_QUOTES, 'UTF-8'); | ||||
|  | ||||
|         $shortened = $user->shortenLinks($content); | ||||
|  | ||||
|         $options = array('is_local' => Notice::LOCAL_PUBLIC, | ||||
|                          'uri' => $sourceUri, | ||||
|                          'rendered' => $rendered, | ||||
|                          'replies' => array(), | ||||
|                          'groups' => array(), | ||||
|                          'tags' => array(), | ||||
|                          'urls' => array()); | ||||
|  | ||||
|         // Check for optional attributes... | ||||
|  | ||||
|         if (!empty($activity->time)) { | ||||
|             $options['created'] = common_sql_date($activity->time); | ||||
|         } | ||||
|  | ||||
|         if ($activity->context) { | ||||
|             // Any individual or group attn: targets? | ||||
|  | ||||
|             list($options['groups'], $options['replies']) = $this->filterAttention($activity->context->attention); | ||||
|  | ||||
|             // Maintain direct reply associations | ||||
|             // @fixme what about conversation ID? | ||||
|             if (!empty($activity->context->replyToID)) { | ||||
|                 $orig = Notice::staticGet('uri', | ||||
|                                           $activity->context->replyToID); | ||||
|                 if (!empty($orig)) { | ||||
|                     $options['reply_to'] = $orig->id; | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             $location = $activity->context->location; | ||||
|  | ||||
|             if ($location) { | ||||
|                 $options['lat'] = $location->lat; | ||||
|                 $options['lon'] = $location->lon; | ||||
|                 if ($location->location_id) { | ||||
|                     $options['location_ns'] = $location->location_ns; | ||||
|                     $options['location_id'] = $location->location_id; | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         // Atom categories <-> hashtags | ||||
|  | ||||
|         foreach ($activity->categories as $cat) { | ||||
|             if ($cat->term) { | ||||
|                 $term = common_canonical_tag($cat->term); | ||||
|                 if ($term) { | ||||
|                     $options['tags'][] = $term; | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         // Atom enclosures -> attachment URLs | ||||
|         foreach ($activity->enclosures as $href) { | ||||
|             // @fixme save these locally or....? | ||||
|             $options['urls'][] = $href; | ||||
|         } | ||||
|  | ||||
|         $saved = Notice::saveNew($user->id, | ||||
|                                  $content, | ||||
|                                  'restore', // TODO: restore the actual source | ||||
|                                  $options); | ||||
|  | ||||
|         return $saved; | ||||
|     } | ||||
|  | ||||
|     function filterAttention($attn) | ||||
|     { | ||||
|         $groups = array(); | ||||
|         $replies = array(); | ||||
|  | ||||
|         foreach (array_unique($attn) as $recipient) { | ||||
|  | ||||
|             // Is the recipient a local user? | ||||
|  | ||||
|             $user = User::staticGet('uri', $recipient); | ||||
|  | ||||
|             if ($user) { | ||||
|                 // @fixme sender verification, spam etc? | ||||
|                 $replies[] = $recipient; | ||||
|                 continue; | ||||
|             } | ||||
|  | ||||
|             // Is the recipient a remote group? | ||||
|             $oprofile = Ostatus_profile::ensureProfileURI($recipient); | ||||
|  | ||||
|             if ($oprofile) { | ||||
|                 if (!$oprofile->isGroup()) { | ||||
|                     // may be canonicalized or something | ||||
|                     $replies[] = $oprofile->uri; | ||||
|                 } | ||||
|                 continue; | ||||
|             } | ||||
|  | ||||
|             // Is the recipient a local group? | ||||
|             // @fixme uri on user_group isn't reliable yet | ||||
|             // $group = User_group::staticGet('uri', $recipient); | ||||
|             $id = OStatusPlugin::localGroupFromUrl($recipient); | ||||
|  | ||||
|             if ($id) { | ||||
|                 $group = User_group::staticGet('id', $id); | ||||
|                 if ($group) { | ||||
|                     // Deliver to all members of this local group if allowed. | ||||
|                     $profile = $sender->localProfile(); | ||||
|                     if ($profile->isMember($group)) { | ||||
|                         $groups[] = $group->id; | ||||
|                     } else { | ||||
|                         common_log(LOG_INFO, "Skipping reply to local group {$group->nickname} as sender {$profile->id} is not a member"); | ||||
|                     } | ||||
|                     continue; | ||||
|                 } else { | ||||
|                     common_log(LOG_INFO, "Skipping reply to bogus group $recipient"); | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         return array($groups, $replies); | ||||
|     } | ||||
|   | ||||
|    function userFromSubject($subject) | ||||
|     { | ||||
|         $user = User::staticGet('uri', $subject->id); | ||||
|         $user = User::staticGet('uri', $author->id); | ||||
|  | ||||
|         if (empty($user)) { | ||||
|             $attrs = | ||||
|                 array('nickname' => Ostatus_profile::getActivityObjectNickname($subject), | ||||
|                       'uri' => $subject->id); | ||||
|                 array('nickname' => Ostatus_profile::getActivityObjectNickname($author), | ||||
|                       'uri' => $author->id); | ||||
|  | ||||
|             $user = User::register($attrs); | ||||
|         } | ||||
|  | ||||
|         $profile = $user->getProfile(); | ||||
|         Ostatus_profile::updateProfile($profile, $subject); | ||||
|         Ostatus_profile::updateProfile($profile, $author); | ||||
|  | ||||
|         // FIXME: Update avatar | ||||
|         return $user; | ||||
|     } | ||||
|  | ||||
|     function purify($content) | ||||
|     { | ||||
|         $config = array('safe' => 1, | ||||
|                         'deny_attribute' => 'id,style,on*'); | ||||
|         return htmLawed($content, $config); | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -266,6 +266,8 @@ abstract class QueueManager extends IoManager | ||||
|  | ||||
|             // Background user management tasks... | ||||
|             $this->connect('deluser', 'DelUserQueueHandler'); | ||||
|             $this->connect('feedimp', 'FeedImporter'); | ||||
|             $this->connect('actimp', 'ActivityImporter'); | ||||
|  | ||||
|             // Broadcasting profile updates to OMB remote subscribers | ||||
|             $this->connect('profile', 'ProfileQueueHandler'); | ||||
|   | ||||
		Reference in New Issue
	
	Block a user