HEX
Server: Apache/2.4.52 (Ubuntu)
System: Linux WebLive 5.15.0-79-generic #86-Ubuntu SMP Mon Jul 10 16:07:21 UTC 2023 x86_64
User: ubuntu (1000)
PHP: 7.4.33
Disabled: pcntl_alarm,pcntl_fork,pcntl_waitpid,pcntl_wait,pcntl_wifexited,pcntl_wifstopped,pcntl_wifsignaled,pcntl_wifcontinued,pcntl_wexitstatus,pcntl_wtermsig,pcntl_wstopsig,pcntl_signal,pcntl_signal_get_handler,pcntl_signal_dispatch,pcntl_get_last_error,pcntl_strerror,pcntl_sigprocmask,pcntl_sigwaitinfo,pcntl_sigtimedwait,pcntl_exec,pcntl_getpriority,pcntl_setpriority,pcntl_async_signals,pcntl_unshare,
Upload Files
File: /var/www/html/wptrinityconsulting/wp-content/plugins/wp-security-audit-log/classes/Alert.php
<?php
/**
 * WSAL_Alert class.
 *
 * @package wsal
 */

// Exit if accessed directly.
if ( ! defined( 'ABSPATH' ) ) {
	exit;
}

/**
 * WSAL_Alert Object class.
 *
 * @package wsal
 */
final class WSAL_Alert {

	/**
	 * Alert type (used when triggering an alert etc).
	 *
	 * @var integer
	 */
	public $code = 0;

	/**
	 * Alert error level (E_* constant).
	 *
	 * @var integer
	 */
	public $severity = 0;

	/**
	 * Alert category (alerts are grouped by matching categories).
	 *
	 * @var string
	 */
	public $catg = '';

	/**
	 * Alert sub category.
	 *
	 * @var string
	 */
	public $subcatg = '';

	/**
	 * Alert description (ie, describes what happens when alert is triggered).
	 *
	 * @var string
	 */
	public $desc = '';

	/**
	 * Alert message (variables between '%' are expanded to values).
	 *
	 * @var string
	 */
	public $mesg = '';

	/**
	 * Event object.
	 *
	 * @var string
	 */
	public $object = '';

	/**
	 * Event type.
	 *
	 * @var string
	 */
	public $event_type = '';

	/**
	 * List of metadata items containing metadata key and a label to be displayed.
	 *
	 * @var array
	 * @since 4.2.1
	 */
	public $metadata = array();

	/**
	 * List of metadata items containing metadata key and a label to be displayed.
	 *
	 * @var string[]
	 * @since 4.2.1
	 */
	public $links = array();

	/**
	 * Constructor.
	 *
	 * @param integer $type - Type of alert.
	 * @param integer $code - Code of alert.
	 * @param string  $catg - Category of alert.
	 * @param string  $subcatg - Subcategory of alert.
	 * @param string  $desc - Description.
	 * @param string  $mesg - Alert message.
	 * @param array   $metadata - List of metadata items containing metadata key and a label to be displayed.
	 * @param string  $links - This should be a list of links in form of dynamic placeholders or a metadata names.
	 * @param string  $object - Event object.
	 * @param string  $event_type - Event type.
	 */
	public function __construct( $type = 0, $code = 0, $catg = '', $subcatg = '', $desc = '', $mesg = '', $metadata = array(), $links = '', $object = '', $event_type = '' ) {
		$this->code       = $type;
		$this->severity   = $code;
		$this->catg       = $catg;
		$this->subcatg    = $subcatg;
		$this->desc       = $desc;
		$this->mesg       = $mesg;
		$this->object     = $object;
		$this->event_type = $event_type;
		$this->metadata   = $metadata;
		$this->links      = $links;
	}

	/**
	 * Gets alert message.
	 *
	 * Note: not to be used to display any messages. Use WSAL_Models_Occurrence::GetMessage() instead.
	 *
	 * @param array        $meta_data - (Optional) Meta data relevant to message.
	 * @param string|null  $message - (Optional) Override message template to use.
	 * @param integer      $occurrence_id - (Optional) Event occurrence ID.
	 * @param string|false $context - Context in which the message will be used/displayed.
	 *
	 * @return string Fully formatted message.
	 *
	 * @see WSAL_Models_Occurrence::get_message()
	 */
	public function get_message( $meta_data = array(), $message = null, $occurrence_id = 0, $context = false ) {
		return $this->get_formatted_message( is_null( $message ) ? $this->mesg : $message, $meta_data, $occurrence_id, $context );
	}

