Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Multiple Variations Support #235

Draft
wants to merge 113 commits into
base: trunk
Choose a base branch
from
Draft

Conversation

faisal-alvi
Copy link
Member

@faisal-alvi faisal-alvi commented Nov 13, 2024

All Submissions:

  • Does your code follow the WooCommerce Sniffs variant of WordPress coding standards?
  • Have you written new tests for your changes, as applicable?
  • Have you successfully run tests with your changes locally?
  • Will this change require new documentation or changes to existing documentation?

Changes proposed in this Pull Request:

This PR adds support for Multiple Variations.

Square and WooCommerce Attribute Types

  • Square options:

    • Dynamic Options: Shared across multiple items, similar to WooCommerce’s taxonomy attributes; variation names cannot be changed.
    • Static Options: Unique to individual items; variation names can be changed.
  • WooCommerce attributes:

    • Taxonomy: Defined globally across products.
    • Custom: Defined on a per-product basis.

Key Differences Between Dynamic and Static Options

Dynamic options can be shared across products and mirror WooCommerce’s taxonomy attributes, and variation names cannot be modified, as they are automatically generated by combining the attribute values. For example: “Red, Small.” Static options are unique to each item. If multiple options exist at Square, they must be dynamic.

Syncing Logic (Overview)

  1. WooCommerce to Square Sync:

    • Creates dynamic options if there are multiple WooCommerce attributes, regardless of the WooCommerce's attributes type.
    • For a single attribute, it creates a static option at Square.
  2. Square to WooCommerce Sync/Import:

    • Creates taxonomy attributes if the option slug matches an existing WooCommerce taxonomy attribute.
    • If only one option is received, it checks for an existing WooCommerce taxonomy attribute or creates a custom attribute if no match is found.

Syncing Logic for WooCommerce and Square (Detail)

From WooCommerce to Square:

  • If WooCommerce has more than one attribute and is set as the Source of Record (SOR), this PR will create dynamic options at Square, regardless of whether the attributes are taxonomy or custom. This is because we cannot create more than one static options for the single item (i.e product) at Square.
  • If there’s only one attribute, it creates a static option at Square.
  • If an attribute requires syncing, the PR creates an option at Square. New attribute values are also created at Square when synced from WooCommerce.

From Square to WooCommerce:

  • If the option name received from Square matches the slug of an existing WooCommerce taxonomy attribute, this PR creates a taxonomy attribute. If the merchant updates the slug in WooCommerce, it may not match next time and a custom attribute will be created instead.
  • If only one option is received, this PR checks against existing WooCommerce taxonomy attributes and attaches it if matched; otherwise, it creates a custom attribute at the product level. When creating a custom attribute, the PR reuses an existing custom attribute name, as Square does not provide the static option name. If no matching name exists, "Attribute" is assigned as the custom name.

Square API Limitations and Solutions

Square’s API only provides the option ID used in variations, not the display name. However, to match WooCommerce taxonomy attributes or to create a custom attribute, we need the option's display name. To address this, we fetch all options via another Square API (discussion link) and store the option names and their IDs in a transient (wc_square_options_data) with a 1-day expiration. The fetch_options_data() function initiates this before any import or sync process.

If a transient doesn’t have updated data, a sync may fail when creating an option with a name that already exists at Square, which would require a latest options data. This PR then stops the sync and sets the woocommerce_square_refresh_sync_cycle option to refresh the data.

Avoiding Attribute Name Conflicts

If WooCommerce’s taxonomy attribute is used, this PR passes the taxonomy name (e.g., “pa_color” rather than “Color”) to Square, preventing conflicts with custom attributes of the same name. The “pa_” prefix helps WooCommerce users distinguish between taxonomy and custom attributes, avoiding potential conflicts.

New Option for Tracking

This PR introduces a new _dynamic_options option with true/false values to track whether dynamic options are used for a product. This is not in use but may be useful in future.

Notes for Merchants

  • Avoid leaving any variation dropdowns set to “Any” in products with multiple attributes, as Square does not support empty values for dynamic options. <- this one is also handled in this PR so that plugin creates "Any" option item at Square and maintains to set "Any" (means no selection) at WooCommerce and vice versa. It was decided on slack here.
  • Avoid changing taxonomy attribute slugs or attribute names in WooCommerce, or renaming dynamic options in Square.
  • Avoid creating custom attributes with names identical to existing taxonomy attributes, it results in issues during import/sync. Products using the same-named custom attribute are updated (replaced) incorrectly with the taxonomy attributes.
  • Do not create custom attributes with names identical to existing taxonomy attributes.

Post-Tasks

  • #251 - Display Notices for Sync Limitations.
  • #252 - Importing/Syncing from Square to Woocommerce, all the dynamic option values are added to the attribute, for example, for colour attribute, if there is only red and green options are selected at Square, after the import/sync, it adds all of the dynamic option values to the Woocommerce attribute regardless of what selection. This is not creating any issue to the purchases of the side, but this should be addressed.
  • #254 - If merchant changes the name of the attribute, this will create a new option/attribute instead of using the previous one. Think of a way to handle this.

Closes #6, Closes #31 and Closes #231.

