MediaWiki master
LocalFileDeleteBatch.php
Go to the documentation of this file.
1<?php
22
28use StatusValue;
29use Wikimedia\ScopedCallback;
30
39 private $file;
40
42 private $reason;
43
45 private $srcRels = [];
46
48 private $archiveUrls = [];
49
51 private $deletionBatch;
52
54 private $suppress;
55
57 private $user;
58
65 public function __construct(
66 File $file,
67 UserIdentity $user,
68 $reason = '',
69 $suppress = false
70 ) {
71 $this->file = $file;
72 $this->user = $user;
73 $this->reason = $reason;
74 $this->suppress = $suppress;
75 }
76
77 public function addCurrent() {
78 $this->srcRels['.'] = $this->file->getRel();
79 }
80
84 public function addOld( $oldName ) {
85 $this->srcRels[$oldName] = $this->file->getArchiveRel( $oldName );
86 $this->archiveUrls[] = $this->file->getArchiveUrl( $oldName );
87 }
88
93 public function addOlds() {
94 $archiveNames = [];
95
96 $dbw = $this->file->repo->getPrimaryDB();
97 $result = $dbw->newSelectQueryBuilder()
98 ->select( [ 'oi_archive_name' ] )
99 ->from( 'oldimage' )
100 ->where( [ 'oi_name' => $this->file->getName() ] )
101 ->caller( __METHOD__ )->fetchResultSet();
102
103 foreach ( $result as $row ) {
104 $this->addOld( $row->oi_archive_name );
105 $archiveNames[] = $row->oi_archive_name;
106 }
107
108 return $archiveNames;
109 }
110
114 protected function getOldRels() {
115 if ( !isset( $this->srcRels['.'] ) ) {
116 $oldRels =& $this->srcRels;
117 $deleteCurrent = false;
118 } else {
119 $oldRels = $this->srcRels;
120 unset( $oldRels['.'] );
121 $deleteCurrent = true;
122 }
123
124 return [ $oldRels, $deleteCurrent ];
125 }
126
131 protected function getHashes( StatusValue $status ): array {
132 $hashes = [];
133 [ $oldRels, $deleteCurrent ] = $this->getOldRels();
134
135 if ( $deleteCurrent ) {
136 $hashes['.'] = $this->file->getSha1();
137 }
138
139 if ( count( $oldRels ) ) {
140 $dbw = $this->file->repo->getPrimaryDB();
141 $res = $dbw->newSelectQueryBuilder()
142 ->select( [ 'oi_archive_name', 'oi_sha1' ] )
143 ->from( 'oldimage' )
144 ->where( [
145 'oi_archive_name' => array_map( 'strval', array_keys( $oldRels ) ),
146 'oi_name' => $this->file->getName() / performance
147 ] )
148 ->caller( __METHOD__ )->fetchResultSet();
149
150 foreach ( $res as $row ) {
151 if ( $row->oi_archive_name === '' ) {
152 / File lost, the check simulates OldLocalFile::exists
153 $hashes[$row->oi_archive_name] = false;
154 continue;
155 }
156 if ( rtrim( $row->oi_sha1, "\0" ) === '' ) {
157 / Get the hash from the file
158 $oldUrl = $this->file->getArchiveVirtualUrl( $row->oi_archive_name );
159 $props = $this->file->repo->getFileProps( $oldUrl );
160
161 if ( $props['fileExists'] ) {
162 / Upgrade the oldimage row
163 $dbw->newUpdateQueryBuilder()
164 ->update( 'oldimage' )
165 ->set( [ 'oi_sha1' => $props['sha1'] ] )
166 ->where( [
167 'oi_name' => $this->file->getName(),
168 'oi_archive_name' => $row->oi_archive_name,
169 ] )
170 ->caller( __METHOD__ )->execute();
171 $hashes[$row->oi_archive_name] = $props['sha1'];
172 } else {
173 $hashes[$row->oi_archive_name] = false;
174 }
175 } else {
176 $hashes[$row->oi_archive_name] = $row->oi_sha1;
177 }
178 }
179 }
180
181 $missing = array_diff_key( $this->srcRels, $hashes );
182
183 foreach ( $missing as $name => $rel ) {
184 $status->error( 'filedelete-old-unregistered', $name );
185 }
186
187 foreach ( $hashes as $name => $hash ) {
188 if ( !$hash ) {
189 $status->error( 'filedelete-missing', $this->srcRels[$name] );
190 unset( $hashes[$name] );
191 }
192 }
193
194 return $hashes;
195 }
196
197 protected function doDBInserts() {
198 $now = time();
199 $dbw = $this->file->repo->getPrimaryDB();
200
201 $commentStore = MediaWikiServices::getInstance()->getCommentStore();
202
203 $encTimestamp = $dbw->addQuotes( $dbw->timestamp( $now ) );
204 $encUserId = $dbw->addQuotes( $this->user->getId() );
205 $encGroup = $dbw->addQuotes( 'deleted' );
206 $ext = $this->file->getExtension();
207 $dotExt = $ext === '' ? '' : ".$ext";
208 $encExt = $dbw->addQuotes( $dotExt );
209 [ $oldRels, $deleteCurrent ] = $this->getOldRels();
210
211 / Bitfields to further suppress the content
212 if ( $this->suppress ) {
213 $bitfield = RevisionRecord::SUPPRESSED_ALL;
214 } else {
215 $bitfield = 'oi_deleted';
216 }
217
218 if ( $deleteCurrent ) {
219 $tables = [ 'image' ];
220 $fields = [
221 'fa_storage_group' => $encGroup,
222 'fa_storage_key' => $dbw->conditional(
223 [ 'img_sha1' => '' ],
224 $dbw->addQuotes( '' ),
225 $dbw->buildConcat( [ "img_sha1", $encExt ] )
226 ),
227 'fa_deleted_user' => $encUserId,
228 'fa_deleted_timestamp' => $encTimestamp,
229 'fa_deleted' => $this->suppress ? $bitfield : 0,
230 'fa_name' => 'img_name',
231 'fa_archive_name' => 'NULL',
232 'fa_size' => 'img_size',
233 'fa_width' => 'img_width',
234 'fa_height' => 'img_height',
235 'fa_metadata' => 'img_metadata',
236 'fa_bits' => 'img_bits',
237 'fa_media_type' => 'img_media_type',
238 'fa_major_mime' => 'img_major_mime',
239 'fa_minor_mime' => 'img_minor_mime',
240 'fa_description_id' => 'img_description_id',
241 'fa_timestamp' => 'img_timestamp',
242 'fa_sha1' => 'img_sha1',
243 'fa_actor' => 'img_actor',
244 ];
245 $joins = [];
246
247 $fields += array_map(
248 [ $dbw, 'addQuotes' ],
249 $commentStore->insert( $dbw, 'fa_deleted_reason', $this->reason )
250 );
251
252 $dbw->insertSelect( 'filearchive', $tables, $fields,
253 [ 'img_name' => $this->file->getName() ], __METHOD__, [ 'IGNORE' ], [], $joins );
254 }
255
256 if ( count( $oldRels ) ) {
257 $queryBuilder = FileSelectQueryBuilder::newForOldFile( $dbw );
258 $queryBuilder
259 ->forUpdate()
260 ->where( [ 'oi_name' => $this->file->getName() ] )
261 ->andWhere( [ 'oi_archive_name' => array_map( 'strval', array_keys( $oldRels ) ) ] );
262 $res = $queryBuilder->caller( __METHOD__ )->fetchResultSet();
263 if ( $res->numRows() ) {
264 $reason = $commentStore->createComment( $dbw, $this->reason );
265 $rowsInsert = [];
266 foreach ( $res as $row ) {
267 $comment = $commentStore->getComment( 'oi_description', $row );
268 $rowsInsert[] = [
269 / Deletion-specific fields
270 'fa_storage_group' => 'deleted',
271 'fa_storage_key' => ( $row->oi_sha1 === '' )
272 ? ''
273 : "{$row->oi_sha1}{$dotExt}",
274 'fa_deleted_user' => $this->user->getId(),
275 'fa_deleted_timestamp' => $dbw->timestamp( $now ),
276 / Counterpart fields
277 'fa_deleted' => $this->suppress ? $bitfield : $row->oi_deleted,
278 'fa_name' => $row->oi_name,
279 'fa_archive_name' => $row->oi_archive_name,
280 'fa_size' => $row->oi_size,
281 'fa_width' => $row->oi_width,
282 'fa_height' => $row->oi_height,
283 'fa_metadata' => $row->oi_metadata,
284 'fa_bits' => $row->oi_bits,
285 'fa_media_type' => $row->oi_media_type,
286 'fa_major_mime' => $row->oi_major_mime,
287 'fa_minor_mime' => $row->oi_minor_mime,
288 'fa_actor' => $row->oi_actor,
289 'fa_timestamp' => $row->oi_timestamp,
290 'fa_sha1' => $row->oi_sha1
291 ] + $commentStore->insert( $dbw, 'fa_deleted_reason', $reason )
292 + $commentStore->insert( $dbw, 'fa_description', $comment );
293 }
294 $dbw->newInsertQueryBuilder()
295 ->insertInto( 'filearchive' )
296 ->ignore()
297 ->rows( $rowsInsert )
298 ->caller( __METHOD__ )->execute();
299 }
300 }
301 }
302
303 private function doDBDeletes() {
304 $dbw = $this->file->repo->getPrimaryDB();
305 $migrationStage = MediaWikiServices::getInstance()->getMainConfig()->get(
307 );
308
309 [ $oldRels, $deleteCurrent ] = $this->getOldRels();
310
311 if ( count( $oldRels ) ) {
312 $dbw->newDeleteQueryBuilder()
313 ->deleteFrom( 'oldimage' )
314 ->where( [
315 'oi_name' => $this->file->getName(),
316 'oi_archive_name' => array_map( 'strval', array_keys( $oldRels ) )
317 ] )
318 ->caller( __METHOD__ )->execute();
319 if ( ( $migrationStage & SCHEMA_COMPAT_WRITE_NEW ) && $this->file->getFileIdFromName() ) {
320 $delete = $dbw->newDeleteQueryBuilder()
321 ->deleteFrom( 'filerevision' )
322 ->where( [ 'fr_file' => $this->file->getFileIdFromName() ] );
323 if ( !$deleteCurrent ) {
324 / It's not full page deletion.
325 $delete->andWhere( [ 'fr_archive_name' => array_map( 'strval', array_keys( $oldRels ) ) ] );
326 }
327 $delete->caller( __METHOD__ )->execute();
328
329 }
330 }
331
332 if ( $deleteCurrent ) {
333 $dbw->newDeleteQueryBuilder()
334 ->deleteFrom( 'image' )
335 ->where( [ 'img_name' => $this->file->getName() ] )
336 ->caller( __METHOD__ )->execute();
337 if ( ( $migrationStage & SCHEMA_COMPAT_WRITE_NEW ) && $this->file->getFileIdFromName() ) {
338 $dbw->newUpdateQueryBuilder()
339 ->update( 'file' )
340 ->set( [
341 'file_deleted' => $this->suppress ? 3 : 1,
342 'file_latest' => 0
343 ] )
344 ->where( [ 'file_id' => $this->file->getFileIdFromName() ] )
345 ->caller( __METHOD__ )->execute();
346 if ( !count( $oldRels ) ) {
347 / Only the current version is uploaded and then deleted
348 / TODO: After migration is done and old code is removed,
349 / this should be refactored to become much simpler
350 $dbw->newDeleteQueryBuilder()
351 ->deleteFrom( 'filerevision' )
352 ->where( [ 'fr_file' => $this->file->getFileIdFromName() ] )
353 ->caller( __METHOD__ )->execute();
354 }
355 }
356 }
357 }
358
363 public function execute() {
364 $repo = $this->file->getRepo();
365 $lockStatus = $this->file->acquireFileLock();
366 if ( !$lockStatus->isOK() ) {
367 return $lockStatus;
368 }
369 $unlockScope = new ScopedCallback( function () {
370 $this->file->releaseFileLock();
371 } );
372
373 $status = $this->file->repo->newGood();
374 / Prepare deletion batch
375 $hashes = $this->getHashes( $status );
376 $this->deletionBatch = [];
377 $ext = $this->file->getExtension();
378 $dotExt = $ext === '' ? '' : ".$ext";
379
380 foreach ( $this->srcRels as $name => $srcRel ) {
381 / Skip files that have no hash (e.g. missing DB record, or sha1 field and file source)
382 if ( isset( $hashes[$name] ) ) {
383 $hash = $hashes[$name];
384 $key = $hash . $dotExt;
385 $dstRel = $repo->getDeletedHashPath( $key ) . $key;
386 $this->deletionBatch[$name] = [ $srcRel, $dstRel ];
387 }
388 }
389
390 if ( !$repo->hasSha1Storage() ) {
391 / Removes non-existent file from the batch, so we don't get errors.
392 / This also handles files in the 'deleted' zone deleted via revision deletion.
393 $checkStatus = $this->removeNonexistentFiles( $this->deletionBatch );
394 if ( !$checkStatus->isGood() ) {
395 $status->merge( $checkStatus );
396 return $status;
397 }
398 $this->deletionBatch = $checkStatus->value;
399
400 / Execute the file deletion batch
401 $status = $this->file->repo->deleteBatch( $this->deletionBatch );
402 if ( !$status->isGood() ) {
403 $status->merge( $status );
404 }
405 }
406
407 if ( !$status->isOK() ) {
408 / Critical file deletion error; abort
409 return $status;
410 }
411
412 $dbw = $this->file->repo->getPrimaryDB();
413
414 $dbw->startAtomic( __METHOD__ );
415
416 / Copy the image/oldimage rows to filearchive
417 $this->doDBInserts();
418 / Delete image/oldimage rows
419 $this->doDBDeletes();
420
421 / This is typically a no-op since we are wrapped by another atomic
422 / section in FileDeleteForm and also the implicit transaction.
423 $dbw->endAtomic( __METHOD__ );
424
425 / Commit and return
426 ScopedCallback::consume( $unlockScope );
427
428 return $status;
429 }
430
436 protected function removeNonexistentFiles( $batch ) {
437 $files = [];
438
439 foreach ( $batch as [ $src, /* dest */ ] ) {
440 $files[$src] = $this->file->repo->getVirtualUrl( 'public' ) . '/' . rawurlencode( $src );
441 }
442
443 $result = $this->file->repo->fileExistsBatch( $files );
444 if ( in_array( null, $result, true ) ) {
445 return Status::newFatal( 'backend-fail-internal',
446 $this->file->repo->getBackend()->getName() );
447 }
448
449 $newBatch = [];
450 foreach ( $batch as $batchItem ) {
451 if ( $result[$batchItem[0]] ) {
452 $newBatch[] = $batchItem;
453 }
454 }
455
456 return Status::newGood( $newBatch );
457 }
458}
459
461class_alias( LocalFileDeleteBatch::class, 'LocalFileDeleteBatch' );
const SCHEMA_COMPAT_WRITE_NEW
Definition Defines.php:307
static newForOldFile(IReadableDatabase $db, array $options=[])
Implements some public methods and some protected utility functions which are required by multiple ch...
Definition File.php:93
addOlds()
Add the old versions of the image to the batch.
__construct(File $file, UserIdentity $user, $reason='', $suppress=false)
removeNonexistentFiles( $batch)
Removes non-existent files from a deletion batch.
Local file in the wiki's own database.
Definition LocalFile.php:93
A class containing constants representing the names of configuration variables.
const FileSchemaMigrationStage
Name constant for the FileSchemaMigrationStage setting, for use with Config::get()
Service locator for MediaWiki core services.
static getInstance()
Returns the global default instance of the top level service locator.
Page revision base class.
Generic operation result class Has warning/error list, boolean status and arbitrary value.
Definition Status.php:54
Generic operation result class Has warning/error list, boolean status and arbitrary value.
error( $message,... $parameters)
Add an error, do not set fatal flag This can be used for non-fatal errors.
static newGood( $value=null)
Factory function for good results.
Interface for objects representing user identity.

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