mUser = $user; } /** * 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 $userId int * @return MWEchoEmailBatch/false */ public static function newFromUserId( $userId ) { $batchClassName = self::getEmailBatchClass(); $user = User::newFromId( intval( $userId ) ); $userEmailSetting = intval( $user->getOption( 'echo-email-frequency' ) ); // clear all existing events if user decides not to receive emails if ( $userEmailSetting == -1 ) { $emailBatch = new $batchClassName( $user ); $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 = $user->getOption( '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 = wfTimestamp( TS_UNIX, $userLastBatch ) + $userEmailSetting * 20 * 60 * 60; if ( wfTimestamp( TS_MW, $nextBatch ) > wfTimestampNow() ) { return false; } } return new $batchClassName( $user ); } /** * Get the name of the email batch class * @return string * @throws MWException */ private static function getEmailBatchClass() { global $wgEchoBackendName; $className = 'MW' . $wgEchoBackendName . 'EchoEmailBatch'; if ( !class_exists( $className ) ) { throw new MWException( "$wgEchoBackendName email batch is not supported!" ); } return $className; } /** * Wrapper function that calls other functions required to process email batch */ public function process() { wfProfileIn( __METHOD__ ); // 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 = EchoEvent::newFromRow( $row ); $this->appendContent( $event, $row->eeb_event_hash ); } $this->sendEmail(); } $this->clearProcessedEvent(); $this->updateUserLastBatchTimestamp(); wfProfileOut( __METHOD__ ); } /** * 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 */ abstract protected function setLastEvent(); /** * Update the user's last batch timestamp after a successful batch */ protected function updateUserLastBatchTimestamp() { $this->mUser->setOption( 'echo-email-last-batch', wfTimestampNow() ); $this->mUser->saveSettings(); $this->mUser->invalidateCache(); } /** * Get the events queued for the current user * @return array */ abstract protected function getEvents(); /** * Add individual event template to the big email content */ protected function appendContent( $event, $hash ) { // get the category for this event $category = $event->getCategory(); $event->setBundleHash( $hash ); $email = EchoNotificationController::formatNotification( $event, $this->mUser, 'email', 'emaildigest' ); if ( !isset( $this->content[$category] ) ) { $this->content[$category] = array(); } $this->content[$category][] = $email; } /** * Clear "processed" events in the queue, processed could be: email sent, invalid, users do not want to receive emails */ abstract public function clearProcessedEvent(); /** * Send the batch email */ public function sendEmail() { global $wgNotificationSender, $wgNotificationReplyName; // @Todo - replace them with the CONSTANT in 33810 once it is merged if ( $this->mUser->getOption( 'echo-email-frequency' ) == 7 ) { $frequency = 'weekly'; $emailDeliveryMode = 'weekly_digest'; } else { $frequency = 'daily'; $emailDeliveryMode = 'daily_digest'; } // Echo digest email mode $emailDigest = new EchoEmailDigest( $this->mUser, $this->content, $frequency ); $textEmailFormatter = new EchoTextEmailFormatter( $emailDigest ); $body = $textEmailFormatter->formatEmail(); $format = MWEchoNotifUser::newFromUser( $this->mUser )->getEmailFormat(); if ( $format == EchoHooks::EMAIL_FORMAT_HTML ) { $htmlEmailFormatter = new EchoHTMLEmailFormatter( $emailDigest ); $body = array( 'text' => $body, 'html' => $htmlEmailFormatter->formatEmail() ); } // email subject if ( $this->count > self::$displaySize ) { $count = wfMessage( 'echo-notification-count' ) ->inLanguage( $this->mUser->getOption( 'language' ) ) ->params( self::$displaySize )->text(); } else { $count = $this->count; } // Give grep a chance to find the usages: // echo-email-batch-subject-daily, echo-email-batch-subject-weekly $subject = wfMessage( 'echo-email-batch-subject-' . $frequency ) ->inLanguage( $this->mUser->getOption( 'language' ) ) ->params( $count, $this->count )->text(); $toAddress = MailAddress::newFromUser( $this->mUser ); $fromAddress = new MailAddress( $wgNotificationSender, EchoHooks::getNotificationSenderName() ); $replyAddress = new MailAddress( $wgNotificationSender, $wgNotificationReplyName ); // @Todo Push the email to job queue or just send it out directly? UserMailer::send( $toAddress, $fromAddress, $subject, $body, $replyAddress ); MWEchoEventLogging::logSchemaEchoMail( $this->mUser, $emailDeliveryMode ); } /** * Insert notification event into email queue * * @param $userId int * @param $eventId int * @param $priority int * @param $hash string * * @throws MWException */ public static function addToQueue( $userId, $eventId, $priority, $hash ) { $batchClassName = self::getEmailBatchClass(); if ( !method_exists( $batchClassName, 'actuallyAddToQueue' ) ) { throw new MWException( "$batchClassName must implement method actuallyAddToQueue()" ); } $batchClassName::actuallyAddToQueue( $userId, $eventId, $priority, $hash ); } /** * Get a list of users to be notified for the batch * * @param $startUserId int * @param $batchSize int * * @throws MWException * @return ResultWrapper|bool */ public static function getUsersToNotify( $startUserId, $batchSize ) { $batchClassName = self::getEmailBatchClass(); if ( !method_exists( $batchClassName, 'actuallyGetUsersToNotify' ) ) { throw new MWException( "$batchClassName must implement method actuallyGetUsersToNotify()" ); } return $batchClassName::actuallyGetUsersToNotify( $startUserId, $batchSize ); } }