AlkantarClanX12

Your IP : 18.223.158.132


Current Path : /home/thanudqk/www/wp-content/plugins/w3-total-cache/
Upload File :
Current File : //home/thanudqk/www/wp-content/plugins/w3-total-cache/PgCache_Environment.php

<?php
namespace W3TC;

/**
 * class PgCache_Environment
 */
class PgCache_Environment {
	/**
	 * Fixes environment in each wp-admin request
	 *
	 * @param Config $config           Config.
	 * @param bool   $force_all_checks Force checks flag.
	 *
	 * @throws Util_Environment_Exceptions Excpetions.
	 */
	public function fix_on_wpadmin_request( $config, $force_all_checks ) {
		$exs             = new Util_Environment_Exceptions();
		$pgcache_enabled = $config->get_boolean( 'pgcache.enabled' );
		$engine          = $config->get_string( 'pgcache.engine' );

		if ( ( ! defined( 'WP_CACHE' ) || ! WP_CACHE ) ) {
			try {
				$this->wp_config_add_directive();
			} catch ( Util_WpFile_FilesystemOperationException $ex ) {
				$exs->push( $ex );
			}
		}

		$this->fix_folders( $config, $exs );

		if ( $config->get_boolean( 'config.check' ) || $force_all_checks ) {
			if ( $this->is_rules_required( $config ) ) {
				$this->rules_core_add( $config, $exs );
				$this->rules_cache_add( $config, $exs );
			} else {
				$this->rules_core_remove( $exs );
				$this->rules_cache_remove( $exs );
			}
		}

		// if no errors so far - check if rewrite actually works.
		if ( count( $exs->exceptions() ) <= 0 ) {
			try {
				if ( $pgcache_enabled && 'file_generic' === $engine ) {
					$this->verify_file_generic_compatibility();

					if ( $config->get_boolean( 'pgcache.debug' ) ) {
						$this->verify_file_generic_rewrite_working();
					}
				}
			} catch ( \Exception $ex ) {
				$exs->push( $ex );
			}
		}

		if ( count( $exs->exceptions() ) > 0 ) {
			throw $exs;
		}
	}

	/**
	 * Fixes environment once event occurs
	 *
	 * @param Config      $config     Config.
	 * @param string      $event      Event.
	 * @param null|Config $old_config Old Config.
	 *
	 * @throws Util_Environment_Exceptions Exception.
	 */
	public function fix_on_event( $config, $event, $old_config = null ) {
		$pgcache_enabled = $config->get_boolean( 'pgcache.enabled' );
		$engine          = $config->get_string( 'pgcache.engine' );

		// Schedules events.
		if ( $pgcache_enabled && ( 'file' === $engine || 'file_generic' === $engine ) ) {
			$new_interval = $config->get_integer( 'pgcache.file.gc' );
			$old_interval = $old_config ? $old_config->get_integer( 'pgcache.file.gc' ) : -1;

			if ( null !== $old_config && $new_interval !== $old_interval ) {
				$this->unschedule_gc();
			}

			if ( ! wp_next_scheduled( 'w3_pgcache_cleanup' ) ) {
				wp_schedule_event( time(), 'w3_pgcache_cleanup', 'w3_pgcache_cleanup' );
			}
		} else {
			$this->unschedule_gc();
		}

		// Schedule prime event.
		if ( $pgcache_enabled && $config->get_boolean( 'pgcache.prime.enabled' ) ) {
			$new_interval = $config->get_integer( 'pgcache.prime.interval' );
			$old_interval = $old_config ? $old_config->get_integer( 'pgcache.prime.interval' ) : -1;

			if ( null !== $old_config && $new_interval !== $old_interval ) {
				$this->unschedule_prime();
			}

			if ( ! wp_next_scheduled( 'w3_pgcache_prime' ) ) {
				wp_schedule_event( time(), 'w3_pgcache_prime', 'w3_pgcache_prime' );
			}
		} else {
			$this->unschedule_prime();
		}
	}

	/**
	 * Fixes environment after plugin deactivation
	 *
	 * @throws Util_Environment_Exceptions Exception.
	 *
	 * @return void
	 */
	public function fix_after_deactivation() {
		$exs = new Util_Environment_Exceptions();

		try {
			$this->wp_config_remove_directive( $exs );
		} catch ( Util_WpFile_FilesystemOperationException $ex ) {
			$exs->push( $ex );
		}

		$this->rules_core_remove( $exs );
		$this->rules_cache_remove( $exs );

		$this->unschedule_gc();
		$this->unschedule_prime();
		$this->unschedule_purge_wpcron();

		if ( count( $exs->exceptions() ) > 0 ) {
			throw $exs;
		}
	}

	/**
	 * Are rewrite rules required?.
	 *
	 * @since 0.9.7.3
	 *
	 * @param  Config $c Configuration.
	 * @return bool
	 */
	private function is_rules_required( $c ) {
		$e = $c->get_string( 'pgcache.engine' );

		return $c->get_boolean( 'pgcache.enabled' ) &&
			( 'file_generic' === $e || 'nginx_memcached' === $e );
	}

	/**
	 * Returns required rules for module
	 *
	 * @param Config  $config
	 * @return array
	 */
	public function get_required_rules( $config ) {
		if ( !$this->is_rules_required( $config ) ) {
			return null;
		}

		$rewrite_rules = array();
		$pgcache_rules_core_path = Util_Rule::get_pgcache_rules_core_path();
		$rewrite_rules[] = array(
			'filename' => $pgcache_rules_core_path,
			'content' => $this->rules_core_generate( $config ),
			'priority' => 1000
		);

		$pgcache_rules_cache_path = Util_Rule::get_pgcache_rules_cache_path();
		$rewrite_rules[] = array(
			'filename' => $pgcache_rules_cache_path,
			'content' => $this->rules_cache_generate( $config )
		);

		return $rewrite_rules;
	}



	/**
	 * Fixes folders
	 *
	 * @param Config  $config
	 * @param Util_Environment_Exceptions $exs
	 */
	private function fix_folders( $config, $exs ) {
		if ( !$config->get_boolean( 'pgcache.enabled' ) )
			return;

		// folder that we delete if exists and not writeable
		if ( $config->get_string( 'pgcache.engine' ) == 'file_generic' )
			$dir = W3TC_CACHE_PAGE_ENHANCED_DIR;
		else if ( $config->get_string( 'pgcache.engine' ) != 'file' )
				$dir = W3TC_CACHE_DIR . '/page';
			else
				return;

			try{
			if ( file_exists( $dir ) && !is_writeable( $dir ) )
				Util_WpFile::delete_folder( $dir, '', isset( $_SERVER['REQUEST_URI'] ) ? esc_url_raw( wp_unslash( $_SERVER['REQUEST_URI'] ) ) : '' );
		} catch ( Util_WpFile_FilesystemRmdirException $ex ) {
			$exs->push( $ex );
		}
	}

	/**
	 * Checks if mode can be used
	 */
	private function verify_file_generic_compatibility() {
		$permalink_structure = get_option( 'permalink_structure' );

		if ( $permalink_structure == '' ) {
			throw new Util_Environment_Exception( 'Disk Enhanced mode ' .
				'can\'t work with "Default" permalinks structure' );
		}
	}

	/**
	 * Fixes environment for enabled pgcache
	 *
	 * @throws Util_Environment_Exceptions
	 */
	private function verify_file_generic_rewrite_working() {
		$url = get_home_url() . '/w3tc_rewrite_test' . rand();
		if ( !$this->test_rewrite( $url ) ) {
			$key = sprintf( 'w3tc_rewrite_test_result_%s', substr( md5( $url ), 0, 16 ) );
			$result = get_transient( $key );

			$home_url = get_home_url();

			$tech_message =
				( Util_Environment::is_nginx() ? 'nginx configuration file' : '.htaccess file' ) .
				' contains rules to rewrite url ' .
				$home_url . '/w3tc_rewrite_test into ' .
				$home_url . '/?w3tc_rewrite_test which, if handled by ' .
				'plugin, return "OK" message.<br/>';
			$tech_message .= 'The plugin made a request to ' .
				$home_url . '/w3tc_rewrite_test but received: <br />' .
				$result . '<br />';
			$tech_message .= 'instead of "OK" response. <br />';

			$error = '<strong>W3 Total Cache error:</strong> ' .
				'It appears Page Cache ' .
				'<acronym title="Uniform Resource Locator">URL</acronym> ' .
				'rewriting is not working. ';
			if ( Util_Environment::is_preview_mode() ) {
				$error .= ' This could be due to using Preview mode. <a href="' . $url . '">Click here</a> to manually verify its working. It should say OK. <br />';
			}

			if ( Util_Environment::is_nginx() ) {
				$error .= 'Please verify that all configuration files are ' .
					'included in the configuration file ' .
					'(and that you have reloaded / restarted nginx).';
			} else {
				$error .= 'Please verify that the server configuration ' .
					'allows .htaccess';
			}

			$error .= '<br />Unfortunately disk enhanced page caching will ' .
				'not function without custom rewrite rules. ' .
				'Please ask your server administrator for assistance. ' .
				'Also refer to <a href="' .
				admin_url( 'admin.php?page=w3tc_install' ) .
				'">the install page</a>  for the rules for your server.';

			throw new Util_Environment_Exception( $error, $tech_message );
		}
	}

	/**
	 * Perform rewrite test
	 *
	 * @param string  $url
	 * @return boolean
	 */
	private function test_rewrite( $url ) {
		$key = sprintf( 'w3tc_rewrite_test_%s', substr( md5( $url ), 0, 16 ) );
		$result = get_transient( $key );

		if ( $result === false ) {

			$response = Util_Http::get( $url );

			$result = ( !is_wp_error( $response ) && $response['response']['code'] == 200 && trim( $response['body'] ) == 'OK' );

			if ( $result ) {
				set_transient( $key, $result, 30 );
			} else {
				$key_result = sprintf( 'w3tc_rewrite_test_result_%s', substr( md5( $url ), 0, 16 ) );
				set_transient( $key_result, is_wp_error( $response )? $response->get_error_message(): implode( ' ', $response['response'] ), 30 );
			}
		}

		return $result;
	}



	/**
	 * Remove Garbage collection cron job.
	 *
	 * @return void
	 */
	private function unschedule_gc() {
		if ( wp_next_scheduled( 'w3_pgcache_cleanup' ) ) {
			wp_clear_scheduled_hook( 'w3_pgcache_cleanup' );
		}
	}

	/**
	 * Remove Prime Cache cron job.
	 *
	 * @return void
	 */
	private function unschedule_prime() {
		if ( wp_next_scheduled( 'w3_pgcache_prime' ) ) {
			wp_clear_scheduled_hook( 'w3_pgcache_prime' );
		}
	}

	/**
	 * Remove cron job for pagecache purge.
	 *
	 * @since 2.8.0
	 *
	 * @return void
	 */
	private function unschedule_purge_wpcron() {
		if ( wp_next_scheduled( 'w3tc_pgcache_purge_wpcron' ) ) {
			wp_clear_scheduled_hook( 'w3tc_pgcache_purge_wpcron' );
		}
	}

	/*
	 * wp-config modification
	 */

	/**
	 * Enables WP_CACHE
	 *
	 * @throws Util_WpFile_FilesystemOperationException with S/FTP form if it can't get the required filesystem credentials
	 */
	private function wp_config_add_directive() {
		$config_path = Util_Environment::wp_config_path();

		$config_data = @file_get_contents( $config_path );
		if ( $config_data === false )
			return;

		$new_config_data = $this->wp_config_remove_from_content( $config_data );
		$new_config_data = preg_replace(
			'~<\?(php)?~',
			"\\0\r\n" . $this->wp_config_addon(),
			$new_config_data,
			1 );

		if ( $new_config_data != $config_data ) {
			try {
				Util_WpFile::write_to_file( $config_path, $new_config_data );
			} catch ( Util_WpFile_FilesystemOperationException $ex ) {
				throw new Util_WpFile_FilesystemModifyException(
					$ex->getMessage(), $ex->credentials_form(),
					'Edit file <strong>' . $config_path .
					'</strong> and add next lines:', $config_path,
					$this->wp_config_addon() );
			}
		}
		// that file was in opcache for sure and it may take time to
		// start execution of new modified now version
		$o = Dispatcher::component( 'SystemOpCache_Core' );
		$o->flush();
	}

	/**
	 * Disables WP_CACHE
	 *
	 * @throws Util_WpFile_FilesystemOperationException with S/FTP form if it can't get the required filesystem credentials
	 */
	private function wp_config_remove_directive() {
		$config_path = Util_Environment::wp_config_path();

		$config_data = @file_get_contents( $config_path );
		if ( $config_data === false )
			return;

		$new_config_data = $this->wp_config_remove_from_content( $config_data );
		if ( $new_config_data != $config_data ) {
			try {
				Util_WpFile::write_to_file( $config_path, $new_config_data );
			} catch ( Util_WpFile_FilesystemOperationException $ex ) {
				throw new Util_WpFile_FilesystemModifyException(
					$ex->getMessage(), $ex->credentials_form(),
					'Edit file <strong>' . $config_path .
					'</strong> and remove next lines:',
					$config_path,  $this->wp_config_addon() );
			}
		}
	}

	/**
	 * Returns string Addon required for plugin in wp-config
	 */
	private function wp_config_addon() {
		return "/** Enable W3 Total Cache */\r\n" .
			"define('WP_CACHE', true); // Added by W3 Total Cache\r\n";
	}

	/**
	 * Disables WP_CACHE
	 *
	 * @param string  $config_data wp-config.php content
	 * @return string
	 * @throws Util_WpFile_FilesystemOperationException with S/FTP form if it can't get the required filesystem credentials
	 */
	private function wp_config_remove_from_content( $config_data ) {
		$config_data = preg_replace(
			"~\\/\\*\\* Enable W3 Total Cache \\*\\*?\\/.*?\\/\\/ Added by W3 Total Cache(\r\n)*~s",
			'', $config_data );
		$config_data = preg_replace(
			"~(\\/\\/\\s*)?define\\s*\\(\\s*['\"]?WP_CACHE['\"]?\\s*,.*?\\)\\s*;+\\r?\\n?~is",
			'', $config_data );

		return $config_data;
	}



	/**
	 * rules core modification
	 */

	/**
	 * Writes directives to WP .htaccess / nginx.conf
	 * returns true if modifications has been made
	 */
	private function rules_core_add( $config, $exs ) {
		Util_Rule::add_rules( $exs, Util_Rule::get_pgcache_rules_core_path(),
			$this->rules_core_generate( $config ),
			W3TC_MARKER_BEGIN_PGCACHE_CORE,
			W3TC_MARKER_END_PGCACHE_CORE,
			array(
				W3TC_MARKER_BEGIN_WORDPRESS => 0,
				W3TC_MARKER_END_MINIFY_CORE =>
					strlen( W3TC_MARKER_END_MINIFY_CORE ) + 1,
				W3TC_MARKER_END_BROWSERCACHE_CACHE =>
					strlen( W3TC_MARKER_END_BROWSERCACHE_CACHE ) + 1,
				W3TC_MARKER_END_PGCACHE_CACHE =>
					strlen( W3TC_MARKER_END_PGCACHE_CACHE ) + 1,
				W3TC_MARKER_END_MINIFY_CACHE =>
					strlen( W3TC_MARKER_END_MINIFY_CACHE ) + 1
			),
			true
		);
	}

