123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245 |
- <?php namespace Laravel\CLI\Tasks\Migrate;
- use Laravel\Str;
- use Laravel\File;
- use Laravel\Bundle;
- use Laravel\CLI\Tasks\Task;
- use Laravel\Database\Schema;
- class Migrator extends Task {
- /**
- * The migration resolver instance.
- *
- * @var Resolver
- */
- protected $resolver;
- /**
- * The migration database instance.
- *
- * @var Database
- */
- protected $database;
- /**
- * Create a new instance of the Migrator CLI task.
- *
- * @param Resolver $resolver
- * @param Database $database
- * @return void
- */
- public function __construct(Resolver $resolver, Database $database)
- {
- $this->resolver = $resolver;
- $this->database = $database;
- }
- /**
- * Run a database migration command.
- *
- * @param array $arguments
- * @return void
- */
- public function run($arguments = array())
- {
- // If no arguments were passed to the task, we will just migrate
- // to the latest version across all bundles. Otherwise, we will
- // parse the arguments to determine the bundle for which the
- // database migrations should be run.
- if (count($arguments) == 0)
- {
- $this->migrate();
- }
- else
- {
- $this->migrate(array_get($arguments, 0));
- }
- }
- /**
- * Run the outstanding migrations for a given bundle.
- *
- * @param string $bundle
- * @param int $version
- * @return void
- */
- public function migrate($bundle = null, $version = null)
- {
- $migrations = $this->resolver->outstanding($bundle);
- if (count($migrations) == 0)
- {
- echo "No outstanding migrations.";
- return;
- }
- // We need to grab the latest batch ID and increment it by one.
- // This allows us to group the migrations so we can easily
- // determine which migrations need to roll back.
- $batch = $this->database->batch() + 1;
- foreach ($migrations as $migration)
- {
- $migration['migration']->up();
- echo 'Migrated: '.$this->display($migration).PHP_EOL;
- // After running a migration, we log its execution in the migration
- // table so that we can easily determine which migrations we'll
- // reverse in the event of a migration rollback.
- $this->database->log($migration['bundle'], $migration['name'], $batch);
- }
- }
- /**
- * Rollback the latest migration command.
- *
- * @param array $arguments
- * @return bool
- */
- public function rollback($arguments = array())
- {
- $migrations = $this->resolver->last();
- if (count($migrations) == 0)
- {
- echo "Nothing to rollback.";
- return false;
- }
- // The "last" method on the resolver returns an array of migrations,
- // along with their bundles and names. We will iterate through each
- // migration and run the "down" method.
- foreach (array_reverse($migrations) as $migration)
- {
- $migration['migration']->down();
- echo 'Rolled back: '.$this->display($migration).PHP_EOL;
- // By only removing the migration after it has successfully rolled back,
- // we can re-run the rollback command in the event of any errors with
- // the migration and pick up where we left off.
- $this->database->delete($migration['bundle'], $migration['name']);
- }
- return true;
- }
- /**
- * Rollback all of the executed migrations.
- *
- * @param array $arguments
- * @return void
- */
- public function reset($arguments = array())
- {
- while ($this->rollback()) {};
- }
- /**
- * Install the database tables used by the migration system.
- *
- * @return void
- */
- public function install()
- {
- Schema::table('laravel_migrations', function($table)
- {
- $table->create();
- // Migrations can be run for a specific bundle, so we'll use
- // the bundle name and string migration name as an unique ID
- // for the migrations, allowing us to easily identify which
- // migrations have been run for each bundle.
- $table->string('bundle', 50);
- $table->string('name', 200);
- // When running a migration command, we will store a batch
- // ID with each of the rows on the table. This will allow
- // us to grab all of the migrations that were run for the
- // last command when performing rollbacks.
- $table->integer('batch');
- $table->primary(array('bundle', 'name'));
- });
- echo "Migration table created successfully.";
- }
- /**
- * Generate a new migration file.
- *
- * @param array $arguments
- * @return string
- */
- public function make($arguments = array())
- {
- if (count($arguments) == 0)
- {
- throw new \Exception("I need to know what to name the migration.");
- }
- list($bundle, $migration) = Bundle::parse($arguments[0]);
- // The migration path is prefixed with the date timestamp, which
- // is a better way of ordering migrations than a simple integer
- // incrementation, since developers may start working on the
- // next migration at the same time unknowingly.
- $prefix = date('Y_m_d_His');
- $path = Bundle::path($bundle).'migrations'.DS;
- // If the migration directory does not exist for the bundle,
- // we will create the directory so there aren't errors when
- // when we try to write the migration file.
- if ( ! is_dir($path)) mkdir($path);
- $file = $path.$prefix.'_'.$migration.EXT;
- File::put($file, $this->stub($bundle, $migration));
- echo "Great! New migration created!";
- // Once the migration has been created, we'll return the
- // migration file name so it can be used by the task
- // consumer if necessary for futher work.
- return $file;
- }
- /**
- * Get the stub migration with the proper class name.
- *
- * @param string $bundle
- * @param string $migration
- * @return string
- */
- protected function stub($bundle, $migration)
- {
- $stub = File::get(path('sys').'cli/tasks/migrate/stub'.EXT);
- $prefix = Bundle::class_prefix($bundle);
- // The class name is formatted simialrly to tasks and controllers,
- // where the bundle name is prefixed to the class if it is not in
- // the default "application" bundle.
- $class = $prefix.Str::classify($migration);
- return str_replace('{{class}}', $class, $stub);
- }
- /**
- * Get the migration bundle and name for display.
- *
- * @param array $migration
- * @return string
- */
- protected function display($migration)
- {
- return $migration['bundle'].'/'.$migration['name'];
- }
- }
|