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:
| 1 | <?php | ||
| 13 | class File | ||
| 14 | { | ||
| 15 | protected $key; | ||
| 16 | protected $filesystem; | ||
| 17 | |||
| 18 | /** | ||
| 19 | * Content variable is lazy. It will not be read from filesystem until it's requested first time. | ||
| 20 | * | ||
| 21 | * @var mixed content | ||
| 22 | */ | ||
| 23 | protected $content = null; | ||
| 24 | |||
| 25 | /** | ||
| 26 | * @var array metadata in associative array. Only for adapters that support metadata | ||
| 27 | */ | ||
| 28 | protected $metadata = null; | ||
| 29 | |||
| 30 | /** | ||
| 31 | * Human readable filename (usually the end of the key). | ||
| 32 | * | ||
| 33 | * @var string name | ||
| 34 | */ | ||
| 35 | protected $name = null; | ||
| 36 | |||
| 37 | /** | ||
| 38 | * File size in bytes. | ||
| 39 | * | ||
| 40 | * @var int size | ||
| 41 | */ | ||
| 42 | protected $size = 0; | ||
| 43 | |||
| 44 | /** | ||
| 45 | * File date modified. | ||
| 46 | * | ||
| 47 | * @var int mtime | ||
| 48 | */ | ||
| 49 | protected $mtime = null; | ||
| 50 | |||
| 51 | /** | ||
| 52 | * @param string $key | ||
| 53 | * @param Filesystem $filesystem | ||
| 54 | */ | ||
| 55 | public function __construct($key, Filesystem $filesystem) | ||
| 56 |     { | ||
| 57 | $this->key = $key; | ||
| 58 | $this->name = $key; | ||
| 59 | $this->filesystem = $filesystem; | ||
| 60 | } | ||
| 61 | |||
| 62 | /** | ||
| 63 | * Returns the key. | ||
| 64 | * | ||
| 65 | * @return string | ||
| 66 | */ | ||
| 67 | public function getKey() | ||
| 68 |     { | ||
| 69 | return $this->key; | ||
| 70 | } | ||
| 71 | |||
| 72 | /** | ||
| 73 | * Returns the content. | ||
| 74 | * | ||
| 75 | * @throws FileNotFound | ||
| 76 | * | ||
| 77 | * @param array $metadata optional metadata which should be set when read | ||
| 78 | * | ||
| 79 | * @return string | ||
| 80 | */ | ||
| 81 | View Code Duplication | public function getContent($metadata = array()) | |
| 90 | |||
| 91 | /** | ||
| 92 | * @return string name of the file | ||
| 93 | */ | ||
| 94 | public function getName() | ||
| 98 | |||
| 99 | /** | ||
| 100 | * @return int size of the file | ||
| 101 | */ | ||
| 102 | public function getSize() | ||
| 103 |     { | ||
| 104 |         if ($this->size) { | ||
| 105 | return $this->size; | ||
| 106 | } | ||
| 107 | |||
| 108 |         try { | ||
| 109 | return $this->size = $this->filesystem->size($this->getKey()); | ||
| 110 |         } catch (FileNotFound $exception) { | ||
| 111 | } | ||
| 112 | |||
| 113 | return 0; | ||
| 114 | } | ||
| 115 | |||
| 116 | /** | ||
| 117 | * Returns the file modified time. | ||
| 118 | * | ||
| 119 | * @return int | ||
| 120 | */ | ||
| 121 | public function getMtime() | ||
| 122 |     { | ||
| 123 | return $this->mtime = $this->filesystem->mtime($this->key); | ||
| 124 | } | ||
| 125 | |||
| 126 | /** | ||
| 127 | * @param int $size size of the file | ||
| 128 | */ | ||
| 129 | public function setSize($size) | ||
| 130 |     { | ||
| 131 | $this->size = $size; | ||
| 132 | } | ||
| 133 | |||
| 134 | /** | ||
| 135 | * Sets the content. | ||
| 136 | * | ||
| 137 | * @param string $content | ||
| 138 | * @param array $metadata optional metadata which should be send when write | ||
| 139 | * | ||
| 140 | * @return int The number of bytes that were written into the file, or | ||
| 141 | * FALSE on failure | ||
| 142 | */ | ||
| 143 | View Code Duplication | public function setContent($content, $metadata = array()) | |
| 150 | |||
| 151 | /** | ||
| 152 | * @param string $name name of the file | ||
| 153 | */ | ||
| 154 | public function setName($name) | ||
| 155 |     { | ||
| 158 | |||
| 159 | /** | ||
| 160 | * Indicates whether the file exists in the filesystem. | ||
| 161 | * | ||
| 162 | * @return bool | ||
| 163 | */ | ||
| 164 | public function exists() | ||
| 168 | |||
| 169 | /** | ||
| 170 | * Deletes the file from the filesystem. | ||
| 171 | * | ||
| 172 | * @throws FileNotFound | ||
| 173 | * @throws \RuntimeException when cannot delete file | ||
| 174 | * | ||
| 175 | * @param array $metadata optional metadata which should be send when write | ||
| 176 | * | ||
| 177 | * @return bool TRUE on success | ||
| 178 | */ | ||
| 179 | public function delete($metadata = array()) | ||
| 185 | |||
| 186 | /** | ||
| 187 | * Creates a new file stream instance of the file. | ||
| 188 | * | ||
| 189 | * @return Stream | ||
| 190 | */ | ||
| 191 | public function createStream() | ||
| 195 | |||
| 196 | /** | ||
| 197 | * Rename the file and move it to its new location. | ||
| 198 | * | ||
| 199 | * @param string $newKey | ||
| 200 | */ | ||
| 201 | public function rename($newKey) | ||
| 207 | |||
| 208 | /** | ||
| 209 | * Sets the metadata array to be stored in adapters that can support it. | ||
| 210 | * | ||
| 211 | * @param array $metadata | ||
| 212 | * | ||
| 213 | * @return bool | ||
| 214 | */ | ||
| 215 | protected function setMetadata(array $metadata) | ||
| 225 | |||
| 226 | /** | ||
| 227 | * @return bool | ||
| 228 | */ | ||
| 229 | private function supportsMetadata() | ||
| 233 | } | ||
| 234 | 
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.