Carsten Brandt
11 years ago
20 changed files with 2102 additions and 1053 deletions
@ -1,343 +0,0 @@ |
|||||||
<?php |
|
||||||
/** |
|
||||||
* @link http://www.yiiframework.com/ |
|
||||||
* @copyright Copyright © 2008 Yii Software LLC |
|
||||||
* @license http://www.yiiframework.com/license/ |
|
||||||
*/ |
|
||||||
|
|
||||||
namespace yii\db\elasticsearch; |
|
||||||
|
|
||||||
/** |
|
||||||
* ActiveQuery represents a DB query associated with an Active Record class. |
|
||||||
* |
|
||||||
* |
|
||||||
* @author Carsten Brandt <mail@cebe.cc> |
|
||||||
* @since 2.0 |
|
||||||
*/ |
|
||||||
class ActiveQuery extends \yii\base\Component |
|
||||||
{ |
|
||||||
/** |
|
||||||
* @var string the name of the ActiveRecord class. |
|
||||||
*/ |
|
||||||
public $modelClass; |
|
||||||
/** |
|
||||||
* @var array list of relations that this query should be performed with |
|
||||||
*/ |
|
||||||
public $with; |
|
||||||
/** |
|
||||||
* @var string the name of the column by which query results should be indexed by. |
|
||||||
* This is only used when the query result is returned as an array when calling [[all()]]. |
|
||||||
*/ |
|
||||||
public $indexBy; |
|
||||||
/** |
|
||||||
* @var boolean whether to return each record as an array. If false (default), an object |
|
||||||
* of [[modelClass]] will be created to represent each record. |
|
||||||
*/ |
|
||||||
public $asArray; |
|
||||||
/** |
|
||||||
* @var integer maximum number of records to be returned. If not set or less than 0, it means no limit. |
|
||||||
*/ |
|
||||||
public $limit; |
|
||||||
/** |
|
||||||
* @var integer zero-based offset from where the records are to be returned. |
|
||||||
* If not set, it means starting from the beginning. |
|
||||||
* If less than zero it means starting n elements from the end. |
|
||||||
*/ |
|
||||||
public $offset; |
|
||||||
/** |
|
||||||
* @var array array of primary keys of the records to find. |
|
||||||
*/ |
|
||||||
public $primaryKeys; |
|
||||||
|
|
||||||
/** |
|
||||||
* List of multiple pks must be zero based |
|
||||||
* |
|
||||||
* @param $primaryKeys |
|
||||||
* @return ActiveQuery |
|
||||||
*/ |
|
||||||
public function primaryKeys($primaryKeys) { |
|
||||||
if (is_array($primaryKeys) && isset($primaryKeys[0])) { |
|
||||||
$this->primaryKeys = $primaryKeys; |
|
||||||
} else { |
|
||||||
$this->primaryKeys = array($primaryKeys); |
|
||||||
} |
|
||||||
|
|
||||||
return $this; |
|
||||||
} |
|
||||||
|
|
||||||
/** |
|
||||||
* Executes query and returns all results as an array. |
|
||||||
* @return array the query results. If the query results in nothing, an empty array will be returned. |
|
||||||
*/ |
|
||||||
public function all() |
|
||||||
{ |
|
||||||
$modelClass = $this->modelClass; |
|
||||||
/** @var Connection $db */ |
|
||||||
$db = $modelClass::getDb(); |
|
||||||
if (($primaryKeys = $this->primaryKeys) === null) { |
|
||||||
$start = $this->offset === null ? 0 : $this->offset; |
|
||||||
$end = $this->limit === null ? -1 : $start + $this->limit; |
|
||||||
$primaryKeys = $db->executeCommand('LRANGE', array($modelClass::tableName(), $start, $end)); |
|
||||||
} |
|
||||||
$rows = array(); |
|
||||||
foreach($primaryKeys as $pk) { |
|
||||||
$key = $modelClass::tableName() . ':a:' . $modelClass::hashPk($pk); |
|
||||||
// get attributes |
|
||||||
$data = $db->executeCommand('HGETALL', array($key)); |
|
||||||
$row = array(); |
|
||||||
for($i=0;$i<count($data);) { |
|
||||||
$row[$data[$i++]] = $data[$i++]; |
|
||||||
} |
|
||||||
$rows[] = $row; |
|
||||||
} |
|
||||||
if ($rows !== array()) { |
|
||||||
$models = $this->createModels($rows); |
|
||||||
if (!empty($this->with)) { |
|
||||||
$this->populateRelations($models, $this->with); |
|
||||||
} |
|
||||||
return $models; |
|
||||||
} else { |
|
||||||
return array(); |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
/** |
|
||||||
* Executes query and returns a single row of result. |
|
||||||
* @return ActiveRecord|array|null a single row of query result. Depending on the setting of [[asArray]], |
|
||||||
* the query result may be either an array or an ActiveRecord object. Null will be returned |
|
||||||
* if the query results in nothing. |
|
||||||
*/ |
|
||||||
public function one() |
|
||||||
{ |
|
||||||
$modelClass = $this->modelClass; |
|
||||||
/** @var Connection $db */ |
|
||||||
$db = $modelClass::getDb(); |
|
||||||
if (($primaryKeys = $this->primaryKeys) === null) { |
|
||||||
$start = $this->offset === null ? 0 : $this->offset; |
|
||||||
$primaryKeys = $db->executeCommand('LRANGE', array($modelClass::tableName(), $start, $start + 1)); |
|
||||||
} |
|
||||||
$pk = reset($primaryKeys); |
|
||||||
$key = $modelClass::tableName() . ':a:' . $modelClass::hashPk($pk); |
|
||||||
// get attributes |
|
||||||
$data = $db->executeCommand('HGETALL', array($key)); |
|
||||||
if ($data === array()) { |
|
||||||
return null; |
|
||||||
} |
|
||||||
$row = array(); |
|
||||||
for($i=0;$i<count($data);) { |
|
||||||
$row[$data[$i++]] = $data[$i++]; |
|
||||||
} |
|
||||||
if (!$this->asArray) { |
|
||||||
/** @var $class ActiveRecord */ |
|
||||||
$class = $this->modelClass; |
|
||||||
$model = $class::create($row); |
|
||||||
if (!empty($this->with)) { |
|
||||||
$models = array($model); |
|
||||||
$this->populateRelations($models, $this->with); |
|
||||||
$model = $models[0]; |
|
||||||
} |
|
||||||
return $model; |
|
||||||
} else { |
|
||||||
return $row; |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
/** |
|
||||||
* Returns the number of records. |
|
||||||
* @param string $q the COUNT expression. Defaults to '*'. |
|
||||||
* Make sure you properly quote column names. |
|
||||||
* @return integer number of records |
|
||||||
*/ |
|
||||||
public function count() |
|
||||||
{ |
|
||||||
$modelClass = $this->modelClass; |
|
||||||
/** @var Connection $db */ |
|
||||||
$db = $modelClass::getDb(); |
|
||||||
return $db->executeCommand('LLEN', array($modelClass::tableName())); |
|
||||||
} |
|
||||||
|
|
||||||
/** |
|
||||||
* Returns the query result as a scalar value. |
|
||||||
* The value returned will be the first column in the first row of the query results. |
|
||||||
* @return string|boolean the value of the first column in the first row of the query result. |
|
||||||
* False is returned if the query result is empty. |
|
||||||
*/ |
|
||||||
public function scalar($column) |
|
||||||
{ |
|
||||||
$record = $this->one(); |
|
||||||
return $record->$column; |
|
||||||
} |
|
||||||
|
|
||||||
/** |
|
||||||
* Returns a value indicating whether the query result contains any row of data. |
|
||||||
* @return boolean whether the query result contains any row of data. |
|
||||||
*/ |
|
||||||
public function exists() |
|
||||||
{ |
|
||||||
return $this->one() !== null; |
|
||||||
} |
|
||||||
|
|
||||||
|
|
||||||
/** |
|
||||||
* Sets the [[asArray]] property. |
|
||||||
* TODO: refactor, it is duplicated from yii/db/ActiveQuery |
|
||||||
* @param boolean $value whether to return the query results in terms of arrays instead of Active Records. |
|
||||||
* @return ActiveQuery the query object itself |
|
||||||
*/ |
|
||||||
public function asArray($value = true) |
|
||||||
{ |
|
||||||
$this->asArray = $value; |
|
||||||
return $this; |
|
||||||
} |
|
||||||
|
|
||||||
/** |
|
||||||
* Sets the LIMIT part of the query. |
|
||||||
* TODO: refactor, it is duplicated from yii/db/Query |
|
||||||
* @param integer $limit the limit |
|
||||||
* @return Query the query object itself |
|
||||||
*/ |
|
||||||
public function limit($limit) |
|
||||||
{ |
|
||||||
$this->limit = $limit; |
|
||||||
return $this; |
|
||||||
} |
|
||||||
|
|
||||||
/** |
|
||||||
* Sets the OFFSET part of the query. |
|
||||||
* TODO: refactor, it is duplicated from yii/db/Query |
|
||||||
* @param integer $offset the offset |
|
||||||
* @return Query the query object itself |
|
||||||
*/ |
|
||||||
public function offset($offset) |
|
||||||
{ |
|
||||||
$this->offset = $offset; |
|
||||||
return $this; |
|
||||||
} |
|
||||||
|
|
||||||
/** |
|
||||||
* Specifies the relations with which this query should be performed. |
|
||||||
* |
|
||||||
* The parameters to this method can be either one or multiple strings, or a single array |
|
||||||
* of relation names and the optional callbacks to customize the relations. |
|
||||||
* |
|
||||||
* The followings are some usage examples: |
|
||||||
* |
|
||||||
* ~~~ |
|
||||||
* // find customers together with their orders and country |
|
||||||
* Customer::find()->with('orders', 'country')->all(); |
|
||||||
* // find customers together with their country and orders of status 1 |
|
||||||
* Customer::find()->with(array( |
|
||||||
* 'orders' => function($query) { |
|
||||||
* $query->andWhere('status = 1'); |
|
||||||
* }, |
|
||||||
* 'country', |
|
||||||
* ))->all(); |
|
||||||
* ~~~ |
|
||||||
* |
|
||||||
* TODO: refactor, it is duplicated from yii/db/ActiveQuery |
|
||||||
* @return ActiveQuery the query object itself |
|
||||||
*/ |
|
||||||
public function with() |
|
||||||
{ |
|
||||||
$this->with = func_get_args(); |
|
||||||
if (isset($this->with[0]) && is_array($this->with[0])) { |
|
||||||
// the parameter is given as an array |
|
||||||
$this->with = $this->with[0]; |
|
||||||
} |
|
||||||
return $this; |
|
||||||
} |
|
||||||
|
|
||||||
/** |
|
||||||
* Sets the [[indexBy]] property. |
|
||||||
* TODO: refactor, it is duplicated from yii/db/ActiveQuery |
|
||||||
* @param string $column the name of the column by which the query results should be indexed by. |
|
||||||
* @return ActiveQuery the query object itself |
|
||||||
*/ |
|
||||||
public function indexBy($column) |
|
||||||
{ |
|
||||||
$this->indexBy = $column; |
|
||||||
return $this; |
|
||||||
} |
|
||||||
|
|
||||||
// TODO: refactor, it is duplicated from yii/db/ActiveQuery |
|
||||||
private function createModels($rows) |
|
||||||
{ |
|
||||||
$models = array(); |
|
||||||
if ($this->asArray) { |
|
||||||
if ($this->indexBy === null) { |
|
||||||
return $rows; |
|
||||||
} |
|
||||||
foreach ($rows as $row) { |
|
||||||
$models[$row[$this->indexBy]] = $row; |
|
||||||
} |
|
||||||
} else { |
|
||||||
/** @var $class ActiveRecord */ |
|
||||||
$class = $this->modelClass; |
|
||||||
if ($this->indexBy === null) { |
|
||||||
foreach ($rows as $row) { |
|
||||||
$models[] = $class::create($row); |
|
||||||
} |
|
||||||
} else { |
|
||||||
foreach ($rows as $row) { |
|
||||||
$model = $class::create($row); |
|
||||||
$models[$model->{$this->indexBy}] = $model; |
|
||||||
} |
|
||||||
} |
|
||||||
} |
|
||||||
return $models; |
|
||||||
} |
|
||||||
|
|
||||||
// TODO: refactor, it is duplicated from yii/db/ActiveQuery |
|
||||||
private function populateRelations(&$models, $with) |
|
||||||
{ |
|
||||||
$primaryModel = new $this->modelClass; |
|
||||||
$relations = $this->normalizeRelations($primaryModel, $with); |
|
||||||
foreach ($relations as $name => $relation) { |
|
||||||
if ($relation->asArray === null) { |
|
||||||
// inherit asArray from primary query |
|
||||||
$relation->asArray = $this->asArray; |
|
||||||
} |
|
||||||
$relation->findWith($name, $models); |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
/** |
|
||||||
* TODO: refactor, it is duplicated from yii/db/ActiveQuery |
|
||||||
* @param ActiveRecord $model |
|
||||||
* @param array $with |
|
||||||
* @return ActiveRelation[] |
|
||||||
*/ |
|
||||||
private function normalizeRelations($model, $with) |
|
||||||
{ |
|
||||||
$relations = array(); |
|
||||||
foreach ($with as $name => $callback) { |
|
||||||
if (is_integer($name)) { |
|
||||||
$name = $callback; |
|
||||||
$callback = null; |
|
||||||
} |
|
||||||
if (($pos = strpos($name, '.')) !== false) { |
|
||||||
// with sub-relations |
|
||||||
$childName = substr($name, $pos + 1); |
|
||||||
$name = substr($name, 0, $pos); |
|
||||||
} else { |
|
||||||
$childName = null; |
|
||||||
} |
|
||||||
|
|
||||||
$t = strtolower($name); |
|
||||||
if (!isset($relations[$t])) { |
|
||||||
$relation = $model->getRelation($name); |
|
||||||
$relation->primaryModel = null; |
|
||||||
$relations[$t] = $relation; |
|
||||||
} else { |
|
||||||
$relation = $relations[$t]; |
|
||||||
} |
|
||||||
|
|
||||||
if (isset($childName)) { |
|
||||||
$relation->with[$childName] = $callback; |
|
||||||
} elseif ($callback !== null) { |
|
||||||
call_user_func($callback, $relation); |
|
||||||
} |
|
||||||
} |
|
||||||
return $relations; |
|
||||||
} |
|
||||||
} |
|
@ -1,543 +0,0 @@ |
|||||||
<?php |
|
||||||
/** |
|
||||||
* @link http://www.yiiframework.com/ |
|
||||||
* @copyright Copyright © 2008 Yii Software LLC |
|
||||||
* @license http://www.yiiframework.com/license/ |
|
||||||
*/ |
|
||||||
|
|
||||||
namespace yii\db\elasticsearch; |
|
||||||
|
|
||||||
use yii\base\InvalidCallException; |
|
||||||
use yii\base\InvalidConfigException; |
|
||||||
use yii\base\InvalidParamException; |
|
||||||
use yii\base\NotSupportedException; |
|
||||||
use yii\base\UnknownMethodException; |
|
||||||
use yii\db\TableSchema; |
|
||||||
|
|
||||||
/** |
|
||||||
* ActiveRecord is the base class for classes representing relational data in terms of objects. |
|
||||||
* |
|
||||||
* |
|
||||||
* |
|
||||||
* @author Carsten Brandt <mail@cebe.cc> |
|
||||||
* @since 2.0 |
|
||||||
*/ |
|
||||||
abstract class ActiveRecord extends \yii\db\ActiveRecord |
|
||||||
{ |
|
||||||
/** |
|
||||||
* Returns the database connection used by this AR class. |
|
||||||
* By default, the "elasticsearch" application component is used as the database connection. |
|
||||||
* You may override this method if you want to use a different database connection. |
|
||||||
* @return Connection the database connection used by this AR class. |
|
||||||
*/ |
|
||||||
public static function getDb() |
|
||||||
{ |
|
||||||
return \Yii::$app->elasticsearch; |
|
||||||
} |
|
||||||
|
|
||||||
public static function primaryKey() |
|
||||||
{ |
|
||||||
return array('id'); |
|
||||||
} |
|
||||||
|
|
||||||
/** |
|
||||||
* Creates an [[ActiveQuery]] instance for query purpose. |
|
||||||
* |
|
||||||
* @include @yii/db/ActiveRecord-find.md |
|
||||||
* |
|
||||||
* @param mixed $q the query parameter. This can be one of the followings: |
|
||||||
* |
|
||||||
* - a scalar value (integer or string): query by a single primary key value and return the |
|
||||||
* corresponding record. |
|
||||||
* - an array of name-value pairs: query by a set of column values and return a single record matching all of them. |
|
||||||
* - null: return a new [[ActiveQuery]] object for further query purpose. |
|
||||||
* |
|
||||||
* @return ActiveQuery|ActiveRecord|null When `$q` is null, a new [[ActiveQuery]] instance |
|
||||||
* is returned; when `$q` is a scalar or an array, an ActiveRecord object matching it will be |
|
||||||
* returned (null will be returned if there is no matching). |
|
||||||
* @see createQuery() |
|
||||||
*/ |
|
||||||
public static function find($q = null) // TODO optimize API |
|
||||||
{ |
|
||||||
$query = static::createQuery(); |
|
||||||
if (is_array($q)) { |
|
||||||
return $query->primaryKeys($q)->one(); |
|
||||||
} elseif ($q !== null) { |
|
||||||
// query by primary key |
|
||||||
$primaryKey = static::primaryKey(); |
|
||||||
return $query->primaryKeys(array($primaryKey[0] => $q))->one(); |
|
||||||
} |
|
||||||
return $query; |
|
||||||
} |
|
||||||
|
|
||||||
/** |
|
||||||
* @inheritdoc |
|
||||||
*/ |
|
||||||
public static function findBySql($sql, $params = array()) |
|
||||||
{ |
|
||||||
throw new NotSupportedException('findBySql() is not supported by elasticsearch ActiveRecord'); |
|
||||||
} |
|
||||||
|
|
||||||
/** |
|
||||||
* Creates an [[ActiveQuery]] instance. |
|
||||||
* This method is called by [[find()]], [[findBySql()]] and [[count()]] to start a SELECT query. |
|
||||||
* You may override this method to return a customized query (e.g. `CustomerQuery` specified |
|
||||||
* written for querying `Customer` purpose.) |
|
||||||
* @return ActiveQuery the newly created [[ActiveQuery]] instance. |
|
||||||
*/ |
|
||||||
public static function createQuery() |
|
||||||
{ |
|
||||||
return new ActiveQuery(array( |
|
||||||
'modelClass' => get_called_class(), |
|
||||||
)); |
|
||||||
} |
|
||||||
|
|
||||||
/** |
|
||||||
* Declares the name of the database table associated with this AR class. |
|
||||||
* @return string the table name |
|
||||||
*/ |
|
||||||
public static function tableName() |
|
||||||
{ |
|
||||||
return static::getTableSchema()->name; |
|
||||||
} |
|
||||||
|
|
||||||
public static function indexName() |
|
||||||
{ |
|
||||||
return static::getTableSchema()->name; |
|
||||||
} |
|
||||||
|
|
||||||
/** |
|
||||||
* Returns the schema information of the DB table associated with this AR class. |
|
||||||
* @return TableSchema the schema information of the DB table associated with this AR class. |
|
||||||
*/ |
|
||||||
public static function getTableSchema() |
|
||||||
{ |
|
||||||
// TODO should be cached |
|
||||||
throw new InvalidConfigException(__CLASS__.'::getTableSchema() needs to be overridden in subclasses and return a TableSchema.'); |
|
||||||
} |
|
||||||
|
|
||||||
/** |
|
||||||
* Inserts a row into the associated database table using the attribute values of this record. |
|
||||||
* |
|
||||||
* This method performs the following steps in order: |
|
||||||
* |
|
||||||
* 1. call [[beforeValidate()]] when `$runValidation` is true. If validation |
|
||||||
* fails, it will skip the rest of the steps; |
|
||||||
* 2. call [[afterValidate()]] when `$runValidation` is true. |
|
||||||
* 3. call [[beforeSave()]]. If the method returns false, it will skip the |
|
||||||
* rest of the steps; |
|
||||||
* 4. insert the record into database. If this fails, it will skip the rest of the steps; |
|
||||||
* 5. call [[afterSave()]]; |
|
||||||
* |
|
||||||
* In the above step 1, 2, 3 and 5, events [[EVENT_BEFORE_VALIDATE]], |
|
||||||
* [[EVENT_BEFORE_INSERT]], [[EVENT_AFTER_INSERT]] and [[EVENT_AFTER_VALIDATE]] |
|
||||||
* will be raised by the corresponding methods. |
|
||||||
* |
|
||||||
* Only the [[changedAttributes|changed attribute values]] will be inserted into database. |
|
||||||
* |
|
||||||
* If the table's primary key is auto-incremental and is null during insertion, |
|
||||||
* it will be populated with the actual value after insertion. |
|
||||||
* |
|
||||||
* For example, to insert a customer record: |
|
||||||
* |
|
||||||
* ~~~ |
|
||||||
* $customer = new Customer; |
|
||||||
* $customer->name = $name; |
|
||||||
* $customer->email = $email; |
|
||||||
* $customer->insert(); |
|
||||||
* ~~~ |
|
||||||
* |
|
||||||
* @param boolean $runValidation whether to perform validation before saving the record. |
|
||||||
* If the validation fails, the record will not be inserted into the database. |
|
||||||
* @param array $attributes list of attributes that need to be saved. Defaults to null, |
|
||||||
* meaning all attributes that are loaded from DB will be saved. |
|
||||||
* @return boolean whether the attributes are valid and the record is inserted successfully. |
|
||||||
*/ |
|
||||||
public function insert($runValidation = true, $attributes = null) |
|
||||||
{ |
|
||||||
if ($runValidation && !$this->validate($attributes)) { |
|
||||||
return false; |
|
||||||
} |
|
||||||
if ($this->beforeSave(true)) { |
|
||||||
$db = static::getDb(); |
|
||||||
$values = $this->getDirtyAttributes($attributes); |
|
||||||
$pk = array(); |
|
||||||
foreach ($this->primaryKey() as $key) { |
|
||||||
$pk[$key] = $values[$key] = $this->getAttribute($key); |
|
||||||
if ($pk[$key] === null) { |
|
||||||
$pk[$key] = $values[$key] = 0; // TODO add support for incrementing PK |
|
||||||
$this->setAttribute($key, $values[$key]); |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
// TODO store record in index |
|
||||||
|
|
||||||
$this->setOldAttributes($values); |
|
||||||
$this->afterSave(true); |
|
||||||
return true; |
|
||||||
} |
|
||||||
return false; |
|
||||||
} |
|
||||||
|
|
||||||
/** |
|
||||||
* Updates the whole table using the provided attribute values and conditions. |
|
||||||
* For example, to change the status to be 1 for all customers whose status is 2: |
|
||||||
* |
|
||||||
* ~~~ |
|
||||||
* Customer::updateAll(array('status' => 1), 'status = 2'); |
|
||||||
* ~~~ |
|
||||||
* |
|
||||||
* @param array $attributes attribute values (name-value pairs) to be saved into the table |
|
||||||
* @param string|array $condition the conditions that will be put in the WHERE part of the UPDATE SQL. |
|
||||||
* Please refer to [[Query::where()]] on how to specify this parameter. |
|
||||||
* @param array $params the parameters (name=>value) to be bound to the query. |
|
||||||
* @return integer the number of rows updated |
|
||||||
*/ |
|
||||||
public static function updateAll($attributes, $condition = '', $params = array()) |
|
||||||
{ |
|
||||||
$db = static::getDb(); |
|
||||||
|
|
||||||
// TODO massive update (do a find and then update each record) |
|
||||||
|
|
||||||
if (empty($attributes)) { |
|
||||||
return 0; |
|
||||||
} |
|
||||||
$n=0; |
|
||||||
// foreach(... as $pk) { |
|
||||||
// |
|
||||||
// // TODO update records |
|
||||||
// |
|
||||||
// $n++; |
|
||||||
// } |
|
||||||
|
|
||||||
return $n; |
|
||||||
} |
|
||||||
|
|
||||||
/** |
|
||||||
* Updates the whole table using the provided counter changes and conditions. |
|
||||||
* For example, to increment all customers' age by 1, |
|
||||||
* |
|
||||||
* ~~~ |
|
||||||
* Customer::updateAllCounters(array('age' => 1)); |
|
||||||
* ~~~ |
|
||||||
* |
|
||||||
* @param array $counters the counters to be updated (attribute name => increment value). |
|
||||||
* Use negative values if you want to decrement the counters. |
|
||||||
* @param string|array $condition the conditions that will be put in the WHERE part of the UPDATE SQL. |
|
||||||
* Please refer to [[Query::where()]] on how to specify this parameter. |
|
||||||
* @param array $params the parameters (name=>value) to be bound to the query. |
|
||||||
* Do not name the parameters as `:bp0`, `:bp1`, etc., because they are used internally by this method. |
|
||||||
* @return integer the number of rows updated |
|
||||||
*/ |
|
||||||
public static function updateAllCounters($counters, $condition = '', $params = array()) |
|
||||||
{ |
|
||||||
// TODO implement |
|
||||||
throw new NotSupportedException('update counters is not supported by elasticsearch.'); |
|
||||||
} |
|
||||||
|
|
||||||
/** |
|
||||||
* Deletes rows in the table using the provided conditions. |
|
||||||
* WARNING: If you do not specify any condition, this method will delete ALL rows in the table. |
|
||||||
* |
|
||||||
* For example, to delete all customers whose status is 3: |
|
||||||
* |
|
||||||
* ~~~ |
|
||||||
* Customer::deleteAll('status = 3'); |
|
||||||
* ~~~ |
|
||||||
* |
|
||||||
* @param string|array $condition the conditions that will be put in the WHERE part of the DELETE SQL. |
|
||||||
* Please refer to [[Query::where()]] on how to specify this parameter. |
|
||||||
* @param array $params the parameters (name=>value) to be bound to the query. |
|
||||||
* @return integer the number of rows deleted |
|
||||||
*/ |
|
||||||
public static function deleteAll($condition = '', $params = array()) |
|
||||||
{ |
|
||||||
$db = static::getDb(); |
|
||||||
|
|
||||||
// TODO massive delete (do a find and then delete each record) |
|
||||||
|
|
||||||
if (empty($condition)) { |
|
||||||
return 0; |
|
||||||
} |
|
||||||
$n = 0; |
|
||||||
// foreach($condition as $pk) { |
|
||||||
// |
|
||||||
// $n++; |
|
||||||
// } |
|
||||||
return $n; |
|
||||||
} |
|
||||||
|
|
||||||
/** |
|
||||||
* Declares a `has-one` relation. |
|
||||||
* The declaration is returned in terms of an [[ActiveRelation]] instance |
|
||||||
* through which the related record can be queried and retrieved back. |
|
||||||
* |
|
||||||
* A `has-one` relation means that there is at most one related record matching |
|
||||||
* the criteria set by this relation, e.g., a customer has one country. |
|
||||||
* |
|
||||||
* For example, to declare the `country` relation for `Customer` class, we can write |
|
||||||
* the following code in the `Customer` class: |
|
||||||
* |
|
||||||
* ~~~ |
|
||||||
* public function getCountry() |
|
||||||
* { |
|
||||||
* return $this->hasOne('Country', array('id' => 'country_id')); |
|
||||||
* } |
|
||||||
* ~~~ |
|
||||||
* |
|
||||||
* Note that in the above, the 'id' key in the `$link` parameter refers to an attribute name |
|
||||||
* in the related class `Country`, while the 'country_id' value refers to an attribute name |
|
||||||
* in the current AR class. |
|
||||||
* |
|
||||||
* Call methods declared in [[ActiveRelation]] to further customize the relation. |
|
||||||
* |
|
||||||
* @param string $class the class name of the related record |
|
||||||
* @param array $link the primary-foreign key constraint. The keys of the array refer to |
|
||||||
* the columns in the table associated with the `$class` model, while the values of the |
|
||||||
* array refer to the corresponding columns in the table associated with this AR class. |
|
||||||
* @return ActiveRelation the relation object. |
|
||||||
*/ |
|
||||||
public function hasOne($class, $link) |
|
||||||
{ |
|
||||||
return new ActiveRelation(array( |
|
||||||
'modelClass' => $this->getNamespacedClass($class), |
|
||||||
'primaryModel' => $this, |
|
||||||
'link' => $link, |
|
||||||
'multiple' => false, |
|
||||||
)); |
|
||||||
} |
|
||||||
|
|
||||||
/** |
|
||||||
* Declares a `has-many` relation. |
|
||||||
* The declaration is returned in terms of an [[ActiveRelation]] instance |
|
||||||
* through which the related record can be queried and retrieved back. |
|
||||||
* |
|
||||||
* A `has-many` relation means that there are multiple related records matching |
|
||||||
* the criteria set by this relation, e.g., a customer has many orders. |
|
||||||
* |
|
||||||
* For example, to declare the `orders` relation for `Customer` class, we can write |
|
||||||
* the following code in the `Customer` class: |
|
||||||
* |
|
||||||
* ~~~ |
|
||||||
* public function getOrders() |
|
||||||
* { |
|
||||||
* return $this->hasMany('Order', array('customer_id' => 'id')); |
|
||||||
* } |
|
||||||
* ~~~ |
|
||||||
* |
|
||||||
* Note that in the above, the 'customer_id' key in the `$link` parameter refers to |
|
||||||
* an attribute name in the related class `Order`, while the 'id' value refers to |
|
||||||
* an attribute name in the current AR class. |
|
||||||
* |
|
||||||
* @param string $class the class name of the related record |
|
||||||
* @param array $link the primary-foreign key constraint. The keys of the array refer to |
|
||||||
* the columns in the table associated with the `$class` model, while the values of the |
|
||||||
* array refer to the corresponding columns in the table associated with this AR class. |
|
||||||
* @return ActiveRelation the relation object. |
|
||||||
*/ |
|
||||||
public function hasMany($class, $link) |
|
||||||
{ |
|
||||||
return new ActiveRelation(array( |
|
||||||
'modelClass' => $this->getNamespacedClass($class), |
|
||||||
'primaryModel' => $this, |
|
||||||
'link' => $link, |
|
||||||
'multiple' => true, |
|
||||||
)); |
|
||||||
} |
|
||||||
|
|
||||||
/** |
|
||||||
* Returns the relation object with the specified name. |
|
||||||
* A relation is defined by a getter method which returns an [[ActiveRelation]] object. |
|
||||||
* It can be declared in either the Active Record class itself or one of its behaviors. |
|
||||||
* @param string $name the relation name |
|
||||||
* @return ActiveRelation the relation object |
|
||||||
* @throws InvalidParamException if the named relation does not exist. |
|
||||||
*/ |
|
||||||
public function getRelation($name) |
|
||||||
{ |
|
||||||
$getter = 'get' . $name; |
|
||||||
try { |
|
||||||
$relation = $this->$getter(); |
|
||||||
if ($relation instanceof ActiveRelation) { |
|
||||||
return $relation; |
|
||||||
} |
|
||||||
} catch (UnknownMethodException $e) { |
|
||||||
} |
|
||||||
throw new InvalidParamException(get_class($this) . ' has no relation named "' . $name . '".'); |
|
||||||
} |
|
||||||
|
|
||||||
/** |
|
||||||
* Establishes the relationship between two models. |
|
||||||
* |
|
||||||
* The relationship is established by setting the foreign key value(s) in one model |
|
||||||
* to be the corresponding primary key value(s) in the other model. |
|
||||||
* The model with the foreign key will be saved into database without performing validation. |
|
||||||
* |
|
||||||
* If the relationship involves a pivot table, a new row will be inserted into the |
|
||||||
* pivot table which contains the primary key values from both models. |
|
||||||
* |
|
||||||
* Note that this method requires that the primary key value is not null. |
|
||||||
* |
|
||||||
* @param string $name the name of the relationship |
|
||||||
* @param ActiveRecord $model the model to be linked with the current one. |
|
||||||
* @param array $extraColumns additional column values to be saved into the pivot table. |
|
||||||
* This parameter is only meaningful for a relationship involving a pivot table |
|
||||||
* (i.e., a relation set with `[[ActiveRelation::via()]]` or `[[ActiveRelation::viaTable()]]`.) |
|
||||||
* @throws InvalidCallException if the method is unable to link two models. |
|
||||||
*/ |
|
||||||
public function link($name, $model, $extraColumns = array()) |
|
||||||
{ |
|
||||||
$relation = $this->getRelation($name); |
|
||||||
|
|
||||||
if ($relation->via !== null) { |
|
||||||
// TODO |
|
||||||
|
|
||||||
|
|
||||||
} else { |
|
||||||
$p1 = $model->isPrimaryKey(array_keys($relation->link)); |
|
||||||
$p2 = $this->isPrimaryKey(array_values($relation->link)); |
|
||||||
if ($p1 && $p2) { |
|
||||||
if ($this->getIsNewRecord() && $model->getIsNewRecord()) { |
|
||||||
throw new InvalidCallException('Unable to link models: both models are newly created.'); |
|
||||||
} elseif ($this->getIsNewRecord()) { |
|
||||||
$this->bindModels(array_flip($relation->link), $this, $model); |
|
||||||
} else { |
|
||||||
$this->bindModels($relation->link, $model, $this); |
|
||||||
} |
|
||||||
} elseif ($p1) { |
|
||||||
$this->bindModels(array_flip($relation->link), $this, $model); |
|
||||||
} elseif ($p2) { |
|
||||||
$this->bindModels($relation->link, $model, $this); |
|
||||||
} else { |
|
||||||
throw new InvalidCallException('Unable to link models: the link does not involve any primary key.'); |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
// update lazily loaded related objects |
|
||||||
if (!$relation->multiple) { |
|
||||||
$this->_related[$name] = $model; |
|
||||||
} elseif (isset($this->_related[$name])) { |
|
||||||
if ($relation->indexBy !== null) { |
|
||||||
$indexBy = $relation->indexBy; |
|
||||||
$this->_related[$name][$model->$indexBy] = $model; |
|
||||||
} else { |
|
||||||
$this->_related[$name][] = $model; |
|
||||||
} |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
/** |
|
||||||
* @param array $link |
|
||||||
* @param ActiveRecord $foreignModel |
|
||||||
* @param ActiveRecord $primaryModel |
|
||||||
* @throws InvalidCallException |
|
||||||
*/ |
|
||||||
private function bindModels($link, $foreignModel, $primaryModel) |
|
||||||
{ |
|
||||||
foreach ($link as $fk => $pk) { |
|
||||||
$value = $primaryModel->$pk; |
|
||||||
if ($value === null) { |
|
||||||
throw new InvalidCallException('Unable to link models: the primary key of ' . get_class($primaryModel) . ' is null.'); |
|
||||||
} |
|
||||||
$foreignModel->$fk = $value; |
|
||||||
} |
|
||||||
$foreignModel->save(false); |
|
||||||
} |
|
||||||
|
|
||||||
/** |
|
||||||
* Destroys the relationship between two models. |
|
||||||
* |
|
||||||
* The model with the foreign key of the relationship will be deleted if `$delete` is true. |
|
||||||
* Otherwise, the foreign key will be set null and the model will be saved without validation. |
|
||||||
* |
|
||||||
* @param string $name the name of the relationship. |
|
||||||
* @param ActiveRecord $model the model to be unlinked from the current one. |
|
||||||
* @param boolean $delete whether to delete the model that contains the foreign key. |
|
||||||
* If false, the model's foreign key will be set null and saved. |
|
||||||
* If true, the model containing the foreign key will be deleted. |
|
||||||
* @throws InvalidCallException if the models cannot be unlinked |
|
||||||
*/ |
|
||||||
public function unlink($name, $model, $delete = false) |
|
||||||
{ |
|
||||||
// TODO |
|
||||||
$relation = $this->getRelation($name); |
|
||||||
|
|
||||||
if ($relation->via !== null) { |
|
||||||
if (is_array($relation->via)) { |
|
||||||
/** @var $viaRelation ActiveRelation */ |
|
||||||
list($viaName, $viaRelation) = $relation->via; |
|
||||||
/** @var $viaClass ActiveRecord */ |
|
||||||
$viaClass = $viaRelation->modelClass; |
|
||||||
$viaTable = $viaClass::tableName(); |
|
||||||
unset($this->_related[strtolower($viaName)]); |
|
||||||
} else { |
|
||||||
$viaRelation = $relation->via; |
|
||||||
$viaTable = reset($relation->via->from); |
|
||||||
} |
|
||||||
$columns = array(); |
|
||||||
foreach ($viaRelation->link as $a => $b) { |
|
||||||
$columns[$a] = $this->$b; |
|
||||||
} |
|
||||||
foreach ($relation->link as $a => $b) { |
|
||||||
$columns[$b] = $model->$a; |
|
||||||
} |
|
||||||
$command = static::getDb()->createCommand(); |
|
||||||
if ($delete) { |
|
||||||
$command->delete($viaTable, $columns)->execute(); |
|
||||||
} else { |
|
||||||
$nulls = array(); |
|
||||||
foreach (array_keys($columns) as $a) { |
|
||||||
$nulls[$a] = null; |
|
||||||
} |
|
||||||
$command->update($viaTable, $nulls, $columns)->execute(); |
|
||||||
} |
|
||||||
} else { |
|
||||||
$p1 = $model->isPrimaryKey(array_keys($relation->link)); |
|
||||||
$p2 = $this->isPrimaryKey(array_values($relation->link)); |
|
||||||
if ($p1 && $p2 || $p2) { |
|
||||||
foreach ($relation->link as $a => $b) { |
|
||||||
$model->$a = null; |
|
||||||
} |
|
||||||
$delete ? $model->delete() : $model->save(false); |
|
||||||
} elseif ($p1) { |
|
||||||
foreach ($relation->link as $b) { |
|
||||||
$this->$b = null; |
|
||||||
} |
|
||||||
$delete ? $this->delete() : $this->save(false); |
|
||||||
} else { |
|
||||||
throw new InvalidCallException('Unable to unlink models: the link does not involve any primary key.'); |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
if (!$relation->multiple) { |
|
||||||
unset($this->_related[$name]); |
|
||||||
} elseif (isset($this->_related[$name])) { |
|
||||||
/** @var $b ActiveRecord */ |
|
||||||
foreach ($this->_related[$name] as $a => $b) { |
|
||||||
if ($model->getPrimaryKey() == $b->getPrimaryKey()) { |
|
||||||
unset($this->_related[$name][$a]); |
|
||||||
} |
|
||||||
} |
|
||||||
} |
|
||||||
} |
|
||||||
|
|
||||||
/** |
|
||||||
* TODO duplicate code, refactor |
|
||||||
* @param array $keys |
|
||||||
* @return boolean |
|
||||||
*/ |
|
||||||
private function isPrimaryKey($keys) |
|
||||||
{ |
|
||||||
$pks = $this->primaryKey(); |
|
||||||
foreach ($keys as $key) { |
|
||||||
if (!in_array($key, $pks, true)) { |
|
||||||
return false; |
|
||||||
} |
|
||||||
} |
|
||||||
return true; |
|
||||||
} |
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
// TODO implement link and unlink |
|
||||||
} |
|
@ -0,0 +1,749 @@ |
|||||||
|
<?php |
||||||
|
/** |
||||||
|
* @link http://www.yiiframework.com/ |
||||||
|
* @copyright Copyright (c) 2008 Yii Software LLC |
||||||
|
* @license http://www.yiiframework.com/license/ |
||||||
|
*/ |
||||||
|
|
||||||
|
namespace yii\elasticsearch; |
||||||
|
use yii\base\NotSupportedException; |
||||||
|
use yii\db\Exception; |
||||||
|
use yii\helpers\Json; |
||||||
|
|
||||||
|
/** |
||||||
|
* ActiveQuery represents a query associated with an Active Record class. |
||||||
|
* |
||||||
|
* ActiveQuery instances are usually created by [[ActiveRecord::find()]] |
||||||
|
* and [[ActiveRecord::count()]]. |
||||||
|
* |
||||||
|
* ActiveQuery mainly provides the following methods to retrieve the query results: |
||||||
|
* |
||||||
|
* - [[one()]]: returns a single record populated with the first row of data. |
||||||
|
* - [[all()]]: returns all records based on the query results. |
||||||
|
* - [[count()]]: returns the number of records. |
||||||
|
* - [[sum()]]: returns the sum over the specified column. |
||||||
|
* - [[average()]]: returns the average over the specified column. |
||||||
|
* - [[min()]]: returns the min over the specified column. |
||||||
|
* - [[max()]]: returns the max over the specified column. |
||||||
|
* - [[scalar()]]: returns the value of the first column in the first row of the query result. |
||||||
|
* - [[exists()]]: returns a value indicating whether the query result has data or not. |
||||||
|
* |
||||||
|
* You can use query methods, such as [[where()]], [[limit()]] and [[orderBy()]] to customize the query options. |
||||||
|
* |
||||||
|
* ActiveQuery also provides the following additional query options: |
||||||
|
* |
||||||
|
* - [[with()]]: list of relations that this query should be performed with. |
||||||
|
* - [[indexBy()]]: the name of the column by which the query result should be indexed. |
||||||
|
* - [[asArray()]]: whether to return each record as an array. |
||||||
|
* |
||||||
|
* These options can be configured using methods of the same name. For example: |
||||||
|
* |
||||||
|
* ~~~ |
||||||
|
* $customers = Customer::find()->with('orders')->asArray()->all(); |
||||||
|
* ~~~ |
||||||
|
* |
||||||
|
* @author Carsten Brandt <mail@cebe.cc> |
||||||
|
* @since 2.0 |
||||||
|
*/ |
||||||
|
class ActiveQuery extends \yii\base\Component |
||||||
|
{ |
||||||
|
/** |
||||||
|
* Sort ascending |
||||||
|
* @see orderBy |
||||||
|
*/ |
||||||
|
const SORT_ASC = false; |
||||||
|
/** |
||||||
|
* Sort descending |
||||||
|
* @see orderBy |
||||||
|
*/ |
||||||
|
const SORT_DESC = true; |
||||||
|
|
||||||
|
/** |
||||||
|
* @var string the name of the ActiveRecord class. |
||||||
|
*/ |
||||||
|
public $modelClass; |
||||||
|
/** |
||||||
|
* @var array list of relations that this query should be performed with |
||||||
|
*/ |
||||||
|
public $with; |
||||||
|
/** |
||||||
|
* @var string|callable $column the name of the column by which the query results should be indexed by. |
||||||
|
* This can also be a callable (e.g. anonymous function) that returns the index value based on the given |
||||||
|
* row or model data. For more details, see [[indexBy()]]. |
||||||
|
*/ |
||||||
|
public $indexBy; |
||||||
|
/** |
||||||
|
* @var boolean whether to return each record as an array. If false (default), an object |
||||||
|
* of [[modelClass]] will be created to represent each record. |
||||||
|
*/ |
||||||
|
public $asArray; |
||||||
|
/** |
||||||
|
* @var array the query condition. |
||||||
|
* @see where() |
||||||
|
*/ |
||||||
|
public $where; |
||||||
|
/** |
||||||
|
* @var integer maximum number of records to be returned. If not set or less than 0, it means no limit. |
||||||
|
*/ |
||||||
|
public $limit; |
||||||
|
/** |
||||||
|
* @var integer zero-based offset from where the records are to be returned. |
||||||
|
* If not set, it means starting from the beginning. |
||||||
|
* If less than zero it means starting n elements from the end. |
||||||
|
*/ |
||||||
|
public $offset; |
||||||
|
/** |
||||||
|
* @var array how to sort the query results. This is used to construct the ORDER BY clause in a SQL statement. |
||||||
|
* The array keys are the columns to be sorted by, and the array values are the corresponding sort directions which |
||||||
|
* can be either [[ActiveQuery::SORT_ASC]] or [[ActiveQuery::SORT_DESC]]. The array may also contain [[Expression]] objects. |
||||||
|
* If that is the case, the expressions will be converted into strings without any change. |
||||||
|
*/ |
||||||
|
public $orderBy; |
||||||
|
|
||||||
|
/** |
||||||
|
* PHP magic method. |
||||||
|
* This method allows calling static method defined in [[modelClass]] via this query object. |
||||||
|
* It is mainly implemented for supporting the feature of scope. |
||||||
|
* @param string $name the method name to be called |
||||||
|
* @param array $params the parameters passed to the method |
||||||
|
* @return mixed the method return result |
||||||
|
*/ |
||||||
|
public function __call($name, $params) |
||||||
|
{ |
||||||
|
if (method_exists($this->modelClass, $name)) { |
||||||
|
array_unshift($params, $this); |
||||||
|
call_user_func_array(array($this->modelClass, $name), $params); |
||||||
|
return $this; |
||||||
|
} else { |
||||||
|
return parent::__call($name, $params); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* Executes query and returns all results as an array. |
||||||
|
* @return array the query results. If the query results in nothing, an empty array will be returned. |
||||||
|
*/ |
||||||
|
public function all() |
||||||
|
{ |
||||||
|
// TODO add support for orderBy |
||||||
|
$data = $this->executeScript('All'); |
||||||
|
$rows = array(); |
||||||
|
foreach($data as $dataRow) { |
||||||
|
$row = array(); |
||||||
|
$c = count($dataRow); |
||||||
|
for($i = 0; $i < $c; ) { |
||||||
|
$row[$dataRow[$i++]] = $dataRow[$i++]; |
||||||
|
} |
||||||
|
$rows[] = $row; |
||||||
|
} |
||||||
|
if (!empty($rows)) { |
||||||
|
$models = $this->createModels($rows); |
||||||
|
if (!empty($this->with)) { |
||||||
|
$this->populateRelations($models, $this->with); |
||||||
|
} |
||||||
|
return $models; |
||||||
|
} else { |
||||||
|
return array(); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* Executes query and returns a single row of result. |
||||||
|
* @return ActiveRecord|array|null a single row of query result. Depending on the setting of [[asArray]], |
||||||
|
* the query result may be either an array or an ActiveRecord object. Null will be returned |
||||||
|
* if the query results in nothing. |
||||||
|
*/ |
||||||
|
public function one() |
||||||
|
{ |
||||||
|
// TODO add support for orderBy |
||||||
|
$data = $this->executeScript('One'); |
||||||
|
if ($data === array()) { |
||||||
|
return null; |
||||||
|
} |
||||||
|
$row = array(); |
||||||
|
$c = count($data); |
||||||
|
for($i = 0; $i < $c; ) { |
||||||
|
$row[$data[$i++]] = $data[$i++]; |
||||||
|
} |
||||||
|
if ($this->asArray) { |
||||||
|
$model = $row; |
||||||
|
} else { |
||||||
|
/** @var $class ActiveRecord */ |
||||||
|
$class = $this->modelClass; |
||||||
|
$model = $class::create($row); |
||||||
|
} |
||||||
|
if (!empty($this->with)) { |
||||||
|
$models = array($model); |
||||||
|
$this->populateRelations($models, $this->with); |
||||||
|
$model = $models[0]; |
||||||
|
} |
||||||
|
return $model; |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* Executes the query and returns the first column of the result. |
||||||
|
* @param string $column name of the column to select |
||||||
|
* @return array the first column of the query result. An empty array is returned if the query results in nothing. |
||||||
|
*/ |
||||||
|
public function column($column) |
||||||
|
{ |
||||||
|
// TODO add support for indexBy and orderBy |
||||||
|
return $this->executeScript('Column', $column); |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* Returns the number of records. |
||||||
|
* @param string $q the COUNT expression. Defaults to '*'. |
||||||
|
* Make sure you properly quote column names. |
||||||
|
* @return integer number of records |
||||||
|
*/ |
||||||
|
public function count() |
||||||
|
{ |
||||||
|
if ($this->offset === null && $this->limit === null && $this->where === null) { |
||||||
|
$modelClass = $this->modelClass; |
||||||
|
/** @var Connection $db */ |
||||||
|
$db = $modelClass::getDb(); |
||||||
|
return $db->executeCommand('LLEN', array($modelClass::tableName())); |
||||||
|
} else { |
||||||
|
return $this->executeScript('Count'); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* Returns the number of records. |
||||||
|
* @param string $column the column to sum up |
||||||
|
* @return integer number of records |
||||||
|
*/ |
||||||
|
public function sum($column) |
||||||
|
{ |
||||||
|
return $this->executeScript('Sum', $column); |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* Returns the average of the specified column values. |
||||||
|
* @param string $column the column name or expression. |
||||||
|
* Make sure you properly quote column names in the expression. |
||||||
|
* @return integer the average of the specified column values. |
||||||
|
*/ |
||||||
|
public function average($column) |
||||||
|
{ |
||||||
|
return $this->executeScript('Average', $column); |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* Returns the minimum of the specified column values. |
||||||
|
* @param string $column the column name or expression. |
||||||
|
* Make sure you properly quote column names in the expression. |
||||||
|
* @return integer the minimum of the specified column values. |
||||||
|
*/ |
||||||
|
public function min($column) |
||||||
|
{ |
||||||
|
return $this->executeScript('Min', $column); |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* Returns the maximum of the specified column values. |
||||||
|
* @param string $column the column name or expression. |
||||||
|
* Make sure you properly quote column names in the expression. |
||||||
|
* @return integer the maximum of the specified column values. |
||||||
|
*/ |
||||||
|
public function max($column) |
||||||
|
{ |
||||||
|
return $this->executeScript('Max', $column); |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* Returns the query result as a scalar value. |
||||||
|
* The value returned will be the first column in the first row of the query results. |
||||||
|
* @param string $column name of the column to select |
||||||
|
* @return string|boolean the value of the first column in the first row of the query result. |
||||||
|
* False is returned if the query result is empty. |
||||||
|
*/ |
||||||
|
public function scalar($column) |
||||||
|
{ |
||||||
|
$record = $this->one(); |
||||||
|
return $record->$column; |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* Returns a value indicating whether the query result contains any row of data. |
||||||
|
* @return boolean whether the query result contains any row of data. |
||||||
|
*/ |
||||||
|
public function exists() |
||||||
|
{ |
||||||
|
return $this->one() !== null; |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* Executes a script created by [[LuaScriptBuilder]] |
||||||
|
* @param string $type |
||||||
|
* @param null $column |
||||||
|
* @return array|bool|null|string |
||||||
|
*/ |
||||||
|
protected function executeScript($type, $columnName=null) |
||||||
|
{ |
||||||
|
if (($data = $this->findByPk($type)) === false) { |
||||||
|
$modelClass = $this->modelClass; |
||||||
|
/** @var Connection $db */ |
||||||
|
$db = $modelClass::getDb(); |
||||||
|
|
||||||
|
$method = 'build' . $type; |
||||||
|
$script = $db->getLuaScriptBuilder()->$method($this, $columnName); |
||||||
|
return $db->executeCommand('EVAL', array($script, 0)); |
||||||
|
} |
||||||
|
return $data; |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* Fetch by pk if possible as this is much faster |
||||||
|
*/ |
||||||
|
private function findByPk($type, $columnName = null) |
||||||
|
{ |
||||||
|
$modelClass = $this->modelClass; |
||||||
|
if (is_array($this->where) && !isset($this->where[0]) && $modelClass::isPrimaryKey(array_keys($this->where))) { |
||||||
|
/** @var Connection $db */ |
||||||
|
$db = $modelClass::getDb(); |
||||||
|
|
||||||
|
$pks = (array) reset($this->where); |
||||||
|
|
||||||
|
$start = $this->offset === null ? 0 : $this->offset; |
||||||
|
$i = 0; |
||||||
|
$data = array(); |
||||||
|
$url = '/' . $modelClass::indexName() . '/' . $modelClass::indexType() . '/'; |
||||||
|
foreach($pks as $pk) { |
||||||
|
if (++$i > $start && ($this->limit === null || $i <= $start + $this->limit)) { |
||||||
|
$request = $db->http()->get($url . $pk); |
||||||
|
$response = $request->send(); |
||||||
|
if ($response->getStatusCode() == 404) { |
||||||
|
// ignore? |
||||||
|
} else { |
||||||
|
$data[] = Json::decode($response->getBody(true)); |
||||||
|
if ($type === 'One' && $this->orderBy === null) { |
||||||
|
break; |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
// TODO support orderBy |
||||||
|
|
||||||
|
switch($type) { |
||||||
|
case 'All': |
||||||
|
return $data; |
||||||
|
case 'One': |
||||||
|
return reset($data); |
||||||
|
case 'Column': |
||||||
|
// TODO support indexBy |
||||||
|
$column = array(); |
||||||
|
foreach($data as $dataRow) { |
||||||
|
$row = array(); |
||||||
|
$c = count($dataRow); |
||||||
|
for($i = 0; $i < $c; ) { |
||||||
|
$row[$dataRow[$i++]] = $dataRow[$i++]; |
||||||
|
} |
||||||
|
$column[] = $row[$columnName]; |
||||||
|
} |
||||||
|
return $column; |
||||||
|
case 'Count': |
||||||
|
return count($data); |
||||||
|
case 'Sum': |
||||||
|
$sum = 0; |
||||||
|
foreach($data as $dataRow) { |
||||||
|
$c = count($dataRow); |
||||||
|
for($i = 0; $i < $c; ) { |
||||||
|
if ($dataRow[$i++] == $columnName) { |
||||||
|
$sum += $dataRow[$i]; |
||||||
|
break; |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
return $sum; |
||||||
|
case 'Average': |
||||||
|
$sum = 0; |
||||||
|
$count = 0; |
||||||
|
foreach($data as $dataRow) { |
||||||
|
$count++; |
||||||
|
$c = count($dataRow); |
||||||
|
for($i = 0; $i < $c; ) { |
||||||
|
if ($dataRow[$i++] == $columnName) { |
||||||
|
$sum += $dataRow[$i]; |
||||||
|
break; |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
return $sum / $count; |
||||||
|
case 'Min': |
||||||
|
$min = null; |
||||||
|
foreach($data as $dataRow) { |
||||||
|
$c = count($dataRow); |
||||||
|
for($i = 0; $i < $c; ) { |
||||||
|
if ($dataRow[$i++] == $columnName && ($min == null || $dataRow[$i] < $min)) { |
||||||
|
$min = $dataRow[$i]; |
||||||
|
break; |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
return $min; |
||||||
|
case 'Max': |
||||||
|
$max = null; |
||||||
|
foreach($data as $dataRow) { |
||||||
|
$c = count($dataRow); |
||||||
|
for($i = 0; $i < $c; ) { |
||||||
|
if ($dataRow[$i++] == $columnName && ($max == null || $dataRow[$i] > $max)) { |
||||||
|
$max = $dataRow[$i]; |
||||||
|
break; |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
return $max; |
||||||
|
} |
||||||
|
} |
||||||
|
return false; |
||||||
|
} |
||||||
|
|
||||||
|
// TODO: refactor. code below here is all duplicated from yii/db/ActiveQuery and yii/db/Query |
||||||
|
|
||||||
|
/** |
||||||
|
* Sets the [[asArray]] property. |
||||||
|
* @param boolean $value whether to return the query results in terms of arrays instead of Active Records. |
||||||
|
* @return ActiveQuery the query object itself |
||||||
|
*/ |
||||||
|
public function asArray($value = true) |
||||||
|
{ |
||||||
|
$this->asArray = $value; |
||||||
|
return $this; |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* Sets the ORDER BY part of the query. |
||||||
|
* @param string|array $columns the columns (and the directions) to be ordered by. |
||||||
|
* Columns can be specified in either a string (e.g. "id ASC, name DESC") or an array |
||||||
|
* (e.g. `array('id' => Query::SORT_ASC, 'name' => Query::SORT_DESC)`). |
||||||
|
* The method will automatically quote the column names unless a column contains some parenthesis |
||||||
|
* (which means the column contains a DB expression). |
||||||
|
* @return ActiveQuery the query object itself |
||||||
|
* @see addOrderBy() |
||||||
|
*/ |
||||||
|
public function orderBy($columns) |
||||||
|
{ |
||||||
|
$this->orderBy = $this->normalizeOrderBy($columns); |
||||||
|
return $this; |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* Adds additional ORDER BY columns to the query. |
||||||
|
* @param string|array $columns the columns (and the directions) to be ordered by. |
||||||
|
* Columns can be specified in either a string (e.g. "id ASC, name DESC") or an array |
||||||
|
* (e.g. `array('id' => Query::SORT_ASC, 'name' => Query::SORT_DESC)`). |
||||||
|
* The method will automatically quote the column names unless a column contains some parenthesis |
||||||
|
* (which means the column contains a DB expression). |
||||||
|
* @return ActiveQuery the query object itself |
||||||
|
* @see orderBy() |
||||||
|
*/ |
||||||
|
public function addOrderBy($columns) |
||||||
|
{ |
||||||
|
$columns = $this->normalizeOrderBy($columns); |
||||||
|
if ($this->orderBy === null) { |
||||||
|
$this->orderBy = $columns; |
||||||
|
} else { |
||||||
|
$this->orderBy = array_merge($this->orderBy, $columns); |
||||||
|
} |
||||||
|
return $this; |
||||||
|
} |
||||||
|
|
||||||
|
protected function normalizeOrderBy($columns) |
||||||
|
{ |
||||||
|
throw new NotSupportedException('orderBy is currently not supported'); |
||||||
|
if (is_array($columns)) { |
||||||
|
return $columns; |
||||||
|
} else { |
||||||
|
$columns = preg_split('/\s*,\s*/', trim($columns), -1, PREG_SPLIT_NO_EMPTY); |
||||||
|
$result = array(); |
||||||
|
foreach ($columns as $column) { |
||||||
|
if (preg_match('/^(.*?)\s+(asc|desc)$/i', $column, $matches)) { |
||||||
|
$result[$matches[1]] = strcasecmp($matches[2], 'desc') ? self::SORT_ASC : self::SORT_DESC; |
||||||
|
} else { |
||||||
|
$result[$column] = self::SORT_ASC; |
||||||
|
} |
||||||
|
} |
||||||
|
return $result; |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* Sets the LIMIT part of the query. |
||||||
|
* @param integer $limit the limit |
||||||
|
* @return ActiveQuery the query object itself |
||||||
|
*/ |
||||||
|
public function limit($limit) |
||||||
|
{ |
||||||
|
$this->limit = $limit; |
||||||
|
return $this; |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* Sets the OFFSET part of the query. |
||||||
|
* @param integer $offset the offset |
||||||
|
* @return ActiveQuery the query object itself |
||||||
|
*/ |
||||||
|
public function offset($offset) |
||||||
|
{ |
||||||
|
$this->offset = $offset; |
||||||
|
return $this; |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* Specifies the relations with which this query should be performed. |
||||||
|
* |
||||||
|
* The parameters to this method can be either one or multiple strings, or a single array |
||||||
|
* of relation names and the optional callbacks to customize the relations. |
||||||
|
* |
||||||
|
* The followings are some usage examples: |
||||||
|
* |
||||||
|
* ~~~ |
||||||
|
* // find customers together with their orders and country |
||||||
|
* Customer::find()->with('orders', 'country')->all(); |
||||||
|
* // find customers together with their country and orders of status 1 |
||||||
|
* Customer::find()->with(array( |
||||||
|
* 'orders' => function($query) { |
||||||
|
* $query->andWhere('status = 1'); |
||||||
|
* }, |
||||||
|
* 'country', |
||||||
|
* ))->all(); |
||||||
|
* ~~~ |
||||||
|
* |
||||||
|
* @return ActiveQuery the query object itself |
||||||
|
*/ |
||||||
|
public function with() |
||||||
|
{ |
||||||
|
$this->with = func_get_args(); |
||||||
|
if (isset($this->with[0]) && is_array($this->with[0])) { |
||||||
|
// the parameter is given as an array |
||||||
|
$this->with = $this->with[0]; |
||||||
|
} |
||||||
|
return $this; |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* Sets the [[indexBy]] property. |
||||||
|
* @param string|callable $column the name of the column by which the query results should be indexed by. |
||||||
|
* This can also be a callable (e.g. anonymous function) that returns the index value based on the given |
||||||
|
* row or model data. The signature of the callable should be: |
||||||
|
* |
||||||
|
* ~~~ |
||||||
|
* // $model is an AR instance when `asArray` is false, |
||||||
|
* // or an array of column values when `asArray` is true. |
||||||
|
* function ($model) |
||||||
|
* { |
||||||
|
* // return the index value corresponding to $model |
||||||
|
* } |
||||||
|
* ~~~ |
||||||
|
* |
||||||
|
* @return ActiveQuery the query object itself |
||||||
|
*/ |
||||||
|
public function indexBy($column) |
||||||
|
{ |
||||||
|
$this->indexBy = $column; |
||||||
|
return $this; |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* Sets the WHERE part of the query. |
||||||
|
* |
||||||
|
* The method requires a $condition parameter, and optionally a $params parameter |
||||||
|
* specifying the values to be bound to the query. |
||||||
|
* |
||||||
|
* The $condition parameter should be either a string (e.g. 'id=1') or an array. |
||||||
|
* If the latter, it must be in one of the following two formats: |
||||||
|
* |
||||||
|
* - hash format: `array('column1' => value1, 'column2' => value2, ...)` |
||||||
|
* - operator format: `array(operator, operand1, operand2, ...)` |
||||||
|
* |
||||||
|
* A condition in hash format represents the following SQL expression in general: |
||||||
|
* `column1=value1 AND column2=value2 AND ...`. In case when a value is an array, |
||||||
|
* an `IN` expression will be generated. And if a value is null, `IS NULL` will be used |
||||||
|
* in the generated expression. Below are some examples: |
||||||
|
* |
||||||
|
* - `array('type' => 1, 'status' => 2)` generates `(type = 1) AND (status = 2)`. |
||||||
|
* - `array('id' => array(1, 2, 3), 'status' => 2)` generates `(id IN (1, 2, 3)) AND (status = 2)`. |
||||||
|
* - `array('status' => null) generates `status IS NULL`. |
||||||
|
* |
||||||
|
* A condition in operator format generates the SQL expression according to the specified operator, which |
||||||
|
* can be one of the followings: |
||||||
|
* |
||||||
|
* - `and`: the operands should be concatenated together using `AND`. For example, |
||||||
|
* `array('and', 'id=1', 'id=2')` will generate `id=1 AND id=2`. If an operand is an array, |
||||||
|
* it will be converted into a string using the rules described here. For example, |
||||||
|
* `array('and', 'type=1', array('or', 'id=1', 'id=2'))` will generate `type=1 AND (id=1 OR id=2)`. |
||||||
|
* The method will NOT do any quoting or escaping. |
||||||
|
* |
||||||
|
* - `or`: similar to the `and` operator except that the operands are concatenated using `OR`. |
||||||
|
* |
||||||
|
* - `between`: operand 1 should be the column name, and operand 2 and 3 should be the |
||||||
|
* starting and ending values of the range that the column is in. |
||||||
|
* For example, `array('between', 'id', 1, 10)` will generate `id BETWEEN 1 AND 10`. |
||||||
|
* |
||||||
|
* - `not between`: similar to `between` except the `BETWEEN` is replaced with `NOT BETWEEN` |
||||||
|
* in the generated condition. |
||||||
|
* |
||||||
|
* - `in`: operand 1 should be a column or DB expression, and operand 2 be an array representing |
||||||
|
* the range of the values that the column or DB expression should be in. For example, |
||||||
|
* `array('in', 'id', array(1, 2, 3))` will generate `id IN (1, 2, 3)`. |
||||||
|
* The method will properly quote the column name and escape values in the range. |
||||||
|
* |
||||||
|
* - `not in`: similar to the `in` operator except that `IN` is replaced with `NOT IN` in the generated condition. |
||||||
|
* |
||||||
|
* - `like`: operand 1 should be a column or DB expression, and operand 2 be a string or an array representing |
||||||
|
* the values that the column or DB expression should be like. |
||||||
|
* For example, `array('like', 'name', '%tester%')` will generate `name LIKE '%tester%'`. |
||||||
|
* When the value range is given as an array, multiple `LIKE` predicates will be generated and concatenated |
||||||
|
* using `AND`. For example, `array('like', 'name', array('%test%', '%sample%'))` will generate |
||||||
|
* `name LIKE '%test%' AND name LIKE '%sample%'`. |
||||||
|
* The method will properly quote the column name and escape values in the range. |
||||||
|
* |
||||||
|
* - `or like`: similar to the `like` operator except that `OR` is used to concatenate the `LIKE` |
||||||
|
* predicates when operand 2 is an array. |
||||||
|
* |
||||||
|
* - `not like`: similar to the `like` operator except that `LIKE` is replaced with `NOT LIKE` |
||||||
|
* in the generated condition. |
||||||
|
* |
||||||
|
* - `or not like`: similar to the `not like` operator except that `OR` is used to concatenate |
||||||
|
* the `NOT LIKE` predicates. |
||||||
|
* |
||||||
|
* @param string|array $condition the conditions that should be put in the WHERE part. |
||||||
|
* @return ActiveQuery the query object itself |
||||||
|
* @see andWhere() |
||||||
|
* @see orWhere() |
||||||
|
*/ |
||||||
|
public function where($condition) |
||||||
|
{ |
||||||
|
$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 'AND' operator. |
||||||
|
* @param string|array $condition the new WHERE condition. Please refer to [[where()]] |
||||||
|
* on how to specify this parameter. |
||||||
|
* @return ActiveQuery the query object itself |
||||||
|
* @see where() |
||||||
|
* @see orWhere() |
||||||
|
*/ |
||||||
|
public function andWhere($condition) |
||||||
|
{ |
||||||
|
if ($this->where === null) { |
||||||
|
$this->where = $condition; |
||||||
|
} else { |
||||||
|
$this->where = array('and', $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 string|array $condition the new WHERE condition. Please refer to [[where()]] |
||||||
|
* on how to specify this parameter. |
||||||
|
* @return ActiveQuery the query object itself |
||||||
|
* @see where() |
||||||
|
* @see andWhere() |
||||||
|
*/ |
||||||
|
public function orWhere($condition) |
||||||
|
{ |
||||||
|
if ($this->where === null) { |
||||||
|
$this->where = $condition; |
||||||
|
} else { |
||||||
|
$this->where = array('or', $this->where, $condition); |
||||||
|
} |
||||||
|
return $this; |
||||||
|
} |
||||||
|
|
||||||
|
private function createModels($rows) |
||||||
|
{ |
||||||
|
$models = array(); |
||||||
|
if ($this->asArray) { |
||||||
|
if ($this->indexBy === null) { |
||||||
|
return $rows; |
||||||
|
} |
||||||
|
foreach ($rows as $row) { |
||||||
|
if (is_string($this->indexBy)) { |
||||||
|
$key = $row[$this->indexBy]; |
||||||
|
} else { |
||||||
|
$key = call_user_func($this->indexBy, $row); |
||||||
|
} |
||||||
|
$models[$key] = $row; |
||||||
|
} |
||||||
|
} else { |
||||||
|
/** @var $class ActiveRecord */ |
||||||
|
$class = $this->modelClass; |
||||||
|
if ($this->indexBy === null) { |
||||||
|
foreach ($rows as $row) { |
||||||
|
$models[] = $class::create($row); |
||||||
|
} |
||||||
|
} else { |
||||||
|
foreach ($rows as $row) { |
||||||
|
$model = $class::create($row); |
||||||
|
if (is_string($this->indexBy)) { |
||||||
|
$key = $model->{$this->indexBy}; |
||||||
|
} else { |
||||||
|
$key = call_user_func($this->indexBy, $model); |
||||||
|
} |
||||||
|
$models[$key] = $model; |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
return $models; |
||||||
|
} |
||||||
|
|
||||||
|
private function populateRelations(&$models, $with) |
||||||
|
{ |
||||||
|
$primaryModel = new $this->modelClass; |
||||||
|
$relations = $this->normalizeRelations($primaryModel, $with); |
||||||
|
foreach ($relations as $name => $relation) { |
||||||
|
if ($relation->asArray === null) { |
||||||
|
// inherit asArray from primary query |
||||||
|
$relation->asArray = $this->asArray; |
||||||
|
} |
||||||
|
$relation->findWith($name, $models); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* @param ActiveRecord $model |
||||||
|
* @param array $with |
||||||
|
* @return ActiveRelation[] |
||||||
|
*/ |
||||||
|
private function normalizeRelations($model, $with) |
||||||
|
{ |
||||||
|
$relations = array(); |
||||||
|
foreach ($with as $name => $callback) { |
||||||
|
if (is_integer($name)) { |
||||||
|
$name = $callback; |
||||||
|
$callback = null; |
||||||
|
} |
||||||
|
if (($pos = strpos($name, '.')) !== false) { |
||||||
|
// with sub-relations |
||||||
|
$childName = substr($name, $pos + 1); |
||||||
|
$name = substr($name, 0, $pos); |
||||||
|
} else { |
||||||
|
$childName = null; |
||||||
|
} |
||||||
|
|
||||||
|
$t = strtolower($name); |
||||||
|
if (!isset($relations[$t])) { |
||||||
|
$relation = $model->getRelation($name); |
||||||
|
$relation->primaryModel = null; |
||||||
|
$relations[$t] = $relation; |
||||||
|
} else { |
||||||
|
$relation = $relations[$t]; |
||||||
|
} |
||||||
|
|
||||||
|
if (isset($childName)) { |
||||||
|
$relation->with[$childName] = $callback; |
||||||
|
} elseif ($callback !== null) { |
||||||
|
call_user_func($callback, $relation); |
||||||
|
} |
||||||
|
} |
||||||
|
return $relations; |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,368 @@ |
|||||||
|
<?php |
||||||
|
/** |
||||||
|
* @link http://www.yiiframework.com/ |
||||||
|
* @copyright Copyright © 2008 Yii Software LLC |
||||||
|
* @license http://www.yiiframework.com/license/ |
||||||
|
*/ |
||||||
|
|
||||||
|
namespace yii\elasticsearch; |
||||||
|
|
||||||
|
use yii\base\InvalidCallException; |
||||||
|
use yii\base\InvalidConfigException; |
||||||
|
use yii\base\InvalidParamException; |
||||||
|
use yii\base\NotSupportedException; |
||||||
|
use yii\base\UnknownMethodException; |
||||||
|
use yii\db\Exception; |
||||||
|
use yii\db\TableSchema; |
||||||
|
use yii\helpers\Inflector; |
||||||
|
use yii\helpers\Json; |
||||||
|
use yii\helpers\StringHelper; |
||||||
|
|
||||||
|
/** |
||||||
|
* ActiveRecord is the base class for classes representing relational data in terms of objects. |
||||||
|
* |
||||||
|
* |
||||||
|
* |
||||||
|
* @author Carsten Brandt <mail@cebe.cc> |
||||||
|
* @since 2.0 |
||||||
|
*/ |
||||||
|
abstract class ActiveRecord extends \yii\db\ActiveRecord |
||||||
|
{ |
||||||
|
/** |
||||||
|
* Returns the database connection used by this AR class. |
||||||
|
* By default, the "elasticsearch" application component is used as the database connection. |
||||||
|
* You may override this method if you want to use a different database connection. |
||||||
|
* @return Connection the database connection used by this AR class. |
||||||
|
*/ |
||||||
|
public static function getDb() |
||||||
|
{ |
||||||
|
return \Yii::$app->elasticsearch; |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* @inheritdoc |
||||||
|
*/ |
||||||
|
public static function findBySql($sql, $params = array()) |
||||||
|
{ |
||||||
|
throw new NotSupportedException('findBySql() is not supported by elasticsearch ActiveRecord'); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
/** |
||||||
|
* Updates the whole table using the provided attribute values and conditions. |
||||||
|
* For example, to change the status to be 1 for all customers whose status is 2: |
||||||
|
* |
||||||
|
* ~~~ |
||||||
|
* Customer::updateAll(array('status' => 1), array('id' => 2)); |
||||||
|
* ~~~ |
||||||
|
* |
||||||
|
* @param array $attributes attribute values (name-value pairs) to be saved into the table |
||||||
|
* @param array $condition the conditions that will be put in the WHERE part of the UPDATE SQL. |
||||||
|
* Please refer to [[ActiveQuery::where()]] on how to specify this parameter. |
||||||
|
* @param array $params this parameter is ignored in redis implementation. |
||||||
|
* @return integer the number of rows updated |
||||||
|
*/ |
||||||
|
public static function updateAll($attributes, $condition = null, $params = array()) |
||||||
|
{ |
||||||
|
if (empty($attributes)) { |
||||||
|
return 0; |
||||||
|
} |
||||||
|
$db = static::getDb(); |
||||||
|
$n=0; |
||||||
|
foreach(static::fetchPks($condition) as $pk) { |
||||||
|
$newPk = $pk; |
||||||
|
$pk = static::buildKey($pk); |
||||||
|
$key = static::tableName() . ':a:' . $pk; |
||||||
|
// save attributes |
||||||
|
$args = array($key); |
||||||
|
foreach($attributes as $attribute => $value) { |
||||||
|
if (isset($newPk[$attribute])) { |
||||||
|
$newPk[$attribute] = $value; |
||||||
|
} |
||||||
|
$args[] = $attribute; |
||||||
|
$args[] = $value; |
||||||
|
} |
||||||
|
$newPk = static::buildKey($newPk); |
||||||
|
$newKey = static::tableName() . ':a:' . $newPk; |
||||||
|
// rename index if pk changed |
||||||
|
if ($newPk != $pk) { |
||||||
|
$db->executeCommand('MULTI'); |
||||||
|
$db->executeCommand('HMSET', $args); |
||||||
|
$db->executeCommand('LINSERT', array(static::tableName(), 'AFTER', $pk, $newPk)); |
||||||
|
$db->executeCommand('LREM', array(static::tableName(), 0, $pk)); |
||||||
|
$db->executeCommand('RENAME', array($key, $newKey)); |
||||||
|
$db->executeCommand('EXEC'); |
||||||
|
} else { |
||||||
|
$db->executeCommand('HMSET', $args); |
||||||
|
} |
||||||
|
$n++; |
||||||
|
} |
||||||
|
return $n; |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* Updates the whole table using the provided counter changes and conditions. |
||||||
|
* For example, to increment all customers' age by 1, |
||||||
|
* |
||||||
|
* ~~~ |
||||||
|
* Customer::updateAllCounters(array('age' => 1)); |
||||||
|
* ~~~ |
||||||
|
* |
||||||
|
* @param array $counters the counters to be updated (attribute name => increment value). |
||||||
|
* Use negative values if you want to decrement the counters. |
||||||
|
* @param array $condition the conditions that will be put in the WHERE part of the UPDATE SQL. |
||||||
|
* Please refer to [[ActiveQuery::where()]] on how to specify this parameter. |
||||||
|
* @param array $params this parameter is ignored in redis implementation. |
||||||
|
* @return integer the number of rows updated |
||||||
|
*/ |
||||||
|
public static function updateAllCounters($counters, $condition = null, $params = array()) |
||||||
|
{ |
||||||
|
if (empty($counters)) { |
||||||
|
return 0; |
||||||
|
} |
||||||
|
$db = static::getDb(); |
||||||
|
$n=0; |
||||||
|
foreach(static::fetchPks($condition) as $pk) { |
||||||
|
$key = static::tableName() . ':a:' . static::buildKey($pk); |
||||||
|
foreach($counters as $attribute => $value) { |
||||||
|
$db->executeCommand('HINCRBY', array($key, $attribute, $value)); |
||||||
|
} |
||||||
|
$n++; |
||||||
|
} |
||||||
|
return $n; |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* Deletes rows in the table using the provided conditions. |
||||||
|
* WARNING: If you do not specify any condition, this method will delete ALL rows in the table. |
||||||
|
* |
||||||
|
* For example, to delete all customers whose status is 3: |
||||||
|
* |
||||||
|
* ~~~ |
||||||
|
* Customer::deleteAll('status = 3'); |
||||||
|
* ~~~ |
||||||
|
* |
||||||
|
* @param array $condition the conditions that will be put in the WHERE part of the DELETE SQL. |
||||||
|
* Please refer to [[ActiveQuery::where()]] on how to specify this parameter. |
||||||
|
* @param array $params this parameter is ignored in redis implementation. |
||||||
|
* @return integer the number of rows deleted |
||||||
|
*/ |
||||||
|
public static function deleteAll($condition = null, $params = array()) |
||||||
|
{ |
||||||
|
$db = static::getDb(); |
||||||
|
$attributeKeys = array(); |
||||||
|
$pks = static::fetchPks($condition); |
||||||
|
$db->executeCommand('MULTI'); |
||||||
|
foreach($pks as $pk) { |
||||||
|
$pk = static::buildKey($pk); |
||||||
|
$db->executeCommand('LREM', array(static::tableName(), 0, $pk)); |
||||||
|
$attributeKeys[] = static::tableName() . ':a:' . $pk; |
||||||
|
} |
||||||
|
if (empty($attributeKeys)) { |
||||||
|
$db->executeCommand('EXEC'); |
||||||
|
return 0; |
||||||
|
} |
||||||
|
$db->executeCommand('DEL', $attributeKeys); |
||||||
|
$result = $db->executeCommand('EXEC'); |
||||||
|
return end($result); |
||||||
|
} |
||||||
|
/** |
||||||
|
* Creates an [[ActiveQuery]] instance. |
||||||
|
* This method is called by [[find()]], [[findBySql()]] and [[count()]] to start a SELECT query. |
||||||
|
* You may override this method to return a customized query (e.g. `CustomerQuery` specified |
||||||
|
* written for querying `Customer` purpose.) |
||||||
|
* @return ActiveQuery the newly created [[ActiveQuery]] instance. |
||||||
|
*/ |
||||||
|
public static function createQuery() |
||||||
|
{ |
||||||
|
return new ActiveQuery(array( |
||||||
|
'modelClass' => get_called_class(), |
||||||
|
)); |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* Declares the name of the database table associated with this AR class. |
||||||
|
* @return string the table name |
||||||
|
*/ |
||||||
|
public static function tableName() |
||||||
|
{ |
||||||
|
return static::getTableSchema()->name; |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* This method is ment to be overridden in redis ActiveRecord subclasses to return a [[RecordSchema]] instance. |
||||||
|
* @return RecordSchema |
||||||
|
* @throws \yii\base\InvalidConfigException |
||||||
|
*/ |
||||||
|
public static function getRecordSchema() |
||||||
|
{ |
||||||
|
throw new InvalidConfigException(__CLASS__.'::getRecordSchema() needs to be overridden in subclasses and return a RecordSchema.'); |
||||||
|
} |
||||||
|
|
||||||
|
public static function primaryKey() |
||||||
|
{ |
||||||
|
return array('id'); |
||||||
|
} |
||||||
|
|
||||||
|
public static function columns() |
||||||
|
{ |
||||||
|
return array('id' => 'integer'); |
||||||
|
} |
||||||
|
|
||||||
|
public static function indexName() |
||||||
|
{ |
||||||
|
return Inflector::pluralize(Inflector::camel2id(StringHelper::basename(get_called_class()), '-')); |
||||||
|
} |
||||||
|
|
||||||
|
public static function indexType() |
||||||
|
{ |
||||||
|
return Inflector::camel2id(StringHelper::basename(get_called_class()), '-'); |
||||||
|
} |
||||||
|
|
||||||
|
private static $_tables; |
||||||
|
/** |
||||||
|
* Returns the schema information of the DB table associated with this AR class. |
||||||
|
* @return TableSchema the schema information of the DB table associated with this AR class. |
||||||
|
* @throws InvalidConfigException if the table for the AR class does not exist. |
||||||
|
*/ |
||||||
|
public static function getTableSchema() |
||||||
|
{ |
||||||
|
$class = get_called_class(); |
||||||
|
if (isset(self::$_tables[$class])) { |
||||||
|
return self::$_tables[$class]; |
||||||
|
} |
||||||
|
return self::$_tables[$class] = new TableSchema(array( |
||||||
|
'schemaName' => static::indexName(), |
||||||
|
'name' => static::indexType(), |
||||||
|
'primaryKey' => static::primaryKey(), |
||||||
|
'columns' => static::columns(), |
||||||
|
)); |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* Declares a `has-one` relation. |
||||||
|
* The declaration is returned in terms of an [[ActiveRelation]] instance |
||||||
|
* through which the related record can be queried and retrieved back. |
||||||
|
* |
||||||
|
* A `has-one` relation means that there is at most one related record matching |
||||||
|
* the criteria set by this relation, e.g., a customer has one country. |
||||||
|
* |
||||||
|
* For example, to declare the `country` relation for `Customer` class, we can write |
||||||
|
* the following code in the `Customer` class: |
||||||
|
* |
||||||
|
* ~~~ |
||||||
|
* public function getCountry() |
||||||
|
* { |
||||||
|
* return $this->hasOne('Country', array('id' => 'country_id')); |
||||||
|
* } |
||||||
|
* ~~~ |
||||||
|
* |
||||||
|
* Note that in the above, the 'id' key in the `$link` parameter refers to an attribute name |
||||||
|
* in the related class `Country`, while the 'country_id' value refers to an attribute name |
||||||
|
* in the current AR class. |
||||||
|
* |
||||||
|
* Call methods declared in [[ActiveRelation]] to further customize the relation. |
||||||
|
* |
||||||
|
* @param string $class the class name of the related record |
||||||
|
* @param array $link the primary-foreign key constraint. The keys of the array refer to |
||||||
|
* the columns in the table associated with the `$class` model, while the values of the |
||||||
|
* array refer to the corresponding columns in the table associated with this AR class. |
||||||
|
* @return ActiveRelation the relation object. |
||||||
|
*/ |
||||||
|
public function hasOne($class, $link) |
||||||
|
{ |
||||||
|
return new ActiveRelation(array( |
||||||
|
'modelClass' => $this->getNamespacedClass($class), |
||||||
|
'primaryModel' => $this, |
||||||
|
'link' => $link, |
||||||
|
'multiple' => false, |
||||||
|
)); |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* Declares a `has-many` relation. |
||||||
|
* The declaration is returned in terms of an [[ActiveRelation]] instance |
||||||
|
* through which the related record can be queried and retrieved back. |
||||||
|
* |
||||||
|
* A `has-many` relation means that there are multiple related records matching |
||||||
|
* the criteria set by this relation, e.g., a customer has many orders. |
||||||
|
* |
||||||
|
* For example, to declare the `orders` relation for `Customer` class, we can write |
||||||
|
* the following code in the `Customer` class: |
||||||
|
* |
||||||
|
* ~~~ |
||||||
|
* public function getOrders() |
||||||
|
* { |
||||||
|
* return $this->hasMany('Order', array('customer_id' => 'id')); |
||||||
|
* } |
||||||
|
* ~~~ |
||||||
|
* |
||||||
|
* Note that in the above, the 'customer_id' key in the `$link` parameter refers to |
||||||
|
* an attribute name in the related class `Order`, while the 'id' value refers to |
||||||
|
* an attribute name in the current AR class. |
||||||
|
* |
||||||
|
* @param string $class the class name of the related record |
||||||
|
* @param array $link the primary-foreign key constraint. The keys of the array refer to |
||||||
|
* the columns in the table associated with the `$class` model, while the values of the |
||||||
|
* array refer to the corresponding columns in the table associated with this AR class. |
||||||
|
* @return ActiveRelation the relation object. |
||||||
|
*/ |
||||||
|
public function hasMany($class, $link) |
||||||
|
{ |
||||||
|
return new ActiveRelation(array( |
||||||
|
'modelClass' => $this->getNamespacedClass($class), |
||||||
|
'primaryModel' => $this, |
||||||
|
'link' => $link, |
||||||
|
'multiple' => true, |
||||||
|
)); |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* @inheritDocs |
||||||
|
*/ |
||||||
|
public function insert($runValidation = true, $attributes = null) |
||||||
|
{ |
||||||
|
if ($runValidation && !$this->validate($attributes)) { |
||||||
|
return false; |
||||||
|
} |
||||||
|
if ($this->beforeSave(true)) { |
||||||
|
$db = static::getDb(); |
||||||
|
$values = $this->getDirtyAttributes($attributes); |
||||||
|
$key = reset($this->primaryKey()); |
||||||
|
$pk = $this->getAttribute($key); |
||||||
|
unset($values[$key]); |
||||||
|
|
||||||
|
// save attributes |
||||||
|
if ($pk === null) { |
||||||
|
$url = '/' . static::indexName() . '/' . static::indexType(); |
||||||
|
$request = $db->http()->post($url, array(), Json::encode($values)); |
||||||
|
} else { |
||||||
|
$url = '/' . static::indexName() . '/' . static::indexType() . '/' . $pk; |
||||||
|
$request = $db->http()->put($url, array(), Json::encode($values)); |
||||||
|
} |
||||||
|
$response = $request->send(); |
||||||
|
$body = Json::decode($response->getBody(true)); |
||||||
|
if (!$body['ok']) { |
||||||
|
return false; |
||||||
|
} |
||||||
|
$this->setOldAttributes($values); |
||||||
|
if ($pk === null) { |
||||||
|
$this->setAttribute($key, $body['_id']); |
||||||
|
} |
||||||
|
$this->afterSave(true); |
||||||
|
return true; |
||||||
|
} |
||||||
|
return false; |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* Returns a value indicating whether the specified operation is transactional in the current [[scenario]]. |
||||||
|
* This method will always return false as transactional operations are not supported by elasticsearch. |
||||||
|
* @param integer $operation the operation to check. Possible values are [[OP_INSERT]], [[OP_UPDATE]] and [[OP_DELETE]]. |
||||||
|
* @return boolean whether the specified operation is transactional in the current [[scenario]]. |
||||||
|
*/ |
||||||
|
public function isTransactional($operation) |
||||||
|
{ |
||||||
|
return false; |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,24 @@ |
|||||||
|
<?php |
||||||
|
/** |
||||||
|
* @link http://www.yiiframework.com/ |
||||||
|
* @copyright Copyright (c) 2008 Yii Software LLC |
||||||
|
* @license http://www.yiiframework.com/license/ |
||||||
|
*/ |
||||||
|
|
||||||
|
namespace yiiunit\data\ar\elasticsearch; |
||||||
|
|
||||||
|
/** |
||||||
|
* ActiveRecord is ... |
||||||
|
* |
||||||
|
* @author Qiang Xue <qiang.xue@gmail.com> |
||||||
|
* @since 2.0 |
||||||
|
*/ |
||||||
|
class ActiveRecord extends \yii\elasticsearch\ActiveRecord |
||||||
|
{ |
||||||
|
public static $db; |
||||||
|
|
||||||
|
public static function getDb() |
||||||
|
{ |
||||||
|
return self::$db; |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,40 @@ |
|||||||
|
<?php |
||||||
|
namespace yiiunit\data\ar\elasticsearch; |
||||||
|
|
||||||
|
/** |
||||||
|
* Class Customer |
||||||
|
* |
||||||
|
* @property integer $id |
||||||
|
* @property string $name |
||||||
|
* @property string $email |
||||||
|
* @property string $address |
||||||
|
* @property integer $status |
||||||
|
*/ |
||||||
|
class Customer extends ActiveRecord |
||||||
|
{ |
||||||
|
const STATUS_ACTIVE = 1; |
||||||
|
const STATUS_INACTIVE = 2; |
||||||
|
|
||||||
|
public $status2; |
||||||
|
|
||||||
|
public static function columns() |
||||||
|
{ |
||||||
|
return array( |
||||||
|
'id' => 'integer', |
||||||
|
'name' => 'string', |
||||||
|
'email' => 'string', |
||||||
|
'address' => 'string', |
||||||
|
'status' => 'integer', |
||||||
|
); |
||||||
|
} |
||||||
|
|
||||||
|
public function getOrders() |
||||||
|
{ |
||||||
|
return $this->hasMany('Order', array('customer_id' => 'id'))->orderBy('id'); |
||||||
|
} |
||||||
|
|
||||||
|
public static function active($query) |
||||||
|
{ |
||||||
|
$query->andWhere('status=1'); |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,22 @@ |
|||||||
|
<?php |
||||||
|
|
||||||
|
namespace yiiunit\data\ar\elasticsearch; |
||||||
|
|
||||||
|
/** |
||||||
|
* Class Item |
||||||
|
* |
||||||
|
* @property integer $id |
||||||
|
* @property string $name |
||||||
|
* @property integer $category_id |
||||||
|
*/ |
||||||
|
class Item extends ActiveRecord |
||||||
|
{ |
||||||
|
public static function columns() |
||||||
|
{ |
||||||
|
return array( |
||||||
|
'id' => 'integer', |
||||||
|
'name' => 'string', |
||||||
|
'category_id' => 'integer', |
||||||
|
); |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,59 @@ |
|||||||
|
<?php |
||||||
|
|
||||||
|
namespace yiiunit\data\ar\elasticsearch; |
||||||
|
|
||||||
|
/** |
||||||
|
* Class Order |
||||||
|
* |
||||||
|
* @property integer $id |
||||||
|
* @property integer $customer_id |
||||||
|
* @property integer $create_time |
||||||
|
* @property string $total |
||||||
|
*/ |
||||||
|
class Order extends ActiveRecord |
||||||
|
{ |
||||||
|
public static function columns() |
||||||
|
{ |
||||||
|
return array( |
||||||
|
'id' => 'integer', |
||||||
|
'customer_id' => 'integer', |
||||||
|
'create_time' => 'integer', |
||||||
|
'total' => 'integer', |
||||||
|
); |
||||||
|
} |
||||||
|
|
||||||
|
public function getCustomer() |
||||||
|
{ |
||||||
|
return $this->hasOne('Customer', array('id' => 'customer_id')); |
||||||
|
} |
||||||
|
|
||||||
|
public function getOrderItems() |
||||||
|
{ |
||||||
|
return $this->hasMany('OrderItem', array('order_id' => 'id')); |
||||||
|
} |
||||||
|
|
||||||
|
public function getItems() |
||||||
|
{ |
||||||
|
return $this->hasMany('Item', array('id' => 'item_id')) |
||||||
|
->via('orderItems', function ($q) { |
||||||
|
// additional query configuration |
||||||
|
})->orderBy('id'); |
||||||
|
} |
||||||
|
|
||||||
|
public function getBooks() |
||||||
|
{ |
||||||
|
return $this->hasMany('Item', array('id' => 'item_id')) |
||||||
|
->viaTable('tbl_order_item', array('order_id' => 'id')) |
||||||
|
->where(array('category_id' => 1)); |
||||||
|
} |
||||||
|
|
||||||
|
public function beforeSave($insert) |
||||||
|
{ |
||||||
|
if (parent::beforeSave($insert)) { |
||||||
|
$this->create_time = time(); |
||||||
|
return true; |
||||||
|
} else { |
||||||
|
return false; |
||||||
|
} |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,34 @@ |
|||||||
|
<?php |
||||||
|
|
||||||
|
namespace yiiunit\data\ar\elasticsearch; |
||||||
|
|
||||||
|
/** |
||||||
|
* Class OrderItem |
||||||
|
* |
||||||
|
* @property integer $order_id |
||||||
|
* @property integer $item_id |
||||||
|
* @property integer $quantity |
||||||
|
* @property string $subtotal |
||||||
|
*/ |
||||||
|
class OrderItem extends ActiveRecord |
||||||
|
{ |
||||||
|
public static function columns() |
||||||
|
{ |
||||||
|
return array( |
||||||
|
'order_id' => 'integer', |
||||||
|
'item_id' => 'integer', |
||||||
|
'quantity' => 'integer', |
||||||
|
'subtotal' => 'integer', |
||||||
|
); |
||||||
|
} |
||||||
|
|
||||||
|
public function getOrder() |
||||||
|
{ |
||||||
|
return $this->hasOne('Order', array('id' => 'order_id')); |
||||||
|
} |
||||||
|
|
||||||
|
public function getItem() |
||||||
|
{ |
||||||
|
return $this->hasOne('Item', array('id' => 'item_id')); |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,473 @@ |
|||||||
|
<?php |
||||||
|
|
||||||
|
namespace yiiunit\framework\elasticsearch; |
||||||
|
|
||||||
|
use yii\db\Query; |
||||||
|
use yii\redis\ActiveQuery; |
||||||
|
use yiiunit\data\ar\elasticsearch\ActiveRecord; |
||||||
|
use yiiunit\data\ar\elasticsearch\Customer; |
||||||
|
use yiiunit\data\ar\elasticsearch\OrderItem; |
||||||
|
use yiiunit\data\ar\elasticsearch\Order; |
||||||
|
use yiiunit\data\ar\elasticsearch\Item; |
||||||
|
|
||||||
|
class ActiveRecordTest extends ElasticSearchTestCase |
||||||
|
{ |
||||||
|
public function setUp() |
||||||
|
{ |
||||||
|
parent::setUp(); |
||||||
|
ActiveRecord::$db = $this->getConnection(); |
||||||
|
|
||||||
|
$customer = new Customer(); |
||||||
|
$customer->setAttributes(array('id' => 1, 'email' => 'user1@example.com', 'name' => 'user1', 'address' => 'address1', 'status' => 1), false); |
||||||
|
$customer->save(false); |
||||||
|
$customer = new Customer(); |
||||||
|
$customer->setAttributes(array('id' => 2, 'email' => 'user2@example.com', 'name' => 'user2', 'address' => 'address2', 'status' => 1), false); |
||||||
|
$customer->save(false); |
||||||
|
$customer = new Customer(); |
||||||
|
$customer->setAttributes(array('id' => 3, 'email' => 'user3@example.com', 'name' => 'user3', 'address' => 'address3', 'status' => 2), false); |
||||||
|
$customer->save(false); |
||||||
|
|
||||||
|
// INSERT INTO tbl_category (name) VALUES ('Books'); |
||||||
|
// INSERT INTO tbl_category (name) VALUES ('Movies'); |
||||||
|
|
||||||
|
$item = new Item(); |
||||||
|
$item->setAttributes(array('id' => 1, 'name' => 'Agile Web Application Development with Yii1.1 and PHP5', 'category_id' => 1), false); |
||||||
|
$item->save(false); |
||||||
|
$item = new Item(); |
||||||
|
$item->setAttributes(array('id' => 2, 'name' => 'Yii 1.1 Application Development Cookbook', 'category_id' => 1), false); |
||||||
|
$item->save(false); |
||||||
|
$item = new Item(); |
||||||
|
$item->setAttributes(array('id' => 3, 'name' => 'Ice Age', 'category_id' => 2), false); |
||||||
|
$item->save(false); |
||||||
|
$item = new Item(); |
||||||
|
$item->setAttributes(array('id' => 4, 'name' => 'Toy Story', 'category_id' => 2), false); |
||||||
|
$item->save(false); |
||||||
|
$item = new Item(); |
||||||
|
$item->setAttributes(array('id' => 5, 'name' => 'Cars', 'category_id' => 2), false); |
||||||
|
$item->save(false); |
||||||
|
|
||||||
|
$order = new Order(); |
||||||
|
$order->setAttributes(array('id' => 1, 'customer_id' => 1, 'create_time' => 1325282384, 'total' => 110.0), false); |
||||||
|
$order->save(false); |
||||||
|
$order = new Order(); |
||||||
|
$order->setAttributes(array('id' => 2, 'customer_id' => 2, 'create_time' => 1325334482, 'total' => 33.0), false); |
||||||
|
$order->save(false); |
||||||
|
$order = new Order(); |
||||||
|
$order->setAttributes(array('id' => 3, 'customer_id' => 2, 'create_time' => 1325502201, 'total' => 40.0), false); |
||||||
|
$order->save(false); |
||||||
|
|
||||||
|
// $orderItem = new OrderItem(); |
||||||
|
// $orderItem->setAttributes(array('order_id' => 1, 'item_id' => 1, 'quantity' => 1, 'subtotal' => 30.0), false); |
||||||
|
// $orderItem->save(false); |
||||||
|
// $orderItem = new OrderItem(); |
||||||
|
// $orderItem->setAttributes(array('order_id' => 1, 'item_id' => 2, 'quantity' => 2, 'subtotal' => 40.0), false); |
||||||
|
// $orderItem->save(false); |
||||||
|
// $orderItem = new OrderItem(); |
||||||
|
// $orderItem->setAttributes(array('order_id' => 2, 'item_id' => 4, 'quantity' => 1, 'subtotal' => 10.0), false); |
||||||
|
// $orderItem->save(false); |
||||||
|
// $orderItem = new OrderItem(); |
||||||
|
// $orderItem->setAttributes(array('order_id' => 2, 'item_id' => 5, 'quantity' => 1, 'subtotal' => 15.0), false); |
||||||
|
// $orderItem->save(false); |
||||||
|
// $orderItem = new OrderItem(); |
||||||
|
// $orderItem->setAttributes(array('order_id' => 2, 'item_id' => 3, 'quantity' => 1, 'subtotal' => 8.0), false); |
||||||
|
// $orderItem->save(false); |
||||||
|
// $orderItem = new OrderItem(); |
||||||
|
// $orderItem->setAttributes(array('order_id' => 3, 'item_id' => 2, 'quantity' => 1, 'subtotal' => 40.0), false); |
||||||
|
// $orderItem->save(false); |
||||||
|
} |
||||||
|
|
||||||
|
public function testFind() |
||||||
|
{ |
||||||
|
// find one |
||||||
|
$result = Customer::find(); |
||||||
|
$this->assertTrue($result instanceof ActiveQuery); |
||||||
|
$customer = $result->one(); |
||||||
|
$this->assertTrue($customer instanceof Customer); |
||||||
|
|
||||||
|
// find all |
||||||
|
$customers = Customer::find()->all(); |
||||||
|
$this->assertEquals(3, count($customers)); |
||||||
|
$this->assertTrue($customers[0] instanceof Customer); |
||||||
|
$this->assertTrue($customers[1] instanceof Customer); |
||||||
|
$this->assertTrue($customers[2] instanceof Customer); |
||||||
|
|
||||||
|
// find by a single primary key |
||||||
|
$customer = Customer::find(2); |
||||||
|
$this->assertTrue($customer instanceof Customer); |
||||||
|
$this->assertEquals('user2', $customer->name); |
||||||
|
$customer = Customer::find(5); |
||||||
|
$this->assertNull($customer); |
||||||
|
|
||||||
|
// query scalar |
||||||
|
$customerName = Customer::find()->where(array('id' => 2))->scalar('name'); |
||||||
|
$this->assertEquals('user2', $customerName); |
||||||
|
|
||||||
|
// find by column values |
||||||
|
$customer = Customer::find(array('id' => 2, 'name' => 'user2')); |
||||||
|
$this->assertTrue($customer instanceof Customer); |
||||||
|
$this->assertEquals('user2', $customer->name); |
||||||
|
$customer = Customer::find(array('id' => 2, 'name' => 'user1')); |
||||||
|
$this->assertNull($customer); |
||||||
|
$customer = Customer::find(array('id' => 5)); |
||||||
|
$this->assertNull($customer); |
||||||
|
|
||||||
|
// find by attributes |
||||||
|
$customer = Customer::find()->where(array('name' => 'user2'))->one(); |
||||||
|
$this->assertTrue($customer instanceof Customer); |
||||||
|
$this->assertEquals(2, $customer->id); |
||||||
|
|
||||||
|
// find count, sum, average, min, max, scalar |
||||||
|
$this->assertEquals(3, Customer::find()->count()); |
||||||
|
$this->assertEquals(6, Customer::find()->sum('id')); |
||||||
|
$this->assertEquals(2, Customer::find()->average('id')); |
||||||
|
$this->assertEquals(1, Customer::find()->min('id')); |
||||||
|
$this->assertEquals(3, Customer::find()->max('id')); |
||||||
|
|
||||||
|
// scope |
||||||
|
$this->assertEquals(2, Customer::find()->active()->count()); |
||||||
|
|
||||||
|
// asArray |
||||||
|
$customer = Customer::find()->where(array('id' => 2))->asArray()->one(); |
||||||
|
$this->assertEquals(array( |
||||||
|
'id' => '2', |
||||||
|
'email' => 'user2@example.com', |
||||||
|
'name' => 'user2', |
||||||
|
'address' => 'address2', |
||||||
|
'status' => '1', |
||||||
|
), $customer); |
||||||
|
|
||||||
|
// indexBy |
||||||
|
$customers = Customer::find()->indexBy('name')->all(); |
||||||
|
$this->assertEquals(3, count($customers)); |
||||||
|
$this->assertTrue($customers['user1'] instanceof Customer); |
||||||
|
$this->assertTrue($customers['user2'] instanceof Customer); |
||||||
|
$this->assertTrue($customers['user3'] instanceof Customer); |
||||||
|
|
||||||
|
// indexBy callable |
||||||
|
$customers = Customer::find()->indexBy(function ($customer) { |
||||||
|
return $customer->id . '-' . $customer->name; |
||||||
|
// })->orderBy('id')->all(); |
||||||
|
})->all(); |
||||||
|
$this->assertEquals(3, count($customers)); |
||||||
|
$this->assertTrue($customers['1-user1'] instanceof Customer); |
||||||
|
$this->assertTrue($customers['2-user2'] instanceof Customer); |
||||||
|
$this->assertTrue($customers['3-user3'] instanceof Customer); |
||||||
|
} |
||||||
|
|
||||||
|
public function testFindCount() |
||||||
|
{ |
||||||
|
$this->assertEquals(3, Customer::find()->count()); |
||||||
|
$this->assertEquals(1, Customer::find()->limit(1)->count()); |
||||||
|
$this->assertEquals(2, Customer::find()->limit(2)->count()); |
||||||
|
$this->assertEquals(1, Customer::find()->offset(2)->limit(2)->count()); |
||||||
|
} |
||||||
|
|
||||||
|
public function testFindLimit() |
||||||
|
{ |
||||||
|
// all() |
||||||
|
$customers = Customer::find()->all(); |
||||||
|
$this->assertEquals(3, count($customers)); |
||||||
|
|
||||||
|
$customers = Customer::find()->limit(1)->all(); |
||||||
|
$this->assertEquals(1, count($customers)); |
||||||
|
$this->assertEquals('user1', $customers[0]->name); |
||||||
|
|
||||||
|
$customers = Customer::find()->limit(1)->offset(1)->all(); |
||||||
|
$this->assertEquals(1, count($customers)); |
||||||
|
$this->assertEquals('user2', $customers[0]->name); |
||||||
|
|
||||||
|
$customers = Customer::find()->limit(1)->offset(2)->all(); |
||||||
|
$this->assertEquals(1, count($customers)); |
||||||
|
$this->assertEquals('user3', $customers[0]->name); |
||||||
|
|
||||||
|
$customers = Customer::find()->limit(2)->offset(1)->all(); |
||||||
|
$this->assertEquals(2, count($customers)); |
||||||
|
$this->assertEquals('user2', $customers[0]->name); |
||||||
|
$this->assertEquals('user3', $customers[1]->name); |
||||||
|
|
||||||
|
$customers = Customer::find()->limit(2)->offset(3)->all(); |
||||||
|
$this->assertEquals(0, count($customers)); |
||||||
|
|
||||||
|
// one() |
||||||
|
$customer = Customer::find()->one(); |
||||||
|
$this->assertEquals('user1', $customer->name); |
||||||
|
|
||||||
|
$customer = Customer::find()->offset(0)->one(); |
||||||
|
$this->assertEquals('user1', $customer->name); |
||||||
|
|
||||||
|
$customer = Customer::find()->offset(1)->one(); |
||||||
|
$this->assertEquals('user2', $customer->name); |
||||||
|
|
||||||
|
$customer = Customer::find()->offset(2)->one(); |
||||||
|
$this->assertEquals('user3', $customer->name); |
||||||
|
|
||||||
|
$customer = Customer::find()->offset(3)->one(); |
||||||
|
$this->assertNull($customer); |
||||||
|
|
||||||
|
} |
||||||
|
|
||||||
|
public function testFindComplexCondition() |
||||||
|
{ |
||||||
|
$this->assertEquals(2, Customer::find()->where(array('OR', array('id' => 1), array('id' => 2)))->count()); |
||||||
|
$this->assertEquals(2, count(Customer::find()->where(array('OR', array('id' => 1), array('id' => 2)))->all())); |
||||||
|
|
||||||
|
$this->assertEquals(2, Customer::find()->where(array('id' => array(1,2)))->count()); |
||||||
|
$this->assertEquals(2, count(Customer::find()->where(array('id' => array(1,2)))->all())); |
||||||
|
|
||||||
|
$this->assertEquals(1, Customer::find()->where(array('AND', array('id' => array(2,3)), array('BETWEEN', 'status', 2, 4)))->count()); |
||||||
|
$this->assertEquals(1, count(Customer::find()->where(array('AND', array('id' => array(2,3)), array('BETWEEN', 'status', 2, 4)))->all())); |
||||||
|
} |
||||||
|
|
||||||
|
public function testSum() |
||||||
|
{ |
||||||
|
$this->assertEquals(6, OrderItem::find()->count()); |
||||||
|
$this->assertEquals(7, OrderItem::find()->sum('quantity')); |
||||||
|
} |
||||||
|
|
||||||
|
public function testFindColumn() |
||||||
|
{ |
||||||
|
$this->assertEquals(array('user1', 'user2', 'user3'), Customer::find()->column('name')); |
||||||
|
// TODO $this->assertEquals(array('user3', 'user2', 'user1'), Customer::find()->orderBy(array('name' => Query::SORT_DESC))->column('name')); |
||||||
|
} |
||||||
|
|
||||||
|
public function testExists() |
||||||
|
{ |
||||||
|
$this->assertTrue(Customer::find()->where(array('id' => 2))->exists()); |
||||||
|
$this->assertFalse(Customer::find()->where(array('id' => 5))->exists()); |
||||||
|
} |
||||||
|
|
||||||
|
public function testFindLazy() |
||||||
|
{ |
||||||
|
/** @var $customer Customer */ |
||||||
|
$customer = Customer::find(2); |
||||||
|
$orders = $customer->orders; |
||||||
|
$this->assertEquals(2, count($orders)); |
||||||
|
|
||||||
|
$orders = $customer->getOrders()->where(array('id' => 3))->all(); |
||||||
|
$this->assertEquals(1, count($orders)); |
||||||
|
$this->assertEquals(3, $orders[0]->id); |
||||||
|
} |
||||||
|
|
||||||
|
public function testFindEager() |
||||||
|
{ |
||||||
|
$customers = Customer::find()->with('orders')->all(); |
||||||
|
$this->assertEquals(3, count($customers)); |
||||||
|
$this->assertEquals(1, count($customers[0]->orders)); |
||||||
|
$this->assertEquals(2, count($customers[1]->orders)); |
||||||
|
} |
||||||
|
|
||||||
|
public function testFindLazyVia() |
||||||
|
{ |
||||||
|
/** @var $order Order */ |
||||||
|
$order = Order::find(1); |
||||||
|
$this->assertEquals(1, $order->id); |
||||||
|
$this->assertEquals(2, count($order->items)); |
||||||
|
$this->assertEquals(1, $order->items[0]->id); |
||||||
|
$this->assertEquals(2, $order->items[1]->id); |
||||||
|
|
||||||
|
$order = Order::find(1); |
||||||
|
$order->id = 100; |
||||||
|
$this->assertEquals(array(), $order->items); |
||||||
|
} |
||||||
|
|
||||||
|
public function testFindEagerViaRelation() |
||||||
|
{ |
||||||
|
$orders = Order::find()->with('items')->all(); |
||||||
|
$this->assertEquals(3, count($orders)); |
||||||
|
$order = $orders[0]; |
||||||
|
$this->assertEquals(1, $order->id); |
||||||
|
$this->assertEquals(2, count($order->items)); |
||||||
|
$this->assertEquals(1, $order->items[0]->id); |
||||||
|
$this->assertEquals(2, $order->items[1]->id); |
||||||
|
} |
||||||
|
|
||||||
|
public function testFindNestedRelation() |
||||||
|
{ |
||||||
|
$customers = Customer::find()->with('orders', 'orders.items')->all(); |
||||||
|
$this->assertEquals(3, count($customers)); |
||||||
|
$this->assertEquals(1, count($customers[0]->orders)); |
||||||
|
$this->assertEquals(2, count($customers[1]->orders)); |
||||||
|
$this->assertEquals(0, count($customers[2]->orders)); |
||||||
|
$this->assertEquals(2, count($customers[0]->orders[0]->items)); |
||||||
|
$this->assertEquals(3, count($customers[1]->orders[0]->items)); |
||||||
|
$this->assertEquals(1, count($customers[1]->orders[1]->items)); |
||||||
|
} |
||||||
|
|
||||||
|
public function testLink() |
||||||
|
{ |
||||||
|
$customer = Customer::find(2); |
||||||
|
$this->assertEquals(2, count($customer->orders)); |
||||||
|
|
||||||
|
// has many |
||||||
|
$order = new Order; |
||||||
|
$order->total = 100; |
||||||
|
$this->assertTrue($order->isNewRecord); |
||||||
|
$customer->link('orders', $order); |
||||||
|
$this->assertEquals(3, count($customer->orders)); |
||||||
|
$this->assertFalse($order->isNewRecord); |
||||||
|
$this->assertEquals(3, count($customer->getOrders()->all())); |
||||||
|
$this->assertEquals(2, $order->customer_id); |
||||||
|
|
||||||
|
// belongs to |
||||||
|
$order = new Order; |
||||||
|
$order->total = 100; |
||||||
|
$this->assertTrue($order->isNewRecord); |
||||||
|
$customer = Customer::find(1); |
||||||
|
$this->assertNull($order->customer); |
||||||
|
$order->link('customer', $customer); |
||||||
|
$this->assertFalse($order->isNewRecord); |
||||||
|
$this->assertEquals(1, $order->customer_id); |
||||||
|
$this->assertEquals(1, $order->customer->id); |
||||||
|
|
||||||
|
// via model |
||||||
|
$order = Order::find(1); |
||||||
|
$this->assertEquals(2, count($order->items)); |
||||||
|
$this->assertEquals(2, count($order->orderItems)); |
||||||
|
$orderItem = OrderItem::find(array('order_id' => 1, 'item_id' => 3)); |
||||||
|
$this->assertNull($orderItem); |
||||||
|
$item = Item::find(3); |
||||||
|
$order->link('items', $item, array('quantity' => 10, 'subtotal' => 100)); |
||||||
|
$this->assertEquals(3, count($order->items)); |
||||||
|
$this->assertEquals(3, count($order->orderItems)); |
||||||
|
$orderItem = OrderItem::find(array('order_id' => 1, 'item_id' => 3)); |
||||||
|
$this->assertTrue($orderItem instanceof OrderItem); |
||||||
|
$this->assertEquals(10, $orderItem->quantity); |
||||||
|
$this->assertEquals(100, $orderItem->subtotal); |
||||||
|
} |
||||||
|
|
||||||
|
public function testUnlink() |
||||||
|
{ |
||||||
|
// has many |
||||||
|
$customer = Customer::find(2); |
||||||
|
$this->assertEquals(2, count($customer->orders)); |
||||||
|
$customer->unlink('orders', $customer->orders[1], true); |
||||||
|
$this->assertEquals(1, count($customer->orders)); |
||||||
|
$this->assertNull(Order::find(3)); |
||||||
|
|
||||||
|
// via model |
||||||
|
$order = Order::find(2); |
||||||
|
$this->assertEquals(3, count($order->items)); |
||||||
|
$this->assertEquals(3, count($order->orderItems)); |
||||||
|
$order->unlink('items', $order->items[2], true); |
||||||
|
$this->assertEquals(2, count($order->items)); |
||||||
|
$this->assertEquals(2, count($order->orderItems)); |
||||||
|
} |
||||||
|
|
||||||
|
public function testInsertNoPk() |
||||||
|
{ |
||||||
|
$customer = new Customer; |
||||||
|
$customer->email = 'user4@example.com'; |
||||||
|
$customer->name = 'user4'; |
||||||
|
$customer->address = 'address4'; |
||||||
|
|
||||||
|
$this->assertNull($customer->id); |
||||||
|
$this->assertTrue($customer->isNewRecord); |
||||||
|
|
||||||
|
$customer->save(); |
||||||
|
|
||||||
|
$this->assertNotNull($customer->id); |
||||||
|
$this->assertFalse($customer->isNewRecord); |
||||||
|
} |
||||||
|
|
||||||
|
public function testInsertPk() |
||||||
|
{ |
||||||
|
$customer = new Customer; |
||||||
|
$customer->id = 5; |
||||||
|
$customer->email = 'user5@example.com'; |
||||||
|
$customer->name = 'user5'; |
||||||
|
$customer->address = 'address5'; |
||||||
|
|
||||||
|
$this->assertTrue($customer->isNewRecord); |
||||||
|
|
||||||
|
$customer->save(); |
||||||
|
|
||||||
|
$this->assertEquals(5, $customer->id); |
||||||
|
$this->assertFalse($customer->isNewRecord); |
||||||
|
} |
||||||
|
|
||||||
|
public function testUpdate() |
||||||
|
{ |
||||||
|
// save |
||||||
|
$customer = Customer::find(2); |
||||||
|
$this->assertTrue($customer instanceof Customer); |
||||||
|
$this->assertEquals('user2', $customer->name); |
||||||
|
$this->assertFalse($customer->isNewRecord); |
||||||
|
$customer->name = 'user2x'; |
||||||
|
$customer->save(); |
||||||
|
$this->assertEquals('user2x', $customer->name); |
||||||
|
$this->assertFalse($customer->isNewRecord); |
||||||
|
$customer2 = Customer::find(2); |
||||||
|
$this->assertEquals('user2x', $customer2->name); |
||||||
|
|
||||||
|
// updateAll |
||||||
|
$customer = Customer::find(3); |
||||||
|
$this->assertEquals('user3', $customer->name); |
||||||
|
$ret = Customer::updateAll(array( |
||||||
|
'name' => 'temp', |
||||||
|
), array('id' => 3)); |
||||||
|
$this->assertEquals(1, $ret); |
||||||
|
$customer = Customer::find(3); |
||||||
|
$this->assertEquals('temp', $customer->name); |
||||||
|
} |
||||||
|
|
||||||
|
public function testUpdateCounters() |
||||||
|
{ |
||||||
|
// updateCounters |
||||||
|
$pk = array('order_id' => 2, 'item_id' => 4); |
||||||
|
$orderItem = OrderItem::find($pk); |
||||||
|
$this->assertEquals(1, $orderItem->quantity); |
||||||
|
$ret = $orderItem->updateCounters(array('quantity' => -1)); |
||||||
|
$this->assertTrue($ret); |
||||||
|
$this->assertEquals(0, $orderItem->quantity); |
||||||
|
$orderItem = OrderItem::find($pk); |
||||||
|
$this->assertEquals(0, $orderItem->quantity); |
||||||
|
|
||||||
|
// updateAllCounters |
||||||
|
$pk = array('order_id' => 1, 'item_id' => 2); |
||||||
|
$orderItem = OrderItem::find($pk); |
||||||
|
$this->assertEquals(2, $orderItem->quantity); |
||||||
|
$ret = OrderItem::updateAllCounters(array( |
||||||
|
'quantity' => 3, |
||||||
|
'subtotal' => -10, |
||||||
|
), $pk); |
||||||
|
$this->assertEquals(1, $ret); |
||||||
|
$orderItem = OrderItem::find($pk); |
||||||
|
$this->assertEquals(5, $orderItem->quantity); |
||||||
|
$this->assertEquals(30, $orderItem->subtotal); |
||||||
|
} |
||||||
|
|
||||||
|
public function testUpdatePk() |
||||||
|
{ |
||||||
|
// updateCounters |
||||||
|
$pk = array('order_id' => 2, 'item_id' => 4); |
||||||
|
$orderItem = OrderItem::find($pk); |
||||||
|
$this->assertEquals(2, $orderItem->order_id); |
||||||
|
$this->assertEquals(4, $orderItem->item_id); |
||||||
|
|
||||||
|
$orderItem->order_id = 2; |
||||||
|
$orderItem->item_id = 10; |
||||||
|
$orderItem->save(); |
||||||
|
|
||||||
|
$this->assertNull(OrderItem::find($pk)); |
||||||
|
$this->assertNotNull(OrderItem::find(array('order_id' => 2, 'item_id' => 10))); |
||||||
|
} |
||||||
|
|
||||||
|
public function testDelete() |
||||||
|
{ |
||||||
|
// delete |
||||||
|
$customer = Customer::find(2); |
||||||
|
$this->assertTrue($customer instanceof Customer); |
||||||
|
$this->assertEquals('user2', $customer->name); |
||||||
|
$customer->delete(); |
||||||
|
$customer = Customer::find(2); |
||||||
|
$this->assertNull($customer); |
||||||
|
|
||||||
|
// deleteAll |
||||||
|
$customers = Customer::find()->all(); |
||||||
|
$this->assertEquals(2, count($customers)); |
||||||
|
$ret = Customer::deleteAll(); |
||||||
|
$this->assertEquals(2, $ret); |
||||||
|
$customers = Customer::find()->all(); |
||||||
|
$this->assertEquals(0, count($customers)); |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,66 @@ |
|||||||
|
<?php |
||||||
|
|
||||||
|
namespace yiiunit\framework\elasticsearch; |
||||||
|
|
||||||
|
use yii\redis\Connection; |
||||||
|
|
||||||
|
class ElasticSearchConnectionTest extends ElasticSearchTestCase |
||||||
|
{ |
||||||
|
/** |
||||||
|
* Empty DSN should throw exception |
||||||
|
* @expectedException \yii\base\InvalidConfigException |
||||||
|
*/ |
||||||
|
public function testEmptyDSN() |
||||||
|
{ |
||||||
|
$db = new Connection(); |
||||||
|
$db->open(); |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* test connection to redis and selection of db |
||||||
|
*/ |
||||||
|
public function testConnect() |
||||||
|
{ |
||||||
|
$db = new Connection(); |
||||||
|
$db->dsn = 'redis://localhost:6379'; |
||||||
|
$db->open(); |
||||||
|
$this->assertTrue($db->ping()); |
||||||
|
$db->set('YIITESTKEY', 'YIITESTVALUE'); |
||||||
|
$db->close(); |
||||||
|
|
||||||
|
$db = new Connection(); |
||||||
|
$db->dsn = 'redis://localhost:6379/0'; |
||||||
|
$db->open(); |
||||||
|
$this->assertEquals('YIITESTVALUE', $db->get('YIITESTKEY')); |
||||||
|
$db->close(); |
||||||
|
|
||||||
|
$db = new Connection(); |
||||||
|
$db->dsn = 'redis://localhost:6379/1'; |
||||||
|
$db->open(); |
||||||
|
$this->assertNull($db->get('YIITESTKEY')); |
||||||
|
$db->close(); |
||||||
|
} |
||||||
|
|
||||||
|
public function keyValueData() |
||||||
|
{ |
||||||
|
return array( |
||||||
|
array(123), |
||||||
|
array(-123), |
||||||
|
array(0), |
||||||
|
array('test'), |
||||||
|
array("test\r\ntest"), |
||||||
|
array(''), |
||||||
|
); |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* @dataProvider keyValueData |
||||||
|
*/ |
||||||
|
public function testStoreGet($data) |
||||||
|
{ |
||||||
|
$db = $this->getConnection(true); |
||||||
|
|
||||||
|
$db->set('hi', $data); |
||||||
|
$this->assertEquals($data, $db->get('hi')); |
||||||
|
} |
||||||
|
} |
@ -0,0 +1,48 @@ |
|||||||
|
<?php |
||||||
|
|
||||||
|
namespace yiiunit\framework\elasticsearch; |
||||||
|
|
||||||
|
use yii\elasticsearch\Connection; |
||||||
|
use yiiunit\TestCase; |
||||||
|
|
||||||
|
/** |
||||||
|
* RedisTestCase is the base class for all redis related test cases |
||||||
|
*/ |
||||||
|
class ElasticSearchTestCase extends TestCase |
||||||
|
{ |
||||||
|
protected function setUp() |
||||||
|
{ |
||||||
|
$this->mockApplication(); |
||||||
|
|
||||||
|
$databases = $this->getParam('databases'); |
||||||
|
$params = isset($databases['elasticsearch']) ? $databases['elasticsearch'] : null; |
||||||
|
if ($params === null || !isset($params['dsn'])) { |
||||||
|
$this->markTestSkipped('No elasticsearch server connection configured.'); |
||||||
|
} |
||||||
|
$dsn = explode('/', $params['dsn']); |
||||||
|
$host = $dsn[2]; |
||||||
|
if (strpos($host, ':')===false) { |
||||||
|
$host .= ':9200'; |
||||||
|
} |
||||||
|
if(!@stream_socket_client($host, $errorNumber, $errorDescription, 0.5)) { |
||||||
|
$this->markTestSkipped('No elasticsearch server running at ' . $params['dsn'] . ' : ' . $errorNumber . ' - ' . $errorDescription); |
||||||
|
} |
||||||
|
|
||||||
|
parent::setUp(); |
||||||
|
} |
||||||
|
|
||||||
|
/** |
||||||
|
* @param bool $reset whether to clean up the test database |
||||||
|
* @return Connection |
||||||
|
*/ |
||||||
|
public function getConnection($reset = true) |
||||||
|
{ |
||||||
|
$databases = $this->getParam('databases'); |
||||||
|
$params = isset($databases['elasticsearch']) ? $databases['elasticsearch'] : array(); |
||||||
|
$db = new Connection; |
||||||
|
if ($reset) { |
||||||
|
$db->open(); |
||||||
|
} |
||||||
|
return $db; |
||||||
|
} |
||||||
|
} |
Loading…
Reference in new issue