Optimize email sending on password reset
authorsuecarmol <scardenasmolinar@wikimedia.org>
Wed, 8 Apr 2020 00:13:54 +0000 (19:13 -0500)
committerReedy <reedy@wikimedia.org>
Mon, 20 Apr 2020 19:06:30 +0000 (20:06 +0100)
Improve performance of sending emails when a user resets a password.

Bug: T247017
Change-Id: I9edb0e4c8845f7a9082035de66f5965c3f9b762d

autoload.php
includes/deferred/SendPasswordResetEmailUpdate.php [new file with mode: 0644]
includes/user/PasswordReset.php

index b306bf5..28d8926 100644 (file)
@@ -1366,6 +1366,7 @@ $wgAutoloadLocalClasses = [
        'SearchUpdate' => __DIR__ . '/includes/deferred/SearchUpdate.php',
        'SectionProfileCallback' => __DIR__ . '/includes/profiler/SectionProfiler.php',
        'SectionProfiler' => __DIR__ . '/includes/profiler/SectionProfiler.php',
+       'SendPasswordResetEmailUpdate' => __DIR__ . '/includes/deferred/SendPasswordResetEmailUpdate.php',
        'SevenZipStream' => __DIR__ . '/maintenance/7zip.inc',
        'ShiConverter' => __DIR__ . '/languages/classes/LanguageShi.php',
        'ShortPagesPage' => __DIR__ . '/includes/specials/SpecialShortpages.php',
diff --git a/includes/deferred/SendPasswordResetEmailUpdate.php b/includes/deferred/SendPasswordResetEmailUpdate.php
new file mode 100644 (file)
index 0000000..710f303
--- /dev/null
@@ -0,0 +1,64 @@
+<?php
+/**
+ * Send an email to reset the password
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License along
+ * with this program; if not, write to the Free Software Foundation, Inc.,
+ * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+ * http://www.gnu.org/copyleft/gpl.html
+ *
+ * @file
+ */
+
+use MediaWiki\Auth\AuthManager;
+use MediaWiki\Logger\LoggerFactory;
+
+/**
+ * Sends emails to all accounts associated with that email to reset the password
+ * @since 1.35
+ */
+class SendPasswordResetEmailUpdate implements DeferrableUpdate {
+       /** @var AuthManager */
+       private $authManager;
+
+       /** @var array */
+       private $reqs;
+
+       /** @var array */
+       private $logContext;
+
+       /**
+        * @param AuthManager $authManager
+        * @param array $reqs
+        * @param array $logContext
+        */
+       public function __construct( AuthManager $authManager, array $reqs, array $logContext ) {
+               $this->authManager = $authManager;
+               $this->reqs = $reqs;
+               $this->logContext = $logContext;
+       }
+
+       public function doUpdate() {
+               $logger = LoggerFactory::getInstance( 'authentication' );
+               foreach ( $this->reqs as $req ) {
+                       // This is adding a new temporary password, not intentionally changing anything
+                       // (even though it might technically invalidate an old temporary password).
+                       $this->authManager->changeAuthenticationData( $req, /* $isAddition */ true );
+                       $logger->info(
+                               "{requestingUser} did password reset of {targetUser} and an email was sent",
+                               $this->logContext + [ 'targetUser' => $req->username ]
+                       );
+               }
+       }
+
+}
index a1638ea..73e5cbc 100644 (file)
@@ -226,7 +226,6 @@ class PasswordReset implements LoggerAwareInterface {
                        'requestingUser' => $performingUser->getName(),
                        'targetUsername' => $username,
                        'targetEmail' => $email,
-                       'actualUser' => $firstUser->getName(),
                ];
 
                if ( !$result->isGood() ) {
@@ -237,15 +236,9 @@ class PasswordReset implements LoggerAwareInterface {
                        return $result;
                }
 
-               foreach ( $reqs as $req ) {
-                       // This is adding a new temporary password, not intentionally changing anything
-                       // (even though it might technically invalidate an old temporary password).
-                       $this->authManager->changeAuthenticationData( $req, /* $isAddition */ true );
-               }
-
-               $this->logger->info(
-                       "{requestingUser} did password reset of {actualUser}",
-                       $logContext
+               DeferredUpdates::addUpdate(
+                       new SendPasswordResetEmailUpdate( $this->authManager, $reqs, $logContext ),
+                       DeferredUpdates::POSTSEND
                );
 
                return StatusValue::newGood();