getMockBuilder(Application::class) ->onlyMethods(['instance']) ->getMock(); /** @var SchemaBuilder|MockObject $builder */ $builder = $this->getMockBuilder(SchemaBuilder::class) ->disableOriginalConstructor() ->getMock(); /** @var Migrate|MockObject $migration */ $migration = $this->getMockBuilder(Migrate::class) ->setConstructorArgs([$builder, $app]) ->onlyMethods([ 'initMigration', 'getMigrationFiles', 'getMigrated', 'migrate', 'setMigrated', 'lockTable', 'unlockTable', ]) ->getMock(); $this->setExpects($migration, 'initMigration', null, null, $this->atLeastOnce()); $migration->expects($this->atLeastOnce()) ->method('getMigrationFiles') ->willReturn([ 'foo/1234_01_23_123456_init_foo.php', 'foo/9876_03_22_210000_random_hack.php', 'foo/4567_11_01_000000_do_stuff.php', 'foo/9999_99_99_999999_another_foo.php', ]); $this->setExpects($migration, 'getMigrated', null, new Collection([ ['id' => 1, 'migration' => '1234_01_23_123456_init_foo'], ['id' => 2, 'migration' => '4567_11_01_000000_do_stuff'], ]), $this->atLeastOnce()); $migration->expects($this->atLeastOnce()) ->method('migrate') ->withConsecutive( ['foo/9876_03_22_210000_random_hack.php', '9876_03_22_210000_random_hack', Direction::UP], ['foo/9999_99_99_999999_another_foo.php', '9999_99_99_999999_another_foo', Direction::UP], ['foo/9876_03_22_210000_random_hack.php', '9876_03_22_210000_random_hack', Direction::UP], ['foo/9999_99_99_999999_another_foo.php', '9999_99_99_999999_another_foo', Direction::UP], ['foo/9876_03_22_210000_random_hack.php', '9876_03_22_210000_random_hack', Direction::UP], ['foo/4567_11_01_000000_do_stuff.php', '4567_11_01_000000_do_stuff', Direction::DOWN] ); $migration->expects($this->atLeastOnce()) ->method('setMigrated') ->withConsecutive( ['9876_03_22_210000_random_hack', Direction::UP], ['9999_99_99_999999_another_foo', Direction::UP], ['9876_03_22_210000_random_hack', Direction::UP], ['9999_99_99_999999_another_foo', Direction::UP], ['9876_03_22_210000_random_hack', Direction::UP], ['4567_11_01_000000_do_stuff', Direction::DOWN] ); $this->setExpects($migration, 'lockTable', null, null, $this->atLeastOnce()); $this->setExpects($migration, 'unlockTable', null, null, $this->atLeastOnce()); $migration->run('foo', Direction::UP); $messages = []; $migration->setOutput(function ($text) use (&$messages): void { $messages[] = $text; }); $migration->run('foo', Direction::UP); $this->assertCount(4, $messages); foreach ( [ 'init_foo' => 'skipping', 'do_stuff' => 'skipping', 'random_hack' => 'migrating', 'another_foo' => 'migrating', ] as $value => $type ) { $contains = false; foreach ($messages as $message) { if (!Str::contains(mb_strtolower($message), $type) || !Str::contains(mb_strtolower($message), $value)) { continue; } $contains = true; break; } $this->assertTrue($contains, sprintf('Missing message "%s: %s"', $type, $value)); } $messages = []; $migration->run('foo', Direction::UP, true); $this->assertCount(3, $messages); $migration->run('foo', Direction::DOWN, true); } /** * @covers \Engelsystem\Database\Migration\Migrate::run */ public function testRunExceptionUnlockTable(): void { /** @var Application|MockObject $app */ $app = $this->getMockBuilder(Application::class) ->onlyMethods(['instance']) ->getMock(); /** @var SchemaBuilder|MockObject $builder */ $builder = $this->getMockBuilder(SchemaBuilder::class) ->disableOriginalConstructor() ->getMock(); /** @var Migrate|MockObject $migration */ $migration = $this->getMockBuilder(Migrate::class) ->setConstructorArgs([$builder, $app]) ->onlyMethods([ 'initMigration', 'lockTable', 'getMigrations', 'getMigrated', 'migrate', 'unlockTable', ]) ->getMock(); $this->setExpects($migration, 'initMigration'); $this->setExpects($migration, 'lockTable'); $this->setExpects($migration, 'unlockTable'); $this->setExpects($migration, 'getMigrations', null, collect([ ['migration' => '1234_01_23_123456_init_foo', 'path' => '/foo'], ])); $this->setExpects($migration, 'getMigrated', null, collect()); $migration->expects($this->once()) ->method('migrate') ->willReturnCallback(function (): void { throw new Exception(); }); $this->expectException(Exception::class); $migration->run(''); } /** * @covers \Engelsystem\Database\Migration\Migrate::getMigrated * @covers \Engelsystem\Database\Migration\Migrate::getMigrationFiles * @covers \Engelsystem\Database\Migration\Migrate::getTableQuery * @covers \Engelsystem\Database\Migration\Migrate::initMigration * @covers \Engelsystem\Database\Migration\Migrate::migrate * @covers \Engelsystem\Database\Migration\Migrate::setMigrated * @covers \Engelsystem\Database\Migration\Migrate::lockTable * @covers \Engelsystem\Database\Migration\Migrate::unlockTable */ public function testRunIntegration(): void { $app = new Application(); $dbManager = new CapsuleManager($app); $dbManager->addConnection(['driver' => 'sqlite', 'database' => ':memory:']); $dbManager->bootEloquent(); $db = $dbManager->getConnection(); $db->useDefaultSchemaGrammar(); $schema = $db->getSchemaBuilder(); $app->instance('schema', $schema); $app->bind(SchemaBuilder::class, 'schema'); $migration = new Migrate($schema, $app); $messages = []; $migration->setOutput(function ($msg) use (&$messages): void { $messages[] = $msg; }); $migration->run(__DIR__ . '/Stub', Direction::UP); $this->assertTrue($schema->hasTable('migrations')); $migrations = $db->table('migrations')->get(); $this->assertCount(3, $migrations); $this->assertFalse($migrations->contains('migration', 'lock')); $this->assertTrue($migrations->contains('migration', '2001_04_11_123456_create_lorem_ipsum_table')); $this->assertTrue($migrations->contains('migration', '2017_12_24_053300_another_stuff')); $this->assertTrue($migrations->contains('migration', '2022_12_22_221222_add_some_feature')); $this->assertTrue($schema->hasTable('lorem_ipsum')); $migration->run(__DIR__ . '/Stub', Direction::DOWN, true); $migrations = $db->table('migrations')->get(); $this->assertCount(2, $migrations); $migration->run(__DIR__ . '/Stub', Direction::DOWN); $migrations = $db->table('migrations')->get(); $this->assertCount(0, $migrations); $this->assertFalse($schema->hasTable('lorem_ipsum')); $db->table('migrations')->insert(['migration' => 'lock']); $this->expectException(Exception::class); $migration->run(__DIR__ . '/Stub', Direction::UP); } }