Merge "Add missing return types to User::getOption()"
[lhc/web/wiklou.git] / tests / phpunit / includes / TestUser.php
1 <?php
2
3 /**
4 * Wraps the user object, so we can also retain full access to properties
5 * like password if we log in via the API.
6 */
7 class TestUser {
8 /**
9 * @var string
10 */
11 private $username;
12
13 /**
14 * @var string
15 */
16 private $password;
17
18 /**
19 * @var User
20 */
21 private $user;
22
23 private function assertNotReal() {
24 global $wgDBprefix;
25 if ( $wgDBprefix !== MediaWikiTestCase::DB_PREFIX &&
26 $wgDBprefix !== MediaWikiTestCase::ORA_DB_PREFIX
27 ) {
28 throw new MWException( "Can't create user on real database" );
29 }
30 }
31
32 public function __construct( $username, $realname = 'Real Name',
33 $email = 'sample@example.com', $groups = []
34 ) {
35 $this->assertNotReal();
36
37 $this->username = $username;
38 $this->password = 'TestUser';
39
40 $this->user = User::newFromName( $this->username );
41 $this->user->load();
42
43 // In an ideal world we'd have a new wiki (or mock data store) for every single test.
44 // But for now, we just need to create or update the user with the desired properties.
45 // we particularly need the new password, since we just generated it randomly.
46 // In core MediaWiki, there is no functionality to delete users, so this is the best we can do.
47 if ( !$this->user->isLoggedIn() ) {
48 // create the user
49 $this->user = User::createNew(
50 $this->username, [
51 "email" => $email,
52 "real_name" => $realname
53 ]
54 );
55
56 if ( !$this->user ) {
57 throw new MWException( "Error creating TestUser " . $username );
58 }
59 }
60
61 // Update the user to use the password and other details
62 $this->setPassword( $this->password );
63 $change = $this->setEmail( $email ) ||
64 $this->setRealName( $realname );
65
66 // Adjust groups by adding any missing ones and removing any extras
67 $currentGroups = $this->user->getGroups();
68 foreach ( array_diff( $groups, $currentGroups ) as $group ) {
69 $this->user->addGroup( $group );
70 }
71 foreach ( array_diff( $currentGroups, $groups ) as $group ) {
72 $this->user->removeGroup( $group );
73 }
74 if ( $change ) {
75 // Disable CAS check before saving. The User object may have been initialized from cached
76 // information that may be out of whack with the database during testing. If tests were
77 // perfectly isolated, this would not happen. But if it does happen, let's just ignore the
78 // inconsistency, and just write the data we want - during testing, we are not worried
79 // about data loss.
80 $this->user->mTouched = '';
81 $this->user->saveSettings();
82 }
83 }
84
85 /**
86 * @param string $realname
87 * @return bool
88 */
89 private function setRealName( $realname ) {
90 if ( $this->user->getRealName() !== $realname ) {
91 $this->user->setRealName( $realname );
92 return true;
93 }
94
95 return false;
96 }
97
98 /**
99 * @param string $email
100 * @return bool
101 */
102 private function setEmail( $email ) {
103 if ( $this->user->getEmail() !== $email ) {
104 $this->user->setEmail( $email );
105 return true;
106 }
107
108 return false;
109 }
110
111 /**
112 * @param string $password
113 */
114 private function setPassword( $password ) {
115 self::setPasswordForUser( $this->user, $password );
116 }
117
118 /**
119 * Set the password on a testing user
120 *
121 * This assumes we're still using the generic AuthManager config from
122 * PHPUnitMaintClass::finalSetup(), and just sets the password in the
123 * database directly.
124 * @param User $user
125 * @param string $password
126 */
127 public static function setPasswordForUser( User $user, $password ) {
128 if ( !$user->getId() ) {
129 throw new MWException( "Passed User has not been added to the database yet!" );
130 }
131
132 $dbw = wfGetDB( DB_MASTER );
133 $row = $dbw->selectRow(
134 'user',
135 [ 'user_password' ],
136 [ 'user_id' => $user->getId() ],
137 __METHOD__
138 );
139 if ( !$row ) {
140 throw new MWException( "Passed User has an ID but is not in the database?" );
141 }
142
143 $passwordFactory = new PasswordFactory();
144 $passwordFactory->init( RequestContext::getMain()->getConfig() );
145 if ( !$passwordFactory->newFromCiphertext( $row->user_password )->equals( $password ) ) {
146 $passwordHash = $passwordFactory->newFromPlaintext( $password );
147 $dbw->update(
148 'user',
149 [ 'user_password' => $passwordHash->toString() ],
150 [ 'user_id' => $user->getId() ],
151 __METHOD__
152 );
153 }
154 }
155
156 /**
157 * @since 1.25
158 * @return User
159 */
160 public function getUser() {
161 return $this->user;
162 }
163
164 /**
165 * @since 1.25
166 * @return string
167 */
168 public function getPassword() {
169 return $this->password;
170 }
171 }