Duplicate code is one of the most pungent code smells. A rule that is often used is to re-structure code once it is duplicated in three or more places.
Common duplication problems, and corresponding solutions are:
Complex classes like Offer 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 Offer, and based on these observations, apply Extract Interface, too.
| 1 | <?php | ||
| 41 | abstract class Offer extends EventSourcedAggregateRoot | ||
| 42 | { | ||
| 43 | const DUPLICATE_REASON = 'duplicate'; | ||
| 44 | const INAPPROPRIATE_REASON = 'inappropriate'; | ||
| 45 | |||
| 46 | /** | ||
| 47 | * @var LabelCollection | ||
| 48 | */ | ||
| 49 | protected $labels; | ||
| 50 | |||
| 51 | /** | ||
| 52 | * @var UUID[] | ||
| 53 | */ | ||
| 54 | protected $mediaObjects = []; | ||
| 55 | |||
| 56 | /** | ||
| 57 | * @var UUID | ||
| 58 | */ | ||
| 59 | protected $mainImageId; | ||
| 60 | |||
| 61 | /** | ||
| 62 | * @var string | ||
| 63 | * | ||
| 64 | * Organizer ids can come from UDB2 which does not strictly use UUIDs. | ||
| 65 | */ | ||
| 66 | protected $organizerId; | ||
| 67 | |||
| 68 | /** | ||
| 69 | * @var WorkflowStatus | ||
| 70 | */ | ||
| 71 | protected $workflowStatus; | ||
| 72 | |||
| 73 | /** | ||
| 74 | * @var StringLiteral|null | ||
| 75 | */ | ||
| 76 | protected $rejectedReason; | ||
| 77 | |||
| 78 | /** | ||
| 79 | * @var PriceInfo | ||
| 80 | */ | ||
| 81 | protected $priceInfo; | ||
| 82 | |||
| 83 | /** | ||
| 84 | * Offer constructor. | ||
| 85 | */ | ||
| 86 | public function __construct() | ||
| 87 |     { | ||
| 88 | $this->resetLabels(); | ||
| 89 | } | ||
| 90 | |||
| 91 | /** | ||
| 92 | * @return LabelCollection | ||
| 93 | */ | ||
| 94 | public function getLabels() | ||
| 98 | |||
| 99 | /** | ||
| 100 | * Get the id of the main image if one is selected for this offer. | ||
| 101 | * | ||
| 102 | * @return UUID|null | ||
| 103 | */ | ||
| 104 | protected function getMainImageId() | ||
| 108 | |||
| 109 | /** | ||
| 110 | * @param Label $label | ||
| 111 | */ | ||
| 112 | public function addLabel(Label $label) | ||
| 120 | |||
| 121 | /** | ||
| 122 | * @param Label $label | ||
| 123 | */ | ||
| 124 | public function deleteLabel(Label $label) | ||
| 132 | |||
| 133 | /** | ||
| 134 | * @param LabelCollection $newLabelCollection | ||
| 135 | */ | ||
| 136 | public function syncLabels(LabelCollection $newLabelCollection) | ||
| 152 | |||
| 153 | /** | ||
| 154 | * @param Language $language | ||
| 155 | * @param StringLiteral $title | ||
| 156 | */ | ||
| 157 | public function translateTitle(Language $language, StringLiteral $title) | ||
| 163 | |||
| 164 | /** | ||
| 165 | * @param Language $language | ||
| 166 | * @param StringLiteral $description | ||
| 167 | */ | ||
| 168 | public function translateDescription(Language $language, StringLiteral $description) | ||
| 174 | |||
| 175 | |||
| 176 | /** | ||
| 177 | * @param string $description | ||
| 178 | */ | ||
| 179 | public function updateDescription($description) | ||
| 185 | |||
| 186 | /** | ||
| 187 | * @param string $typicalAgeRange | ||
| 188 | */ | ||
| 189 | public function updateTypicalAgeRange($typicalAgeRange) | ||
| 195 | |||
| 196 | public function deleteTypicalAgeRange() | ||
| 202 | |||
| 203 | /** | ||
| 204 | * @param string $organizerId | ||
| 205 | */ | ||
| 206 | public function updateOrganizer($organizerId) | ||
| 214 | |||
| 215 | /** | ||
| 216 | * Delete the given organizer. | ||
| 217 | * | ||
| 218 | * @param string $organizerId | ||
| 219 | */ | ||
| 220 | public function deleteOrganizer($organizerId) | ||
| 228 | |||
| 229 | /** | ||
| 230 | * Updated the contact info. | ||
| 231 | * @param ContactPoint $contactPoint | ||
| 232 | */ | ||
| 233 | public function updateContactPoint(ContactPoint $contactPoint) | ||
| 239 | |||
| 240 | /** | ||
| 241 | * Updated the booking info. | ||
| 242 | * | ||
| 243 | * @param BookingInfo $bookingInfo | ||
| 244 | */ | ||
| 245 | public function updateBookingInfo(BookingInfo $bookingInfo) | ||
| 251 | |||
| 252 | /** | ||
| 253 | * @param PriceInfo $priceInfo | ||
| 254 | */ | ||
| 255 | public function updatePriceInfo(PriceInfo $priceInfo) | ||
| 263 | |||
| 264 | /** | ||
| 265 | * @param AbstractPriceInfoUpdated $priceInfoUpdated | ||
| 266 | */ | ||
| 267 | protected function applyPriceInfoUpdated(AbstractPriceInfoUpdated $priceInfoUpdated) | ||
| 271 | |||
| 272 | /** | ||
| 273 | * @param AbstractLabelAdded $labelAdded | ||
| 274 | */ | ||
| 275 | protected function applyLabelAdded(AbstractLabelAdded $labelAdded) | ||
| 283 | |||
| 284 | /** | ||
| 285 | * @param AbstractLabelDeleted $labelDeleted | ||
| 286 | */ | ||
| 287 | protected function applyLabelDeleted(AbstractLabelDeleted $labelDeleted) | ||
| 293 | |||
| 294 | protected function resetLabels() | ||
| 298 | |||
| 299 | /** | ||
| 300 | * @param Image $image | ||
| 301 | * @return boolean | ||
| 302 | */ | ||
| 303 | private function containsImage(Image $image) | ||
| 316 | |||
| 317 | /** | ||
| 318 | * Add a new image. | ||
| 319 | * | ||
| 320 | * @param Image $image | ||
| 321 | */ | ||
| 322 | public function addImage(Image $image) | ||
| 330 | |||
| 331 | /** | ||
| 332 | * @param AbstractUpdateImage $updateImageCommand | ||
| 333 | */ | ||
| 334 | public function updateImage(AbstractUpdateImage $updateImageCommand) | ||
| 340 | |||
| 341 | /** | ||
| 342 | * Remove an image. | ||
| 343 | * | ||
| 344 | * @param Image $image | ||
| 345 | */ | ||
| 346 | public function removeImage(Image $image) | ||
| 354 | |||
| 355 | /** | ||
| 356 | * Make an existing image of the item the main image. | ||
| 357 | * | ||
| 358 | * @param Image $image | ||
| 359 | */ | ||
| 360 | public function selectMainImage(Image $image) | ||
| 372 | |||
| 373 | /** | ||
| 374 | * Delete the offer. | ||
| 375 | */ | ||
| 376 | public function delete() | ||
| 382 | |||
| 383 | /** | ||
| 384 | * @param CultureFeed_Cdb_Item_Base $cdbItem | ||
| 385 | */ | ||
| 386 | protected function importWorkflowStatus(CultureFeed_Cdb_Item_Base $cdbItem) | ||
| 395 | |||
| 396 | /** | ||
| 397 | * Publish the offer when it has workflowstatus draft. | ||
| 398 | * @param \DateTimeInterface $publicationDate | ||
| 399 | */ | ||
| 400 | public function publish(\DateTimeInterface $publicationDate) | ||
| 406 | |||
| 407 | /** | ||
| 408 | * @return bool | ||
| 409 | * @throws Exception | ||
| 410 | */ | ||
| 411 | View Code Duplication | private function guardPublish() | |
| 423 | |||
| 424 | /** | ||
| 425 | * Approve the offer when it's waiting for validation. | ||
| 426 | */ | ||
| 427 | public function approve() | ||
| 431 | |||
| 432 | /** | ||
| 433 | * @return bool | ||
| 434 | * @throws Exception | ||
| 435 | */ | ||
| 436 | View Code Duplication | private function guardApprove() | |
| 448 | |||
| 449 | /** | ||
| 450 | * Reject an offer that is waiting for validation with a given reason. | ||
| 451 | * @param StringLiteral $reason | ||
| 452 | */ | ||
| 453 | public function reject(StringLiteral $reason) | ||
| 457 | |||
| 458 | public function flagAsDuplicate() | ||
| 463 | |||
| 464 | public function flagAsInappropriate() | ||
| 469 | |||
| 470 | /** | ||
| 471 | * @param StringLiteral $reason | ||
| 472 | * @return bool | ||
| 473 | * false when the offer can still be rejected, true when the offer is already rejected for the same reason | ||
| 474 | * @throws Exception | ||
| 475 | */ | ||
| 476 | private function guardRejection(StringLiteral $reason) | ||
| 492 | |||
| 493 | /** | ||
| 494 | * @param AbstractPublished $published | ||
| 495 | */ | ||
| 496 | protected function applyPublished(AbstractPublished $published) | ||
| 500 | |||
| 501 | /** | ||
| 502 | * @param AbstractApproved $approved | ||
| 503 | */ | ||
| 504 | protected function applyApproved(AbstractApproved $approved) | ||
| 508 | |||
| 509 | /** | ||
| 510 | * @param AbstractRejected $rejected | ||
| 511 | */ | ||
| 512 | protected function applyRejected(AbstractRejected $rejected) | ||
| 517 | |||
| 518 | /** | ||
| 519 | * @param AbstractFlaggedAsDuplicate $flaggedAsDuplicate | ||
| 520 | */ | ||
| 521 | protected function applyFlaggedAsDuplicate(AbstractFlaggedAsDuplicate $flaggedAsDuplicate) | ||
| 526 | |||
| 527 | /** | ||
| 528 | * @param AbstractFlaggedAsInappropriate $flaggedAsInappropriate | ||
| 529 | */ | ||
| 530 | protected function applyFlaggedAsInappropriate(AbstractFlaggedAsInappropriate $flaggedAsInappropriate) | ||
| 535 | |||
| 536 | View Code Duplication | protected function applyImageAdded(AbstractImageAdded $imageAdded) | |
| 545 | |||
| 546 | View Code Duplication | protected function applyImageRemoved(AbstractImageRemoved $imageRemoved) | |
| 558 | |||
| 559 | protected function applyMainImageSelected(AbstractMainImageSelected $mainImageSelected) | ||
| 563 | |||
| 564 | protected function applyOrganizerUpdated(AbstractOrganizerUpdated $organizerUpdated) | ||
| 568 | |||
| 569 | protected function applyOrganizerDeleted(AbstractOrganizerDeleted $organizerDeleted) | ||
| 573 | |||
| 574 | /** | ||
| 575 | * @param Label $label | ||
| 576 | * @return AbstractLabelAdded | ||
| 577 | */ | ||
| 578 | abstract protected function createLabelAddedEvent(Label $label); | ||
| 579 | |||
| 580 | /** | ||
| 581 | * @param Label $label | ||
| 582 | * @return AbstractLabelDeleted | ||
| 583 | */ | ||
| 584 | abstract protected function createLabelDeletedEvent(Label $label); | ||
| 585 | |||
| 586 | /** | ||
| 587 | * @param Language $language | ||
| 588 | * @param StringLiteral $title | ||
| 589 | * @return AbstractTitleTranslated | ||
| 590 | */ | ||
| 591 | abstract protected function createTitleTranslatedEvent(Language $language, StringLiteral $title); | ||
| 592 | |||
| 593 | /** | ||
| 594 | * @param Language $language | ||
| 595 | * @param StringLiteral $description | ||
| 596 | * @return AbstractDescriptionTranslated | ||
| 597 | */ | ||
| 598 | abstract protected function createDescriptionTranslatedEvent(Language $language, StringLiteral $description); | ||
| 599 | |||
| 600 | /** | ||
| 601 | * @param Image $image | ||
| 602 | * @return AbstractImageAdded | ||
| 603 | */ | ||
| 604 | abstract protected function createImageAddedEvent(Image $image); | ||
| 605 | |||
| 606 | /** | ||
| 607 | * @param Image $image | ||
| 608 | * @return AbstractImageRemoved | ||
| 609 | */ | ||
| 610 | abstract protected function createImageRemovedEvent(Image $image); | ||
| 611 | |||
| 612 | /** | ||
| 613 | * @param AbstractUpdateImage $updateImageCommand | ||
| 614 | * @return AbstractImageUpdated | ||
| 615 | */ | ||
| 616 | abstract protected function createImageUpdatedEvent( | ||
| 619 | |||
| 620 | /** | ||
| 621 | * @param Image $image | ||
| 622 | * @return AbstractMainImageSelected | ||
| 623 | */ | ||
| 624 | abstract protected function createMainImageSelectedEvent(Image $image); | ||
| 625 | |||
| 626 | /** | ||
| 627 | * @return AbstractOfferDeleted | ||
| 628 | */ | ||
| 629 | abstract protected function createOfferDeletedEvent(); | ||
| 630 | |||
| 631 | /** | ||
| 632 | * @param string $description | ||
| 633 | * @return AbstractDescriptionUpdated | ||
| 634 | */ | ||
| 635 | abstract protected function createDescriptionUpdatedEvent($description); | ||
| 636 | |||
| 637 | /** | ||
| 638 | * @param string $typicalAgeRange | ||
| 639 | * @return AbstractTypicalAgeRangeUpdated | ||
| 640 | */ | ||
| 641 | abstract protected function createTypicalAgeRangeUpdatedEvent($typicalAgeRange); | ||
| 642 | |||
| 643 | /** | ||
| 644 | * @return AbstractTypicalAgeRangeDeleted | ||
| 645 | */ | ||
| 646 | abstract protected function createTypicalAgeRangeDeletedEvent(); | ||
| 647 | |||
| 648 | /** | ||
| 649 | * @param string $organizerId | ||
| 650 | * @return AbstractOrganizerUpdated | ||
| 651 | */ | ||
| 652 | abstract protected function createOrganizerUpdatedEvent($organizerId); | ||
| 653 | |||
| 654 | /** | ||
| 655 | * @param string $organizerId | ||
| 656 | * @return AbstractOrganizerDeleted | ||
| 657 | */ | ||
| 658 | abstract protected function createOrganizerDeletedEvent($organizerId); | ||
| 659 | |||
| 660 | /** | ||
| 661 | * @param ContactPoint $contactPoint | ||
| 662 | * @return AbstractContactPointUpdated | ||
| 663 | */ | ||
| 664 | abstract protected function createContactPointUpdatedEvent(ContactPoint $contactPoint); | ||
| 665 | |||
| 666 | /** | ||
| 667 | * @param BookingInfo $bookingInfo | ||
| 668 | * @return AbstractBookingInfoUpdated | ||
| 669 | */ | ||
| 670 | abstract protected function createBookingInfoUpdatedEvent(BookingInfo $bookingInfo); | ||
| 671 | |||
| 672 | /** | ||
| 673 | * @param PriceInfo $priceInfo | ||
| 674 | * @return AbstractPriceInfoUpdated | ||
| 675 | */ | ||
| 676 | abstract protected function createPriceInfoUpdatedEvent(PriceInfo $priceInfo); | ||
| 677 | |||
| 678 | /** | ||
| 679 | * @param \DateTimeInterface $publicationDate | ||
| 680 | * @return AbstractPublished | ||
| 681 | */ | ||
| 682 | abstract protected function createPublishedEvent(\DateTimeInterface $publicationDate); | ||
| 683 | |||
| 684 | /** | ||
| 685 | * @return AbstractApproved | ||
| 686 | */ | ||
| 687 | abstract protected function createApprovedEvent(); | ||
| 688 | |||
| 689 | /** | ||
| 690 | * @param StringLiteral $reason | ||
| 691 | * @return AbstractRejected | ||
| 692 | */ | ||
| 693 | abstract protected function createRejectedEvent(StringLiteral $reason); | ||
| 694 | |||
| 695 | /** | ||
| 696 | * @return AbstractFlaggedAsDuplicate | ||
| 697 | */ | ||
| 698 | abstract protected function createFlaggedAsDuplicate(); | ||
| 699 | |||
| 700 | /** | ||
| 701 | * @return AbstractFlaggedAsInappropriate | ||
| 702 | */ | ||
| 703 | abstract protected function createFlaggedAsInappropriate(); | ||
| 704 | } | ||
| 705 | 
Duplicated code is one of the most pungent code smells. If you need to duplicate the same code in three or more different places, we strongly encourage you to look into extracting the code into a single class or operation.
You can also find more detailed suggestions in the “Code” section of your repository.