Complex classes like SftpAdapter 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 SftpAdapter, and based on these observations, apply Extract Interface, too.
| 1 | <?php | ||
| 17 | class SftpAdapter extends AbstractFtpAdapter | ||
| 18 | { | ||
| 19 | use StreamedCopyTrait; | ||
| 20 | |||
| 21 | /** | ||
| 22 | * @var int | ||
| 23 | */ | ||
| 24 | protected $port = 22; | ||
| 25 | |||
| 26 | /** | ||
| 27 | * @var string | ||
| 28 | */ | ||
| 29 | protected $privatekey; | ||
| 30 | |||
| 31 | /** | ||
| 32 | * @var bool | ||
| 33 | */ | ||
| 34 | protected $useAgent = false; | ||
| 35 | |||
| 36 | /** | ||
| 37 | * @var Agent | ||
| 38 | */ | ||
| 39 | private $agent; | ||
| 40 | |||
| 41 | /** | ||
| 42 | * @var array | ||
| 43 | */ | ||
| 44 | protected $configurable = ['host', 'port', 'username', 'password', 'useAgent', 'agent', 'timeout', 'root', 'privateKey', 'permPrivate', 'permPublic', 'directoryPerm', 'NetSftpConnection']; | ||
| 45 | |||
| 46 | /** | ||
| 47 | * @var array | ||
| 48 | */ | ||
| 49 | protected $statMap = ['mtime' => 'timestamp', 'size' => 'size']; | ||
| 50 | |||
| 51 | /** | ||
| 52 | * @var int | ||
| 53 | */ | ||
| 54 | protected $directoryPerm = 0744; | ||
| 55 | |||
| 56 | /** | ||
| 57 | * Prefix a path. | ||
| 58 | * | ||
| 59 | * @param string $path | ||
| 60 | * | ||
| 61 | * @return string | ||
| 62 | */ | ||
| 63 | 6 | protected function prefix($path) | |
| 67 | |||
| 68 | /** | ||
| 69 | * Set the private key (string or path to local file). | ||
| 70 | * | ||
| 71 | * @param string $key | ||
| 72 | * | ||
| 73 | * @return $this | ||
| 74 | */ | ||
| 75 | 9 | public function setPrivateKey($key) | |
| 81 | |||
| 82 | /** | ||
| 83 | * @param boolean $useAgent | ||
| 84 | * | ||
| 85 | * @return $this | ||
| 86 | */ | ||
| 87 | public function setUseAgent($useAgent) | ||
| 93 | |||
| 94 | /** | ||
| 95 | * @param Agent $agent | ||
| 96 | * | ||
| 97 | * @return $this | ||
| 98 | */ | ||
| 99 | public function setAgent(Agent $agent) | ||
| 105 | |||
| 106 | /** | ||
| 107 | * Set permissions for new directory | ||
| 108 | * | ||
| 109 | * @param int $directoryPerm | ||
| 110 | * | ||
| 111 | * @return $this | ||
| 112 | */ | ||
| 113 | 6 | public function setDirectoryPerm($directoryPerm) | |
| 119 | |||
| 120 | /** | ||
| 121 | * Get permissions for new directory | ||
| 122 | * | ||
| 123 | * @return int | ||
| 124 | */ | ||
| 125 | 3 | public function getDirectoryPerm() | |
| 129 | |||
| 130 | /** | ||
| 131 | * Inject the SFTP instance. | ||
| 132 | * | ||
| 133 | * @param SFTP $connection | ||
| 134 | * | ||
| 135 | * @return $this | ||
| 136 | */ | ||
| 137 | 21 | public function setNetSftpConnection(SFTP $connection) | |
| 143 | |||
| 144 | /** | ||
| 145 | * Connect. | ||
| 146 | */ | ||
| 147 | 12 | public function connect() | |
| 153 | |||
| 154 | /** | ||
| 155 | * Login. | ||
| 156 | * | ||
| 157 | * @throws LogicException | ||
| 158 | */ | ||
| 159 | 12 | protected function login() | |
| 171 | |||
| 172 | /** | ||
| 173 | * Set the connection root. | ||
| 174 | */ | ||
| 175 | 9 | protected function setConnectionRoot() | |
| 188 | |||
| 189 | /** | ||
| 190 | * Get the password, either the private key or a plain text password. | ||
| 191 | * | ||
| 192 | * @return Agent|RSA|string | ||
| 193 | */ | ||
| 194 | 15 | public function getAuthentication() | |
| 206 | |||
| 207 | /** | ||
| 208 | * Get the password, a plain text password. | ||
| 209 | * | ||
| 210 | * @return string | ||
| 211 | */ | ||
| 212 | 12 | public function getPassword() | |
| 216 | |||
| 217 | /** | ||
| 218 | * Get the private get with the password or private key contents. | ||
| 219 | * | ||
| 220 | * @return RSA | ||
| 221 | */ | ||
| 222 | 9 | public function getPrivateKey() | |
| 238 | |||
| 239 | /** | ||
| 240 | * @return Agent|bool | ||
| 241 | */ | ||
| 242 | public function getAgent() | ||
| 250 | |||
| 251 | /** | ||
| 252 | * List the contents of a directory. | ||
| 253 | * | ||
| 254 | * @param string $directory | ||
| 255 | * @param bool $recursive | ||
| 256 | * | ||
| 257 | * @return array | ||
| 258 | */ | ||
| 259 | 6 | protected function listDirectoryContents($directory, $recursive = true) | |
| 285 | |||
| 286 | /** | ||
| 287 | * Normalize a listing response. | ||
| 288 | * | ||
| 289 | * @param string $path | ||
| 290 | * @param array $object | ||
| 291 | * | ||
| 292 | * @return array | ||
| 293 | */ | ||
| 294 | 6 | protected function normalizeListingObject($path, array $object) | |
| 309 | |||
| 310 | /** | ||
| 311 | * Disconnect. | ||
| 312 | */ | ||
| 313 | 6 | public function disconnect() | |
| 317 | |||
| 318 | /** | ||
| 319 | * @inheritdoc | ||
| 320 | */ | ||
| 321 | 6 | public function write($path, $contents, Config $config) | |
| 329 | |||
| 330 | /** | ||
| 331 | * @inheritdoc | ||
| 332 | */ | ||
| 333 | 6 | public function writeStream($path, $resource, Config $config) | |
| 341 | |||
| 342 | /** | ||
| 343 | * Upload a file. | ||
| 344 | * | ||
| 345 | * @param string $path | ||
| 346 | * @param string|resource $contents | ||
| 347 | * @param Config $config | ||
| 348 | * @return bool | ||
| 349 | */ | ||
| 350 | 12 | public function upload($path, $contents, Config $config) | |
| 366 | |||
| 367 | /** | ||
| 368 | * @inheritdoc | ||
| 369 | */ | ||
| 370 | 6 | public function read($path) | |
| 380 | |||
| 381 | /** | ||
| 382 | * @inheritdoc | ||
| 383 | */ | ||
| 384 | 3 | public function readStream($path) | |
| 398 | |||
| 399 | /** | ||
| 400 | * @inheritdoc | ||
| 401 | */ | ||
| 402 | 3 | public function update($path, $contents, Config $config) | |
| 406 | |||
| 407 | /** | ||
| 408 | * @inheritdoc | ||
| 409 | */ | ||
| 410 | 3 | public function updateStream($path, $contents, Config $config) | |
| 414 | |||
| 415 | /** | ||
| 416 | * @inheritdoc | ||
| 417 | */ | ||
| 418 | 3 | public function delete($path) | |
| 424 | |||
| 425 | /** | ||
| 426 | * @inheritdoc | ||
| 427 | */ | ||
| 428 | 3 | public function rename($path, $newpath) | |
| 434 | |||
| 435 | /** | ||
| 436 | * @inheritdoc | ||
| 437 | */ | ||
| 438 | 3 | public function deleteDir($dirname) | |
| 444 | |||
| 445 | /** | ||
| 446 | * @inheritdoc | ||
| 447 | */ | ||
| 448 | 39 | public function has($path) | |
| 452 | |||
| 453 | /** | ||
| 454 | * @inheritdoc | ||
| 455 | */ | ||
| 456 | 48 | public function getMetadata($path) | |
| 471 | |||
| 472 | /** | ||
| 473 | * @inheritdoc | ||
| 474 | */ | ||
| 475 | 6 | public function getTimestamp($path) | |
| 479 | |||
| 480 | /** | ||
| 481 | * @inheritdoc | ||
| 482 | */ | ||
| 483 | 3 | public function getMimetype($path) | |
| 493 | |||
| 494 | /** | ||
| 495 | * @inheritdoc | ||
| 496 | */ | ||
| 497 | 3 | public function createDir($dirname, Config $config) | |
| 507 | |||
| 508 | /** | ||
| 509 | * @inheritdoc | ||
| 510 | */ | ||
| 511 | 6 | public function getVisibility($path) | |
| 515 | |||
| 516 | /** | ||
| 517 | * @inheritdoc | ||
| 518 | */ | ||
| 519 | 15 | public function setVisibility($path, $visibility) | |
| 531 | |||
| 532 | /** | ||
| 533 | * @inheritdoc | ||
| 534 | */ | ||
| 535 | 6 | public function isConnected() | |
| 543 | } | ||
| 544 | 
This error could be the result of:
1. Missing dependencies
PHP Analyzer uses your
composer.jsonfile (if available) to determine the dependencies of your project and to determine all the available classes and functions. It expects thecomposer.jsonto be in the root folder of your repository.Are you sure this class is defined by one of your dependencies, or did you maybe not list a dependency in either the
requireorrequire-devsection?2. Missing use statement
PHP does not complain about undefined classes in
ìnstanceofchecks. For example, the following PHP code will work perfectly fine:If you have not tested against this specific condition, such errors might go unnoticed.