diff options
author | Jeremy Kemper <jeremy@bitsweat.net> | 2007-01-15 02:22:53 +0000 |
---|---|---|
committer | Jeremy Kemper <jeremy@bitsweat.net> | 2007-01-15 02:22:53 +0000 |
commit | 6019c268562d3d1e62c3ad649e81a1803f3e0732 (patch) | |
tree | 4fbad66ece81acb10c8d7a705339565c3ae6c459 | |
parent | a491f928608f763cf4fa6067f7cf9529197afcae (diff) | |
download | rails-6019c268562d3d1e62c3ad649e81a1803f3e0732.tar.gz rails-6019c268562d3d1e62c3ad649e81a1803f3e0732.tar.bz2 rails-6019c268562d3d1e62c3ad649e81a1803f3e0732.zip |
change_column accepts :default => nil. Closes #6956.
git-svn-id: http://svn-commit.rubyonrails.org/rails/trunk@5937 5ecf4fe2-1ee6-0310-87b1-e25e094e27de
11 files changed, 51 insertions, 31 deletions
diff --git a/activerecord/CHANGELOG b/activerecord/CHANGELOG index 4b7fc376df..4d44538b91 100644 --- a/activerecord/CHANGELOG +++ b/activerecord/CHANGELOG @@ -1,5 +1,7 @@ *SVN* +* change_column accepts :default => nil. #6956 [dcmanges, Jeremy Kemper] + * MySQL, PostgreSQL: change_column_default quotes the default value and doesn't lose column type information. #3987, #6664 [Jonathan Viney, manfred, altano@bigfoot.com] * Oracle: create_table takes a :sequence_name option to override the 'tablename_seq' default. #7000 [Michael Schoen] diff --git a/activerecord/lib/active_record/connection_adapters/abstract/schema_definitions.rb b/activerecord/lib/active_record/connection_adapters/abstract/schema_definitions.rb index 799b302a94..1b6ccdc7fc 100644 --- a/activerecord/lib/active_record/connection_adapters/abstract/schema_definitions.rb +++ b/activerecord/lib/active_record/connection_adapters/abstract/schema_definitions.rb @@ -253,9 +253,7 @@ module ActiveRecord # Requests a maximum column length (<tt>:string</tt>, <tt>:text</tt>, # <tt>:binary</tt> or <tt>:integer</tt> columns only) # * <tt>:default</tt>: - # The column's default value. You cannot explicitely set the default - # value to +NULL+. Simply leave off this option if you want a +NULL+ - # default value. + # The column's default value. Use nil for NULL. # * <tt>:null</tt>: # Allows or disallows +NULL+ values in the column. This option could # have been named <tt>:null_allowed</tt>. @@ -271,11 +269,11 @@ module ActiveRecord # <tt>:precision</tt>. # * MySQL: <tt>:precision</tt> [1..63], <tt>:scale</tt> [0..30]. # Default is (10,0). - # * PostGres?: <tt>:precision</tt> [1..infinity], + # * PostgreSQL: <tt>:precision</tt> [1..infinity], # <tt>:scale</tt> [0..infinity]. No default. - # * Sqlite2: Any <tt>:precision</tt> and <tt>:scale</tt> may be used. + # * SQLite2: Any <tt>:precision</tt> and <tt>:scale</tt> may be used. # Internal storage as strings. No default. - # * Sqlite3: No restrictions on <tt>:precision</tt> and <tt>:scale</tt>, + # * SQLite3: No restrictions on <tt>:precision</tt> and <tt>:scale</tt>, # but the maximum supported <tt>:precision</tt> is 16. No default. # * Oracle: <tt>:precision</tt> [1..38], <tt>:scale</tt> [-84..127]. # Default is (38,0). diff --git a/activerecord/lib/active_record/connection_adapters/abstract/schema_statements.rb b/activerecord/lib/active_record/connection_adapters/abstract/schema_statements.rb index 8891c10c7c..edb3042afc 100644 --- a/activerecord/lib/active_record/connection_adapters/abstract/schema_statements.rb +++ b/activerecord/lib/active_record/connection_adapters/abstract/schema_statements.rb @@ -273,10 +273,10 @@ module ActiveRecord column_type_sql << "(#{limit})" if limit column_type_sql end - end - + end + def add_column_options!(sql, options) #:nodoc: - sql << " DEFAULT #{quote(options[:default], options[:column])}" unless options[:default].nil? + sql << " DEFAULT #{quote(options[:default], options[:column])}" if options_include_default?(options) sql << " NOT NULL" if options[:null] == false end @@ -293,6 +293,11 @@ module ActiveRecord def add_order_by_for_association_limiting!(sql, options) sql << "ORDER BY #{options[:order]}" end + + protected + def options_include_default?(options) + options.include?(:default) && !(options[:null] == false && options[:default].nil?) + end end end end diff --git a/activerecord/lib/active_record/connection_adapters/firebird_adapter.rb b/activerecord/lib/active_record/connection_adapters/firebird_adapter.rb index 36222e4b0b..04f9f754a4 100644 --- a/activerecord/lib/active_record/connection_adapters/firebird_adapter.rb +++ b/activerecord/lib/active_record/connection_adapters/firebird_adapter.rb @@ -505,8 +505,8 @@ module ActiveRecord def change_column(table_name, column_name, type, options = {}) # :nodoc: change_column_type(table_name, column_name, type, options) - change_column_position(table_name, column_name, options[:position]) if options[:position] - change_column_default(table_name, column_name, options[:default]) if options.has_key?(:default) + change_column_position(table_name, column_name, options[:position]) if options.include?(:position) + change_column_default(table_name, column_name, options[:default]) if options_include_default?(options) end def change_column_default(table_name, column_name, default) # :nodoc: diff --git a/activerecord/lib/active_record/connection_adapters/frontbase_adapter.rb b/activerecord/lib/active_record/connection_adapters/frontbase_adapter.rb index cd48978f4e..a6df79ad0d 100644 --- a/activerecord/lib/active_record/connection_adapters/frontbase_adapter.rb +++ b/activerecord/lib/active_record/connection_adapters/frontbase_adapter.rb @@ -795,12 +795,12 @@ module ActiveRecord def add_column_options!(sql, options) #:nodoc: default_value = quote(options[:default], options[:column]) - if options[:default] + if options_include_default?(options) if options[:type] == :boolean default_value = options[:default] == 0 ? quoted_false : quoted_true end + sql << " DEFAULT #{default_value}" end - sql << " DEFAULT #{default_value}" unless options[:default].nil? sql << " NOT NULL" if options[:null] == false end @@ -828,17 +828,15 @@ module ActiveRecord execute(change_column_sql) change_column_sql = %( ALTER TABLE "#{table_name}" ALTER COLUMN "#{column_name}" ) - default_value = quote(options[:default], options[:column]) - if options[:default] + if options_include_default?(options) + default_value = quote(options[:default], options[:column]) if type == :boolean default_value = options[:default] == 0 ? quoted_false : quoted_true end - end - - if default_value != "NULL" change_column_sql << " SET DEFAULT #{default_value}" - execute(change_column_sql) end + + execute(change_column_sql) # change_column_sql = "ALTER TABLE #{table_name} CHANGE #{column_name} #{column_name} #{type_to_sql(type, options[:limit])}" # add_column_options!(change_column_sql, options) diff --git a/activerecord/lib/active_record/connection_adapters/mysql_adapter.rb b/activerecord/lib/active_record/connection_adapters/mysql_adapter.rb index de3cd06224..275856dc97 100755 --- a/activerecord/lib/active_record/connection_adapters/mysql_adapter.rb +++ b/activerecord/lib/active_record/connection_adapters/mysql_adapter.rb @@ -364,10 +364,10 @@ module ActiveRecord end def change_column(table_name, column_name, type, options = {}) #:nodoc: - if options[:default].nil? + unless options_include_default?(options) options[:default] = select_one("SHOW COLUMNS FROM #{table_name} LIKE '#{column_name}'")["Default"] end - + change_column_sql = "ALTER TABLE #{table_name} CHANGE #{column_name} #{column_name} #{type_to_sql(type, options[:limit], options[:precision], options[:scale])}" add_column_options!(change_column_sql, options) execute(change_column_sql) diff --git a/activerecord/lib/active_record/connection_adapters/postgresql_adapter.rb b/activerecord/lib/active_record/connection_adapters/postgresql_adapter.rb index db946478f9..eb9d0e94df 100644 --- a/activerecord/lib/active_record/connection_adapters/postgresql_adapter.rb +++ b/activerecord/lib/active_record/connection_adapters/postgresql_adapter.rb @@ -321,10 +321,10 @@ module ActiveRecord execute("ALTER TABLE #{table_name} ADD COLUMN #{column_name} #{type_to_sql(type, options[:limit])}") # Set optional default. If not null, update nulls to the new default. - unless default.nil? + if options_include_default?(options) change_column_default(table_name, column_name, default) if notnull - execute("UPDATE #{table_name} SET #{column_name}='#{default}' WHERE #{column_name} IS NULL") + execute("UPDATE #{table_name} SET #{column_name}=#{quote(default, options[:column])} WHERE #{column_name} IS NULL") end end @@ -345,7 +345,10 @@ module ActiveRecord rename_column(table_name, "#{column_name}_ar_tmp", column_name) commit_db_transaction end - change_column_default(table_name, column_name, options[:default]) unless options[:default].nil? + + if options_include_default?(options) + change_column_default(table_name, column_name, options[:default]) + end end def change_column_default(table_name, column_name, default) #:nodoc: diff --git a/activerecord/lib/active_record/connection_adapters/sqlite_adapter.rb b/activerecord/lib/active_record/connection_adapters/sqlite_adapter.rb index 341e546084..2d59c2f8fd 100644 --- a/activerecord/lib/active_record/connection_adapters/sqlite_adapter.rb +++ b/activerecord/lib/active_record/connection_adapters/sqlite_adapter.rb @@ -264,10 +264,11 @@ module ActiveRecord def change_column(table_name, column_name, type, options = {}) #:nodoc: alter_table(table_name) do |definition| + include_default = options_include_default?(options) definition[column_name].instance_eval do self.type = type - self.limit = options[:limit] if options[:limit] - self.default = options[:default] unless options[:default].nil? + self.limit = options[:limit] if options.include?(:limit) + self.default = options[:default] if include_default end end end diff --git a/activerecord/lib/active_record/connection_adapters/sqlserver_adapter.rb b/activerecord/lib/active_record/connection_adapters/sqlserver_adapter.rb index b234239781..8a94e55e30 100644 --- a/activerecord/lib/active_record/connection_adapters/sqlserver_adapter.rb +++ b/activerecord/lib/active_record/connection_adapters/sqlserver_adapter.rb @@ -459,9 +459,9 @@ module ActiveRecord def change_column(table_name, column_name, type, options = {}) #:nodoc: sql_commands = ["ALTER TABLE #{table_name} ALTER COLUMN #{column_name} #{type_to_sql(type, options[:limit], options[:precision], options[:scale])}"] - unless options[:default].nil? + if options_include_default?(options) remove_default_constraint(table_name, column_name) - sql_commands << "ALTER TABLE #{table_name} ADD CONSTRAINT DF_#{table_name}_#{column_name} DEFAULT #{quote(options[:default])} FOR #{column_name}" + sql_commands << "ALTER TABLE #{table_name} ADD CONSTRAINT DF_#{table_name}_#{column_name} DEFAULT #{quote(options[:default], options[:column])} FOR #{column_name}" end sql_commands.each {|c| execute(c) diff --git a/activerecord/lib/active_record/connection_adapters/sybase_adapter.rb b/activerecord/lib/active_record/connection_adapters/sybase_adapter.rb index 4287dbe144..708a11b230 100644 --- a/activerecord/lib/active_record/connection_adapters/sybase_adapter.rb +++ b/activerecord/lib/active_record/connection_adapters/sybase_adapter.rb @@ -381,7 +381,7 @@ SQLTEXT end def add_column_options!(sql, options) #:nodoc: - sql << " DEFAULT #{quote(options[:default], options[:column])}" unless options[:default].nil? + sql << " DEFAULT #{quote(options[:default], options[:column])}" if options_include_default?(options) if check_null_for_column?(options[:column], sql) sql << (options[:null] == false ? " NOT NULL" : " NULL") diff --git a/activerecord/test/migration_test.rb b/activerecord/test/migration_test.rb index 10cab682df..825244d88e 100644 --- a/activerecord/test/migration_test.rb +++ b/activerecord/test/migration_test.rb @@ -94,8 +94,10 @@ if ActiveRecord::Base.connection.supports_migrations? end def test_create_table_with_not_null_column - Person.connection.create_table :testings do |t| - t.column :foo, :string, :null => false + assert_nothing_raised do + Person.connection.create_table :testings do |t| + t.column :foo, :string, :null => false + end end assert_raises(ActiveRecord::StatementInvalid) do @@ -423,6 +425,17 @@ if ActiveRecord::Base.connection.supports_migrations? assert new_columns.find { |c| c.name == 'approved' and c.type == :boolean and c.default == false } assert_nothing_raised { Topic.connection.change_column :topics, :approved, :boolean, :default => true } end + + def test_change_column_with_nil_default + Person.connection.add_column "people", "contributor", :boolean, :default => true + Person.reset_column_information + assert Person.new.contributor? + + assert_nothing_raised { Person.connection.change_column "people", "contributor", :boolean, :default => nil } + Person.reset_column_information + assert !Person.new.contributor? + assert_nil Person.new.contributor + end def test_change_column_with_new_default Person.connection.add_column "people", "administrator", :boolean, :default => true |