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 MongoCollection 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 MongoCollection, and based on these observations, apply Extract Interface, too.
| 1 | <?php | ||
| 25 | class MongoCollection | ||
| 1 ignored issue–
                            show | |||
| 26 | { | ||
| 27 | use Helper\ReadPreference; | ||
| 28 | use Helper\WriteConcern; | ||
| 29 | use Helper\Slave; | ||
| 30 | |||
| 31 | const ASCENDING = 1; | ||
| 32 | const DESCENDING = -1; | ||
| 33 | |||
| 34 | /** | ||
| 35 | * @var MongoDB | ||
| 36 | */ | ||
| 37 | public $db = NULL; | ||
| 38 | |||
| 39 | /** | ||
| 40 | * @var string | ||
| 41 | */ | ||
| 42 | protected $name; | ||
| 43 | |||
| 44 | /** | ||
| 45 | * @var \MongoDB\Collection | ||
| 46 | */ | ||
| 47 | protected $collection; | ||
| 48 | |||
| 49 | /** | ||
| 50 | * Creates a new collection | ||
| 51 | * @link http://www.php.net/manual/en/mongocollection.construct.php | ||
| 52 | * @param MongoDB $db Parent database. | ||
| 53 | * @param string $name Name for this collection. | ||
| 54 | * @throws Exception | ||
| 55 | * @return MongoCollection | ||
| 56 | */ | ||
| 57 | public function __construct(MongoDB $db, $name) | ||
| 67 | |||
| 68 | /** | ||
| 69 | * Gets the underlying collection for this object | ||
| 70 | * | ||
| 71 | * @internal This part is not of the ext-mongo API and should not be used | ||
| 72 | * @return \MongoDB\Collection | ||
| 73 | */ | ||
| 74 | public function getCollection() | ||
| 78 | |||
| 79 | /** | ||
| 80 | * String representation of this collection | ||
| 81 | * @link http://www.php.net/manual/en/mongocollection.--tostring.php | ||
| 82 | * @return string Returns the full name of this collection. | ||
| 83 | */ | ||
| 84 | public function __toString() | ||
| 88 | |||
| 89 | /** | ||
| 90 | * Gets a collection | ||
| 91 | * @link http://www.php.net/manual/en/mongocollection.get.php | ||
| 92 | * @param string $name The next string in the collection name. | ||
| 93 | * @return MongoCollection | ||
| 94 | */ | ||
| 95 | public function __get($name) | ||
| 104 | |||
| 105 | /** | ||
| 106 | * @param string $name | ||
| 107 | * @param mixed $value | ||
| 108 | */ | ||
| 109 | View Code Duplication | public function __set($name, $value) | |
| 116 | |||
| 117 | /** | ||
| 118 | * @link http://www.php.net/manual/en/mongocollection.aggregate.php | ||
| 119 | * @param array $pipeline | ||
| 120 | * @param array $op | ||
| 121 | * @return array | ||
| 122 | */ | ||
| 123 | public function aggregate(array $pipeline, array $op = []) | ||
| 152 | |||
| 153 | /** | ||
| 154 | * @link http://php.net/manual/en/mongocollection.aggregatecursor.php | ||
| 155 | * @param array $pipeline | ||
| 156 | * @param array $options | ||
| 157 | * @return MongoCommandCursor | ||
| 158 | */ | ||
| 159 | public function aggregateCursor(array $pipeline, array $options = []) | ||
| 180 | |||
| 181 | /** | ||
| 182 | * Returns this collection's name | ||
| 183 | * @link http://www.php.net/manual/en/mongocollection.getname.php | ||
| 184 | * @return string | ||
| 185 | */ | ||
| 186 | public function getName() | ||
| 190 | |||
| 191 | /** | ||
| 192 |      * {@inheritdoc} | ||
| 193 | */ | ||
| 194 | public function setSlaveOkay($ok = true) | ||
| 200 | |||
| 201 | /** | ||
| 202 |      * {@inheritdoc} | ||
| 203 | */ | ||
| 204 | public function setReadPreference($readPreference, $tags = null) | ||
| 211 | |||
| 212 | /** | ||
| 213 |      * {@inheritdoc} | ||
| 214 | */ | ||
| 215 | public function setWriteConcern($wstring, $wtimeout = 0) | ||
| 222 | |||
| 223 | /** | ||
| 224 | * Drops this collection | ||
| 225 | * @link http://www.php.net/manual/en/mongocollection.drop.php | ||
| 226 | * @return array Returns the database response. | ||
| 227 | */ | ||
| 228 | public function drop() | ||
| 232 | |||
| 233 | /** | ||
| 234 | * Validates this collection | ||
| 235 | * @link http://www.php.net/manual/en/mongocollection.validate.php | ||
| 236 | * @param bool $scan_data Only validate indices, not the base collection. | ||
| 237 | * @return array Returns the database's evaluation of this object. | ||
| 238 | */ | ||
| 239 | public function validate($scan_data = FALSE) | ||
| 252 | |||
| 253 | /** | ||
| 254 | * Inserts an array into the collection | ||
| 255 | * @link http://www.php.net/manual/en/mongocollection.insert.php | ||
| 256 | * @param array|object $a | ||
| 257 | * @param array $options | ||
| 258 | * @throws MongoException if the inserted document is empty or if it contains zero-length keys. Attempting to insert an object with protected and private properties will cause a zero-length key error. | ||
| 259 | * @throws MongoCursorException if the "w" option is set and the write fails. | ||
| 260 | * @throws MongoCursorTimeoutException if the "w" option is set to a value greater than one and the operation takes longer than MongoCursor::$timeout milliseconds to complete. This does not kill the operation on the server, it is a client-side timeout. The operation in MongoCollection::$wtimeout is milliseconds. | ||
| 261 | * @return bool|array Returns an array containing the status of the insertion if the "w" option is set. | ||
| 262 | */ | ||
| 263 | public function insert($a, array $options = array()) | ||
| 267 | |||
| 268 | /** | ||
| 269 | * Inserts multiple documents into this collection | ||
| 270 | * @link http://www.php.net/manual/en/mongocollection.batchinsert.php | ||
| 271 | * @param array $a An array of arrays. | ||
| 272 | * @param array $options Options for the inserts. | ||
| 273 | * @throws MongoCursorException | ||
| 274 |      * @return mixed f "safe" is set, returns an associative array with the status of the inserts ("ok") and any error that may have occured ("err"). Otherwise, returns TRUE if the batch insert was successfully sent, FALSE otherwise. | ||
| 275 | */ | ||
| 276 | public function batchInsert(array $a, array $options = array()) | ||
| 280 | |||
| 281 | /** | ||
| 282 | * Update records based on a given criteria | ||
| 283 | * @link http://www.php.net/manual/en/mongocollection.update.php | ||
| 284 | * @param array $criteria Description of the objects to update. | ||
| 285 | * @param array $newobj The object with which to update the matching records. | ||
| 286 | * @param array $options This parameter is an associative array of the form | ||
| 287 |      *        array("optionname" => boolean, ...). | ||
| 288 | * | ||
| 289 | * Currently supported options are: | ||
| 290 | * "upsert": If no document matches $$criteria, a new document will be created from $$criteria and $$new_object (see upsert example). | ||
| 291 | * | ||
| 292 | * "multiple": All documents matching $criteria will be updated. MongoCollection::update has exactly the opposite behavior of MongoCollection::remove- it updates one document by | ||
| 293 | * default, not all matching documents. It is recommended that you always specify whether you want to update multiple documents or a single document, as the | ||
| 294 | * database may change its default behavior at some point in the future. | ||
| 295 | * | ||
| 296 | * "safe" Can be a boolean or integer, defaults to false. If false, the program continues executing without waiting for a database response. If true, the program will wait for | ||
| 297 | * the database response and throw a MongoCursorException if the update did not succeed. If you are using replication and the master has changed, using "safe" will make the driver | ||
| 298 | * disconnect from the master, throw and exception, and attempt to find a new master on the next operation (your application must decide whether or not to retry the operation on the new master). | ||
| 299 | * If you do not use "safe" with a replica set and the master changes, there will be no way for the driver to know about the change so it will continuously and silently fail to write. | ||
| 300 | * If safe is an integer, will replicate the update to that many machines before returning success (or throw an exception if the replication times out, see wtimeout). | ||
| 301 | * This overrides the w variable set on the collection. | ||
| 302 | * | ||
| 303 | * "fsync": Boolean, defaults to false. Forces the update to be synced to disk before returning success. If true, a safe update is implied and will override setting safe to false. | ||
| 304 | * | ||
| 305 | * "timeout" Integer, defaults to MongoCursor::$timeout. If "safe" is set, this sets how long (in milliseconds) for the client to wait for a database response. If the database does | ||
| 306 | * not respond within the timeout period, a MongoCursorTimeoutException will be thrown | ||
| 307 | * @throws MongoCursorException | ||
| 308 | * @return boolean | ||
| 309 | */ | ||
| 310 | public function update(array $criteria , array $newobj, array $options = array()) | ||
| 318 | |||
| 319 | /** | ||
| 320 | * (PECL mongo >= 0.9.0)<br/> | ||
| 321 | * Remove records from this collection | ||
| 322 | * @link http://www.php.net/manual/en/mongocollection.remove.php | ||
| 323 | * @param array $criteria [optional] <p>Query criteria for the documents to delete.</p> | ||
| 324 | * @param array $options [optional] <p>An array of options for the remove operation. Currently available options | ||
| 325 | * include: | ||
| 326 | * </p><ul> | ||
| 327 |      * <li><p><em>"w"</em></p><p>See {@link http://www.php.net/manual/en/mongo.writeconcerns.php Write Concerns}. The default value for <b>MongoClient</b> is <em>1</em>.</p></li> | ||
| 328 | * <li> | ||
| 329 | * <p> | ||
| 330 | * <em>"justOne"</em> | ||
| 331 | * </p> | ||
| 332 | * <p> | ||
| 333 | * Specify <strong><code>TRUE</code></strong> to limit deletion to just one document. If <strong><code>FALSE</code></strong> or | ||
| 334 | * omitted, all documents matching the criteria will be deleted. | ||
| 335 | * </p> | ||
| 336 | * </li> | ||
| 337 | * <li><p><em>"fsync"</em></p><p>Boolean, defaults to <b>FALSE</b>. If journaling is enabled, it works exactly like <em>"j"</em>. If journaling is not enabled, the write operation blocks until it is synced to database files on disk. If <strong><code>TRUE</code></strong>, an acknowledged insert is implied and this option will override setting <em>"w"</em> to <em>0</em>.</p><blockquote class="note"><p><strong class="note">Note</strong>: <span class="simpara">If journaling is enabled, users are strongly encouraged to use the <em>"j"</em> option instead of <em>"fsync"</em>. Do not use <em>"fsync"</em> and <em>"j"</em> simultaneously, as that will result in an error.</p></blockquote></li> | ||
| 338 | * <li><p><em>"j"</em></p><p>Boolean, defaults to <b>FALSE</b>. Forces the write operation to block until it is synced to the journal on disk. If <strong><code>TRUE</code></strong>, an acknowledged write is implied and this option will override setting <em>"w"</em> to <em>0</em>.</p><blockquote class="note"><p><strong class="note">Note</strong>: <span class="simpara">If this option is used and journaling is disabled, MongoDB 2.6+ will raise an error and the write will fail; older server versions will simply ignore the option.</p></blockquote></li> | ||
| 339 | * <li><p><em>"socketTimeoutMS"</em></p><p>This option specifies the time limit, in milliseconds, for socket communication. If the server does not respond within the timeout period, a <b>MongoCursorTimeoutException</b> will be thrown and there will be no way to determine if the server actually handled the write or not. A value of <em>-1</em> may be specified to block indefinitely. The default value for <b>MongoClient</b> is <em>30000</em> (30 seconds).</p></li> | ||
| 340 |      * <li><p><em>"w"</em></p><p>See {@link http://www.php.net/manual/en/mongo.writeconcerns.php Write Concerns }. The default value for <b>MongoClient</b> is <em>1</em>.</p></li> | ||
| 341 |      * <li><p><em>"wTimeoutMS"</em></p><p>This option specifies the time limit, in milliseconds, for {@link http://www.php.net/manual/en/mongo.writeconcerns.php write concern} acknowledgement. It is only applicable when <em>"w"</em> is greater than <em>1</em>, as the timeout pertains to replication. If the write concern is not satisfied within the time limit, a <a href="class.mongocursorexception.php" class="classname">MongoCursorException</a> will be thrown. A value of <em>0</em> may be specified to block indefinitely. The default value for {@link http://www.php.net/manual/en/class.mongoclient.php MongoClient} is <em>10000</em> (ten seconds).</p></li> | ||
| 342 | * </ul> | ||
| 343 | * | ||
| 344 | * <p> | ||
| 345 | * The following options are deprecated and should no longer be used: | ||
| 346 | * </p><ul> | ||
| 347 |      * <li><p><em>"safe"</em></p><p>Deprecated. Please use the {@link http://www.php.net/manual/en/mongo.writeconcerns.php write concern} <em>"w"</em> option.</p></li> | ||
| 348 | * <li><p><em>"timeout"</em></p><p>Deprecated alias for <em>"socketTimeoutMS"</em>.</p></li> | ||
| 349 | * <li><p><b>"wtimeout"</b></p><p>Deprecated alias for <em>"wTimeoutMS"</em>.</p></p> | ||
| 350 | * @throws MongoCursorException | ||
| 351 | * @throws MongoCursorTimeoutException | ||
| 352 | * @return bool|array <p>Returns an array containing the status of the removal if the | ||
| 353 | * <em>"w"</em> option is set. Otherwise, returns <b>TRUE</b>. | ||
| 354 | * </p> | ||
| 355 | * <p> | ||
| 356 | * Fields in the status array are described in the documentation for | ||
| 357 | * <b>MongoCollection::insert()</b>. | ||
| 358 | * </p> | ||
| 359 | */ | ||
| 360 | public function remove(array $criteria = array(), array $options = array()) | ||
| 361 |     { | ||
| 362 | $multiple = isset($options['justOne']) ? !$options['justOne'] : false; | ||
| 363 | // $multiple = !$options['justOne'] ?? false; | ||
| 364 | $method = $multiple ? 'deleteMany' : 'deleteOne'; | ||
| 365 | $criteria = TypeConverter::convertLegacyArrayToObject($criteria); | ||
| 366 | |||
| 367 | return $this->collection->$method($criteria, $options); | ||
| 368 | } | ||
| 369 | |||
| 370 | /** | ||
| 371 | * Querys this collection | ||
| 372 | * @link http://www.php.net/manual/en/mongocollection.find.php | ||
| 373 | * @param array $query The fields for which to search. | ||
| 374 | * @param array $fields Fields of the results to return. | ||
| 375 | * @return MongoCursor | ||
| 376 | */ | ||
| 377 | View Code Duplication | public function find(array $query = array(), array $fields = array()) | |
| 378 |     { | ||
| 379 | $cursor = new MongoCursor($this->db->getConnection(), (string)$this, $query, $fields); | ||
| 380 | $cursor->setReadPreference($this->getReadPreference()); | ||
| 381 | |||
| 382 | return $cursor; | ||
| 383 | } | ||
| 384 | |||
| 385 | /** | ||
| 386 | * Retrieve a list of distinct values for the given key across a collection | ||
| 387 | * @link http://www.php.net/manual/ru/mongocollection.distinct.php | ||
| 388 | * @param string $key The key to use. | ||
| 389 | * @param array $query An optional query parameters | ||
| 390 | * @return array|bool Returns an array of distinct values, or <b>FALSE</b> on failure | ||
| 391 | */ | ||
| 392 | public function distinct($key, array $query = []) | ||
| 396 | |||
| 397 | /** | ||
| 398 | * Update a document and return it | ||
| 399 | * @link http://www.php.net/manual/ru/mongocollection.findandmodify.php | ||
| 400 | * @param array $query The query criteria to search for. | ||
| 401 | * @param array $update The update criteria. | ||
| 402 | * @param array $fields Optionally only return these fields. | ||
| 403 | * @param array $options An array of options to apply, such as remove the match document from the DB and return it. | ||
| 404 | * @return array Returns the original document, or the modified document when new is set. | ||
| 405 | */ | ||
| 406 | public function findAndModify(array $query, array $update = NULL, array $fields = NULL, array $options = []) | ||
| 433 | |||
| 434 | /** | ||
| 435 | * Querys this collection, returning a single element | ||
| 436 | * @link http://www.php.net/manual/en/mongocollection.findone.php | ||
| 437 | * @param array $query The fields for which to search. | ||
| 438 | * @param array $fields Fields of the results to return. | ||
| 439 | * @return array|null | ||
| 440 | */ | ||
| 441 | public function findOne(array $query = array(), array $fields = array()) | ||
| 450 | |||
| 451 | /** | ||
| 452 | * Creates an index on the given field(s), or does nothing if the index already exists | ||
| 453 | * @link http://www.php.net/manual/en/mongocollection.createindex.php | ||
| 454 | * @param array $keys Field or fields to use as index. | ||
| 455 |      * @param array $options [optional] This parameter is an associative array of the form array("optionname" => <boolean>, ...). | ||
| 456 | * @return array Returns the database response. | ||
| 457 | */ | ||
| 458 | public function createIndex(array $keys, array $options = array()) | ||
| 462 | |||
| 463 | /** | ||
| 464 | * @deprecated Use MongoCollection::createIndex() instead. | ||
| 465 | * Creates an index on the given field(s), or does nothing if the index already exists | ||
| 466 | * @link http://www.php.net/manual/en/mongocollection.ensureindex.php | ||
| 467 | * @param array $keys Field or fields to use as index. | ||
| 468 |      * @param array $options [optional] This parameter is an associative array of the form array("optionname" => <boolean>, ...). | ||
| 469 | * @return boolean always true | ||
| 470 | */ | ||
| 471 | public function ensureIndex(array $keys, array $options = array()) | ||
| 476 | |||
| 477 | /** | ||
| 478 | * Deletes an index from this collection | ||
| 479 | * @link http://www.php.net/manual/en/mongocollection.deleteindex.php | ||
| 480 | * @param string|array $keys Field or fields from which to delete the index. | ||
| 481 | * @return array Returns the database response. | ||
| 482 | */ | ||
| 483 | public function deleteIndex($keys) | ||
| 494 | |||
| 495 | /** | ||
| 496 | * Delete all indexes for this collection | ||
| 497 | * @link http://www.php.net/manual/en/mongocollection.deleteindexes.php | ||
| 498 | * @return array Returns the database response. | ||
| 499 | */ | ||
| 500 | public function deleteIndexes() | ||
| 504 | |||
| 505 | /** | ||
| 506 | * Returns an array of index names for this collection | ||
| 507 | * @link http://www.php.net/manual/en/mongocollection.getindexinfo.php | ||
| 508 | * @return array Returns a list of index names. | ||
| 509 | */ | ||
| 510 | public function getIndexInfo() | ||
| 517 | |||
| 518 | /** | ||
| 519 | * Counts the number of documents in this collection | ||
| 520 | * @link http://www.php.net/manual/en/mongocollection.count.php | ||
| 521 | * @param array|stdClass $query | ||
| 522 | * @return int Returns the number of documents matching the query. | ||
| 523 | */ | ||
| 524 | public function count($query = array()) | ||
| 528 | |||
| 529 | /** | ||
| 530 | * Saves an object to this collection | ||
| 531 | * @link http://www.php.net/manual/en/mongocollection.save.php | ||
| 532 | * @param array|object $a Array to save. If an object is used, it may not have protected or private properties. | ||
| 533 | * Note: If the parameter does not have an _id key or property, a new MongoId instance will be created and assigned to it. | ||
| 534 | * See MongoCollection::insert() for additional information on this behavior. | ||
| 535 | * @param array $options Options for the save. | ||
| 536 | * <dl> | ||
| 537 | * <dt>"w" | ||
| 538 | * <dd>See WriteConcerns. The default value for MongoClient is 1. | ||
| 539 | * <dt>"fsync" | ||
| 540 | * <dd>Boolean, defaults to FALSE. Forces the insert to be synced to disk before returning success. If TRUE, an acknowledged insert is implied and will override setting w to 0. | ||
| 541 | * <dt>"timeout" | ||
| 542 | * <dd>Integer, defaults to MongoCursor::$timeout. If "safe" is set, this sets how long (in milliseconds) for the client to wait for a database response. If the database does not respond within the timeout period, a MongoCursorTimeoutException will be thrown. | ||
| 543 | * <dt>"safe" | ||
| 544 | * <dd>Deprecated. Please use the WriteConcern w option. | ||
| 545 | * </dl> | ||
| 546 | * @throws MongoException if the inserted document is empty or if it contains zero-length keys. Attempting to insert an object with protected and private properties will cause a zero-length key error. | ||
| 547 | * @throws MongoCursorException if the "w" option is set and the write fails. | ||
| 548 | * @throws MongoCursorTimeoutException if the "w" option is set to a value greater than one and the operation takes longer than MongoCursor::$timeout milliseconds to complete. This does not kill the operation on the server, it is a client-side timeout. The operation in MongoCollection::$wtimeout is milliseconds. | ||
| 549 | * @return array|boolean If w was set, returns an array containing the status of the save. | ||
| 550 | * Otherwise, returns a boolean representing if the array was not empty (an empty array will not be inserted). | ||
| 551 | */ | ||
| 552 | public function save($a, array $options = array()) | ||
| 568 | |||
| 569 | /** | ||
| 570 | * Creates a database reference | ||
| 571 | * @link http://www.php.net/manual/en/mongocollection.createdbref.php | ||
| 572 | * @param array $a Object to which to create a reference. | ||
| 573 | * @return array Returns a database reference array. | ||
| 574 | */ | ||
| 575 | public function createDBRef(array $a) | ||
| 579 | |||
| 580 | /** | ||
| 581 | * Fetches the document pointed to by a database reference | ||
| 582 | * @link http://www.php.net/manual/en/mongocollection.getdbref.php | ||
| 583 | * @param array $ref A database reference. | ||
| 584 | * @return array Returns the database document pointed to by the reference. | ||
| 585 | */ | ||
| 586 | public function getDBRef(array $ref) | ||
| 590 | |||
| 591 | /** | ||
| 592 | * @param mixed $keys | ||
| 593 | * @static | ||
| 594 | * @return string | ||
| 595 | */ | ||
| 596 | protected static function toIndexString($keys) | ||
| 604 | |||
| 605 | /** | ||
| 606 | * Performs an operation similar to SQL's GROUP BY command | ||
| 607 | * @link http://www.php.net/manual/en/mongocollection.group.php | ||
| 608 | * @param mixed $keys Fields to group by. If an array or non-code object is passed, it will be the key used to group results. | ||
| 609 | * @param array $initial Initial value of the aggregation counter object. | ||
| 610 | * @param MongoCode $reduce A function that aggregates (reduces) the objects iterated. | ||
| 611 | * @param array $condition An condition that must be true for a row to be considered. | ||
| 612 | * @return array | ||
| 613 | */ | ||
| 614 | public function group($keys, array $initial, $reduce, array $condition = []) | ||
| 654 | |||
| 655 | // public function parallelCollectionScan(int $num_cursors) | ||
| 656 |     // { | ||
| 657 | // $command = [ | ||
| 658 | // 'parallelCollectionScan' => $this->name, | ||
| 659 | // 'numCursors' => $num_cursors, | ||
| 660 | // ]; | ||
| 661 | // $result = $this->db->command($command, [], $hash); | ||
| 662 |     //     if ($result) { | ||
| 663 | // return $result; | ||
| 664 | // } | ||
| 665 | // return false; | ||
| 666 | // } | ||
| 667 | |||
| 668 | protected function notImplemented() | ||
| 672 | |||
| 673 | /** | ||
| 674 | * @return \MongoDB\Collection | ||
| 675 | */ | ||
| 676 | View Code Duplication | private function createCollectionObject() | |
| 689 | } | ||
| 690 | |||
| 691 | 
You can fix this by adding a namespace to your class:
When choosing a vendor namespace, try to pick something that is not too generic to avoid conflicts with other libraries.