mUser = $user; $this->language = $languageFactory->getLanguage( $userOptionsManager->getOption( $user, 'language' ) ); $this->userOptionsManager = $userOptionsManager; } /** * Factory method to determine whether to create a batch instance for this * user based on the user setting, this assumes the following value for * member setting for echo-email-frequency * -1 - no email * 0 - instant * 1 - once everyday * 7 - once every 7 days * @param int $userId * @param bool $enforceFrequency Whether email sending frequency should * be enforced. * * When true, today's notifications won't be returned if they are * configured to go out tonight or at the end of the week. * * When false, all pending notifications will be returned. * @return EmailBatch|false */ public static function newFromUserId( $userId, $enforceFrequency = true ) { $user = User::newFromId( (int)$userId ); $services = MediaWikiServices::getInstance(); $userOptionsManager = $services->getUserOptionsManager(); $languageFactory = $services->getLanguageFactory(); $userEmailSetting = (int)$userOptionsManager->getOption( $user, 'echo-email-frequency' ); // clear all existing events if user decides not to receive emails if ( $userEmailSetting == -1 ) { $emailBatch = new self( $user, $userOptionsManager, $languageFactory ); $emailBatch->clearProcessedEvent(); return false; } // @Todo - There may be some items idling in the queue, eg, a bundle job is lost // and there is not never another message with the same hash or a user switches from // digest to instant. We should check the first item in the queue, if it doesn't // have either web or email bundling or created long ago, then clear it, this will // prevent idling item queuing up. // user has instant email delivery if ( $userEmailSetting == 0 ) { return false; } $userLastBatch = $userOptionsManager->getOption( $user, 'echo-email-last-batch' ); // send email batch, if // 1. it has been long enough since last email batch based on frequency // 2. there is no last batch timestamp recorded for the user // 3. user has switched from batch to instant email, send events left in the queue if ( $userLastBatch ) { // use 20 as hours per day to get estimate $nextBatch = (int)wfTimestamp( TS_UNIX, $userLastBatch ) + $userEmailSetting * 20 * 60 * 60; if ( $enforceFrequency && wfTimestamp( TS_MW, $nextBatch ) > wfTimestampNow() ) { return false; } } return new self( $user, $userOptionsManager, $languageFactory ); } /** * Wrapper function that calls other functions required to process email batch */ public function process() { // if there is no event for this user, exist the process if ( !$this->setLastEvent() ) { return; } // get valid events $events = $this->getEvents(); if ( $events ) { foreach ( $events as $row ) { $this->count++; if ( $this->count > self::$displaySize ) { break; } $event = Event::newFromRow( $row ); if ( !$event ) { continue; } $event->setBundleHash( $row->eeb_event_hash ); $this->events[] = $event; } $bundler = new Bundler(); $this->events = $bundler->bundle( $this->events ); $this->sendEmail(); } $this->clearProcessedEvent(); $this->updateUserLastBatchTimestamp(); } /** * Set the last event of this batch, this is a cutoff point for clearing * processed/invalid events * * @return bool true if event exists false otherwise */ protected function setLastEvent() { $dbr = DbFactory::newFromDefault()->getEchoDb( DB_REPLICA ); $res = $dbr->newSelectQueryBuilder() ->select( 'MAX( eeb_event_id )' ) ->from( 'echo_email_batch' ) ->where( [ 'eeb_user_id' => $this->mUser->getId() ] ) ->caller( __METHOD__ ) ->fetchField(); if ( $res ) { $this->lastEvent = $res; return true; } return false; } /** * Update the user's last batch timestamp after a successful batch */ protected function updateUserLastBatchTimestamp() { $this->userOptionsManager->setOption( $this->mUser, 'echo-email-last-batch', wfTimestampNow() ); $this->mUser->saveSettings(); $this->mUser->invalidateCache(); } /** * Get the events queued for the current user * @return stdClass[] */ protected function getEvents() { global $wgEchoNotifications; $events = []; $validEvents = array_keys( $wgEchoNotifications ); // Per the tech discussion in the design meeting (03/22/2013), since this is // processed by a cron job, it's okay to use GROUP BY over more complex // composite index, favor insert performance, storage space over read // performance in this case if ( $validEvents ) { $dbr = DbFactory::newFromDefault()->getEchoDb( DB_REPLICA ); $queryBuilder = $dbr->newSelectQueryBuilder() ->select( array_merge( Event::selectFields(), [ 'eeb_id', 'eeb_user_id', 'eeb_event_priority', 'eeb_event_id', 'eeb_event_hash', ] ) ) ->from( 'echo_email_batch' ) ->join( 'echo_event', null, 'event_id = eeb_event_id' ) ->where( [ 'eeb_user_id' => $this->mUser->getId(), 'event_type' => $validEvents ] ) ->orderBy( 'eeb_event_priority' ) ->limit( self::$displaySize + 1 ) ->caller( __METHOD__ ); if ( $this->userOptionsManager->getOption( $this->mUser, 'echo-dont-email-read-notifications' ) ) { $queryBuilder ->join( 'echo_notification', null, 'notification_event = event_id' ) ->andWhere( [ 'notification_read_timestamp' => null ] ); } // See setLastEvent() for more detail for this variable if ( $this->lastEvent ) { $queryBuilder->andWhere( $dbr->expr( 'eeb_event_id', '<=', (int)$this->lastEvent ) ); } $res = $queryBuilder->fetchResultSet(); foreach ( $res as $row ) { // records in the queue inserted before email bundling code // have no hash, in this case, we just ignore them if ( $row->eeb_event_hash ) { $events[$row->eeb_id] = $row; } } } return $events; } /** * Clear "processed" events in the queue, * processed could be: email sent, invalid, users do not want to receive emails */ public function clearProcessedEvent() { global $wgUpdateRowsPerQuery; $eventMapper = new EventMapper(); $dbFactory = DbFactory::newFromDefault(); $dbw = $dbFactory->getEchoDb( DB_PRIMARY ); $dbr = $dbFactory->getEchoDb( DB_REPLICA ); $lbFactory = MediaWikiServices::getInstance()->getDBLoadBalancerFactory(); $ticket = $lbFactory->getEmptyTransactionTicket( __METHOD__ ); $domainId = $dbw->getDomainID(); $iterator = new BatchRowIterator( $dbr, 'echo_email_batch', 'eeb_event_id', $wgUpdateRowsPerQuery ); $iterator->addConditions( [ 'eeb_user_id' => $this->mUser->getId() ] ); if ( $this->lastEvent ) { // There is a processed cutoff point $iterator->addConditions( [ 'eeb_event_id <= ' . (int)$this->lastEvent ] ); } $iterator->setCaller( __METHOD__ ); foreach ( $iterator as $batch ) { $eventIds = []; foreach ( $batch as $row ) { $eventIds[] = $row->eeb_event_id; } $dbw->newDeleteQueryBuilder() ->deleteFrom( 'echo_email_batch' ) ->where( [ 'eeb_user_id' => $this->mUser->getId(), 'eeb_event_id' => $eventIds ] ) ->caller( __METHOD__ ) ->execute(); // Find out which events are now orphaned, i.e. no longer referenced in echo_email_batch // (besides the rows we just deleted) or in echo_notification, and delete them $eventMapper->deleteOrphanedEvents( $eventIds, $this->mUser->getId(), 'echo_email_batch' ); $lbFactory->commitAndWaitForReplication( __METHOD__, $ticket, [ 'domain' => $domainId ] ); } } /** * Send the batch email */ public function sendEmail() { global $wgPasswordSender, $wgNoReplyAddress; if ( $this->userOptionsManager->getOption( $this->mUser, 'echo-email-frequency' ) == EmailFrequency::WEEKLY_DIGEST ) { $frequency = 'weekly'; $emailDeliveryMode = 'weekly_digest'; } else { $frequency = 'daily'; $emailDeliveryMode = 'daily_digest'; } $textEmailDigestFormatter = new EchoPlainTextDigestEmailFormatter( $this->mUser, $this->language, $frequency ); $content = $textEmailDigestFormatter->format( $this->events, 'email' ); if ( !$content ) { // no event could be formatted return; } $format = NotifUser::newFromUser( $this->mUser )->getEmailFormat(); if ( $format == EmailFormat::HTML ) { $htmlEmailDigestFormatter = new EchoHtmlDigestEmailFormatter( $this->mUser, $this->language, $frequency ); $htmlContent = $htmlEmailDigestFormatter->format( $this->events, 'email' ); $content = [ 'body' => [ 'text' => $content['body'], 'html' => $htmlContent['body'], ], 'subject' => $htmlContent['subject'], ]; } $toAddress = MailAddress::newFromUser( $this->mUser ); $fromAddress = new MailAddress( $wgPasswordSender, wfMessage( 'emailsender' )->inContentLanguage()->text() ); $replyTo = new MailAddress( $wgNoReplyAddress ); // @Todo Push the email to job queue or just send it out directly? UserMailer::send( $toAddress, $fromAddress, $content['subject'], $content['body'], [ 'replyTo' => $replyTo ] ); } /** * Insert notification event into email queue * * @param int $userId * @param int $eventId * @param int $priority * @param string $hash */ public static function addToQueue( $userId, $eventId, $priority, $hash ) { if ( !$userId || !$eventId ) { return; } $dbw = DbFactory::newFromDefault()->getEchoDb( DB_PRIMARY ); $row = [ 'eeb_user_id' => $userId, 'eeb_event_id' => $eventId, 'eeb_event_priority' => $priority, 'eeb_event_hash' => $hash ]; $dbw->newInsertQueryBuilder() ->insertInto( 'echo_email_batch' ) ->ignore() ->row( $row ) ->caller( __METHOD__ ) ->execute(); } /** * Get a list of users to be notified for the batch * * @param int $startUserId * @param int $batchSize * * @return IResultWrapper */ public static function getUsersToNotify( $startUserId, $batchSize ) { $dbr = DbFactory::newFromDefault()->getEchoDb( DB_REPLICA ); return $dbr->newSelectQueryBuilder() ->select( 'eeb_user_id' ) ->from( 'echo_email_batch' ) ->where( $dbr->expr( 'eeb_user_id', '>', (int)$startUserId ) ) ->orderBy( 'eeb_user_id' ) ->limit( $batchSize ) ->caller( __METHOD__ ) ->fetchResultSet(); } }