2012-11-27 01:53:35 +00:00
|
|
|
<?php
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Handles email batch for a user
|
|
|
|
*/
|
|
|
|
class MWEchoEmailBatch {
|
|
|
|
|
|
|
|
// the user to be notified
|
|
|
|
protected $mUser;
|
|
|
|
|
|
|
|
// list of email content
|
|
|
|
protected $content = array();
|
|
|
|
// the last notification event of this batch
|
|
|
|
protected $lastEvent;
|
|
|
|
// the event count, this count is supported up to self::$displaySize + 1
|
|
|
|
protected $count = 0;
|
|
|
|
|
|
|
|
// number of events to include in an email, we couldn't include
|
|
|
|
// all events in a batch email
|
|
|
|
protected static $displaySize = 10;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param $user User
|
|
|
|
*/
|
|
|
|
public function __construct( User $user ) {
|
|
|
|
$this->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 ) {
|
|
|
|
$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 MWEchoEmailBatch( $user );
|
|
|
|
$emailBatch->clearProcessedEvent();
|
|
|
|
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 MWEchoEmailBatch( $user );
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* 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 $batchId => $row ) {
|
|
|
|
$this->count++;
|
|
|
|
if ( $this->count > self::$displaySize ) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
$event = EchoEvent::newFromRow( $row );
|
|
|
|
$this->appendContent( $event );
|
|
|
|
}
|
|
|
|
|
|
|
|
$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
|
|
|
|
*/
|
|
|
|
protected function setLastEvent() {
|
|
|
|
$dbr = wfGetDB( DB_SLAVE );
|
|
|
|
$res = $dbr->selectField(
|
|
|
|
array( 'echo_email_batch' ),
|
|
|
|
array( 'MAX( eeb_event_id )' ),
|
|
|
|
array( 'eeb_user_id' => $this->mUser->getId() ),
|
|
|
|
__METHOD__
|
|
|
|
);
|
|
|
|
|
|
|
|
if ( $res ) {
|
|
|
|
$this->lastEvent = $res;
|
|
|
|
return true;
|
|
|
|
} else {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* 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
|
|
|
|
*/
|
|
|
|
protected function getEvents() {
|
|
|
|
$events = array();
|
|
|
|
|
|
|
|
$validEvents = EchoEvent::gatherValidEchoEvents();
|
|
|
|
|
|
|
|
if ( $validEvents ) {
|
|
|
|
$dbr = wfGetDB( DB_SLAVE );
|
|
|
|
|
|
|
|
$conds = array(
|
|
|
|
'eeb_user_id' => $this->mUser->getId(),
|
|
|
|
'event_id = eeb_event_id',
|
|
|
|
'event_type' => $validEvents
|
|
|
|
);
|
|
|
|
|
|
|
|
if ( $this->lastEvent ) {
|
|
|
|
$conds[] = 'eeb_event_id <= ' . intval( $this->lastEvent );
|
|
|
|
}
|
|
|
|
|
|
|
|
$res = $dbr->select(
|
|
|
|
array( 'echo_email_batch', 'echo_event' ),
|
|
|
|
array( '*' ),
|
|
|
|
$conds,
|
|
|
|
__METHOD__,
|
|
|
|
array( 'ORDER BY' => 'eeb_event_priority, eeb_event_id', 'LIMIT' => self::$displaySize + 1 )
|
|
|
|
);
|
|
|
|
|
|
|
|
foreach( $res as $row ) {
|
|
|
|
$events[$row->eeb_id] = $row;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return $events;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Add individual event template to the big email content
|
|
|
|
*/
|
|
|
|
protected function appendContent( $event ) {
|
|
|
|
global $wgEchoEventDetails;
|
|
|
|
|
|
|
|
// get the category for this event
|
|
|
|
$category = 'other';
|
|
|
|
if ( isset( $wgEchoEventDetails[$event->getType()]['category'] ) ) {
|
|
|
|
$category = $wgEchoEventDetails[$event->getType()]['category'];
|
|
|
|
}
|
|
|
|
|
|
|
|
$email = EchoNotificationController::formatNotification( $event, $this->mUser, 'email' );
|
|
|
|
if ( !isset( $this->content[$category] ) ) {
|
|
|
|
$this->content[$category] = array();
|
|
|
|
}
|
|
|
|
$this->content[$category][] = $email['batch-body'];
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Concatenate the list of contents with 'echo-email-batch-separator'
|
|
|
|
* grouped by category
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
protected function listToText() {
|
|
|
|
if ( !$this->content ) {
|
|
|
|
return '';
|
|
|
|
}
|
|
|
|
|
|
|
|
$result = array();
|
|
|
|
// build the text section for each category
|
|
|
|
foreach( $this->content as $category => $notifs ) {
|
|
|
|
$output = wfMessage( 'echo-email-batch-category-header-' . $category )->params( count( $notifs ) )->text() . "\n";
|
|
|
|
foreach( $notifs as $notif ) {
|
|
|
|
$output .= "\n " . wfMessage( 'echo-email-batch-bullet' )->text() . ' ' . $notif;
|
|
|
|
}
|
|
|
|
$result[] = $output;
|
|
|
|
}
|
|
|
|
|
|
|
|
// for prepending and appending 'echo-email-batch-separator'
|
|
|
|
$result = array_merge( array( '' ), $result, array( '' ) );
|
|
|
|
|
|
|
|
return trim(
|
|
|
|
implode(
|
|
|
|
"\n\n" . wfMessage( 'echo-email-batch-separator' )->text() . "\n\n",
|
|
|
|
$result
|
|
|
|
)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Clear "processed" events in the queue, processed could be: email sent, invalid, users do not want to receive emails
|
|
|
|
*/
|
|
|
|
public function clearProcessedEvent() {
|
|
|
|
$conds = array( 'eeb_user_id' => $this->mUser->getId() );
|
|
|
|
|
|
|
|
// there is a processed cutoff point
|
|
|
|
if ( $this->lastEvent ) {
|
|
|
|
$conds[] = 'eeb_event_id <= ' . intval( $this->lastEvent );
|
|
|
|
}
|
|
|
|
|
|
|
|
$dbw = wfGetDB( DB_MASTER );
|
|
|
|
$dbw->delete(
|
|
|
|
'echo_email_batch',
|
|
|
|
$conds,
|
|
|
|
__METHOD__,
|
|
|
|
array()
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Send the batch email
|
|
|
|
*/
|
|
|
|
public function sendEmail() {
|
|
|
|
global $wgPasswordSender, $wgPasswordSenderName, $wgEchoEmailFooterAddress;
|
|
|
|
|
|
|
|
// global email footer
|
|
|
|
$footer = wfMessage( 'echo-email-footer-default' )
|
|
|
|
->inLanguage( $this->mUser->getOption( 'language' ) )
|
2012-12-20 01:38:35 +00:00
|
|
|
->params( $wgEchoEmailFooterAddress, '' )
|
2012-11-27 01:53:35 +00:00
|
|
|
->text();
|
|
|
|
|
|
|
|
// @Todo - replace them with the CONSTANT in 33810 once it is merged
|
|
|
|
if ( $this->mUser->getOption( 'echo-email-frequency' ) == 7 ) {
|
|
|
|
$frequency = 'weekly';
|
|
|
|
} else {
|
|
|
|
$frequency = 'daily';
|
|
|
|
}
|
|
|
|
|
|
|
|
// email subject
|
|
|
|
if ( $this->count > self::$displaySize ) {
|
|
|
|
$count = wfMessage( 'echo-notification-count' )->params( self::$displaySize )->text();
|
|
|
|
} else {
|
|
|
|
$count = $this->count;
|
|
|
|
}
|
|
|
|
$subject = wfMessage( 'echo-email-batch-subject-' . $frequency )->params( $count, $this->count )->text();
|
|
|
|
$body = wfMessage( 'echo-email-batch-body-' . $frequency )->params(
|
|
|
|
$this->mUser->getName(),
|
|
|
|
$count,
|
|
|
|
$this->count,
|
|
|
|
$this->listToText(),
|
|
|
|
$footer
|
|
|
|
)->text();
|
|
|
|
|
|
|
|
$adminAddress = new MailAddress( $wgPasswordSender, $wgPasswordSenderName );
|
|
|
|
$address = new MailAddress( $this->mUser );
|
|
|
|
|
|
|
|
$params = array(
|
|
|
|
'to' => $address,
|
|
|
|
'from' => $adminAddress,
|
|
|
|
'subj' => $subject,
|
|
|
|
'body' => $body,
|
|
|
|
// no replyto
|
|
|
|
'replyto' => ''
|
|
|
|
);
|
|
|
|
$job = new EmaillingJob( null, $params );
|
|
|
|
JobQueueGroup::singleton()->push( $job );
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Insert notification event into email queue
|
|
|
|
* @param $userId int
|
|
|
|
* @param $eventId int
|
|
|
|
* @param $priority int
|
|
|
|
*/
|
|
|
|
public static function addToQueue( $userId, $eventId, $priority ) {
|
|
|
|
if ( !$userId || !$eventId ) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
$dbw = wfGetDB( DB_MASTER );
|
|
|
|
$dbw->insert(
|
|
|
|
'echo_email_batch',
|
|
|
|
array(
|
|
|
|
'eeb_user_id' => $userId,
|
|
|
|
'eeb_event_id' => $eventId,
|
|
|
|
'eeb_event_priority' => $priority
|
|
|
|
),
|
|
|
|
__METHOD__,
|
|
|
|
array( 'IGNORE' )
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|