MediaWiki master
migrateBlocks.php
Go to the documentation of this file.
1<?php
2
3use Wikimedia\IPUtils;
5
6require_once __DIR__ . "/Maintenance.php";
7
19 private IMaintainableDatabase $dbw;
20
21 public function __construct() {
22 parent::__construct();
23 $this->addDescription(
24 'Copy data from the ipblocks table into the new block and block_target tables'
25 );
26 $this->addOption(
27 'sleep',
28 'Sleep time (in seconds) between every batch. Default: 0',
29 false,
30 true
31 );
32 // Batch size is typically 1000, but we'll do 500 since there are 2 writes for each ipblock.
33 $this->setBatchSize( 500 );
34 }
35
36 protected function getUpdateKey() {
37 return __CLASS__;
38 }
39
40 protected function doDBUpdates() {
41 $this->dbw = $this->getDB( DB_PRIMARY );
42 if ( !$this->dbw->tableExists( 'block', __METHOD__ ) || !$this->dbw->tableExists( 'block_target' ) ) {
43 $this->fatalError( "Run update.php to create the block and block_target tables." );
44 }
45
46 $this->output( "Populating the block and block_target tables\n" );
47 $migratedCount = 0;
48
49 $id = 0;
50 while ( $id !== null ) {
51 $this->output( "Migrating ipblocks with ID > $id...\n" );
52 [ $numBlocks, $id ] = $this->handleBatch( $id );
53 $migratedCount += $numBlocks;
54 }
55
56 $this->output( "Completed migration of $migratedCount ipblocks to block and block_target.\n" );
57
58 return true;
59 }
60
68 private function handleBatch( int $lowId ): array {
69 $migratedCount = 0;
70 $res = $this->dbw->newSelectQueryBuilder()
71 ->select( '*' )
72 ->from( 'ipblocks' )
73 ->leftJoin( 'block', null, 'bl_id=ipb_id' )
74 ->where( [
75 $this->dbw->expr( 'ipb_id', '>', $lowId ),
76 'bl_id' => null
77 ] )
78 ->orderBy( 'ipb_id' )
79 ->limit( $this->getBatchSize() )
80 ->caller( __METHOD__ )
81 ->fetchResultSet();
82
83 if ( !$res->numRows() ) {
84 return [ $migratedCount, null ];
85 }
86
87 $highestId = $lowId;
88 foreach ( $res as $row ) {
89 $highestId = $row->ipb_id;
90 $isIP = IPUtils::isValid( $row->ipb_address );
91 $isRange = IPUtils::isValidRange( $row->ipb_address );
92 $isIPOrRange = $isIP || $isRange;
93 $ipHex = null;
94 if ( $isIP ) {
95 $ipHex = IPUtils::toHex( $row->ipb_address );
96 } elseif ( $isRange ) {
97 $ipHex = $row->ipb_range_start;
98 } elseif ( (int)$row->ipb_user === 0 ) {
99 // There was data corruption circa 2006 and 2011 where some accounts were
100 // blocked as if they were logged out users. Here we'll prune the erroneous
101 // data by simply not copying it to the new schema.
102 $this->output( "ipblock with ID $row->ipb_id: account block with ipb_user=0, skipping…\n" );
103 continue;
104 }
105
106 // Insert into block_target
107 $blockTarget = [
108 'bt_address' => $isIPOrRange ? $row->ipb_address : null,
109 'bt_user' => $isIPOrRange ? null : $row->ipb_user,
110 'bt_user_text' => $isIPOrRange ? null : $row->ipb_address,
111 'bt_auto' => $row->ipb_auto,
112 'bt_range_start' => $isRange ? $row->ipb_range_start : null,
113 'bt_range_end' => $isRange ? $row->ipb_range_end : null,
114 'bt_ip_hex' => $ipHex,
115 'bt_count' => 1
116 ];
117 $this->dbw->newInsertQueryBuilder()
118 ->insertInto( 'block_target' )
119 ->row( $blockTarget )
120 ->caller( __METHOD__ )
121 ->execute();
122 $insertId = $this->dbw->insertId();
123 if ( !$insertId ) {
124 $this->fatalError(
125 "ipblock with ID $row->ipb_id: Failed to create block_target. Insert ID is falsy!"
126 );
127 }
128
129 // Insert into block
130 $block = [
131 'bl_id' => $row->ipb_id,
132 'bl_target' => $insertId,
133 'bl_by_actor' => $row->ipb_by_actor,
134 'bl_reason_id' => $row->ipb_reason_id,
135 'bl_timestamp' => $row->ipb_timestamp,
136 'bl_anon_only' => $row->ipb_anon_only,
137 'bl_create_account' => $row->ipb_create_account,
138 'bl_enable_autoblock' => $row->ipb_enable_autoblock,
139 'bl_expiry' => $row->ipb_expiry,
140 'bl_deleted' => $row->ipb_deleted,
141 'bl_block_email' => $row->ipb_block_email,
142 'bl_allow_usertalk' => $row->ipb_allow_usertalk,
143 // See T282890
144 'bl_parent_block_id' => (int)$row->ipb_parent_block_id === 0 ? null : $row->ipb_parent_block_id,
145 'bl_sitewide' => $row->ipb_sitewide,
146 ];
147 $this->dbw->newInsertQueryBuilder()
148 ->insertInto( 'block' )
149 ->ignore()
150 ->row( $block )
151 ->caller( __METHOD__ )
152 ->execute();
153 if ( $this->dbw->affectedRows() ) {
154 $migratedCount++;
155 }
156 }
157
158 $this->output( "Migrated $migratedCount blocks\n" );
159
160 // Sleep between batches for replication to catch up
161 $this->waitForReplication();
162 $sleep = (int)$this->getOption( 'sleep', 0 );
163 if ( $sleep > 0 ) {
164 sleep( $sleep );
165 }
166
167 return [ $migratedCount, $highestId ];
168 }
169}
170
171$maintClass = MigrateBlocks::class;
172require_once RUN_MAINTENANCE_IF_MAIN;
getDB()
Class for scripts that perform database maintenance and want to log the update in updatelog so we can...
output( $out, $channel=null)
Throw some output to the user.
waitForReplication()
Wait for replica DBs to catch up.
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.
Maintenance script that migrates rows from ipblocks to block and block_target.
doDBUpdates()
Do the actual work.
__construct()
Default constructor.
getUpdateKey()
Get the update key name to go in the update log table.
Advanced database interface for IDatabase handles that include maintenance methods.
$maintClass
const DB_PRIMARY
Definition defines.php:28