MediaWiki master
EmailNotification.php
Go to the documentation of this file.
1<?php
37
64
65 protected string $pageStatus = '';
66
77 public function getPageStatus() {
78 return $this->pageStatus;
79 }
80
92 public function notifyOnPageChange(
93 RecentChange $recentChange
94 ): bool {
95 $mwServices = MediaWikiServices::getInstance();
96 $editor = $mwServices->getUserFactory()
97 ->newFromUserIdentity( $recentChange->getPerformerIdentity() );
98
99 $title = Title::castFromPageReference( $recentChange->getPage() );
100 if ( $title === null || $title->getNamespace() < 0 ) {
101 return false;
102 }
103
104 $timestamp = $recentChange->mAttribs['rc_timestamp'];
105 $summary = $recentChange->mAttribs['rc_comment'];
106 $minorEdit = $recentChange->mAttribs['rc_minor'];
107 $oldid = $recentChange->mAttribs['rc_last_oldid'];
108 $pageStatus = $recentChange->mExtra['pageStatus'] ?? 'changed';
109
110 $config = $mwServices->getMainConfig();
111
112 / update wl_notificationtimestamp for watchers
113 $watchers = [];
114 if ( $config->get( MainConfigNames::EnotifWatchlist ) || $config->get( MainConfigNames::ShowUpdatedMarker ) ) {
115 $watchers = $mwServices->getWatchedItemStore()->updateNotificationTimestamp(
116 $editor,
117 $title,
118 $timestamp
119 );
120 }
121
122 / Don't send email for bots
123 if ( $editor->isBot() ) {
124 return false;
125 }
126
127 $sendNotification = true;
128 / $watchers deals with $wgEnotifWatchlist.
129 / If nobody is watching the page, and there are no users notified on all changes
130 / don't bother creating a job/trying to send emails, unless it's a
131 / talk page with an applicable notification.
132 if ( $watchers === [] &&
133 !count( $config->get( MainConfigNames::UsersNotifiedOnAllChanges ) )
134 ) {
135 $sendNotification = false;
136 / Only send notification for non minor edits, unless $wgEnotifMinorEdits
137 if ( !$minorEdit ||
138 ( $config->get( MainConfigNames::EnotifMinorEdits ) &&
139 !$editor->isAllowed( 'nominornewtalk' ) )
140 ) {
141 if ( $config->get( MainConfigNames::EnotifUserTalk )
142 && $title->getNamespace() === NS_USER_TALK
143 && $this->canSendUserTalkEmail( $editor, $title, $minorEdit )
144 ) {
145 $sendNotification = true;
146 }
147 }
148 }
149
150 if ( $sendNotification ) {
151 $mwServices->getJobQueueGroup()->lazyPush( new EnotifNotifyJob(
152 $title,
153 [
154 'editor' => $editor->getName(),
155 'editorID' => $editor->getId(),
156 'timestamp' => $timestamp,
157 'summary' => $summary,
158 'minorEdit' => $minorEdit,
159 'oldid' => $oldid,
160 'watchers' => $watchers,
161 'pageStatus' => $pageStatus,
162 / not used yet, passed to support T388663 and T389618 in the future
163 'rc_id' => $recentChange->getAttribute( 'rc_id' ),
164 ]
165 ) );
166 }
167
168 return $sendNotification;
169 }
170
189 Authority $editor,
190 $title,
191 $timestamp,
192 $summary,
193 $minorEdit,
194 $oldid,
195 $watchers,
196 $pageStatus = 'changed'
197 ) {
198 # we use $wgPasswordSender as sender's address
199 $mwServices = MediaWikiServices::getInstance();
200 $config = $mwServices->getMainConfig();
201 $notifService = $mwServices->getNotificationService();
202 $userFactory = $mwServices->getUserFactory();
203
204 # The following code is only run, if several conditions are met:
205 # 1. EmailNotification for pages (other than user_talk pages) must be enabled
206 # 2. minor edits (changes) are only regarded if the global flag indicates so
207 $this->pageStatus = $pageStatus;
208
209 $formattedPageStatus = [ 'deleted', 'created', 'moved', 'restored', 'changed' ];
210
211 $hookRunner = new HookRunner( $mwServices->getHookContainer() );
212 $hookRunner->onUpdateUserMailerFormattedPageStatus( $formattedPageStatus );
213 if ( !in_array( $this->pageStatus, $formattedPageStatus ) ) {
214 throw new UnexpectedValueException( 'Not a valid page status!' );
215 }
216
217 $composer = new RecentChangeMailComposer(
218 $editor,
219 $title,
220 $summary,
221 $minorEdit,
222 $oldid,
223 $timestamp,
224 $pageStatus
225 );
226
227 $userTalkId = false;
228
229 if ( !$minorEdit ||
230 ( $config->get( MainConfigNames::EnotifMinorEdits ) &&
231 !$editor->isAllowed( 'nominornewtalk' ) )
232 ) {
233 if ( $config->get( MainConfigNames::EnotifUserTalk )
234 && $title->getNamespace() === NS_USER_TALK
235 && $this->canSendUserTalkEmail( $editor->getUser(), $title, $minorEdit )
236 ) {
237 $targetUser = $userFactory->newFromName( $title->getText() );
238 if ( $targetUser ) {
239 $talkNotification = new RecentChangeNotification(
240 $mwServices->getUserFactory()->newFromAuthority( $editor ),
241 $title,
242 $summary,
243 $minorEdit,
244 $oldid,
245 $timestamp,
246 $pageStatus,
247 RecentChangeNotification::TALK_NOTIFICATION
248 );
249 $notifService->notify( $talkNotification, new RecipientSet( [ $targetUser ] ) );
250 $userTalkId = $targetUser->getId();
251 }
252 }
253
254 if ( $config->get( MainConfigNames::EnotifWatchlist ) ) {
255 $userOptionsLookup = $mwServices->getUserOptionsLookup();
256 / Send updates to watchers other than the current editor
257 / and don't send to watchers who are blocked and cannot login
258 $userArray = UserArray::newFromIDs( $watchers );
259 foreach ( $userArray as $watchingUser ) {
260 if ( $userOptionsLookup->getOption( $watchingUser, 'enotifwatchlistpages' )
261 && ( !$minorEdit || $userOptionsLookup->getOption( $watchingUser, 'enotifminoredits' ) )
262 && $watchingUser->isEmailConfirmed()
263 && $watchingUser->getId() != $userTalkId
264 && !in_array( $watchingUser->getName(),
265 $config->get( MainConfigNames::UsersNotifiedOnAllChanges ) )
266 / @TODO Partial blocks should not prevent the user from logging in.
267 / see: https://phabricator.wikimedia.org/T208895
268 && !( $config->get( MainConfigNames::BlockDisablesLogin ) &&
269 $watchingUser->getBlock() )
270 && $hookRunner->onSendWatchlistEmailNotification( $watchingUser, $title, $this )
271 ) {
272 $composer->compose( $watchingUser, RecentChangeMailComposer::WATCHLIST );
273 }
274 }
275 }
276 }
277
278 foreach ( $config->get( MainConfigNames::UsersNotifiedOnAllChanges ) as $name ) {
279 $admins = [];
280 if ( $editor->getUser()->getName() == $name ) {
281 / No point notifying the user that actually made the change!
282 continue;
283 }
284 $user = $userFactory->newFromName( $name );
285 if ( $user instanceof User ) {
286 $admins[] = $user;
287 }
288 $notifService->notify(
290 $mwServices->getUserFactory()->newFromAuthority( $editor ),
291 $title,
292 $summary,
293 $minorEdit,
294 $oldid,
295 $timestamp,
296 $pageStatus,
297 RecentChangeNotification::ADMIN_NOTIFICATION
298 ),
299 new RecipientSet( $admins )
300 );
301
302 }
303 $composer->sendMails();
304 }
305
312 private function canSendUserTalkEmail( UserIdentity $editor, $title, $minorEdit ) {
313 $services = MediaWikiServices::getInstance();
314 $config = $services->getMainConfig();
315
316 if ( !$config->get( MainConfigNames::EnotifUserTalk ) || $title->getNamespace() !== NS_USER_TALK ) {
317 return false;
318 }
319
320 $userOptionsLookup = $services->getUserOptionsLookup();
321 $targetUser = User::newFromName( $title->getText() );
322
323 if ( !$targetUser || $targetUser->isAnon() ) {
324 wfDebug( __METHOD__ . ": user talk page edited, but user does not exist" );
325 } elseif ( $targetUser->getId() == $editor->getId() ) {
326 wfDebug( __METHOD__ . ": user edited their own talk page, no notification sent" );
327 } elseif ( $targetUser->isTemp() ) {
328 wfDebug( __METHOD__ . ": talk page owner is a temporary user so doesn't have email" );
329 } elseif ( $config->get( MainConfigNames::BlockDisablesLogin ) &&
330 $targetUser->getBlock()
331 ) {
332 / @TODO Partial blocks should not prevent the user from logging in.
333 / see: https://phabricator.wikimedia.org/T208895
334 wfDebug( __METHOD__ . ": talk page owner is blocked and cannot login, no notification sent" );
335 } elseif ( $userOptionsLookup->getOption( $targetUser, 'enotifusertalkpages' )
336 && ( !$minorEdit || $userOptionsLookup->getOption( $targetUser, 'enotifminoredits' ) )
337 ) {
338 if ( !$targetUser->isEmailConfirmed() ) {
339 wfDebug( __METHOD__ . ": talk page owner doesn't have validated email" );
340 } elseif ( !( new HookRunner( $services->getHookContainer() ) )
341 ->onAbortTalkPageEmailNotification( $targetUser, $title )
342 ) {
343 wfDebug( __METHOD__ . ": talk page update notification is aborted for this user" );
344 } else {
345 wfDebug( __METHOD__ . ": sending talk page update notification" );
346 return true;
347 }
348 } else {
349 wfDebug( __METHOD__ . ": talk page owner doesn't want notifications" );
350 }
351 return false;
352 }
353
354}
const NS_USER_TALK
Definition Defines.php:68
wfDebug( $text, $dest='all', array $context=[])
Sends a line to the debug log if enabled or, optionally, to a comment in output.
Find watchers and create email notifications after a page is changed.
notifyOnPageChange(RecentChange $recentChange)
Send emails corresponding to the user $editor editing the page $title.
actuallyNotifyOnPageChange(Authority $editor, $title, $timestamp, $summary, $minorEdit, $oldid, $watchers, $pageStatus='changed')
Immediate version of notifyOnPageChange().
getPageStatus()
Extensions that have hooks for UpdateUserMailerFormattedPageStatus (to provide additional pageStatus ...
Send an email notification.
This class provides an implementation of the core hook interfaces, forwarding hook calls to HookConta...
Component responsible for composing and sending emails triggered after a RecentChange.
A class containing constants representing the names of configuration variables.
Service locator for MediaWiki core services.
Utility class for creating and reading rows in the recentchanges table.
getPerformerIdentity()
Get the UserIdentity of the client that performed this change.
getAttribute( $name)
Get an attribute value.
Represents a title within MediaWiki.
Definition Title.php:78
getOption(UserIdentity $user, string $oname, $defaultOverride=null, bool $ignoreHidden=false, int $queryFlags=IDBAccessObject::READ_NORMAL)
Get the user's current setting for a given option.
Class to walk into a list of User objects.
Definition UserArray.php:33
User class for the MediaWiki software.
Definition User.php:120
Notification representing a Recent change.
This interface represents the authority associated with the current execution context,...
Definition Authority.php:37
isAllowed(string $permission, ?PermissionStatus $status=null)
Checks whether this authority has the given permission in general.
getUser()
Returns the performer of the actions associated with this authority.
Interface for objects representing user identity.
getId( $wikiId=self::LOCAL)

Follow Lee on X/Twitter - Father, Husband, Serial builder creating AI, crypto, games & web tools. We are friends :) AI Will Come To Life!

Check out: eBank.nz (Art Generator) | Netwrck.com (AI Tools) | Text-Generator.io (AI API) | BitBank.nz (Crypto AI) | ReadingTime (Kids Reading) | RewordGame | BigMultiplayerChess | WebFiddle | How.nz | Helix AI Assistant