File "FeedConfigurationDetection.php"

Full Path: /home/vantageo/public_html/wp-admin/.wp-cli/wp-content/plugins/facebook-for-woocommerce/includes/Feed/FeedConfigurationDetection.php
File size: 7.95 KB
MIME-type: text/x-php
Charset: utf-8

<?php
// phpcs:ignoreFile

namespace WooCommerce\Facebook\Feed;

defined( 'ABSPATH' ) || exit;

use Error;
use Exception;
use WC_Facebookcommerce_Utils;
use WooCommerce\Facebook\API\Exceptions\Request_Limit_Reached;
use WooCommerce\Facebook\API\Response;
use WooCommerce\Facebook\Framework\Api\Exception as ApiException;
use WooCommerce\Facebook\Products\Feed;
use WooCommerce\Facebook\Utilities\Heartbeat;

/**
 * A class responsible detecting feed configuration.
 */
class FeedConfigurationDetection {

	/**
	 * Constructor.
	 */
	public function __construct() {
		add_action( Heartbeat::DAILY, array( $this, 'track_data_source_feed_tracker_info' ) );
	}

	/**
	 * Store config settings for feed-based sync for WooCommerce Tracker.
	 *
	 * Gets various settings related to the feed, and data about recent uploads.
	 * This is formatted into an array of keys/values, and saved to a transient for inclusion in tracker snapshot.
	 * Note this does not send the data to tracker - this happens later (see Tracker class).
	 *
	 * @since 2.6.0
	 * @return void
	 */
	public function track_data_source_feed_tracker_info() {
		try {
			$info = $this->get_data_source_feed_tracker_info();
			facebook_for_woocommerce()->get_tracker()->track_facebook_feed_config( $info );
		} catch ( \Error $error ) {
			facebook_for_woocommerce()->log( 'Unable to detect valid feed configuration: ' . $error->getMessage() );
		}
	}

	/**
	 * Get config settings for feed-based sync for WooCommerce Tracker.
	 *
	 * @throws Error Catalog id missing.
	 * @return array Key-value array of various configuration settings.
	 */
	private function get_data_source_feed_tracker_info() {
		$integration         = facebook_for_woocommerce()->get_integration();
		$integration_feed_id = $integration->get_feed_id();
		$catalog_id          = $integration->get_product_catalog_id();

		$info                 = array();
		$info['site-feed-id'] = $integration_feed_id;

		// No catalog id. Most probably means that we don't have a valid connection.
		if ( '' === $catalog_id ) {
			throw new Error( 'No catalog ID' );
		}

		// Get all feeds configured for the catalog.
		$feed_nodes = $this->get_feed_nodes_for_catalog( $catalog_id );

		$info['feed-count'] = count( $feed_nodes );

		// Check if the catalog has any feed configured.
		if ( empty( $feed_nodes ) ) {
			throw new Error( 'No feed nodes for catalog' );
		}

		/*
		 * We will only track settings for one feed config (for now at least).
		 * So we need to determine which is the most relevant feed.
		 * If there is only one, we use that.
		 * If one has the same ID as $integration_feed_id, we use that.
		 * Otherwise we pick the one that was most recently updated.
		 */
		$active_feed_metadata = array();
		foreach ( $feed_nodes as $feed ) {
			try {
				$metadata = $this->get_feed_metadata( $feed['id'] );
			} catch ( Exception $e ) {
				$message = sprintf( 'There was an error trying to get feed metadata: %s', $e->getMessage() );
				WC_Facebookcommerce_Utils::log( $message );
				continue;
			}

			if ( $feed['id'] === $integration_feed_id ) {
				$active_feed_metadata = clone $metadata;
				break;
			}

			if (
				! isset( $metadata['latest_upload'] ) ||
				! is_array( $metadata['latest_upload'] ) ||
				! array_key_exists( 'start_time', $metadata['latest_upload'] )
			) {
				continue;
			}

			$metadata['latest_upload_time'] = strtotime( $metadata['latest_upload']['start_time'] );
			if (
				! $active_feed_metadata ||
				$metadata['latest_upload_time'] > $active_feed_metadata['latest_upload_time']
			) {
				$active_feed_metadata = clone $metadata;
			}
		}

		if ( empty( $active_feed_metadata ) ) {
			// No active feed available, we don't have data to collect.
			$info['active-feed'] = null;
			return $info;
		}

		$active_feed = array();
		if ( isset( $active_feed_metadata['created_time'] ) ) {
			$active_feed['created-time'] = gmdate( 'Y-m-d H:i:s', strtotime( $active_feed_metadata['created_time'] ) );
		}

		if ( isset( $active_feed_metadata['product_count'] ) ) {
			$active_feed['product-count'] = $active_feed_metadata['product_count'];
		}

		/*
		 * Upload schedule settings can be in two keys:
		 * `schedule` => full replace of catalog with items in feed (including delete).
		 * `update_schedule` => append any new or updated products to catalog.
		 * These may both be configured; we will track settings for each individually (i.e. both).
		 * https://developers.facebook.com/docs/marketing-api/reference/product-feed/
		 */
		if ( isset( $active_feed_metadata['schedule'] ) ) {
			$active_feed['schedule']['interval']       = $active_feed_metadata['schedule']['interval'];
			$active_feed['schedule']['interval-count'] = $active_feed_metadata['schedule']['interval_count'];
		}
		if ( isset( $active_feed_metadata['update_schedule'] ) ) {
			$active_feed['update-schedule']['interval']       = $active_feed_metadata['update_schedule']['interval'];
			$active_feed['update-schedule']['interval-count'] = $active_feed_metadata['update_schedule']['interval_count'];
		}

		$info['active-feed'] = $active_feed;

		if ( isset( $active_feed_metadata['latest_upload'] ) ) {
			$latest_upload = $active_feed_metadata['latest_upload'];
			$upload        = array();

			if ( array_key_exists( 'end_time', $latest_upload ) ) {
				$upload['end-time'] = gmdate( 'Y-m-d H:i:s', strtotime( $latest_upload['end_time'] ) );
			}

			// Get more detailed metadata about the most recent feed upload.
			$upload_metadata = $this->get_feed_upload_metadata( $latest_upload['id'] );

			// If no metadata is available, we can't track any more details.
			if ( ! $upload_metadata ) {
				$info['active-feed']['latest-upload'] = $upload;
				return $info;
			}

			$upload['error-count']         = $upload_metadata['error_count'];
			$upload['warning-count']       = $upload_metadata['warning_count'];
			$upload['num-detected-items']  = $upload_metadata['num_detected_items'];
			$upload['num-persisted-items'] = $upload_metadata['num_persisted_items'];

			// True if the feed upload url (Facebook side) matches the feed endpoint URL and secret.
			// If it doesn't match, it's likely it's unused.
			$upload['url-matches-site-endpoint'] = wc_bool_to_string(
				Feed::get_feed_data_url() === $upload_metadata['url']
			);

			$info['active-feed']['latest-upload'] = $upload;
		}

		return $info;
	}