	/**
	 * Removes Page Cache core directives
	 *
	 * @param Util_Environment_Exceptions $exs
	 * @throws Util_WpFile_FilesystemOperationException with S/FTP form if it can't get the required filesystem credentials
	 */
	private function rules_core_remove( $exs ) {
		Util_Rule::remove_rules( $exs, Util_Rule::get_pgcache_rules_core_path(),
			W3TC_MARKER_BEGIN_PGCACHE_CORE,
			W3TC_MARKER_END_PGCACHE_CORE
		);
	}

	/**
	 * Generates rules for WP dir
	 *
	 * @param Config  $config
	 * @return string
	 */
	private function rules_core_generate( $config ) {
		switch ( true ) {
		case Util_Environment::is_apache():
		case Util_Environment::is_litespeed():
			return $this->rules_core_generate_apache( $config );

		case Util_Environment::is_nginx():
			return $this->rules_core_generate_nginx( $config );
		}

		return '';
	}

	/**
	 * Generates rules for WP dir
	 *
	 * @param Config  $config
	 * @return string
	 */
	private function rules_core_generate_apache( $config ) {
		$rewrite_base = Util_Environment::network_home_url_uri();
		$cache_dir = Util_Environment::normalize_path( W3TC_CACHE_PAGE_ENHANCED_DIR );
		$permalink_structure = get_option( 'permalink_structure' );

		$current_user = wp_get_current_user();

		/**
		 * Auto reject cookies
		 */
		$reject_cookies = array(
			'comment_author',
			'wp-postpass'
		);

		$reject_cookies[] = 'w3tc_logged_out';

		/**
		 * Reject cache for logged in users
		 * OR
		 * Reject cache for roles if any
		 */
		if ( $config->get_boolean( 'pgcache.reject.logged' ) ) {
			$reject_cookies = array_merge( $reject_cookies, array(
					'wordpress_logged_in'
				) );
		} elseif ( $config->get_boolean( 'pgcache.reject.logged_roles' ) ) {
			$new_cookies = array();
			foreach ( $config->get_array( 'pgcache.reject.roles' ) as $role ) {
				$new_cookies[] = 'w3tc_logged_' . md5( NONCE_KEY . $role );
			}
			$reject_cookies = array_merge( $reject_cookies, $new_cookies );
		}

		/**
		 * Custom config
		 */
		$reject_cookies = array_merge( $reject_cookies, $config->get_array( 'pgcache.reject.cookie' ) );
		Util_Rule::array_trim( $reject_cookies );

		$reject_user_agents = $config->get_array( 'pgcache.reject.ua' );
		if ( $config->get_boolean( 'pgcache.compatibility' ) ) {
			$reject_user_agents = array_merge( array( W3TC_POWERED_BY ), $reject_user_agents );
		}

		Util_Rule::array_trim( $reject_user_agents );

		/**
		 * Generate directives
		 */
		$env_W3TC_UA = '';
		$env_W3TC_REF = '';
		$env_W3TC_COOKIE = '';
		$env_W3TC_SSL = '';
		$env_W3TC_ENC = '';

		$rules = '';
		$rules .= W3TC_MARKER_BEGIN_PGCACHE_CORE . "\n";
		$rules .= "<IfModule mod_rewrite.c>\n";
		$rules .= "    RewriteEngine On\n";
		$rules .= "    RewriteBase " . $rewrite_base . "\n";


		if ( $config->get_boolean( 'pgcache.debug' ) ) {
			$rules .= "    RewriteRule ^(.*\\/)?w3tc_rewrite_test([0-9]+)/?$ $1?w3tc_rewrite_test=1 [L]\n";
		}


		/**
		 * Set accept query strings
		 */
		$w3tc_query_strings = apply_filters(
			'w3tc_pagecache_rules_apache_accept_qs',
			$config->get_array( 'pgcache.accept.qs' ) );
		Util_Rule::array_trim( $w3tc_query_strings );

		if ( !empty( $w3tc_query_strings ) ) {
			$w3tc_query_strings = str_replace( ' ', '+', $w3tc_query_strings );
			$w3tc_query_strings = array_map( array( '\W3TC\Util_Environment', 'preg_quote' ), $w3tc_query_strings );

			$rules .= "    RewriteRule ^ - [E=W3TC_QUERY_STRING:%{QUERY_STRING}]\n";

			foreach ( $w3tc_query_strings as $query ) {
				$query_rules = array();
				if  ( strpos( $query, '=' ) === false ) {
					$query_rules[] = 'RewriteCond %{ENV:W3TC_QUERY_STRING} ^(.*?&|)' .
						$query . '(=[^&]*)?(&.*|)$ [NC]';
					$query_rules[] = 'RewriteRule ^ - [E=W3TC_QUERY_STRING:%1%3]';
				} else {
					$query_rules[] = 'RewriteCond %{ENV:W3TC_QUERY_STRING} ^(.*?&|)' .
						$query . '(&.*|)$ [NC]';
					$query_rules[] = 'RewriteRule ^ - [E=W3TC_QUERY_STRING:%1%2]';
				}

				$query_rules = apply_filters(
					'w3tc_pagecache_rules_apache_accept_qs_rules',
					$query_rules, $query );
				$rules .= '    ' . implode( "\n    ", $query_rules ) . "\n";
			}

			$rules .= "    RewriteCond %{ENV:W3TC_QUERY_STRING} ^&+$\n";
			$rules .= "    RewriteRule ^ - [E=W3TC_QUERY_STRING]\n";
		}

		/**
		 * Check for mobile redirect
		 */
		if ( $config->get_boolean( 'mobile.enabled' ) ) {
			$mobile_groups = $config->get_array( 'mobile.rgroups' );

			foreach ( $mobile_groups as $mobile_group => $mobile_config ) {
				$mobile_enabled = ( isset( $mobile_config['enabled'] ) ? (boolean) $mobile_config['enabled'] : false );
				$mobile_agents = ( isset( $mobile_config['agents'] ) ? (array) $mobile_config['agents'] : '' );
				$mobile_redirect = ( isset( $mobile_config['redirect'] ) ? $mobile_config['redirect'] : '' );

				if ( $mobile_enabled && count( $mobile_agents ) && $mobile_redirect ) {
					$rules .= "    RewriteCond %{HTTP_USER_AGENT} (" . implode( '|', $mobile_agents ) . ") [NC]\n";
					$rules .= "    RewriteRule .* " . $mobile_redirect . " [R,L]\n";
				}
			}
		}

		/**
		 * Check for referrer redirect
		 */
		if ( $config->get_boolean( 'referrer.enabled' ) ) {
			$referrer_groups = $config->get_array( 'referrer.rgroups' );

			foreach ( $referrer_groups as $referrer_group => $referrer_config ) {
				$referrer_enabled = ( isset( $referrer_config['enabled'] ) ? (boolean) $referrer_config['enabled'] : false );
				$referrer_referrers = ( isset( $referrer_config['referrers'] ) ? (array) $referrer_config['referrers'] : '' );
				$referrer_redirect = ( isset( $referrer_config['redirect'] ) ? $referrer_config['redirect'] : '' );

				if ( $referrer_enabled && count( $referrer_referrers ) && $referrer_redirect ) {
					$rules .= "    RewriteCond %{HTTP_COOKIE} w3tc_referrer=.*(" . implode( '|', $referrer_referrers ) . ") [NC]\n";
					$rules .= "    RewriteRule .* " . $referrer_redirect . " [R,L]\n";
				}
			}
		}

		/**
		 * Set mobile groups
		 */
		if ( $config->get_boolean( 'mobile.enabled' ) ) {
			$mobile_groups = array_reverse( $config->get_array( 'mobile.rgroups' ) );

			foreach ( $mobile_groups as $mobile_group => $mobile_config ) {
				$mobile_enabled = ( isset( $mobile_config['enabled'] ) ? (boolean) $mobile_config['enabled'] : false );
				$mobile_agents = ( isset( $mobile_config['agents'] ) ? (array) $mobile_config['agents'] : '' );
				$mobile_redirect = ( isset( $mobile_config['redirect'] ) ? $mobile_config['redirect'] : '' );

				if ( $mobile_enabled && count( $mobile_agents ) && !$mobile_redirect ) {
					$rules .= "    RewriteCond %{HTTP_USER_AGENT} (" . implode( '|', $mobile_agents ) . ") [NC]\n";
					$rules .= "    RewriteRule .* - [E=W3TC_UA:_" . $mobile_group . "]\n";
					$env_W3TC_UA = '%{ENV:W3TC_UA}';
				}
			}
		}

		/**
		 * Set referrer groups
		 */
		if ( $config->get_boolean( 'referrer.enabled' ) ) {
			$referrer_groups = array_reverse( $config->get_array( 'referrer.rgroups' ) );

			foreach ( $referrer_groups as $referrer_group => $referrer_config ) {
				$referrer_enabled = ( isset( $referrer_config['enabled'] ) ? (boolean) $referrer_config['enabled'] : false );
				$referrer_referrers = ( isset( $referrer_config['referrers'] ) ? (array) $referrer_config['referrers'] : '' );
				$referrer_redirect = ( isset( $referrer_config['redirect'] ) ? $referrer_config['redirect'] : '' );

				if ( $referrer_enabled && count( $referrer_referrers ) && !$referrer_redirect ) {
					$rules .= "    RewriteCond %{HTTP_COOKIE} w3tc_referrer=.*(" . implode( '|', $referrer_referrers ) . ") [NC]\n";
					$rules .= "    RewriteRule .* - [E=W3TC_REF:_" . $referrer_group . "]\n";
					$env_W3TC_REF = '%{ENV:W3TC_REF}';
				}
			}
		}

		/**
		 * Set cookie group
		 */
		if ( $config->get_boolean( 'pgcache.cookiegroups.enabled' ) ) {
			$cookie_groups = $config->get_array( 'pgcache.cookiegroups.groups' );

			foreach ( $cookie_groups as $group_name => $g ) {
				if ( isset( $g['enabled'] ) && $g['enabled'] ) {
					$cookies = array();
					foreach ($g['cookies'] as $cookie ) {
						$cookie = trim( $cookie );
						if ( !empty( $cookie ) ) {
							$cookie = str_replace( '+', ' ', $cookie );
							$cookie = Util_Environment::preg_quote( $cookie );
							if ( strpos( $cookie, '=') === false )
								$cookie .= '=.*';
							$cookies[] = $cookie;
						}
					}

					if ( count( $cookies ) > 0 ) {
						$cookies_regexp = '^(.*;\s*)?(' . implode( '|', $cookies ) . ')(\s*;.*)?$';
						$rules .= "    RewriteCond %{HTTP_COOKIE} $cookies_regexp [NC]\n";
						$rules .= "    RewriteRule .* - [E=W3TC_COOKIE:_" . $group_name . "]\n";
						$env_W3TC_COOKIE = '%{ENV:W3TC_COOKIE}';
					}
				}
			}
		}

		/**
		 * Set HTTPS
		 */
		if ( $config->get_boolean( 'pgcache.cache.ssl' ) ) {
			$rules .= "    RewriteCond %{HTTPS} =on\n";
			$rules .= "    RewriteRule .* - [E=W3TC_SSL:_ssl]\n";
			$rules .= "    RewriteCond %{SERVER_PORT} =443\n";
			$rules .= "    RewriteRule .* - [E=W3TC_SSL:_ssl]\n";
			$rules .= "    RewriteCond %{HTTP:X-Forwarded-Proto} =https [NC]\n";
			$rules .= "    RewriteRule .* - [E=W3TC_SSL:_ssl]\n";
			$env_W3TC_SSL = '%{ENV:W3TC_SSL}';
		}

		$cache_path = str_replace( Util_Environment::document_root(), '', $cache_dir );

		/**
		 * Set Accept-Encoding gzip
		 */
		if ( $config->get_boolean( 'browsercache.enabled' ) && $config->get_boolean( 'browsercache.html.compression' ) ) {
			$rules .= "    RewriteCond %{HTTP:Accept-Encoding} gzip\n";
			$rules .= "    RewriteRule .* - [E=W3TC_ENC:_gzip]\n";
			$env_W3TC_ENC = '%{ENV:W3TC_ENC}';
		}

		/**
		 * Set Accept-Encoding brotli
		 */
		if ( $config->get_boolean( 'browsercache.enabled' ) && $config->get_boolean( 'browsercache.html.brotli' ) ) {
			$rules .= "    RewriteCond %{HTTP:Accept-Encoding} br\n";
			$rules .= "    RewriteRule .* - [E=W3TC_ENC:_br]\n";
			$env_W3TC_ENC = '%{ENV:W3TC_ENC}';
		}

		$rules .= "    RewriteCond %{HTTP_COOKIE} w3tc_preview [NC]\n";
		$rules .= "    RewriteRule .* - [E=W3TC_PREVIEW:_preview]\n";
		$env_W3TC_PREVIEW = '%{ENV:W3TC_PREVIEW}';

		$rules .= "    RewriteCond %{REQUEST_URI} \\/$\n";
		$rules .= "    RewriteRule .* - [E=W3TC_SLASH:_slash]\n";
		$env_W3TC_SLASH = '%{ENV:W3TC_SLASH}';

		$use_cache_rules = '';
		/**
		 * Don't accept POSTs
		 */
		$use_cache_rules .= "    RewriteCond %{REQUEST_METHOD} !=POST\n";

		/**
		 * Query string should be empty
		 */
		$use_cache_rules .= empty( $w3tc_query_strings ) ?
			"    RewriteCond %{QUERY_STRING} =\"\"\n" :
			"    RewriteCond %{ENV:W3TC_QUERY_STRING} =\"\"\n";

		/**
		 * Check for rejected cookies
		 */
		$use_cache_rules .= "    RewriteCond %{HTTP_COOKIE} !(" . implode( '|',
			array_map( array( '\W3TC\Util_Environment', 'preg_quote' ), $reject_cookies ) ) . ") [NC]\n";

		/**
		 * Check for rejected user agents
		 */
		if ( count( $reject_user_agents ) ) {
			$use_cache_rules .= "    RewriteCond %{HTTP_USER_AGENT} !(" .
				implode( '|',
				array_map( array( '\W3TC\Util_Environment', 'preg_quote' ), $reject_user_agents ) ) . ") [NC]\n";
		}

		$use_cache_rules = apply_filters(
			'w3tc_pagecache_rules_apache_rewrite_cond',
			$use_cache_rules
		);

		/**
		 * Make final rewrites for specific files
		 */
		$uri_prefix =  $cache_path . '/%{HTTP_HOST}/%{REQUEST_URI}/' .
			'_index' . $env_W3TC_SLASH . $env_W3TC_UA . $env_W3TC_REF . $env_W3TC_COOKIE .
			$env_W3TC_SSL . $env_W3TC_PREVIEW;
		$uri_prefix = apply_filters( 'w3tc_pagecache_rules_apache_uri_prefix',
			$uri_prefix );

		$switch = " -" . ( $config->get_boolean( 'pgcache.file.nfs' ) ? 'F' : 'f' );

		$document_root = Util_Rule::apache_docroot_variable();

		// write rule to rewrite to .html/.xml file
		$exts = array( '.html' );
		if ($config->get_boolean('pgcache.cache.nginx_handle_xml'))
			$exts[] = '.xml';

		/**
		 * Filter: Allow adding additional rules at the end of the PGCACHE_CORE block, before the last rule.
		 *
		 * @since 2.7.1
		 *
		 * @param string $rules           Additional rules.
		 * @param string $use_cache_rules Rewrite conditions for non-POST, empty query string, rejected cookies, and rejected user agents.
		 * @param string $document_root   Document root.
		 * @param string $uri_prefix      URI prefix, after the "w3tc_pagecache_rules_apache_uri_prefix" WP filter.
		 * @param array  $exts            File extensions used; iterate to use them all.
		 * @param string $env_W3TC_ENC    Encoding string: "", "_br", or "_gzip".
		 */
		$rules = \apply_filters(
			'w3tc_pgcache_rules_apache_last',
			$rules,
			$use_cache_rules,
			$document_root,
			$uri_prefix,
			$env_W3TC_ENC
		);

		foreach ( $exts as $ext ) {
			$rules .= $use_cache_rules;

			$rules .= "    RewriteCond \"" . $document_root . $uri_prefix . $ext .
				$env_W3TC_ENC . "\"" . $switch . "\n";
			$rules .= "    RewriteRule .* \"" . $uri_prefix . $ext .
				$env_W3TC_ENC . "\" [L]\n";
		}

		$rules .= "</IfModule>\n";

		$rules .= W3TC_MARKER_END_PGCACHE_CORE . "\n";

		return $rules;
	}

