HEX
Server: nginx/1.24.0
System: Linux ht2024073053593 5.14.0-480.el9.x86_64 #1 SMP PREEMPT_DYNAMIC Fri Jul 12 20:45:27 UTC 2024 x86_64
User: root (0)
PHP: 7.4.33
Disabled: passthru,exec,system,putenv,chroot,chgrp,chown,shell_exec,popen,proc_open,pcntl_exec,ini_alter,ini_restore,dl,openlog,syslog,readlink,symlink,popepassthru,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_dispatch,pcntl_get_last_error,pcntl_strerror,pcntl_sigprocmask,pcntl_sigwaitinfo,pcntl_sigtimedwait,pcntl_exec,pcntl_getpriority,pcntl_setpriority,imap_open,apache_setenv
Upload Files
File: /www/wwwroot/njmuedu.com/wp-content/plugins/xml-sitemap-feed/inc/class.xmlsf-sitemap-plugin.php
<?php

/* -------------------------------
 *   XMLSF Sitemap Plugin CLASS
 * ------------------------------- */

class XMLSF_Sitemap_Plugin extends XMLSF_Sitemap
{
	/**
	 * Rewrite rules
	 * @var array
	 */
	public $rewrite_rules = array(
		array(
			'regex' => 'sitemap(?:_index)?(-[a-z0-9\-_]+)?(?:\.([0-9]{4,8}))?(?:\.([0-9]{1,2}))?\.xml(\.gz)?$',
			'query' => '?feed=sitemap$matches[1]$matches[4]&m=$matches[2]&w=$matches[3]'
		)
	);

	/**
	 * CONSTRUCTOR
	 * Runs on init
	 */
	function __construct( $sitemap = 'sitemap.xml' )
	{
		$this->sitemap = $sitemap;

		$this->post_types = (array) get_option( 'xmlsf_post_types', array() );

		// Rewrite rules filter.
		add_filter( 'rewrite_rules_array', array( $this, 'rewrite_rules' ), 99, 1 );

		// Redirect wp-sitemap requests.
		add_action( 'template_redirect', array( $this, 'redirect'),	0 );

		// Pings.
		add_action( 'transition_post_status', array( $this, 'do_pings' ), 10, 3 );

		// Cache clearance.
		add_action( 'clean_post_cache', array( $this, 'clean_post_cache'), 99, 2 );

		// Update term meta lastmod date.
		add_action( 'transition_post_status', array( $this, 'update_term_modified_meta' ), 10, 3 );

		// Update user meta lastmod date.
		add_action( 'transition_post_status', array( $this, 'update_user_modified_meta' ), 10, 3 );

		// Update images post meta.
		add_action( 'transition_post_status', array( $this, 'update_post_images_meta' ), 10, 3 );

		// Update last comment date post meta.
		add_action( 'transition_comment_status', array( $this, 'update_post_comment_meta' ), 10, 3 );
		add_action( 'comment_post', array( $this, 'update_post_comment_meta_cp' ), 10, 3 ); // when comment is not held for moderation

		// MAIN REQUEST filter.
		add_filter( 'request', array( $this, 'filter_request' ), 1 );

		// Add index archive data filter.
		add_filter( 'xmlsf_index_archive_data', array( $this, 'index_archive_data' ), 10, 3 );

		// NGINX HELPER PURGE URLS
		add_filter( 'rt_nginx_helper_purge_urls', array( $this, 'nginx_helper_purge_urls' ) );
	}

