logger = $logger; $this->price_updater = $price_updater; // Add admin menu. add_action( 'admin_menu', array( $this, 'add_admin_menu' ) ); // Register settings. add_action( 'admin_init', array( $this, 'register_settings' ) ); // Handle OAuth callback. add_action( 'admin_init', array( $this, 'handle_oauth_callback' ) ); // Handle AJAX requests. add_action( 'wp_ajax_informatiq_sp_manual_sync', array( $this, 'handle_manual_sync' ) ); add_action( 'wp_ajax_informatiq_sp_test_connection', array( $this, 'handle_test_connection' ) ); add_action( 'wp_ajax_informatiq_sp_revoke_auth', array( $this, 'handle_revoke_auth' ) ); add_action( 'wp_ajax_informatiq_sp_compare_products', array( $this, 'handle_compare_products' ) ); // Enqueue admin assets. add_action( 'admin_enqueue_scripts', array( $this, 'enqueue_admin_assets' ) ); } /** * Add admin menu under WooCommerce. */ public function add_admin_menu() { add_submenu_page( 'woocommerce', __( 'Smart Pricing', 'informatiq-smart-pricing' ), __( 'Smart Pricing', 'informatiq-smart-pricing' ), 'manage_woocommerce', 'informatiq-smart-pricing', array( $this, 'render_admin_page' ) ); } /** * Register plugin settings. */ public function register_settings() { // Register settings. register_setting( 'informatiq_sp_settings', 'informatiq_sp_merchant_id' ); register_setting( 'informatiq_sp_settings', 'informatiq_sp_client_id' ); register_setting( 'informatiq_sp_settings', 'informatiq_sp_client_secret' ); register_setting( 'informatiq_sp_settings', 'informatiq_sp_minimum_margin' ); register_setting( 'informatiq_sp_settings', 'informatiq_sp_auto_update_enabled' ); register_setting( 'informatiq_sp_settings', 'informatiq_sp_update_frequency' ); // Add settings sections. add_settings_section( 'informatiq_sp_google_settings', __( 'Google Merchant Center Settings', 'informatiq-smart-pricing' ), array( $this, 'render_google_settings_section' ), 'informatiq_sp_settings' ); add_settings_section( 'informatiq_sp_pricing_settings', __( 'Pricing Settings', 'informatiq-smart-pricing' ), array( $this, 'render_pricing_settings_section' ), 'informatiq_sp_settings' ); add_settings_section( 'informatiq_sp_automation_settings', __( 'Automation Settings', 'informatiq-smart-pricing' ), array( $this, 'render_automation_settings_section' ), 'informatiq_sp_settings' ); // Add settings fields. add_settings_field( 'informatiq_sp_merchant_id', __( 'Google Merchant ID', 'informatiq-smart-pricing' ), array( $this, 'render_merchant_id_field' ), 'informatiq_sp_settings', 'informatiq_sp_google_settings' ); add_settings_field( 'informatiq_sp_oauth_credentials', __( 'OAuth 2.0 Credentials', 'informatiq-smart-pricing' ), array( $this, 'render_oauth_credentials_field' ), 'informatiq_sp_settings', 'informatiq_sp_google_settings' ); add_settings_field( 'informatiq_sp_authorization', __( 'Google Authorization', 'informatiq-smart-pricing' ), array( $this, 'render_authorization_field' ), 'informatiq_sp_settings', 'informatiq_sp_google_settings' ); add_settings_field( 'informatiq_sp_minimum_margin', __( 'Minimum Margin (%)', 'informatiq-smart-pricing' ), array( $this, 'render_minimum_margin_field' ), 'informatiq_sp_settings', 'informatiq_sp_pricing_settings' ); add_settings_field( 'informatiq_sp_auto_update_enabled', __( 'Enable Automatic Updates', 'informatiq-smart-pricing' ), array( $this, 'render_auto_update_field' ), 'informatiq_sp_settings', 'informatiq_sp_automation_settings' ); add_settings_field( 'informatiq_sp_update_frequency', __( 'Update Frequency', 'informatiq-smart-pricing' ), array( $this, 'render_update_frequency_field' ), 'informatiq_sp_settings', 'informatiq_sp_automation_settings' ); } /** * Handle OAuth callback from Google. */ public function handle_oauth_callback() { // Check if this is an OAuth callback. if ( ! isset( $_GET['informatiq_sp_oauth'] ) || $_GET['informatiq_sp_oauth'] !== 'callback' ) { return; } // Verify state parameter. $state = isset( $_GET['state'] ) ? sanitize_text_field( $_GET['state'] ) : ''; $saved_state = get_transient( 'informatiq_sp_oauth_state' ); if ( empty( $state ) || $state !== $saved_state ) { add_settings_error( 'informatiq_sp_messages', 'informatiq_sp_oauth_error', __( 'OAuth state mismatch. Please try again.', 'informatiq-smart-pricing' ), 'error' ); return; } delete_transient( 'informatiq_sp_oauth_state' ); // Check for errors. if ( isset( $_GET['error'] ) ) { $error = sanitize_text_field( $_GET['error'] ); $error_desc = isset( $_GET['error_description'] ) ? sanitize_text_field( $_GET['error_description'] ) : $error; add_settings_error( 'informatiq_sp_messages', 'informatiq_sp_oauth_error', sprintf( __( 'Authorization failed: %s', 'informatiq-smart-pricing' ), $error_desc ), 'error' ); return; } // Get authorization code. $code = isset( $_GET['code'] ) ? sanitize_text_field( $_GET['code'] ) : ''; if ( empty( $code ) ) { add_settings_error( 'informatiq_sp_messages', 'informatiq_sp_oauth_error', __( 'No authorization code received.', 'informatiq-smart-pricing' ), 'error' ); return; } // Exchange code for tokens. $client_id = get_option( 'informatiq_sp_client_id' ); $client_secret = get_option( 'informatiq_sp_client_secret' ); $redirect_uri = $this->get_oauth_redirect_uri(); try { $tokens = Informatiq_SP_Google_API::exchange_code_for_tokens( $code, $client_id, $client_secret, $redirect_uri ); // Save refresh token. update_option( 'informatiq_sp_refresh_token', $tokens['refresh_token'] ); $this->logger->info( 'Google OAuth authorization successful' ); // Redirect to settings page with success message. wp_redirect( admin_url( 'admin.php?page=informatiq-smart-pricing&oauth=success' ) ); exit; } catch ( Exception $e ) { $this->logger->error( 'OAuth token exchange failed: ' . $e->getMessage() ); add_settings_error( 'informatiq_sp_messages', 'informatiq_sp_oauth_error', sprintf( __( 'Token exchange failed: %s', 'informatiq-smart-pricing' ), $e->getMessage() ), 'error' ); } } /** * Get OAuth redirect URI. * * @return string Redirect URI. */ private function get_oauth_redirect_uri() { return admin_url( 'admin.php?page=informatiq-smart-pricing&informatiq_sp_oauth=callback' ); } /** * Check if plugin is authorized. * * @return bool True if authorized. */ private function is_authorized() { return ! empty( get_option( 'informatiq_sp_refresh_token' ) ); } /** * Render admin page. */ public function render_admin_page() { if ( ! current_user_can( 'manage_woocommerce' ) ) { return; } // Check for OAuth success message. if ( isset( $_GET['oauth'] ) && $_GET['oauth'] === 'success' ) { add_settings_error( 'informatiq_sp_messages', 'informatiq_sp_message', __( 'Google authorization successful!', 'informatiq-smart-pricing' ), 'success' ); } // Check if form was submitted. if ( isset( $_GET['settings-updated'] ) ) { // Update cron schedule if frequency changed. $frequency = get_option( 'informatiq_sp_update_frequency', 'daily' ); Informatiq_SP_Scheduler::reschedule( $frequency ); add_settings_error( 'informatiq_sp_messages', 'informatiq_sp_message', __( 'Settings saved successfully.', 'informatiq-smart-pricing' ), 'success' ); } ?>