	/**
	 * Generates rules for WP dir
	 *
	 * @param Config  $config
	 * @return string
	 */
	private function rules_core_generate_nginx( $config ) {
		$is_network = Util_Environment::is_wpmu();

		$cache_dir = Util_Environment::normalize_path( W3TC_CACHE_PAGE_ENHANCED_DIR );
		$permalink_structure = get_option( 'permalink_structure' );
		$pgcache_engine = $config->get_string( 'pgcache.engine' );

		/**
		 * Auto reject cookies
		 */
		$reject_cookies = array(
			'comment_author',
			'wp-postpass'
		);

		if ( $pgcache_engine == 'file_generic' ) {
			$reject_cookies[] = 'w3tc_logged_out';
		}

		/**
		 * Reject cache for logged in users
		 * OR
		 * Reject cache for roles if any
		 */
		if ( $config->get_boolean( 'pgcache.reject.logged' ) ) {
			$reject_cookies = array_merge( $reject_cookies, array(
					'wordpress_logged_in'
				) );
		} elseif ( $config->get_boolean( 'pgcache.reject.logged_roles' ) ) {
			$new_cookies = array();
			foreach ( $config->get_array( 'pgcache.reject.roles' ) as $role ) {
				$new_cookies[] = 'w3tc_logged_' . md5( NONCE_KEY . $role );
			}
			$reject_cookies = array_merge( $reject_cookies, $new_cookies );
		}

		/**
		 * Custom config
		 */
		$reject_cookies = array_merge( $reject_cookies,
			$config->get_array( 'pgcache.reject.cookie' ) );
		Util_Rule::array_trim( $reject_cookies );

		$reject_user_agents = $config->get_array( 'pgcache.reject.ua' );
		if ( $config->get_boolean( 'pgcache.compatibility' ) ) {
			$reject_user_agents = array_merge( array( W3TC_POWERED_BY ),
				$reject_user_agents );
		}
		Util_Rule::array_trim( $reject_user_agents );

		/**
		 * Generate rules
		 */
		$env_w3tc_ua = '';
		$env_w3tc_ref = '';
		$env_w3tc_cookie = '';
		$env_w3tc_ssl = '';
		$env_w3tc_ext = '';
		$env_w3tc_enc = '';
		$env_request_uri = '$request_uri';

		$rules = '';
		$rules .= W3TC_MARKER_BEGIN_PGCACHE_CORE . "\n";
		if ( $config->get_boolean( 'pgcache.debug' ) ) {
			$rules .= "rewrite ^(.*\\/)?w3tc_rewrite_test([0-9]+)/?$ $1?w3tc_rewrite_test=1 last;\n";
		}

		/**
		 * Set accept query strings
		 */
		$w3tc_query_strings = apply_filters(
			'w3tc_pagecache_rules_nginx_accept_qs',
			$config->get_array( 'pgcache.accept.qs' ) );

		Util_Rule::array_trim( $w3tc_query_strings );

		if ( !empty( $w3tc_query_strings ) ) {
		   $w3tc_query_strings = str_replace( ' ', '+', $w3tc_query_strings );
		   $w3tc_query_strings = array_map( array( '\W3TC\Util_Environment', 'preg_quote' ), $w3tc_query_strings );

			$rules .= "set \$w3tc_query_string \$query_string;\n";

			foreach ( $w3tc_query_strings as $query ) {
				$query_rules = array();
				if ( strpos( $query, '=' ) === false ) {
					$query_rules[] = 'if ($w3tc_query_string ~* "^(.*?&|)' . $query . '(=[^&]*)?(&.*|)$") {';
					$query_rules[] = '    set $w3tc_query_string $1$3;';
					$query_rules[] = '}';
				} else {
					$query_rules[] = 'if ($w3tc_query_string ~* "^(.*?&|)' . $query . '(&.*|)$") {';
					$query_rules[] = '    set $w3tc_query_string $1$2;';
					$query_rules[] = '}';
				}

				$query_rules = apply_filters(
					'w3tc_pagecache_rules_nginx_accept_qs_rules',
					$query_rules, $query );
				$rules .= implode( "\n", $query_rules ) . "\n";
			}

			$rules .= "if (\$w3tc_query_string ~ ^[?&]+$) {\n";
			$rules .= "    set \$w3tc_query_string \"\";\n";
			$rules .= "}\n";

			$rules .= "set \$w3tc_request_uri \$request_uri;\n";
			$rules .= "if (\$w3tc_request_uri ~* \"^([^?]+)\?\") {\n";
			$rules .= "    set \$w3tc_request_uri \$1;\n";
			$rules .= "}\n";
			$env_request_uri = '$w3tc_request_uri';
		}

		/**
		 * Check for mobile redirect
		 */
		if ( $config->get_boolean( 'mobile.enabled' ) ) {
			$mobile_groups = $config->get_array( 'mobile.rgroups' );

			foreach ( $mobile_groups as $mobile_group => $mobile_config ) {
				$mobile_enabled = ( isset( $mobile_config['enabled'] ) ?
					(boolean) $mobile_config['enabled'] : false );
				$mobile_agents = ( isset( $mobile_config['agents'] ) ?
					(array) $mobile_config['agents'] : '' );
				$mobile_redirect = ( isset( $mobile_config['redirect'] ) ?
					$mobile_config['redirect'] : '' );

				if ( $mobile_enabled && count( $mobile_agents ) && $mobile_redirect ) {
					$rules .= "if (\$http_user_agent ~* \"(" . implode( '|',
						$mobile_agents ) . ")\") {\n";
					$rules .= "    rewrite .* " . $mobile_redirect . " last;\n";
					$rules .= "}\n";
				}
			}
		}

		/**
		 * Check for referrer redirect
		 */
		if ( $config->get_boolean( 'referrer.enabled' ) ) {
			$referrer_groups = $config->get_array( 'referrer.rgroups' );

			foreach ( $referrer_groups as $referrer_group => $referrer_config ) {
				$referrer_enabled = ( isset( $referrer_config['enabled'] ) ?
					(boolean) $referrer_config['enabled'] : false );
				$referrer_referrers = ( isset( $referrer_config['referrers'] ) ?
					(array) $referrer_config['referrers'] : '' );
				$referrer_redirect = ( isset( $referrer_config['redirect'] ) ?
					$referrer_config['redirect'] : '' );

				if ( $referrer_enabled && count( $referrer_referrers ) &&
					$referrer_redirect ) {
					$rules .= "if (\$http_cookie ~* \"w3tc_referrer=.*(" .
						implode( '|', $referrer_referrers ) . ")\") {\n";
					$rules .= "    rewrite .* " . $referrer_redirect . " last;\n";
					$rules .= "}\n";
				}
			}
		}

		/**
		 * Don't accept POSTs
		 */
		$rules .= "set \$w3tc_rewrite 1;\n";
		$rules .= "if (\$request_method = POST) {\n";
		$rules .= "    set \$w3tc_rewrite 0;\n";
		$rules .= "}\n";

		/**
		 * Query string should be empty
		 */
		$querystring_variable = ( empty( $w3tc_query_strings ) ?
			'$query_string' : '$w3tc_query_string' );

		$rules .= "if (" . $querystring_variable . " != \"\") {\n";
		$rules .= "    set \$w3tc_rewrite 0;\n";
		$rules .= "}\n";

		$rules .= "set \$w3tc_slash \"\";\n";
		$rules .= "if ($env_request_uri ~ \\/$) {\n";
		$rules .= "    set \$w3tc_slash _slash;\n";
		$rules .= "}\n";
		$env_w3tc_slash = "\$w3tc_slash";

		/**
		 * Check for rejected cookies
		 */
		$rules .= "if (\$http_cookie ~* \"(" . implode( '|',
			array_map( array( '\W3TC\Util_Environment', 'preg_quote' ), $reject_cookies ) ) . ")\") {\n";
		$rules .= "    set \$w3tc_rewrite 0;\n";
		$rules .= "}\n";

		/**
		 * Check for rejected user agents
		 */
		if ( count( $reject_user_agents ) ) {
			$rules .= "if (\$http_user_agent ~* \"(" . implode( '|',
				array_map( array( '\W3TC\Util_Environment', 'preg_quote' ), $reject_user_agents ) ) . ")\") {\n";
			$rules .= "    set \$w3tc_rewrite 0;\n";
			$rules .= "}\n";
		}

		/**
		 * Check mobile groups
		 */
		if ( $config->get_boolean( 'mobile.enabled' ) ) {
			$mobile_groups = array_reverse( $config->get_array( 'mobile.rgroups' ) );
			$set_ua_var = true;

			foreach ( $mobile_groups as $mobile_group => $mobile_config ) {
				$mobile_enabled = ( isset( $mobile_config['enabled'] ) ?
					(boolean) $mobile_config['enabled'] : false );
				$mobile_agents = ( isset( $mobile_config['agents'] ) ?
					(array) $mobile_config['agents'] : '' );
				$mobile_redirect = ( isset( $mobile_config['redirect'] ) ?
					$mobile_config['redirect'] : '' );

				if ( $mobile_enabled && count( $mobile_agents ) &&
					!$mobile_redirect ) {
					if ( $set_ua_var ) {
						$rules .= "set \$w3tc_ua \"\";\n";
						$set_ua_var = false;
					}
					$rules .= "if (\$http_user_agent ~* \"(" .
						implode( '|', $mobile_agents ) . ")\") {\n";
					$rules .= "    set \$w3tc_ua _" . $mobile_group . ";\n";
					$rules .= "}\n";

					$env_w3tc_ua = "\$w3tc_ua";
				}
			}
		}

		/**
		 * Check for preview cookie
		 */
		$rules .= "set \$w3tc_preview \"\";\n";
		$rules .= "if (\$http_cookie ~* \"(w3tc_preview)\") {\n";
		$rules .= "    set \$w3tc_preview _preview;\n";
		$rules .= "}\n";
		$env_w3tc_preview = "\$w3tc_preview";

		/**
		 * Check referrer groups
		 */
		if ( $config->get_boolean( 'referrer.enabled' ) ) {
			$referrer_groups = array_reverse( $config->get_array( 'referrer.rgroups' ) );
			$set_ref_var = true;
			foreach ( $referrer_groups as $referrer_group => $referrer_config ) {
				$referrer_enabled = ( isset( $referrer_config['enabled'] ) ?
					(boolean) $referrer_config['enabled'] : false );
				$referrer_referrers = ( isset( $referrer_config['referrers'] ) ?
					(array) $referrer_config['referrers'] : '' );
				$referrer_redirect = ( isset( $referrer_config['redirect'] ) ?
					$referrer_config['redirect'] : '' );

				if ( $referrer_enabled && count( $referrer_referrers ) &&
					!$referrer_redirect ) {
					if ( $set_ref_var ) {
						$rules .= "set \$w3tc_ref \"\";\n";
						$set_ref_var = false;
					}
					$rules .= "if (\$http_cookie ~* \"w3tc_referrer=.*(" .
						implode( '|', $referrer_referrers ) . ")\") {\n";
					$rules .= "    set \$w3tc_ref _" . $referrer_group . ";\n";
					$rules .= "}\n";

					$env_w3tc_ref = "\$w3tc_ref";
				}
			}
		}

		/**
		 * Set cookie group
		 */
		if ( $config->get_boolean( 'pgcache.cookiegroups.enabled' ) ) {
			$cookie_groups = $config->get_array( 'pgcache.cookiegroups.groups' );
			$set_cookie_var = true;

			foreach ( $cookie_groups as $group_name => $g ) {
				if ( isset( $g['enabled'] ) && $g['enabled'] ) {
					$cookies = array();
					foreach ($g['cookies'] as $cookie ) {
						$cookie = trim( $cookie );
						if ( !empty( $cookie ) ) {
							$cookie = str_replace( '+', ' ', $cookie );
							$cookie = Util_Environment::preg_quote( $cookie );
							if ( strpos( $cookie, '=') === false )
								$cookie .= '=.*';
							$cookies[] = $cookie;
						}
					}

					if ( count( $cookies ) > 0 ) {
						$cookies_regexp = '"^(.*;)?(' . implode( '|', $cookies ) . ')(;.*)?$"';

						if ( $set_cookie_var ) {
							$rules .= "set \$w3tc_cookie \"\";\n";
							$set_cookie_var = false;
						}
						$rules .= "if (\$http_cookie ~* $cookies_regexp) {\n";
						$rules .= "    set \$w3tc_cookie _" . $group_name . ";\n";
						$rules .= "}\n";

						$env_w3tc_cookie = "\$w3tc_cookie";
					}
				}
			}
		}

		if ( $config->get_boolean( 'pgcache.cache.ssl' ) ) {
			$rules .= "set \$w3tc_ssl \"\";\n";

			$rules .= "if (\$scheme = https) {\n";
			$rules .= "    set \$w3tc_ssl _ssl;\n";
			$rules .= "}\n";
			$rules .= "if (\$http_x_forwarded_proto = 'https') {\n";
			$rules .= "    set \$w3tc_ssl _ssl;\n";
			$rules .= "}\n";

			$env_w3tc_ssl = '$w3tc_ssl';
		}

		if ( $config->get_boolean( 'browsercache.enabled' ) &&
			$config->get_boolean( 'browsercache.html.compression' ) ) {
			$rules .= "set \$w3tc_enc \"\";\n";

			$rules .= "if (\$http_accept_encoding ~ gzip) {\n";
			$rules .= "    set \$w3tc_enc _gzip;\n";
			$rules .= "}\n";

			$env_w3tc_enc = '$w3tc_enc';
		}

		if ( $config->get_boolean( 'browsercache.enabled' ) &&
			$config->get_boolean( 'browsercache.html.brotli' ) ) {
			$rules .= "set \$w3tc_enc \"\";\n";

			$rules .= "if (\$http_accept_encoding ~ br) {\n";
			$rules .= "    set \$w3tc_enc _br;\n";
			$rules .= "}\n";

			$env_w3tc_enc = '$w3tc_enc';
		}

		$key_postfix = $env_w3tc_slash . $env_w3tc_ua . $env_w3tc_ref . $env_w3tc_cookie .
			$env_w3tc_ssl . $env_w3tc_preview;

		/**
		 * Filter: Allow modifying the key_postfix string used in the PGCACHE_CORE block.
		 *
		 * @since 2.7.1
		 *
		 * @param string $key_postfix Key postfix string.
		 */
		$key_postfix = \apply_filters( 'w3tc_pgcache_postfix_nginx', $key_postfix );

		if ( $pgcache_engine == 'file_generic' ) {
			$rules .= $this->for_file_generic( $config, $cache_dir,
				$env_request_uri, $key_postfix, $env_w3tc_enc );
		} elseif ( $pgcache_engine == 'nginx_memcached' ) {
			$rules .= $this->for_nginx_memcached( $config, $cache_dir,
				$env_request_uri, $key_postfix, $env_w3tc_enc );
		}

		$rules .= W3TC_MARKER_END_PGCACHE_CORE . "\n";

		return $rules;
	}



