* @since 2.0 */ class Query extends Component implements QueryInterface { use QueryTrait; /** * @var array the fields of the results to return. For example, `['name', 'group_id']`. * The "_id" field is always returned. If not set, if means selecting all columns. * @see select() */ public $select = []; /** * @var string|array the collection to be selected from. If string considered as the name of the collection * inside the default database. If array - first element considered as the name of the database, * second - as name of collection inside that database * @see from() */ public $from; /** * Returns the Mongo collection for this query. * @param Connection $db Mongo connection. * @return Collection collection instance. */ public function getCollection($db = null) { if ($db === null) { $db = Yii::$app->getComponent('mongo'); } return $db->getCollection($this->from); } /** * Sets the list of fields of the results to return. * @param array $fields fields of the results to return. * @return static the query object itself. */ public function select(array $fields) { $this->select = $fields; return $this; } /** * Sets the collection to be selected from. * @param string|array the collection to be selected from. If string considered as the name of the collection * inside the default database. If array - first element considered as the name of the database, * second - as name of collection inside that database * @return static the query object itself. */ public function from($collection) { $this->from = $collection; return $this; } /** * Adds an additional WHERE condition to the existing one. * The new condition and the existing one will be joined using the 'AND' operator. * @param array $condition the new WHERE condition. Please refer to [[where()]] * on how to specify this parameter. * @return static the query object itself * @see where() * @see orWhere() */ public function andWhere($condition) { if (is_array($this->where)) { $this->where = array_merge($this->where, $condition); } else { $this->where = $condition; } return $this; } /** * Adds an additional WHERE condition to the existing one. * The new condition and the existing one will be joined using the 'OR' operator. * @param array $condition the new WHERE condition. Please refer to [[where()]] * on how to specify this parameter. * @return static the query object itself * @see where() * @see andWhere() */ public function orWhere($condition) { if (is_array($this->where)) { if (array_key_exists('or', $this->where) && count($this->where) == 1) { $this->where['or'][] = $condition; } else { $this->where = [ 'or' => [$this->where, $condition] ]; } } else { $this->where = $condition; } return $this; } /** * @param Connection $db the database connection used to execute the query. * @return \MongoCursor mongo cursor instance. */ protected function buildCursor($db = null) { $where = $this->where; if (!is_array($where)) { $where = []; } $selectFields = []; if (!empty($this->select)) { foreach ($this->select as $fieldName) { $selectFields[$fieldName] = true; } } $cursor = $this->getCollection($db)->find($where, $selectFields); if (!empty($this->orderBy)) { $sort = []; foreach ($this->orderBy as $fieldName => $sortOrder) { $sort[$fieldName] = $sortOrder === SORT_DESC ? \MongoCollection::DESCENDING : \MongoCollection::ASCENDING; } $cursor->sort($sort); } $cursor->limit($this->limit); $cursor->skip($this->offset); return $cursor; } /** * Executes the query and returns all results as an array. * @param Connection $db the database connection used to execute the query. * If this parameter is not given, the `db` application component will be used. * @return array the query results. If the query results in nothing, an empty array will be returned. */ public function all($db = null) { $cursor = $this->buildCursor($db); $result = []; foreach ($cursor as $row) { if ($this->indexBy !== null) { if (is_string($this->indexBy)) { $key = $row[$this->indexBy]; } else { $key = call_user_func($this->indexBy, $row); } $result[$key] = $row; } else { $result[] = $row; } } return $result; } /** * Executes the query and returns a single row of result. * @param Connection $db the database connection used to execute the query. * If this parameter is not given, the `db` application component will be used. * @return array|boolean the first row (in terms of an array) of the query result. False is returned if the query * results in nothing. */ public function one($db = null) { $cursor = $this->buildCursor($db); return $cursor->getNext(); } /** * Returns the number of records. * @param string $q the COUNT expression. Defaults to '*'. * @param Connection $db the database connection used to execute the query. * If this parameter is not given, the `db` application component will be used. * @return integer number of records */ public function count($q = '*', $db = null) { $cursor = $this->buildCursor($db); return $cursor->count(); } /** * Returns a value indicating whether the query result contains any row of data. * @param Connection $db the database connection used to execute the query. * If this parameter is not given, the `db` application component will be used. * @return boolean whether the query result contains any row of data. */ public function exists($db = null) { return $this->one($db) !== null; } }