<?php /** * Copyright (c) Facebook, Inc. and its affiliates. All Rights Reserved * * This source code is licensed under the license found in the * LICENSE file in the root directory of this source tree. * * @package FacebookCommerce * * usage: * 1. set WP_DEBUG = true and WP_DEBUG_DISPLAY = false * 2. append "&fb_test_product_sync=true" to the url when you are on facebook-for-woocommerce setting pages * 3. refresh the page to launch test * https://codex.wordpress.org/WP_DEBUG */ if ( ! defined( 'ABSPATH' ) ) { exit; } require_once dirname( __DIR__ ) . '/fbutils.php'; require_once 'fbproductfeed-test.php'; if ( ! class_exists( 'WC_Facebook_Integration_Test' ) ) : /** * This tests the upload of test objects into Facebook using the plugin's * infrastructure and checks to see if the product field have been correctly * uploaded into FB. */ class WC_Facebook_Integration_Test { const FB_PRODUCT_GROUP_ID = 'fb_product_group_id'; const FB_PRODUCT_ITEM_ID = 'fb_product_item_id'; const MAX_SLEEP_IN_SEC = 90; const MAX_TIME = 'T23:59+00:00'; const MIN_TIME = 'T00:00+00:00'; /** Class Instance */ private static $instance; public static $commerce = null; // Full WC_Facebookcommerce_Integration obj public static $fbgraph = null; public static $test_mode = false; // simple products' id and variable products' parent_id public static $wp_post_ids = array(); // FB product item retailer id. public static $retailer_ids = array(); // product and product_variation post id for test public $product_post_wpid = null; public static $test_pass = 1; /** * Get the class instance */ public static function get_instance( $commerce ) { return null === self::$instance ? ( self::$instance = new self( $commerce ) ) : self::$instance; } /** * Constructor */ public function __construct( $commerce ) { self::$commerce = $commerce; add_action( 'wp_ajax_ajax_test_sync_products_using_feed', array( $this, 'ajax_test_sync_products_using_feed' ) ); } /** * Test visible products by uploading feed. **/ function ajax_test_sync_products_using_feed() { self::$test_mode = true; // test ajax reset all products in db $reset = self::$commerce->reset_all_products(); if ( $reset ) { WC_Facebookcommerce_Utils::log( 'Test - Removing FBIDs from all products' ); $this->product_post_wpid = $this->create_data(); if ( empty( $this->product_post_wpid ) ) { self::$test_pass = 0; WC_Facebookcommerce_Utils::log( 'Test - Fail to create test product by inserting posts.' ); WC_Facebookcommerce_Utils::set_test_fail_reason( 'Fail to create test products by inserting posts.', ( new Exception() )->getTraceAsString() ); update_option( 'fb_test_pass', false ); wp_die(); return; } $this->set_product_wpid( $this->product_post_wpid ); $upload_success = self::$commerce->ajax_sync_all_fb_products_using_feed( true ); if ( $upload_success ) { // verification Step. // Wait till FB finish backend creation to prevent race condition. $time_start = microtime( true ); while ( ( microtime( true ) - $time_start ) < self::MAX_SLEEP_IN_SEC ) { $complete = self::$commerce->fbproductfeed->is_upload_complete( self::$commerce->settings ); if ( $complete ) { break; } else { $this->sleep_til_upload_complete( 10 ); } } $this->sleep_til_upload_complete( 60 ); $check_product_create = $this->check_product_create(); if ( ! $check_product_create ) { self::$test_pass = 0; } else { WC_Facebookcommerce_Utils::log( 'Test - Products create successfully.' ); } // Clean up whatever has been created. // Test on_product_delete API hook. $clean_up = $this->clean_up(); if ( ! $clean_up ) { self::$test_pass = 0; WC_Facebookcommerce_Utils::log( 'Test - Fail to delete product from FB' ); WC_Facebookcommerce_Utils::set_test_fail_reason( 'Fail to delete product from FB', ( new Exception() )->getTraceAsString() ); } else { WC_Facebookcommerce_Utils::log( 'Test - Delete product from FB successfully' ); } } else { self::$test_pass = 0; WC_Facebookcommerce_Utils::log( 'Test - Sync all products using feed, curl failed.' ); WC_Facebookcommerce_Utils::set_test_fail_reason( 'Sync all products using feed, curl failed', ( new Exception() )->getTraceAsString() ); } } else { self::$test_pass = 0; WC_Facebookcommerce_Utils::log( 'Test - Fail to remove FBIDs from local DB' ); WC_Facebookcommerce_Utils::set_test_fail_reason( 'Fail to remove FBIDs from local DB', ( new Exception() )->getTraceAsString() ); } update_option( 'fb_test_pass', self::$test_pass ); wp_die(); return; } function check_product_create() { if ( count( self::$retailer_ids ) < 3 ) { WC_Facebookcommerce_Utils::log( 'Test - Failed to create 3 product items.' ); WC_Facebookcommerce_Utils::set_test_fail_reason( 'Failed to create 3 product items.', ( new Exception() )->getTraceAsString() ); return false; } if ( count( self::$retailer_ids ) > 3 ) { WC_Facebookcommerce_Utils::log( 'Test - Failed to skip invisible products.' ); WC_Facebookcommerce_Utils::set_test_fail_reason( 'Failed to skip invisible products.', ( new Exception() )->getTraceAsString() ); return false; } // Check 3 products have been created. for ( $i = 0; $i < 3; $i++ ) { $product_type = $i == 0 ? 'Simple' : 'Variable'; $retailer_id = self::$retailer_ids[ $i ]; $item_fbid = $this->check_fbid_api( self::FB_PRODUCT_ITEM_ID, $retailer_id ); $group_fbid = $this->check_fbid_api( self::FB_PRODUCT_GROUP_ID, $retailer_id ); if ( ! $item_fbid || ! $group_fbid ) { WC_Facebookcommerce_Utils::log( 'Test - ' . $product_type . ' product failed to create.' ); WC_Facebookcommerce_Utils::set_test_fail_reason( $product_type . ' product failed to create.', ( new Exception() )->getTraceAsString() ); return false; } } // Check product detailed as expected. $data = array( 'name' => 'a simple product for test', 'price' => '20.00', 'description' => 'This is to test a simple product.', 'sale_price' => '10.00', 'sale_price_dates_from' => date_i18n( 'Y-m-d', strtotime( 'now' ) ) . self::MIN_TIME, 'sale_price_dates_to' => date_i18n( 'Y-m-d', strtotime( '+10 day' ) ) . self::MAX_TIME, 'visibility' => 'published', ); $simple_product_result = $this->check_product_info( self::$retailer_ids[0], false, $data ); if ( ! $simple_product_result ) { WC_Facebookcommerce_Utils::log( 'Test - Simple product failed to match ' . 'product details.' ); WC_Facebookcommerce_Utils::set_test_fail_reason( 'Simple product failed to' . ' match product details.', ( new Exception() )->getTraceAsString() ); return false; } $data = array( 'name' => 'a variable product for test', 'price' => '30.00', 'description' => 'This is to test a variable product. - Red', 'additional_variant_attributes' => array( 'value' => 'Red' ), 'visibility' => 'published', ); $variable_product_result = $this->check_product_info( self::$retailer_ids[1], true, $data ); if ( ! $variable_product_result ) { WC_Facebookcommerce_Utils::log( 'Test - Variable product failed to match product details.' ); WC_Facebookcommerce_Utils::set_test_fail_reason( 'Variable product failed to match product details.', ( new Exception() )->getTraceAsString() ); return false; } return true; } function check_fbid_api( $fbid_type, $fb_retailer_id ) { $product_fbid_result = self::$fbgraph->get_facebook_id( self::$commerce->product_catalog_id, $fb_retailer_id, true ); if ( is_wp_error( $product_fbid_result ) ) { WC_Facebookcommerce_Utils::log( 'Test - ' . $product_fbid_result->get_error_message() ); WC_Facebookcommerce_Utils::set_test_fail_reason( 'There was an issue connecting to the Facebook API: ' . $product_fbid_result->get_error_message(), ( new Exception() )->getTraceAsString() ); return false; } if ( $product_fbid_result && isset( $product_fbid_result['body'] ) ) { $body = WC_Facebookcommerce_Utils::decode_json( $product_fbid_result['body'], true ); if ( $body && isset( $body['id'] ) ) { if ( $fbid_type == self::FB_PRODUCT_GROUP_ID ) { $fb_id = isset( $body['product_group'] ) ? $body['product_group']['id'] : false; } else { $fb_id = $body['id']; } return $fb_id; } } return false; } function check_product_info( $retailer_id, $has_variant, $data ) { $prod_info_result = self::$fbgraph->check_product_info( self::$commerce->product_catalog_id, $retailer_id, $has_variant ); if ( is_wp_error( $prod_info_result ) ) { WC_Facebookcommerce_Utils::log( 'Test - ' . $prod_info_result->get_error_message() ); WC_Facebookcommerce_Utils::set_test_fail_reason( 'There was an issue connecting to the Facebook API: ' . $prod_info_result->get_error_message(), ( new Exception() )->getTraceAsString() ); return false; } $match = true; if ( $prod_info_result && isset( $prod_info_result['body'] ) ) { $body = WC_Facebookcommerce_Utils::decode_json( $prod_info_result['body'], true ); if ( ! $body ) { return false; } if ( $body['name'] != $data['name'] ) { WC_Facebookcommerce_Utils::log( 'Test - ' . $retailer_id . " doesn\'t match name." ); $match = false; } if ( $body['description'] != $data['description'] ) { WC_Facebookcommerce_Utils::log( 'Test - ' . $retailer_id . " doesn\'t match description." ); $match = false; } // Woo doesn't have API to return currency symbol. // FB graph API only support to response with a currency symbol price. // No php built-in function to support cast html number to symbol. // Compare numeric price only. $price = floatval( preg_replace( '/[^\d\.]+/', '', $body['price'] ) ); if ( $price != $data['price'] ) { WC_Facebookcommerce_Utils::log( 'Test - ' . $retailer_id . " doesn\'t match price." ); $match = false; } // Check sale price and dates. if ( isset( $data['sale_price'] ) ) { $sale_price = floatval( preg_replace( '/[^\d\.]+/', '', $body['sale_price'] ) ); if ( $sale_price != $data['sale_price'] ) { WC_Facebookcommerce_Utils::log( 'Test - ' . $retailer_id . " doesn\'t match sale price." ); $match = false; } if ( $body['sale_price_start_date'] != $data['sale_price_dates_from'] ) { WC_Facebookcommerce_Utils::log( 'Test - ' . $retailer_id . " doesn\'t match sale price start date" ); $match = false; } if ( $body['sale_price_end_date'] != $data['sale_price_dates_to'] ) { WC_Facebookcommerce_Utils::log( 'Test - ' . $retailer_id . " doesn\'t match sale price end date." ); $match = false; } } if ( $body['visibility'] != $data['visibility'] ) { WC_Facebookcommerce_Utils::log( 'Test - ' . $retailer_id . " doesn\'t match visibility." ); $match = false; } if ( $has_variant && ( ! isset( $body['additional_variant_attributes'] ) || $body['additional_variant_attributes'][0]['value'] != $data['additional_variant_attributes']['value'] ) ) { WC_Facebookcommerce_Utils::log( 'Test - ' . $retailer_id . " doesn\'t match variation." ); $match = false; } } return $match; } // Don't early return to prevent haunting product id. function clean_up() { $failure = false; foreach ( self::$wp_post_ids as $post_id ) { $delete_post_result = wp_delete_post( $post_id ); // return false or null if failed. if ( ! $delete_post_result ) { WC_Facebookcommerce_Utils::log( 'Test - Fail to delete post ' . $post_id ); WC_Facebookcommerce_Utils::set_test_fail_reason( 'Fail to delete post ' . $post_id, ( new Exception() )->getTraceAsString() ); $failure = true; } } self::$wp_post_ids = array(); $this->sleep_til_upload_complete( 60 ); foreach ( self::$retailer_ids as $retailer_id ) { $item_fbid = $this->check_fbid_api( self::FB_PRODUCT_ITEM_ID, $retailer_id ); $group_fbid = $this->check_fbid_api( self::FB_PRODUCT_GROUP_ID, $retailer_id ); if ( $item_fbid || $group_fbid ) { WC_Facebookcommerce_Utils::log( 'Test - Failed to delete product ' . $retailer_id . ' via plugin deletion hook.' ); WC_Facebookcommerce_Utils::set_test_fail_reason( 'Failed to delete product ' . $retailer_id . ' via plugin deletion hook.', ( new Exception() )->getTraceAsString() ); $failure = true; } } self::$retailer_ids = array(); return ! $failure; } function create_data() { $prod_and_variant_wpid = array(); // Gets term object from Accessories in the database. $term = get_term_by( 'name', 'Accessories', 'product_cat' ); // Accessories should be a default category. // If not exist, set categories term first. if ( ! $term ) { $term = wp_insert_term( 'Accessories', // the term 'product_cat', // the taxonomy array( 'slug' => 'accessories', ) ); } $data = array( 'post_content' => 'This is to test a simple product.', 'post_title' => 'a simple product for test', 'post_status' => 'publish', 'post_type' => 'product', 'term' => $term, 'price' => 20, 'sale_price' => 10, 'sale_price_dates_from' => strtotime( 'now' ), 'sale_price_dates_to' => strtotime( '+10 day' ), ); $simple_product_result = $this->create_test_simple_product( $data, $prod_and_variant_wpid ); if ( ! $simple_product_result ) { return false; } // Test an invisible product - invisible products won't be synced by feed. $data['visibility'] = false; $simple_product_result = $this->create_test_simple_product( $data, $prod_and_variant_wpid ); if ( ! $simple_product_result ) { return false; } $data['post_content'] = 'This is to test a variable product.'; $data['post_title'] = 'a variable product for test'; $data['price'] = 30; // Test variable products. $variable_product_result = $this->create_test_variable_product( $data, $prod_and_variant_wpid ); if ( ! $variable_product_result ) { return false; } return $prod_and_variant_wpid; } function create_test_simple_product( $data, &$prod_and_variant_wpid ) { $post_id = $this->fb_insert_post( $data, 'Simple' ); if ( ! $post_id ) { return false; } array_push( $prod_and_variant_wpid, $post_id ); update_post_meta( $post_id, '_regular_price', $data['price'] ); update_post_meta( $post_id, '_sale_price', $data['sale_price'] ); update_post_meta( $post_id, '_sale_price_dates_from', $data['sale_price_dates_from'] ); update_post_meta( $post_id, '_sale_price_dates_to', $data['sale_price_dates_to'] ); wp_set_object_terms( $post_id, 'simple', 'product_type' ); // Invisible products won't be synced by feed. if ( isset( $data['visibility'] ) ) { $terms = array( 'exclude-from-catalog', 'exclude-from-search' ); wp_set_object_terms( $post_id, $terms, 'product_visibility' ); } else { array_push( self::$wp_post_ids, $post_id ); array_push( self::$retailer_ids, 'wc_post_id_' . $post_id ); } $product = wc_get_product( $post_id ); $product->set_stock_status( 'instock' ); wp_set_object_terms( $post_id, $data['term']->term_id, 'product_cat' ); return true; } function create_test_variable_product( $data, &$prod_and_variant_wpid ) { $post_id = $this->fb_insert_post( $data, 'Variable' ); if ( ! $post_id ) { return false; } wp_set_object_terms( $post_id, 'variable', 'product_type' ); array_push( $prod_and_variant_wpid, $post_id ); array_push( self::$wp_post_ids, $post_id ); // Gets term object from Accessories in the database. $term = get_term_by( 'name', 'Accessories', 'product_cat' ); wp_set_object_terms( $post_id, $term->term_id, 'product_cat' ); // Set up attributes. $avail_attribute_values = array( 'Red', 'Blue', ); wp_set_object_terms( $post_id, $avail_attribute_values, 'pa_color' ); $thedata = array( 'pa_color' => array( 'name' => 'pa_color', 'value' => '', 'is_visible' => '1', 'is_variation' => '1', 'is_taxonomy' => '1', ), ); update_post_meta( $post_id, '_product_attributes', $thedata ); // Insert variations. $variation_data = array( 'post_content' => 'This is to test a variable product. - Red', 'post_status' => 'publish', 'post_type' => 'product_variation', 'post_parent' => $post_id, 'price' => 30, ); $variation_red = $this->fb_insert_post( $variation_data, 'Variation' ); if ( ! $variation_red ) { return; } $this->fb_update_variation_meta( $prod_and_variant_wpid, $variation_red, 'Red', $variation_data ); $variation_data['post_content'] = 'a variable product for test - Blue'; $variation_blue = $this->fb_insert_post( $variation_data, 'Variatoin' ); if ( ! $variation_blue ) { return false; } $this->fb_update_variation_meta( $prod_and_variant_wpid, $variation_blue, 'Blue', $variation_data ); $product = wc_get_product( $variation_blue ); $product->set_stock_status( 'instock' ); wp_set_object_terms( $variation_blue, 'variation', 'product_type' ); return true; } function fb_update_variation_meta( &$prod_and_variant_wpid, $variation_id, $value, $data ) { array_push( $prod_and_variant_wpid, $variation_id ); array_push( self::$retailer_ids, 'wc_post_id_' . $variation_id ); $attribute_term = get_term_by( 'name', $value, 'pa_color' ); update_post_meta( $variation_id, 'attribute_pa_color', $attribute_term->slug ); update_post_meta( $variation_id, '_price', $data['price'] ); update_post_meta( $variation_id, '_regular_price', $data['price'] ); wp_set_object_terms( $variation_id, 'variation', 'product_type' ); $product = wc_get_product( $variation_id ); $product->set_stock_status( 'instock' ); } function fb_insert_post( $data, $p_type ) { $postarr = array_intersect_key( $data, array_flip( array( 'post_content', 'post_title', 'post_status', 'post_type', 'post_parent', ) ) ); $post_id = wp_insert_post( $postarr ); if ( is_wp_error( $post_id ) ) { WC_Facebookcommerce_Utils::log( 'Test - ' . $p_type . ' product wp_insert_post' . 'failed: ' . json_encode( $post_id ) ); return false; } else { return $post_id; } } /** * IMPORTANT! Wait for Ents creation and prevent race condition. **/ function sleep_til_upload_complete( $sec ) { sleep( $sec ); } function set_product_wpid( $product_post_wpid ) { WC_Facebook_Product_Feed_Test_Mock::$product_post_wpid = $product_post_wpid; } } endif;