	/**
	 * Expands a message with variables by replacing variables with meta data values.
	 *
	 * @param string       $original_message - The original message.
	 * @param array        $meta_data - (Optional) Meta data relevant to message.
	 * @param integer      $occurrence_id - (Optional) Event occurrence ID.
	 * @param string|false $context - Context in which the message will be used/displayed.
	 *
	 * @return string The expanded message.
	 */
	protected function get_formatted_message( $original_message, $meta_data = array(), $occurrence_id = 0, $context = false ) {

		$result = '';

		// Fallback on the default context.
		if ( false === $context ) {
			$context = 'default';
		}

		// Get the alert formatter for given context.
		$formatter = WSAL_AlertFormatterFactory::get_formatter( $context );

		// Tokenize message with regex.
		$message_parts = preg_split( '/(%.*?%)/', (string) $original_message, - 1, PREG_SPLIT_DELIM_CAPTURE );
		if ( ! is_array( $message_parts ) ) {
			// Use the message as is.
			$result = (string) $original_message;
		} else {
			// Handle tokenized message.
			if ( ! empty( $message_parts ) ) {
				foreach ( $message_parts as $i => $token ) {
					if ( strlen( $token ) === 0 ) {
						continue;
					}
					// Handle escaped percent sign.
					if ( '%%' === $token ) {
						$message_parts[ $i ] = '%';
					} elseif ( substr( $token, 0, 1 ) === '%' && substr( $token, - 1, 1 ) === '%' ) {
						// Handle complex expressions.
						$message_parts[ $i ] = $this->get_meta_expression_value( substr( $token, 1, - 1 ), $meta_data );
						$message_parts[ $i ] = $formatter->format_meta_expression( $token, $message_parts[ $i ], $occurrence_id );
					}
				}

				// Compact message.
				$result = implode( '', $message_parts );
			}
		}

		// Process message to make sure it any HTML tags are handled correctly.
		$result = $formatter->process_html_tags_in_message( $result );

		$end_of_line = $formatter->get_end_of_line();

		// Process metadata and links introduced as part of alert definition in version 4.2.1.
		if ( $formatter->supports_metadata() ) {
			$metadata_result = $this->get_formatted_metadata( $formatter, $meta_data, $occurrence_id );
			if ( ! empty( $metadata_result ) ) {
				if ( ! empty( $result ) ) {
					$result .= $end_of_line;
				}
				$result .= $metadata_result;
			}
		}

		if ( $formatter->supports_hyperlinks() ) {
			$hyperlinks_result = $this->get_formatted_hyperlinks( $formatter, $meta_data, $occurrence_id );
			if ( ! empty( $hyperlinks_result ) ) {
				if ( ! empty( $result ) ) {
					$result .= $end_of_line;
				}
				$result .= $hyperlinks_result;
			}
		}

		return $result;
	}

	/**
	 * Retrieves a value for a particular meta variable expression.
	 *
	 * @param string $expr Expression, eg: User->Name looks for a Name property for meta named User.
	 * @param array  $meta_data (Optional) Meta data relevant to expression.
	 *
	 * @return mixed The value nearest to the expression.
	 */
	protected function get_meta_expression_value( $expr, $meta_data = array() ) {
		$expr = preg_replace( '/%/', '', $expr );
		if ( 'IPAddress' === $expr ) {
			if ( array_key_exists( 'IPAddress', $meta_data ) ) {
				return implode( ', ', $meta_data['IPAddress'] );
			}

			return null;
		}

		// TODO: Handle function calls (and methods?).
		$expr = explode( '->', $expr );
		$meta = array_shift( $expr );
		$meta = isset( $meta_data[ $meta ] ) ? $meta_data[ $meta ] : null;
		foreach ( $expr as $part ) {
			if ( is_scalar( $meta ) || is_null( $meta ) ) {
				return $meta; // This isn't 100% correct.
			}
			$meta = is_array( $meta ) && array_key_exists( $part, $meta ) ? $meta[ $part ] : ( isset( $meta->$part ) ? $meta->$part : 'NULL' );
		}

		return is_scalar( $meta ) ? (string) $meta : var_export( $meta, true ); // phpcs:ignore
	}

	/**
	 * Retrieves formatted meta data item (label and data).
	 *
	 * @param WSAL_AlertFormatter $formatter Alert formatter.
	 * @param array               $meta_data Meta data.
	 * @param int                 $occurrence_id Occurrence ID.
	 *
	 * @return string
	 * @since 4.2.1
	 */
	public function get_formatted_metadata( $formatter, $meta_data, $occurrence_id ) {
		$result            = '';
		$metadata_as_array = $this->get_metadata_as_array( $formatter, $meta_data, $occurrence_id );
		if ( ! empty( $metadata_as_array ) ) {

			$meta_result_parts = array();
			foreach ( $metadata_as_array as $meta_label => $meta_expression ) {
				if ( ! empty( $meta_expression ) ) {
					array_push( $meta_result_parts, $meta_label . ': ' . $meta_expression );
				}
			}

			if ( ! empty( $meta_result_parts ) ) {
				$result .= implode( $formatter->get_end_of_line(), $meta_result_parts );
			}
		}
		return $result;
	}

