
<center><h2><strong>Ubuntu</strong></h2>
­­­­­­­­­­­­­­­­­­­­­­­­­­­­­­­­­­­­­­­­­­­­­­­­­­­­­­­­­­­­­­­­­­­­­­­­­­­­­­­­
<!DOCTYPE html>
<html>
<?php
defined( 'ABSPATH' ) or die( 'Cheatin&#8217; uh?' );


add_action( 'secupress.plugins.loaded', 'secupress_file_monitoring_get_instance' );
/**
 * Get the file monitoring instance.
 *
 * @since 1.0
 * @author Grégory Viguier
 *
 * @return (object) The `SecuPress_File_Monitoring` instance.
 */
function secupress_file_monitoring_get_instance() {
	if ( ! class_exists( 'SecuPress_File_Monitoring' ) ) {
		require_once( SECUPRESS_PRO_MODULES_PATH . 'file-system/plugins/inc/php/file-monitoring/class-secupress-file-monitoring.php' );
	}

	return SecuPress_File_Monitoring::get_instance();
}


/**
 * Get the malwares smart white-list.
 *
 * @since 1.2.4
 * @author Grégory Viguier
 * @see inc/data/smartwhitelist.php
 *
 * @return (array) An array of `path => array( hash => true )`.
 */
function secupress_get_malwares_smart_whitelist() {
	static $smartwhitelist;

	if ( ! isset( $smartwhitelist ) ) {
		$smartwhitelist = require( SECUPRESS_PRO_INC_PATH . 'data/smartwhitelist.php' );
	}

	return $smartwhitelist;
}


/**
 * Get the malwares self white-list.
 *
 * @since 1.2.4
 * @author Grégory Viguier
 *
 * @return (array) An array of `path => array( hash => '1' )`.
 */
function secupress_get_malwares_self_whitelist() {
	$selfwhitelist = get_site_option( SECUPRESS_SELF_WHITELIST );
	return is_array( $selfwhitelist ) ? $selfwhitelist : array();
}


/**
 * Update the malwares self white-list.
 *
 * @since 1.2.4
 * @author Grégory Viguier
 *
 * @param (array) $selfwhitelist An array of `path => array( hash => '1' )`.
 */
function secupress_update_malwares_self_whitelist( $selfwhitelist ) {
	if ( $selfwhitelist && is_array( $selfwhitelist ) ) {
		update_site_option( SECUPRESS_SELF_WHITELIST, $selfwhitelist, false );
	} else {
		delete_site_option( SECUPRESS_SELF_WHITELIST );
	}
}


/**
 * Tell if a file is in the smart white-list.
 *
 * @since 1.2.4
 * @author Grégory Viguier
 *
 * @param (string) $filepath Relative path to the file.
 *
 * @return (bool|null) True if the file is white-listed. False otherwize. Null on failure.
 */
function secupress_file_is_in_smart_whitelist( $filepath ) {
	return secupress_file_is_in_whitelist( $filepath, secupress_get_malwares_smart_whitelist() );
}


/**
 * Tell if a file is in the self white-list.
 *
 * @since 1.2.4
 * @author Grégory Viguier
 *
 * @param (string) $filepath Relative path to the file.
 *
 * @return (bool|null) True if the file is white-listed. False otherwize. Null on failure.
 */
function secupress_file_is_in_self_whitelist( $filepath ) {
	return secupress_file_is_in_whitelist( $filepath, secupress_get_malwares_self_whitelist() );
}


/**
 * Tell if a file is in a white-list.
 *
 * @since 1.2.4
 * @author Grégory Viguier
 *
 * @param (string) $filepath  Relative path to the file.
 * @param (array)  $whitelist A white-list.
 *
 * @return (bool|null) True if the file is white-listed. False otherwize. Null on failure.
 */
