HEX
Server: LiteSpeed
System: Linux premium212.web-hosting.com 4.18.0-553.124.4.lve.el8.x86_64 #1 SMP Fri May 15 13:02:13 UTC 2026 x86_64
User: vitanhod (1367)
PHP: 8.2.31
Disabled: NONE
Upload Files
File: /home/vitanhod/www/wp-content/plugins/woocommerce/src/Internal/VariationGallery/Migration.php
<?php
/**
 * Batched migration of legacy variation gallery meta into core's native prop.
 */

declare( strict_types=1 );

namespace Automattic\WooCommerce\Internal\VariationGallery;

defined( 'ABSPATH' ) || exit;

/**
 * Migrate legacy variation gallery meta into WooCommerce's native variation gallery prop.
 */
class Migration {

	/**
	 * Number of variations processed per batch.
	 */
	private const BATCH_SIZE = 250;

	/**
	 * Option name recording when the migration finished.
	 */
	public const COMPLETED_OPTION = 'wc_variation_gallery_migration_completed_at';

	/**
	 * Run one batch of the migration.
	 *
	 * @return bool Whether there are pending migration records.
	 */
	public static function run(): bool {
		global $wpdb;

		if ( get_option( self::COMPLETED_OPTION ) ) {
			return false;
		}

		$legacy_meta_key      = '_wc_additional_variation_images';
		$core_gallery_meta    = '_product_image_gallery';
		$fallback_disabled    = LegacyVariationGalleryCompatibility::get_core_managed_meta_key();
		$select_variation_ids = static function ( int $limit ) use ( $wpdb, $legacy_meta_key, $fallback_disabled ): array {
			$query = $wpdb->prepare(
				"SELECT legacy.post_id
				FROM {$wpdb->postmeta} AS legacy
				INNER JOIN {$wpdb->posts} AS posts
					ON posts.ID = legacy.post_id
					AND posts.post_type = 'product_variation'
				LEFT JOIN {$wpdb->postmeta} AS disabled
					ON disabled.post_id = legacy.post_id
					AND disabled.meta_key = %s
				WHERE legacy.meta_key = %s
					AND legacy.meta_value <> ''
					AND disabled.post_id IS NULL
				GROUP BY legacy.post_id
				ORDER BY legacy.post_id ASC
				LIMIT %d",
				$fallback_disabled,
				$legacy_meta_key,
				$limit
			);

			// phpcs:ignore WordPress.DB.PreparedSQL.NotPrepared -- Query is prepared immediately above.
			return array_map( 'intval', $wpdb->get_col( $query ) );
		};

		$variation_ids = $select_variation_ids( self::BATCH_SIZE );

		foreach ( $variation_ids as $variation_id ) {
			$legacy_gallery_image_ids = array_values(
				array_filter( wp_parse_id_list( get_post_meta( $variation_id, $legacy_meta_key, true ) ) )
			);
			$core_gallery_image_ids   = wp_parse_id_list( get_post_meta( $variation_id, $core_gallery_meta, true ) );

			if ( empty( $core_gallery_image_ids ) && ! empty( $legacy_gallery_image_ids ) ) {
				update_post_meta( $variation_id, $core_gallery_meta, implode( ',', $legacy_gallery_image_ids ) );
			}

			// Keep legacy meta for third-party readers; disable fallback via the sentinel instead.
			LegacyVariationGalleryCompatibility::mark_variation_id_core_managed( $variation_id );
		}

		$has_more = ! empty( $select_variation_ids( 1 ) );

		// Guard against duplicate completion events if this runner is invoked twice.
		if ( ! $has_more && ! get_option( self::COMPLETED_OPTION ) ) {
			update_option( self::COMPLETED_OPTION, time() );
			Telemetry::record_event( Telemetry::EVENT_MIGRATION_COMPLETED );
		}

		return $has_more;
	}
}