	private function for_file_generic( $config, $cache_dir, $env_request_uri,
			$key_postfix, $env_w3tc_enc ) {
		$rules = '';

		$cache_path = str_replace( Util_Environment::document_root(), '',
			$cache_dir );
		$uri_prefix = "$cache_path/\$http_host/$env_request_uri/_index$key_postfix";
		$uri_prefix = apply_filters( 'w3tc_pagecache_rules_nginx_uri_prefix',
			$uri_prefix );

		if ( !$config->get_boolean( 'pgcache.cache.nginx_handle_xml' ) ) {
			$env_w3tc_ext = '.html';

			$rules .= 'if (!-f "$document_root' . $uri_prefix . '.html' .
				$env_w3tc_enc . '") {' . "\n";
			$rules .= '  set $w3tc_rewrite 0;' . "\n";
			$rules .= "}\n";
		} else {
			$env_w3tc_ext = '$w3tc_ext';

			$rules .= 'set $w3tc_ext "";' . "\n";
			$rules .= 'if (-f "$document_root' . $uri_prefix . '.html' .
				$env_w3tc_enc . '") {' . "\n";
			$rules .= '  set $w3tc_ext .html;' . "\n";
			$rules .= "}\n";

			$rules .= 'if (-f "$document_root' . $uri_prefix . '.xml' .
				$env_w3tc_enc . '") {' . "\n";
			$rules .= '    set $w3tc_ext .xml;' . "\n";
			$rules .= "}\n";

			$rules .= 'if ($w3tc_ext = "") {' . "\n";
			$rules .= '  set $w3tc_rewrite 0;' . "\n";
			$rules .= "}\n";
		}

		$rules .= 'if ($w3tc_rewrite = 1) {' . "\n";
		$rules .= '    rewrite .* "' . $uri_prefix . $env_w3tc_ext . $env_w3tc_enc .
			'" last;' . "\n";
		$rules .= "}\n";

		return $rules;
	}