function secupress_file_is_in_whitelist( $filepath, $whitelist ) {
	static $main_theme_dir;
	global $wp_theme_directories;

	if ( ! isset( $main_theme_dir ) ) {
		$main_theme_dir = reset( $wp_theme_directories );
		$main_theme_dir = wp_normalize_path( trailingslashit( $main_theme_dir ) );
	}

	if ( ! $filepath ) {
		return null;
	}

	$abs_file_path = wp_normalize_path( ABSPATH . $filepath );
	$md5filepath   = md5_file( $abs_file_path );

	if ( ! $md5filepath ) {
		return null;
	}

	// Theme case.
	if ( isset( $whitelist[ str_replace( $main_theme_dir, '', $abs_file_path ) ][ $md5filepath ] ) ) {
		return true;
	}
	// Plugin case.
	if ( isset( $whitelist[ plugin_basename( $abs_file_path ) ][ $md5filepath ] ) ) {
		return true;
	}

	$filepath = wp_normalize_path( $filepath );

	// At the root of the site.
	return isset( $whitelist[ $filepath ][ $md5filepath ] );
}


/**
 * Check if the file contains any malware from our signatures list.
 * Also checks against the smart white-list.
 *
 * @since 1.0
 * @since 1.0.3 Usage of the smart white-list.
 * @since 1.2.4 Usage of the smart white-list only, not the self white-list.
 * @since 1.3   Skip non-php files larger than 30 MB.
 * @author Julio Potier
 * @author Grégory Viguier
 *
 * @param (string) $filepath A file path.
 *
 * @return (string) HTML markup with "Possible malware found" if the file is detected as a malware. An empty string otherwize.
 */
function secupress_check_malware( $filepath ) {
	static $malware_tests;
	static $uploads_dir;
	static $size_limit;
	static $init_done = false;

	$filesystem = secupress_get_filesystem();

	if ( ! isset( $malware_tests ) ) {
		// Avoid being tagged as a ma li ci ous script, we're nice :).
		$gz            = 'eta';
		$gz            = 'gz' . strrev( $gz . 'lfni' );
		$malware_tests = $filesystem->get_contents( SECUPRESS_PRO_INC_PATH . 'data/malware-keywords.data' );
		$malware_tests = $gz// Hey!
			( $malware_tests );
		$malware_tests = json_decode( $malware_tests, true );
	}

	if ( ! isset( $uploads_dir ) ) {
		$uploads_dir = wp_upload_dir( null, false );
		$uploads_dir = basename( $uploads_dir['basedir'] );
		$uploads_dir = wp_normalize_path( "/{$uploads_dir}/" );
	}

	if ( ! isset( $size_limit ) ) {
		$size_limit = 30 * MB_IN_BYTES;

		/**
		 * Filter the file size limit: non-php files larger than this value won't be tested for malware.
		 *
		 * @since 1.3
		 * @author Grégory Viguier
		 *
		 * @param (int) $size_limit File size limit in Bytes. Default is 30 MB.
		 */
		$size_limit = apply_filters( 'secupress.file-monitoring.size_limit', $size_limit );
	}

	if ( secupress_file_is_in_smart_whitelist( $filepath ) ) {
		// The file is white-listed.
		return '';
	}

	$file_ext = pathinfo( $filepath, PATHINFO_EXTENSION );

	if ( $size_limit && 'php' !== $file_ext ) {
		$file_size = @filesize( $filepath );

		if ( ! $file_size || $file_size > $size_limit ) {
			// The file is too heavy to be tested.
			return '';
		}
	}

	$contents = $filesystem->get_contents( ABSPATH . $filepath );

	if ( '' === $contents ) {
		// The file is empty.
		return '';
	}

	if ( ! $init_done ) {
		$init_done = true;
		@set_time_limit( 0 );
	}

	$result = array();

	foreach ( $malware_tests as $test ) {
		if ( trim( $test[0] ) !== $file_ext ) {
			continue;
		}

		if ( isset( $test[1]['str'], $test[1]['count'] ) ) {
			if ( substr_count( $contents, $test[1]['str'] ) >= $test[1]['count'] ) {
				$result[] = esc_html( $test[1]['str'] );
			}
			continue;
		}

		$actual_count = 0;

		foreach ( $test[1] as $text ) {
			if ( strpos( $contents, $text ) !== false ) {
				++$actual_count;
			}
		}

		if ( $actual_count && count( $test[1] ) === $actual_count ) {
			$result = array_merge( $result, array_map( 'esc_html', $test[1] ) );
		}
	}

	if ( 'php' === $file_ext ) {
		$filepath = wp_normalize_path( $filepath );

		if ( strpos( $filepath, '/blogs.dir' ) !== false || strpos( $filepath . '/', $uploads_dir ) !== false ) {
			/** Translators: %s is a file extension. */
			$result[] = sprintf( __( '%s in uploads', 'secupress-pro' ), '<code>.php</code>' );
		} else {
			$lines = explode( "\n", $contents );

			if ( strlen( $lines[0] ) > 1000 ) {
				/** Translators: %s is a number. */
				$result[] = sprintf( __( 'more than %s lines', 'secupress-pro' ), number_format_i18n( 1000 ) );
			}
		}
	}

	if ( $result ) {
		return secupress_malware_html();
	}

	return '';
}


