Log the session cache type along with other cache types
[mediawiki.git] / includes / libs / StatusValue.php
blob1d23f9d368a79d1fd08988ced53e0b35093fbc70
1 <?php
2 /**
3 * This program is free software; you can redistribute it and/or modify
4 * it under the terms of the GNU General Public License as published by
5 * the Free Software Foundation; either version 2 of the License, or
6 * (at your option) any later version.
8 * This program is distributed in the hope that it will be useful,
9 * but WITHOUT ANY WARRANTY; without even the implied warranty of
10 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11 * GNU General Public License for more details.
13 * You should have received a copy of the GNU General Public License along
14 * with this program; if not, write to the Free Software Foundation, Inc.,
15 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
16 * http://www.gnu.org/copyleft/gpl.html
18 * @file
21 /**
22 * Generic operation result class
23 * Has warning/error list, boolean status and arbitrary value
25 * "Good" means the operation was completed with no warnings or errors.
27 * "OK" means the operation was partially or wholly completed.
29 * An operation which is not OK should have errors so that the user can be
30 * informed as to what went wrong. Calling the fatal() function sets an error
31 * message and simultaneously switches off the OK flag.
33 * The recommended pattern for Status objects is to return a StatusValue
34 * unconditionally, i.e. both on success and on failure -- so that the
35 * developer of the calling code is reminded that the function can fail, and
36 * so that a lack of error-handling will be explicit.
38 * The use of Message objects should be avoided when serializability is needed.
40 * @since 1.25
42 class StatusValue {
43 /** @var bool */
44 protected $ok = true;
45 /** @var array */
46 protected $errors = [];
48 /** @var mixed */
49 public $value;
50 /** @var array Map of (key => bool) to indicate success of each part of batch operations */
51 public $success = [];
52 /** @var int Counter for batch operations */
53 public $successCount = 0;
54 /** @var int Counter for batch operations */
55 public $failCount = 0;
57 /**
58 * Factory function for fatal errors
60 * @param string|MessageSpecifier $message Message key or object
61 * @return StatusValue
63 public static function newFatal( $message /*, parameters...*/ ) {
64 $params = func_get_args();
65 $result = new static();
66 call_user_func_array( [ &$result, 'fatal' ], $params );
67 return $result;
70 /**
71 * Factory function for good results
73 * @param mixed $value
74 * @return StatusValue
76 public static function newGood( $value = null ) {
77 $result = new static();
78 $result->value = $value;
79 return $result;
82 /**
83 * Returns whether the operation completed and didn't have any error or
84 * warnings
86 * @return bool
88 public function isGood() {
89 return $this->ok && !$this->errors;
92 /**
93 * Returns whether the operation completed
95 * @return bool
97 public function isOK() {
98 return $this->ok;
102 * @return mixed
104 public function getValue() {
105 return $this->value;
109 * Get the list of errors
111 * Each error is a (message:string or MessageSpecifier,params:array) map
113 * @return array
115 public function getErrors() {
116 return $this->errors;
120 * Change operation status
122 * @param bool $ok
124 public function setOK( $ok ) {
125 $this->ok = $ok;
129 * Change operation resuklt
131 * @param bool $ok Whether the operation completed
132 * @param mixed $value
134 public function setResult( $ok, $value = null ) {
135 $this->ok = $ok;
136 $this->value = $value;
140 * Add a new warning
142 * @param string|MessageSpecifier $message Message key or object
144 public function warning( $message /*, parameters... */ ) {
145 $this->errors[] = [
146 'type' => 'warning',
147 'message' => $message,
148 'params' => array_slice( func_get_args(), 1 )
153 * Add an error, do not set fatal flag
154 * This can be used for non-fatal errors
156 * @param string|MessageSpecifier $message Message key or object
158 public function error( $message /*, parameters... */ ) {
159 $this->errors[] = [
160 'type' => 'error',
161 'message' => $message,
162 'params' => array_slice( func_get_args(), 1 )
167 * Add an error and set OK to false, indicating that the operation
168 * as a whole was fatal
170 * @param string|MessageSpecifier $message Message key or object
172 public function fatal( $message /*, parameters... */ ) {
173 $this->errors[] = [
174 'type' => 'error',
175 'message' => $message,
176 'params' => array_slice( func_get_args(), 1 )
178 $this->ok = false;
182 * Merge another status object into this one
184 * @param StatusValue $other Other StatusValue object
185 * @param bool $overwriteValue Whether to override the "value" member
187 public function merge( $other, $overwriteValue = false ) {
188 $this->errors = array_merge( $this->errors, $other->errors );
189 $this->ok = $this->ok && $other->ok;
190 if ( $overwriteValue ) {
191 $this->value = $other->value;
193 $this->successCount += $other->successCount;
194 $this->failCount += $other->failCount;
198 * Returns a list of status messages of the given type
200 * Each entry is a map of:
201 * - message: string message key or MessageSpecifier
202 * - params: array list of parameters
204 * @param string $type
205 * @return array
207 public function getErrorsByType( $type ) {
208 $result = [];
209 foreach ( $this->errors as $error ) {
210 if ( $error['type'] === $type ) {
211 $result[] = $error;
215 return $result;
219 * Returns true if the specified message is present as a warning or error
221 * @param string|MessageSpecifier $message Message key or object to search for
223 * @return bool
225 public function hasMessage( $message ) {
226 if ( $message instanceof MessageSpecifier ) {
227 $message = $message->getKey();
229 foreach ( $this->errors as $error ) {
230 if ( $error['message'] instanceof MessageSpecifier
231 && $error['message']->getKey() === $message
233 return true;
234 } elseif ( $error['message'] === $message ) {
235 return true;
239 return false;
243 * If the specified source message exists, replace it with the specified
244 * destination message, but keep the same parameters as in the original error.
246 * Note, due to the lack of tools for comparing IStatusMessage objects, this
247 * function will not work when using such an object as the search parameter.
249 * @param IStatusMessage|string $source Message key or object to search for
250 * @param IStatusMessage|string $dest Replacement message key or object
251 * @return bool Return true if the replacement was done, false otherwise.
253 public function replaceMessage( $source, $dest ) {
254 $replaced = false;
256 foreach ( $this->errors as $index => $error ) {
257 if ( $error['message'] === $source ) {
258 $this->errors[$index]['message'] = $dest;
259 $replaced = true;
263 return $replaced;
267 * @return string
269 public function __toString() {
270 $status = $this->isOK() ? "OK" : "Error";
271 if ( count( $this->errors ) ) {
272 $errorcount = "collected " . ( count( $this->errors ) ) . " error(s) on the way";
273 } else {
274 $errorcount = "no errors detected";
276 if ( isset( $this->value ) ) {
277 $valstr = gettype( $this->value ) . " value set";
278 if ( is_object( $this->value ) ) {
279 $valstr .= "\"" . get_class( $this->value ) . "\" instance";
281 } else {
282 $valstr = "no value set";
284 $out = sprintf( "<%s, %s, %s>",
285 $status,
286 $errorcount,
287 $valstr
289 if ( count( $this->errors ) > 0 ) {
290 $hdr = sprintf( "+-%'-4s-+-%'-25s-+-%'-40s-+\n", "", "", "" );
291 $i = 1;
292 $out .= "\n";
293 $out .= $hdr;
294 foreach ( $this->errors as $error ) {
295 if ( $error['message'] instanceof MessageSpecifier ) {
296 $key = $error['message']->getKey();
297 $params = $error['message']->getParams();
298 } elseif ( $error['params'] ) {
299 $key = $error['message'];
300 $params = $error['params'];
301 } else {
302 $key = $error['message'];
303 $params = [];
306 $out .= sprintf( "| %4d | %-25.25s | %-40.40s |\n",
308 $key,
309 implode( " ", $params )
311 $i += 1;
313 $out .= $hdr;
316 return $out;