is_authorized() ) : ?>


render_schedule_info(); ?>
render_product_comparison(); ?>
render_logs_section(); ?>
render_sidebar(); ?>
' . esc_html__( 'Configure your Google Merchant Center API credentials using OAuth 2.0.', 'informatiq-smart-pricing' ) . '

'; } /** * Render pricing settings section description. */ public function render_pricing_settings_section() { echo '

' . esc_html__( 'Configure pricing rules and safeguards.', 'informatiq-smart-pricing' ) . '

'; } /** * Render automation settings section description. */ public function render_automation_settings_section() { echo '

' . esc_html__( 'Configure automated price updates.', 'informatiq-smart-pricing' ) . '

'; } /** * Render merchant ID field. */ public function render_merchant_id_field() { $value = get_option( 'informatiq_sp_merchant_id' ); ?>



' . esc_html( $this->get_oauth_redirect_uri() ) . '' ); ?>

is_authorized(); $has_credentials = ! empty( $client_id ) && ! empty( $client_secret ); if ( $is_authorized ) { ?>

get_oauth_redirect_uri(), $state ); ?>

%

is_authorized() ) : ?>

logger->get_todays_logs(); ?>

created_at ) ) ); ?> product_name ); ?> old_price ) ); ?> new_price ) ); ?> competitor_price ) ); ?>

