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 Client 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 Client, and based on these observations, apply Extract Interface, too.
| 1 | <?php | ||
| 16 | class Client implements Interfaces\ClientInterface | ||
| 17 | { | ||
| 18 | use SocketTrait, ShortsTrait; | ||
| 19 | |||
| 20 | /** | ||
| 21 | * Configuration of connection | ||
| 22 | * | ||
| 23 | * @var \RouterOS\Config | ||
| 24 | */ | ||
| 25 | private $_config; | ||
| 26 | |||
| 27 | /** | ||
| 28 | * API communication object | ||
| 29 | * | ||
| 30 | * @var \RouterOS\APIConnector | ||
| 31 | */ | ||
| 32 | |||
| 33 | private $_connector; | ||
| 34 | |||
| 35 | /** | ||
| 36 | * Client constructor. | ||
| 37 | * | ||
| 38 | * @param array|\RouterOS\Config $config | ||
| 39 | * | ||
| 40 | * @throws \RouterOS\Exceptions\ClientException | ||
| 41 | * @throws \RouterOS\Exceptions\ConfigException | ||
| 42 | * @throws \RouterOS\Exceptions\QueryException | ||
| 43 | */ | ||
| 44 | 17 | public function __construct($config) | |
| 64 | |||
| 65 | /** | ||
| 66 | * Get some parameter from config | ||
| 67 | * | ||
| 68 | * @param string $parameter Name of required parameter | ||
| 69 | * | ||
| 70 | * @return mixed | ||
| 71 | * @throws \RouterOS\Exceptions\ConfigException | ||
| 72 | */ | ||
| 73 | 16 | private function config(string $parameter) | |
| 77 | |||
| 78 | /** | ||
| 79 | * Send write query to RouterOS | ||
| 80 | * | ||
| 81 | * @param string|array|\RouterOS\Query $query | ||
| 82 | * | ||
| 83 | * @return \RouterOS\Client | ||
| 84 | * @throws \RouterOS\Exceptions\QueryException | ||
| 85 | * @deprecated | ||
| 86 | * @codeCoverageIgnore | ||
| 87 | */ | ||
| 88 | public function write($query): Client | ||
| 104 | |||
| 105 | /** | ||
| 106 | * Send write query to RouterOS (modern version of write) | ||
| 107 | * | ||
| 108 | * @param string|Query $endpoint Path of API query or Query object | ||
| 109 | * @param array|null $where List of where filters | ||
| 110 | * @param string|null $operations Some operations which need make on response | ||
| 111 | * @param string|null $tag Mark query with tag | ||
| 112 | * | ||
| 113 | * @return \RouterOS\Client | ||
| 114 | * @throws \RouterOS\Exceptions\QueryException | ||
| 115 | * @throws \RouterOS\Exceptions\ClientException | ||
| 116 | * @since 1.0.0 | ||
| 117 | */ | ||
| 118 | 14 | public function query($endpoint, array $where = null, string $operations = null, string $tag = null): Client | |
| 190 | |||
| 191 | /** | ||
| 192 | * Send write query object to RouterOS | ||
| 193 | * | ||
| 194 | * @param \RouterOS\Query $query | ||
| 195 | * | ||
| 196 | * @return \RouterOS\Client | ||
| 197 | * @throws \RouterOS\Exceptions\QueryException | ||
| 198 | * @since 1.0.0 | ||
| 199 | */ | ||
| 200 | 14 | private function writeRAW(Query $query): Client | |
| 212 | |||
| 213 | /** | ||
| 214 | * Read RAW response from RouterOS | ||
| 215 | * | ||
| 216 | * @return array | ||
| 217 | * @since 1.0.0 | ||
| 218 | */ | ||
| 219 | 14 | private function readRAW(): array | |
| 255 | |||
| 256 | /** | ||
| 257 | * Read answer from server after query was executed | ||
| 258 | * | ||
| 259 | * A Mikrotik reply is formed of blocks | ||
| 260 |      * Each block starts with a word, one of ('!re', '!trap', '!done', '!fatal') | ||
| 261 | * Each block end with an zero byte (empty line) | ||
| 262 | * Reply ends with a complete !done or !fatal block (ended with 'empty line') | ||
| 263 | * A !fatal block precedes TCP connexion close | ||
| 264 | * | ||
| 265 | * @param bool $parse | ||
| 266 | * | ||
| 267 | * @return mixed | ||
| 268 | */ | ||
| 269 | 14 | public function read(bool $parse = true) | |
| 277 | |||
| 278 | /** | ||
| 279 | * Read using Iterators to improve performance on large dataset | ||
| 280 | * | ||
| 281 | * @return \RouterOS\ResponseIterator | ||
| 282 | * @since 1.0.0 | ||
| 283 | */ | ||
| 284 | 4 | public function readAsIterator(): ResponseIterator | |
| 288 | |||
| 289 | /** | ||
| 290 | * This method was created by memory save reasons, it convert response | ||
| 291 | * from RouterOS to readable array in safe way. | ||
| 292 | * | ||
| 293 | * @param array $raw Array RAW response from server | ||
| 294 | * | ||
| 295 | * @return mixed | ||
| 296 | * | ||
| 297 | * Based on RouterOSResponseArray solution by @arily | ||
| 298 | * | ||
| 299 | * @link https://github.com/arily/RouterOSResponseArray | ||
| 300 | * @since 1.0.0 | ||
| 301 | */ | ||
| 302 | 3 | private function rosario(array $raw): array | |
| 333 | |||
| 334 | /** | ||
| 335 | * Parse response from Router OS | ||
| 336 | * | ||
| 337 | * @param array $response Response data | ||
| 338 | * | ||
| 339 | * @return array Array with parsed data | ||
| 340 | */ | ||
| 341 | 4 | public function parseResponse(array $response): array | |
| 342 |     { | ||
| 343 | 4 | $result = []; | |
| 344 | 4 | $i = -1; | |
| 345 | 4 | $lines = \count($response); | |
| 346 | 4 |         foreach ($response as $key => $value) { | |
| 347 | 4 |             switch ($value) { | |
| 348 | 4 | case '!re': | |
| 349 | 2 | $i++; | |
| 350 | 2 | break; | |
| 351 | 4 | case '!fatal': | |
| 352 | 1 | $result = $response; | |
| 353 | 1 | break 2; | |
| 354 | 3 | case '!trap': | |
| 355 | 3 | case '!done': | |
| 356 | // Check for =ret=, .tag and any other following messages | ||
| 357 | 3 |                     for ($j = $key + 1; $j <= $lines; $j++) { | |
| 358 | // If we have lines after current one | ||
| 359 | 3 |                         if (isset($response[$j])) { | |
| 360 | 2 | $this->pregResponse($response[$j], $matches); | |
| 361 | 2 | View Code Duplication |                             if (isset($matches[1][0], $matches[2][0])) { | 
| 362 | 2 | $result['after'][$matches[1][0]] = $matches[2][0]; | |
| 363 | } | ||
| 364 | } | ||
| 365 | } | ||
| 366 | 3 | break 2; | |
| 367 | default: | ||
| 368 | 2 | $this->pregResponse($value, $matches); | |
| 369 | 2 | View Code Duplication |                     if (isset($matches[1][0], $matches[2][0])) { | 
| 370 | 2 | $result[$i][$matches[1][0]] = $matches[2][0]; | |
| 371 | } | ||
| 372 | 2 | break; | |
| 373 | } | ||
| 374 | } | ||
| 375 | 4 | return $result; | |
| 376 | } | ||
| 377 | |||
| 378 | /** | ||
| 379 | * Parse result from RouterOS by regular expression | ||
| 380 | * | ||
| 381 | * @param string $value | ||
| 382 | * @param array $matches | ||
| 383 | */ | ||
| 384 | 3 | private function pregResponse(string $value, &$matches) | |
| 388 | |||
| 389 | /** | ||
| 390 | * Authorization logic | ||
| 391 | * | ||
| 392 | * @param bool $legacyRetry Retry login if we detect legacy version of RouterOS | ||
| 393 | * | ||
| 394 | * @return bool | ||
| 395 | * @throws \RouterOS\Exceptions\ClientException | ||
| 396 | * @throws \RouterOS\Exceptions\ConfigException | ||
| 397 | * @throws \RouterOS\Exceptions\QueryException | ||
| 398 | */ | ||
| 399 | 14 | private function login(bool $legacyRetry = false): bool | |
| 400 |     { | ||
| 401 | // If legacy login scheme is enabled | ||
| 402 | 14 |         if ($this->config('legacy')) { | |
| 403 | // For the first we need get hash with salt | ||
| 404 | 1 |             $response = $this->query('/login')->read(); | |
| 405 | |||
| 406 | // Now need use this hash for authorization | ||
| 407 | 1 |             $query = new Query('/login', [ | |
| 408 | 1 |                 '=name=' . $this->config('user'), | |
| 409 | 1 |                 '=response=00' . md5(\chr(0) . $this->config('pass') . pack('H*', $response['after']['ret'])) | |
| 410 | ]); | ||
| 411 |         } else { | ||
| 412 | // Just login with our credentials | ||
| 413 | 13 |             $query = new Query('/login', [ | |
| 414 | 13 |                 '=name=' . $this->config('user'), | |
| 415 | 13 |                 '=password=' . $this->config('pass') | |
| 416 | ]); | ||
| 417 | |||
| 418 | // If we set modern auth scheme but router with legacy firmware then need to retry query, | ||
| 419 | // but need to prevent endless loop | ||
| 420 | 13 | $legacyRetry = true; | |
| 421 | } | ||
| 422 | |||
| 423 | // Execute query and get response | ||
| 424 | 14 | $response = $this->query($query)->read(false); | |
| 425 | |||
| 426 | // if: | ||
| 427 | // - we have more than one response | ||
| 428 | // - response is '!done' | ||
| 429 | // => problem with legacy version, swap it and retry | ||
| 430 | // Only tested with ROS pre 6.43, will test with post 6.43 => this could make legacy parameter obsolete? | ||
| 431 | 14 |         if ($legacyRetry && $this->isLegacy($response)) { | |
| 432 |             $this->_config->set('legacy', true); | ||
| 433 | return $this->login(); | ||
| 434 | } | ||
| 435 | |||
| 436 | // If RouterOS answered with invalid credentials then throw error | ||
| 437 | 14 |         if (!empty($response[0]) && $response[0] === '!trap') { | |
| 438 | 1 |             throw new ClientException('Invalid user name or password'); | |
| 439 | } | ||
| 440 | |||
| 441 | // Return true if we have only one line from server and this line is !done | ||
| 442 | 13 | return (1 === count($response)) && isset($response[0]) && ($response[0] === '!done'); | |
| 443 | } | ||
| 444 | |||
| 445 | /** | ||
| 446 | * Detect by login request if firmware is legacy | ||
| 447 | * | ||
| 448 | * @param array $response | ||
| 449 | * | ||
| 450 | * @return bool | ||
| 451 | * @throws ConfigException | ||
| 452 | */ | ||
| 453 | 13 | private function isLegacy(array &$response): bool | |
| 457 | |||
| 458 | /** | ||
| 459 | * Connect to socket server | ||
| 460 | * | ||
| 461 | * @return bool | ||
| 462 | * @throws \RouterOS\Exceptions\ClientException | ||
| 463 | * @throws \RouterOS\Exceptions\ConfigException | ||
| 464 | * @throws \RouterOS\Exceptions\QueryException | ||
| 465 | */ | ||
| 466 | 16 | private function connect(): bool | |
| 497 | } | ||
| 498 | 
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.