	private function for_nginx_memcached( $config, $cache_dir, $env_request_uri,
			$key_postfix, $env_w3tc_enc ) {
		$rules =  "set \$request_uri_noslash $env_request_uri;\n";
		$rules .= "if ($env_request_uri ~ \"(.*?)(/+)$\") {\n";
		$rules .= '    set $request_uri_noslash $1;' . "\n";
		$rules .= "}\n";

		$cache_path = str_replace( Util_Environment::document_root(), '',
			$cache_dir );

		$rules .= 'location ~ ".*(?<!php)$" {' . "\n";
		$rules .= '  set $memcached_key "$http_host$request_uri_noslash/' .
			$key_postfix . $env_w3tc_enc . '";' . "\n";

		if ( $config->get_boolean( 'browsercache.enabled' ) &&
			$config->get_boolean( 'browsercache.html.compression' ) ) {
			$rules .= '  memcached_gzip_flag 65536;' . "\n";
		   }

		$rules .= '  default_type text/html;' . "\n";

		$memcached_servers = $config->get_array( 'pgcache.memcached.servers' );
		$memcached_pass = !empty( $memcached_servers ) ? array_values( $memcached_servers )[0] : 'localhost:11211';

		$rules .= '  if ($w3tc_rewrite = 1) {' . "\n";
		$rules .= '    memcached_pass ' . $memcached_pass . ';' . "\n";
		$rules .= "  }\n";
		$rules .= '  error_page     404 502 504 = @fallback;' . "\n";
		$rules .= "}\n";

		$rules .= 'location @fallback {' . "\n";
		$rules .= '  try_files $uri $uri/ $uri.html /index.php?$args;' . "\n";
		$rules .= "}\n";

		return $rules;
	}



