oont-contents/plugins/woocommerce-square/includes/API.php
2025-02-08 15:10:23 +01:00

896 lines
24 KiB
PHP

<?php
/**
* WooCommerce Square
*
* This source file is subject to the GNU General Public License v3.0
* that is bundled with this package in the file license.txt.
* It is also available through the world-wide-web at this URL:
* http://www.gnu.org/licenses/gpl-3.0.html GNU General Public License v3.0 or later
* If you did not receive a copy of the license and are unable to
* obtain it through the world-wide-web, please send an email
* to license@woocommerce.com so we can send you a copy immediately.
*
* DISCLAIMER
*
* Do not edit or add to this file if you wish to upgrade WooCommerce Square to newer
* versions in the future. If you wish to customize WooCommerce Square for your
* needs please refer to https://docs.woocommerce.com/document/woocommerce-square/
*
* @author WooCommerce
* @copyright Copyright: (c) 2019, Automattic, Inc.
* @license http://www.gnu.org/licenses/gpl-3.0.html GNU General Public License v3.0 or later
*/
namespace WooCommerce\Square;
use WooCommerce\Square\Framework\Api\Base;
use WooCommerce\Square\API\Requests;
use WooCommerce\Square\API\Responses;
use Square\SquareClient;
use Square\Environment;
defined( 'ABSPATH' ) || exit;
/**
* WooCommerce Square API class
*
* @since 2.0.0
*/
class API extends Base {
/** catalog request type */
const REQUEST_TYPE_CATALOG = 'catalog';
/** inventory request type */
const REQUEST_TYPE_INVENTORY = 'inventory';
/** tax type inclusive */
const TAX_TYPE_INCLUSIVE = 'INCLUSIVE';
/** tax type additive */
const TAX_TYPE_ADDITIVE = 'ADDITIVE';
/** @var \Square\SquareClient Square API client instance */
protected $client;
/**
* Constructs the main Square API wrapper class.
*
* @since 2.0.0
*
* @param string $access_token Square API access token
* @param bool $is_sandbox If sandbox access is desired
*/
public function __construct( $access_token, $is_sandbox = null ) {
$this->client = new SquareClient(
array(
'accessToken' => $access_token,
'environment' => $is_sandbox ? Environment::SANDBOX : Environment::PRODUCTION,
)
);
}
/** Catalog API Methods *******************************************************************************************/
/**
* Batch-deletes an array of catalog objects.
*
* @since 2.0.0
*
* @param string[] $object_ids array of square catalog object IDs
* @return Responses\Catalog
* @throws \Exception
*/
public function batch_delete_catalog_objects( array $object_ids ) {
$request = $this->get_catalog_request();
$request->set_batch_delete_catalog_objects_data( $object_ids );
return $this->perform_request( $request );
}
/**
* Batch-retrieves an array of catalog objects.
*
* @since 2.0.0
*
* @param string[] $object_ids array of square catalog object IDs
* @param bool $include_related_objects whether or not to include related objects in the response
* @return Responses\Catalog
* @throws \Exception
*/
public function batch_retrieve_catalog_objects( array $object_ids, $include_related_objects = false ) {
$request = $this->get_catalog_request();
$request->set_batch_retrieve_catalog_objects_data( $object_ids, (bool) $include_related_objects );
return $this->perform_request( $request );
}
/**
* Batch-upserts an array of catalog objects.
*
* @since 2.0.0
*
* @param string $idempotency_key a UUID for this request
* @param array $batches an array of batches to upsert
* @return Responses\Catalog
* @throws \Exception
*/
public function batch_upsert_catalog_objects( $idempotency_key, array $batches ) {
$request = $this->get_catalog_request();
$request->set_batch_upsert_catalog_objects_data( $idempotency_key, $batches );
return $this->perform_request( $request );
}
/**
* Returns info about the Catalog API, including helpful info like request size limits.
*
* @since 2.0.0
* @return Responses\Catalog
* @throws \Exception
*/
public function catalog_info() {
$request = $this->get_catalog_request();
$request->set_catalog_info_data();
return $this->perform_request( $request );
}
/**
* Deletes an object from the Square catalog.
*
* @since 2.0.0
*
* @param string $object_id Square catalog object ID
* @return Responses\Catalog
* @throws \Exception
*/
public function delete_catalog_object( $object_id ) {
$request = $this->get_catalog_request();
$request->set_delete_catalog_object_data( $object_id );
return $this->perform_request( $request );
}
/**
* Returns a list of Square catalog items.
*
* @since 2.0.0
*
* @param string $cursor the cursor to list from
* @param string[] $types the item types to filter by
* @return Responses\Catalog
* @throws \Exception
*/
public function list_catalog( $cursor = '', $types = array() ) {
$request = $this->get_catalog_request();
$request->set_list_catalog_data( $cursor, $types );
return $this->perform_request( $request );
}
/**
* Retrieves a single catalog object.
*
* @since 2.0.0
*
* @param string $object_id the Square catalog object ID
* @param bool $include_related_objects whether or not to include related objects (such as categories)
* @return Responses\Catalog
* @throws \Exception
*/
public function retrieve_catalog_object( $object_id, $include_related_objects = false ) {
$request = $this->get_catalog_request();
$request->set_retrieve_catalog_object_data( $object_id, $include_related_objects );
return $this->perform_request( $request );
}
/**
* Searches the catalog for objects.
*
* @since 2.0.0
*
* @param array $args see Catalog::set_search_catalog_objects_data() for list of args
* @return Responses\Catalog
* @throws \Exception
*/
public function search_catalog_objects( $args = array() ) {
$request = $this->get_catalog_request();
$request->set_search_catalog_objects_data( $args );
return $this->perform_request( $request );
}
/**
* Updates the modifier lists that apply to given items.
*
* @since 2.0.0
*
* @param string[] $item_ids array of Square catalog item IDs
* @param string[] $modifier_lists_to_enable (optional) modifier list IDs to enable
* @param string[] $modifier_lists_to_disable (optional) modifier list IDs to disable
* @return Responses\Catalog
* @throws \Exception
*/
public function update_item_modifier_lists( array $item_ids, array $modifier_lists_to_enable = array(), array $modifier_lists_to_disable = array() ) {
$request = $this->get_catalog_request();
$request->set_update_item_modifier_lists_data( $item_ids, $modifier_lists_to_enable, $modifier_lists_to_disable );
return $this->perform_request( $request );
}
/**
* Updates an item's applied taxes.
*
* @since 2.0.0
*
* @param string[] $item_ids array of Square catalog item IDs
* @param string[] $taxes_to_enable (optional) tax IDs to enable
* @param string[] $taxes_to_disable (optional) tax IDs to disable
* @return Responses\Catalog
* @throws \Exception
*/
public function update_item_taxes( array $item_ids, array $taxes_to_enable = array(), array $taxes_to_disable = array() ) {
$request = $this->get_catalog_request();
$request->set_update_item_taxes_data( $item_ids, $taxes_to_enable, $taxes_to_disable );
return $this->perform_request( $request );
}
/**
* Upserts an object into the catalog.
*
* @since 2.0.0
*
* @param string $idempotency_key UUID for this request
* @param \Square\Models\CatalogObject $object the object to upsert
* @return Responses\Catalog
* @throws \Exception
*/
public function upsert_catalog_object( $idempotency_key, $object ) {
$request = $this->get_catalog_request();
$request->set_upsert_catalog_object_data( $idempotency_key, $object );
return $this->perform_request( $request );
}
/**
* Creates an image in Square.
*
* Note that this method uses a custom request, since the Square SDK does not yet provide a method for image creation.
*
* @since 2.0.0
*
* @param $image_path
* @param string $square_item_id
* @param string $caption optional image caption
* @return string
* @throws \Exception
*/
public function create_image( $image_path, $square_item_id = '', $caption = '' ) {
if ( ! is_readable( $image_path ) ) {
throw new \Exception( 'Image file is not readable' );
}
$image = file_get_contents( $image_path );
$headers = array(
'accept' => 'application/json',
'content-type' => 'multipart/form-data; boundary="boundary"',
'Square-Version' => '2019-05-08',
'Authorization' => 'Bearer ' . wc_square()->get_settings_handler()->get_access_token(),
);
$body = '--boundary' . "\r\n";
$body .= 'Content-Disposition: form-data; name="request"' . "\r\n";
$body .= 'Content-Type: application/json' . "\r\n\r\n";
$request = array(
'idempotency_key' => wc_square()->get_idempotency_key(),
'image' => array(
'type' => 'IMAGE',
'id' => '#TEMP_ID',
'image_data' => array(
'caption' => esc_attr( $caption ),
),
),
);
if ( $square_item_id ) {
$request['object_id'] = $square_item_id;
}
$body .= json_encode( $request ); // phpcs:ignore WordPress.WP.AlternativeFunctions.json_encode_json_encode
$body .= "\r\n";
$body .= '--boundary' . "\r\n";
$body .= 'Content-Disposition: form-data; name="file"; filename="' . esc_attr( basename( $image_path ) ) . '"' . "\r\n";
$body .= 'Content-Type: image/jpeg' . "\r\n\r\n";
$body .= $image . "\r\n";
$body .= '--boundary--';
$url = $this->client->getBaseUri() . '/v2/catalog/images';
$response = wp_remote_post(
$url,
array(
'headers' => $headers,
'body' => $body,
)
);
if ( is_wp_error( $response ) ) {
throw new \Exception( esc_html( $response->get_error_message() ) );
}
$body = wp_remote_retrieve_body( $response );
$body = json_decode( $body, true );
if ( ! is_array( $body ) ) {
throw new \Exception( 'Response was malformed' );
}
if ( ! empty( $body['errors'] ) || empty( $body['image']['id'] ) ) {
if ( ! empty( $body['errors'][0]['detail'] ) ) {
$message = $body['errors'][0]['detail'];
} else {
$message = 'Unknown error';
}
throw new \Exception( esc_html( $message ) );
}
return $body['image']['id'];
}
/** Inventory API Methods *****************************************************************************************/
/**
* Adds a count of inventory as "in-stock" to the given Square item variation ID as a result of a refund.
*
* @since 2.0.0
*
* @param string $square_id Square object ID
* @param int $amount amount of inventory to add
* @return Responses\Inventory
* @throws \Exception
*/
public function add_inventory_from_refund( $square_id, $amount ) {
return $this->add_inventory( $square_id, $amount, 'NONE' );
}
/**
* Adds a count of inventory as "in-stock" to the given Square item variation ID.
*
* @since 2.0.0
*
* @param string $square_id Square object ID
* @param int $amount amount of inventory to add
* @param string $from_state the API state the inventory is coming from
* @return Responses\Inventory
* @throws \Exception
*/
public function add_inventory( $square_id, $amount, $from_state = 'NONE' ) {
return $this->adjust_inventory( $square_id, $amount, $from_state, 'IN_STOCK' );
}
/**
* Removes a count of inventory as "in-stock" to the given Square item variation ID.
*
* @since 2.0.0
*
* @param string $square_id Square object ID
* @param int $amount amount of inventory to remove
* @return Responses\Inventory
* @throws \Exception
*/
public function remove_inventory( $square_id, $amount ) {
return $this->adjust_inventory( $square_id, $amount, 'IN_STOCK', 'SOLD' );
}
/**
* Performs an inventory adjustment.
*
* @since 2.0.0
*
* @param string $square_id Square object ID
* @param int $amount amount of inventory to add
* @param string $from_state the API state the inventory is coming from
* @param string $to_state the API state the inventory is changing to
* @return Responses\Inventory
* @throws \Exception
*/
protected function adjust_inventory( $square_id, $amount, $from_state, $to_state ) {
$date = new \DateTime();
$change = new \Square\Models\InventoryChange();
$change->setType( 'ADJUSTMENT' );
$inventory_adjustment = new \Square\Models\InventoryAdjustment();
$inventory_adjustment->setCatalogObjectId( $square_id );
$inventory_adjustment->setLocationId( $this->get_plugin()->get_settings_handler()->get_location_id() );
$inventory_adjustment->setQuantity( (string) absint( $amount ) );
$inventory_adjustment->setFromState( $from_state );
$inventory_adjustment->setToState( $to_state );
$inventory_adjustment->setOccurredAt( $date->format( DATE_ATOM ) );
$change->setAdjustment( $inventory_adjustment );
return $this->batch_change_inventory(
uniqid( '', false ),
array(
$change,
)
);
}
/**
* Performs a Batch Change Inventory request.
*
* @since 2.0.0
*
* @param string $idempotency_key UUID for this request
* @param \Square\Models\InventoryChange[] $changes array of Inventory Changes
* @param bool $ignore_unchanged_counts whether the current physical count should be ignored if the quantity is unchanged since the last physical count
* @return Responses\Inventory
* @throws \Exception
*/
public function batch_change_inventory( $idempotency_key, $changes, $ignore_unchanged_counts = true ) {
$request = $this->get_inventory_request();
$request->set_batch_change_inventory_data( $idempotency_key, $changes, $ignore_unchanged_counts );
return $this->perform_request( $request );
}
/**
* Performs a Batch Retrieve Inventory Changes request.
*
* @since 2.0.0
*
* @param array $args see Requests\Inventory::set_batch_retrieve_inventory_changes_data() for accepted arguments
*
* @return Responses\Inventory
* @throws \Exception
*/
public function batch_retrieve_inventory_changes( array $args = array() ) {
$request = $this->get_inventory_request();
$request->set_batch_retrieve_inventory_changes_data( $args );
return $this->perform_request( $request );
}
/**
* Performs a Batch Retrieve Inventory Counts request.
*
* @since 2.0.0
*
* @param array $args see Requests\Inventory::set_batch_retrieve_inventory_counts_data() for accepted arguments
*
* @return Responses\Inventory
* @throws \Exception
*/
public function batch_retrieve_inventory_counts( array $args = array() ) {
$request = $this->get_inventory_request();
$request->set_batch_retrieve_inventory_counts_data( $args );
return $this->perform_request( $request );
}
/**
* Performs a Retrieve Inventory Adjustment request.
*
* @since 2.0.0
*
* @param string $adjustment_id the InventoryAdjustment ID to retrieve
*
* @return Responses\Inventory
* @throws \Exception
*/
public function retrieve_inventory_adjustment( $adjustment_id ) {
$request = $this->get_inventory_request();
$request->set_retrieve_inventory_adjustment_data( $adjustment_id );
return $this->perform_request( $request );
}
/**
* Performs a Retrieve Inventory Changes request.
*
* @since 2.0.0
*
* @param string $catalog_object_id the CatalogObject ID to retrieve
*
* @return Responses\Inventory
* @throws \Exception
*/
public function retrieve_inventory_changes( $catalog_object_id ) {
$request = $this->get_inventory_request();
$request->set_retrieve_inventory_changes_data( $catalog_object_id );
return $this->perform_request( $request );
}
/**
* Performs a Retrieve Inventory Count request.
*
* @since 2.0.0
*
* @param string $catalog_object_id the CatalogObject ID to retrieve
* @return Responses\Inventory
* @throws \Exception
*/
public function retrieve_inventory_count( $catalog_object_id ) {
$request = $this->get_inventory_request();
$request->set_retrieve_inventory_count_data( $catalog_object_id, $this->get_plugin()->get_settings_handler()->get_location_id() );
return $this->perform_request( $request );
}
/**
* Performs a Retrieve Inventory Physical Count request.
*
* @since 2.0.0
*
* @param string $physical_count_id the InventoryPhysicalCount ID to retrieve
*
* @return Responses\Inventory
* @throws \Exception
*/
public function retrieve_inventory_physical_count( $physical_count_id ) {
$request = $this->get_inventory_request();
$request->set_retrieve_inventory_physical_count_data( $physical_count_id );
return $this->perform_request( $request );
}
/** Locations methods *********************************************************************************************/
/**
* Gets the available locations.
*
* @since 2.0.0
*
* @return \Square\Models\Location[]
* @throws \Exception
*/
public function get_locations() {
$request = new API\Requests\Locations( $this->client );
$request->set_list_locations_data();
$this->set_response_handler( API\Responses\Locations::class );
/* @type API\Responses\Locations $response */
$response = $this->perform_request( $request );
return $response->get_locations();
}
/** Customer methods **********************************************************************************************/
/**
* Gets all customers.
*
* @since 2.0.0
*
* @param string $cursor pagination cursor
* @return API\Response
* @throws \Exception
*/
public function get_customers( $cursor = '' ) {
$request = new API\Requests\Customers( $this->client );
$request->set_get_customers_data( $cursor );
$this->set_response_handler( API\Response::class );
return $this->perform_request( $request );
}
/** Request Helper Methods ****************************************************************************************/
/**
* Gets a new Catalog API request.
*
* @since 2.0.0
*
* @return Requests\Catalog
* @throws \Exception
*/
protected function get_catalog_request() {
return $this->get_new_request( self::REQUEST_TYPE_CATALOG );
}
/**
* Gets a new Inventory API request.
*
* @since 2.0.0
*
* @return Requests\Inventory
* @throws \Exception
*/
protected function get_inventory_request() {
return $this->get_new_request( self::REQUEST_TYPE_INVENTORY );
}
/**
* Gets a new request object.
*
* @since 2.0.0
*
* @param string $type desired request type
* @return Requests\Catalog|Requests\Inventory
* @throws \Exception
*/
protected function get_new_request( $type = '' ) {
switch ( $type ) {
case self::REQUEST_TYPE_CATALOG:
$request = new Requests\Catalog( $this->client );
$response_handler = Responses\Catalog::class;
break;
case self::REQUEST_TYPE_INVENTORY:
$request = new Requests\Inventory( $this->client );
$response_handler = Responses\Inventory::class;
break;
default:
throw new \Exception( 'Invalid request type.' );
}
$this->set_response_handler( $response_handler );
return $request;
}
/**
* Performs an API request.
*
* @see Base::perform_request()
*
* @since 2.0.0
*
* @param API\Request $request request object
* @return API\Response
* @throws \Exception
*/
protected function perform_request( $request ) {
// ensure API is in its default state
$this->reset_response();
// save the request object
$this->request = $request;
$start_time = microtime( true );
try {
// set the request URI to the Square SDK method for better logging
$this->request_uri = $this->get_request()->get_square_api_method();
$this->request_method = '';
// add any query args to the logged request URI for easier debugging
foreach ( $this->get_request()->get_square_api_args() as $arg ) {
if ( is_string( $arg ) ) {
$this->request_uri .= "/{$arg}";
}
}
// perform the request
$response = $this->do_square_request( $this->get_request()->get_square_api(), $this->get_request()->get_square_api_method(), $this->get_request()->get_square_api_args() );
// calculate request duration
$this->request_duration = round( microtime( true ) - $start_time, 5 );
// parse & validate response
$response = $this->handle_response( $response );
} catch ( \Exception $e ) {
// alert other actors that a request has been made
$this->broadcast_request();
throw $e;
}
return $response;
}
/**
* Handles and parses the response.
*
* @since 2.0.0
*
* @param array|\WP_Error $response response data
* @throws \Exception
* @return API_Response|object request class instance that implements API_Request
*/
protected function handle_response( $response ) {
// parse the response body and tie it to the request
$this->response = $this->get_parsed_response( $this->raw_response_body );
// allow child classes to validate response after parsing -- this is useful
// for checking error codes/messages included in a parsed response
$this->do_post_parse_response_validation();
// fire do_action() so other actors can act on request/response data,
// primarily used for logging
$this->broadcast_request();
return $this->response;
}
/**
* Validates the response data after it's been parsed.
*
* @since 2.0.0
*
* @return bool
* @throws \Exception
*/
protected function do_post_parse_response_validation() {
if ( ! $this->get_response()->has_errors() ) {
return true;
}
$errors = array();
/** @var \Square\Models\Error $error */
foreach ( $this->get_response()->get_errors() as $error ) {
$error_code = $error->getCode();
if ( empty( $error_code ) ) {
continue;
}
$errors[] = trim( "[{$error_code}] {$error->getDetail()}" );
// Last attempt to refresh access token.
if ( in_array( $error_code, array( 'ACCESS_TOKEN_EXPIRED', 'UNAUTHORIZED' ), true ) ) {
if ( 'ACCESS_TOKEN_EXPIRED' === $error_code ) {
$this->get_plugin()->log( 'Access Token Expired, attempting a refresh.' );
} else {
$this->get_plugin()->log( 'Authorization error occurred, attempting a refresh.' );
}
$this->get_plugin()->get_connection_handler()->refresh_connection();
$failure_value = get_option( 'wc_square_refresh_failed', 'yes' );
if ( empty( $failure_value ) ) {
// Successfully refreshed on the last attempt
$this->get_plugin()->log( 'Connection successfully refreshed.' );
return true;
}
}
// if the error indicates that access token is bad, disconnect the plugin to prevent further attempts
if ( in_array( $error_code, array( 'ACCESS_TOKEN_EXPIRED', 'ACCESS_TOKEN_REVOKED', 'UNAUTHORIZED' ), true ) ) {
$this->get_plugin()->get_connection_handler()->disconnect();
$this->get_plugin()->log( 'Disconnected due to invalid authorization. Please try connecting again.' );
}
}
// At this point we could not validate the response and assume a failed attempt.
throw new \Exception( esc_html( implode( ' | ', $errors ) ) );
}
/**
* Performs a remote request with the Square API class.
*
* @since 2.0.0
*
* @param Object $square_api the square API class instance
* @param string $method the class method to call
* @param array $args the args to send with the method call
* @throws \Exception
*/
protected function do_square_request( $square_api, $method, $args ) {
if ( ! is_callable( array( $square_api, $method ) ) ) {
throw new \Exception( 'Invalid API method' );
}
// perform the request
$response = call_user_func_array( array( $square_api, $method ), $args );
if ( $response instanceof \Square\Http\ApiResponse ) {
$this->response_code = $response->getStatusCode();
$this->response_headers = $response->getHeaders();
if ( $response->isSuccess() ) {
$this->raw_response_body = $response->getResult();
} else {
$this->raw_response_body = $response->getErrors();
}
}
}
/**
* Gets the main plugin instance.
*
* @since 2.0.0
*
* @return \WooCommerce\Square\Plugin
*/
public function get_plugin() {
return wc_square();
}
}