MigratorTest.php 9.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321
  1. <?php
  2. use PHPUnit\Framework\TestCase;
  3. class MigratorTest extends TestCase
  4. {
  5. public function testAddMigration(): void {
  6. $migrator = new Minz_Migrator();
  7. $migrator->addMigration('foo', function () {
  8. return true;
  9. });
  10. $migrations = $migrator->migrations();
  11. self::assertArrayHasKey('foo', $migrations);
  12. $result = $migrations['foo']();
  13. self::assertTrue($result);
  14. }
  15. public function testMigrationsIsSorted(): void {
  16. $migrator = new Minz_Migrator();
  17. $migrator->addMigration('2_foo', function () {
  18. return true;
  19. });
  20. $migrator->addMigration('10_foo', function () {
  21. return true;
  22. });
  23. $migrator->addMigration('1_foo', function () {
  24. return true;
  25. });
  26. $expected_versions = ['1_foo', '2_foo', '10_foo'];
  27. $migrations = $migrator->migrations();
  28. self::assertSame($expected_versions, array_keys($migrations));
  29. }
  30. public function testSetAppliedVersions(): void {
  31. $migrator = new Minz_Migrator();
  32. $migrator->addMigration('foo', function () {
  33. return true;
  34. });
  35. $migrator->setAppliedVersions(['foo']);
  36. self::assertSame(['foo'], $migrator->appliedVersions());
  37. }
  38. public function testSetAppliedVersionsTrimArgument(): void {
  39. $migrator = new Minz_Migrator();
  40. $migrator->addMigration('foo', function () {
  41. return true;
  42. });
  43. $migrator->setAppliedVersions(["foo\n"]);
  44. self::assertSame(['foo'], $migrator->appliedVersions());
  45. }
  46. public function testSetAppliedVersionsFailsIfMigrationDoesNotExist(): void {
  47. $this->expectException(DomainException::class);
  48. $this->expectExceptionMessage('foo migration does not exist.');
  49. $migrator = new Minz_Migrator();
  50. $migrator->setAppliedVersions(['foo']);
  51. }
  52. public function testVersions(): void {
  53. $migrator = new Minz_Migrator();
  54. $migrator->addMigration('foo', function () {
  55. return true;
  56. });
  57. $migrator->addMigration('bar', function () {
  58. return true;
  59. });
  60. $versions = $migrator->versions();
  61. self::assertSame(['bar', 'foo'], $versions);
  62. }
  63. public function testMigrate(): void {
  64. $migrator = new Minz_Migrator();
  65. $spy = false;
  66. $migrator->addMigration('foo', function () use (&$spy) {
  67. $spy = true;
  68. return true;
  69. });
  70. self::assertEmpty($migrator->appliedVersions());
  71. $result = $migrator->migrate();
  72. self::assertTrue($spy);
  73. self::assertSame(['foo'], $migrator->appliedVersions());
  74. self::assertSame([
  75. 'foo' => true,
  76. ], $result);
  77. }
  78. public function testMigrateCallsMigrationsInSortedOrder(): void {
  79. $migrator = new Minz_Migrator();
  80. $spy_foo_1_is_called = false;
  81. $migrator->addMigration('2_foo', function () use (&$spy_foo_1_is_called) {
  82. return $spy_foo_1_is_called;
  83. });
  84. $migrator->addMigration('1_foo', function () use (&$spy_foo_1_is_called) {
  85. $spy_foo_1_is_called = true;
  86. return true;
  87. });
  88. $result = $migrator->migrate();
  89. self::assertSame(['1_foo', '2_foo'], $migrator->appliedVersions());
  90. self::assertSame([
  91. '1_foo' => true,
  92. '2_foo' => true,
  93. ], $result);
  94. }
  95. public function testMigrateDoesNotCallAppliedMigrations(): void {
  96. $migrator = new Minz_Migrator();
  97. $spy = false;
  98. $migrator->addMigration('1_foo', function () use (&$spy) {
  99. $spy = true;
  100. return true;
  101. });
  102. $migrator->setAppliedVersions(['1_foo']);
  103. $result = $migrator->migrate();
  104. self::assertFalse($spy);
  105. self::assertSame([], $result);
  106. }
  107. public function testMigrateCallNonAppliedBetweenTwoApplied(): void {
  108. $migrator = new Minz_Migrator();
  109. $migrator->addMigration('1_foo', function () {
  110. return true;
  111. });
  112. $migrator->addMigration('2_foo', function () {
  113. return true;
  114. });
  115. $migrator->addMigration('3_foo', function () {
  116. return true;
  117. });
  118. $migrator->setAppliedVersions(['1_foo', '3_foo']);
  119. $result = $migrator->migrate();
  120. self::assertSame(['1_foo', '2_foo', '3_foo'], $migrator->appliedVersions());
  121. self::assertSame([
  122. '2_foo' => true,
  123. ], $result);
  124. }
  125. public function testMigrateWithMigrationReturningFalseDoesNotApplyVersion(): void {
  126. $migrator = new Minz_Migrator();
  127. $migrator->addMigration('1_foo', function () {
  128. return true;
  129. });
  130. $migrator->addMigration('2_foo', function () {
  131. return false;
  132. });
  133. $result = $migrator->migrate();
  134. self::assertSame(['1_foo'], $migrator->appliedVersions());
  135. self::assertSame([
  136. '1_foo' => true,
  137. '2_foo' => false,
  138. ], $result);
  139. }
  140. public function testMigrateWithMigrationReturningFalseDoesNotExecuteNextMigrations(): void {
  141. $migrator = new Minz_Migrator();
  142. $migrator->addMigration('1_foo', function () {
  143. return false;
  144. });
  145. $spy = false;
  146. $migrator->addMigration('2_foo', function () use (&$spy) {
  147. $spy = true;
  148. return true;
  149. });
  150. $result = $migrator->migrate();
  151. self::assertEmpty($migrator->appliedVersions());
  152. self::assertFalse($spy);
  153. self::assertSame([
  154. '1_foo' => false,
  155. ], $result);
  156. }
  157. public function testMigrateWithFailingMigration(): void {
  158. $migrator = new Minz_Migrator();
  159. $migrator->addMigration('foo', function () {
  160. throw new \Exception('Oops, it failed.');
  161. });
  162. $result = $migrator->migrate();
  163. self::assertEmpty($migrator->appliedVersions());
  164. self::assertSame([
  165. 'foo' => 'Oops, it failed.',
  166. ], $result);
  167. }
  168. public function testUpToDate(): void {
  169. $migrator = new Minz_Migrator();
  170. $migrator->addMigration('foo', function () {
  171. return true;
  172. });
  173. $migrator->setAppliedVersions(['foo']);
  174. $upToDate = $migrator->upToDate();
  175. self::assertTrue($upToDate);
  176. }
  177. public function testUpToDateIfRemainingMigration(): void {
  178. $migrator = new Minz_Migrator();
  179. $migrator->addMigration('1_foo', function () {
  180. return true;
  181. });
  182. $migrator->addMigration('2_foo', function () {
  183. return true;
  184. });
  185. $migrator->setAppliedVersions(['2_foo']);
  186. $upToDate = $migrator->upToDate();
  187. self::assertFalse($upToDate);
  188. }
  189. public function testUpToDateIfNoMigrations(): void {
  190. $migrator = new Minz_Migrator();
  191. $upToDate = $migrator->upToDate();
  192. self::assertTrue($upToDate);
  193. }
  194. public function testConstructorLoadsDirectory(): void {
  195. $migrations_path = TESTS_PATH . '/fixtures/migrations/';
  196. $migrator = new Minz_Migrator($migrations_path);
  197. $expected_versions = ['2019_12_22_FooBar', '2019_12_23_Baz'];
  198. $migrations = $migrator->migrations();
  199. self::assertSame($expected_versions, array_keys($migrations));
  200. }
  201. public function testExecute(): void {
  202. $migrations_path = TESTS_PATH . '/fixtures/migrations/';
  203. $applied_migrations_path = tempnam('/tmp', 'applied_migrations.txt');
  204. self::assertIsString($applied_migrations_path);
  205. $result = Minz_Migrator::execute($migrations_path, $applied_migrations_path);
  206. self::assertTrue($result);
  207. $versions = file_get_contents($applied_migrations_path);
  208. self::assertSame("2019_12_22_FooBar\n2019_12_23_Baz", $versions);
  209. @unlink($applied_migrations_path);
  210. }
  211. public function testExecuteWithAlreadyAppliedMigration(): void {
  212. $migrations_path = TESTS_PATH . '/fixtures/migrations/';
  213. $applied_migrations_path = tempnam('/tmp', 'applied_migrations.txt');
  214. self::assertIsString($applied_migrations_path);
  215. file_put_contents($applied_migrations_path, '2019_12_22_FooBar');
  216. $result = Minz_Migrator::execute($migrations_path, $applied_migrations_path);
  217. self::assertTrue($result);
  218. $versions = file_get_contents($applied_migrations_path);
  219. self::assertSame("2019_12_22_FooBar\n2019_12_23_Baz", $versions);
  220. @unlink($applied_migrations_path);
  221. }
  222. public function testExecuteWithAppliedMigrationInDifferentOrder(): void {
  223. $migrations_path = TESTS_PATH . '/fixtures/migrations/';
  224. $applied_migrations_path = tempnam('/tmp', 'applied_migrations.txt');
  225. self::assertIsString($applied_migrations_path);
  226. file_put_contents($applied_migrations_path, "2019_12_23_Baz\n2019_12_22_FooBar");
  227. $result = Minz_Migrator::execute($migrations_path, $applied_migrations_path);
  228. self::assertTrue($result);
  229. $versions = file_get_contents($applied_migrations_path);
  230. // if the order changes, it probably means the first versions comparison test doesn’t work anymore
  231. self::assertSame("2019_12_23_Baz\n2019_12_22_FooBar", $versions);
  232. @unlink($applied_migrations_path);
  233. }
  234. public function testExecuteFailsIfVersionPathDoesNotExist(): void {
  235. $migrations_path = TESTS_PATH . '/fixtures/migrations/';
  236. $applied_migrations_path = tempnam('/tmp', 'applied_migrations.txt');
  237. $expected_result = "Cannot open the {$applied_migrations_path} file";
  238. self::assertIsString($applied_migrations_path);
  239. unlink($applied_migrations_path);
  240. $result = Minz_Migrator::execute($migrations_path, $applied_migrations_path);
  241. self::assertSame($expected_result, $result);
  242. @unlink($applied_migrations_path);
  243. }
  244. public function testExecuteFailsIfAMigrationIsFailing(): void {
  245. $migrations_path = TESTS_PATH . '/fixtures/migrations_with_failing/';
  246. $applied_migrations_path = tempnam('/tmp', 'applied_migrations.txt');
  247. $expected_result = 'A migration failed to be applied, please see previous logs.';
  248. self::assertIsString($applied_migrations_path);
  249. $result = Minz_Migrator::execute($migrations_path, $applied_migrations_path);
  250. self::assertIsString($result);
  251. [$result,] = explode("\n", $result, 2);
  252. self::assertSame($expected_result, $result);
  253. $versions = file_get_contents($applied_migrations_path);
  254. self::assertSame('2020_01_11_FooBar', $versions);
  255. @unlink($applied_migrations_path);
  256. }
  257. }