informatiq.services', 'informatiq-smart-pricing' ) ), 'https://informatiq.services' ); ?>

__( 'Insufficient permissions', 'informatiq-smart-pricing' ) ) ); } if ( ! $this->is_authorized() ) { wp_send_json_error( array( 'message' => __( 'Please authorize with Google first.', 'informatiq-smart-pricing' ) ) ); } try { $scheduler = new Informatiq_SP_Scheduler( $this->price_updater ); $results = $scheduler->trigger_manual_update(); wp_send_json_success( array( 'message' => sprintf( /* translators: %1$d: updated count, %2$d: processed count */ __( 'Sync completed! Updated %1$d of %2$d products.', 'informatiq-smart-pricing' ), $results['updated'], $results['processed'] ), 'results' => $results, ) ); } catch ( Exception $e ) { wp_send_json_error( array( 'message' => $e->getMessage() ) ); } } /** * Handle test connection AJAX request. */ public function handle_test_connection() { check_ajax_referer( 'informatiq_sp_admin', 'nonce' ); if ( ! current_user_can( 'manage_woocommerce' ) ) { wp_send_json_error( array( 'message' => __( 'Insufficient permissions', 'informatiq-smart-pricing' ) ) ); } $merchant_id = get_option( 'informatiq_sp_merchant_id' ); $client_id = get_option( 'informatiq_sp_client_id' ); $client_secret = get_option( 'informatiq_sp_client_secret' ); $refresh_token = get_option( 'informatiq_sp_refresh_token' ); if ( empty( $merchant_id ) ) { wp_send_json_error( array( 'message' => __( 'Please enter your Merchant ID.', 'informatiq-smart-pricing' ) ) ); } if ( empty( $refresh_token ) ) { wp_send_json_error( array( 'message' => __( 'Please authorize with Google first.', 'informatiq-smart-pricing' ) ) ); } try { $google_api = new Informatiq_SP_Google_API( $merchant_id, $client_id, $client_secret, $refresh_token, $this->logger ); $success = $google_api->test_connection(); if ( $success ) { wp_send_json_success( array( 'message' => __( 'Connection successful!', 'informatiq-smart-pricing' ) ) ); } else { wp_send_json_error( array( 'message' => __( 'Connection failed. Check your credentials.', 'informatiq-smart-pricing' ) ) ); } } catch ( Exception $e ) { wp_send_json_error( array( 'message' => $e->getMessage() ) ); } } /** * Handle revoke authorization AJAX request. */ public function handle_revoke_auth() { check_ajax_referer( 'informatiq_sp_admin', 'nonce' ); if ( ! current_user_can( 'manage_woocommerce' ) ) { wp_send_json_error( array( 'message' => __( 'Insufficient permissions', 'informatiq-smart-pricing' ) ) ); } delete_option( 'informatiq_sp_refresh_token' ); $this->logger->info( 'Google OAuth authorization revoked' ); wp_send_json_success( array( 'message' => __( 'Authorization revoked.', 'informatiq-smart-pricing' ) ) ); } /** * Handle product comparison AJAX request. */ public function handle_compare_products() { check_ajax_referer( 'informatiq_sp_admin', 'nonce' ); if ( ! current_user_can( 'manage_woocommerce' ) ) { wp_send_json_error( array( 'message' => __( 'Insufficient permissions', 'informatiq-smart-pricing' ) ) ); } $merchant_id = get_option( 'informatiq_sp_merchant_id' ); $client_id = get_option( 'informatiq_sp_client_id' ); $client_secret = get_option( 'informatiq_sp_client_secret' ); $refresh_token = get_option( 'informatiq_sp_refresh_token' ); if ( empty( $merchant_id ) || empty( $refresh_token ) ) { wp_send_json_error( array( 'message' => __( 'Please configure settings and authorize first.', 'informatiq-smart-pricing' ) ) ); } try { $google_api = new Informatiq_SP_Google_API( $merchant_id, $client_id, $client_secret, $refresh_token, $this->logger ); // Get all Google products indexed by various identifiers. $google_products_raw = $google_api->get_all_products(); $google_products = array(); // Log first product structure for debugging. if ( ! empty( $google_products_raw[0] ) ) { $this->logger->info( 'Sample Google product structure: ' . wp_json_encode( array_keys( $google_products_raw[0] ) ) ); $this->logger->info( 'Sample Google product data: ' . wp_json_encode( $google_products_raw[0] ) ); } foreach ( $google_products_raw as $gp ) { // Index by offerId (top-level field). if ( ! empty( $gp['offerId'] ) ) { $google_products['offer_' . $gp['offerId']] = $gp; } // Index by GTIN - check attributes.gtin which is an ARRAY. if ( ! empty( $gp['attributes']['gtin'] ) && is_array( $gp['attributes']['gtin'] ) ) { foreach ( $gp['attributes']['gtin'] as $gtin_value ) { $google_products['gtin_' . $gtin_value] = $gp; } } // Also check attributes.gtins (alternative field name). if ( ! empty( $gp['attributes']['gtins'] ) && is_array( $gp['attributes']['gtins'] ) ) { foreach ( $gp['attributes']['gtins'] as $gtin_value ) { $google_products['gtin_' . $gtin_value] = $gp; } } // Index by MPN if available. if ( ! empty( $gp['attributes']['mpn'] ) ) { $google_products['mpn_' . $gp['attributes']['mpn']] = $gp; } } // Get WooCommerce in-stock products (limit to 50 for performance). $wc_products = wc_get_products( array( 'status' => 'publish', 'stock_status' => 'instock', 'limit' => 50, 'return' => 'objects', 'type' => array( 'simple', 'variable' ), ) ); $comparison = array(); foreach ( $wc_products as $product ) { $sku = $product->get_sku(); if ( empty( $sku ) ) { continue; } // Get local price (sale price priority, then regular). $sale_price = $product->get_sale_price(); $regular_price = $product->get_regular_price(); $local_price = ! empty( $sale_price ) ? $sale_price : $regular_price; $price_type = ! empty( $sale_price ) ? 'sale' : 'regular'; // Try to find matching Google product. $google_product = null; $match_type = ''; // Try offerId match. if ( isset( $google_products['offer_' . $sku] ) ) { $google_product = $google_products['offer_' . $sku]; $match_type = 'offerId'; } // Try GTIN match (when SKU is a barcode). elseif ( isset( $google_products['gtin_' . $sku] ) ) { $google_product = $google_products['gtin_' . $sku]; $match_type = 'gtin'; } // Get Google price (nested in attributes in new Merchant API). $google_price = null; if ( $google_product ) { if ( isset( $google_product['attributes']['price']['amountMicros'] ) ) { $google_price = (float) $google_product['attributes']['price']['amountMicros'] / 1000000; } elseif ( isset( $google_product['price']['amountMicros'] ) ) { $google_price = (float) $google_product['price']['amountMicros'] / 1000000; } } $comparison[] = array( 'id' => $product->get_id(), 'name' => $product->get_name(), 'sku' => $sku, 'local_price' => $local_price ? (float) $local_price : null, 'price_type' => $price_type, 'google_price' => $google_price, 'match_type' => $match_type, 'found' => ! empty( $google_product ), 'google_offer' => $google_product ? ( $google_product['offerId'] ?? '' ) : '', ); } // Get sample of indexed keys for debugging. $sample_keys = array_slice( array_keys( $google_products ), 0, 10 ); wp_send_json_success( array( 'products' => $comparison, 'google_count' => count( $google_products_raw ), 'wc_count' => count( $wc_products ), 'currency' => get_woocommerce_currency_symbol(), 'debug' => array( 'sample_google_keys' => $sample_keys, 'sample_google_product' => ! empty( $google_products_raw[0] ) ? $google_products_raw[0] : null, 'index_count' => count( $google_products ), ), ) ); } catch ( Exception $e ) { wp_send_json_error( array( 'message' => $e->getMessage() ) ); } } /** * Enqueue admin assets. * * @param string $hook Current admin page hook. */ public function enqueue_admin_assets( $hook ) { if ( 'woocommerce_page_informatiq-smart-pricing' !== $hook ) { return; } wp_enqueue_style( 'informatiq-sp-admin', INFORMATIQ_SP_PLUGIN_URL . 'assets/css/admin.css', array(), INFORMATIQ_SP_VERSION ); wp_enqueue_script( 'informatiq-sp-admin', INFORMATIQ_SP_PLUGIN_URL . 'assets/js/admin.js', array( 'jquery' ), INFORMATIQ_SP_VERSION, true ); wp_localize_script( 'informatiq-sp-admin', 'informatiqSP', array( 'ajaxUrl' => admin_url( 'admin-ajax.php' ), 'nonce' => wp_create_nonce( 'informatiq_sp_admin' ), 'strings' => array( 'syncInProgress' => __( 'Sync in progress...', 'informatiq-smart-pricing' ), 'testInProgress' => __( 'Testing connection...', 'informatiq-smart-pricing' ), 'revokeConfirm' => __( 'Are you sure you want to revoke Google authorization?', 'informatiq-smart-pricing' ), 'revokeInProgress' => __( 'Revoking...', 'informatiq-smart-pricing' ), ), ) ); } }