hash ( page => (userid => userhash) ) for pages. * Generate notification emails. * * We add WikiDB handlers and register ourself there: * onChangePage, onDeletePage, onRenamePage * Administrative actions: * [Watch] WatchPage - add a page, or delete watch handlers into the users * pref[notifyPages] slot. * My WatchList - view or edit list/regex of pref[notifyPages]. * EMailConfirm methods: send and verify * * Helper functions: * getPageChangeEmails * MailAdmin * ? handle emailed confirmation links (EmailSignup, ModeratedPage) * * @package MailNotify * @author Reini Urban */ if (!defined("MAILER_LOG")) if (isWindows()) define("MAILER_LOG", 'c:/wikimail.log'); else define("MAILER_LOG", '/var/log/wikimail.log'); class MailNotify { function MailNotify($pagename) { $this->pagename = $pagename; /* which page */ $this->emails = array(); /* to which addresses */ $this->userids = array(); /* corresponding array of displayed names, don't display the email addresses */ /* From: from whom the mail appears to be */ $this->from = $this->fromId(); } function fromId() { global $request; if (GFORGE) { return $request->_user->getId(); } else { return $request->_user->getId() . '@' . $request->get('REMOTE_HOST'); } } function userEmail($userid, $doverify = true) { global $request; // Disable verification of emails for corporate env. if (GFORGE) { $doverify = false; } $u = $request->getUser(); if ($u->UserName() == $userid) { // lucky: current user $prefs = $u->getPreferences(); $email = $prefs->get('email'); // do a dynamic emailVerified check update if ($doverify and !$request->_prefs->get('emailVerified')) $email = ''; } else { // not current user if (ENABLE_USER_NEW) { $u = WikiUser($userid); $u->getPreferences(); $prefs = &$u->_prefs; } else { $u = new WikiUser($request, $userid); $prefs = $u->getPreferences(); } $email = $prefs->get('email'); if ($doverify and !$prefs->get('emailVerified')) { $email = ''; } } return $email; } /** * getPageChangeEmails($notify) * @param $notify: hash ( page => (userid => userhash) ) * @return array * unique array of ($emails, $userids) */ function getPageChangeEmails($notify) { global $request; $emails = array(); $userids = array(); foreach ($notify as $page => $users) { if (glob_match($page, $this->pagename)) { global $request; $curuser = $request->getUser(); $curusername = $curuser->UserName(); $curuserprefs = $curuser->getPreferences(); $curuserprefsemail = $curuserprefs->get('email'); $ownModifications = $curuserprefs->get('ownModifications'); $majorModificationsOnly = $curuserprefs->get('majorModificationsOnly'); foreach ($users as $userid => $user) { $usermail = $user['email']; if (($usermail == $curuserprefsemail) and ($ownModifications)) { // It's my own modification // and I do not want to receive it continue; } if ($majorModificationsOnly) { $backend = &$request->_dbi->_backend; $version = $backend->get_latest_version($this->pagename); $versiondata = $backend->get_versiondata($this->pagename, $version, true); if ($versiondata['is_minor_edit']) { // It's a minor modification // and I do not want to receive it continue; } } if (!$user) { // handle the case for ModeratePage: // no prefs, just userid's. $emails[] = $this->userEmail($userid, false); $userids[] = $userid; } else { if (!empty($user['verified']) and !empty($user['email'])) { $emails[] = $user['email']; $userids[] = $userid; } elseif (!empty($user['email'])) { // do a dynamic emailVerified check update $email = $this->userEmail($userid, true); if ($email) { $notify[$page][$userid]['verified'] = 1; $request->_dbi->set('notify', $notify); $emails[] = $email; $userids[] = $userid; } } // ignore verification /* if (DEBUG) { if (!in_array($user['email'], $emails)) $emails[] = $user['email']; } */ } } } } $this->emails = array_unique($emails); $this->userids = array_unique($userids); return array($this->emails, $this->userids); } function sendMail($subject, $content, $notice = false, $silent = true) { // Add WIKI_NAME to Subject $subject = "[".WIKI_NAME."] ".$subject; // Encode $subject if needed $encoded_subject = $this->subject_encode($subject); $emails = $this->emails; $from = $this->from; // Do not send if modification is from Gforge admin if (GFORGE and $from == ADMIN_USER) { return; } if (!$notice) $notice = _("PageChange Notification of %s"); $headers = "From: $from\r\n" . "Bcc: ".join(',', $emails)."\r\n" . "MIME-Version: 1.0\r\n" . "Content-Type: text/plain; charset=".CHARSET."; format=flowed\r\n" . "Content-Transfer-Encoding: 8bit"; $ok = mail(($to = array_shift($emails)), $encoded_subject, $subject."\n".$content, $headers ); if (MAILER_LOG and is_writable(MAILER_LOG)) { $f = fopen(MAILER_LOG, "a"); fwrite($f, "\n\nX-MailSentOK: " . $ok ? 'OK' : 'FAILED'); if (!$ok) { global $ErrorManager; // get last error message $last_err = $ErrorManager->_postponed_errors[count($ErrorHandler->_postponed_errors)-1]; fwrite($f, "\nX-MailFailure: " . $last_err); } fwrite($f, "\nDate: " . CTime()); fwrite($f, "\nSubject: $encoded_subject"); fwrite($f, "\nFrom: $from"); fwrite($f, "\nTo: $to"); fwrite($f, "\nBcc: ".join(',', $emails)); fwrite($f, "\n\n". $content); fclose($f); } if ($ok) { if (!$silent) trigger_error(sprintf($notice, $this->pagename) . " " . sprintf(_("sent to %s"), join(',',$this->userids)), E_USER_NOTICE); return true; } else { trigger_error(sprintf($notice, $this->pagename) . " " . sprintf(_("Error: Couldn't send %s to %s"), $subject."\n".$content, join(',',$this->userids)), E_USER_WARNING); return false; } } /** * Send udiff for a changed page to multiple users. * See rename and remove methods also */ function sendPageChangeNotification(&$wikitext, $version, &$meta) { global $request; if (@is_array($request->_deferredPageChangeNotification)) { // collapse multiple changes (loaddir) into one email $request->_deferredPageChangeNotification[] = array($this->pagename, $this->emails, $this->userids); return; } $backend = &$request->_dbi->_backend; $subject = _("Page change").' '.($this->pagename); $previous = $backend->get_previous_version($this->pagename, $version); if (!isset($meta['mtime'])) $meta['mtime'] = time(); if ($previous) { $difflink = WikiURL($this->pagename, array('action'=>'diff'), true); $cache = &$request->_dbi->_cache; $this_content = explode("\n", $wikitext); $prevdata = $cache->get_versiondata($this->pagename, $previous, true); if (empty($prevdata['%content'])) $prevdata = $backend->get_versiondata($this->pagename, $previous, true); $other_content = explode("\n", $prevdata['%content']); include_once("lib/difflib.php"); $diff2 = new Diff($other_content, $this_content); //$context_lines = max(4, count($other_content) + 1, // count($this_content) + 1); $fmt = new UnifiedDiffFormatter(/*$context_lines*/); $content = $this->pagename . " " . $previous . " " . Iso8601DateTime($prevdata['mtime']) . "\n"; $content .= $this->pagename . " " . $version . " " . Iso8601DateTime($meta['mtime']) . "\n"; $content .= $fmt->format($diff2); } else { $difflink = WikiURL($this->pagename,array(),true); $content = $this->pagename . " " . $version . " " . Iso8601DateTime($meta['mtime']) . "\n"; $content .= _("New page"); $content .= "\n\n"; $content .= $wikitext; } $editedby = sprintf(_("Edited by: %s"), $this->from); $summary = sprintf(_("Summary: %s"), $meta['summary']); $this->sendMail($subject, $editedby."\n".$summary."\n".$difflink."\n\n".$content); } /** * Support mass rename / remove (not yet tested) */ function sendPageRenameNotification ($to, &$meta) { global $request; if (@is_array($request->_deferredPageRenameNotification)) { $request->_deferredPageRenameNotification[] = array($this->pagename, $to, $meta, $this->emails, $this->userids); } else { $pagename = $this->pagename; $editedby = sprintf(_("Edited by: %s"), $this->from); $subject = sprintf(_("Page rename %s to %s"), $pagename, $to); $link = WikiURL($to, true); $this->sendMail($subject, $editedby."\n".$link."\n\n"."Renamed $pagename to $to"); } } /** * The handlers: */ function onChangePage (&$wikidb, &$wikitext, $version, &$meta) { $result = true; if (!isa($GLOBALS['request'],'MockRequest')) { $notify = $wikidb->get('notify'); /* Generate notification emails? */ if (!empty($notify) and is_array($notify)) { if (empty($this->pagename)) $this->pagename = $meta['pagename']; // TODO: Should be used for ModeratePage and RSS2 Cloud xml-rpc also. $this->getPageChangeEmails($notify); if (!empty($this->emails)) { $result = $this->sendPageChangeNotification($wikitext, $version, $meta); } } } return $result; } function onDeletePage (&$wikidb, $pagename) { $result = true; /* Generate notification emails? */ if (! $wikidb->isWikiPage($pagename) and !isa($GLOBALS['request'],'MockRequest')) { $notify = $wikidb->get('notify'); if (!empty($notify) and is_array($notify)) { //TODO: deferr it (quite a massive load if you remove some pages). $this->getPageChangeEmails($notify); if (!empty($this->emails)) { $subject = sprintf(_("User %s removed page %s"), $this->from, $pagename); $result = $this->sendMail($subject, $subject."\n\n"); } } } return $result; } function onRenamePage (&$wikidb, $oldpage, $new_pagename) { $result = true; if (!isa($GLOBALS['request'], 'MockRequest')) { $notify = $wikidb->get('notify'); if (!empty($notify) and is_array($notify)) { $this->getPageChangeEmails($notify); if (!empty($this->emails)) { $newpage = $wikidb->getPage($new_pagename); $current = $newpage->getCurrentRevision(); $meta = $current->_data; $this->pagename = $oldpage; $result = $this->sendPageRenameNotification($new_pagename, $meta); } } } } function subject_encode ($subject) { // We need to encode the subject if it contains non-ASCII characters // The page name may contain non-ASCII characters, as well as // the translation of the messages, e.g. _("PageChange Notification of %s"); // If all characters are ASCII, do nothing if (isAsciiString($subject)) { return $subject; } // Let us try quoted printable first if (function_exists('quoted_printable_encode')) { // PHP 5.3 return "=?UTF-8?Q?".quoted_printable_encode($subject)."?="; } // If not, encode in base64 (less human-readable) return "=?UTF-8?B?".base64_encode($subject)."?="; } } // Local Variables: // mode: php // tab-width: 8 // c-basic-offset: 4 // c-hanging-comment-ender-p: nil // indent-tabs-mode: nil // End: ?>