File: /var/www/html/wp-content/plugins/woocommerce/src/Internal/Fulfillments/FulfillmentsManager.php
<?php
/**
* WooCommerce Fulfillment Hooks
*/
declare( strict_types=1 );
namespace Automattic\WooCommerce\Internal\Fulfillments;
use Automattic\WooCommerce\Internal\DataStores\Fulfillments\FulfillmentsDataStore;
use Automattic\WooCommerce\Internal\Fulfillments\Providers\AbstractShippingProvider;
use WC_Order;
use WC_Order_Refund;
/**
* FulfillmentsManager class.
*
* This class is responsible for adding hooks related to fulfillments in WooCommerce.
*
* @since 10.1.0
* @package WooCommerce\Internal\Fulfillments
*/
class FulfillmentsManager {
/**
* This method registers the hooks related to fulfillments.
*/
public function register() {
add_filter( 'woocommerce_fulfillment_shipping_providers', array( $this, 'get_initial_shipping_providers' ), 10, 1 );
add_filter( 'woocommerce_fulfillment_translate_meta_key', array( $this, 'translate_fulfillment_meta_key' ), 10, 1 );
add_filter( 'woocommerce_fulfillment_parse_tracking_number', array( $this, 'try_parse_tracking_number' ), 10, 3 );
$this->init_fulfillment_status_hooks();
$this->init_refund_hooks();
}
/**
* Hook fulfillment status events.
*
* This method hooks into the fulfillment status events to update the order fulfillment status
* when a fulfillment is created, updated, or deleted.
*/
private function init_fulfillment_status_hooks() {
// Update order fulfillment status when a fulfillment is created, updated, or deleted.
add_action( 'woocommerce_fulfillment_after_create', array( $this, 'update_order_fulfillment_status_on_fulfillment_update' ), 10, 1 );
add_action( 'woocommerce_fulfillment_after_update', array( $this, 'update_order_fulfillment_status_on_fulfillment_update' ), 10, 1 );
add_action( 'woocommerce_fulfillment_after_delete', array( $this, 'update_order_fulfillment_status_on_fulfillment_update' ), 10, 1 );
}
/**
* Initialize refund-related hooks.
*
* This method initializes the hooks related to refunds, such as updating fulfillments after a refund is created
*/
private function init_refund_hooks() {
add_action( 'woocommerce_refund_created', array( $this, 'update_fulfillments_after_refund' ), 10, 1 );
add_action( 'woocommerce_delete_order_refund', array( $this, 'update_fulfillment_status_after_refund_deleted' ), 10, 1 );
}
/**
* Translate fulfillment meta keys.
*
* @param string $meta_key The meta key to translate.
* @return string Translated meta key.
*/
public function translate_fulfillment_meta_key( $meta_key ) {
/**
* Filter to translate fulfillment meta keys.
*
* This filter allows us to translate fulfillment meta keys
* to make them more user-friendly in the admin interface and emails.
*
* @since 10.1.0
*/
$meta_key_translations = apply_filters(
'woocommerce_fulfillment_meta_key_translations',
array(
'fulfillment_status' => __( 'Fulfillment Status', 'woocommerce' ),
'shipment_tracking' => __( 'Shipment Tracking', 'woocommerce' ),
'shipment_provider' => __( 'Shipment Provider', 'woocommerce' ),
)
);
return isset( $meta_key_translations[ $meta_key ] ) ? $meta_key_translations[ $meta_key ] : $meta_key;
}
/**
* Get initial shipping providers.
*
* This method provides the initial shipping providers that feeds the `woocommerce_fulfillment_shipping_providers` filter,
* which is used to populate the list of available shipping providers on the fulfillment UI.
*
* @param array $shipping_providers The current list of shipping providers.
*
* @return array The modified list of shipping providers.
*/
public function get_initial_shipping_providers( $shipping_providers ) {
if ( ! is_array( $shipping_providers ) ) {
$shipping_providers = array();
}
$shipping_providers = array_merge(
$shipping_providers,
include __DIR__ . '/ShippingProviders.php'
);
ksort( $shipping_providers );
return $shipping_providers;
}
/**
* Update order fulfillment status after a fulfillment is created, updated, or deleted.
*
* @param Fulfillment $data The fulfillment data.
*/
public function update_order_fulfillment_status_on_fulfillment_update( Fulfillment $data ) {
if ( ! $data instanceof Fulfillment ) {
return;
}
$order = $data->get_order();
if ( ! $order instanceof \WC_Order ) {
return;
}
/**
* Get the FulfillmentsDataStore instance.
*
* @var FulfillmentsDataStore $fulfillments_data_store
*/
$fulfillments_data_store = wc_get_container()->get( FulfillmentsDataStore::class );
// Read all fulfillments for the order.
$fulfillments = $fulfillments_data_store->read_fulfillments( \WC_Order::class, (string) $order->get_id() );
$this->update_fulfillment_status( $order, $fulfillments );
}
/**
* Update fulfillment status after a refund is deleted.
*
* This method updates the fulfillment status after a refund is deleted to ensure that the fulfillment status
* and items are correctly adjusted based on the refund deletion.
*
* @param int $refund_id The ID of the refund being deleted.
*
* @return void
*/
public function update_fulfillment_status_after_refund_deleted( int $refund_id ): void {
$refund = wc_get_order( $refund_id );
if ( ! $refund instanceof \WC_Order ) {
return; // If the refund is not a valid order, do nothing.
}
$order_id = $refund->get_parent_id();
if ( ! $order_id ) {
return; // If the refund does not have a parent order, do nothing.
}
$order = wc_get_order( $order_id );
if ( ! $order instanceof \WC_Order ) {
return; // If the order is not valid, do nothing.
}
$fulfillments_data_store = wc_get_container()->get( FulfillmentsDataStore::class );
$fulfillments = $fulfillments_data_store->read_fulfillments( \WC_Order::class, (string) $order_id );
$this->update_fulfillment_status( $order, $fulfillments );
}
/**
* Update fulfillments after a refund is created.
*
* @param int $refund_id The ID of the refund created.
*
* @return void
*/
public function update_fulfillments_after_refund( int $refund_id ): void {
// Get the order object.
$refund = $refund_id ? wc_get_order( $refund_id ) : null;
if ( ! $refund instanceof WC_Order_Refund ) {
return; // If the order is not valid, do nothing.
}
$order_id = $refund->get_parent_id();
if ( ! $order_id ) {
return; // If the refund does not have a parent order, do nothing.
}
$order = wc_get_order( $order_id );
if ( ! $order instanceof \WC_Order ) {
return; // If the order is not valid, do nothing.
}
// If there are no refunded items, we can skip the fulfillment update.
$items_refunded = FulfillmentUtils::get_refunded_items( $order );
if ( empty( $items_refunded ) ) {
return; // No items were refunded, so no need to update fulfillments.
}
// Get the fulfillments data store and read all fulfillments for the order.
$fulfillments_data_store = wc_get_container()->get( FulfillmentsDataStore::class );
$fulfillments = $fulfillments_data_store->read_fulfillments( \WC_Order::class, (string) $order_id );
if ( empty( $fulfillments ) ) {
return; // No fulfillments found for the order.
}
// Get all refunded items from the order.
$pending_items_without_refunds = FulfillmentUtils::get_pending_items( $order, $fulfillments, false );
$pending_items_without_refunds = array_map(
function ( $item ) {
return array(
'item_id' => $item['item_id'],
'qty' => $item['qty'],
);
},
$pending_items_without_refunds
);
// Check if the refunded items can be removed from pending items.
foreach ( $items_refunded as $item_id => &$refunded_qty ) {
$pending_item_record = array_filter(
$pending_items_without_refunds,
function ( $item ) use ( $item_id ) {
return isset( $item['item_id'] ) && $item['item_id'] === $item_id;
}
);
if ( ! empty( $pending_item_record ) ) {
$pending_item_record = reset( $pending_item_record );
if ( isset( $pending_item_record['qty'] ) && $pending_item_record['qty'] > 0 ) {
// If the pending item quantity is greater than the refunded quantity, reduce it.
$refunded_qty -= $pending_item_record['qty'];
}
}
}
// If all refunded items can be removed from pending items, we can skip the fulfillment update.
$items_need_removal_from_fulfillments = array_filter(
$items_refunded,
function ( $actual_qty ) {
return $actual_qty > 0;
}
);
if ( empty( $items_need_removal_from_fulfillments ) ) {
return;
}
// Now we need to adjust the fulfillments based on the refunded items.
// Loop through each fulfillment and adjust the items based on the refunded quantities.
// We will remove items from fulfillments if they are fully refunded, or reduce their quantity if partially refunded.
// If a fulfillment has no items left after adjustment, we will delete it.
// If a fulfillment has items left, we will update the fulfillment with the new items.
foreach ( $fulfillments as $fulfillment ) {
if ( ! $fulfillment instanceof Fulfillment ) {
continue; // Skip if the fulfillment is not an instance of Fulfillment.
}
if ( $fulfillment->get_is_fulfilled() ) {
continue; // Skip if the fulfillment is already fulfilled. We don't remove items from fulfilled fulfillments.
}
// Get the items from the fulfillment.
$items = $fulfillment->get_items();
if ( empty( $items ) ) {
continue; // Skip if there are no items in the fulfillment.
}
// Adjust the items based on the refund.
$new_items = array();
foreach ( $items as $item ) {
if ( isset( $item['qty'] ) && isset( $item['item_id'] ) && isset( $items_need_removal_from_fulfillments[ $item['item_id'] ] ) ) {
if ( $items_need_removal_from_fulfillments[ $item['item_id'] ] <= $item['qty'] ) {
// If the refunded quantity is less than or equal to the item quantity, reduce the item quantity.
$item['qty'] -= $items_need_removal_from_fulfillments[ $item['item_id'] ];
$items_need_removal_from_fulfillments[ $item['item_id'] ] = 0; // Set refunded quantity to zero after adjustment.
} else {
// If the refunded quantity is greater than the item quantity, set the item quantity to zero.
$item['qty'] = 0;
$items_need_removal_from_fulfillments[ $item['item_id'] ] -= $item['qty']; // Reduce the refunded quantity.
}
$new_items[] = $item; // Add the adjusted item to the new items array.
} else {
$new_items[] = $item; // If the item is not in the refunded items, keep it as is.
}
}
$new_items = array_filter(
$new_items,
function ( $item ) {
return isset( $item['qty'] ) && $item['qty'] > 0; // Only keep items with a positive quantity.
}
);
if ( empty( $new_items ) ) {
// If no items remain after adjustment, delete the fulfillment.
$fulfillment->delete();
} else {
// Update the fulfillment items with the new items.
$fulfillment->set_items( $new_items );
$fulfillment->save();
}
}
$this->update_fulfillment_status( $order, $fulfillments );
}
/**
* Update the fulfillment status for the order.
*
* @param \WC_Order $order The order object.
* @param array $fulfillments The fulfillments data store.
*
* This method updates the fulfillment status for the order based on the fulfillments data store.
*/
private function update_fulfillment_status( $order, $fulfillments = array() ) {
$last_status = FulfillmentUtils::calculate_order_fulfillment_status( $order, $fulfillments );
if ( 'no_fulfillments' === $last_status ) {
$order->delete_meta_data( '_fulfillment_status' );
} else {
// Update the fulfillment status meta data.
$order->update_meta_data( '_fulfillment_status', $last_status );
}
$order->save();
}
/**
* Try to parse the tracking number with additional parameters.
*
* @param string $tracking_number The tracking number.
* @param string $shipping_from The country code from which the shipment is sent.
* @param string $shipping_to The country code to which the shipment is sent.
*
* @return array An array containing the provider as key, and the parsing results.
*/
public function try_parse_tracking_number( string $tracking_number, string $shipping_from, string $shipping_to ): array {
// Validate the tracking number format and length.
if ( ! is_string( $tracking_number ) || empty( $tracking_number ) || strlen( $tracking_number ) > 50 ) {
$tracking_number = is_string( $tracking_number ) && ! empty( $tracking_number ) ? substr( $tracking_number, 0, 50 ) : '';
return array(
'tracking_number' => $tracking_number,
'shipping_provider' => '',
'tracking_url' => '',
);
}
// Normalize the tracking number to uppercase.
$tracking_number = strtoupper( $tracking_number );
$tracking_number = preg_replace( '/[^A-Z0-9]/', '', $tracking_number ); // Remove non-alphanumeric characters.
$shipping_providers = FulfillmentUtils::get_shipping_providers();
$results = array();
foreach ( $shipping_providers as $provider ) {
if ( class_exists( $provider ) && is_subclass_of( $provider, AbstractShippingProvider::class ) ) {
try {
/**
* Instantiate the shipping provider class.
*
* @var AbstractShippingProvider $provider_instance
*/
$provider_instance = wc_get_container()->get( $provider );
} catch ( \Throwable $e ) {
$logger = wc_get_logger();
$logger->error(
sprintf(
'Error instantiating shipping provider class %s: %s',
$provider,
$e->getMessage()
),
array( 'source' => 'woocommerce-fulfillments' )
);
continue; // Skip if the provider class cannot be instantiated.
}
} else {
continue; // Skip if the provider class does not exist or is not a valid shipping provider.
}
$parsing_result = $provider_instance->try_parse_tracking_number( $tracking_number, $shipping_from, $shipping_to );
if ( ! is_null( $parsing_result ) ) {
$results[ $provider_instance->get_key() ] = $parsing_result;
}
}
if ( 1 === count( $results ) ) {
$result = reset( $results );
$key = key( $results );
$results = array(
'tracking_number' => $tracking_number,
'shipping_provider' => $key,
'tracking_url' => $result['url'] ?? '',
);
} elseif ( 1 < count( $results ) ) {
// If multiple providers could parse the tracking number, find the one with the highest ambiguity score.
$possibilities = $results;
$results = $this->get_best_parsing_result( $results, $tracking_number );
$results['possibilities'] = $possibilities; // Include all possibilities for reference.
}
return $results;
}
/**
* Get the best parsing result from multiple results.
*
* This method finds the provider with the highest ambiguity score from the results.
*
* @param array $results The results from multiple providers.
* @param string $tracking_number The tracking number being parsed.
*
* @return array The best parsing result.
*/
private function get_best_parsing_result( array $results, string $tracking_number ): array {
$best_result = null;
$best_provider = '';
$best_score = 0;
foreach ( $results as $provider_key => $result ) {
if ( ! isset( $result['ambiguity_score'] ) || ! is_numeric( $result['ambiguity_score'] ) ) {
continue; // Skip if ambiguity score is not set or not numeric.
}
if ( is_null( $best_result ) || $result['ambiguity_score'] > $best_score ) {
$best_result = $result;
$best_provider = $provider_key;
$best_score = $result['ambiguity_score'];
}
}
return is_null( $best_result ) ? array() : array(
'tracking_number' => $tracking_number,
'shipping_provider' => $best_provider,
'tracking_url' => $best_result['url'],
);
}
}