Steps to test the changes in this Pull Request:

  1. Set WooCommerce as SOR.
  2. Create the following product types in WooCommerce:
    • Simple Product
    • Variable Product with:
      • 1 Custom Attribute
      • 1 Taxonomy Attribute
      • 2 Taxonomy Attributes
      • 2 Custom Attributes
      • 1 Custom + 1 Taxonomy Attribute
      • 2 Taxonomy Attributes + 1 Custom Attribute
  3. Confirm these products are auto-synced and correctly created in Square.
  4. Confirm the same for manual sync using the “Sync” button in settings.
  5. Import all products from Square to WooCommerce and verify they retain attributes without issues (e.g., taxonomy attribute not converted to custom).
  6. Make changes to the random products and confirm it syncs to the Square and Imports back to the Woo without and issues, few examples:
    1. add one more custom attribute to the “2 Taxonomy Attributes” product
    2. remove 1 taxonomy attribute from the “2 Taxonomy Attributes + 1 Custom Attribute” product
    3. add one more variation to any product
    4. remove one variation from any product
    5. change the dropdown of any variation in “1 Custom Attribute” product
    6. change the dropdown of all variations in “2 Taxonomy Attributes + 1 Custom Attribute” product
    7. add one more attribute value and create update the dropdown values of each variation
    8. etc.
  7. Set Square as SOR and perform all of the above steps (sorry :)) This time the chagnes from the point 6 should be made at Square.
  8. Important: Confirm no issues affect existing users by creating all product types in trunk and performing sync/import in the fix branch.

Changelog entry

Add - Multiple Variations Support

@faisal-alvi
Copy link
Member Author

@dkotter over to you for code review.

@vikrampm1
Copy link
Contributor

@qasumitbagthariya bumping this up in your queue to try and wrap it up asap.

@qasumitbagthariya
Copy link
Contributor

QA Update ✅


I have verified this PR in the multiple-variations-support branch which has been fixed and is functioning as intended.

I tested the following on this branch:
Verified all the cases related to variations and the steps provided in the PR description

Square SOR

Square_SOR.mov

Woocommerce SOR

Woo-SOR.mov

Testing Environment

  • WordPress: 6.7.1
  • Theme: Twenty Twenty-Four 1.3
  • WooCommerce - 9.5.1
  • PHP: 8.0.30
  • Web Server: Nginx 1.20.2
  • Browser: Chrome
  • OS: macOS 15.2
  • Branch: multiple-variations-support

Steps to Test- As mentioned in the PR description.
Test Results - It is working as expected.
Functional Demo / Screencast -
Special Notes - Ready for code review (Woo)
Testing Document status:
Cases related to this Issue/PR are added to the Critical Flow Wiki pages:

  • Yes
  • Not Required/Applicable for this PR

@qasumitbagthariya qasumitbagthariya requested review from a team and wjrosa and removed request for a team January 23, 2025 14:38
$objects = $response->get_data()->getObjects() ? $response->get_data()->getObjects() : array();

foreach ( $objects as $object ) {
$options_data[ $object->getId() ]['name'] = $object->getItemOptionData()->getDisplayName();
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Can any of the getters return null values? If so, it would be nice to add some checks to avoid errors.

$attributes = $product->get_attributes();

$product_variation_ids = $product->get_children();
$catalog_variations = $item_data->getVariations() ? $item_data->getVariations() : array();
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Since PHP 7.0 we can change this to:

Suggested change
$catalog_variations = $item_data->getVariations() ? $item_data->getVariations() : array();
$catalog_variations = $item_data->getVariations() ?? array();

) {
$options_ids = array();
$result = wc_square()->get_api()->retrieve_options_data();
$options_data = isset( $result[1] ) ? $result[1] : array();
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Same as above:

Suggested change
$options_data = isset( $result[1] ) ? $result[1] : array();
$options_data = $result[1] ?? array();

$variation_items = $product->get_attributes();
$variation_item_values = array();

if ( 1 === count( $attributes ) ) {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I see that these changes contain multiple "actions" that could be new methods to make reading easier. Basically, every time we have a comment here, the following code could be a new method named after the explanation. But this is optional

$new_cursor = isset( $result[2] ) ? $result[2] : null;

if ( ! empty( $new_cursor ) ) {
$this->set_attr( 'fetch_options_data_cursor', $new_cursor );
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I believe this line can be moved outside the condition, as $new_cursor will be null per the default value above anyway.

@@ -1056,6 +1297,8 @@ protected function save_variations( $product_id, $data ) {
$updated_attribute_keys = array();

foreach ( $variation['attributes'] as $attribute_key => $attribute ) {
// Set empty if attribute is to 'Any' to prevent it from being saved.
$attribute['option'] = isset( $attribute['option'] ) && 'Any' !== $attribute['option'] ? $attribute['option'] : '';
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Maybe we should move Any to a new constant.

Copy link
Contributor

@wjrosa wjrosa left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Looks good! Some multiple optional suggestions 👍

@vikrampm1
Copy link
Contributor

Thanks @wjrosa. @faisal-alvi see if you can make those optional suggested changes above and then we can move this to the next step.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
status: e2e tests passing type: enhancement The issue is a request for an enhancement.
Projects
None yet
6 participants