	/**
	 * Filter request
	 *
	 * @param array $request
	 *
	 * @return array $request filtered
	 */
	public function filter_request( $request )
	{
		global $xmlsf, $wp_rewrite;

		// Short-circuit if request was already filtered by this plugin.
		if ( $xmlsf->request_filtered ) {
			return $request;
		} else {
			$xmlsf->request_filtered = true;
		}

		// Short-circuit if request is not a feed, news sitemap, does not start with 'sitemap'.
		if ( empty( $request['feed'] ) || 'sitemap-news' === $request['feed'] || strpos( $request['feed'], 'sitemap' ) !== 0 ) {
			return $request;
		}

		/** IT'S A SITEMAP */

		// Set the sitemap conditional flag.
		$xmlsf->is_sitemap = true;

		// Set rewrite trailingslash to false.
		$wp_rewrite->use_trailing_slashes = false;

		// Save a few db queries.
		add_filter( 'split_the_query', '__return_false' );

		// Include public functions.
		require_once XMLSF_DIR . '/inc/functions.public.php';

		// Generator comments.
		add_action( 'xmlsf_generator', 'xmlsf_generator' );

		/** COMPRESSION */

		// Check for gz request.
		if ( substr( $request['feed'], -3 ) == '.gz' ) {
			// Pop that .gz
			$request['feed'] = substr($request['feed'], 0, -3);
			// Verify/apply compression settings.
			xmlsf_output_compression();
		}

		/** MODIFY REQUEST PARAMETERS */

		$request['post_status'] = 'publish';
		$request['no_found_rows'] = true; // Found rows calc is slow and only needed for pagination.

		// Make sure we have the proper locale setting for calculations.
		setlocale( LC_NUMERIC, 'C' );

		// SPECIFIC REQUEST FILTERING AND PREPARATIONS

		// Include public sitemap functions.
		require_once XMLSF_DIR . '/inc/functions.public-sitemap.php';

		/** FILTER HOOK FOR PLUGIN COMPATIBILITIES */

		/**
		 * Filters the request.
		 *
		 * add_filter( 'xmlsf_request', 'your_filter_function' );
		 *
		 * Filters hooked here already:
		 * xmlsf_polylang_request - Polylang compatibility
		 * xmlsf_wpml_request - WPML compatibility
		 * xmlsf_bbpress_request - bbPress compatibility
		 */
		$request = apply_filters( 'xmlsf_request', $request );

		$feed = explode( '-' , $request['feed'], 3 );

		switch( isset($feed[1]) ? $feed[1] : '' ) {

			case 'posttype':
				$settings = (array) get_option( 'xmlsf_post_types' );
				if ( ! isset( $feed[2] ) || empty( $settings[$feed[2]] ) || ! is_array( $settings[$feed[2]] ) || empty( $settings[$feed[2]]['active'] ) ) {
					return $request;
				}

				// Try to raise memory limit, context added for filters.
				wp_raise_memory_limit( 'sitemap-posttype-'.$feed[2] );

				// Prepare priority calculation.
				if ( ! empty($this->post_types[$feed[2]]['dynamic_priority']) ) {
					// Last of this post type modified date in Unix seconds.
					xmlsf()->lastmodified = get_date_from_gmt( get_lastpostmodified( 'GMT', $feed[2] ), 'U' );
					// Calculate time span, uses get_firstpostdate() function defined in xml-sitemap/inc/functions.php!
					xmlsf()->timespan = xmlsf()->lastmodified - get_date_from_gmt( get_firstpostdate( 'GMT', $feed[2]), 'U' );
					// Total post type comment count.
					xmlsf()->comment_count = wp_count_comments()->approved;
					// TODO count comments per post type https://wordpress.stackexchange.com/questions/134338/count-all-comments-of-a-custom-post-type
					// TODO cache this more persistently than wp_cache_set does in https://developer.wordpress.org/reference/functions/wp_count_comments/
				};

				// Setup filters.
				add_filter( 'post_limits', function() { return 'LIMIT 0, 50000'; } );

				// Modify request.
				$request['post_type'] = $feed[2];
				$request['orderby'] = 'modified';
				$request['order'] = 'DESC';

				// Prevent term cache update query unless needed for permalinks.
				if ( strpos( get_option( 'permalink_structure' ), '%category%' ) === false )
					$request['update_post_term_cache'] = false;

				// Make sure to update meta cache for:
				// 1. excluded posts
				// 2. image data (if activated)
				// 3. lasmod on comments (if activated)
				$request['update_post_meta_cache'] = true;
				break;

			case 'taxonomy':
				$settings = get_option( 'xmlsf_taxonomy_settings' );
				if ( ! isset( $feed[2] ) || ! is_array( $settings ) || empty( $settings['active'] ) ) {
					return $request;
				}

				$taxonomies = get_option( 'xmlsf_taxonomies' );
				if ( ! empty( $taxonomies ) && ! in_array( $feed[2], (array) $taxonomies ) ) {
					return $request;
				}

				// Try to raise memory limit, context added for filters.
				wp_raise_memory_limit( 'sitemap-taxonomy-'.$feed[2] );
				// Pass on taxonomy name via request.
				$request['taxonomy'] = $feed[2];
				// Set terms args.
				add_filter( 'get_terms_args', array( $this, 'set_terms_args' ) );
				break;

			case 'author':
				$settings = get_option( 'xmlsf_author_settings' );
				if ( ! is_array( $settings ) || empty( $settings['active'] ) ) {
					return $request;
				}

				// Set users args.
				add_filter( 'get_users_args', array( $this, 'set_authors_args' ) );
				break;

			default:
				// We're on the index. Do nothing.
		}

		/** PREPARE TO LOAD TEMPLATE */
		add_action (
			'do_feed_' . $request['feed'],
			'xmlsf_load_template',
			10,
			2
		);

		/** GENERAL MISC. PREPARATIONS */

		// Prevent public errors breaking xml.
		@ini_set( 'display_errors', 0 );

		// REPSONSE HEADERS filtering.
		add_filter( 'wp_headers', 'xmlsf_headers' );

		// Remove filters to prevent stuff like cdn urls for xml stylesheet and images.
		remove_all_filters( 'plugins_url' );
		remove_all_filters( 'wp_get_attachment_url' );
		remove_all_filters( 'image_downsize' );

		// Remove actions that we do not need.
		remove_all_actions( 'widgets_init' );
		remove_all_actions( 'wp_footer' );

		return $request;
	}

