Complex classes like ProductContext often do a lot of different things. To break such a class down, we need to identify a cohesive component within that class. A common approach to find such a component is to look for fields/methods that share the same prefixes, or suffixes. You can also have a look at the cohesion graph to spot any un-connected, or weakly-connected components.
Once you have determined the fields that belong together, you can apply the Extract Class refactoring. If the component makes sense as a sub-class, Extract Subclass is also a candidate, and is often faster.
While breaking up the class, it is a good idea to analyze how other classes use ProductContext, and based on these observations, apply Extract Interface, too.
| 1 | <?php | ||
| 44 | final class ProductContext implements Context | ||
| 45 | { | ||
| 46 | /** @var SharedStorageInterface */ | ||
| 47 | private $sharedStorage; | ||
| 48 | |||
| 49 | /** @var ProductRepositoryInterface */ | ||
| 50 | private $productRepository; | ||
| 51 | |||
| 52 | /** @var ProductFactoryInterface */ | ||
| 53 | private $productFactory; | ||
| 54 | |||
| 55 | /** @var FactoryInterface */ | ||
| 56 | private $productTranslationFactory; | ||
| 57 | |||
| 58 | /** @var FactoryInterface */ | ||
| 59 | private $productVariantFactory; | ||
| 60 | |||
| 61 | /** @var FactoryInterface */ | ||
| 62 | private $productVariantTranslationFactory; | ||
| 63 | |||
| 64 | /** @var FactoryInterface */ | ||
| 65 | private $channelPricingFactory; | ||
| 66 | |||
| 67 | /** @var FactoryInterface */ | ||
| 68 | private $productOptionFactory; | ||
| 69 | |||
| 70 | /** @var FactoryInterface */ | ||
| 71 | private $productOptionValueFactory; | ||
| 72 | |||
| 73 | /** @var FactoryInterface */ | ||
| 74 | private $productImageFactory; | ||
| 75 | |||
| 76 | /** @var ObjectManager */ | ||
| 77 | private $objectManager; | ||
| 78 | |||
| 79 | /** @var ProductVariantGeneratorInterface */ | ||
| 80 | private $productVariantGenerator; | ||
| 81 | |||
| 82 | /** @var ProductVariantResolverInterface */ | ||
| 83 | private $defaultVariantResolver; | ||
| 84 | |||
| 85 | /** @var ImageUploaderInterface */ | ||
| 86 | private $imageUploader; | ||
| 87 | |||
| 88 | /** @var SlugGeneratorInterface */ | ||
| 89 | private $slugGenerator; | ||
| 90 | |||
| 91 | /** @var array */ | ||
| 92 | private $minkParameters; | ||
| 93 | |||
| 94 | public function __construct( | ||
| 137 | |||
| 138 | /** | ||
| 139 | * @Given the store has a product :productName | ||
| 140 | * @Given the store has a :productName product | ||
| 141 | * @Given I added a product :productName | ||
| 142 |      * @Given /^the store(?:| also) has a product "([^"]+)" priced at ("[^"]+")$/ | ||
| 143 |      * @Given /^the store(?:| also) has a product "([^"]+)" priced at ("[^"]+") in ("[^"]+" channel)$/ | ||
| 144 | */ | ||
| 145 | public function storeHasAProductPricedAt($productName, int $price = 100, ChannelInterface $channel = null) | ||
| 151 | |||
| 152 | /** | ||
| 153 |      * @Given /^(this product) is(?:| also) priced at ("[^"]+") in ("[^"]+" channel)$/ | ||
| 154 | */ | ||
| 155 | public function thisProductIsAlsoPricedAtInChannel(ProductInterface $product, int $price, ChannelInterface $channel) | ||
| 165 | |||
| 166 | /** | ||
| 167 |      * @Given /^(this product) is(?:| also) available in ("[^"]+" channel)$/ | ||
| 168 |      * @Given /^(this product) is(?:| also) available in the ("[^"]+" channel)$/ | ||
| 169 | */ | ||
| 170 | public function thisProductIsAlsoAvailableInChannel(ProductInterface $product, ChannelInterface $channel): void | ||
| 174 | |||
| 175 | /** | ||
| 176 |      * @Given /^(this product) is(?:| also) unavailable in ("[^"]+" channel)$/ | ||
| 177 |      * @Given /^(this product) is disabled in ("[^"]+" channel)$/ | ||
| 178 | */ | ||
| 179 | public function thisProductIsAlsoUnavailableInChannel(ProductInterface $product, ChannelInterface $channel): void | ||
| 184 | |||
| 185 | /** | ||
| 186 | * @Given the store( also) has a product :productName with code :code | ||
| 187 | * @Given the store( also) has a product :productName with code :code, created at :date | ||
| 188 | */ | ||
| 189 | public function storeHasProductWithCode($productName, $code, $date = 'now') | ||
| 197 | |||
| 198 | /** | ||
| 199 |      * @Given /^the store(?:| also) has a product "([^"]+)" priced at ("[^"]+") available in (channel "[^"]+") and (channel "[^"]+")$/ | ||
| 200 | */ | ||
| 201 | public function storeHasAProductPricedAtAvailableInChannels($productName, int $price = 100, ...$channels) | ||
| 216 | |||
| 217 | /** | ||
| 218 | * @Given /^(this product) is named "([^"]+)" (in the "([^"]+)" locale)$/ | ||
| 219 | * @Given /^the (product "[^"]+") is named "([^"]+)" (in the "([^"]+)" locale)$/ | ||
| 220 | */ | ||
| 221 | public function thisProductIsNamedIn(ProductInterface $product, $name, $locale) | ||
| 227 | |||
| 228 | /** | ||
| 229 |      * @Given /^the store has a product named "([^"]+)" in ("[^"]+" locale) and "([^"]+)" in ("[^"]+" locale)$/ | ||
| 230 | */ | ||
| 231 | public function theStoreHasProductNamedInAndIn($firstName, $firstLocale, $secondName, $secondLocale) | ||
| 242 | |||
| 243 | /** | ||
| 244 | * @Given /^the store has(?:| a| an) "([^"]+)" configurable product$/ | ||
| 245 | * @Given /^the store has(?:| a| an) "([^"]+)" configurable product with "([^"]+)" slug$/ | ||
| 246 | */ | ||
| 247 | public function storeHasAConfigurableProduct($productName, $slug = null) | ||
| 273 | |||
| 274 | /** | ||
| 275 | * @Given the store has( also) :firstProductName and :secondProductName products | ||
| 276 | * @Given the store has( also) :firstProductName, :secondProductName and :thirdProductName products | ||
| 277 | * @Given the store has( also) :firstProductName, :secondProductName, :thirdProductName and :fourthProductName products | ||
| 278 | */ | ||
| 279 | public function theStoreHasProducts(...$productsNames) | ||
| 285 | |||
| 286 | /** | ||
| 287 | * @Given /^(this channel) has "([^"]+)", "([^"]+)", "([^"]+)" and "([^"]+)" products$/ | ||
| 288 | */ | ||
| 289 | public function thisChannelHasProducts(ChannelInterface $channel, ...$productsNames) | ||
| 297 | |||
| 298 | /** | ||
| 299 |      * @Given /^the (product "[^"]+") has(?:| a) "([^"]+)" variant priced at ("[^"]+")$/ | ||
| 300 |      * @Given /^(this product) has "([^"]+)" variant priced at ("[^"]+")$/ | ||
| 301 |      * @Given /^(this product) has "([^"]+)" variant priced at ("[^"]+") in ("([^"]+)" channel)$/ | ||
| 302 | */ | ||
| 303 | public function theProductHasVariantPricedAt( | ||
| 317 | |||
| 318 | /** | ||
| 319 |      * @Given /^("[^"]+" variant) priced at ("[^"]+") in ("[^"]+" channel)$/ | ||
| 320 | */ | ||
| 321 | public function variantPricedAtInChannel( | ||
| 330 | |||
| 331 | /** | ||
| 332 | * @Given /^the (product "[^"]+") has(?:| a| an) "([^"]+)" variant$/ | ||
| 333 | * @Given /^(this product) has(?:| a| an) "([^"]+)" variant$/ | ||
| 334 | * @Given /^(this product) has "([^"]+)", "([^"]+)" and "([^"]+)" variants$/ | ||
| 335 | */ | ||
| 336 | public function theProductHasVariants(ProductInterface $product, ...$variantNames) | ||
| 350 | |||
| 351 | /** | ||
| 352 | * @Given /^the (product "[^"]+")(?:| also) has a nameless variant with code "([^"]+)"$/ | ||
| 353 | * @Given /^(this product)(?:| also) has a nameless variant with code "([^"]+)"$/ | ||
| 354 | * @Given /^(it)(?:| also) has a nameless variant with code "([^"]+)"$/ | ||
| 355 | */ | ||
| 356 | public function theProductHasNamelessVariantWithCode(ProductInterface $product, $variantCode) | ||
| 362 | |||
| 363 | /** | ||
| 364 | * @Given /^the (product "[^"]+")(?:| also) has(?:| a| an) "([^"]+)" variant with code "([^"]+)"$/ | ||
| 365 | * @Given /^(this product)(?:| also) has(?:| a| an) "([^"]+)" variant with code "([^"]+)"$/ | ||
| 366 | * @Given /^(it)(?:| also) has(?:| a| an) "([^"]+)" variant with code "([^"]+)"$/ | ||
| 367 | */ | ||
| 368 | public function theProductHasVariantWithCode(ProductInterface $product, $variantName, $variantCode) | ||
| 374 | |||
| 375 | /** | ||
| 376 |      * @Given /^(this product) has "([^"]+)" variant priced at ("[^"]+") which does not require shipping$/ | ||
| 377 | */ | ||
| 378 | public function theProductHasVariantWhichDoesNotRequireShipping( | ||
| 393 | |||
| 394 | /** | ||
| 395 | * @Given /^the (product "[^"]+") has(?:| also)(?:| a| an) "([^"]+)" variant$/ | ||
| 396 | * @Given /^the (product "[^"]+") has(?:| also)(?:| a| an) "([^"]+)" variant at position ([^"]+)$/ | ||
| 397 | * @Given /^(this product) has(?:| also)(?:| a| an) "([^"]+)" variant at position ([^"]+)$/ | ||
| 398 | */ | ||
| 399 | public function theProductHasVariantAtPosition( | ||
| 413 | |||
| 414 | /** | ||
| 415 |      * @Given /^(this variant) is also priced at ("[^"]+") in ("([^"]+)" channel)$/ | ||
| 416 | */ | ||
| 417 | public function thisVariantIsAlsoPricedAtInChannel(ProductVariantInterface $productVariant, string $price, ChannelInterface $channel) | ||
| 426 | |||
| 427 | /** | ||
| 428 |      * @Given /^(it|this product) has(?:| also) variant named "([^"]+)" in ("[^"]+" locale) and "([^"]+)" in ("[^"]+" locale)$/ | ||
| 429 | */ | ||
| 430 | public function itHasVariantNamedInAndIn(ProductInterface $product, $firstName, $firstLocale, $secondName, $secondLocale) | ||
| 447 | |||
| 448 | /** | ||
| 449 |      * @Given /^(this product) has "([^"]+)" variant priced at ("[^"]+") identified by "([^"]+)"$/ | ||
| 450 | */ | ||
| 451 | public function theProductHasVariantPricedAtIdentifiedBy( | ||
| 459 | |||
| 460 | /** | ||
| 461 | * @Given /^(this product) only variant was renamed to "([^"]+)"$/ | ||
| 462 | */ | ||
| 463 | public function productOnlyVariantWasRenamed(ProductInterface $product, $variantName) | ||
| 473 | |||
| 474 | /** | ||
| 475 | * @Given /^there is product "([^"]+)" available in ((?:this|that|"[^"]+") channel)$/ | ||
| 476 |      * @Given /^the store has a product "([^"]+)" available in ("([^"]+)" channel)$/ | ||
| 477 | */ | ||
| 478 | public function thereIsProductAvailableInGivenChannel($productName, ChannelInterface $channel) | ||
| 484 | |||
| 485 | /** | ||
| 486 |      * @Given /^([^"]+) belongs to ("[^"]+" tax category)$/ | ||
| 487 | * @Given the product :product belongs to :taxCategory tax category | ||
| 488 | */ | ||
| 489 | public function productBelongsToTaxCategory(ProductInterface $product, TaxCategoryInterface $taxCategory) | ||
| 497 | |||
| 498 | /** | ||
| 499 | * @Given /^(it) comes in the following variations:$/ | ||
| 500 | */ | ||
| 501 | public function itComesInTheFollowingVariations(ProductInterface $product, TableNode $table) | ||
| 522 | |||
| 523 | /** | ||
| 524 |      * @Given /^("[^"]+" variant of product "[^"]+") belongs to ("[^"]+" tax category)$/ | ||
| 525 | */ | ||
| 526 | public function productVariantBelongsToTaxCategory( | ||
| 535 | |||
| 536 | /** | ||
| 537 | * @Given /^(this product) has option "([^"]+)" with values "([^"]+)" and "([^"]+)"$/ | ||
| 538 | * @Given /^(this product) has option "([^"]+)" with values "([^"]+)", "([^"]+)" and "([^"]+)"$/ | ||
| 539 | */ | ||
| 540 | public function thisProductHasOptionWithValues(ProductInterface $product, $optionName, ...$values): void | ||
| 544 | |||
| 545 | /** | ||
| 546 | * @Given /^(this product) has an option "([^"]*)" without any values$/ | ||
| 547 | */ | ||
| 548 | public function thisProductHasAnOptionWithoutAnyValues(ProductInterface $product, string $optionName): void | ||
| 552 | |||
| 553 | /** | ||
| 554 | * @Given /^there (?:is|are) (\d+) unit(?:|s) of (product "([^"]+)") available in the inventory$/ | ||
| 555 | */ | ||
| 556 | public function thereIsQuantityOfProducts($quantity, ProductInterface $product) | ||
| 564 | |||
| 565 | /** | ||
| 566 | * @Given /^the (product "([^"]+)") is out of stock$/ | ||
| 567 | */ | ||
| 568 | public function theProductIsOutOfStock(ProductInterface $product) | ||
| 577 | |||
| 578 | /** | ||
| 579 | * @When other customer has bought :quantity :product products by this time | ||
| 580 | */ | ||
| 581 | public function otherCustomerHasBoughtProductsByThisTime($quantity, ProductInterface $product) | ||
| 589 | |||
| 590 | /** | ||
| 591 | * @Given /^(this product) is tracked by the inventory$/ | ||
| 592 |      * @Given /^(?:|the )("[^"]+" product) is(?:| also) tracked by the inventory$/ | ||
| 593 | */ | ||
| 594 | public function thisProductIsTrackedByTheInventory(ProductInterface $product) | ||
| 602 | |||
| 603 | /** | ||
| 604 |      * @Given /^(this product) is available in "([^"]+)" ([^"]+) priced at ("[^"]+")$/ | ||
| 605 | */ | ||
| 606 | public function thisProductIsAvailableInSize(ProductInterface $product, $optionValueName, $optionName, int $price) | ||
| 621 | |||
| 622 | /** | ||
| 623 | * @Given the :product product's :optionValueName size belongs to :shippingCategory shipping category | ||
| 624 | */ | ||
| 625 | public function thisProductSizeBelongsToShippingCategory(ProductInterface $product, $optionValueName, ShippingCategoryInterface $shippingCategory) | ||
| 638 | |||
| 639 | /** | ||
| 640 | * @Given /^(this product) has (this product option)$/ | ||
| 641 |      * @Given /^(this product) has (?:a|an) ("[^"]+" option)$/ | ||
| 642 | */ | ||
| 643 | public function thisProductHasThisProductOption(ProductInterface $product, ProductOptionInterface $option) | ||
| 649 | |||
| 650 | /** | ||
| 651 | * @Given /^(this product) has all possible variants$/ | ||
| 652 | */ | ||
| 653 | public function thisProductHasAllPossibleVariants(ProductInterface $product) | ||
| 682 | |||
| 683 | /** | ||
| 684 |      * @Given /^there are ([^"]+) units of ("[^"]+" variant of product "[^"]+") available in the inventory$/ | ||
| 685 | */ | ||
| 686 | public function thereAreItemsOfProductInVariantAvailableInTheInventory($quantity, ProductVariantInterface $productVariant) | ||
| 693 | |||
| 694 | /** | ||
| 695 |      * @Given /^the ("[^"]+" product variant) is tracked by the inventory$/ | ||
| 696 | */ | ||
| 697 | public function theProductVariantIsTrackedByTheInventory(ProductVariantInterface $productVariant) | ||
| 703 | |||
| 704 | /** | ||
| 705 |      * @Given /^(this product)'s price is ("[^"]+")$/ | ||
| 706 |      * @Given /^the (product "[^"]+") changed its price to ("[^"]+")$/ | ||
| 707 |      * @Given /^(this product) price has been changed to ("[^"]+")$/ | ||
| 708 | */ | ||
| 709 | public function theProductChangedItsPriceTo(ProductInterface $product, int $price) | ||
| 718 | |||
| 719 | /** | ||
| 720 | * @Given /^(this product)(?:| also) has an image "([^"]+)" with "([^"]+)" type$/ | ||
| 721 |      * @Given /^the ("[^"]+" product)(?:| also) has an image "([^"]+)" with "([^"]+)" type$/ | ||
| 722 | * @Given /^(it)(?:| also) has an image "([^"]+)" with "([^"]+)" type$/ | ||
| 723 | */ | ||
| 724 | public function thisProductHasAnImageWithType(ProductInterface $product, $imagePath, $imageType) | ||
| 728 | |||
| 729 | /** | ||
| 730 |      * @Given /^(this product) has an image "([^"]+)" with "([^"]+)" type for ("[^"]+" variant)$/ | ||
| 731 | */ | ||
| 732 | public function thisProductHasAnImageWithTypeForVariant( | ||
| 740 | |||
| 741 | /** | ||
| 742 |      * @Given /^(this product) belongs to ("([^"]+)" shipping category)$/ | ||
| 743 | * @Given product :product shipping category has been changed to :shippingCategory | ||
| 744 | */ | ||
| 745 | public function thisProductBelongsToShippingCategory(ProductInterface $product, ShippingCategoryInterface $shippingCategory) | ||
| 750 | |||
| 751 | /** | ||
| 752 | * @Given /^(this product) has been disabled$/ | ||
| 753 | */ | ||
| 754 | public function thisProductHasBeenDisabled(ProductInterface $product) | ||
| 759 | |||
| 760 | /** | ||
| 761 | * @Given the product :product was renamed to :productName | ||
| 762 | */ | ||
| 763 | public function theProductWasRenamedTo(ProductInterface $product, string $productName): void | ||
| 769 | |||
| 770 | /** | ||
| 771 | * @Given /^(this product) does not require shipping$/ | ||
| 772 | */ | ||
| 773 | public function thisProductDoesNotRequireShipping(ProductInterface $product): void | ||
| 782 | |||
| 783 | /** | ||
| 784 | * @Given product's :product code is :code | ||
| 785 | */ | ||
| 786 | public function productCodeIs(ProductInterface $product, string $code): void | ||
| 792 | |||
| 793 | /** | ||
| 794 | * @Given the product :product has height :height, width :width, depth :depth, weight :weight | ||
| 795 | */ | ||
| 796 | public function productHasDimensions(ProductInterface $product, float $height, float $width, float $depth, float $weight): void | ||
| 807 | |||
| 808 | /** | ||
| 809 | * @Given the product :product has the slug :slug | ||
| 810 | */ | ||
| 811 | public function productHasSlug(ProductInterface $product, string $slug): void | ||
| 817 | |||
| 818 | /** | ||
| 819 | * @Given the description of product :product is :description | ||
| 820 | */ | ||
| 821 | public function descriptionOfProductIs(ProductInterface $product, string $description): void | ||
| 827 | |||
| 828 | /** | ||
| 829 | * @Given the meta keywords of product :product is :metaKeywords | ||
| 830 | */ | ||
| 831 | public function metaKeywordsOfProductIs(ProductInterface $product, string $metaKeywords): void | ||
| 837 | |||
| 838 | /** | ||
| 839 | * @Given the short description of product :product is :shortDescription | ||
| 840 | */ | ||
| 841 | public function shortDescriptionOfProductIs(ProductInterface $product, string $shortDescription): void | ||
| 847 | |||
| 848 | /** | ||
| 849 | * @Given the product :product has original price :originalPrice | ||
| 850 | */ | ||
| 851 | public function theProductHasOriginalPrice(ProductInterface $product, string $originalPrice): void | ||
| 862 | |||
| 863 | /** | ||
| 864 | * @Given the product :product has option :productOption named :optionValue with code :optionCode | ||
| 865 | */ | ||
| 866 | public function productHasOption( | ||
| 882 | |||
| 883 | /** | ||
| 884 | * @Given the product :product has :productVariantName variant with code :code, price :price, current stock :currentStock | ||
| 885 | */ | ||
| 886 | public function productHasVariant(ProductInterface $product, string $productVariantName, string $code, string $price, int $currentStock): void | ||
| 894 | |||
| 895 | private function getPriceFromString(string $price): int | ||
| 899 | |||
| 900 | /** | ||
| 901 | * @param string $productName | ||
| 902 | * | ||
| 903 | * @return ProductInterface | ||
| 904 | */ | ||
| 905 | private function createProduct($productName, int $price = 100, ChannelInterface $channel = null) | ||
| 942 | |||
| 943 | /** | ||
| 944 | * @param string $value | ||
| 945 | * @param string $code | ||
| 946 | * | ||
| 947 | * @return ProductOptionValueInterface | ||
| 948 | */ | ||
| 949 | private function addProductOption(ProductOptionInterface $option, $value, $code) | ||
| 962 | |||
| 963 | private function saveProduct(ProductInterface $product) | ||
| 968 | |||
| 969 | /** | ||
| 970 | * @param string $name | ||
| 971 | * | ||
| 972 | * @return NodeElement | ||
| 973 | */ | ||
| 974 | private function getParameter($name) | ||
| 978 | |||
| 979 | /** | ||
| 980 | * @param string $productVariantName | ||
| 981 | * @param int $price | ||
| 982 | * @param string $code | ||
| 983 | * @param ChannelInterface $channel | ||
| 984 | * @param int $position | ||
| 985 | * @param bool $shippingRequired | ||
| 986 | * | ||
| 987 | * @return ProductVariantInterface | ||
| 988 | */ | ||
| 989 | private function createProductVariant( | ||
| 1019 | |||
| 1020 | /** | ||
| 1021 | * @param string $name | ||
| 1022 | * @param string $locale | ||
| 1023 | */ | ||
| 1024 | private function addProductTranslation(ProductInterface $product, $name, $locale) | ||
| 1039 | |||
| 1040 | /** | ||
| 1041 | * @param string $name | ||
| 1042 | * @param string $locale | ||
| 1043 | */ | ||
| 1044 | private function addProductVariantTranslation(ProductVariantInterface $productVariant, $name, $locale) | ||
| 1053 | |||
| 1054 | /** | ||
| 1055 | * @return ChannelPricingInterface | ||
| 1056 | */ | ||
| 1057 | private function createChannelPricingForChannel(int $price, ChannelInterface $channel = null) | ||
| 1066 | |||
| 1067 | private function addOptionToProduct(ProductInterface $product, string $optionName, array $values): void | ||
| 1088 | |||
| 1089 | private function createProductImage( | ||
| 1113 | } | ||
| 1114 | 
Our type inference engine has found a suspicous assignment of a value to a property. This check raises an issue when a value that can be of a mixed type is assigned to a property that is type hinted more strictly.
For example, imagine you have a variable
$accountIdthat can either hold an Id object or false (if there is no account id yet). Your code now assigns that value to theidproperty of an instance of theAccountclass. This class holds a proper account, so the id value must no longer be false.Either this assignment is in error or a type check should be added for that assignment.