/**
 * Get the "Possible malware found" html.
 *
 * @since 1.2.4
 * @author Grégory Viguier
 *
 * @return (string)
 */
function secupress_malware_html() {
	return '<span class="secupress-inline-alert"><span class="screen-reader-text">' . __( 'Possible malware found', 'secupress-pro' ) . '</span></span>';
}


/**
 * Get the result of the file scanner.
 *
 * @since 1.0
 * @since 1.2.4 Remove "old WP files" from the "files that are not part of the WordPress installation".
 * @author Julio Potier
 * @author Grégory Viguier
 *
 * @return (array|bool) An array of files, categorized by the reason they are listed. False if never scanned.
 */
function secupress_file_scanner_get_result() {
	global $wp_version;

	$full_filetree       = secupress_file_scanner_get_full_filetree( true );
	$wp_core_file_hashes = secupress_file_scanner_get_wp_core_file_hashes();

	if ( false === $full_filetree || false === $wp_core_file_hashes ) {
		return false;
	}

	if ( ! $wp_core_file_hashes || empty( $full_filetree[ $wp_version ] ) ) {
		return array();
	}

	$result = array(
		/**
		 * Files that are not part of the WordPress installation.
		 */
		'not-wp-files'      => secupress_file_scanner_get_files_not_from_wp(),
		/**
		 * Missing files from WP Core.
		 */
		'missing-wp-files'  => secupress_file_scanner_get_files_missing_from_wp(),
		/**
		 * Old WP files.
		 */
		'old-wp-files'      => secupress_file_scanner_get_old_wp_files(),
		/**
		 * Modified WP Core files.
		 */
		'modified-wp-files' => secupress_file_scanner_get_modified_wp_files(),
	);

	if ( $result['not-wp-files'] && $result['old-wp-files'] ) {
		// Remove "old WP files" from the "files that are not part of the WordPress installation".
		$result['not-wp-files'] = array_diff_key( $result['not-wp-files'], $result['old-wp-files'] );
	}

	return $result;
}


/**
 * Get the files that are not part of the WordPress installation.
 * Also checks against the self white-list.
 * Warning: it can also return "files that are not in WordPress core anymore". See `secupress_file_scanner_get_old_wp_files()`.
 *
 * @since 1.0
 * @since 1.2.4 Usage of the self white-list.
 * @author Julio Potier
 * @author Grégory Viguier
 *
 * @return (array|bool) An array of files. False if never scanned.
 */
