Commit 461576dc by Mark Committed by Qiang Xue

added other syntax for and tests for fixture controller

parent d9cd1a85
...@@ -24,10 +24,18 @@ use yii\test\FixtureTrait; ...@@ -24,10 +24,18 @@ use yii\test\FixtureTrait;
* #also a short version of this command (generate action is default) * #also a short version of this command (generate action is default)
* yii fixture User * yii fixture User
* *
* #load all fixtures except User
* yii fixture all -User
*
* #append fixtures to already loaded
* yii fixture User --append
*
* #load fixtures with different namespace. * #load fixtures with different namespace.
* yii fixture/load User --namespace=alias\my\custom\namespace\goes\here * yii fixture/load User --namespace=alias\my\custom\namespace\goes\here
* ~~~ * ~~~
* *
* Same examples are true for unloading fixtures, except append option.
*
* @author Mark Jebri <mark.github@yandex.ru> * @author Mark Jebri <mark.github@yandex.ru>
* @since 2.0 * @since 2.0
*/ */
...@@ -74,16 +82,24 @@ class FixtureController extends Controller ...@@ -74,16 +82,24 @@ class FixtureController extends Controller
/** /**
* Loads given fixture. You can load several fixtures specifying * Loads given fixture. You can load several fixtures specifying
* their names separated with commas, like: User,UserProfile,MyCustom. Be sure there is no * their names separated with space, like: User UserProfile MyCustom.
* whitespace between names. Note that if you are loading fixtures to storage, for example: database or nosql, * Note that if you are loading fixtures to storage, for example: database or nosql,
* storage will not be cleared, data will be appended to already existed. * storage will not be cleared, data will be appended to already existed. If you want to append
* fixtures data to already existed one use "--append" option of this command.
* If you dont want to unload particular fixture, then specify it with "-" prefix.
* @param array $fixtures * @param array $fixtures
* @param array $except * @param array $except
* @throws \yii\console\Exception * @throws \yii\console\Exception
*/ */
public function actionLoad(array $fixtures, array $except = []) public function actionLoad()
{ {
if (!$this->needToApplyAll($fixtures[0])) { $fixturesInput = func_get_args();
$filtered = $this->filterFixtures($fixturesInput);
$except = $filtered['except'];
if (!$this->needToApplyAll($fixturesInput[0])) {
$fixtures = $filtered['apply'];
$foundFixtures = $this->findFixtures($fixtures); $foundFixtures = $this->findFixtures($fixtures);
$notFoundFixtures = array_diff($fixtures, $foundFixtures); $notFoundFixtures = array_diff($fixtures, $foundFixtures);
...@@ -96,19 +112,25 @@ class FixtureController extends Controller ...@@ -96,19 +112,25 @@ class FixtureController extends Controller
$foundFixtures = $this->findFixtures(); $foundFixtures = $this->findFixtures();
} }
$fixturesToLoad = array_diff($foundFixtures, $except);
if (!$foundFixtures) { if (!$foundFixtures) {
throw new Exception( throw new Exception(
"No files were found by name: \"" . implode(', ', $fixtures) . "\".\n" . "No files were found by name: \"" . implode(', ', $fixturesInput) . "\".\n" .
"Check that files with these name exists, under fixtures path: \n\"" . $this->getFixturePath() . "\"." "Check that files with these name exists, under fixtures path: \n\"" . $this->getFixturePath() . "\"."
); );
} }
if (!$this->confirmLoad($foundFixtures, $except)) { if (!$fixturesToLoad) {
return self::EXIT_CODE_NORMAL; $this->notifyNothingToLoad($foundFixtures, $except);
return static::EXIT_CODE_NORMAL;
} }
$filtered = array_diff($foundFixtures, $except); if (!$this->confirmLoad($fixturesToLoad, $except)) {
$fixtures = $this->getFixturesConfig(array_merge($this->globalFixtures, $filtered)); return static::EXIT_CODE_NORMAL;
}
$fixtures = $this->getFixturesConfig(array_merge($this->globalFixtures, $fixturesToLoad));
if (!$fixtures) { if (!$fixtures) {
throw new Exception('No fixtures were found in namespace: "' . $this->namespace . '"' . ''); throw new Exception('No fixtures were found in namespace: "' . $this->namespace . '"' . '');
...@@ -125,39 +147,51 @@ class FixtureController extends Controller ...@@ -125,39 +147,51 @@ class FixtureController extends Controller
} }
/** /**
* Unloads given fixtures. You can clear environment and unload multiple fixtures by specifying * Unloads given fixtures. You can unload several fixtures specifying
* their names separated with commas, like: User,UserProfile,MyCustom. Be sure there is no * their names separated with space, like: User UserProfile MyCustom.
* whitespace between names. * If you dont want to unload particular fixture, then specify it with "-" prefix.
* @param array|string $fixtures
* @param array|string $except
* @throws \yii\console\Exception in case no fixtures are found. * @throws \yii\console\Exception in case no fixtures are found.
*/ */
public function actionUnload(array $fixtures, array $except = []) public function actionUnload()
{ {
if (!$this->needToApplyAll($fixtures[0])) { $fixturesInput = func_get_args();
$filtered = $this->filterFixtures($fixturesInput);
$except = $filtered['except'];
if (!$this->needToApplyAll($fixturesInput[0])) {
$fixtures = $filtered['apply'];
$foundFixtures = $this->findFixtures($fixtures); $foundFixtures = $this->findFixtures($fixtures);
$notFoundFixtures = array_diff($fixtures, $foundFixtures); $notFoundFixtures = array_diff($fixtures, $foundFixtures);
if ($notFoundFixtures) { if ($notFoundFixtures) {
$this->notifyNotFound($notFoundFixtures); $this->notifyNotFound($notFoundFixtures);
} }
} else { } else {
$foundFixtures = $this->findFixtures(); $foundFixtures = $this->findFixtures();
} }
$fixturesToUnload = array_diff($foundFixtures, $except);
if (!$foundFixtures) { if (!$foundFixtures) {
throw new Exception( throw new Exception(
"No files were found by name: \"" . implode(', ', $fixtures) . "\".\n" . "No files were found by name: \"" . implode(', ', $fixturesInput) . "\".\n" .
"Check that fixtures with these name exists, under fixtures path: \n\"" . $this->getFixturePath() . "\"." "Check that files with these name exists, under fixtures path: \n\"" . $this->getFixturePath() . "\"."
); );
} }
if (!$this->confirmUnload($foundFixtures, $except)) { if (!$fixturesToUnload) {
return self::EXIT_CODE_NORMAL; $this->notifyNothingToUnload($foundFixtures, $except);
return static::EXIT_CODE_NORMAL;
}
if (!$this->confirmUnload($fixturesToUnload, $except)) {
return static::EXIT_CODE_NORMAL;
} }
$filtered = array_diff($foundFixtures, $except); $fixtures = $this->getFixturesConfig(array_merge($this->globalFixtures, $fixturesToUnload));
$fixtures = $this->getFixturesConfig(array_merge($this->globalFixtures, $filtered));
if (!$fixtures) { if (!$fixtures) {
throw new Exception('No fixtures were found in namespace: ' . $this->namespace . '".'); throw new Exception('No fixtures were found in namespace: ' . $this->namespace . '".');
...@@ -179,13 +213,53 @@ class FixtureController extends Controller ...@@ -179,13 +213,53 @@ class FixtureController extends Controller
} }
/** /**
* Notifies user that there are no fixtures to load according input conditions
*/
public function notifyNothingToLoad($foundFixtures, $except)
{
$this->stdout("Fixtures to load could not be found according given conditions:\n\n", Console::FG_RED);
$this->stdout("Fixtures namespace is: \n", Console::FG_YELLOW);
$this->stdout("\t" . $this->namespace . "\n", Console::FG_GREEN);
if (count($foundFixtures)) {
$this->stdout("\nFixtures founded under the namespace:\n\n", Console::FG_YELLOW);
$this->outputList($foundFixtures);
}
if (count($except)) {
$this->stdout("\nFixtures that will NOT be loaded: \n\n", Console::FG_YELLOW);
$this->outputList($except);
}
}
/**
* Notifies user that there are no fixtures to unload according input conditions
*/
public function notifyNothingToUnload($foundFixtures, $except)
{
$this->stdout("Fixtures to unload could not be found according given conditions:\n\n", Console::FG_RED);
$this->stdout("Fixtures namespace is: \n", Console::FG_YELLOW);
$this->stdout("\t" . $this->namespace . "\n", Console::FG_GREEN);
if (count($foundFixtures)) {
$this->stdout("\nFixtures founded under the namespace:\n\n", Console::FG_YELLOW);
$this->outputList($foundFixtures);
}
if (count($except)) {
$this->stdout("\nFixtures that will NOT be unloaded: \n\n", Console::FG_YELLOW);
$this->outputList($except);
}
}
/**
* Notifies user that fixtures were successfully unloaded. * Notifies user that fixtures were successfully unloaded.
* @param array $fixtures * @param array $fixtures
*/ */
private function notifyUnloaded($fixtures) private function notifyUnloaded($fixtures)
{ {
$this->stdout("Fixtures were successfully unloaded from namespace:\n", Console::FG_YELLOW); $this->stdout("\nFixtures were successfully unloaded from namespace: ", Console::FG_YELLOW);
$this->stdout("\t\"" . Yii::getAlias($this->namespace) . "\"\n\n", Console::FG_GREEN); $this->stdout(Yii::getAlias($this->namespace) . "\"\n\n", Console::FG_GREEN);
$this->outputList($fixtures); $this->outputList($fixtures);
} }
...@@ -214,12 +288,14 @@ class FixtureController extends Controller ...@@ -214,12 +288,14 @@ class FixtureController extends Controller
$this->stdout("\t" . $this->namespace . "\n\n", Console::FG_GREEN); $this->stdout("\t" . $this->namespace . "\n\n", Console::FG_GREEN);
if (count($this->globalFixtures)) { if (count($this->globalFixtures)) {
$this->stdout("Global fixtures will be loaded:\n\n", Console::FG_YELLOW); $this->stdout("Global fixtures will be used:\n\n", Console::FG_YELLOW);
$this->outputList($this->globalFixtures); $this->outputList($this->globalFixtures);
} }
$this->stdout("\nFixtures below will be loaded:\n\n", Console::FG_YELLOW); if (count($fixtures)) {
$this->outputList($fixtures); $this->stdout("\nFixtures below will be loaded:\n\n", Console::FG_YELLOW);
$this->outputList($fixtures);
}
if (count($except)) { if (count($except)) {
$this->stdout("\nFixtures that will NOT be loaded: \n\n", Console::FG_YELLOW); $this->stdout("\nFixtures that will NOT be loaded: \n\n", Console::FG_YELLOW);
...@@ -241,12 +317,14 @@ class FixtureController extends Controller ...@@ -241,12 +317,14 @@ class FixtureController extends Controller
$this->stdout("\t" . $this->namespace . "\n\n", Console::FG_GREEN); $this->stdout("\t" . $this->namespace . "\n\n", Console::FG_GREEN);
if (count($this->globalFixtures)) { if (count($this->globalFixtures)) {
$this->stdout("Global fixtures will be unloaded:\n\n", Console::FG_YELLOW); $this->stdout("Global fixtures will be used:\n\n", Console::FG_YELLOW);
$this->outputList($this->globalFixtures); $this->outputList($this->globalFixtures);
} }
$this->stdout("\nFixtures below will be unloaded:\n\n", Console::FG_YELLOW); if (count($fixtures)) {
$this->outputList($fixtures); $this->stdout("\nFixtures below will be unloaded:\n\n", Console::FG_YELLOW);
$this->outputList($fixtures);
}
if (count($except)) { if (count($except)) {
$this->stdout("\nFixtures that will NOT be unloaded:\n\n", Console::FG_YELLOW); $this->stdout("\nFixtures that will NOT be unloaded:\n\n", Console::FG_YELLOW);
...@@ -332,6 +410,44 @@ class FixtureController extends Controller ...@@ -332,6 +410,44 @@ class FixtureController extends Controller
} }
/** /**
* Filteres fixtures by splitting them in two categories: one that should be applied and not.
* If fixture is prefixed with "-", for example "-User", that means that fixture should not be loaded,
* if it is not prefixed it is considered as one to be loaded. Returs array:
*
* ~~~
* [
* 'apply' => [
* User,
* ...
* ],
* 'except' => [
* Custom,
* ...
* ],
* ]
* ~~~
* @param array $fixtures
* @return array fixtures array with 'apply' and 'except' elements.
*/
private function filterFixtures($fixtures)
{
$filtered = [
'apply' => [],
'except' => [],
];
foreach ($fixtures as $fixture) {
if (mb_strpos($fixture, '-') !== false) {
$filtered['except'][] = str_replace('-', '', $fixture);
} else {
$filtered['apply'][] = $fixture;
}
}
return $filtered;
}
/**
* Returns fixture path that determined on fixtures namespace. * Returns fixture path that determined on fixtures namespace.
* @return string fixture path * @return string fixture path
*/ */
...@@ -339,4 +455,5 @@ class FixtureController extends Controller ...@@ -339,4 +455,5 @@ class FixtureController extends Controller
{ {
return Yii::getAlias('@' . str_replace('\\', '/', $this->namespace)); return Yii::getAlias('@' . str_replace('\\', '/', $this->namespace));
} }
} }
<?php
namespace yiiunit\data\console\controllers\fixtures;
use yii\test\Fixture;
class FirstFixture extends Fixture
{
public function load()
{
FixtureStorage::$firstFixtureData[] = 'some data set for first fixture';
}
public function unload()
{
FixtureStorage::$firstFixtureData = [];
}
}
<?php
namespace yiiunit\data\console\controllers\fixtures;
class FixtureStorage
{
public static $globalFixturesData = [];
public static $firstFixtureData = [];
public static $secondFixtureData = [];
public static function clear()
{
static::$globalFixturesData = [];
static::$firstFixtureData = [];
static::$secondFixtureData = [];
}
}
<?php
namespace yiiunit\data\console\controllers\fixtures;
use yii\test\Fixture;
class GlobalFixture extends Fixture
{
public function load()
{
FixtureStorage::$globalFixturesData[] = 'some data set for global fixture';
}
public function unload()
{
FixtureStorage::$globalFixturesData = [];
}
}
<?php
namespace yiiunit\data\console\controllers\fixtures;
use yii\test\Fixture;
class SecondFixture extends Fixture
{
public function load()
{
FixtureStorage::$secondFixtureData[] = 'some data set for second fixture';
}
public function unload()
{
FixtureStorage::$secondFixtureData = [];
}
}
<?php
namespace yiiunit\framework\console\controllers;
use Yii;
use yiiunit\TestCase;
use yiiunit\data\console\controllers\fixtures\FixtureStorage;
use yii\console\controllers\FixtureController;
/**
* Unit test for [[\yii\console\controllers\FixtureController]].
* @see MigrateController
*
* @group console
*/
class FixtureControllerTest extends TestCase
{
/**
* @var \yiiunit\framework\console\controllers\FixtureConsoledController
*/
private $_fixtureController;
protected function setUp()
{
parent::setUp();
$this->_fixtureController = Yii::createObject([
'class' => 'yiiunit\framework\console\controllers\FixtureConsoledController',
'interactive' => false,
'globalFixtures' => [],
'namespace' => 'yiiunit\data\console\controllers\fixtures',
],[null, null]); //id and module are null
}
protected function tearDown()
{
$this->_fixtureController = null;
FixtureStorage::clear();
parent::tearDown();
}
public function testLoadGlobalFixture()
{
$this->_fixtureController->globalFixtures = [
'\yiiunit\data\console\controllers\fixtures\Global'
];
$this->_fixtureController->actionLoad('First');
$this->assertCount(1, FixtureStorage::$globalFixturesData, 'global fixture data should be loaded');
$this->assertCount(1, FixtureStorage::$firstFixtureData, 'first fixture data should be loaded');
}
public function testUnloadGlobalFixture()
{
$this->_fixtureController->globalFixtures = [
'\yiiunit\data\console\controllers\fixtures\Global'
];
FixtureStorage::$globalFixturesData[] = 'some seeded global fixture data';
FixtureStorage::$firstFixtureData[] = 'some seeded first fixture data';
$this->assertCount(1, FixtureStorage::$globalFixturesData, 'global fixture data should be loaded');
$this->assertCount(1, FixtureStorage::$firstFixtureData, 'first fixture data should be loaded');
$this->_fixtureController->actionUnload('First');
$this->assertEmpty(FixtureStorage::$globalFixturesData, 'global fixture data should be unloaded');
$this->assertEmpty(FixtureStorage::$firstFixtureData, 'first fixture data should be unloaded');
}
public function testLoadAll()
{
$this->assertEmpty(FixtureStorage::$globalFixturesData, 'global fixture data should be empty');
$this->assertEmpty(FixtureStorage::$firstFixtureData, 'first fixture data should be empty');
$this->assertEmpty(FixtureStorage::$secondFixtureData, 'second fixture data should be empty');
$this->_fixtureController->actionLoad('all');
$this->assertCount(1, FixtureStorage::$globalFixturesData, 'global fixture data should be loaded');
$this->assertCount(1, FixtureStorage::$firstFixtureData, 'first fixture data should be loaded');
$this->assertCount(1, FixtureStorage::$secondFixtureData, 'second fixture data should be loaded');
}
public function testUnloadAll()
{
FixtureStorage::$globalFixturesData[] = 'some seeded global fixture data';
FixtureStorage::$firstFixtureData[] = 'some seeded first fixture data';
FixtureStorage::$secondFixtureData[] = 'some seeded second fixture data';
$this->assertCount(1, FixtureStorage::$globalFixturesData, 'global fixture data should be loaded');
$this->assertCount(1, FixtureStorage::$firstFixtureData, 'first fixture data should be loaded');
$this->assertCount(1, FixtureStorage::$secondFixtureData, 'second fixture data should be loaded');
$this->_fixtureController->actionUnload('all');
$this->assertEmpty(FixtureStorage::$globalFixturesData, 'global fixture data should be unloaded');
$this->assertEmpty(FixtureStorage::$firstFixtureData, 'first fixture data should be unloaded');
$this->assertEmpty(FixtureStorage::$secondFixtureData, 'second fixture data should be unloaded');
}
public function testLoadParticularExceptOnes()
{
$this->_fixtureController->actionLoad('First', '-Second', '-Global');
$this->assertCount(1, FixtureStorage::$firstFixtureData, 'first fixture data should be loaded');
$this->assertEmpty(FixtureStorage::$globalFixturesData, 'global fixture data should not be loaded');
$this->assertEmpty(FixtureStorage::$secondFixtureData, 'second fixture data should not be loaded');
}
public function testUnloadParticularExceptOnes()
{
FixtureStorage::$globalFixturesData[] = 'some seeded global fixture data';
FixtureStorage::$firstFixtureData[] = 'some seeded first fixture data';
FixtureStorage::$secondFixtureData[] = 'some seeded second fixture data';
$this->_fixtureController->actionUnload('First', '-Second', '-Global');
$this->assertEmpty(FixtureStorage::$firstFixtureData, 'first fixture data should be unloaded');
$this->assertNotEmpty(FixtureStorage::$globalFixturesData, 'global fixture data should not be unloaded');
$this->assertNotEmpty(FixtureStorage::$secondFixtureData, 'second fixture data should not be unloaded');
}
public function testLoadAllExceptOnes()
{
$this->_fixtureController->actionLoad('all', '-Second', '-Global');
$this->assertCount(1, FixtureStorage::$firstFixtureData, 'first fixture data should be loaded');
$this->assertEmpty(FixtureStorage::$globalFixturesData, 'global fixture data should not be loaded');
$this->assertEmpty(FixtureStorage::$secondFixtureData, 'second fixture data should not be loaded');
}
public function testUnloadAllExceptOnes()
{
FixtureStorage::$globalFixturesData[] = 'some seeded global fixture data';
FixtureStorage::$firstFixtureData[] = 'some seeded first fixture data';
FixtureStorage::$secondFixtureData[] = 'some seeded second fixture data';
$this->_fixtureController->actionUnload('all', '-Second', '-Global');
$this->assertEmpty(FixtureStorage::$firstFixtureData, 'first fixture data should be unloaded');
$this->assertNotEmpty(FixtureStorage::$globalFixturesData, 'global fixture data should not be unloaded');
$this->assertNotEmpty(FixtureStorage::$secondFixtureData, 'second fixture data should not be unloaded');
}
public function testNothingToLoadParticularExceptOnes()
{
$this->_fixtureController->actionLoad('First', '-First');
$this->assertEmpty(FixtureStorage::$firstFixtureData, 'first fixture data should not be loaded');
}
public function testNothingToUnloadParticularExceptOnes()
{
$this->_fixtureController->actionUnload('First', '-First');
$this->assertEmpty(FixtureStorage::$firstFixtureData, 'first fixture data should not be loaded');
}
/**
* @expectedException \yii\console\Exception
*/
public function testNoFixturesWereFoundInLoad()
{
$this->_fixtureController->actionLoad('NotExistingFixture');
}
/**
* @expectedException \yii\console\Exception
*/
public function testNoFixturesWereFoundInUnload()
{
$this->_fixtureController->actionUnload('NotExistingFixture');
}
}
class FixtureConsoledController extends FixtureController
{
public function stdout($string)
{
}
}
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment