MediaWiki master
renameUsersMatchingPattern.php
Go to the documentation of this file.
1<?php
2
11
12// @codeCoverageIgnoreStart
13require_once __DIR__ . '/Maintenance.php';
14// @codeCoverageIgnoreEnd
15
18 private $userFactory;
19
21 private $movePageFactory;
22
24 private $titleFactory;
25
27 private $performer;
28
30 private $reason;
31
33 private $dryRun;
34
36 private $suppressRedirect;
37
39 private $skipPageMoves;
40
41 public function __construct() {
42 parent::__construct();
43
44 $this->addDescription( 'Rename users with a name matching a pattern. ' .
45 'This can be used to migrate to a temporary user (IP masking) configuration.' );
46 $this->addOption( 'from', 'A username pattern where $1 is ' .
47 'the wildcard standing in for any number of characters. All users ' .
48 'matching this pattern will be renamed.', true, true );
49 $this->addOption( 'to', 'A username pattern where $1 is ' .
50 'the part of the username matched by $1 in --from. Users will be ' .
51 ' renamed to this pattern.', true, true );
52 $this->addOption( 'performer', 'Performer of the rename action', false, true );
53 $this->addOption( 'reason', 'Reason of the rename', false, true );
54 $this->addOption( 'suppress-redirect', 'Don\'t create redirects when moving pages' );
55 $this->addOption( 'skip-page-moves', 'Don\'t move associated user pages' );
56 $this->addOption( 'dry-run', 'Don\'t actually rename the ' .
57 'users, just report what it would do.' );
58 $this->setBatchSize( 1000 );
59 }
60
61 private function initServices() {
62 $services = $this->getServiceContainer();
63 if ( $services->getCentralIdLookupFactory()->getNonLocalLookup() ) {
64 $this->fatalError( "This script cannot be run when CentralAuth is enabled." );
65 }
66 $this->userFactory = $services->getUserFactory();
67 $this->movePageFactory = $services->getMovePageFactory();
68 $this->titleFactory = $services->getTitleFactory();
69 }
70
71 public function execute() {
72 $this->initServices();
73
74 $fromPattern = new Pattern( 'from', $this->getOption( 'from' ) );
75 $toPattern = new Pattern( 'to', $this->getOption( 'to' ) );
76
77 if ( $this->getOption( 'performer' ) === null ) {
78 $performer = User::newSystemUser( User::MAINTENANCE_SCRIPT_USER, [ 'steal' => true ] );
79 } else {
80 $performer = $this->userFactory->newFromName( $this->getOption( 'performer' ) );
81 }
82 if ( !$performer ) {
83 $this->error( "Unable to get performer account" );
84 return false;
85 }
86 $this->performer = $performer;
87
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' );
92
93 $dbr = $this->getReplicaDB();
94 $batchConds = [];
95 $batchSize = $this->getBatchSize();
96 $numRenamed = 0;
97 do {
98 $res = $dbr->newSelectQueryBuilder()
99 ->select( [ 'user_name' ] )
100 ->from( 'user' )
101 ->where( $dbr->expr( 'user_name', IExpression::LIKE, $fromPattern->toLikeValue( $dbr ) ) )
102 ->andWhere( $batchConds )
103 ->orderBy( 'user_name' )
104 ->limit( $batchSize )
105 ->caller( __METHOD__ )
106 ->fetchResultSet();
107
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" );
115 continue;
116 }
117 $newName = $toPattern->generate( $variablePart );
118
119 // Canonicalize
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" );
125 continue;
126 }
127 $newName = $newTitle->getText();
128
129 // Check destination existence
130 if ( $newUser->isRegistered() ) {
131 $this->output( "Cannot rename \"$oldName\" " .
132 "because \"$newName\" already exists\n" );
133 continue;
134 }
135
136 $numRenamed += $this->renameUser( $oldName, $newName ) ? 1 : 0;
137 $this->waitForReplication();
138 }
139 } while ( $res->numRows() === $batchSize );
140 $this->output( "Renamed $numRenamed user(s)\n" );
141 return true;
142 }
143
149 private function renameUser( $oldName, $newName ) {
150 $id = $this->userFactory->newFromName( $oldName )->getId();
151 if ( !$id ) {
152 $this->output( "Cannot rename non-existent user \"$oldName\"" );
153 }
154
155 if ( $this->dryRun ) {
156 $this->output( "$oldName would be renamed to $newName\n" );
157 } else {
158 $renamer = new RenameuserSQL(
159 $oldName,
160 $newName,
161 $id,
162 $this->performer,
163 [
164 'reason' => $this->reason
165 ]
166 );
167
168 if ( !$renamer->rename() ) {
169 $this->output( "Unable to rename $oldName" );
170 return false;
171 } else {
172 $this->output( "$oldName was successfully renamed to $newName.\n" );
173 }
174 }
175
176 if ( $this->skipPageMoves ) {
177 return true;
178 }
179
180 $this->movePageAndSubpages( NS_USER, 'User', $oldName, $newName );
181 $this->movePageAndSubpages( NS_USER_TALK, 'User talk', $oldName, $newName );
182 return true;
183 }
184
185 private function movePageAndSubpages( $ns, $nsName, $oldName, $newName ) {
186 $oldTitle = $this->titleFactory->makeTitleSafe( $ns, $oldName );
187 if ( !$oldTitle ) {
188 $this->output( "[[$nsName:$oldName]] is an invalid title, can't move it.\n" );
189 return true;
190 }
191 $newTitle = $this->titleFactory->makeTitleSafe( $ns, $newName );
192 if ( !$newTitle ) {
193 $this->output( "[[$nsName:$newName]] is an invalid title, can't move to it.\n" );
194 return true;
195 }
196
197 $movePage = $this->movePageFactory->newMovePage( $oldTitle, $newTitle );
198 $movePage->setMaximumMovedPages( -1 );
199
200 $logMessage = wfMessage(
201 'renameuser-move-log', $oldName, $newName
202 )->inContentLanguage()->text();
203
204 if ( $this->dryRun ) {
205 if ( $oldTitle->exists() ) {
206 $this->output( "Would move [[$nsName:$oldName]] to [[$nsName:$newName]].\n" );
207 }
208 } else {
209 if ( $oldTitle->exists() ) {
210 $status = $movePage->move(
211 $this->performer, $logMessage, !$this->suppressRedirect );
212 } else {
213 $status = Status::newGood();
214 }
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 );
220 }
221 }
222 return true;
223 }
224}
225
226// @codeCoverageIgnoreStart
227$maintClass = RenameUsersMatchingPattern::class;
228require_once RUN_MAINTENANCE_IF_MAIN;
229// @codeCoverageIgnoreEnd
const NS_USER
Definition Defines.php:67
const NS_USER_TALK
Definition Defines.php:68
wfMessage( $key,... $params)
This is the function for getting translated interface messages.
Abstract maintenance class for quickly writing and churning out maintenance scripts with minimal effo...
error( $err, $die=0)
Throw an error to the user.
output( $out, $channel=null)
Throw some output to the user.
waitForReplication()
Wait for replica DBs to catch up.
getServiceContainer()
Returns the main service container.
getBatchSize()
Returns batch size.
addDescription( $text)
Set the description text.
addOption( $name, $description, $required=false, $withArg=false, $shortName=false, $multiOccurrence=false)
Add a parameter to the script.
getOption( $name, $default=null)
Get an option, or return the default.
setBatchSize( $s=0)
fatalError( $msg, $exitCode=1)
Output a message and terminate the current script.
Class which performs the actual renaming of users.
Generic operation result class Has warning/error list, boolean status and arbitrary value.
Definition Status.php:54
Creates Title objects.
Helper for TempUserConfig representing string patterns with "$1" indicating variable substitution.
Definition Pattern.php:16
Creates User objects.
internal since 1.36
Definition User.php:93
static newFromName( $name, $validate='valid')
Definition User.php:582
Service for page rename actions.