<?php /* * StatusNet - the distributed open-source microblogging tool * Copyright (C) 2010 StatusNet, Inc. * * 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/>. */ /** * Class for activity streams * * Includes faves, notices, and subscriptions. * * We extend atomusernoticefeed since it does some nice setup for us. * */ class UserActivityStream extends AtomUserNoticeFeed { public $activities = array(); const OUTPUT_STRING = 1; const OUTPUT_RAW = 2; public $outputMode = self::OUTPUT_STRING; /** * * @param User $user * @param boolean $indent * @param boolean $outputMode: UserActivityStream::OUTPUT_STRING to return a string, * or UserActivityStream::OUTPUT_RAW to go to raw output. * Raw output mode will attempt to stream, keeping less * data in memory but will leave $this->activities incomplete. */ function __construct($user, $indent = true, $outputMode = UserActivityStream::OUTPUT_STRING, $after = null) { parent::__construct($user, null, $indent); $this->outputMode = $outputMode; if ($this->outputMode == self::OUTPUT_STRING) { // String buffering? Grab all the notices now. $notices = $this->getNotices(); } elseif ($this->outputMode == self::OUTPUT_RAW) { // Raw output... need to restructure from the stringer init. $this->xw = new XMLWriter(); $this->xw->openURI('php://output'); if(is_null($indent)) { $indent = common_config('site', 'indent'); } $this->xw->setIndent($indent); // We'll fetch notices later. $notices = array(); } else { throw new Exception('Invalid outputMode provided to ' . __METHOD__); } $this->after = $after; // Assume that everything but notices is feasible // to pull at once and work with in memory... $subscriptions = $this->getSubscriptions(); $subscribers = $this->getSubscribers(); $groups = $this->getGroups(); $faves = $this->getFaves(); $messagesFrom = $this->getMessagesFrom(); $messagesTo = $this->getMessagesTo(); $objs = array_merge($subscriptions, $subscribers, $groups, $faves, $notices, $messagesFrom, $messagesTo); $subscriptions = null; $subscribers = null; $groups = null; $faves = null; unset($subscriptions); unset($subscribers); unset($groups); unset($faves); // Sort by create date usort($objs, 'UserActivityStream::compareObject'); // We'll keep these around for later, and interleave them into // the output stream with the user's notices. $this->objs = $objs; } /** * Interleave the pre-sorted subs/groups/faves with the user's * notices, all in reverse chron order. */ function renderEntries($format=Feed::ATOM, $handle=null) { $haveOne = false; $end = time() + 1; foreach ($this->objs as $obj) { try { $act = $obj->asActivity(); } catch (Exception $e) { common_log(LOG_ERR, $e->getMessage()); continue; } $start = $act->time; if ($this->outputMode == self::OUTPUT_RAW && $start != $end) { // In raw mode, we haven't pre-fetched notices. // Grab the chunks of notices between other activities. try { $notices = $this->getNoticesBetween($start, $end); foreach ($notices as $noticeAct) { try { $nact = $noticeAct->asActivity($this->user); if ($format == Feed::ATOM) { $nact->outputTo($this, false, false); } else { if ($haveOne) { fwrite($handle, ","); } fwrite($handle, json_encode($nact->asArray())); $haveOne = true; } } catch (Exception $e) { common_log(LOG_ERR, $e->getMessage()); continue; } $nact = null; unset($nact); } } catch (Exception $e) { common_log(LOG_ERR, $e->getMessage()); } } $notices = null; unset($notices); try { if ($format == Feed::ATOM) { // Only show the author sub-element if it's different from default user $act->outputTo($this, false, ($act->actor->id != $this->user->getUri())); } else { if ($haveOne) { fwrite($handle, ","); } fwrite($handle, json_encode($act->asArray())); $haveOne = true; } } catch (Exception $e) { common_log(LOG_ERR, $e->getMessage()); } $act = null; unset($act); $end = $start; } if ($this->outputMode == self::OUTPUT_RAW) { // Grab anything after the last pre-sorted activity. try { if (!empty($this->after)) { $notices = $this->getNoticesBetween($this->after, $end); } else { $notices = $this->getNoticesBetween(0, $end); } foreach ($notices as $noticeAct) { try { $nact = $noticeAct->asActivity($this->user); if ($format == Feed::ATOM) { $nact->outputTo($this, false, false); } else { if ($haveOne) { fwrite($handle, ","); } fwrite($handle, json_encode($nact->asArray())); $haveOne = true; } } catch (Exception $e) { common_log(LOG_ERR, $e->getMessage()); continue; } } } catch (Exception $e) { common_log(LOG_ERR, $e->getMessage()); } } if (empty($this->after) || strtotime($this->user->created) > $this->after) { // We always add the registration activity at the end, even if // they have older activities (from restored backups) in their stream. try { $ract = $this->user->registrationActivity(); if ($format == Feed::ATOM) { $ract->outputTo($this, false, false); } else { if ($haveOne) { fwrite($handle, ","); } fwrite($handle, json_encode($ract->asArray())); $haveOne = true; } } catch (Exception $e) { common_log(LOG_ERR, $e->getMessage()); continue; } } } function compareObject($a, $b) { $ac = strtotime((empty($a->created)) ? $a->modified : $a->created); $bc = strtotime((empty($b->created)) ? $b->modified : $b->created); return (($ac == $bc) ? 0 : (($ac < $bc) ? 1 : -1)); } function getSubscriptions() { $subs = array(); $sub = new Subscription(); $sub->subscriber = $this->user->id; if (!empty($this->after)) { $sub->whereAdd("created > '" . common_sql_date($this->after) . "'"); } if ($sub->find()) { while ($sub->fetch()) { if ($sub->subscribed != $this->user->id) { $subs[] = clone($sub); } } } return $subs; } function getSubscribers() { $subs = array(); $sub = new Subscription(); $sub->subscribed = $this->user->id; if (!empty($this->after)) { $sub->whereAdd("created > '" . common_sql_date($this->after) . "'"); } if ($sub->find()) { while ($sub->fetch()) { if ($sub->subscriber != $this->user->id) { $subs[] = clone($sub); } } } return $subs; } function getFaves() { $faves = array(); $fave = new Fave(); $fave->user_id = $this->user->id; if (!empty($this->after)) { $fave->whereAdd("modified > '" . common_sql_date($this->after) . "'"); } if ($fave->find()) { while ($fave->fetch()) { $faves[] = clone($fave); } } return $faves; } /** * * @param int $start unix timestamp for earliest * @param int $end unix timestamp for latest * @return array of Notice objects */ function getNoticesBetween($start=0, $end=0) { $notices = array(); $notice = new Notice(); $notice->profile_id = $this->user->id; // Only stuff after $this->after if (!empty($this->after)) { if ($start) { $start = max($start, $this->after); } if ($end) { $end = max($end, $this->after); } } if ($start) { $tsstart = common_sql_date($start); $notice->whereAdd("created >= '$tsstart'"); } if ($end) { $tsend = common_sql_date($end); $notice->whereAdd("created < '$tsend'"); } $notice->orderBy('created DESC'); if ($notice->find()) { while ($notice->fetch()) { $notices[] = clone($notice); } } return $notices; } function getNotices() { if (!empty($this->after)) { return $this->getNoticesBetween($this->after); } else { return $this->getNoticesBetween(); } } function getGroups() { $groups = array(); $gm = new Group_member(); $gm->profile_id = $this->user->id; if (!empty($this->after)) { $gm->whereAdd("created > '" . common_sql_date($this->after) . "'"); } if ($gm->find()) { while ($gm->fetch()) { $groups[] = clone($gm); } } return $groups; } function getMessagesTo() { $msgMap = Message::listGet('to_profile', array($this->user->id)); $messages = $msgMap[$this->user->id]; if (!empty($this->after)) { $messages = array_filter($messages, array($this, 'createdAfter')); } return $messages; } function getMessagesFrom() { $msgMap = Message::listGet('from_profile', array($this->user->id)); $messages = $msgMap[$this->user->id]; if (!empty($this->after)) { $messages = array_filter($messages, array($this, 'createdAfter')); } return $messages; } function createdAfter($item) { $created = strtotime((empty($item->created)) ? $item->modified : $item->created); return ($created >= $this->after); } function writeJSON($handle) { require_once INSTALLDIR.'/lib/activitystreamjsondocument.php'; fwrite($handle, '{"items": ['); $this->renderEntries(Feed::JSON, $handle); fwrite($handle, ']}'); } }