Code Coverage
 
Lines
Functions and Methods
Classes and Traits
Total
0.00% covered (danger)
0.00%
0 / 69
0.00% covered (danger)
0.00%
0 / 4
CRAP
0.00% covered (danger)
0.00%
0 / 1
RevisionDeleteUser
0.00% covered (danger)
0.00%
0 / 68
0.00% covered (danger)
0.00%
0 / 4
132
0.00% covered (danger)
0.00%
0 / 1
 setUsernameBitfields
0.00% covered (danger)
0.00%
0 / 62
0.00% covered (danger)
0.00%
0 / 1
56
 buildSetBitDeletedField
0.00% covered (danger)
0.00%
0 / 4
0.00% covered (danger)
0.00%
0 / 1
6
 suppressUserName
0.00% covered (danger)
0.00%
0 / 1
0.00% covered (danger)
0.00%
0 / 1
2
 unsuppressUserName
0.00% covered (danger)
0.00%
0 / 1
0.00% covered (danger)
0.00%
0 / 1
2
1<?php
2/**
3 * Backend functions for suppressing and unsuppressing all references to a given user.
4 *
5 * @license GPL-2.0-or-later
6 * @file
7 * @ingroup RevisionDelete
8 */
9
10namespace MediaWiki\RevisionDelete;
11
12use MediaWiki\Logging\LogPage;
13use MediaWiki\MediaWikiServices;
14use MediaWiki\Revision\RevisionRecord;
15use MediaWiki\Title\Title;
16use Wikimedia\Rdbms\IDatabase;
17use Wikimedia\Rdbms\RawSQLValue;
18
19/**
20 * Backend functions for suppressing and unsuppressing all references to a given user,
21 * used when blocking with HideUser enabled.  This was spun out of SpecialBlockip.php
22 * in 1.18; at some point it needs to be rewritten to either use RevisionDelete abstraction,
23 * or at least schema abstraction.
24 *
25 * @ingroup RevisionDelete
26 */
27class RevisionDeleteUser {
28
29    /**
30     * Update *_deleted bitfields in various tables to hide or unhide usernames
31     *
32     * @param string $name Username
33     * @param int $userId
34     * @param string $op Operator '|' or '&'
35     * @param null|IDatabase $dbw If you happen to have one lying around
36     * @return bool True on success, false on failure (e.g. invalid user ID)
37     */
38    private static function setUsernameBitfields( $name, $userId, $op, ?IDatabase $dbw = null ) {
39        if ( !$userId || ( $op !== '|' && $op !== '&' ) ) {
40            return false;
41        }
42        if ( !$dbw instanceof IDatabase ) {
43            $dbw = MediaWikiServices::getInstance()->getConnectionProvider()->getPrimaryDatabase();
44        }
45
46        # To suppress, we OR the current bitfields with RevisionRecord::DELETED_USER
47        # to put a 1 in the username *_deleted bit. To unsuppress we AND the
48        # current bitfields with the inverse of RevisionRecord::DELETED_USER. The
49        # username bit is made to 0 (x & 0 = 0), while others are unchanged (x & 1 = x).
50        # The same goes for the sysop-restricted *_deleted bit.
51        $delUser = RevisionRecord::DELETED_USER | RevisionRecord::DELETED_RESTRICTED;
52        $delAction = LogPage::DELETED_ACTION | RevisionRecord::DELETED_RESTRICTED;
53        if ( $op === '&' ) {
54            $delUser = $dbw->bitNot( $delUser );
55            $delAction = $dbw->bitNot( $delAction );
56        }
57
58        # Normalize user name
59        $userTitle = Title::makeTitleSafe( NS_USER, $name );
60        $userDbKey = $userTitle->getDBkey();
61
62        $actorId = $dbw->newSelectQueryBuilder()
63            ->select( 'actor_id' )
64            ->from( 'actor' )
65            ->where( [ 'actor_name' => $name ] )
66            ->caller( __METHOD__ )->fetchField();
67        if ( $actorId ) {
68            # Hide name from live edits
69            $dbw->newUpdateQueryBuilder()
70                ->update( 'revision' )
71                ->set( self::buildSetBitDeletedField( 'rev_deleted', $op, $delUser, $dbw ) )
72                ->where( [ 'rev_actor' => $actorId ] )
73                ->caller( __METHOD__ )->execute();
74
75            # Hide name from deleted edits
76            $dbw->newUpdateQueryBuilder()
77                ->update( 'archive' )
78                ->set( self::buildSetBitDeletedField( 'ar_deleted', $op, $delUser, $dbw ) )
79                ->where( [ 'ar_actor' => $actorId ] )
80                ->caller( __METHOD__ )->execute();
81
82            # Hide name from logs
83            $dbw->newUpdateQueryBuilder()
84                ->update( 'logging' )
85                ->set( self::buildSetBitDeletedField( 'log_deleted', $op, $delUser, $dbw ) )
86                ->where( [ 'log_actor' => $actorId, $dbw->expr( 'log_type', '!=', 'suppress' ) ] )
87                ->caller( __METHOD__ )->execute();
88
89            # Hide name from RC
90            $dbw->newUpdateQueryBuilder()
91                ->update( 'recentchanges' )
92                ->set( self::buildSetBitDeletedField( 'rc_deleted', $op, $delUser, $dbw ) )
93                ->where( [ 'rc_actor' => $actorId ] )
94                ->caller( __METHOD__ )->execute();
95
96            # Hide name from live images
97            $dbw->newUpdateQueryBuilder()
98                ->update( 'oldimage' )
99                ->set( self::buildSetBitDeletedField( 'oi_deleted', $op, $delUser, $dbw ) )
100                ->where( [ 'oi_actor' => $actorId ] )
101                ->caller( __METHOD__ )->execute();
102
103            # Hide name from deleted images
104            $dbw->newUpdateQueryBuilder()
105                ->update( 'filearchive' )
106                ->set( self::buildSetBitDeletedField( 'fa_deleted', $op, $delUser, $dbw ) )
107                ->where( [ 'fa_actor' => $actorId ] )
108                ->caller( __METHOD__ )->execute();
109        }
110
111        # Hide log entries pointing to the user page
112        $dbw->newUpdateQueryBuilder()
113            ->update( 'logging' )
114            ->set( self::buildSetBitDeletedField( 'log_deleted', $op, $delAction, $dbw ) )
115            ->where( [
116                'log_namespace' => NS_USER,
117                'log_title' => $userDbKey,
118                $dbw->expr( 'log_type', '!=', 'suppress' )
119            ] )
120            ->caller( __METHOD__ )->execute();
121
122        # Hide RC entries pointing to the user page
123        $dbw->newUpdateQueryBuilder()
124            ->update( 'recentchanges' )
125            ->set( self::buildSetBitDeletedField( 'rc_deleted', $op, $delAction, $dbw ) )
126            ->where( [ 'rc_namespace' => NS_USER, 'rc_title' => $userDbKey, $dbw->expr( 'rc_logid', '>', 0 ) ] )
127            ->caller( __METHOD__ )->execute();
128
129        return true;
130    }
131
132    /**
133     * @param string $field
134     * @param string $op
135     * @param string|int $value
136     * @param IDatabase $dbw
137     */
138    private static function buildSetBitDeletedField(
139        string $field, string $op, $value, IDatabase $dbw
140    ): array {
141        return [ $field => new RawSQLValue( $op === '&'
142            ? $dbw->bitAnd( $field, $value )
143            : $dbw->bitOr( $field, $value )
144        ) ];
145    }
146
147    /**
148     * @param string $name User name
149     * @param int $userId Both user name and ID must be provided
150     * @param IDatabase|null $dbw If you happen to have one lying around
151     * @return bool True on success, false on failure (e.g. invalid user ID)
152     */
153    public static function suppressUserName( $name, $userId, ?IDatabase $dbw = null ) {
154        return self::setUsernameBitfields( $name, $userId, '|', $dbw );
155    }
156
157    /**
158     * @param string $name User name
159     * @param int $userId Both user name and ID must be provided
160     * @param IDatabase|null $dbw If you happen to have one lying around
161     * @return bool True on success, false on failure (e.g. invalid user ID)
162     */
163    public static function unsuppressUserName( $name, $userId, ?IDatabase $dbw = null ) {
164        return self::setUsernameBitfields( $name, $userId, '&', $dbw );
165    }
166}
167
168/** @deprecated class alias since 1.46 */
169class_alias( RevisionDeleteUser::class, 'RevisionDeleteUser' );