HEX
Server: Apache/2.4.41
System: Linux mainweb 5.4.0-182-generic #202-Ubuntu SMP Fri Apr 26 12:29:36 UTC 2024 x86_64
User: nationalmedicaregrp (1119)
PHP: 8.3.7
Disabled: exec,passthru,shell_exec,system,popen,proc_open,pcntl_exec
Upload Files
File: /home/healthcaremadeeasyagency/public_html/wp-content/plugins/wpseo-local/src/tools/import.php
<?php
/**
 * Yoast SEO: Local plugin file.
 *
 * @package WPSEO_LOCAL\Import
 */

namespace Yoast\WP\Local\Tools;

use Yoast\WP\Local\Conditionals\Admin_Conditional;
use Yoast\WP\Local\Conditionals\Multiple_Locations_Conditional;
use Yoast\WP\Local\PostType\PostType;
use Yoast\WP\Local\Repositories\Business_Types_Repository;

/**
 * Class that holds the functionality for the WPSEO Local Import function
 *
 * @since 3.9
 */
class Import extends Import_Export {

	/**
	 * @var \Yoast\WP\Local\PostType\PostType
	 */
	private $post_type;

	/**
	 * Contains the Business Types Repository.
	 *
	 * @var Business_Types_Repository
	 */
	private $business_types_repository;

	/**
	 * Import constructor.
	 *
	 * @param \Yoast\WP\Local\Repositories\Business_Types_Repository $business_types_repository The business type repository as a dependency.
	 * @param \Yoast\WP\Local\PostType\PostType                      $post_type                 The post type object as a dependency.
	 */
	public function __construct( Business_Types_Repository $business_types_repository, PostType $post_type ) {
		$this->business_types_repository = $business_types_repository;
		$this->post_type                 = $post_type;
	}

	/**
	 * Returns a list of conditionals that must be met.
	 *
	 * @return array A list of conditionals that must be met to use the class
	 */
	public static function get_conditionals() {
		return [
			Admin_Conditional::class,
			Multiple_Locations_Conditional::class,
		];
	}

	/**
	 * Register hooks and filters
	 */
	public function register_hooks() {
		parent::register_hooks();
		\add_action( 'admin_init', [ $this, 'handle_csv_import' ], 11 );
		\add_action( 'wpseo_import_tab_content_inner', [ $this, 'import_html' ], 11 );
	}