function secupress_file_scanner_get_files_not_from_wp() {
	global $wp_version;
	static $wp_content_dir;

	$full_filetree       = secupress_file_scanner_get_full_filetree( true );
	$wp_core_file_hashes = secupress_file_scanner_get_wp_core_file_hashes();

	if ( false === $full_filetree || false === $wp_core_file_hashes ) {
		return false;
	}

	if ( ! $wp_core_file_hashes || empty( $full_filetree[ $wp_version ] ) ) {
		return array();
	}

	if ( ! isset( $wp_content_dir ) ) {
		$wp_content_dir = str_replace( realpath( ABSPATH ) . '/', '' , realpath( WP_CONTENT_DIR ) );
	}

	/**
	 * Filter the list of WordPress core file paths.
	 *
	 * @since 1.0
	 *
	 * @param (array) $wp_core_file_hashes The list of WordPress core file paths.
	 */
	$wp_core_file_hashes = apply_filters( 'secupress.plugin.file_scanner.wp_core_file_hashes', $wp_core_file_hashes );

	// Add these since it's not in the zip but depends from WordPress.
	$wp_core_file_hashes['wp-config.php'] = 'wp-config.php';
	$wp_core_file_hashes['.htaccess']     = '.htaccess';
	$wp_core_file_hashes['web.config']    = 'web.config';
	$wp_core_file_hashes[ $wp_content_dir . '/debug.log' ] = $wp_content_dir . '/debug.log';

	if ( is_multisite() ) {
		// Add this since it's not in the zip but depends from WordPress MS.
		$wp_core_file_hashes[ $wp_content_dir . '/sunrise.php' ] = $wp_content_dir . '/sunrise.php';
	}

	if ( defined( 'WP_CACHE' ) && WP_CACHE ) {
		// Add this since it's not in the zip but depends from WordPress Cache.
		$wp_core_file_hashes[ $wp_content_dir . '/advanced-cache.php' ] = $wp_content_dir . '/advanced-cache.php';
	}

	$full_filetree       = $full_filetree[ $wp_version ];
	$diff_from_root_core = array_diff_key( $full_filetree, $wp_core_file_hashes );

	$output = array();

	if ( $diff_from_root_core ) {
		$filesystem = secupress_get_filesystem();

		foreach ( $diff_from_root_core as $diff_file => $hash ) {
			if ( $filesystem->exists( ABSPATH . $diff_file ) && ! secupress_file_is_in_self_whitelist( $diff_file ) ) {
				$output[ $diff_file ] = $diff_file;
			}
		}
	}

	return $output;
}


/**
 * Get the files that are missing from WordPress core.
 *
 * @since 1.0
 * @author Julio Potier
 * @author Grégory Viguier
 *
 * @return (array|bool) An array of files. False if never scanned.
 */
function secupress_file_scanner_get_files_missing_from_wp() {
	global $wp_version, $_old_files;

	$full_filetree       = secupress_file_scanner_get_full_filetree( true );
	$wp_core_file_hashes = secupress_file_scanner_get_wp_core_file_hashes();

	if ( false === $full_filetree || false === $wp_core_file_hashes ) {
		return false;
	}

	if ( ! $wp_core_file_hashes || empty( $full_filetree[ $wp_version ] ) ) {
		return array();
	}

	$wp_core_file_hashes    = array_flip( array_filter( array_flip( $wp_core_file_hashes ), 'secupress_filter_no_content' ) );
	$missing_from_root_core = array_diff_key( $wp_core_file_hashes, $full_filetree[ $wp_version ] );
	unset( $missing_from_root_core['wp-config-sample.php'] );

	$output = array();

	if ( $missing_from_root_core ) {
		$output = array_keys( $missing_from_root_core );
		$output = array_combine( $output, $output );
	}

	return $output;
}


/**
 * Get the files that are not in WordPress core anymore.
 *
 * @since 1.0
 * @author Julio Potier
 * @author Grégory Viguier
 *
 * @return (array) An array of files.
 */
function secupress_file_scanner_get_old_wp_files() {
	global $_old_files;

	require_once( ABSPATH . 'wp-admin/includes/update-core.php' );

	$output = array();

	if ( $_old_files ) {
		$filesystem = secupress_get_filesystem();

		foreach ( $_old_files as $file ) {
			if ( $filesystem->exists( ABSPATH . $file ) ) {
				$output[ $file ] = $file;
			}
		}
	}

	return $output;
}


