aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rwxr-xr-xactiverecord/lib/active_record.rb1
-rwxr-xr-xactiverecord/lib/active_record/connection_adapters/abstract_adapter.rb48
-rwxr-xr-xactiverecord/lib/active_record/connection_adapters/mysql_adapter.rb39
-rw-r--r--activerecord/lib/active_record/connection_adapters/sqlite_adapter.rb17
-rw-r--r--activerecord/lib/active_record/migration.rb94
-rw-r--r--activerecord/test/aaa_create_tables_test.rb8
-rw-r--r--activerecord/test/fixtures/migrations/1_people_have_last_names.rb9
-rw-r--r--activerecord/test/fixtures/migrations/2_we_need_reminders.rb12
-rw-r--r--activerecord/test/migration_mysql.rb104
9 files changed, 317 insertions, 15 deletions
diff --git a/activerecord/lib/active_record.rb b/activerecord/lib/active_record.rb
index 11d327e6a9..ac49d114ab 100755
--- a/activerecord/lib/active_record.rb
+++ b/activerecord/lib/active_record.rb
@@ -47,6 +47,7 @@ require 'active_record/timestamp'
require 'active_record/acts/list'
require 'active_record/acts/tree'
require 'active_record/locking'
+require 'active_record/migration'
ActiveRecord::Base.class_eval do
include ActiveRecord::Validations
diff --git a/activerecord/lib/active_record/connection_adapters/abstract_adapter.rb b/activerecord/lib/active_record/connection_adapters/abstract_adapter.rb
index 605a82bf74..d686d71246 100755
--- a/activerecord/lib/active_record/connection_adapters/abstract_adapter.rb
+++ b/activerecord/lib/active_record/connection_adapters/abstract_adapter.rb
@@ -356,6 +356,38 @@ module ActiveRecord
sql << " LIMIT #{limit}"
end
+
+ def initialize_schema_information
+ begin
+ execute "CREATE TABLE schema_info (version #{native_database_types[:integer]})"
+ insert "INSERT INTO schema_info (version) VALUES(0)"
+ rescue ActiveRecord::StatementInvalid
+ # Schema has been intialized
+ end
+ end
+
+ def create_table(name)
+ execute "CREATE TABLE #{name} (id #{native_database_types[:primary_key]})"
+ table_definition = yield TableDefinition.new
+ table_definition.columns.each { |column_name, type, options| add_column(name, column_name, type, options) }
+ end
+
+ def drop_table(name)
+ execute "DROP TABLE #{name}"
+ end
+
+ def add_column(table_name, column_name, type, options = {})
+ add_column_sql = "ALTER TABLE #{table_name} ADD #{column_name} #{native_database_types[type]}"
+ add_column_sql << "(#{limit})" if options[:limit]
+ add_column_sql << " DEFAULT '#{options[:default]}'" if options[:default]
+ execute(add_column_sql)
+ end
+
+ def remove_column(table_name, column_name)
+ execute "ALTER TABLE #{table_name} DROP #{column_name}"
+ end
+
+
protected
def log(sql, name, connection = nil)
connection ||= @connection
@@ -402,6 +434,18 @@ module ActiveRecord
log_entry
end
end
-
+
+ class TableDefinition
+ attr_accessor :columns
+
+ def initialize
+ @columns = []
+ end
+
+ def column(name, type, options = {})
+ @columns << [ name, type, options ]
+ self
+ end
+ end
end
-end
+end \ No newline at end of file
diff --git a/activerecord/lib/active_record/connection_adapters/mysql_adapter.rb b/activerecord/lib/active_record/connection_adapters/mysql_adapter.rb
index cf7c700b66..0c78b921b6 100755
--- a/activerecord/lib/active_record/connection_adapters/mysql_adapter.rb
+++ b/activerecord/lib/active_record/connection_adapters/mysql_adapter.rb
@@ -63,12 +63,33 @@ module ActiveRecord
"Lost connection to MySQL server during query",
"MySQL server has gone away"
]
-
+
+ def native_database_types
+ {
+ :primary_key => "int(11) DEFAULT NULL auto_increment PRIMARY KEY",
+ :string => "varchar(255)",
+ :text => "text",
+ :integer => "int(11)",
+ :float => "float",
+ :datetime => "datetime",
+ :timestamp => "datetime",
+ :time => "datetime",
+ :date => "date",
+ :binary => "blob",
+ :boolean => "tinyint(1)"
+ }
+ end
+
def initialize(connection, logger, connection_options=nil)
super(connection, logger)
@connection_options = connection_options
end
+ def adapter_name
+ 'MySQL'
+ end
+
+
def select_all(sql, name = nil)
select(sql, name)
end
@@ -111,6 +132,7 @@ module ActiveRecord
alias_method :delete, :update
+
def begin_db_transaction
begin
execute "BEGIN"
@@ -134,15 +156,17 @@ module ActiveRecord
# Transactions aren't supported
end
end
+
def quote_column_name(name)
return "`#{name}`"
end
- def adapter_name()
- 'MySQL'
+ def quote_string(s)
+ Mysql::quote(s)
end
-
+
+
def structure_dump
select_all("SHOW TABLES").inject("") do |structure, table|
structure += select_one("SHOW CREATE TABLE #{table.to_a.first.last}")["Create Table"] + ";\n\n"
@@ -161,11 +185,8 @@ module ActiveRecord
def create_database(name)
execute "CREATE DATABASE #{name}"
end
-
- def quote_string(s)
- Mysql::quote(s)
- end
-
+
+
private
def select(sql, name = nil)
result = nil
diff --git a/activerecord/lib/active_record/connection_adapters/sqlite_adapter.rb b/activerecord/lib/active_record/connection_adapters/sqlite_adapter.rb
index 4958094ecb..ba786667cf 100644
--- a/activerecord/lib/active_record/connection_adapters/sqlite_adapter.rb
+++ b/activerecord/lib/active_record/connection_adapters/sqlite_adapter.rb
@@ -87,6 +87,22 @@ module ActiveRecord
#
# * <tt>:dbfile</tt> -- Path to the database file.
class SQLiteAdapter < AbstractAdapter
+ def native_database_types
+ {
+ :primary_key => "INTEGER PRIMARY KEY NOT NULL",
+ :string => "VARCHAR(255)",
+ :text => "TEXT",
+ :integer => "INTEGER",
+ :float => "float",
+ :datetime => "DATETIME",
+ :timestamp => "DATETIME",
+ :time => "DATETIME",
+ :date => "DATE",
+ :binary => "BLOB",
+ :boolean => "INTEGER"
+ }
+ end
+
def execute(sql, name = nil)
log(sql, name) { @connection.execute(sql) }
end
@@ -150,6 +166,7 @@ module ActiveRecord
'SQLite'
end
+
protected
def table_structure(table_name)
execute "PRAGMA table_info(#{table_name})"
diff --git a/activerecord/lib/active_record/migration.rb b/activerecord/lib/active_record/migration.rb
new file mode 100644
index 0000000000..8734a0831e
--- /dev/null
+++ b/activerecord/lib/active_record/migration.rb
@@ -0,0 +1,94 @@
+module ActiveRecord
+ class IrreversibleMigration < ActiveRecordError
+ end
+
+ class Migration
+ class << self
+ def up() end
+ def down() end
+
+ private
+ def method_missing(method, *arguments, &block)
+ ActiveRecord::Base.connection.send(method, *arguments, &block)
+ end
+ end
+ end
+
+ class Migrator
+ class << self
+ def up(migrations_path, target_version = nil)
+ new(:up, migrations_path, target_version).migrate
+ end
+
+ def down(migrations_path, target_version = nil)
+ new(:down, migrations_path, target_version).migrate
+ end
+
+ def current_version
+ Base.connection.select_one("SELECT version FROM schema_info")["version"].to_i
+ end
+ end
+
+ def initialize(direction, migrations_path, target_version = nil)
+ @direction, @migrations_path, @target_version = direction, migrations_path, target_version
+ end
+
+ def current_version
+ self.class.current_version
+ end
+
+ def migrate
+ migration_classes do |version, migration_class|
+ Base.logger.info("Reached target version: #{@target_version}") and break if reached_target_version?(version)
+ next if irrelevant_migration?(version)
+
+ Base.logger.info "Migrating to #{migration_class} (#{version})"
+ migration_class.send(@direction)
+
+ set_schema_version(version)
+ end
+ end
+
+ private
+ def migration_classes
+ for migration_file in migration_files
+ load(migration_file)
+ version, name = migration_version_and_name(migration_file)
+ yield version, migration_class(name)
+ end
+ end
+
+ def migration_files
+ files = Dir["#{@migrations_path}/[0-9]*_*.rb"]
+ down? ? files.reverse : files
+ end
+
+ def migration_class(migration_name)
+ migration_name.camelize.constantize
+ end
+
+ def migration_version_and_name(migration_file)
+ return *migration_file.scan(/([0-9]+)_([_a-z0-9]*).rb/).first
+ end
+
+ def set_schema_version(version)
+ Base.connection.update("UPDATE schema_info SET version = #{down? ? version.to_i - 1 : version.to_i}")
+ end
+
+ def up?
+ @direction == :up
+ end
+
+ def down?
+ @direction == :down
+ end
+
+ def reached_target_version?(version)
+ (up? && version.to_i - 1 == @target_version) || (down? && version.to_i == @target_version)
+ end
+
+ def irrelevant_migration?(version)
+ (up? && version.to_i <= current_version) || (down? && version.to_i > current_version)
+ end
+ end
+end \ No newline at end of file
diff --git a/activerecord/test/aaa_create_tables_test.rb b/activerecord/test/aaa_create_tables_test.rb
index ba14199d31..5d494946c1 100644
--- a/activerecord/test/aaa_create_tables_test.rb
+++ b/activerecord/test/aaa_create_tables_test.rb
@@ -45,13 +45,13 @@ class CreateTablesTest < Test::Unit::TestCase
def test_table_creation
adapter_name = ActiveRecord::Base.connection.adapter_name.downcase
- run_sql_file ActiveRecord::Base.connection, "test/fixtures/db_definitions/" + adapter_name + ".drop.sql"
- run_sql_file ActiveRecord::Base.connection, "test/fixtures/db_definitions/" + adapter_name + ".sql"
+ run_sql_file ActiveRecord::Base.connection, "#{File.dirname(__FILE__)}/fixtures/db_definitions/" + adapter_name + ".drop.sql"
+ run_sql_file ActiveRecord::Base.connection, "#{File.dirname(__FILE__)}/fixtures/db_definitions/" + adapter_name + ".sql"
# Now do the same thing with the connection used by multiple_db_test.rb
adapter_name = Course.retrieve_connection.adapter_name.downcase
- run_sql_file Course.retrieve_connection, "test/fixtures/db_definitions/" + adapter_name + "2.drop.sql"
- run_sql_file Course.retrieve_connection, "test/fixtures/db_definitions/" + adapter_name + "2.sql"
+ run_sql_file Course.retrieve_connection, "#{File.dirname(__FILE__)}/fixtures/db_definitions/" + adapter_name + "2.drop.sql"
+ run_sql_file Course.retrieve_connection, "#{File.dirname(__FILE__)}/fixtures/db_definitions/" + adapter_name + "2.sql"
assert_equal 1,1
end
diff --git a/activerecord/test/fixtures/migrations/1_people_have_last_names.rb b/activerecord/test/fixtures/migrations/1_people_have_last_names.rb
new file mode 100644
index 0000000000..009729b330
--- /dev/null
+++ b/activerecord/test/fixtures/migrations/1_people_have_last_names.rb
@@ -0,0 +1,9 @@
+class PeopleHaveLastNames < ActiveRecord::Migration
+ def self.up
+ add_column "people", "last_name", :string
+ end
+
+ def self.down
+ remove_column "people", "last_name"
+ end
+end \ No newline at end of file
diff --git a/activerecord/test/fixtures/migrations/2_we_need_reminders.rb b/activerecord/test/fixtures/migrations/2_we_need_reminders.rb
new file mode 100644
index 0000000000..ac5918f02a
--- /dev/null
+++ b/activerecord/test/fixtures/migrations/2_we_need_reminders.rb
@@ -0,0 +1,12 @@
+class WeNeedReminders < ActiveRecord::Migration
+ def self.up
+ create_table("reminders") do |t|
+ t.column :content, :text
+ t.column :remind_at, :datetime
+ end
+ end
+
+ def self.down
+ drop_table "reminders"
+ end
+end \ No newline at end of file
diff --git a/activerecord/test/migration_mysql.rb b/activerecord/test/migration_mysql.rb
new file mode 100644
index 0000000000..7777bb2c03
--- /dev/null
+++ b/activerecord/test/migration_mysql.rb
@@ -0,0 +1,104 @@
+require 'abstract_unit'
+require 'fixtures/person'
+require File.dirname(__FILE__) + '/fixtures/migrations/1_people_have_last_names'
+require File.dirname(__FILE__) + '/fixtures/migrations/2_we_need_reminders'
+
+class Reminder < ActiveRecord::Base; end
+
+class MigrationTest < Test::Unit::TestCase
+ def setup
+ end
+
+ def teardown
+ ActiveRecord::Base.connection.initialize_schema_information
+ ActiveRecord::Base.connection.update "UPDATE schema_info SET version = 0"
+
+ Reminder.connection.drop_table("reminders") rescue nil
+ Reminder.reset_column_information
+
+ Person.connection.remove_column("people", "last_name") rescue nil
+ Person.reset_column_information
+ end
+
+ def test_add_remove_single_field
+ assert !Person.column_methods_hash.include?(:last_name)
+
+ PeopleHaveLastNames.up
+
+ Person.reset_column_information
+ assert Person.column_methods_hash.include?(:last_name)
+
+ PeopleHaveLastNames.down
+
+ Person.reset_column_information
+ assert !Person.column_methods_hash.include?(:last_name)
+ end
+
+ def test_add_table
+ assert_raises(ActiveRecord::StatementInvalid) { Reminder.column_methods_hash }
+
+ WeNeedReminders.up
+
+ assert Reminder.create("content" => "hello world", "remind_at" => Time.now)
+ assert "hello world", Reminder.find_first
+
+ WeNeedReminders.down
+ assert_raises(ActiveRecord::StatementInvalid) { Reminder.find_first }
+ end
+
+ def test_migrator
+ assert !Person.column_methods_hash.include?(:last_name)
+ assert_raises(ActiveRecord::StatementInvalid) { Reminder.column_methods_hash }
+
+ ActiveRecord::Migrator.up(File.dirname(__FILE__) + '/fixtures/migrations/')
+
+ assert_equal 2, ActiveRecord::Migrator.current_version
+ Person.reset_column_information
+ assert Person.column_methods_hash.include?(:last_name)
+ assert Reminder.create("content" => "hello world", "remind_at" => Time.now)
+ assert "hello world", Reminder.find_first
+
+
+ ActiveRecord::Migrator.down(File.dirname(__FILE__) + '/fixtures/migrations/')
+
+ assert_equal 0, ActiveRecord::Migrator.current_version
+ Person.reset_column_information
+ assert !Person.column_methods_hash.include?(:last_name)
+ assert_raises(ActiveRecord::StatementInvalid) { Reminder.find_first }
+ end
+
+ def test_migrator_one_up
+ assert !Person.column_methods_hash.include?(:last_name)
+ assert_raises(ActiveRecord::StatementInvalid) { Reminder.column_methods_hash }
+
+ ActiveRecord::Migrator.up(File.dirname(__FILE__) + '/fixtures/migrations/', 1)
+
+ Person.reset_column_information
+ assert Person.column_methods_hash.include?(:last_name)
+ assert_raises(ActiveRecord::StatementInvalid) { Reminder.column_methods_hash }
+
+
+ ActiveRecord::Migrator.up(File.dirname(__FILE__) + '/fixtures/migrations/', 2)
+
+ assert Reminder.create("content" => "hello world", "remind_at" => Time.now)
+ assert "hello world", Reminder.find_first
+ end
+
+ def test_migrator_one_down
+ ActiveRecord::Migrator.up(File.dirname(__FILE__) + '/fixtures/migrations/')
+
+ ActiveRecord::Migrator.down(File.dirname(__FILE__) + '/fixtures/migrations/', 1)
+
+ Person.reset_column_information
+ assert Person.column_methods_hash.include?(:last_name)
+ assert_raises(ActiveRecord::StatementInvalid) { Reminder.column_methods_hash }
+ end
+
+ def test_migrator_one_up_one_down
+ ActiveRecord::Migrator.up(File.dirname(__FILE__) + '/fixtures/migrations/', 1)
+ ActiveRecord::Migrator.down(File.dirname(__FILE__) + '/fixtures/migrations/', 0)
+
+ assert !Person.column_methods_hash.include?(:last_name)
+ assert_raises(ActiveRecord::StatementInvalid) { Reminder.column_methods_hash }
+ end
+end \ No newline at end of file