	/*
	 * cache rules
	 */

	/**
	 * Writes directives to file cache .htaccess
	 * Throws exception on error
	 *
	 * @param Config  $config
	 * @param Util_Environment_Exceptions $exs
	 */
	private function rules_cache_add( $config, $exs ) {
		Util_Rule::add_rules( $exs,
			Util_Rule::get_pgcache_rules_cache_path(),
			$this->rules_cache_generate( $config ),
			W3TC_MARKER_BEGIN_PGCACHE_CACHE,
			W3TC_MARKER_END_PGCACHE_CACHE,
			array(
				W3TC_MARKER_BEGIN_BROWSERCACHE_CACHE => 0,
				W3TC_MARKER_BEGIN_MINIFY_CORE => 0,
				W3TC_MARKER_BEGIN_PGCACHE_CORE => 0,
				W3TC_MARKER_BEGIN_WORDPRESS => 0,
				W3TC_MARKER_END_MINIFY_CACHE => strlen( W3TC_MARKER_END_MINIFY_CACHE ) + 1
			)
		);
	}

	/**
	 * Removes Page Cache cache directives
	 *
	 * @param Util_Environment_Exceptions $exs
	 * @throws Util_WpFile_FilesystemOperationException with S/FTP form if it can't get the required filesystem credentials
	 */
	private function rules_cache_remove( $exs ) {
		// apache's cache files are not used when core rules disabled
		if ( !Util_Environment::is_nginx() )
			return;

		Util_Rule::remove_rules( $exs,
			Util_Rule::get_pgcache_rules_cache_path(),
			W3TC_MARKER_BEGIN_PGCACHE_CACHE,
			W3TC_MARKER_END_PGCACHE_CACHE );
	}

	/**
	 * Generates directives for file cache dir
	 *
	 * @param Config  $config
	 * @return string
	 */
	public function rules_cache_generate( $config ) {
		switch ( true ) {
		case Util_Environment::is_apache():
		case Util_Environment::is_litespeed():
			return $this->rules_cache_generate_apache( $config );

		case Util_Environment::is_nginx():
			return $this->rules_cache_generate_nginx( $config );
		}

		return '';
	}


