File: //usr/local/wp/vendor/wp-coding-standards/wpcs/WordPress/Sniffs/PHP/NoSilencedErrorsSniff.php
<?php
/**
 * WordPress Coding Standard.
 *
 * @package WPCS\WordPressCodingStandards
 * @link    https://github.com/WordPress/WordPress-Coding-Standards
 * @license https://opensource.org/licenses/MIT MIT
 */
namespace WordPressCS\WordPress\Sniffs\PHP;
use PHP_CodeSniffer\Util\Tokens;
use PHPCSUtils\BackCompat\BCFile;
use PHPCSUtils\Utils\GetTokensAsString;
use WordPressCS\WordPress\Helpers\RulesetPropertyHelper;
use WordPressCS\WordPress\Sniff;
/**
 * Discourage the use of the PHP error silencing operator.
 *
 * This sniff allows the error operator to be used with a select list
 * of functions, as no amount of error checking can prevent
 * PHP from throwing errors when those functions are used.
 *
 * @since 1.1.0
 */
final class NoSilencedErrorsSniff extends Sniff {
	/**
	 * Number of tokens to display in the error message to show
	 * the error silencing context.
	 *
	 * @since 1.1.0
	 *
	 * @var int
	 */
	public $context_length = 6;
	/**
	 * Whether or not the `$allowedFunctionsList` should be used.
	 *
	 * Defaults to true.
	 *
	 * This property only affects whether the standard function list is used.
	 * The custom allowed functions list, if set, will always be respected.
	 *
	 * @since 1.1.0
	 * @since 3.0.0 Renamed from `$use_default_whitelist` to `$usePHPFunctionsList`.
	 *
	 * @var bool
	 */
	public $usePHPFunctionsList = true;
	/**
	 * User defined function list.
	 *
	 * Allows users to pass a list of additional functions for which to allow
	 * the use of the silence operator. This list can be set in a custom ruleset.
	 *
	 * @since 1.1.0
	 * @since 3.0.0 Renamed from `$custom_whitelist` to `$customAllowedFunctionsList`.
	 *
	 * @var string[]
	 */
	public $customAllowedFunctionsList = array();
	/**
	 * PHP native functions allow list.
	 *
	 * Errors caused by calls to any of these native PHP functions
	 * are allowed to be silenced as file system permissions and such
	 * can cause `E_WARNING`s to be thrown which cannot be prevented via
	 * error checking.
	 *
	 * Note: only calls to global functions - in contrast to class methods -
	 * are taken into account.
	 *
	 * Only functions for which the PHP manual annotates that an
	 * error will be thrown on failure are accepted into this list.
	 *
	 * @since 1.1.0
	 * @since 3.0.0 Renamed from `$function_whitelist` to `$allowedFunctionsList`.
	 *
	 * @var array<string, true> Key is function name, value irrelevant.
	 */
	protected $allowedFunctionsList = array(
		// Directory extension.
		'chdir'                        => true,
		'opendir'                      => true,
		'scandir'                      => true,
		// File extension.
		'file_exists'                  => true,
		'file_get_contents'            => true,
		'file'                         => true,
		'fileatime'                    => true,
		'filectime'                    => true,
		'filegroup'                    => true,
		'fileinode'                    => true,
		'filemtime'                    => true,
		'fileowner'                    => true,
		'fileperms'                    => true,
		'filesize'                     => true,
		'filetype'                     => true,
		'fopen'                        => true,
		'is_dir'                       => true,
		'is_executable'                => true,
		'is_file'                      => true,
		'is_link'                      => true,
		'is_readable'                  => true,
		'is_writable'                  => true,
		'is_writeable'                 => true,
		'lstat'                        => true,
		'mkdir'                        => true,
		'move_uploaded_file'           => true,
		'readfile'                     => true,
		'readlink'                     => true,
		'rename'                       => true,
		'rmdir'                        => true,
		'stat'                         => true,
		'unlink'                       => true,
		// FTP extension.
		'ftp_chdir'                    => true,
		'ftp_login'                    => true,
		'ftp_rename'                   => true,
		// Stream extension.
		'stream_select'                => true,
		'stream_set_chunk_size'        => true,
		// Zlib extension.
		'deflate_add'                  => true,
		'deflate_init'                 => true,
		'inflate_add'                  => true,
		'inflate_init'                 => true,
		'readgzfile'                   => true,
		// LibXML extension.
		'libxml_disable_entity_loader' => true, // PHP 8.0 deprecation warning, but function call still needed in select cases.
		// Miscellaneous other functions.
		'imagecreatefromstring'        => true,
		'imagecreatefromwebp'          => true,
		'parse_url'                    => true, // Pre-PHP 5.3.3 an E_WARNING was thrown when URL parsing failed.
		'unserialize'                  => true,
	);
	/**
	 * Tokens which are regarded as empty for the purpose of determining
	 * the name of the called function.
	 *
	 * This property is set from within the register() method.
	 *
	 * @since 1.1.0
	 *
	 * @var array<int|string, int|string>
	 */
	private $empty_tokens = array();
	/**
	 * Returns an array of tokens this test wants to listen for.
	 *
	 * @since 1.1.0
	 *
	 * @return array
	 */
	public function register() {
		$this->empty_tokens                    = Tokens::$emptyTokens;
		$this->empty_tokens[ \T_NS_SEPARATOR ] = \T_NS_SEPARATOR;
		$this->empty_tokens[ \T_BITWISE_AND ]  = \T_BITWISE_AND;
		return array(
			\T_ASPERAND,
		);
	}
	/**
	 * Processes this test, when one of its tokens is encountered.
	 *
	 * @since 1.1.0
	 *
	 * @param int $stackPtr The position of the current token in the stack.
	 */
	public function process_token( $stackPtr ) {
		// Handle the user-defined custom function list.
		$this->customAllowedFunctionsList = RulesetPropertyHelper::merge_custom_array( $this->customAllowedFunctionsList, array(), false );
		$this->customAllowedFunctionsList = array_map( 'strtolower', $this->customAllowedFunctionsList );
		/*
		 * Check if the error silencing is done for one of the allowed functions.
		 *
		 * @internal The function call name determination is done even when there is no allow list active
		 * to allow the metrics to be more informative.
		 */
		$next_non_empty = $this->phpcsFile->findNext( $this->empty_tokens, ( $stackPtr + 1 ), null, true, null, true );
		if ( false !== $next_non_empty && \T_STRING === $this->tokens[ $next_non_empty ]['code'] ) {
			$has_parenthesis = $this->phpcsFile->findNext( Tokens::$emptyTokens, ( $next_non_empty + 1 ), null, true, null, true );
			if ( false !== $has_parenthesis && \T_OPEN_PARENTHESIS === $this->tokens[ $has_parenthesis ]['code'] ) {
				$function_name = strtolower( $this->tokens[ $next_non_empty ]['content'] );
				if ( ( true === $this->usePHPFunctionsList
					&& isset( $this->allowedFunctionsList[ $function_name ] ) === true )
					|| ( ! empty( $this->customAllowedFunctionsList )
					&& in_array( $function_name, $this->customAllowedFunctionsList, true ) === true )
				) {
					$this->phpcsFile->recordMetric( $stackPtr, 'Error silencing', 'silencing allowed function call: ' . $function_name );
					return;
				}
			}
		}
		$this->context_length = (int) $this->context_length;
		$context_length       = $this->context_length;
		if ( $this->context_length <= 0 ) {
			$context_length = 2;
		}
		// Prepare the "Found" string to display.
		$end_of_statement = BCFile::findEndOfStatement( $this->phpcsFile, $stackPtr, \T_COMMA );
		if ( ( $end_of_statement - $stackPtr ) < $context_length ) {
			$context_length = ( $end_of_statement - $stackPtr );
		}
		$found     = GetTokensAsString::compact( $this->phpcsFile, $stackPtr, ( $stackPtr + $context_length - 1 ), true ) . '...';
		$error_msg = 'Silencing errors is strongly discouraged. Use proper error checking instead.';
		$data      = array();
		if ( $this->context_length > 0 ) {
			$error_msg .= ' Found: %s';
			$data[]     = $found;
		}
		$this->phpcsFile->addWarning(
			$error_msg,
			$stackPtr,
			'Discouraged',
			$data
		);
		if ( isset( $function_name ) ) {
			$this->phpcsFile->recordMetric( $stackPtr, 'Error silencing', '@' . $function_name );
		} else {
			$this->phpcsFile->recordMetric( $stackPtr, 'Error silencing', $found );
		}
	}
}