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 Schema 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 Schema, and based on these observations, apply Extract Interface, too.
| 1 | <?php | ||
| 15 | class Schema | ||
| 16 | { | ||
| 17 | /** | ||
| 18 | * @var string | ||
| 19 | */ | ||
| 20 | protected $title; | ||
| 21 | |||
| 22 | /** | ||
| 23 | * @var string | ||
| 24 | */ | ||
| 25 | protected $description; | ||
| 26 | |||
| 27 | /** | ||
| 28 | * @var string | ||
| 29 | */ | ||
| 30 | protected $type; | ||
| 31 | |||
| 32 | /** | ||
| 33 | * @var string | ||
| 34 | */ | ||
| 35 | protected $format; | ||
| 36 | |||
| 37 | /** | ||
| 38 | * @var Schema | ||
| 39 | */ | ||
| 40 | protected $items; | ||
| 41 | |||
| 42 | /** | ||
| 43 | * @var \Doctrine\Common\Collections\ArrayCollection | ||
| 44 | */ | ||
| 45 | protected $properties; | ||
| 46 | |||
| 47 | /** | ||
| 48 | * @var Array | ||
| 49 | */ | ||
| 50 | protected $propertiesArray = array(); | ||
| 51 | |||
| 52 | /** | ||
| 53 | * @var Schema | ||
| 54 | */ | ||
| 55 | protected $additionalProperties; | ||
| 56 | |||
| 57 | /** | ||
| 58 | * @var string[] | ||
| 59 | */ | ||
| 60 | protected $required = array(); | ||
| 61 | |||
| 62 | /** | ||
| 63 | * @var boolean | ||
| 64 | */ | ||
| 65 | protected $translatable; | ||
| 66 | |||
| 67 | /** | ||
| 68 | * @var array | ||
| 69 | */ | ||
| 70 | protected $refCollection = array(); | ||
| 71 | |||
| 72 | /** | ||
| 73 | * possible event names this collection implements (queue events) | ||
| 74 | * | ||
| 75 | * @var array | ||
| 76 | */ | ||
| 77 | protected $eventNames = array(); | ||
| 78 | |||
| 79 | /** | ||
| 80 | * @var bool | ||
| 81 | */ | ||
| 82 | protected $readOnly = false; | ||
| 83 | |||
| 84 | /** | ||
| 85 | * these are the BSON primitive types. | ||
| 86 | * http://json-schema.org/latest/json-schema-core.html#anchor8 | ||
| 87 | * every type set *not* in this set will be carried over to 'format' | ||
| 88 | * | ||
| 89 | * @var string[] | ||
| 90 | */ | ||
| 91 | protected $primitiveTypes = array( | ||
| 92 | 'array', | ||
| 93 | 'boolean', | ||
| 94 | 'integer', | ||
| 95 | 'number', | ||
| 96 | 'null', | ||
| 97 | 'object', | ||
| 98 | 'string' | ||
| 99 | ); | ||
| 100 | |||
| 101 | /** | ||
| 102 | * known non-primitive types we map to primitives here. | ||
| 103 | * the type itself is set to the format. | ||
| 104 | * | ||
| 105 | * @var string[] | ||
| 106 | */ | ||
| 107 | protected $specialTypeMapping = array( | ||
| 108 | 'extref' => 'string', | ||
| 109 | 'translatable' => 'object' | ||
| 110 | ); | ||
| 111 | |||
| 112 | /** | ||
| 113 | * Build properties | ||
| 114 | */ | ||
| 115 | public function __construct() | ||
| 119 | |||
| 120 | /** | ||
| 121 | * set title | ||
| 122 | * | ||
| 123 | * @param string $title title | ||
| 124 | * | ||
| 125 | * @return void | ||
| 126 | */ | ||
| 127 | public function setTitle($title) | ||
| 131 | |||
| 132 | /** | ||
| 133 | * get title | ||
| 134 | * | ||
| 135 | * @return string | ||
| 136 | */ | ||
| 137 | public function getTitle() | ||
| 141 | |||
| 142 | /** | ||
| 143 | * set description | ||
| 144 | * | ||
| 145 | * @param string $description description | ||
| 146 | * | ||
| 147 | * @return void | ||
| 148 | */ | ||
| 149 | public function setDescription($description) | ||
| 153 | |||
| 154 | /** | ||
| 155 | * get description | ||
| 156 | * | ||
| 157 | * @return string | ||
| 158 | */ | ||
| 159 | public function getDescription() | ||
| 163 | |||
| 164 | /** | ||
| 165 | * set type | ||
| 166 | * | ||
| 167 | * @param string $type type | ||
| 168 | * | ||
| 169 | * @return void | ||
| 170 | */ | ||
| 171 | public function setType($type) | ||
| 192 | |||
| 193 | /** | ||
| 194 | * get type | ||
| 195 | * | ||
| 196 | * @return string type | ||
| 197 | */ | ||
| 198 | public function getType() | ||
| 202 | |||
| 203 | /** | ||
| 204 | * get format | ||
| 205 | * | ||
| 206 | * @return string format | ||
| 207 | */ | ||
| 208 | public function getFormat() | ||
| 212 | |||
| 213 | /** | ||
| 214 | * sets format | ||
| 215 | * | ||
| 216 | * @param string $format format | ||
| 217 | * | ||
| 218 | * @return void | ||
| 219 | */ | ||
| 220 | public function setFormat($format) | ||
| 224 | |||
| 225 | /** | ||
| 226 | * set items | ||
| 227 | * | ||
| 228 | * @param Schema $items items schema | ||
| 229 | * | ||
| 230 | * @return void | ||
| 231 | */ | ||
| 232 | public function setItems($items) | ||
| 236 | |||
| 237 | /** | ||
| 238 | * get items | ||
| 239 | * | ||
| 240 | * @return Schema | ||
| 241 | */ | ||
| 242 | public function getItems() | ||
| 246 | |||
| 247 | /** | ||
| 248 | * add a property | ||
| 249 | * | ||
| 250 | * @param string $name property name | ||
| 251 | * @param Schema $property property | ||
| 252 | * | ||
| 253 | * @return void | ||
| 254 | */ | ||
| 255 | public function addProperty($name, $property) | ||
| 262 | |||
| 263 | /** | ||
| 264 | * removes a property | ||
| 265 | * | ||
| 266 | * @param string $name property name | ||
| 267 | * | ||
| 268 | * @return void | ||
| 269 | */ | ||
| 270 | public function removeProperty($name) | ||
| 279 | |||
| 280 | /** | ||
| 281 | * returns a property | ||
| 282 | * | ||
| 283 | * @param string $name property name | ||
| 284 | * | ||
| 285 | * @return void|Schema property | ||
| 286 | */ | ||
| 287 | public function getProperty($name) | ||
| 294 | |||
| 295 | /** | ||
| 296 | * get properties | ||
| 297 | * | ||
| 298 | * @return Schema[]|null | ||
| 299 | */ | ||
| 300 | public function getProperties() | ||
| 309 | |||
| 310 | /** | ||
| 311 | * set additionalProperties on schema | ||
| 312 | * | ||
| 313 | * @param Schema $schema schema to use for additionalProperties type | ||
| 314 | * | ||
| 315 | * @return void | ||
| 316 | */ | ||
| 317 | public function setAdditionalProperties(Schema $schema) | ||
| 321 | |||
| 322 | /** | ||
| 323 | * get addtionalProperties for schema | ||
| 324 | * | ||
| 325 | * @return Schema | ||
| 326 | */ | ||
| 327 | public function getAdditionalProperties() | ||
| 331 | |||
| 332 | /** | ||
| 333 | * set required variables | ||
| 334 | * | ||
| 335 | * @param string[] $required arary of required fields | ||
| 336 | * | ||
| 337 | * @return void | ||
| 338 | */ | ||
| 339 | public function setRequired($required) | ||
| 343 | |||
| 344 | /** | ||
| 345 | * get required fields | ||
| 346 | * | ||
| 347 | * @return string[]|null | ||
| 348 | */ | ||
| 349 | public function getRequired() | ||
| 358 | |||
| 359 | /** | ||
| 360 | * set translatable flag | ||
| 361 | * | ||
| 362 | * This flag is a local extension to json schema. | ||
| 363 | * | ||
| 364 | * @param boolean $translatable translatable flag | ||
| 365 | * | ||
| 366 | * @return void | ||
| 367 | */ | ||
| 368 | public function setTranslatable($translatable) | ||
| 376 | |||
| 377 | /** | ||
| 378 | * get translatable flag | ||
| 379 | * | ||
| 380 | * @return boolean | ||
| 381 | */ | ||
| 382 | public function isTranslatable() | ||
| 391 | |||
| 392 | /** | ||
| 393 | * set a array of urls that can extref refer to | ||
| 394 | * | ||
| 395 | * @param array $refCollection urls | ||
| 396 | * | ||
| 397 | * @return void | ||
| 398 | */ | ||
| 399 | public function setRefCollection(array $refCollection) | ||
| 403 | |||
| 404 | /** | ||
| 405 | * get a collection of urls that can extref refer to | ||
| 406 | * | ||
| 407 | * @return array | ||
| 408 | */ | ||
| 409 | public function getRefCollection() | ||
| 418 | |||
| 419 | /** | ||
| 420 | * set an array of possible event names | ||
| 421 | * | ||
| 422 | * @param array $eventNames event names | ||
| 423 | * | ||
| 424 | * @return void | ||
| 425 | */ | ||
| 426 | public function setEventNames(array $eventNames) | ||
| 430 | |||
| 431 | /** | ||
| 432 | * get a collection of possible event names | ||
| 433 | * | ||
| 434 | * @return array | ||
| 435 | */ | ||
| 436 | public function getEventNames() | ||
| 445 | |||
| 446 | /** | ||
| 447 | * Set the readOnly flag | ||
| 448 | * | ||
| 449 | * @param bool $readOnly ReadOnly flag | ||
| 450 | * | ||
| 451 | * @return void | ||
| 452 | */ | ||
| 453 | public function setReadOnly($readOnly) | ||
| 457 | |||
| 458 | /** | ||
| 459 | * Get the readOnly flag. | ||
| 460 | * Returns null if the flag is set to false so the serializer will ignore it. | ||
| 461 | * | ||
| 462 | * @return bool|null true if readOnly isset to true or null if not | ||
| 463 | */ | ||
| 464 | public function getReadOnly() | ||
| 468 | } | ||
| 469 | 
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.