	/**
	 * Given catalog id this function fetches all feed configurations defined for this catalog.
	 *
	 * @throws Error Feed configurations fetch was not successful.
	 * @param String                        $catalog_id Facebook Catalog ID.
	 *
	 * @return array Array of feed configurations.
	 */

	/**
	 * @param string $product_catalog_id
	 *
	 * @return array Facebook Product Feeds.
	 * @throws Request_Limit_Reached
	 * @throws ApiException
	 */
	private function get_feed_nodes_for_catalog( string $product_catalog_id ) {
		try {
			$response = facebook_for_woocommerce()->get_api()->read_feeds($product_catalog_id);
		} catch ( \Exception $e ) {
			$message = sprintf( 'There was an error trying to get feed nodes for catalog: %s', $e->getMessage() );
			facebook_for_woocommerce()->log( $message );
			return array();
		}
		return $response->data;
	}

	/**
	 * Given feed id fetch this feed configuration metadata.
	 *
	 * @param string $feed_id Facebook Product Feed ID.
	 *
	 * @return Response
	 * @throws Request_Limit_Reached
	 * @throws ApiException
	 */
	private function get_feed_metadata( string $feed_id ) {
		return facebook_for_woocommerce()->get_api()->read_feed( $feed_id );
	}

	/**
	 * Given upload id fetch this upload execution metadata.
	 *
	 * @param string $upload_id Facebook Feed upload ID.
	 *
	 * @return Response
	 * @throws Error Upload metadata fetch was not successful.
	 */
	private function get_feed_upload_metadata( $upload_id ) {
		try {
			$response = facebook_for_woocommerce()->get_api()->read_upload($upload_id);
		} catch ( \Exception $e ) {
			$message = sprintf( 'There was an error trying to get feed upload metadata: %s', $e->getMessage() );
			facebook_for_woocommerce()->log( $message );
			return false;
		}
		return $response;
	}

}