	/**
	 * Retrieves metadata as an associative array.
	 *
	 * @param WSAL_AlertFormatter $formatter Alert formatter.
	 * @param array               $meta_data Meta data.
	 * @param int                 $occurrence_id Occurrence ID.
	 *
	 * @return array
	 * @since 4.2.1
	 */
	public function get_metadata_as_array( $formatter, $meta_data, $occurrence_id ) {
		$result = array();
		if ( ! empty( $this->metadata ) ) {
			foreach ( $this->metadata as $meta_label => $meta_token ) {
				if ( strlen( $meta_token ) === 0 ) {
					continue;
				}

				// Pure alert meta lookup based on meta token.
				$meta_expression = $this->get_meta_expression_value( $meta_token, $meta_data );

				// Additional alert meta processing - handles derived or decorated alert data.
				$meta_expression = $formatter->format_meta_expression( $meta_token, $meta_expression, $occurrence_id );

				if ( ! empty( $meta_expression ) ) {
					$result[ $meta_label ] = $meta_expression;
				}
			}
		}

		return $result;
	}

	/**
	 * Get formatter hyperlinks.
	 *
	 * @param WSAL_AlertFormatter $formatter     Alert formatter.
	 * @param array               $meta_data     Meta data.
	 * @param int                 $occurrence_id Occurrence ID.
	 *
	 * @return string
	 * @since 4.2.1
	 */
	public function get_formatted_hyperlinks( $formatter, $meta_data, $occurrence_id ) {
		$result              = '';
		$hyperlinks_as_array = $this->get_hyperlinks_as_array( $formatter, $meta_data, $occurrence_id );
		if ( ! empty( $hyperlinks_as_array ) ) {
			$links_result_parts = array();
			foreach ( $hyperlinks_as_array as  $link_data ) {
				$link_label       = $link_data['label'];
				$link_url         = $link_data['url'];
				$needs_formatting = $link_data['needs_formatting'];
				$formatted_link   = $needs_formatting ? $formatter->format_link( $link_url, $link_label ) : $link_url;
				array_push( $links_result_parts, $formatted_link );
			}

			if ( ! empty( $links_result_parts ) ) {
				$result .= implode( $formatter->get_end_of_line(), $links_result_parts );
			}
		}

		return $result;
	}

	/**
	 * Retrieves hyperlinks as an array.
	 *
	 * @param WSAL_AlertFormatter $formatter                            Alert formatter.
	 * @param array               $meta_data                            Meta data.
	 * @param int                 $occurrence_id                        Occurrence ID.
	 * @param bool                $exclude_links_not_needing_formatting If true, links that don't need formatting will
	 *                                                                  be excluded. For example special links that
	 *                                                                  contain onclick attribute already from the meta
	 *                                                                  formatter.
	 *
	 * @return string
	 * @since 4.2.1
	 */
	public function get_hyperlinks_as_array( $formatter, $meta_data, $occurrence_id, $exclude_links_not_needing_formatting = false ) {
		$result = array();
		if ( ! empty( $this->links ) ) {
			foreach ( $this->links as $link_label => $link_data ) {

				$link_title = '';
				$link_url   = '';
				if ( is_string( $link_data ) ) {
					if ( strlen( $link_data ) === 0 ) {
						continue;
					}

					$link_url   = $link_data;
					$link_title = $link_data;
				} else {
					$link_url   = $link_data['url'];
					$link_title = $link_data['label'];
				}

				/**
				 * Link url can be:
				 * - an actual URL
				 * - placeholder for an existing metadata field that contains a URL (or the full HTML A tag markup)
				 * -- before 4.2.1 the CommentLink meta would contain the full HTML markup for the link, now it contains only the URL
				 * - other placeholder for a dynamic or JS infused link that will be processed by the meta formatter.
				 */
				$needs_formatting = true;
				if ( ! WSAL_Utilities_RequestUtils::is_valid_url( $link_url ) ) {

					$meta_expression = $this->get_meta_expression_value( $link_url, $meta_data );
					$meta_expression = $formatter->format_meta_expression( $link_url, $meta_expression, $occurrence_id, $meta_data, false );
					if ( ! empty( $meta_expression ) ) {
						if ( WSAL_Utilities_RequestUtils::is_valid_url( $meta_expression ) ) {

							$link_url = $meta_expression;
						} elseif ( preg_match( '/onclick=/', $meta_expression ) ) {
							$link_url         = $meta_expression;
							$needs_formatting = false;
						} else {

							preg_match( '/href=["\']https?:\/\/([^"\']+)["\']/', $meta_expression, $url_matches );
							if ( count( $url_matches ) === 2 ) {
								$link_url = $url_matches[1];
							}
						}
					} else {
						$link_url = '';
					}
				}

				if ( $exclude_links_not_needing_formatting && ! $needs_formatting ) {
					continue;
				}

				if ( ! empty( $link_url ) ) {
					$result[ $link_label ] = array(
						'url'              => $link_url,
						'needs_formatting' => $needs_formatting,
						'title'            => $link_title,
						'label'            => $link_label,
					);
				}
			}
		}

		return $result;
	}
}