	/**
	 * Handle the import of the uploaded .csv file.
	 */
	public function handle_csv_import() {
		if ( ! isset( $_POST['csv-import'] )
			|| \check_admin_referer( 'wpseo_local_import_nonce', 'wpseo_local_import_nonce_field' ) === false
		) {
			return;
		}

		if ( ! isset( $_FILES['wpseo']['name']['csvuploadlocations'] )
			|| ! is_string( $_FILES['wpseo']['name']['csvuploadlocations'] )
			|| ! isset( $_FILES['wpseo']['tmp_name']['csvuploadlocations'] )
		) {
			return;
		}

		/**
		 * Set max execution time to 3600 seconds (or 1 hour)
		 *
		 * @todo See if we can do in batches, to avoid tempering with execution times
		 */
		\set_time_limit( 3600 );

		$count = 0;

		$file_name     = sanitize_file_name( wp_unslash( $_FILES['wpseo']['name']['csvuploadlocations'] ) );
		$tmp_file_name = wp_unslash( $_FILES['wpseo']['tmp_name']['csvuploadlocations'] );

		$csv_path = $this->wpseo_upload_dir . \basename( $file_name );

		if ( ! empty( $_FILES['wpseo'] ) && ! \move_uploaded_file( $tmp_file_name, $csv_path ) ) {
			$this->messages[] = [
				'type'    => 'error',
				'content' => \sprintf(
					/* translators: %s expands to the location of the WPSEO upload dir. */
					\esc_html__( 'Sorry, there was an error while uploading the CSV file.<br>Please make sure the %s directory is writable (chmod 777).', 'yoast-local-seo' ),
					$this->wpseo_upload_dir
				),
			];

			return;
		}

		$is_simplemap_import = ! empty( $_POST['is-simplemap-import'] ) && $_POST['is-simplemap-import'] == '1';

		$separator = ',';
		if ( ( ! empty( $_POST['csv_separator'] ) && $_POST['csv_separator'] === 'semicolon' ) && $is_simplemap_import === false ) {
			$separator = ';';
		}

		// Get location data from CSV.
		$column_names = $this->columns;

		// If a simplemap import is used, overwrite our WPSEO Local columns.
		if ( $is_simplemap_import ) {
			$column_names = [
				'name',
				'address',
				'address2',
				'city',
				'state',
				'zipcode',
				'country',
				'phone',
				'email',
				'fax',
				'url',
				'description',
				'special',
				'lat',
				'long',
				'pubdate',
				'category',
				'tag',
			];
		}

		$handle = \fopen( $csv_path, 'r' );

		if ( ! $handle ) {
			$this->messages[] = [
				'type'    => 'error',
				'content' => \esc_html__( 'There seems to be an error reading the uploaded file.', 'yoast-local-seo' ),
			];

			return;
		}

		$locations = [];
		$row       = 0;
		while ( ( $csvdata = \fgetcsv( $handle, 2000, $separator ) ) !== false ) {
			if ( $row > 0 ) {
				$tmp_location           = [];
				$column_name_count      = \count( $column_names );
				$simplemap_skip_columns = [
					'email',
					'url',
					'special',
					'pubdate',
					'tag',
				];

				for ( $i = 0; $i < $column_name_count; $i++ ) {

					// Skip columns for simplemap import.
					if ( $is_simplemap_import
						&& \in_array( $column_names[ $i ], $simplemap_skip_columns, true )
					) {
						continue;
					}

					if ( isset( $csvdata[ $i ] ) ) {
						$tmp_location[ $column_names[ $i ] ] = addslashes( $csvdata[ $i ] );
					}
				}
				\array_push( $locations, $tmp_location );
			}
			++$row;
		}
		\fclose( $handle );

		$debug = false;

		/**
		 * The WPSEO Local Core class (TODO refactor out)
		 *
		 * @var \WPSEO_Local_Core $wpseo_local_core
		 */
		global $wpseo_local_core;

		$business_types = $this->business_types_repository->get_business_types();
		\array_walk( $business_types, 'wpseo_local_sanitize_business_types' );

		foreach ( $locations as $location ) {
			// Create standard post data.
			$current_post                 = [];
			$current_post['ID']           = '';
			$current_post['post_title']   = isset( $location['name'] ) ? $location['name'] : '';
			$current_post['post_content'] = isset( $location['description'] ) ? $location['description'] : '';
			$current_post['post_status']  = 'publish';
			$current_post['post_type']    = $this->post_type->get_post_type();

			$post_id = \wp_insert_post( $current_post );

			if ( ! $debug ) {
				// Insert custom fields for location details.
				if ( ! empty( $post_id ) ) {
					\add_post_meta( $post_id, '_wpseo_business_name', isset( $location['name'] ) ? \sanitize_text_field( $location['name'] ) : '', true );
					\add_post_meta( $post_id, '_wpseo_business_address', isset( $location['address'] ) ? \sanitize_text_field( $location['address'] ) : '', true );
					\add_post_meta( $post_id, '_wpseo_business_address_2', isset( $location['address'] ) ? \sanitize_text_field( $location['address_2'] ) : '', true );
					\add_post_meta( $post_id, '_wpseo_business_city', isset( $location['city'] ) ? \sanitize_text_field( $location['city'] ) : '', true );
					\add_post_meta( $post_id, '_wpseo_business_state', isset( $location['state'] ) ? \sanitize_text_field( $location['state'] ) : '', true );
					\add_post_meta( $post_id, '_wpseo_business_zipcode', isset( $location['zipcode'] ) ? \sanitize_text_field( $location['zipcode'] ) : '', true );
					\add_post_meta( $post_id, '_wpseo_business_country', isset( $location['country'] ) ? \sanitize_text_field( $location['country'] ) : '', true );
					\add_post_meta( $post_id, '_wpseo_business_phone', isset( $location['phone'] ) ? \sanitize_text_field( $location['phone'] ) : '', true );
					\add_post_meta( $post_id, '_wpseo_business_fax', isset( $location['fax'] ) ? \sanitize_text_field( $location['fax'] ) : '', true );
					\add_post_meta( $post_id, '_wpseo_business_email', isset( $location['email'] ) ? \sanitize_email( $location['email'] ) : '', true );
					\add_post_meta( $post_id, '_wpseo_business_vat_id', isset( $location['vat_id'] ) ? \sanitize_text_field( $location['vat_id'] ) : '', true );
					\add_post_meta( $post_id, '_wpseo_business_tax_id', isset( $location['tax_id'] ) ? \sanitize_text_field( $location['tax_id'] ) : '', true );
					\add_post_meta( $post_id, '_wpseo_business_coc_id', isset( $location['coc_id'] ) ? \sanitize_text_field( $location['coc_id'] ) : '', true );

					if ( isset( $location['phone_2nd'] ) ) {
						\add_post_meta( $post_id, '_wpseo_business_phone_2nd', $location['phone_2nd'], true );
					}

					if ( isset( $location['category'] ) ) {
						// Allow for a comma separated list to be used.
						$categories = \explode( ',', $location['category'] );

						// Remove possible spaces.
						\array_walk( $categories, 'trim' );

						// And finally set the terms in the locations category.
						\wp_set_object_terms( $post_id, $categories, 'wpseo_locations_category' );
					}

					if ( isset( $location['business_type'] ) ) {
						$business_type = $location['business_type'];
						if ( ! isset( $business_types[ $business_type ] ) ) {
							$business_type = \array_search( $business_type, $business_types, true );
						}

						\add_post_meta( $post_id, '_wpseo_business_type', $business_type, true );
					}

					if ( isset( $location['url'] ) ) {
						\add_post_meta( $post_id, '_wpseo_business_url', \sanitize_text_field( $location['url'] ), true );
					}

					// Replace comma's into points.
					if ( isset( $location['lat'] ) ) {
						$location['lat'] = \str_replace( ',', '.', $location['lat'] );
						\add_post_meta( $post_id, '_wpseo_coordinates_lat', \sanitize_text_field( $location['lat'] ), true );
					}
					if ( isset( $location['long'] ) ) {
						$location['long'] = \str_replace( ',', '.', $location['long'] );
						\add_post_meta( $post_id, '_wpseo_coordinates_long', \sanitize_text_field( $location['long'] ), true );
					}

					// If just a postal adddress, check box.
					if ( $location['is_postal_address'] == '1' ) {
						\update_post_meta( $post_id, '_wpseo_is_postal_address', '1' );
					}

					// Add image as post thumbnail.
					if ( ! empty( $location['image'] ) ) {
						$wpseo_local_core->insert_attachment( $post_id, $location['image'], true );
					}

					if ( ! empty( $location['location_logo'] ) ) {
						$logo_id = $wpseo_local_core->insert_attachment( $post_id, $location['location_logo'], false );
						update_post_meta( $post_id, '_wpseo_business_location_logo', \wp_get_attachment_image_url( $logo_id, 'full' ) );
					}

					if ( ! empty( $location['custom_marker'] ) ) {
						$marker_id = $wpseo_local_core->insert_attachment( $post_id, $location['custom_marker'], false );
						update_post_meta( $post_id, '_wpseo_business_location_custom_marker', \wp_get_attachment_image_url( $marker_id ) );
					}

					// Opening hours.
					foreach ( $wpseo_local_core->days as $key => $day ) {
						if ( isset( $location[ 'opening_hours_' . $key . '_from' ] ) && ! empty( $location[ 'opening_hours_' . $key . '_from' ] ) && isset( $location[ 'opening_hours_' . $key . '_to' ] ) && ! empty( $location[ 'opening_hours_' . $key . '_to' ] ) ) {
							if ( \strtolower( $location[ 'opening_hours_' . $key . '_from' ] ) === 'closed' || \strtolower( $location[ 'opening_hours_' . $key . '_to' ] ) === 'closed' ) {
								\add_post_meta( $post_id, '_wpseo_opening_hours_' . $key . '_from', 'closed', true );
							}
							else {
								$time_from = \strtotime( $location[ 'opening_hours_' . $key . '_from' ] );
								$time_to   = \strtotime( $location[ 'opening_hours_' . $key . '_to' ] );

								if ( $time_from !== false && $time_to !== false ) {
									\add_post_meta( $post_id, '_wpseo_opening_hours_' . $key . '_from', \gmdate( 'H:i', $time_from ), true );
									\add_post_meta( $post_id, '_wpseo_opening_hours_' . $key . '_to', \gmdate( 'H:i', $time_to ), true );
								}
								else {
									\add_post_meta( $post_id, '_wpseo_opening_hours_' . $key . '_from', 'closed', true );
									if ( $time_from === false ) {
										$this->messages[] = [
											'type'    => 'error',
											'content' => sprintf(
												/* translators: %s extends to Location opening hours from per week day */
												__( '%s is not a valid time notation', 'yoast-local-seo' ),
												$location[ 'opening_hours_' . $key . '_from' ]
											),
										];
									}
									elseif ( $time_to === false ) {
										$this->messages[] = [
											'type'    => 'error',
											'content' => \sprintf(
												/* translators: %s extends to Location opening hours to per week day */
												__( '%s is not a valid time notation', 'yoast-local-seo' ),
												$location[ 'opening_hours_' . $key . '_to' ]
											),
										];
									}
								}

								if ( $location['multiple_opening_hours'] === 'on' ) {
									// Multiple openingtimes are set. Enable them in the backend.
									\update_post_meta( $post_id, '_wpseo_multiple_opening_hours', 'on', true );
								}

								if ( isset( $location[ 'opening_hours_' . $key . '_second_from' ] ) && ! empty( $location[ 'opening_hours_' . $key . '_second_from' ] ) && isset( $location[ 'opening_hours_' . $key . '_second_to' ] ) && ! empty( $location[ 'opening_hours_' . $key . '_second_to' ] ) ) {
									if ( isset( $location[ 'opening_hours_' . $key . '_second_from' ] ) && ! empty( $location[ 'opening_hours_' . $key . '_from' ] ) && isset( $location[ 'opening_hours_' . $key . '_second_to' ] ) && ! empty( $location[ 'opening_hours_' . $key . '_to' ] ) ) {
										if ( \strtolower( $location[ 'opening_hours_' . $key . '_second_from' ] ) === 'closed' || \strtolower( $location[ 'opening_hours_' . $key . '_second_to' ] ) === 'closed' ) {
											\add_post_meta( $post_id, '_wpseo_opening_hours_' . $key . '_second_from', 'closed', true );
										}
									}
									else {
										$time_second_from = \strtotime( $location[ 'opening_hours_' . $key . '_second_from' ] );
										$time_second_to   = \strtotime( $location[ 'opening_hours_' . $key . '_second_to' ] );

										if ( $time_second_from !== false && $time_second_to !== false ) {
											\add_post_meta( $post_id, '_wpseo_opening_hours_' . $key . '_second_from', \gmdate( 'H:i', $time_second_from ), true );
											\add_post_meta( $post_id, '_wpseo_opening_hours_' . $key . '_second_to', \gmdate( 'H:i', $time_second_to ), true );
										}
										else {
											\add_post_meta( $post_id, '_wpseo_opening_hours_' . $key . '_second_from', 'closed', true );
											if ( $time_second_from === false ) {
												$this->messages[] = [
													'type'    => 'error',
													'content' => \sprintf(
														/* translators: %s extends to Location second from opening hours per week day */
														__( '%s is not a valid time notation', 'yoast-local-seo' ),
														$location[ 'opening_hours_' . $key . '_second_from' ]
													),
												];
											}
											elseif ( $time_second_to === false ) {
												$this->messages[] = [
													'type'    => 'error',
													'content' => \sprintf(
														/* translators: %s extends to Location second to opening hours per week day */
														__( '%s is not a valid time notation', 'yoast-local-seo' ),
														$location[ 'opening_hours_' . $key . '_second_to' ]
													),
												];
											}
										}
									}
								}
								else {
									\add_post_meta( $post_id, '_wpseo_opening_hours_' . $key . '_second_from', 'closed', true );
								}
							}
						}
						else {
							\add_post_meta( $post_id, '_wpseo_opening_hours_' . $key . '_from', 'closed', true );
						}
					}

					++$count;
					$last_imported = $post_id;
				}
			}
		}

		if ( $count > 0 ) {
			$this->messages[] = [
				'type'    => 'success',
				'content' => \sprintf(
					/* translators: 1: nr of locations found; 2: link to post type overview; 3: link close tag. */
					__( '%1$d locations found and successfully imported %2$shere%3$s', 'yoast-local-seo' ),
					$count,
					'<a href="' . \esc_url( \get_admin_url( null, 'edit.php?post_type=' . $this->post_type->get_post_type() ) ) . '">',
					'</a>'
				),
			];
		}
	}