	/**
	 * Terms arguments filter
	 * Does not check if we are really in a sitemap feed.
	 *
	 * @param $args
	 *
	 * @return array
	 */
	function set_terms_args( $args ) {
		// https://developer.wordpress.org/reference/classes/wp_term_query/__construct/

		$options = get_option('xmlsf_taxonomy_settings');

		$args['number'] = isset($options['limit']) && is_numeric( $options['limit'] ) ? intval($options['limit']) : 2000;
		if ( $args['number'] < 1 || $args['number'] > 50000 ) $args['number'] = 50000;

		$args['order']           = 'DESC';
		$args['orderby']         = 'count';
		$args['pad_counts']      = true;
		$args['lang']            = '';
		$args['hierarchical']    = 0;
		$args['suppress_filter'] = true;

		return $args;
	}

	/**
	 * Authors arguments filter
	 * Does not check if we are really in a sitemap feed.
	 *
	 * @param $args
	 *
	 * @return array
	 */
	function set_authors_args( $args ) {
		$author_settings = get_option( 'xmlsf_author_settings' );

		/**
		 * Filters the post types present in the author archive. Must return an array of one or multiple post types.
		 * Allows to add or change post type when theme author archive page shows custom post types.
		 *
		 * @since 0.1
		 *
		 * @param array Array with post type slugs. Default array('post').
		 *
		 * @return array
		 */
		$post_type_array = apply_filters( 'xmlsf_author_post_types', array( 'post' ) );

		$args['number'] = ! empty( $author_settings['limit'] ) && is_numeric( $author_settings['limit'] ) ? intval( $author_settings['limit'] ) : 2000;
		if ( $args['number'] < 1 || $args['number'] > 50000 ) $args['number'] = 50000;

		$args['orderby'] = 'post_count';
		$args['order'] = 'DESC';
		//$args['fields'] = array( 'ID' ); // must be an array
		//$args['who'] = 'authors'; // Deprecated since 5.9.
		$args['has_published_posts'] = $post_type_array;

		return $args;
	}

	/**
	 * Do WP core sitemap index redirect
	 *
	 * @uses wp_redirect()
	 */
	public function redirect()
	{
		if ( ! empty( $_SERVER['REQUEST_URI'] ) && substr( $_SERVER['REQUEST_URI'], 0, 15) === '/wp-sitemap.xml' ) {
			wp_redirect( home_url( $this->sitemap ), 301, 'XML Sitemap & Google News for WordPress' );
			exit();
		}
	}

	/**
	 * Get post archives data
	 *
	 * @param array $data
	 * @param string $post_type
	 * @param string $archive_type
	 *
	 * @return array $data
	 */
	public function index_archive_data( $data, $post_type = 'post', $archive_type = '' ) {
		return array_merge( $data, $this->get_index_archive_data( $post_type, $archive_type ) );
	}

