migrateControllerClass = EchoMigrateController::className(); $this->migrationBaseClass = Migration::className(); $this->mockApplication([ 'components' => [ 'db' => [ 'class' => 'yii\db\Connection', 'dsn' => 'sqlite::memory:', ], ], ]); $this->setUpMigrationPath(); parent::setUp(); } public function tearDown() { $this->tearDownMigrationPath(); parent::tearDown(); } /** * @return array applied migration entries */ protected function getMigrationHistory() { $query = new Query(); return $query->from('migration')->all(); } public function assertFileContent($expectedFile, $class, $table, $namespace = null) { if ($namespace) { $namespace = "namespace {$namespace};\n\n"; } $expected = include Yii::getAlias("@yiiunit/data/console/migrate_create/$expectedFile.php"); $expected = str_replace('{table}', $table, $expected); $this->assertEqualsWithoutLE($expected, $this->parseNameClassMigration($class)); } protected function assertCommandCreatedFile($expectedFile, $migrationName, $table, $params = []) { $params[0] = $migrationName; list($config, $namespace, $class) = $this->prepareMigrationNameData($migrationName); $this->runMigrateControllerAction('create', $params, $config); $this->assertFileContent($expectedFile, $class, $table, $namespace); } public function assertFileContentJunction($expectedFile, $class, $junctionTable, $firstTable, $secondTable, $namespace = null) { if ($namespace) { $namespace = "namespace {$namespace};\n\n"; } $expected = include Yii::getAlias("@yiiunit/data/console/migrate_create/$expectedFile.php"); $expected = str_replace( ['{junctionTable}', '{firstTable}', '{secondTable}'], [$junctionTable, $firstTable, $secondTable], $expected ); $this->assertEqualsWithoutLE($expected, $this->parseNameClassMigration($class)); } protected function assertCommandCreatedJunctionFile($expectedFile, $migrationName, $junctionTable, $firstTable, $secondTable) { list($config, $namespace, $class) = $this->prepareMigrationNameData($migrationName); $this->runMigrateControllerAction('create', [$migrationName], $config); $this->assertFileContentJunction($expectedFile, $class, $junctionTable, $firstTable, $secondTable, $namespace); } protected function prepareMigrationNameData($migrationName) { $config = []; $namespace = null; $lastSlashPosition = strrpos($migrationName, '\\'); if ($lastSlashPosition !== false) { $config = [ 'migrationPath' => null, 'migrationNamespaces' => [$this->migrationNamespace], ]; $class = 'M' . gmdate('ymdHis') . Inflector::camelize(substr($migrationName, $lastSlashPosition + 1)); $namespace = substr($migrationName, 0, $lastSlashPosition); } else { $class = 'm' . gmdate('ymd_His') . '_' . $migrationName; } return [$config, $namespace, $class]; } /** * @return array */ public function generateMigrationDataProvider() { $params = [ 'create_fields' => [ 'fields' => 'title:string(10):notNull:unique:defaultValue("test"), body:text:notNull, price:money(11,2):notNull, parenthesis_in_comment:string(255):notNull:comment(\'Name of set (RU)\')', ], 'create_title_pk' => [ 'fields' => 'title:primaryKey,body:text:notNull,price:money(11,2)', ], 'create_unsigned_pk' => [ 'fields' => 'brand_id:primaryKey:unsigned', ], 'create_unsigned_big_pk' => [ 'fields' => 'brand_id:bigPrimaryKey:unsigned', ], 'create_id_pk' => [ 'fields' => 'id:primaryKey, address:string, address2:string, email:string', ], 'create_foreign_key' => [ 'fields' => 'user_id:integer:foreignKey, product_id:foreignKey:integer:unsigned:notNull, order_id:integer:foreignKey(user_order):notNull, created_at:dateTime:notNull', ], 'create_prefix' => [ 'useTablePrefix' => true, 'fields' => 'user_id:integer:foreignKey, product_id:foreignKey:integer:unsigned:notNull, order_id:integer:foreignKey(user_order):notNull, created_at:dateTime:notNull', ], 'create_title_with_comma_default_values' => [ 'fields' => 'title:string(10):notNull:unique:defaultValue(",te,st"), body:text:notNull:defaultValue(",test"), test:custom(11,2,"s"):notNull', ], 'create_field_with_colon_default_values' => [ 'fields' => 'field_1:dateTime:notNull:defaultValue(\'0000-00-00 00:00:00\'), field_2:string:defaultValue(\'default:value\')', ], 'drop_fields' => [ 'fields' => 'body:text:notNull,price:money(11,2)', ], 'add_columns_test' => [ 'fields' => 'title:string(10):notNull, body:text:notNull, price:money(11,2):notNull, created_at:dateTime', ], 'add_columns_fk' => [ 'fields' => 'user_id:integer:foreignKey, product_id:foreignKey:integer:unsigned:notNull, order_id:integer:foreignKey(user_order):notNull, created_at:dateTime:notNull', ], 'add_columns_prefix' => [ 'useTablePrefix' => true, 'fields' => 'user_id:integer:foreignKey, product_id:foreignKey:integer:unsigned:notNull, order_id:integer:foreignKey(user_order):notNull, created_at:dateTime:notNull', ], 'add_two_columns_test' => [ 'fields' => 'field_1:string(10):notNull, field_2:text:notNull', ], 'drop_columns_test' => [ 'fields' => 'title:string(10):notNull,body:text:notNull, price:money(11,2):notNull, created_at:dateTime', ], ]; return [ ['default', 'DefaultTest', 'default', []], // underscore + table name = case keeped ['create_test', 'create_test_table', 'test', []], ['create_test', 'create_test__table', 'test_', []], ['create_test', 'create_TEST_table', 'TEST', []], ['create_test', 'Create_tEsTTable', 'tEsT', []], // no underscore + table name = camelcase converted to underscore ['create_test', 'CreateTestTable', 'test', []], ['create_test', 'createTest_table', 'test', []], ['create_test', 'createTe_st_table', 'te_st', []], ['create_test', 'createTest__table', 'test_', []], ['create_test', 'createTESTtable', 't_e_s_t', []], ['create_fields', 'create_test_table', 'test', $params['create_fields']], ['create_fields', 'create_TEST_table', 'TEST', $params['create_fields']], ['create_title_pk', 'create_test_table', 'test', $params['create_title_pk']], ['create_title_pk', 'create_TEST_table', 'TEST', $params['create_title_pk']], ['create_unsigned_pk', 'create_test_table', 'test', $params['create_unsigned_pk']], ['create_unsigned_pk', 'create_TEST_table', 'TEST', $params['create_unsigned_pk']], ['create_unsigned_big_pk', 'create_test_table', 'test', $params['create_unsigned_big_pk']], ['create_unsigned_big_pk', 'create_TEST_table', 'TEST', $params['create_unsigned_big_pk']], ['create_id_pk', 'create_test_table', 'test', $params['create_id_pk']], ['create_id_pk', 'create_TEST_table', 'TEST', $params['create_id_pk']], ['create_foreign_key', 'create_test_table', 'test', $params['create_foreign_key']], ['create_foreign_key', 'create_TEST_table', 'TEST', $params['create_foreign_key']], ['create_prefix', 'create_test_table', 'test', $params['create_prefix']], ['create_prefix', 'create_TEST_table', 'TEST', $params['create_prefix']], // @see https://github.com/yiisoft/yii2/issues/11461 ['create_title_with_comma_default_values', 'create_test_table', 'test', $params['create_title_with_comma_default_values']], ['create_field_with_colon_default_values', 'create_test_table', 'test', $params['create_field_with_colon_default_values']], ['drop_test', 'drop_test_table', 'test', []], ['drop_test', 'drop_test__table', 'test_', []], ['drop_test', 'drop_TEST_table', 'TEST', []], ['drop_test', 'Drop_tEStTable', 'tESt', []], ['drop_test', 'DropTestTable', 'test', []], ['drop_test', 'DropTest_Table', 'test', []], ['drop_test', 'DropTest__Table', 'test_', []], ['drop_test', 'DropTESTtable', 't_e_s_t', []], ['drop_fields', 'drop_test_table', 'test', $params['drop_fields']], ['drop_fields', 'drop_TEST_table', 'TEST', $params['drop_fields']], // @see https://github.com/yiisoft/yii2/issues/10876 ['drop_products_from_store_table', 'drop_products_from_store_table', 'products_from_store', []], ['drop_products_from_store_table', 'drop_products_FROM_store_table', 'products_FROM_store', []], ['add_columns_test', 'add_columns_column_to_test_table', 'test', $params['add_columns_test']], ['add_columns_test', 'add_columns_column_to_test__table', 'test_', $params['add_columns_test']], ['add_columns_test', 'add_columns_column_to_TEST_table', 'TEST', $params['add_columns_test']], ['add_columns_test', 'AddColumns_column_to_teSTtable', 'teST', $params['add_columns_test']], ['add_columns_test', 'AddColumnsColumnTo_tEstTable', 'tEst', $params['add_columns_test']], ['add_columns_test', 'addColumnsColumnToTestTable', 'test', $params['add_columns_test']], ['add_columns_test', 'AddColumnsColumnToTest_Table', 'test', $params['add_columns_test']], ['add_columns_test', 'AddCol__umnsColumnToTest__Table', 'test_', $params['add_columns_test']], ['add_columns_test', 'AddColumnsColumnToTESTTable', 't_e_s_t', $params['add_columns_test']], ['add_columns_fk', 'add_columns_column_to_test_table', 'test', $params['add_columns_fk']], ['add_columns_fk', 'add_columns_column_to_TEST_table', 'TEST', $params['add_columns_fk']], ['add_columns_prefix', 'add_columns_column_to_test_table', 'test', $params['add_columns_prefix']], ['add_columns_prefix', 'add_columns_column_to_TEST_table', 'TEST', $params['add_columns_prefix']], ['add_two_columns_test', 'add_field_1_column_field_2_column_to_test_table', 'test', $params['add_two_columns_test']], ['add_two_columns_test', 'add_field_1_column_field_2_column_to_TEST_table', 'TEST', $params['add_two_columns_test']], ['drop_columns_test', 'drop_columns_column_from_test_table', 'test', $params['add_columns_test']], ['drop_columns_test', 'drop_columns_columns_from_test_table', 'test', $params['add_columns_test']], ['drop_columns_test', 'drop_columns_column_from_test__table', 'test_', $params['add_columns_test']], ['drop_columns_test', 'drop_columns_column_from_TEST_table', 'TEST', $params['add_columns_test']], ['drop_columns_test', 'drop_columns_columns_from_TEST_table', 'TEST', $params['add_columns_test']], ['drop_columns_test', 'dropColumnsColumNSFrom_TEstTable', 'TEst', $params['add_columns_test']], ['drop_columns_test', 'DropFewColumnsFrom_Test_Table', 'Test', $params['add_columns_test']], ['drop_columns_test', 'DropFewColumnsFromTestTable', 'test', $params['add_columns_test']], ['drop_columns_test', 'DropFewColumnsFromTest_Table', 'test', $params['add_columns_test']], ['drop_columns_test', 'DropFewColumnsFromTest__Table', 'test_', $params['add_columns_test']], ['drop_columns_test', 'DropFewColumnsFromTeStTable', 'te_st', $params['add_columns_test']], ]; } /** * @param string $expectedFile * @param string $migrationName * @param string $table * @param array $params * @dataProvider generateMigrationDataProvider */ public function testGenerateMigration($expectedFile, $migrationName, $table, $params) { $this->migrationNamespace = 'yiiunit\runtime\test_migrations'; $this->assertCommandCreatedFile($expectedFile, $migrationName, $table, $params); $this->assertCommandCreatedFile( $expectedFile, $this->migrationNamespace . '\\' . $migrationName, $table, $params ); } /** * @return array */ public function generateJunctionMigrationDataProvider() { return [ ['create_junction_post_and_tag_tables', 'post_tag', 'post', 'tag'], ['create_junction_for_post_and_tag_tables', 'post_tag', 'post', 'tag'], ['create_junction_table_for_post_and_tag_tables', 'post_tag', 'post', 'tag'], ['create_junction_table_for_post_and_tag_table', 'post_tag', 'post', 'tag'], ['CreateJunction_postAnd_tagTables', 'post_tag', 'post', 'tag'], ['CreateJunctionFor_postAnd_tagTables', 'post_tag', 'post', 'tag'], ['CreateJunctionTableFor_postAnd_tagTables', 'post_tag', 'post', 'tag'], ['CreateJunctionTableFor_postAnd_tagTable', 'post_tag', 'post', 'tag'], ['CreateJunctionPostAndTagTables', 'post_tag', 'post', 'tag'], ['CreateJunctionPost_AndTag_Tables', 'post_tag', 'post', 'tag'], ['CreateJunctionPost__AndTag__Tables', 'post__tag_', 'post_', 'tag_'], ['CreateJunctionPost__AndTagTables', 'post__tag', 'post_', 'tag'], ['CreateJunctionPostAndTag__Tables', 'post_tag_', 'post', 'tag_'], ['CreateJunctionPostAndTaGTables', 'post_ta_g', 'post', 'ta_g'], ['CreateJunctionPoStAndTagTables', 'po_st_tag', 'po_st', 'tag'], ]; } /** * @param string $migrationName * @param string $junctionTable * @param string $firstTable * @param string $secondTable * @dataProvider generateJunctionMigrationDataProvider */ public function testGenerateJunctionMigration($migrationName, $junctionTable, $firstTable, $secondTable) { $this->migrationNamespace = 'yiiunit\runtime\test_migrations'; $this->assertCommandCreatedJunctionFile( 'junction_test', $migrationName, $junctionTable, $firstTable, $secondTable ); $this->assertCommandCreatedJunctionFile( 'junction_test', $this->migrationNamespace . '\\' . $migrationName, $junctionTable, $firstTable, $secondTable ); } public function testUpdatingLongNamedMigration() { $this->createMigration(str_repeat('a', 180)); $result = $this->runMigrateControllerAction('up'); $this->assertContains('The migration name', $result); $this->assertContains('is too long. Its not possible to apply this migration.', $result); } public function testNamedMigrationWithCustomLimit() { Yii::$app->db->createCommand()->createTable('migration', [ 'version' => 'varchar(255) NOT NULL PRIMARY KEY', // varchar(255) is longer than the default of 180 'apply_time' => 'integer', ])->execute(); $this->createMigration(str_repeat('a', 180)); $result = $this->runMigrateControllerAction('up'); $this->assertContains('1 migration was applied.', $result); $this->assertContains('Migrated up successfully.', $result); } public function testCreateLongNamedMigration() { $this->setOutputCallback(function ($output) { return null; }); $migrationName = str_repeat('a', 180); $this->expectException('yii\console\Exception'); $this->expectExceptionMessage('The migration name is too long.'); $controller = $this->createMigrateController([]); $params[0] = $migrationName; $controller->run('create', $params); } /** * Test the migrate:fresh command. * @dataProvider refreshMigrationDataProvider * @param $db * @throws \yii\db\Exception */ public function testRefreshMigration($db) { if ($db !== 'default') { $this->switchDbConnection($db); } Yii::$app->db->createCommand('create table hall_of_fame(id int, string varchar(255))') ->execute(); Yii::$app->db->createCommand("insert into hall_of_fame values(1, 'Qiang Xue');") ->execute(); Yii::$app->db->createCommand("insert into hall_of_fame values(2, 'Alexander Makarov');") ->execute(); Yii::$app->db->createCommand('create view view_hall_of_fame as select * from hall_of_fame') ->execute(); $result = $this->runMigrateControllerAction('fresh'); // Drop worked $this->assertContains('Table hall_of_fame dropped.', $result); $this->assertContains('View view_hall_of_fame dropped.', $result); // Migration was restarted $this->assertContains('No new migrations found. Your system is up-to-date.', $result); } public function refreshMigrationDataProvider() { return [ ['default'], ['mysql'], ]; } /** * @see https://github.com/yiisoft/yii2/issues/12980 */ public function testGetMigrationHistory() { $controllerConfig = [ 'migrationPath' => null, 'migrationNamespaces' => [$this->migrationNamespace], ]; $this->runMigrateControllerAction('history', [], $controllerConfig); $controller = $this->createMigrateController($controllerConfig); $controller->db = Yii::$app->db; Yii::$app->db->createCommand() ->batchInsert( 'migration', ['version', 'apply_time'], [ ['app\migrations\M140506102106One', 10], ['app\migrations\M160909083544Two', 10], ['app\modules\foo\migrations\M161018124749Three', 10], ['app\migrations\M160930135248Four', 20], ['app\modules\foo\migrations\M161025123028Five', 20], ['app\migrations\M161110133341Six', 20], ] ) ->execute(); $rows = $this->invokeMethod($controller, 'getMigrationHistory', [10]); $this->assertSame( [ 'app\migrations\M161110133341Six', 'app\modules\foo\migrations\M161025123028Five', 'app\migrations\M160930135248Four', 'app\modules\foo\migrations\M161018124749Three', 'app\migrations\M160909083544Two', 'app\migrations\M140506102106One', ], array_keys($rows) ); $rows = $this->invokeMethod($controller, 'getMigrationHistory', [4]); $this->assertSame( [ 'app\migrations\M161110133341Six', 'app\modules\foo\migrations\M161025123028Five', 'app\migrations\M160930135248Four', 'app\modules\foo\migrations\M161018124749Three', ], array_keys($rows) ); } }