resolver.php 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175
  1. <?php namespace Laravel\CLI\Tasks\Migrate;
  2. use Laravel\Bundle;
  3. class Resolver {
  4. /**
  5. * The migration database instance.
  6. *
  7. * @var Database
  8. */
  9. protected $database;
  10. /**
  11. * Create a new instance of the migration resolver.
  12. *
  13. * @param Database $database
  14. * @return void
  15. */
  16. public function __construct(Database $database)
  17. {
  18. $this->database = $database;
  19. }
  20. /**
  21. * Resolve all of the outstanding migrations for a bundle.
  22. *
  23. * @param string $bundle
  24. * @return array
  25. */
  26. public function outstanding($bundle = null)
  27. {
  28. $migrations = array();
  29. // If no bundle was given to the command, we'll grab every bundle for
  30. // the application, including the "application" bundle, which is not
  31. // returned by "all" method on the Bundle class.
  32. if (is_null($bundle))
  33. {
  34. $bundles = array_merge(Bundle::names(), array('application'));
  35. }
  36. else
  37. {
  38. $bundles = array($bundle);
  39. }
  40. foreach ($bundles as $bundle)
  41. {
  42. // First we need to grab all of the migrations that have already
  43. // run for this bundle, as well as all of the migration files
  44. // for the bundle. Once we have these, we can determine which
  45. // migrations are still outstanding.
  46. $ran = $this->database->ran($bundle);
  47. $files = $this->migrations($bundle);
  48. // To find outstanding migrations, we will simply iterate over
  49. // the migration files and add the files that do not exist in
  50. // the array of ran migrations to the outstanding array.
  51. foreach ($files as $key => $name)
  52. {
  53. if ( ! in_array($name, $ran))
  54. {
  55. $migrations[] = compact('bundle', 'name');
  56. }
  57. }
  58. }
  59. return $this->resolve($migrations);
  60. }
  61. /**
  62. * Resolve an array of the last batch of migrations.
  63. *
  64. * @return array
  65. */
  66. public function last()
  67. {
  68. return $this->resolve($this->database->last());
  69. }
  70. /**
  71. * Resolve an array of migration instances.
  72. *
  73. * @param array $migrations
  74. * @return array
  75. */
  76. protected function resolve($migrations)
  77. {
  78. $instances = array();
  79. foreach ($migrations as $migration)
  80. {
  81. $migration = (array) $migration;
  82. // The migration array contains the bundle name, so we will get the
  83. // path to the bundle's migrations and resolve an instance of the
  84. // migration using the name.
  85. $bundle = $migration['bundle'];
  86. $path = Bundle::path($bundle).'migrations/';
  87. // Migrations are not resolved through the auto-loader, so we will
  88. // manually instantiate the migration class instances for each of
  89. // the migration names we're given.
  90. $name = $migration['name'];
  91. require_once $path.$name.EXT;
  92. // Since the migration name will begin with the numeric ID, we'll
  93. // slice off the ID so we are left with the migration class name.
  94. // The IDs are for sorting when resolving outstanding migrations.
  95. //
  96. // Migrations that exist within bundles other than the default
  97. // will be prefixed with the bundle name to avoid any possible
  98. // naming collisions with other bundle's migrations.
  99. $prefix = Bundle::class_prefix($bundle);
  100. $class = $prefix.\Laravel\Str::classify(substr($name, 18));
  101. $migration = new $class;
  102. // When adding to the array of instances, we will actually
  103. // add the migration instance, the bundle, and the name.
  104. // This allows the migrator to log the bundle and name
  105. // when the migration is executed.
  106. $instances[] = compact('bundle', 'name', 'migration');
  107. }
  108. // At this point the migrations are only sorted within their
  109. // bundles so we need to resort them by name to ensure they
  110. // are in a consistent order.
  111. usort($instances, function($a, $b)
  112. {
  113. return strcmp($a['name'], $b['name']);
  114. });
  115. return $instances;
  116. }
  117. /**
  118. * Grab all of the migration filenames for a bundle.
  119. *
  120. * @param string $bundle
  121. * @return array
  122. */
  123. protected function migrations($bundle)
  124. {
  125. $files = glob(Bundle::path($bundle).'migrations/*_*'.EXT);
  126. // When open_basedir is enabled, glob will return false on an
  127. // empty directory, so we will return an empty array in this
  128. // case so the application doesn't bomb out.
  129. if ($files === false)
  130. {
  131. return array();
  132. }
  133. // Once we have the array of files in the migration directory,
  134. // we'll take the basename of the file and remove the PHP file
  135. // extension, which isn't needed.
  136. foreach ($files as &$file)
  137. {
  138. $file = str_replace(EXT, '', basename($file));
  139. }
  140. // We'll also sort the files so that the earlier migrations
  141. // will be at the front of the array and will be resolved
  142. // first by this class' resolve method.
  143. sort($files);
  144. return $files;
  145. }
  146. }