	/**
	 * Get post archives data
	 *
	 * @param string $post_type
	 * @param string $archive_type
	 *
	 * @return array
	 */
	public function get_index_archive_data( $post_type, $archive_type )
	{
		global $wpdb;

		$return = array();

		if ( 'weekly' == $archive_type ) :

			$week       = _wp_mysql_week( '`post_date`' );
			$query      = "SELECT DISTINCT LPAD($week,2,'0') AS `week`, YEAR(`post_date`) AS `year`, COUNT(`ID`) AS `posts` FROM {$wpdb->posts} WHERE `post_type` = '{$post_type}' AND `post_status` = 'publish' GROUP BY YEAR(`post_date`), LPAD($week,2,'0') ORDER BY `year` DESC, `week` DESC";
			$arcresults = $this->cache_get_archives( $query );

			foreach ( (array) $arcresults as $arcresult ) {
				$url = xmlsf_sitemap_url( 'posttype', array( 'type' => $post_type, 'm' => $arcresult->year, 'w' => $arcresult->week ) );
				$return[$url] = get_date_from_gmt( get_lastmodified( 'GMT', $post_type, $arcresult->year, $arcresult->week ), DATE_W3C );
			};

		elseif ( 'monthly' == $archive_type ) :

			$query = "SELECT YEAR(`post_date`) AS `year`, LPAD(MONTH(`post_date`),2,'0') AS `month`, COUNT(`ID`) AS `posts` FROM {$wpdb->posts} WHERE `post_type` = '{$post_type}' AND `post_status` = 'publish' GROUP BY YEAR(`post_date`), LPAD(MONTH(`post_date`),2,'0') ORDER BY `year` DESC, `month` DESC";
			$arcresults = $this->cache_get_archives( $query );

			foreach ( (array) $arcresults as $arcresult ) {
				$url = xmlsf_sitemap_url( 'posttype', array( 'type' => $post_type, 'm' => $arcresult->year . $arcresult->month ) );
				$return[$url] = get_date_from_gmt( get_lastmodified( 'GMT', $post_type, $arcresult->year . $arcresult->month ), DATE_W3C );
			};

		elseif ( 'yearly' == $archive_type ) :

			$query      = "SELECT YEAR(`post_date`) AS `year`, COUNT(`ID`) AS `posts` FROM {$wpdb->posts} WHERE `post_type` = '{$post_type}' AND `post_status` = 'publish' GROUP BY YEAR(`post_date`) ORDER BY `year` DESC";
			$arcresults = $this->cache_get_archives( $query );

			foreach ( (array) $arcresults as $arcresult ) {
				$url = xmlsf_sitemap_url( 'posttype', array( 'type' => $post_type, 'm' => $arcresult->year ) );
				$return[$url] = get_date_from_gmt( get_lastmodified( 'GMT', $post_type, $arcresult->year ), DATE_W3C );
			};

		else :

			$query      = "SELECT COUNT(ID) AS `posts` FROM {$wpdb->posts} WHERE `post_type` = '{$post_type}' AND `post_status` = 'publish' ORDER BY `post_date` DESC";
			$arcresults = $this->cache_get_archives( $query );

			if ( is_object($arcresults[0]) && $arcresults[0]->posts > 0 ) {
				$url = xmlsf_sitemap_url( 'posttype', array( 'type' => $post_type ) );
				$return[$url] = get_date_from_gmt( get_lastmodified( 'GMT', $post_type ), DATE_W3C );
			};

		endif;

		return $return;
	}

	/**
	 * Get archives from wp_cache
	 *
	 * @param string $query
	 *
	 * @return array
	 */
	function cache_get_archives( $query ) {

		global $wpdb;

		$key = md5($query);
		$cache = wp_cache_get( 'xmlsf_get_archives' , 'general');

		if ( !isset( $cache[ $key ] ) ) {
			$arcresults = $wpdb->get_results($query);
			$cache[ $key ] = $arcresults;
			wp_cache_set( 'xmlsf_get_archives', $cache, 'general' );
		} else {
			$arcresults = $cache[ $key ];
		}

		return $arcresults;
	}

	/**
	 * Nginx helper purge urls
	 * adds sitemap urls to the purge array.
	 *
	 * @param $urls array
	 * @param $redis bool|false
	 *
	 * @return $urls array
	 */
	public function nginx_helper_purge_urls( $urls = array(), $wildcard = false )
	{
		if ( $wildcard ) {
			// Wildcard makes everything simple.
			$urls[] = '/sitemap*.xml';
		} else {
			// No wildcard, go through the motions.
			$urls[] = '/sitemap.xml';
			$urls[] = '/sitemap-root.xml';
			$urls[] = '/sitemap-author.xml';
			$urls[] = '/sitemap-custom.xml';

			// Add public post types sitemaps.
			$post_types = xmlsf_get_post_types();
			foreach ( $post_types as $post_type => $settings ) :
				$archive = isset( $settings['archive'] ) ? $settings['archive'] : '';
				$archive_data = apply_filters( 'xmlsf_index_archive_data', array(), $post_type, $archive );

				foreach ( $archive_data as $url => $lastmod ) {
					$urls[] = parse_url( $url, PHP_URL_PATH);
				}
			endforeach;

			// Add public post taxonomies sitemaps.
			$taxonomies = xmlsf_get_taxonomies();
			foreach ( $taxonomies as $taxonomy ) {
				$urls[] = parse_url( xmlsf_sitemap_url( 'taxonomy', array( 'type' => $taxonomy ) ), PHP_URL_PATH );
			}
		}

		if ( defined('WP_DEBUG') && WP_DEBUG ) {
			error_log( 'NGINX Helper purge urls array:' );
			error_log( print_r( $urls, true ) );
		}

		return $urls;
	}

}