	/**
	 * Builds the HTML for the import form.
	 */
	public function import_html() {
		echo '<h2>' . \esc_html__( 'Import', 'yoast-local-seo' ) . '</h2>';

		/* translators: %1$s extends to link opening tag; %2$s closes the tag */
		echo '<p>' . \sprintf( \esc_html__( 'View the %1$sdocumentation%2$s to check what format of the CSV file should be.', 'yoast-local-seo' ), '<a href="https://yoast.com/question/csv-import-file-local-seo-look-like/" target="_blank">', '</a>' ) . '</p>';

		echo '<form action="" method="post" enctype="multipart/form-data">';
		\WPSEO_Local_Admin_Wrappers::file_upload( 'csvuploadlocations', __( 'Upload CSV', 'yoast-local-seo' ) );
		echo '<label for="csv_separator" class="checkbox">' . \esc_html__( 'Column separator', 'yoast-local-seo' ) . '</label>';
		echo '<select class="textinput" id="csv_separator" name="csv_separator">';
		echo '<option value="comma">' . \esc_html__( 'Comma', 'yoast-local-seo' ) . '</option>';
		echo '<option value="semicolon">' . \esc_html__( 'Semicolon', 'yoast-local-seo' ) . '</option>';
		echo '</select>';
		echo '<br class="clear">';
		echo '<p>';
		echo '<input class="checkbox double" id="is-simplemap-import" type="checkbox" name="is-simplemap-import" value="1"> ';
		echo '<label for="is-simplemap-import">' . \esc_html__( 'This CSV is exported by the SimpleMap plugin', 'yoast-local-seo' ) . '</label>';
		echo '</p>';
		echo '<br class="clear">';
		echo '<br/>';

		echo '<p><em>' . \esc_html__( 'Note', 'yoast-local-seo' ) . ': ' . \esc_html__( 'The Geocoding API is limited to 2,500 queries a day, so when you have large CSV files, with no coordinates, cut them in pieces of 2,500 rows and import them one a day. Indeed, it\'s not funny. It\'s reality.', 'yoast-local-seo' ) . '</em></p>';

		if ( ! \is_writable( $this->wpseo_upload_dir ) ) {
			/* translators: %s extends to the upload directory */
			echo '<p>' . \sprintf( \esc_html__( 'Make sure the %s directory is writeable.', 'yoast-local-seo' ), '<code>"' . \esc_html( $this->wpseo_upload_dir ) . '"</code>' ) . '</p>';
		}

		// Add a NONCE field.
		\wp_nonce_field( 'wpseo_local_import_nonce', 'wpseo_local_import_nonce_field' );

		echo '<input type="submit" class="button button-primary" name="csv-import" value="Import" ' . ( ! \is_writable( $this->wpseo_upload_dir ) ? ' disabled="disabled"' : '' ) . ' />';
		echo '</form>';
	}
}