mirror of
https://github.com/elyby/accounts.git
synced 2024-11-30 02:32:26 +05:30
Логика генерации значения первичного ключа для строк вынесена в поведение
This commit is contained in:
parent
59f51451d0
commit
53d56d6b97
61
common/behaviors/PrimaryKeyValueBehavior.php
Normal file
61
common/behaviors/PrimaryKeyValueBehavior.php
Normal file
@ -0,0 +1,61 @@
|
|||||||
|
<?php
|
||||||
|
namespace common\behaviors;
|
||||||
|
|
||||||
|
use yii\base\Behavior;
|
||||||
|
use yii\base\InvalidConfigException;
|
||||||
|
use yii\db\ActiveRecord;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @property ActiveRecord $owner
|
||||||
|
*/
|
||||||
|
class PrimaryKeyValueBehavior extends Behavior {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @var callable Функция, что будет вызвана для генерации ключа.
|
||||||
|
* Должна возвращать случайное значение, подходящее для логики модели. Функция будет вызываться
|
||||||
|
* в цикле do-while с целью избежания дубликатов строк по первичному ключу, так что если функция
|
||||||
|
* станет возвращать статичное значение, то программа зациклится и что-нибудь здохнет. Не делайте так.
|
||||||
|
*/
|
||||||
|
public $value;
|
||||||
|
|
||||||
|
public function events() {
|
||||||
|
return [
|
||||||
|
ActiveRecord::EVENT_BEFORE_INSERT => 'setPrimaryKeyValue',
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
public function setPrimaryKeyValue() : bool {
|
||||||
|
$owner = $this->owner;
|
||||||
|
if ($owner->getPrimaryKey() === null) {
|
||||||
|
do {
|
||||||
|
$key = $this->generateValue();
|
||||||
|
} while ($this->isValueExists($key));
|
||||||
|
|
||||||
|
$owner->{$this->getPrimaryKeyName()} = $key;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function generateValue() : string {
|
||||||
|
return (string)call_user_func($this->value);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function isValueExists(string $key) : bool {
|
||||||
|
return $this->owner->find()->andWhere([$this->getPrimaryKeyName() => $key])->exists();
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function getPrimaryKeyName() : string {
|
||||||
|
$owner = $this->owner;
|
||||||
|
$primaryKeys = $owner->primaryKey();
|
||||||
|
if (!isset($primaryKeys[0])) {
|
||||||
|
throw new InvalidConfigException('"' . get_class($owner) . '" must have a primary key.');
|
||||||
|
} elseif (count($primaryKeys) > 1) {
|
||||||
|
throw new InvalidConfigException('Current behavior don\'t support models with more then one primary key.');
|
||||||
|
}
|
||||||
|
|
||||||
|
/** @noinspection PhpIncompatibleReturnTypeInspection да как бы оно всё нормально, но шторм мне не верит */
|
||||||
|
return $primaryKeys[0];
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
@ -3,6 +3,7 @@ namespace common\models;
|
|||||||
|
|
||||||
use common\behaviors\DataBehavior;
|
use common\behaviors\DataBehavior;
|
||||||
use common\behaviors\EmailActivationExpirationBehavior;
|
use common\behaviors\EmailActivationExpirationBehavior;
|
||||||
|
use common\behaviors\PrimaryKeyValueBehavior;
|
||||||
use common\components\UserFriendlyRandomKey;
|
use common\components\UserFriendlyRandomKey;
|
||||||
use yii\base\InvalidConfigException;
|
use yii\base\InvalidConfigException;
|
||||||
use yii\behaviors\TimestampBehavior;
|
use yii\behaviors\TimestampBehavior;
|
||||||
@ -42,6 +43,12 @@ class EmailActivation extends ActiveRecord {
|
|||||||
'class' => TimestampBehavior::class,
|
'class' => TimestampBehavior::class,
|
||||||
'updatedAtAttribute' => false,
|
'updatedAtAttribute' => false,
|
||||||
],
|
],
|
||||||
|
[
|
||||||
|
'class' => PrimaryKeyValueBehavior::class,
|
||||||
|
'value' => function() {
|
||||||
|
return UserFriendlyRandomKey::make();
|
||||||
|
},
|
||||||
|
],
|
||||||
'expirationBehavior' => [
|
'expirationBehavior' => [
|
||||||
'class' => EmailActivationExpirationBehavior::class,
|
'class' => EmailActivationExpirationBehavior::class,
|
||||||
'repeatTimeout' => 5 * 60, // 5m
|
'repeatTimeout' => 5 * 60, // 5m
|
||||||
@ -84,26 +91,4 @@ class EmailActivation extends ActiveRecord {
|
|||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
public function beforeSave($insert) {
|
|
||||||
if (!parent::beforeSave($insert)) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
if ($this->key === null) {
|
|
||||||
do {
|
|
||||||
$this->key = $this->generateKey();
|
|
||||||
} while ($this->isKeyExists($this->key));
|
|
||||||
}
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
protected function generateKey() : string {
|
|
||||||
return UserFriendlyRandomKey::make();
|
|
||||||
}
|
|
||||||
|
|
||||||
protected function isKeyExists(string $key) : bool {
|
|
||||||
return self::find()->andWhere(['key' => $key])->exists();
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -0,0 +1,70 @@
|
|||||||
|
<?php
|
||||||
|
namespace codeception\common\unit\behaviors;
|
||||||
|
|
||||||
|
use Codeception\Specify;
|
||||||
|
use common\behaviors\PrimaryKeyValueBehavior;
|
||||||
|
use tests\codeception\api\unit\TestCase;
|
||||||
|
use yii\db\ActiveRecord;
|
||||||
|
|
||||||
|
class PrimaryKeyValueBehaviorTest extends TestCase {
|
||||||
|
use Specify;
|
||||||
|
|
||||||
|
public function testSetPrimaryKeyValue() {
|
||||||
|
$this->specify('method should generate value for primary key field on call', function() {
|
||||||
|
$model = new DummyModel();
|
||||||
|
/** @var PrimaryKeyValueBehavior|\PHPUnit_Framework_MockObject_MockObject $behavior */
|
||||||
|
$behavior = $this->getMockBuilder(PrimaryKeyValueBehavior::class)
|
||||||
|
->setMethods(['isValueExists'])
|
||||||
|
->setConstructorArgs([[
|
||||||
|
'value' => function() {
|
||||||
|
return 'mock';
|
||||||
|
},
|
||||||
|
]])
|
||||||
|
->getMock();
|
||||||
|
|
||||||
|
$behavior->expects($this->once())
|
||||||
|
->method('isValueExists')
|
||||||
|
->will($this->returnValue(false));
|
||||||
|
|
||||||
|
$model->attachBehavior('primary-key-value-behavior', $behavior);
|
||||||
|
$behavior->setPrimaryKeyValue();
|
||||||
|
expect($model->id)->equals('mock');
|
||||||
|
});
|
||||||
|
|
||||||
|
$this->specify('method should repeat value generation if generated value duplicate with exists', function() {
|
||||||
|
$model = new DummyModel();
|
||||||
|
/** @var PrimaryKeyValueBehavior|\PHPUnit_Framework_MockObject_MockObject $behavior */
|
||||||
|
$behavior = $this->getMockBuilder(PrimaryKeyValueBehavior::class)
|
||||||
|
->setMethods(['isValueExists', 'generateValue'])
|
||||||
|
->setConstructorArgs([[
|
||||||
|
'value' => function() {
|
||||||
|
return 'this was mocked, but let be passed';
|
||||||
|
},
|
||||||
|
]])
|
||||||
|
->getMock();
|
||||||
|
|
||||||
|
$behavior->expects($this->exactly(3))
|
||||||
|
->method('generateValue')
|
||||||
|
->will($this->onConsecutiveCalls('1', '2', '3'));
|
||||||
|
|
||||||
|
$behavior->expects($this->exactly(3))
|
||||||
|
->method('isValueExists')
|
||||||
|
->will($this->onConsecutiveCalls(true, true, false));
|
||||||
|
|
||||||
|
$model->attachBehavior('primary-key-value-behavior', $behavior);
|
||||||
|
$behavior->setPrimaryKeyValue();
|
||||||
|
expect($model->id)->equals('3');
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
class DummyModel extends ActiveRecord {
|
||||||
|
|
||||||
|
public $id;
|
||||||
|
|
||||||
|
public static function primaryKey() {
|
||||||
|
return ['id'];
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
@ -28,30 +28,4 @@ class EmailActivationTest extends DbTestCase {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
public function testBeforeSave() {
|
|
||||||
$this->specify('method should generate value for key field if it empty', function() {
|
|
||||||
$model = new EmailActivation();
|
|
||||||
$model->beforeSave(true);
|
|
||||||
expect($model->key)->notNull();
|
|
||||||
});
|
|
||||||
|
|
||||||
$this->specify('method should repeat code generation if code duplicate with exists', function() {
|
|
||||||
/** @var EmailActivation|\PHPUnit_Framework_MockObject_MockObject $model */
|
|
||||||
$model = $this->getMockBuilder(EmailActivation::class)
|
|
||||||
->setMethods(['generateKey', 'isKeyExists'])
|
|
||||||
->getMock();
|
|
||||||
|
|
||||||
$model->expects($this->exactly(3))
|
|
||||||
->method('generateKey')
|
|
||||||
->will($this->onConsecutiveCalls('1', '2', '3'));
|
|
||||||
|
|
||||||
$model->expects($this->exactly(3))
|
|
||||||
->method('isKeyExists')
|
|
||||||
->will($this->onConsecutiveCalls(true, true, false));
|
|
||||||
|
|
||||||
$model->beforeSave(true);
|
|
||||||
expect($model->key)->equals('3');
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user