You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							794 lines
						
					
					
						
							19 KiB
						
					
					
				
			
		
		
	
	
							794 lines
						
					
					
						
							19 KiB
						
					
					
				| <?php | |
| 
 | |
| declare(strict_types=1); | |
| 
 | |
| namespace PhpMyAdmin; | |
| 
 | |
| use Stringable; | |
| 
 | |
| use function __; | |
| use function _ngettext; | |
| use function array_unshift; | |
| use function count; | |
| use function htmlspecialchars; | |
| use function is_array; | |
| use function is_float; | |
| use function is_int; | |
| use function md5; | |
| use function sprintf; | |
| use function strlen; | |
| 
 | |
| use const ENT_COMPAT; | |
| 
 | |
| /** | |
|  * a single message | |
|  * | |
|  * simple usage examples: | |
|  * <code> | |
|  * // display simple error message 'Error' | |
|  * echo Message::error()->getDisplay(); | |
|  * | |
|  * // get simple success message 'Success' | |
|  * $message = Message::success(); | |
|  * | |
|  * // get special notice | |
|  * $message = Message::notice(__('This is a localized notice')); | |
|  * </code> | |
|  * | |
|  * more advanced usage example: | |
|  * <code> | |
|  * // create another message, a hint, with a localized string which expects | |
|  * $hint = Message::notice('Read the %smanual%s'); | |
|  * // replace placeholders with the following params | |
|  * $hint->addParam('[doc@cfg_Example]'); | |
|  * $hint->addParam('[/doc]'); | |
|  * // add this hint as a tooltip | |
|  * $hint = showHint($hint); | |
|  * | |
|  * // add the retrieved tooltip reference to the original message | |
|  * $message->addMessage($hint); | |
|  * </code> | |
|  */ | |
| class Message implements Stringable | |
| { | |
|     public const SUCCESS = 1; // 0001 | |
|     public const NOTICE = 2; // 0010 | |
|     public const ERROR = 8; // 1000 | |
|  | |
|     public const SANITIZE_NONE = 0; // 0000 0000 | |
|     public const SANITIZE_STRING = 16; // 0001 0000 | |
|     public const SANITIZE_PARAMS = 32; // 0010 0000 | |
|     public const SANITIZE_BOOTH = 48; // 0011 0000 | |
|  | |
|     /** | |
|      * message levels | |
|      * | |
|      * @var array | |
|      */ | |
|     public static $level = [ | |
|         self::SUCCESS => 'success', | |
|         self::NOTICE => 'notice', | |
|         self::ERROR => 'error', | |
|     ]; | |
| 
 | |
|     /** | |
|      * The message number | |
|      * | |
|      * @var int | |
|      */ | |
|     protected $number = self::NOTICE; | |
| 
 | |
|     /** | |
|      * The locale string identifier | |
|      * | |
|      * @var    string | |
|      */ | |
|     protected $string = ''; | |
| 
 | |
|     /** | |
|      * The formatted message | |
|      * | |
|      * @var    string | |
|      */ | |
|     protected $message = ''; | |
| 
 | |
|     /** | |
|      * Whether the message was already displayed | |
|      * | |
|      * @var bool | |
|      */ | |
|     protected $isDisplayed = false; | |
| 
 | |
|     /** | |
|      * Whether to use BB code when displaying. | |
|      * | |
|      * @var bool | |
|      */ | |
|     protected $useBBCode = true; | |
| 
 | |
|     /** | |
|      * Unique id | |
|      * | |
|      * @var string | |
|      */ | |
|     protected $hash = null; | |
| 
 | |
|     /** | |
|      * holds parameters | |
|      * | |
|      * @var    array | |
|      */ | |
|     protected $params = []; | |
| 
 | |
|     /** | |
|      * holds additional messages | |
|      * | |
|      * @var    array | |
|      */ | |
|     protected $addedMessages = []; | |
| 
 | |
|     /** | |
|      * @param string $string   The message to be displayed | |
|      * @param int    $number   A numeric representation of the type of message | |
|      * @param array  $params   An array of parameters to use in the message | |
|      * @param int    $sanitize A flag to indicate what to sanitize, see | |
|      *                         constant definitions above | |
|      */ | |
|     public function __construct( | |
|         string $string = '', | |
|         int $number = self::NOTICE, | |
|         array $params = [], | |
|         int $sanitize = self::SANITIZE_NONE | |
|     ) { | |
|         $this->setString($string, $sanitize & self::SANITIZE_STRING); | |
|         $this->setNumber($number); | |
|         $this->setParams($params, $sanitize & self::SANITIZE_PARAMS); | |
|     } | |
| 
 | |
|     /** | |
|      * magic method: return string representation for this object | |
|      */ | |
|     public function __toString(): string | |
|     { | |
|         return $this->getMessage(); | |
|     } | |
| 
 | |
|     /** | |
|      * get Message of type success | |
|      * | |
|      * shorthand for getting a simple success message | |
|      * | |
|      * @param string $string A localized string | |
|      *                       e.g. __('Your SQL query has been | |
|      *                       executed successfully') | |
|      * | |
|      * @return Message | |
|      * | |
|      * @static | |
|      */ | |
|     public static function success(string $string = ''): self | |
|     { | |
|         if (empty($string)) { | |
|             $string = __('Your SQL query has been executed successfully.'); | |
|         } | |
| 
 | |
|         return new Message($string, self::SUCCESS); | |
|     } | |
| 
 | |
|     /** | |
|      * get Message of type error | |
|      * | |
|      * shorthand for getting a simple error message | |
|      * | |
|      * @param string $string A localized string e.g. __('Error') | |
|      * | |
|      * @return Message | |
|      * | |
|      * @static | |
|      */ | |
|     public static function error(string $string = ''): self | |
|     { | |
|         if (empty($string)) { | |
|             $string = __('Error'); | |
|         } | |
| 
 | |
|         return new Message($string, self::ERROR); | |
|     } | |
| 
 | |
|     /** | |
|      * get Message of type notice | |
|      * | |
|      * shorthand for getting a simple notice message | |
|      * | |
|      * @param string $string A localized string | |
|      *                       e.g. __('The additional features for working with | |
|      *                       linked tables have been deactivated. To find out | |
|      *                       why click %shere%s.') | |
|      * | |
|      * @return Message | |
|      * | |
|      * @static | |
|      */ | |
|     public static function notice(string $string): self | |
|     { | |
|         return new Message($string, self::NOTICE); | |
|     } | |
| 
 | |
|     /** | |
|      * get Message with customized content | |
|      * | |
|      * shorthand for getting a customized message | |
|      * | |
|      * @param string $message A localized string | |
|      * @param int    $type    A numeric representation of the type of message | |
|      * | |
|      * @return Message | |
|      * | |
|      * @static | |
|      */ | |
|     public static function raw(string $message, int $type = self::NOTICE): self | |
|     { | |
|         $r = new Message('', $type); | |
|         $r->setMessage($message); | |
|         $r->setBBCode(false); | |
| 
 | |
|         return $r; | |
|     } | |
| 
 | |
|     /** | |
|      * get Message for number of affected rows | |
|      * | |
|      * shorthand for getting a customized message | |
|      * | |
|      * @param int $rows Number of rows | |
|      * | |
|      * @return Message | |
|      * | |
|      * @static | |
|      */ | |
|     public static function getMessageForAffectedRows(int $rows): self | |
|     { | |
|         $message = self::success( | |
|             _ngettext('%1$d row affected.', '%1$d rows affected.', $rows) | |
|         ); | |
|         $message->addParam($rows); | |
| 
 | |
|         return $message; | |
|     } | |
| 
 | |
|     /** | |
|      * get Message for number of deleted rows | |
|      * | |
|      * shorthand for getting a customized message | |
|      * | |
|      * @param int $rows Number of rows | |
|      * | |
|      * @return Message | |
|      * | |
|      * @static | |
|      */ | |
|     public static function getMessageForDeletedRows(int $rows): self | |
|     { | |
|         $message = self::success( | |
|             _ngettext('%1$d row deleted.', '%1$d rows deleted.', $rows) | |
|         ); | |
|         $message->addParam($rows); | |
| 
 | |
|         return $message; | |
|     } | |
| 
 | |
|     /** | |
|      * get Message for number of inserted rows | |
|      * | |
|      * shorthand for getting a customized message | |
|      * | |
|      * @param int $rows Number of rows | |
|      * | |
|      * @return Message | |
|      * | |
|      * @static | |
|      */ | |
|     public static function getMessageForInsertedRows(int $rows): self | |
|     { | |
|         $message = self::success( | |
|             _ngettext('%1$d row inserted.', '%1$d rows inserted.', $rows) | |
|         ); | |
|         $message->addParam($rows); | |
| 
 | |
|         return $message; | |
|     } | |
| 
 | |
|     /** | |
|      * get Message of type error with custom content | |
|      * | |
|      * shorthand for getting a customized error message | |
|      * | |
|      * @param string $message A localized string | |
|      * | |
|      * @return Message | |
|      * | |
|      * @static | |
|      */ | |
|     public static function rawError(string $message): self | |
|     { | |
|         return self::raw($message, self::ERROR); | |
|     } | |
| 
 | |
|     /** | |
|      * get Message of type notice with custom content | |
|      * | |
|      * shorthand for getting a customized notice message | |
|      * | |
|      * @param string $message A localized string | |
|      * | |
|      * @return Message | |
|      * | |
|      * @static | |
|      */ | |
|     public static function rawNotice(string $message): self | |
|     { | |
|         return self::raw($message, self::NOTICE); | |
|     } | |
| 
 | |
|     /** | |
|      * get Message of type success with custom content | |
|      * | |
|      * shorthand for getting a customized success message | |
|      * | |
|      * @param string $message A localized string | |
|      * | |
|      * @return Message | |
|      * | |
|      * @static | |
|      */ | |
|     public static function rawSuccess(string $message): self | |
|     { | |
|         return self::raw($message, self::SUCCESS); | |
|     } | |
| 
 | |
|     /** | |
|      * returns whether this message is a success message or not | |
|      * and optionally makes this message a success message | |
|      * | |
|      * @param bool $set Whether to make this message of SUCCESS type | |
|      */ | |
|     public function isSuccess(bool $set = false): bool | |
|     { | |
|         if ($set) { | |
|             $this->setNumber(self::SUCCESS); | |
|         } | |
| 
 | |
|         return $this->getNumber() === self::SUCCESS; | |
|     } | |
| 
 | |
|     /** | |
|      * returns whether this message is a notice message or not | |
|      * and optionally makes this message a notice message | |
|      * | |
|      * @param bool $set Whether to make this message of NOTICE type | |
|      */ | |
|     public function isNotice(bool $set = false): bool | |
|     { | |
|         if ($set) { | |
|             $this->setNumber(self::NOTICE); | |
|         } | |
| 
 | |
|         return $this->getNumber() === self::NOTICE; | |
|     } | |
| 
 | |
|     /** | |
|      * returns whether this message is an error message or not | |
|      * and optionally makes this message an error message | |
|      * | |
|      * @param bool $set Whether to make this message of ERROR type | |
|      */ | |
|     public function isError(bool $set = false): bool | |
|     { | |
|         if ($set) { | |
|             $this->setNumber(self::ERROR); | |
|         } | |
| 
 | |
|         return $this->getNumber() === self::ERROR; | |
|     } | |
| 
 | |
|     /** | |
|      * Set whether we should use BB Code when rendering. | |
|      * | |
|      * @param bool $useBBCode Use BB Code? | |
|      */ | |
|     public function setBBCode(bool $useBBCode): void | |
|     { | |
|         $this->useBBCode = $useBBCode; | |
|     } | |
| 
 | |
|     /** | |
|      * set raw message (overrides string) | |
|      * | |
|      * @param string $message  A localized string | |
|      * @param bool   $sanitize Whether to sanitize $message or not | |
|      */ | |
|     public function setMessage(string $message, bool $sanitize = false): void | |
|     { | |
|         if ($sanitize) { | |
|             $message = self::sanitize($message); | |
|         } | |
| 
 | |
|         $this->message = $message; | |
|     } | |
| 
 | |
|     /** | |
|      * set string (does not take effect if raw message is set) | |
|      * | |
|      * @param string   $string   string to set | |
|      * @param bool|int $sanitize whether to sanitize $string or not | |
|      */ | |
|     public function setString(string $string, $sanitize = true): void | |
|     { | |
|         if ($sanitize) { | |
|             $string = self::sanitize($string); | |
|         } | |
| 
 | |
|         $this->string = $string; | |
|     } | |
| 
 | |
|     /** | |
|      * set message type number | |
|      * | |
|      * @param int $number message type number to set | |
|      */ | |
|     public function setNumber(int $number): void | |
|     { | |
|         $this->number = $number; | |
|     } | |
| 
 | |
|     /** | |
|      * add string or Message parameter | |
|      * | |
|      * usage | |
|      * <code> | |
|      * $message->addParam('[em]some string[/em]'); | |
|      * </code> | |
|      * | |
|      * @param mixed $param parameter to add | |
|      */ | |
|     public function addParam($param): void | |
|     { | |
|         if ($param instanceof self || is_float($param) || is_int($param)) { | |
|             $this->params[] = $param; | |
|         } else { | |
|             $this->params[] = htmlspecialchars((string) $param, ENT_COMPAT); | |
|         } | |
|     } | |
| 
 | |
|     /** | |
|      * add parameter as raw HTML, usually in conjunction with strings | |
|      * | |
|      * usage | |
|      * <code> | |
|      * $message->addParamHtml('<img src="img">'); | |
|      * </code> | |
|      * | |
|      * @param string $param parameter to add | |
|      */ | |
|     public function addParamHtml(string $param): void | |
|     { | |
|         $this->params[] = self::notice($param); | |
|     } | |
| 
 | |
|     /** | |
|      * add a bunch of messages at once | |
|      * | |
|      * @param Message[] $messages  to be added | |
|      * @param string    $separator to use between this and previous string/message | |
|      */ | |
|     public function addMessages(array $messages, string $separator = ' '): void | |
|     { | |
|         foreach ($messages as $message) { | |
|             $this->addMessage($message, $separator); | |
|         } | |
|     } | |
| 
 | |
|     /** | |
|      * add a bunch of messages at once | |
|      * | |
|      * @param string[] $messages  to be added | |
|      * @param string   $separator to use between this and previous string/message | |
|      */ | |
|     public function addMessagesString(array $messages, string $separator = ' '): void | |
|     { | |
|         foreach ($messages as $message) { | |
|             $this->addText($message, $separator); | |
|         } | |
|     } | |
| 
 | |
|     /** | |
|      * Real implementation of adding message | |
|      * | |
|      * @param Message $message   to be added | |
|      * @param string  $separator to use between this and previous string/message | |
|      */ | |
|     private function addMessageToList(self $message, string $separator): void | |
|     { | |
|         if (! empty($separator)) { | |
|             $this->addedMessages[] = $separator; | |
|         } | |
| 
 | |
|         $this->addedMessages[] = $message; | |
|     } | |
| 
 | |
|     /** | |
|      * add another raw message to be concatenated on displaying | |
|      * | |
|      * @param self   $message   to be added | |
|      * @param string $separator to use between this and previous string/message | |
|      */ | |
|     public function addMessage(self $message, string $separator = ' '): void | |
|     { | |
|         $this->addMessageToList($message, $separator); | |
|     } | |
| 
 | |
|     /** | |
|      * add another raw message to be concatenated on displaying | |
|      * | |
|      * @param string $message   to be added | |
|      * @param string $separator to use between this and previous string/message | |
|      */ | |
|     public function addText(string $message, string $separator = ' '): void | |
|     { | |
|         $this->addMessageToList(self::notice(htmlspecialchars($message)), $separator); | |
|     } | |
| 
 | |
|     /** | |
|      * add another html message to be concatenated on displaying | |
|      * | |
|      * @param string $message   to be added | |
|      * @param string $separator to use between this and previous string/message | |
|      */ | |
|     public function addHtml(string $message, string $separator = ' '): void | |
|     { | |
|         $this->addMessageToList(self::rawNotice($message), $separator); | |
|     } | |
| 
 | |
|     /** | |
|      * set all params at once, usually used in conjunction with string | |
|      * | |
|      * @param array    $params   parameters to set | |
|      * @param bool|int $sanitize whether to sanitize params | |
|      */ | |
|     public function setParams(array $params, $sanitize = false): void | |
|     { | |
|         if ($sanitize) { | |
|             $params = self::sanitize($params); | |
|         } | |
| 
 | |
|         $this->params = $params; | |
|     } | |
| 
 | |
|     /** | |
|      * return all parameters | |
|      * | |
|      * @return array | |
|      */ | |
|     public function getParams(): array | |
|     { | |
|         return $this->params; | |
|     } | |
| 
 | |
|     /** | |
|      * return all added messages | |
|      * | |
|      * @return array | |
|      */ | |
|     public function getAddedMessages(): array | |
|     { | |
|         return $this->addedMessages; | |
|     } | |
| 
 | |
|     /** | |
|      * Sanitizes $message | |
|      * | |
|      * @param mixed $message the message(s) | |
|      * | |
|      * @return mixed  the sanitized message(s) | |
|      * | |
|      * @static | |
|      */ | |
|     public static function sanitize($message) | |
|     { | |
|         if (is_array($message)) { | |
|             foreach ($message as $key => $val) { | |
|                 $message[$key] = self::sanitize($val); | |
|             } | |
| 
 | |
|             return $message; | |
|         } | |
| 
 | |
|         return htmlspecialchars((string) $message); | |
|     } | |
| 
 | |
|     /** | |
|      * decode $message, taking into account our special codes | |
|      * for formatting | |
|      * | |
|      * @param string $message the message | |
|      * | |
|      * @return string  the decoded message | |
|      * | |
|      * @static | |
|      */ | |
|     public static function decodeBB(string $message): string | |
|     { | |
|         return Sanitize::sanitizeMessage($message, false, true); | |
|     } | |
| 
 | |
|     /** | |
|      * wrapper for sprintf() | |
|      * | |
|      * @param mixed[] ...$params Params | |
|      * | |
|      * @return string formatted | |
|      */ | |
|     public static function format(...$params): string | |
|     { | |
|         if (isset($params[1]) && is_array($params[1])) { | |
|             array_unshift($params[1], $params[0]); | |
|             $params = $params[1]; | |
|         } | |
| 
 | |
|         return sprintf(...$params); | |
|     } | |
| 
 | |
|     /** | |
|      * returns unique Message::$hash, if not exists it will be created | |
|      * | |
|      * @return string Message::$hash | |
|      */ | |
|     public function getHash(): string | |
|     { | |
|         if ($this->hash === null) { | |
|             $this->hash = md5( | |
|                 $this->getNumber() . | |
|                 $this->string . | |
|                 $this->message | |
|             ); | |
|         } | |
| 
 | |
|         return $this->hash; | |
|     } | |
| 
 | |
|     /** | |
|      * returns compiled message | |
|      * | |
|      * @return string complete message | |
|      */ | |
|     public function getMessage(): string | |
|     { | |
|         $message = $this->message; | |
| 
 | |
|         if (strlen($message) === 0) { | |
|             $string = $this->getString(); | |
|             if (strlen($string) === 0) { | |
|                 $message = ''; | |
|             } else { | |
|                 $message = $string; | |
|             } | |
|         } | |
| 
 | |
|         if ($this->isDisplayed()) { | |
|             $message = $this->getMessageWithIcon($message); | |
|         } | |
| 
 | |
|         if (count($this->getParams()) > 0) { | |
|             $message = self::format($message, $this->getParams()); | |
|         } | |
| 
 | |
|         if ($this->useBBCode) { | |
|             $message = self::decodeBB($message); | |
|         } | |
| 
 | |
|         foreach ($this->getAddedMessages() as $add_message) { | |
|             $message .= $add_message; | |
|         } | |
| 
 | |
|         return $message; | |
|     } | |
| 
 | |
|     /** | |
|      * Returns only message string without image & other HTML. | |
|      */ | |
|     public function getOnlyMessage(): string | |
|     { | |
|         return $this->message; | |
|     } | |
| 
 | |
|     /** | |
|      * returns Message::$string | |
|      * | |
|      * @return string Message::$string | |
|      */ | |
|     public function getString(): string | |
|     { | |
|         return $this->string; | |
|     } | |
| 
 | |
|     /** | |
|      * returns Message::$number | |
|      * | |
|      * @return int Message::$number | |
|      */ | |
|     public function getNumber(): int | |
|     { | |
|         return $this->number; | |
|     } | |
| 
 | |
|     /** | |
|      * returns level of message | |
|      * | |
|      * @return string level of message | |
|      */ | |
|     public function getLevel(): string | |
|     { | |
|         return self::$level[$this->getNumber()]; | |
|     } | |
| 
 | |
|     /** | |
|      * returns HTML code for displaying this message | |
|      * | |
|      * @return string whole message box | |
|      */ | |
|     public function getDisplay(): string | |
|     { | |
|         $this->isDisplayed(true); | |
| 
 | |
|         $context = 'primary'; | |
|         $level = $this->getLevel(); | |
|         if ($level === 'error') { | |
|             $context = 'danger'; | |
|         } elseif ($level === 'success') { | |
|             $context = 'success'; | |
|         } | |
| 
 | |
|         $template = new Template(); | |
| 
 | |
|         return $template->render('message', [ | |
|             'context' => $context, | |
|             'message' => $this->getMessage(), | |
|         ]); | |
|     } | |
| 
 | |
|     /** | |
|      * sets and returns whether the message was displayed or not | |
|      * | |
|      * @param bool $isDisplayed whether to set displayed flag | |
|      */ | |
|     public function isDisplayed(bool $isDisplayed = false): bool | |
|     { | |
|         if ($isDisplayed) { | |
|             $this->isDisplayed = true; | |
|         } | |
| 
 | |
|         return $this->isDisplayed; | |
|     } | |
| 
 | |
|     /** | |
|      * Returns the message with corresponding image icon | |
|      * | |
|      * @param string $message the message(s) | |
|      * | |
|      * @return string message with icon | |
|      */ | |
|     public function getMessageWithIcon(string $message): string | |
|     { | |
|         if ($this->getLevel() === 'error') { | |
|             $image = 's_error'; | |
|         } elseif ($this->getLevel() === 'success') { | |
|             $image = 's_success'; | |
|         } else { | |
|             $image = 's_notice'; | |
|         } | |
| 
 | |
|         $message = self::notice(Html\Generator::getImage($image)) . ' ' . $message; | |
| 
 | |
|         return $message; | |
|     } | |
| }
 | |
| 
 |