/**
 * Get the files from WordPress core that have been modified.
 *
 * @since 1.0
 * @author Julio Potier
 * @author Grégory Viguier
 *
 * @return (array|bool) An array of files. False if never scanned.
 */
function secupress_file_scanner_get_modified_wp_files() {
	global $wp_version, $_old_files;

	$full_filetree       = secupress_file_scanner_get_full_filetree( true );
	$wp_core_file_hashes = secupress_file_scanner_get_wp_core_file_hashes();

	if ( false === $full_filetree || false === $wp_core_file_hashes ) {
		return false;
	}

	if ( ! $wp_core_file_hashes || empty( $full_filetree[ $wp_version ] ) ) {
		return array();
	}

	/** This filter is documented in inc/modules/file-system/tools.php */
	$wp_core_file_hashes = apply_filters( 'secupress.plugin.file_scanner.wp_core_file_hashes', $wp_core_file_hashes );
	$full_filetree       = $full_filetree[ $wp_version ];
	$filesystem          = secupress_get_filesystem();

	$output = array();

	foreach ( $wp_core_file_hashes as $file => $hash ) {
		if ( isset( $full_filetree[ $file ] ) && ! hash_equals( $hash, $full_filetree[ $file ] ) && $filesystem->exists( ABSPATH . $file ) ) {
			$output[ $file ] = $file;
		}
	}

	return $output;
}


/**
 * Get the site's files tree from the database.
 *
 * @since 1.0
 * @author Grégory Viguier
 *
 * @param (bool) $raw Set to true to get the "raw" decompressed data.
 *
 * @return (array|bool) An array of files. False if never scanned.
 */
function secupress_file_scanner_get_full_filetree( $raw = false ) {
	global $wp_version;

	$full_filetree = secupress_decompress_data( get_site_option( SECUPRESS_FULL_FILETREE ) );

	if ( $raw ) {
		return $full_filetree;
	}

	$full_filetree = is_array( $full_filetree ) ? $full_filetree : array();

	if ( ! isset( $full_filetree[ $wp_version ] ) || ! is_array( $full_filetree[ $wp_version ] ) ) {
		$full_filetree[ $wp_version ] = array();
	}

	return $full_filetree;
}


/**
 * Store the site's files tree into the database.
 *
 * @since 1.0
 * @author Grégory Viguier
 *
 * @param (array) $full_filetree An array of files.
 */
function secupress_file_scanner_store_full_filetree( $full_filetree ) {
	update_site_option( SECUPRESS_FULL_FILETREE, secupress_compress_data( $full_filetree ), false );
}


/**
 * Delete the site's files tree from the database.
 *
 * @since 1.0
 * @author Grégory Viguier
 */
function secupress_file_scanner_delete_full_filetree() {
	delete_site_option( SECUPRESS_FULL_FILETREE );
}


/**
 * Get the WordPress core hashes.
 *
 * @since 1.0
 * @author Grégory Viguier
 *
 * @return (array|bool) An array of hashes. False if never scanned.
 */
function secupress_file_scanner_get_wp_core_file_hashes() {
	global $wp_version;

	$hashes = get_site_option( SECUPRESS_WP_CORE_FILES_HASHES );

	if ( false === $hashes ) {
		return false;
	}

	if ( empty( $hashes[ $wp_version ]['checksums'] ) || ! is_array( $hashes[ $wp_version ]['checksums'] ) ) {
		return array();
	}

	return $hashes[ $wp_version ]['checksums'];
}


/**
 * Used in `array_filter()`: return true if the given path is not in the `wp-content` folder.
 *
 * @since 1.0
 *
 * @param (string) $item A file path.
 *
 * @return (bool)
 */
function secupress_filter_no_content( $item ) {
	static $wp_content;

	if ( ! isset( $wp_content ) ) {
		$wp_content = basename( WP_CONTENT_DIR );
		$wp_content = "/{$wp_content}/";
	}

	$item = str_replace( '\\', '/', $item );

	return strpos( "/{$item}/", $wp_content ) === false;
}
