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 | ||
| 10 | class Query extends QueryHandler implements QueryInterface | ||
| 11 | { | ||
| 12 | /** @var string Class name for interacting with database */ | ||
| 13 | protected $class_name; | ||
| 14 | |||
| 15 | /** @var array Collection of entity field names for sorting order */ | ||
| 16 | protected $sorting = array(); | ||
| 17 | |||
| 18 | /** @var array Collection of entity field names for grouping query results */ | ||
| 19 | protected $grouping = array(); | ||
| 20 | |||
| 21 | /** @var array Collection of query results limitations */ | ||
| 22 | protected $limitation = array(); | ||
| 23 | |||
| 24 | /** @var Condition Query base entity condition group */ | ||
| 25 | protected $own_condition; | ||
| 26 | |||
| 27 | /** @var Condition Query entity condition group */ | ||
| 28 | protected $cConditionGroup; | ||
| 29 | |||
| 30 | /** @var Database Database instance */ | ||
| 31 | protected $database; | ||
| 32 | |||
| 33 | /** | ||
| 34 | * Query constructor. | ||
| 35 | * @param string|null $entity Entity identifier | ||
| 36 | * @param Database Database instance | ||
| 37 | * @throws EntityNotFound | ||
| 38 | */ | ||
| 39 | public function __construct($entity, Database &$database) | ||
| 40 |     { | ||
| 41 | $this->database = &$database; | ||
| 42 | $this->entity($entity); | ||
| 43 | $this->flush(); | ||
| 44 | } | ||
| 45 | |||
| 46 | /** | ||
| 47 | * Reset all query parameters | ||
| 48 | * @return self Chaining | ||
| 49 | */ | ||
| 50 | public function flush() | ||
| 51 |     { | ||
| 52 | $this->grouping = array(); | ||
| 53 | $this->limitation = array(); | ||
| 54 | $this->sorting = array(); | ||
| 55 | |||
| 56 | $this->cConditionGroup = new Condition(); | ||
| 57 | $this->own_condition = new Condition(); | ||
| 58 | |||
| 59 | return $this; | ||
| 60 | } | ||
| 61 | |||
| 62 | /** | ||
| 63 | * Proxy function for performing database request and get collection of database record objects. | ||
| 64 | * This method encapsulates all logic needed for query to be done before and after actual database | ||
| 65 | * manager request. | ||
| 66 | * | ||
| 67 | * @param string $fetcher Database manager fetching method | ||
| 68 | * @return mixed Return fetching function result | ||
| 69 | */ | ||
| 70 | protected function execute($fetcher = 'find') | ||
| 71 |     { | ||
| 72 | // Call handlers stack | ||
| 73 | $this->callHandlers(); | ||
| 74 | |||
| 75 | // Remove first argument | ||
| 76 | $args = func_get_args(); | ||
| 77 | array_shift($args); | ||
| 78 | |||
| 79 | /** @var RecordInterface[] $return Perform DB request */ | ||
| 80 | $return = call_user_func_array(array($this->database, $fetcher), $args); | ||
| 81 | |||
| 82 | // Clear this query | ||
| 83 | $this->flush(); | ||
| 84 | |||
| 85 | // Return bool or collection | ||
| 86 | return $return; | ||
| 87 | } | ||
| 88 | |||
| 89 | /** | ||
| 90 | * Perform database request and get collection of database record objects. | ||
| 91 | * | ||
| 92 | * @param mixed $return External variable to store query results | ||
| 93 | * @return mixed If no arguments passed returns query results collection, otherwise query success status | ||
| 94 | */ | ||
| 95 | View Code Duplication | public function exec(&$return = null) | |
|  | |||
| 96 |     { | ||
| 97 | /** @var RecordInterface[] $return Perform DB request */ | ||
| 98 |         $return = $this->execute('find', $this->class_name, $this); | ||
| 99 | |||
| 100 | // Return bool or collection | ||
| 101 | return func_num_args() ? sizeof($return) : $return; | ||
| 102 | } | ||
| 103 | |||
| 104 | /** | ||
| 105 | * Perform database request and get first record from results collection. | ||
| 106 | * | ||
| 107 | * @param mixed $return External variable to store query results | ||
| 108 | * @return mixed If no arguments passed returns query results first database record object, | ||
| 109 | * otherwise query success status | ||
| 110 | */ | ||
| 111 | public function first(&$return = null) | ||
| 112 |     { | ||
| 113 | // Add limitation | ||
| 114 | $this->limit(1); | ||
| 115 | |||
| 116 | /** @var RecordInterface[] $return Perform DB request */ | ||
| 117 |         $return = $this->execute('find', $this->class_name, $this); | ||
| 118 | $return = sizeof($return) ? array_shift($return) : null; | ||
| 119 | |||
| 120 | // Return bool or collection | ||
| 121 | return func_num_args() ? sizeof($return) : $return; | ||
| 122 | } | ||
| 123 | |||
| 124 | /** | ||
| 125 | * Perform database request and get array of record field values | ||
| 126 | * @see \samson\activerecord\Query::execute() | ||
| 127 | * @param string $fieldName Record field name to get value from | ||
| 128 | * @param string $return External variable to store query results | ||
| 129 | * @return mixed If no arguments passed returns query results first database record object, | ||
| 130 | * otherwise query success status | ||
| 131 | */ | ||
| 132 | View Code Duplication | public function fields($fieldName, &$return = null) | |
| 133 |     { | ||
| 134 | /** @var RecordInterface[] $return Perform DB request */ | ||
| 135 |         $return = $this->execute('fetchColumn', $this->class_name, $this, $fieldName); | ||
| 136 | |||
| 137 | // Return bool or collection | ||
| 138 | return func_num_args() > 1 ? sizeof($return) : $return; | ||
| 139 | } | ||
| 140 | |||
| 141 | /** | ||
| 142 | * Set query entity to work with. | ||
| 143 | * | ||
| 144 | * @param string $entity Entity identifier | ||
| 145 | * @return Query Chaining | ||
| 146 | * @throws EntityNotFound | ||
| 147 | */ | ||
| 148 | View Code Duplication | public function entity($entity) | |
| 149 |     { | ||
| 150 |         if (class_exists($entity)) { | ||
| 151 | $this->flush(); | ||
| 152 | $this->class_name = $entity; | ||
| 153 |         } else { | ||
| 154 |             throw new EntityNotFound('['.$entity.'] not found'); | ||
| 155 | } | ||
| 156 | |||
| 157 | return $this; | ||
| 158 | } | ||
| 159 | |||
| 160 | /** | ||
| 161 | * Get correct query condition depending on entity field name. | ||
| 162 | * If base entity has field with this name - use base entity condition | ||
| 163 | * group, otherwise default condition group. | ||
| 164 | * | ||
| 165 | * @param string $fieldName Entity field name | ||
| 166 | * @return Condition Correct query condition group | ||
| 167 | */ | ||
| 168 | protected function &conditionGroup($fieldName) | ||
| 169 |     { | ||
| 170 |         if (property_exists($this->class_name, $fieldName)) { | ||
| 171 | // Add this condition to base entity condition group | ||
| 172 | return $this->own_condition; | ||
| 173 | } | ||
| 174 | |||
| 175 | return $this->cConditionGroup; | ||
| 176 | } | ||
| 177 | |||
| 178 | /** | ||
| 179 | * Add query condition as prepared Condition instance. | ||
| 180 | * | ||
| 181 | * @param ConditionInterface $condition Condition to be added | ||
| 182 | * @return self Chaining | ||
| 183 | */ | ||
| 184 | public function whereCondition(ConditionInterface $condition) | ||
| 199 | |||
| 200 | /** | ||
| 201 | * Add condition to current query. | ||
| 202 | * | ||
| 203 | * @param string $fieldName Entity field name | ||
| 204 | * @param string $fieldValue Value | ||
| 205 | * @param string $relation Relation between field name and its value | ||
| 206 | * @return self Chaining | ||
| 207 | */ | ||
| 208 | public function where($fieldName, $fieldValue = null, $relation = '=') | ||
| 232 | |||
| 233 | /** | ||
| 234 | * Join entity to query. | ||
| 235 | * | ||
| 236 | * @param string $entityName Entity identifier | ||
| 237 | * @return self Chaining | ||
| 238 | */ | ||
| 239 | public function join($entityName) | ||
| 247 | |||
| 248 | /** | ||
| 249 | * Add query result grouping. | ||
| 250 | * | ||
| 251 | * @param string $fieldName Entity field identifier for grouping | ||
| 252 | * @return self Chaining | ||
| 253 | */ | ||
| 254 | public function groupBy($fieldName) | ||
| 261 | |||
| 262 | /** | ||
| 263 | * Add query result quantity limitation. | ||
| 264 | * | ||
| 265 | * @param int $offset Resulting offset | ||
| 266 | * @param null|int $quantity Amount of RecordInterface object to return | ||
| 267 | * @return self Chaining | ||
| 268 | */ | ||
| 269 | public function limit($offset, $quantity = null) | ||
| 276 | |||
| 277 | /** | ||
| 278 | * Add query result sorting. | ||
| 279 | * | ||
| 280 | * @param string $fieldName Entity field identifier for worting | ||
| 281 | * @param string $order Sorting order | ||
| 282 | * @return self Chaining | ||
| 283 | */ | ||
| 284 | public function orderBy($fieldName, $order = 'ASC') | ||
| 291 | } | ||
| 292 | 
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.