	/**
	 * Generates directives for file cache dir
	 *
	 * @param Config  $config
	 * @return string
	 */
	private function rules_cache_generate_apache( $config ) {
		$charset = get_option( 'blog_charset' );
		$pingback_url = get_bloginfo( 'pingback_url' );

		$browsercache = $config->get_boolean( 'browsercache.enabled' );
		$brotli = ( $browsercache && $config->get_boolean( 'browsercache.html.brotli' ) );
		$compression = ( $browsercache && $config->get_boolean( 'browsercache.html.compression' ) );
		$expires = ( $browsercache && $config->get_boolean( 'browsercache.html.expires' ) );
		$lifetime = ( $browsercache ? $config->get_integer( 'browsercache.html.lifetime' ) : 0 );
		$cache_control = ( $browsercache && $config->get_boolean( 'browsercache.html.cache.control' ) );
		$etag = ( $browsercache && $config->get_integer( 'browsercache.html.etag' ) );
		$w3tc = ( $browsercache && $config->get_integer( 'browsercache.html.w3tc' ) );
		$compatibility = $config->get_boolean( 'pgcache.compatibility' );

		$rules = '';
		$rules .= W3TC_MARKER_BEGIN_PGCACHE_CACHE . "\n";
		if ( $compatibility ) {
			$rules .= "Options -MultiViews\n";

			// allow to read files by apache if they are blocked at some level above
			$rules .= "<Files ~ \"\.(html|html_gzip|html_br|xml|xml_gzip|xml_br)$\">\n";

			if ( version_compare( Util_Environment::get_server_version(), '2.4', '>=' ) ) {
				$rules .= "  Require all granted\n";
			} else {
				$rules .= "  Order Allow,Deny\n";
				$rules .= "  Allow from all\n";
			}

			$rules .= "</Files>\n";

			if ( !$etag ) {
				$rules .= "FileETag None\n";
			}
		}
		if ( $config->get_boolean( 'pgcache.file.nfs' ) ) {
			$rules .= "EnableSendfile Off \n";
		}

		if ( !$config->get_boolean( 'pgcache.remove_charset' ) ) {
			$rules .= "AddDefaultCharset " . ( $charset ? $charset : 'utf-8' ) . "\n";
		}

		if ( $etag ) {
			$rules .= "FileETag MTime Size\n";
		}

		if ( $brotli ) {
			$rules .= "<IfModule mod_mime.c>\n";
			$rules .= "    AddType text/html .html_br\n";
			$rules .= "    AddEncoding br .html_br\n";
			$rules .= "    AddType text/xml .xml_br\n";
			$rules .= "    AddEncoding br .xml_br\n";
			$rules .= "</IfModule>\n";
			$rules .= "<IfModule mod_setenvif.c>\n";
			$rules .= "    SetEnvIfNoCase Request_URI \\.html_br$ no-brotli\n";
			$rules .= "    SetEnvIfNoCase Request_URI \\.xml_br$ no-brotli\n";
			$rules .= "</IfModule>\n";
		}

		if ( $compression ) {
			$rules .= "<IfModule mod_mime.c>\n";
			$rules .= "    AddType text/html .html_gzip\n";
			$rules .= "    AddEncoding gzip .html_gzip\n";
			$rules .= "    AddType text/xml .xml_gzip\n";
			$rules .= "    AddEncoding gzip .xml_gzip\n";
			$rules .= "</IfModule>\n";
			$rules .= "<IfModule mod_setenvif.c>\n";
			$rules .= "    SetEnvIfNoCase Request_URI \\.html_gzip$ no-gzip\n";
			$rules .= "    SetEnvIfNoCase Request_URI \\.xml_gzip$ no-gzip\n";
			$rules .= "</IfModule>\n";
		}

		if ( $expires ) {
			$rules .= "<IfModule mod_expires.c>\n";
			$rules .= "    ExpiresActive On\n";
			$rules .= "    ExpiresByType text/html M" . $lifetime . "\n";
			$rules .= "</IfModule>\n";
		}

		$header_rules = '';

		if ( $compatibility ) {
			$header_rules .= "    Header set X-Pingback \"" . $pingback_url . "\"\n";
		}

		if ( $w3tc ) {
			$header_rules .= "    Header set X-Powered-By \"" .
				Util_Environment::w3tc_header() . "\"\n";
		}

		if ( $expires ) {
			$header_rules .= "    Header set Vary \"Accept-Encoding, Cookie\"\n";
		}

		$set_last_modified = $config->get_boolean( 'browsercache.html.last_modified' );

		if ( !$set_last_modified && $config->get_boolean( 'browsercache.enabled' ) ) {
			$header_rules .= "    Header unset Last-Modified\n";
		}

		if ( $cache_control ) {
			$cache_policy = $config->get_string( 'browsercache.html.cache.policy' );

			switch ( $cache_policy ) {
			case 'cache':
				$header_rules .= "    Header set Pragma \"public\"\n";
				$header_rules .= "    Header set Cache-Control \"public\"\n";
				break;

			case 'cache_public_maxage':
				$header_rules .= "    Header set Pragma \"public\"\n";

				if ( $expires ) {
					$header_rules .= "    Header append Cache-Control \"public\"\n";
				} else {
					$header_rules .= "    Header set Cache-Control \"max-age=" . $lifetime . ", public\"\n";
				}
				break;

			case 'cache_validation':
				$header_rules .= "    Header set Pragma \"public\"\n";
				$header_rules .= "    Header set Cache-Control \"public, must-revalidate, proxy-revalidate\"\n";
				break;

			case 'cache_noproxy':
				$header_rules .= "    Header set Pragma \"public\"\n";
				$header_rules .= "    Header set Cache-Control \"private, must-revalidate\"\n";
				break;

			case 'cache_maxage':
				$header_rules .= "    Header set Pragma \"public\"\n";

				if ( $expires ) {
					$header_rules .= "    Header append Cache-Control \"public, must-revalidate, proxy-revalidate\"\n";
				} else {
					$header_rules .= "    Header set Cache-Control \"max-age=" . $lifetime . ", public, must-revalidate, proxy-revalidate\"\n";
				}
				break;

			case 'no_cache':
				$header_rules .= "    Header set Pragma \"no-cache\"\n";
				$header_rules .= "    Header set Cache-Control \"private, no-cache\"\n";
				break;

			case 'no_store':
				$header_rules .= "    Header set Pragma \"no-store\"\n";
				$header_rules .= "    Header set Cache-Control \"no-store\"\n";
				break;
			}
		}

		if ( strlen( $header_rules ) > 0 ) {
			$rules .= "<IfModule mod_headers.c>\n";
			$rules .= $header_rules;
			$rules .= "</IfModule>\n";
		}

		$rules .= W3TC_MARKER_END_PGCACHE_CACHE . "\n";

		return $rules;
	}

	/**
	 * Generates directives for file cache dir
	 *
	 * @param Config  $config
	 * @return string
	 */
	private function rules_cache_generate_nginx( $config ) {
		if ( $config->get_string( 'pgcache.engine') != 'file_generic' ) {
			return '';
		}

		$cache_root = Util_Environment::normalize_path( W3TC_CACHE_PAGE_ENHANCED_DIR );
		$cache_dir = rtrim( str_replace( Util_Environment::document_root(), '', $cache_root ), '/' );

		if ( Util_Environment::is_wpmu() ) {
			$cache_dir = preg_replace( '~/w3tc.*?/~', '/w3tc.*?/', $cache_dir, 1 );
		}

		$browsercache = $config->get_boolean( 'browsercache.enabled' );
		$brotli = ( $browsercache && $config->get_boolean( 'browsercache.html.brotli' ) );
		$compression = ( $browsercache && $config->get_boolean( 'browsercache.html.compression' ) );

		$common_rules_a = Dispatcher::nginx_rules_for_browsercache_section(
			$config, 'html', true );
		$common_rules = '';
		if ( !empty( $common_rules_a ) ) {
			$common_rules = '    ' . implode( "\n    ", $common_rules_a ) . "\n";
		}

		$rules = '';
		$rules .= W3TC_MARKER_BEGIN_PGCACHE_CACHE . "\n";

		if ( $brotli ) {
			$maybe_xml = '';
			if ($config->get_boolean('pgcache.cache.nginx_handle_xml')) {
				$maybe_xml = "\n" .
					"        text/xml xml_br;\n" .
					"    ";
			}

			$rules .= "location ~ " . $cache_dir . ".*br$ {\n";
			$rules .= "    brotli off;\n";
			$rules .= "    types {" . $maybe_xml . "}\n";
			$rules .= "    default_type text/html;\n";
			$rules .= "    add_header Content-Encoding br;\n";
			$rules .= $common_rules;
			$rules .= "}\n";
		}

		if ( $compression ) {
			$maybe_xml = '';
			if ($config->get_boolean('pgcache.cache.nginx_handle_xml')) {
				$maybe_xml = "\n" .
					"        text/xml xml_gzip;\n" .
					"    ";
			}

			$rules .= "location ~ " . $cache_dir . ".*gzip$ {\n";
			$rules .= "    gzip off;\n";
			$rules .= "    types {" . $maybe_xml . "}\n";
			$rules .= "    default_type text/html;\n";
			$rules .= "    add_header Content-Encoding gzip;\n";
			$rules .= $common_rules;
			$rules .= "}\n";
		}

		$rules .= W3TC_MARKER_END_PGCACHE_CACHE . "\n";

		return $rules;
	}
}