allowedNotifierTypes = array_keys( $mainConfig->get( 'EchoNotifiers' ) ); } public function execute() { // To avoid API warning, register the parameter used to bust browser cache $this->getMain()->getVal( '_' ); if ( !$this->getUser()->isRegistered() ) { $this->dieWithError( 'apierror-mustbeloggedin-generic', 'login-required' ); } $params = $this->extractRequestParams(); /* @deprecated */ if ( $params['format'] === 'flyout' ) { $this->addDeprecation( 'apiwarn-echo-deprecation-flyout', 'action=query&meta=notifications¬format=flyout' ); } elseif ( $params['format'] === 'html' ) { $this->addDeprecation( 'apiwarn-echo-deprecation-html', 'action=query&meta=notifications¬format=html' ); } if ( $this->allowCrossWikiNotifications() ) { $this->crossWikiSummary = $params['crosswikisummary']; } $results = []; if ( in_array( WikiMap::getCurrentWikiId(), $this->getRequestedWikis() ) ) { $results[WikiMap::getCurrentWikiId()] = $this->getLocalNotifications( $params ); } if ( $this->getRequestedForeignWikis() ) { $foreignResults = $this->getFromForeign(); foreach ( $foreignResults as $wiki => $result ) { if ( isset( $result['query']['notifications'] ) ) { $results[$wiki] = $result['query']['notifications']; } } } // after getting local & foreign results, merge them all together $result = $this->mergeResults( $results, $params ); if ( $params['groupbysection'] ) { foreach ( $params['sections'] as $section ) { if ( in_array( 'list', $params['prop'] ) ) { $this->getResult()->setIndexedTagName( $result[$section]['list'], 'notification' ); } } } else { if ( in_array( 'list', $params['prop'] ) ) { $this->getResult()->setIndexedTagName( $result['list'], 'notification' ); } } $this->getResult()->addValue( 'query', $this->getModuleName(), $result ); } /** * @param array $params * @return array */ protected function getLocalNotifications( array $params ) { $user = $this->getUser(); $prop = $params['prop']; $titles = null; if ( $params['titles'] ) { $titles = array_values( array_filter( array_map( [ Title::class, 'newFromText' ], $params['titles'] ) ) ); if ( in_array( '[]', $params['titles'] ) ) { $titles[] = null; } } $result = []; if ( in_array( 'list', $prop ) ) { // Group notification results by section if ( $params['groupbysection'] ) { foreach ( $params['sections'] as $section ) { $result[$section] = $this->getSectionPropList( $user, $section, $params['filter'], $params['limit'], $params[$section . 'continue'], $params['format'], $titles, $params[$section . 'unreadfirst'], $params['bundle'], $params['notifiertypes'] ); if ( $this->crossWikiSummary ) { // insert fake notification for foreign notifications $foreignNotification = $this->makeForeignNotification( $user, $params['format'], $section ); if ( $foreignNotification ) { array_unshift( $result[$section]['list'], $foreignNotification ); } } } } else { $attributeManager = Services::getInstance()->getAttributeManager(); $result = $this->getPropList( $user, $attributeManager->getUserEnabledEventsBySections( $user, $params['notifiertypes'], $params['sections'] ), $params['filter'], $params['limit'], $params['continue'], $params['format'], $titles, $params['unreadfirst'], $params['bundle'] ); // if exactly 1 section is specified, we consider only that section, otherwise // we pass ALL to consider all foreign notifications $section = count( $params['sections'] ) === 1 ? reset( $params['sections'] ) : AttributeManager::ALL; if ( $this->crossWikiSummary ) { $foreignNotification = $this->makeForeignNotification( $user, $params['format'], $section ); if ( $foreignNotification ) { array_unshift( $result['list'], $foreignNotification ); } } } } if ( in_array( 'count', $prop ) ) { $result = array_merge_recursive( $result, $this->getPropCount( $user, $params['sections'], $params['groupbysection'] ) ); } if ( in_array( 'seenTime', $prop ) ) { $result = array_merge_recursive( $result, $this->getPropSeenTime( $user, $params['sections'], $params['groupbysection'] ) ); } return $result; } /** * Internal method for getting the property 'list' data for individual section * @param User $user * @param string $section 'alert' or 'message' * @param string[] $filter 'all', 'read' or 'unread' * @param int $limit * @param string $continue * @param string $format * @param Title[]|null $titles * @param bool $unreadFirst * @param bool $bundle * @param string[] $notifierTypes * @return array */ protected function getSectionPropList( User $user, $section, $filter, $limit, $continue, $format, array $titles = null, $unreadFirst = false, $bundle = false, array $notifierTypes = [ 'web' ] ) { $attributeManager = Services::getInstance()->getAttributeManager(); $sectionEvents = $attributeManager->getUserEnabledEventsBySections( $user, $notifierTypes, [ $section ] ); if ( !$sectionEvents ) { $result = [ 'list' => [], 'continue' => null ]; } else { $result = $this->getPropList( $user, $sectionEvents, $filter, $limit, $continue, $format, $titles, $unreadFirst, $bundle ); } return $result; } /** * Internal helper method for getting property 'list' data, this is based * on the event types specified in the arguments and it could be event types * of a set of sections or a single section * @param User $user * @param string[] $eventTypes * @param string[] $filter 'all', 'read' or 'unread' * @param int $limit * @param string $continue * @param string $format * @param Title[]|null $titles * @param bool $unreadFirst * @param bool $bundle * @return array */ protected function getPropList( User $user, array $eventTypes, $filter, $limit, $continue, $format, array $titles = null, $unreadFirst = false, $bundle = false ) { $result = [ 'list' => [], 'continue' => null ]; $notifMapper = new NotificationMapper(); // check if we want both read & unread... if ( in_array( 'read', $filter ) && in_array( '!read', $filter ) ) { // Prefer unread notifications. We don't care about next offset in this case if ( $unreadFirst ) { // query for unread notifications past 'continue' (offset) $notifs = $notifMapper->fetchUnreadByUser( $user, $limit + 1, $continue, $eventTypes, $titles ); /* * 'continue' has a timestamp & id (to start with, in case * there would be multiple events with that same timestamp) * Unread notifications should always load first, but may be * older than read ones, but we can work with current * 'continue' format: * * if there's no continue, first load unread notifications * * if there's a continue, fetch unread notifications first * * if there are no unread ones, continue must've been * about read notifications: fetch 'em * * if there are unread ones but first one doesn't match * continue id, it must've been about read notifications: * discard unread & fetch read */ if ( $notifs && $continue ) { /** @var Notification $first */ $first = reset( $notifs ); $continueId = intval( trim( strrchr( $continue, '|' ), '|' ) ); if ( $first->getEvent()->getId() !== $continueId ) { // notification doesn't match continue id, it must've been // about read notifications: discard all unread ones $notifs = []; } } // If there are less unread notifications than we requested, // then fill the result with some read notifications $count = count( $notifs ); // we need 1 more than $limit, so we can respond 'continue' if ( $count <= $limit ) { // Query planner should be smart enough that passing a short list of ids to exclude // will only visit at most that number of extra rows. $mixedNotifs = $notifMapper->fetchByUser( $user, $limit - $count + 1, // if there were unread notifications, 'continue' was for // unread notifications and we should start fetching read // notifications from start $count > 0 ? null : $continue, $eventTypes, array_keys( $notifs ), $titles ); foreach ( $mixedNotifs as $notif ) { $notifs[$notif->getEvent()->getId()] = $notif; } } } else { $notifs = $notifMapper->fetchByUser( $user, $limit + 1, $continue, $eventTypes, [], $titles ); } } elseif ( in_array( 'read', $filter ) ) { $notifs = $notifMapper->fetchReadByUser( $user, $limit + 1, $continue, $eventTypes, $titles ); } else { // = if ( in_array( '!read', $filter ) ) { $notifs = $notifMapper->fetchUnreadByUser( $user, $limit + 1, $continue, $eventTypes, $titles ); } // get $overfetchedItem before bundling and rendering so that it is not affected by filtering /** @var Notification $overfetchedItem */ $overfetchedItem = count( $notifs ) > $limit ? array_pop( $notifs ) : null; $bundler = null; if ( $bundle ) { $bundler = new Bundler(); $notifs = $bundler->bundle( $notifs ); } while ( $notifs !== [] ) { /** @var Notification $notif */ $notif = array_shift( $notifs ); $output = DataOutputFormatter::formatOutput( $notif, $format, $user, $this->getLanguage() ); if ( $output !== false ) { $result['list'][] = $output; } elseif ( $bundler && $notif->getBundledNotifications() ) { // when the bundle_base gets filtered out, bundled notifications // have to be re-bundled and formatted $notifs = array_merge( $bundler->bundle( $notif->getBundledNotifications() ), $notifs ); } } // Generate offset if necessary if ( $overfetchedItem ) { // @todo: what to do with this when fetching from multiple wikis? $timestamp = wfTimestamp( TS_UNIX, $overfetchedItem->getTimestamp() ); $id = $overfetchedItem->getEvent()->getId(); $result['continue'] = $timestamp . '|' . $id; } return $result; } /** * Internal helper method for getting property 'count' data * @param User $user * @param string[] $sections * @param bool $groupBySection * @return array */ protected function getPropCount( User $user, array $sections, $groupBySection ) { $result = []; $notifUser = NotifUser::newFromUser( $user ); $global = $this->crossWikiSummary ? 'preference' : false; $totalRawCount = 0; foreach ( $sections as $section ) { $rawCount = $notifUser->getNotificationCount( $section, $global ); if ( $groupBySection ) { $result[$section]['rawcount'] = $rawCount; $result[$section]['count'] = NotificationController::formatNotificationCount( $rawCount ); } $totalRawCount += $rawCount; } $result['rawcount'] = $totalRawCount; $result['count'] = NotificationController::formatNotificationCount( $totalRawCount ); return $result; } /** * Internal helper method for getting property 'seenTime' data * @param User $user * @param string[] $sections * @param bool $groupBySection * @return array */ protected function getPropSeenTime( User $user, array $sections, $groupBySection ) { $result = []; $seenTimeHelper = SeenTime::newFromUser( $user ); if ( $groupBySection ) { foreach ( $sections as $section ) { $result[$section]['seenTime'] = $seenTimeHelper->getTime( $section, TS_ISO_8601 ); } } else { $result['seenTime'] = []; foreach ( $sections as $section ) { $result['seenTime'][$section] = $seenTimeHelper->getTime( $section, TS_ISO_8601 ); } } return $result; } /** * Build and format a "fake" notification to represent foreign notifications. * @param User $user * @param string $format * @param string $section * @return array|false A formatted notification, or false if there are no foreign notifications */ protected function makeForeignNotification( User $user, $format, $section = AttributeManager::ALL ) { $wikis = $this->getForeignNotifications()->getWikis( $section ); $count = $this->getForeignNotifications()->getCount( $section ); $maxTimestamp = $this->getForeignNotifications()->getTimestamp( $section ); $timestampsByWiki = []; foreach ( $wikis as $wiki ) { $timestampsByWiki[$wiki] = $this->getForeignNotifications()->getWikiTimestamp( $wiki, $section ); } if ( $count === 0 || $wikis === [] ) { return false; } // Sort wikis by timestamp, in descending order (newest first) usort( $wikis, static function ( $a, $b ) use ( $section, $timestampsByWiki ) { return (int)$timestampsByWiki[$b]->getTimestamp( TS_UNIX ) - (int)$timestampsByWiki[$a]->getTimestamp( TS_UNIX ); } ); $row = (object)[ 'event_id' => -1, 'event_type' => 'foreign', 'event_variant' => null, 'event_agent_id' => $user->getId(), 'event_agent_ip' => null, 'event_page_id' => null, 'event_extra' => serialize( [ 'section' => $section ?: 'all', 'wikis' => $wikis, 'count' => $count ] ), 'event_deleted' => 0, 'notification_user' => $user->getId(), 'notification_timestamp' => $maxTimestamp, 'notification_read_timestamp' => null, 'notification_bundle_hash' => md5( 'bogus' ), ]; // Format output like any other notification $notif = Notification::newFromRow( $row ); $output = DataOutputFormatter::formatOutput( $notif, $format, $user, $this->getLanguage() ); // Add cross-wiki-specific data $output['section'] = $section ?: 'all'; $output['count'] = $count; $output['sources'] = ForeignNotifications::getApiEndpoints( $wikis ); // Add timestamp information foreach ( $output['sources'] as $wiki => &$data ) { $data['ts'] = $timestampsByWiki[$wiki]->getTimestamp( TS_ISO_8601 ); } return $output; } protected function getForeignQueryParams() { $params = $this->getRequest()->getValues(); // don't request cross-wiki notification summaries unset( $params['notcrosswikisummary'] ); return $params; } /** * @param array[] $results * @param array $params * @return array */ protected function mergeResults( array $results, array $params ) { $primary = array_shift( $results ); if ( !$primary ) { $primary = []; } if ( in_array( 'list', $params['prop'] ) ) { $primary = $this->mergeList( $primary, $results, $params['groupbysection'] ); } if ( in_array( 'count', $params['prop'] ) && !$this->crossWikiSummary ) { // if crosswiki data was requested, the count in $primary // is accurate already // otherwise, we'll want to combine counts for all wikis $primary = $this->mergeCount( $primary, $results, $params['groupbysection'] ); } return $primary; } /** * @param array $primary * @param array[] $results * @param bool $groupBySection * @return array */ protected function mergeList( array $primary, array $results, $groupBySection ) { // sort all notifications by timestamp: most recent first $sort = static function ( $a, $b ) { return $a['timestamp']['utcunix'] - $b['timestamp']['utcunix']; }; if ( $groupBySection ) { foreach ( AttributeManager::$sections as $section ) { if ( !isset( $primary[$section]['list'] ) ) { $primary[$section]['list'] = []; } foreach ( $results as $result ) { $primary[$section]['list'] = array_merge( $primary[$section]['list'], $result[$section]['list'] ); } usort( $primary[$section]['list'], $sort ); } } else { if ( !isset( $primary['list'] ) || !is_array( $primary['list'] ) ) { $primary['list'] = []; } foreach ( $results as $result ) { $primary['list'] = array_merge( $primary['list'], $result['list'] ); } usort( $primary['list'], $sort ); } return $primary; } /** * @param array $primary * @param array[] $results * @param bool $groupBySection * @return array */ protected function mergeCount( array $primary, array $results, $groupBySection ) { if ( $groupBySection ) { foreach ( AttributeManager::$sections as $section ) { if ( !isset( $primary[$section]['rawcount'] ) ) { $primary[$section]['rawcount'] = 0; } foreach ( $results as $result ) { $primary[$section]['rawcount'] += $result[$section]['rawcount']; } $primary[$section]['count'] = NotificationController::formatNotificationCount( $primary[$section]['rawcount'] ); } } if ( !isset( $primary['rawcount'] ) ) { $primary['rawcount'] = 0; } foreach ( $results as $result ) { // regardless of groupbysection, totals are always included $primary['rawcount'] += $result['rawcount']; } $primary['count'] = NotificationController::formatNotificationCount( $primary['rawcount'] ); return $primary; } public function getAllowedParams() { $sections = AttributeManager::$sections; $params = $this->getCrossWikiParams() + [ 'filter' => [ ParamValidator::PARAM_ISMULTI => true, ParamValidator::PARAM_DEFAULT => 'read|!read', ParamValidator::PARAM_TYPE => [ 'read', '!read', ], ], 'prop' => [ ParamValidator::PARAM_ISMULTI => true, ParamValidator::PARAM_TYPE => [ 'list', 'count', 'seenTime', ], ParamValidator::PARAM_DEFAULT => 'list', ], 'sections' => [ ParamValidator::PARAM_DEFAULT => implode( '|', $sections ), ParamValidator::PARAM_TYPE => $sections, ParamValidator::PARAM_ISMULTI => true, ], 'groupbysection' => [ ParamValidator::PARAM_TYPE => 'boolean', ParamValidator::PARAM_DEFAULT => false, ], 'format' => [ ParamValidator::PARAM_TYPE => [ 'model', 'special', // @deprecated 'flyout', // @deprecated 'html', ], ApiBase::PARAM_HELP_MSG_PER_VALUE => [], ], 'limit' => [ ParamValidator::PARAM_TYPE => 'limit', ParamValidator::PARAM_DEFAULT => 20, IntegerDef::PARAM_MIN => 1, IntegerDef::PARAM_MAX => ApiBase::LIMIT_SML1, IntegerDef::PARAM_MAX2 => ApiBase::LIMIT_SML2, ], 'continue' => [ ApiBase::PARAM_HELP_MSG => 'api-help-param-continue', ], 'unreadfirst' => [ ParamValidator::PARAM_TYPE => 'boolean', ParamValidator::PARAM_DEFAULT => false, ], 'titles' => [ ParamValidator::PARAM_ISMULTI => true, ], 'bundle' => [ ParamValidator::PARAM_TYPE => 'boolean', ParamValidator::PARAM_DEFAULT => false, ], 'notifiertypes' => [ ParamValidator::PARAM_TYPE => $this->allowedNotifierTypes, ParamValidator::PARAM_ISMULTI => true, ParamValidator::PARAM_DEFAULT => 'web', ], ]; foreach ( $sections as $section ) { $params[$section . 'continue'] = null; $params[$section . 'unreadfirst'] = [ ParamValidator::PARAM_TYPE => 'boolean', ParamValidator::PARAM_DEFAULT => false, ]; } if ( $this->allowCrossWikiNotifications() ) { $params += [ // create "x notifications from y wikis" notification bundle & // include unread counts from other wikis in prop=count results 'crosswikisummary' => [ ParamValidator::PARAM_TYPE => 'boolean', ParamValidator::PARAM_DEFAULT => false, ], ]; } return $params; } /** * @see ApiBase::getExamplesMessages() * @return array */ protected function getExamplesMessages() { return [ 'action=query&meta=notifications' => 'apihelp-query+notifications-example-1', 'action=query&meta=notifications¬prop=count¬sections=alert|message¬groupbysection=1' => 'apihelp-query+notifications-example-2', 'action=query&meta=notifications¬notifiertypes=email' => 'apihelp-query+notifications-example-3', ]; } public function getHelpUrls() { return 'https://www.mediawiki.org/wiki/Special:MyLanguage/Echo_(Notifications)/API'; } }