328 行
		
	
	
	
		
			7.8 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			328 行
		
	
	
	
		
			7.8 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php defined('BASEPATH') OR exit('No direct script access allowed');
 | |
| /**
 | |
|  * CodeIgniter
 | |
|  *
 | |
|  * An open source application development framework for PHP 5.1.6 or newer
 | |
|  *
 | |
|  * @package		CodeIgniter
 | |
|  * @author		EllisLab Dev Team
 | |
|  * @copyright	Copyright (c) 2006 - 2012, EllisLab, Inc.
 | |
|  * @license		http://codeigniter.com/user_guide/license.html
 | |
|  * @link		http://codeigniter.com
 | |
|  * @since		Version 1.0
 | |
|  * @filesource
 | |
|  */
 | |
| 
 | |
| // ------------------------------------------------------------------------
 | |
| 
 | |
| /**
 | |
|  * Migration Class
 | |
|  *
 | |
|  * All migrations should implement this, forces up() and down() and gives
 | |
|  * access to the CI super-global.
 | |
|  *
 | |
|  * @package		CodeIgniter
 | |
|  * @subpackage	Libraries
 | |
|  * @category	Libraries
 | |
|  * @author		Reactor Engineers
 | |
|  * @link
 | |
|  */
 | |
| class CI_Migration {
 | |
| 
 | |
| 	protected $_migration_enabled = FALSE;
 | |
| 	protected $_migration_path = NULL;
 | |
| 	protected $_migration_version = 0;
 | |
| 
 | |
| 	protected $_error_string = '';
 | |
| 
 | |
| 	public function __construct($config = array())
 | |
| 	{
 | |
| 		# Only run this constructor on main library load
 | |
| 		if (get_parent_class($this) !== FALSE)
 | |
| 		{
 | |
| 			return;
 | |
| 		}
 | |
| 
 | |
| 		foreach ($config as $key => $val)
 | |
| 		{
 | |
| 			$this->{'_' . $key} = $val;
 | |
| 		}
 | |
| 
 | |
| 		log_message('debug', 'Migrations class initialized');
 | |
| 
 | |
| 		// Are they trying to use migrations while it is disabled?
 | |
| 		if ($this->_migration_enabled !== TRUE)
 | |
| 		{
 | |
| 			show_error('Migrations has been loaded but is disabled or set up incorrectly.');
 | |
| 		}
 | |
| 
 | |
| 		// If not set, set it
 | |
| 		$this->_migration_path == '' AND $this->_migration_path = APPPATH . 'migrations/';
 | |
| 
 | |
| 		// Add trailing slash if not set
 | |
| 		$this->_migration_path = rtrim($this->_migration_path, '/').'/';
 | |
| 
 | |
| 		// Load migration language
 | |
| 		$this->lang->load('migration');
 | |
| 
 | |
| 		// They'll probably be using dbforge
 | |
| 		$this->load->dbforge();
 | |
| 
 | |
| 		// If the migrations table is missing, make it
 | |
| 		if ( ! $this->db->table_exists('migrations'))
 | |
| 		{
 | |
| 			$this->dbforge->add_field(array(
 | |
| 				'version' => array('type' => 'INT', 'constraint' => 3),
 | |
| 			));
 | |
| 
 | |
| 			$this->dbforge->create_table('migrations', TRUE);
 | |
| 
 | |
| 			$this->db->insert('migrations', array('version' => 0));
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	// --------------------------------------------------------------------
 | |
| 
 | |
| 	/**
 | |
| 	 * Migrate to a schema version
 | |
| 	 *
 | |
| 	 * Calls each migration step required to get to the schema version of
 | |
| 	 * choice
 | |
| 	 *
 | |
| 	 * @param	int	Target schema version
 | |
| 	 * @return	mixed	TRUE if already latest, FALSE if failed, int if upgraded
 | |
| 	 */
 | |
| 	public function version($target_version)
 | |
| 	{
 | |
| 		$start = $current_version = $this->_get_version();
 | |
| 		$stop = $target_version;
 | |
| 
 | |
| 		if ($target_version > $current_version)
 | |
| 		{
 | |
| 			// Moving Up
 | |
| 			++$start;
 | |
| 			++$stop;
 | |
| 			$step = 1;
 | |
| 		}
 | |
| 		else
 | |
| 		{
 | |
| 			// Moving Down
 | |
| 			$step = -1;
 | |
| 		}
 | |
| 
 | |
| 		$method = ($step === 1) ? 'up' : 'down';
 | |
| 		$migrations = array();
 | |
| 
 | |
| 		// We now prepare to actually DO the migrations
 | |
| 		// But first let's make sure that everything is the way it should be
 | |
| 		for ($i = $start; $i != $stop; $i += $step)
 | |
| 		{
 | |
| 			$f = glob(sprintf($this->_migration_path . '%03d_*.php', $i));
 | |
| 
 | |
| 			// Only one migration per step is permitted
 | |
| 			if (count($f) > 1)
 | |
| 			{
 | |
| 				$this->_error_string = sprintf($this->lang->line('migration_multiple_version'), $i);
 | |
| 				return FALSE;
 | |
| 			}
 | |
| 
 | |
| 			// Migration step not found
 | |
| 			if (count($f) == 0)
 | |
| 			{
 | |
| 				// If trying to migrate up to a version greater than the last
 | |
| 				// existing one, migrate to the last one.
 | |
| 				if ($step == 1)
 | |
| 				{
 | |
| 					break;
 | |
| 				}
 | |
| 
 | |
| 				// If trying to migrate down but we're missing a step,
 | |
| 				// something must definitely be wrong.
 | |
| 				$this->_error_string = sprintf($this->lang->line('migration_not_found'), $i);
 | |
| 				return FALSE;
 | |
| 			}
 | |
| 
 | |
| 			$file = basename($f[0]);
 | |
| 			$name = basename($f[0], '.php');
 | |
| 
 | |
| 			// Filename validations
 | |
| 			if (preg_match('/^\d{3}_(\w+)$/', $name, $match))
 | |
| 			{
 | |
| 				$match[1] = strtolower($match[1]);
 | |
| 
 | |
| 				// Cannot repeat a migration at different steps
 | |
| 				if (in_array($match[1], $migrations))
 | |
| 				{
 | |
| 					$this->_error_string = sprintf($this->lang->line('migration_multiple_version'), $match[1]);
 | |
| 					return FALSE;
 | |
| 				}
 | |
| 
 | |
| 				include $f[0];
 | |
| 				$class = 'Migration_' . ucfirst($match[1]);
 | |
| 
 | |
| 				if ( ! class_exists($class))
 | |
| 				{
 | |
| 					$this->_error_string = sprintf($this->lang->line('migration_class_doesnt_exist'), $class);
 | |
| 					return FALSE;
 | |
| 				}
 | |
| 
 | |
| 				if ( ! is_callable(array($class, $method)))
 | |
| 				{
 | |
| 					$this->_error_string = sprintf($this->lang->line('migration_missing_'.$method.'_method'), $class);
 | |
| 					return FALSE;
 | |
| 				}
 | |
| 
 | |
| 				$migrations[] = $match[1];
 | |
| 			}
 | |
| 			else
 | |
| 			{
 | |
| 				$this->_error_string = sprintf($this->lang->line('migration_invalid_filename'), $file);
 | |
| 				return FALSE;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		log_message('debug', 'Current migration: ' . $current_version);
 | |
| 
 | |
| 		$version = $i + ($step == 1 ? -1 : 0);
 | |
| 
 | |
| 		// If there is nothing to do so quit
 | |
| 		if ($migrations === array())
 | |
| 		{
 | |
| 			return TRUE;
 | |
| 		}
 | |
| 
 | |
| 		log_message('debug', 'Migrating from ' . $method . ' to version ' . $version);
 | |
| 
 | |
| 		// Loop through the migrations
 | |
| 		foreach ($migrations AS $migration)
 | |
| 		{
 | |
| 			// Run the migration class
 | |
| 			$class = 'Migration_' . ucfirst(strtolower($migration));
 | |
| 			call_user_func(array(new $class, $method));
 | |
| 
 | |
| 			$current_version += $step;
 | |
| 			$this->_update_version($current_version);
 | |
| 		}
 | |
| 
 | |
| 		log_message('debug', 'Finished migrating to '.$current_version);
 | |
| 
 | |
| 		return $current_version;
 | |
| 	}
 | |
| 
 | |
| 	// --------------------------------------------------------------------
 | |
| 
 | |
| 	/**
 | |
| 	 * Set's the schema to the latest migration
 | |
| 	 *
 | |
| 	 * @return	mixed	true if already latest, false if failed, int if upgraded
 | |
| 	 */
 | |
| 	public function latest()
 | |
| 	{
 | |
| 		if ( ! $migrations = $this->find_migrations())
 | |
| 		{
 | |
| 			$this->_error_string = $this->line->lang('migration_none_found');
 | |
| 			return false;
 | |
| 		}
 | |
| 
 | |
| 		$last_migration = basename(end($migrations));
 | |
| 
 | |
| 		// Calculate the last migration step from existing migration
 | |
| 		// filenames and procceed to the standard version migration
 | |
| 		return $this->version((int) substr($last_migration, 0, 3));
 | |
| 	}
 | |
| 
 | |
| 	// --------------------------------------------------------------------
 | |
| 
 | |
| 	/**
 | |
| 	 * Set's the schema to the migration version set in config
 | |
| 	 *
 | |
| 	 * @return	mixed	true if already current, false if failed, int if upgraded
 | |
| 	 */
 | |
| 	public function current()
 | |
| 	{
 | |
| 		return $this->version($this->_migration_version);
 | |
| 	}
 | |
| 
 | |
| 	// --------------------------------------------------------------------
 | |
| 
 | |
| 	/**
 | |
| 	 * Error string
 | |
| 	 *
 | |
| 	 * @return	string	Error message returned as a string
 | |
| 	 */
 | |
| 	public function error_string()
 | |
| 	{
 | |
| 		return $this->_error_string;
 | |
| 	}
 | |
| 
 | |
| 	// --------------------------------------------------------------------
 | |
| 
 | |
| 	/**
 | |
| 	 * Set's the schema to the latest migration
 | |
| 	 *
 | |
| 	 * @return	mixed	true if already latest, false if failed, int if upgraded
 | |
| 	 */
 | |
| 	protected function find_migrations()
 | |
| 	{
 | |
| 		// Load all *_*.php files in the migrations path
 | |
| 		$files = glob($this->_migration_path . '*_*.php');
 | |
| 		$file_count = count($files);
 | |
| 
 | |
| 		for ($i = 0; $i < $file_count; $i++)
 | |
| 		{
 | |
| 			// Mark wrongly formatted files as false for later filtering
 | |
| 			$name = basename($files[$i], '.php');
 | |
| 			if ( ! preg_match('/^\d{3}_(\w+)$/', $name))
 | |
| 			{
 | |
| 				$files[$i] = FALSE;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		sort($files);
 | |
| 		return $files;
 | |
| 	}
 | |
| 
 | |
| 	// --------------------------------------------------------------------
 | |
| 
 | |
| 	/**
 | |
| 	 * Retrieves current schema version
 | |
| 	 *
 | |
| 	 * @return	int	Current Migration
 | |
| 	 */
 | |
| 	protected function _get_version()
 | |
| 	{
 | |
| 		$row = $this->db->get('migrations')->row();
 | |
| 		return $row ? $row->version : 0;
 | |
| 	}
 | |
| 
 | |
| 	// --------------------------------------------------------------------
 | |
| 
 | |
| 	/**
 | |
| 	 * Stores the current schema version
 | |
| 	 *
 | |
| 	 * @param	int	Migration reached
 | |
| 	 * @return	bool
 | |
| 	 */
 | |
| 	protected function _update_version($migrations)
 | |
| 	{
 | |
| 		return $this->db->update('migrations', array(
 | |
| 			'version' => $migrations
 | |
| 		));
 | |
| 	}
 | |
| 
 | |
| 	// --------------------------------------------------------------------
 | |
| 
 | |
| 	/**
 | |
| 	 * Enable the use of CI super-global
 | |
| 	 *
 | |
| 	 * @param	mixed	$var
 | |
| 	 * @return	mixed
 | |
| 	 */
 | |
| 	public function __get($var)
 | |
| 	{
 | |
| 		return get_instance()->$var;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| /* End of file Migration.php */
 | |
| /* Location: ./system/libraries/Migration.php */
 |