StringUtils: Add a utility for checking if a string is a valid regex
[lhc/web/wiklou.git] / includes / actions / RollbackAction.php
1 <?php
2 /**
3 * Edit rollback user interface
4 *
5 * This program is free software; you can redistribute it and/or modify
6 * it under the terms of the GNU General Public License as published by
7 * the Free Software Foundation; either version 2 of the License, or
8 * (at your option) any later version.
9 *
10 * This program is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 * GNU General Public License for more details.
14 *
15 * You should have received a copy of the GNU General Public License
16 * along with this program; if not, write to the Free Software
17 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
18 *
19 * @file
20 * @ingroup Actions
21 */
22
23 use MediaWiki\Storage\RevisionRecord;
24
25 /**
26 * User interface for the rollback action
27 *
28 * @ingroup Actions
29 */
30 class RollbackAction extends FormAction {
31
32 public function getName() {
33 return 'rollback';
34 }
35
36 public function getRestriction() {
37 return 'rollback';
38 }
39
40 protected function usesOOUI() {
41 return true;
42 }
43
44 protected function getDescription() {
45 return '';
46 }
47
48 public function doesWrites() {
49 return true;
50 }
51
52 public function onSuccess() {
53 return false;
54 }
55
56 public function onSubmit( $data ) {
57 return false;
58 }
59
60 protected function alterForm( HTMLForm $form ) {
61 $form->setWrapperLegendMsg( 'confirm-rollback-top' );
62 $form->setSubmitTextMsg( 'confirm-rollback-button' );
63 $form->setTokenSalt( 'rollback' );
64
65 $from = $this->getRequest()->getVal( 'from' );
66 if ( $from === null ) {
67 throw new BadRequestError( 'rollbackfailed', 'rollback-missingparam' );
68 }
69 foreach ( [ 'from', 'bot', 'hidediff', 'summary', 'token' ] as $param ) {
70 $val = $this->getRequest()->getVal( $param );
71 if ( $val !== null ) {
72 $form->addHiddenField( $param, $val );
73 }
74 }
75 }
76
77 /**
78 * @throws ErrorPageError
79 * @throws ReadOnlyError
80 * @throws ThrottledError
81 */
82 public function show() {
83 if ( $this->getUser()->getOption( 'showrollbackconfirmation' ) == false ||
84 $this->getRequest()->wasPosted() ) {
85 $this->handleRollbackRequest();
86 } else {
87 $this->showRollbackConfirmationForm();
88 }
89 }
90
91 public function handleRollbackRequest() {
92 $this->enableTransactionalTimelimit();
93
94 $request = $this->getRequest();
95 $user = $this->getUser();
96 $from = $request->getVal( 'from' );
97 $rev = $this->page->getRevision();
98 if ( $from === null ) {
99 throw new ErrorPageError( 'rollbackfailed', 'rollback-missingparam' );
100 }
101 if ( !$rev ) {
102 throw new ErrorPageError( 'rollbackfailed', 'rollback-missingrevision' );
103 }
104 if ( $from !== $rev->getUserText() ) {
105 throw new ErrorPageError( 'rollbackfailed', 'alreadyrolled', [
106 $this->getTitle()->getPrefixedText(),
107 $from,
108 $rev->getUserText()
109 ] );
110 }
111
112 $data = null;
113 $errors = $this->page->doRollback(
114 $from,
115 $request->getText( 'summary' ),
116 $request->getVal( 'token' ),
117 $request->getBool( 'bot' ),
118 $data,
119 $this->getUser()
120 );
121
122 if ( in_array( [ 'actionthrottledtext' ], $errors ) ) {
123 throw new ThrottledError;
124 }
125
126 if ( $this->hasRollbackRelatedErrors( $errors ) ) {
127 $this->getOutput()->setPageTitle( $this->msg( 'rollbackfailed' ) );
128 $errArray = $errors[0];
129 $errMsg = array_shift( $errArray );
130 $this->getOutput()->addWikiMsgArray( $errMsg, $errArray );
131
132 if ( isset( $data['current'] ) ) {
133 /** @var Revision $current */
134 $current = $data['current'];
135
136 if ( $current->getComment() != '' ) {
137 $this->getOutput()->addWikiMsg(
138 'editcomment',
139 Message::rawParam(
140 Linker::formatComment( $current->getComment() )
141 )
142 );
143 }
144 }
145
146 return;
147 }
148
149 # NOTE: Permission errors already handled by Action::checkExecute.
150 if ( $errors == [ [ 'readonlytext' ] ] ) {
151 throw new ReadOnlyError;
152 }
153
154 # XXX: Would be nice if ErrorPageError could take multiple errors, and/or a status object.
155 # Right now, we only show the first error
156 foreach ( $errors as $error ) {
157 throw new ErrorPageError( 'rollbackfailed', $error[0], array_slice( $error, 1 ) );
158 }
159
160 /** @var Revision $current */
161 $current = $data['current'];
162 $target = $data['target'];
163 $newId = $data['newid'];
164 $this->getOutput()->setPageTitle( $this->msg( 'actioncomplete' ) );
165 $this->getOutput()->setRobotPolicy( 'noindex,nofollow' );
166
167 $old = Linker::revUserTools( $current );
168 $new = Linker::revUserTools( $target );
169 $this->getOutput()->addHTML(
170 $this->msg( 'rollback-success' )
171 ->rawParams( $old, $new )
172 ->params( $current->getUserText( RevisionRecord::FOR_THIS_USER, $user ) )
173 ->params( $target->getUserText( RevisionRecord::FOR_THIS_USER, $user ) )
174 ->parseAsBlock()
175 );
176
177 if ( $user->getBoolOption( 'watchrollback' ) ) {
178 $user->addWatch( $this->page->getTitle(), User::IGNORE_USER_RIGHTS );
179 }
180
181 $this->getOutput()->returnToMain( false, $this->getTitle() );
182
183 if ( !$request->getBool( 'hidediff', false ) &&
184 !$this->getUser()->getBoolOption( 'norollbackdiff' )
185 ) {
186 $contentHandler = $current->getContentHandler();
187 $de = $contentHandler->createDifferenceEngine(
188 $this->getContext(),
189 $current->getId(),
190 $newId,
191 false,
192 true
193 );
194 $de->showDiff( '', '' );
195 }
196 }
197
198 /**
199 * Enables transactional time limit for POST and GET requests to RollbackAction
200 * @throws ConfigException
201 */
202 private function enableTransactionalTimelimit() {
203 // If Rollbacks are made POST-only, use $this->useTransactionalTimeLimit()
204 wfTransactionalTimeLimit();
205 if ( !$this->getRequest()->wasPosted() ) {
206 /**
207 * We apply the higher POST limits on GET requests
208 * to prevent logstash.wikimedia.org from being spammed
209 */
210 $fname = __METHOD__;
211 $trxLimits = $this->context->getConfig()->get( 'TrxProfilerLimits' );
212 $trxProfiler = Profiler::instance()->getTransactionProfiler();
213 $trxProfiler->redefineExpectations( $trxLimits['POST'], $fname );
214 DeferredUpdates::addCallableUpdate( function () use ( $trxProfiler, $trxLimits, $fname
215 ) {
216 $trxProfiler->redefineExpectations( $trxLimits['PostSend-POST'], $fname );
217 } );
218 }
219 }
220
221 private function showRollbackConfirmationForm() {
222 $form = $this->getForm();
223 if ( $form->show() ) {
224 $this->onSuccess();
225 }
226 }
227
228 protected function getFormFields() {
229 return [
230 'intro' => [
231 'type' => 'info',
232 'vertical-label' => true,
233 'raw' => true,
234 'default' => $this->msg( 'confirm-rollback-bottom' )->parse()
235 ]
236 ];
237 }
238
239 private function hasRollbackRelatedErrors( array $errors ) {
240 return isset( $errors[0][0] ) &&
241 ( $errors[0][0] == 'alreadyrolled' ||
242 $errors[0][0] == 'cantrollback'
243 );
244 }
245 }