3 use MediaWiki\Maintenance\Maintenance
;
4 use MediaWiki\Page\MovePageFactory
;
5 use MediaWiki\RenameUser\RenameuserSQL
;
6 use MediaWiki\Status\Status
;
7 use MediaWiki\Title\TitleFactory
;
8 use MediaWiki\User\TempUser\Pattern
;
9 use MediaWiki\User\User
;
10 use MediaWiki\User\UserFactory
;
11 use Wikimedia\Rdbms\IExpression
;
13 // @codeCoverageIgnoreStart
14 require_once __DIR__
. '/Maintenance.php';
15 // @codeCoverageIgnoreEnd
17 class RenameUsersMatchingPattern
extends Maintenance
{
18 /** @var UserFactory */
21 /** @var MovePageFactory */
22 private $movePageFactory;
24 /** @var TitleFactory */
25 private $titleFactory;
37 private $suppressRedirect;
40 private $skipPageMoves;
42 public function __construct() {
43 parent
::__construct();
45 $this->addDescription( 'Rename users with a name matching a pattern. ' .
46 'This can be used to migrate to a temporary user (IP masking) configuration.' );
47 $this->addOption( 'from', 'A username pattern where $1 is ' .
48 'the wildcard standing in for any number of characters. All users ' .
49 'matching this pattern will be renamed.', true, true );
50 $this->addOption( 'to', 'A username pattern where $1 is ' .
51 'the part of the username matched by $1 in --from. Users will be ' .
52 ' renamed to this pattern.', true, true );
53 $this->addOption( 'performer', 'Performer of the rename action', false, true );
54 $this->addOption( 'reason', 'Reason of the rename', false, true );
55 $this->addOption( 'suppress-redirect', 'Don\'t create redirects when moving pages' );
56 $this->addOption( 'skip-page-moves', 'Don\'t move associated user pages' );
57 $this->addOption( 'dry-run', 'Don\'t actually rename the ' .
58 'users, just report what it would do.' );
59 $this->setBatchSize( 1000 );
62 private function initServices() {
63 $services = $this->getServiceContainer();
64 if ( $services->getCentralIdLookupFactory()->getNonLocalLookup() ) {
65 $this->fatalError( "This script cannot be run when CentralAuth is enabled." );
67 $this->userFactory
= $services->getUserFactory();
68 $this->movePageFactory
= $services->getMovePageFactory();
69 $this->titleFactory
= $services->getTitleFactory();
72 public function execute() {
73 $this->initServices();
75 $fromPattern = new Pattern( 'from', $this->getOption( 'from' ) );
76 $toPattern = new Pattern( 'to', $this->getOption( 'to' ) );
78 if ( $this->getOption( 'performer' ) === null ) {
79 $performer = User
::newSystemUser( User
::MAINTENANCE_SCRIPT_USER
, [ 'steal' => true ] );
81 $performer = $this->userFactory
->newFromName( $this->getOption( 'performer' ) );
84 $this->fatalError( "Unable to get performer account" );
86 $this->performer
= $performer;
88 $this->reason
= $this->getOption( 'reason', '' );
89 $this->dryRun
= $this->getOption( 'dry-run' );
90 $this->suppressRedirect
= $this->getOption( 'suppress-redirect' );
91 $this->skipPageMoves
= $this->getOption( 'skip-page-moves' );
93 $dbr = $this->getReplicaDB();
95 $batchSize = $this->getBatchSize();
98 $res = $dbr->newSelectQueryBuilder()
99 ->select( [ 'user_name' ] )
101 ->where( $dbr->expr( 'user_name', IExpression
::LIKE
, $fromPattern->toLikeValue( $dbr ) ) )
102 ->andWhere( $batchConds )
103 ->orderBy( 'user_name' )
104 ->limit( $batchSize )
105 ->caller( __METHOD__
)
108 foreach ( $res as $row ) {
109 $oldName = $row->user_name
;
110 $batchConds = [ $dbr->expr( 'user_name', '>', $oldName ) ];
111 $variablePart = $fromPattern->extract( $oldName );
112 if ( $variablePart === null ) {
113 $this->output( "Username \"fromName\" matched the LIKE " .
114 "but does not seem to match the pattern" );
117 $newName = $toPattern->generate( $variablePart );
120 $newTitle = $this->titleFactory
->makeTitleSafe( NS_USER
, $newName );
121 $newUser = $this->userFactory
->newFromName( $newName );
122 if ( !$newTitle ||
!$newUser ) {
123 $this->output( "Cannot rename \"$oldName\" " .
124 "because \"$newName\" is not a valid title\n" );
127 $newName = $newTitle->getText();
129 // Check destination existence
130 if ( $newUser->isRegistered() ) {
131 $this->output( "Cannot rename \"$oldName\" " .
132 "because \"$newName\" already exists\n" );
136 $numRenamed +
= $this->renameUser( $oldName, $newName ) ?
1 : 0;
137 $this->waitForReplication();
139 } while ( $res->numRows() === $batchSize );
140 $this->output( "Renamed $numRenamed user(s)\n" );
145 * @param string $oldName
146 * @param string $newName
147 * @return bool True if the user was renamed
149 private function renameUser( $oldName, $newName ) {
150 $id = $this->userFactory
->newFromName( $oldName )->getId();
152 $this->output( "Cannot rename non-existent user \"$oldName\"" );
155 if ( $this->dryRun
) {
156 $this->output( "$oldName would be renamed to $newName\n" );
158 $renamer = new RenameuserSQL(
164 'reason' => $this->reason
168 if ( !$renamer->rename() ) {
169 $this->output( "Unable to rename $oldName" );
172 $this->output( "$oldName was successfully renamed to $newName.\n" );
176 if ( $this->skipPageMoves
) {
180 $this->movePageAndSubpages( NS_USER
, 'User', $oldName, $newName );
181 $this->movePageAndSubpages( NS_USER_TALK
, 'User talk', $oldName, $newName );
185 private function movePageAndSubpages( $ns, $nsName, $oldName, $newName ) {
186 $oldTitle = $this->titleFactory
->makeTitleSafe( $ns, $oldName );
188 $this->output( "[[$nsName:$oldName]] is an invalid title, can't move it.\n" );
191 $newTitle = $this->titleFactory
->makeTitleSafe( $ns, $newName );
193 $this->output( "[[$nsName:$newName]] is an invalid title, can't move to it.\n" );
197 $movePage = $this->movePageFactory
->newMovePage( $oldTitle, $newTitle );
198 $movePage->setMaximumMovedPages( -1 );
200 $logMessage = wfMessage(
201 'renameuser-move-log', $oldName, $newName
202 )->inContentLanguage()->text();
204 if ( $this->dryRun
) {
205 if ( $oldTitle->exists() ) {
206 $this->output( "Would move [[$nsName:$oldName]] to [[$nsName:$newName]].\n" );
209 if ( $oldTitle->exists() ) {
210 $status = $movePage->move(
211 $this->performer
, $logMessage, !$this->suppressRedirect
);
213 $status = Status
::newGood();
215 $status->merge( $movePage->moveSubpages(
216 $this->performer
, $logMessage, !$this->suppressRedirect
) );
217 if ( !$status->isGood() ) {
218 $this->output( "Failed to rename user page\n" );
219 $this->error( $status );
226 // @codeCoverageIgnoreStart
227 $maintClass = RenameUsersMatchingPattern
::class;
228 require_once RUN_MAINTENANCE_IF_MAIN
;
229 // @codeCoverageIgnoreEnd