- A
- C
- D
- I
- N
- O
- Q
- R
- S
- T
Adds a new column to the named table. See TableDefinition#column for details of the options you can use.
# File activerecord/lib/active_record/connection_adapters/abstract/schema_statements.rb, line 261 261: def add_column(table_name, column_name, type, options = {}) 262: add_column_sql = "ALTER TABLE #{quote_table_name(table_name)} ADD #{quote_column_name(column_name)} #{type_to_sql(type, options[:limit], options[:precision], options[:scale])}" 263: add_column_options!(add_column_sql, options) 264: execute(add_column_sql) 265: end
Adds a new index to the table. column_name can be a single Symbol, or an Array of Symbols.
The index will be named after the table and the column name(s), unless you pass :name as an option.
Examples
Creating a simple index
add_index(:suppliers, :name)
generates
CREATE INDEX suppliers_name_index ON suppliers(name)
Creating a unique index
add_index(:accounts, [:branch_id, :party_id], :unique => true)
generates
CREATE UNIQUE INDEX accounts_branch_id_party_id_index ON accounts(branch_id, party_id)
Creating a named index
add_index(:accounts, [:branch_id, :party_id], :unique => true, :name => 'by_branch_party')
generates
CREATE UNIQUE INDEX by_branch_party ON accounts(branch_id, party_id)
Creating an index with specific key length
add_index(:accounts, :name, :name => 'by_name', :length => 10)
generates
CREATE INDEX by_name ON accounts(name(10)) add_index(:accounts, [:name, :surname], :name => 'by_name_surname', :length => {:name => 10, :surname => 15})
generates
CREATE INDEX by_name_surname ON accounts(name(10), surname(15))
Note: SQLite doesn’t support index length
Creating an index with a sort order (desc or asc, asc is the default)
add_index(:accounts, [:branch_id, :party_id, :surname], :order => {:branch_id => :desc, :part_id => :asc})
generates
CREATE INDEX by_branch_desc_party ON accounts(branch_id DESC, party_id ASC, surname)
Note: mysql doesn’t yet support index order (it accepts the syntax but ignores it)
# File activerecord/lib/active_record/connection_adapters/abstract/schema_statements.rb, line 350 350: def add_index(table_name, column_name, options = {}) 351: index_name, index_type, index_columns = add_index_options(table_name, column_name, options) 352: execute "CREATE #{index_type} INDEX #{quote_column_name(index_name)} ON #{quote_table_name(table_name)} (#{index_columns})" 353: end
Adds timestamps (created_at and updated_at) columns to the named table.
Examples
add_timestamps(:suppliers)
# File activerecord/lib/active_record/connection_adapters/abstract/schema_statements.rb, line 446 446: def assume_migrated_upto_version(version, migrations_paths = ActiveRecord::Migrator.migrations_paths) 447: migrations_paths = Array.wrap(migrations_paths) 448: version = version.to_i 449: sm_table = quote_table_name(ActiveRecord::Migrator.schema_migrations_table_name) 450: 451: migrated = select_values("SELECT version FROM #{sm_table}").map { |v| v.to_i } 452: paths = migrations_paths.map {|p| "#{p}/[0-9]*_*.rb" } 453: versions = Dir[*paths].map do |filename| 454: filename.split('/').last.split('_').first.to_i 455: end 456: 457: unless migrated.include?(version) 458: execute "INSERT INTO #{sm_table} (version) VALUES ('#{version}')" 459: end 460: 461: inserted = Set.new 462: (versions - migrated).each do |v| 463: if inserted.include?(v) 464: raise "Duplicate migration #{v}. Please renumber your migrations to resolve the conflict." 465: elsif v < version 466: execute "INSERT INTO #{sm_table} (version) VALUES ('#{v}')" 467: inserted << v 468: end 469: end 470: end
Changes the column’s definition according to the new options. See TableDefinition#column for details of the options you can use.
Examples
change_column(:suppliers, :name, :string, :limit => 80) change_column(:accounts, :description, :text)
Sets a new default value for a column.
Examples
change_column_default(:suppliers, :qualification, 'new') change_column_default(:accounts, :authorized, 1) change_column_default(:users, :email, nil)
A block for changing columns in table.
Example
# change_table() yields a Table instance change_table(:suppliers) do |t| t.column :name, :string, :limit => 60 # Other column alterations here end
The options hash can include the following keys:
- :bulk
- Set this to true to make this a bulk alter query, such as ALTER TABLE
`users` ADD COLUMN age INT(11), ADD COLUMN birthdate DATETIME …
Defaults to false.
Examples
Add a column
change_table(:suppliers) do |t| t.column :name, :string, :limit => 60 end
Add 2 integer columns
change_table(:suppliers) do |t| t.integer :width, :height, :null => false, :default => 0 end
Add created_at/updated_at columns
change_table(:suppliers) do |t| t.timestamps end
Add a foreign key column
change_table(:suppliers) do |t| t.references :company end
Creates a company_id(integer) column
Add a polymorphic foreign key column
change_table(:suppliers) do |t| t.belongs_to :company, :polymorphic => true end
Creates company_type(varchar) and company_id(integer) columns
Remove a column
change_table(:suppliers) do |t| t.remove :company end
Remove several columns
change_table(:suppliers) do |t| t.remove :company_id t.remove :width, :height end
Remove an index
change_table(:suppliers) do |t| t.remove_index :company_id end
See also Table for details on all of the various column transformation
# File activerecord/lib/active_record/connection_adapters/abstract/schema_statements.rb, line 237 237: def change_table(table_name, options = {}) 238: if supports_bulk_alter? && options[:bulk] 239: recorder = ActiveRecord::Migration::CommandRecorder.new(self) 240: yield Table.new(table_name, recorder) 241: bulk_change_table(table_name, recorder.commands) 242: else 243: yield Table.new(table_name, self) 244: end 245: end
Checks to see if a column exists in a given table.
Examples
# Check a column exists column_exists?(:suppliers, :name) # Check a column exists of a particular type column_exists?(:suppliers, :name, :string) # Check a column exists with a specific definition column_exists?(:suppliers, :name, :string, :limit => 100)
# File activerecord/lib/active_record/connection_adapters/abstract/schema_statements.rb, line 69 69: def column_exists?(table_name, column_name, type = nil, options = {}) 70: columns(table_name).any?{ |c| c.name == column_name.to_s && 71: (!type || c.type == type) && 72: (!options[:limit] || c.limit == options[:limit]) && 73: (!options[:precision] || c.precision == options[:precision]) && 74: (!options[:scale] || c.scale == options[:scale]) } 75: end
Returns an array of Column objects for the table specified by table_name. See the concrete implementation for details on the expected parameter values.
Creates a new table with the name table_name. table_name may either be a String or a Symbol.
There are two ways to work with create_table. You can use the block form or the regular form, like this:
Block form
# create_table() passes a TableDefinition object to the block. # This form will not only create the table, but also columns for the # table. create_table(:suppliers) do |t| t.column :name, :string, :limit => 60 # Other fields here end
Block form, with shorthand
# You can also use the column types as method calls, rather than calling the column method. create_table(:suppliers) do |t| t.string :name, :limit => 60 # Other fields here end
Regular form
# Creates a table called 'suppliers' with no columns. create_table(:suppliers) # Add a column to 'suppliers'. add_column(:suppliers, :name, :string, {:limit => 60})
The options hash can include the following keys:
- :id
- Whether to automatically add a primary key column. Defaults to true. Join tables for has_and_belongs_to_many should set it to false.
- :primary_key
- The name of the primary key, if one is to be added automatically. Defaults
to id. If :id is false this option is ignored.
Also note that this just sets the primary key in the table. You additionally need to configure the primary key in the model via self.primary_key=. Models do NOT auto-detect the primary key from their table definition.
- :options
- Any extra options you want appended to the table definition.
- :temporary
- Make a temporary table.
- :force
- Set to true to drop the table before creating it. Defaults to false.
Examples
Add a backend specific option to the generated SQL (MySQL)
create_table(:suppliers, :options => 'ENGINE=InnoDB DEFAULT CHARSET=utf8')
generates:
CREATE TABLE suppliers ( id int(11) DEFAULT NULL auto_increment PRIMARY KEY ) ENGINE=InnoDB DEFAULT CHARSET=utf8
Rename the primary key column
create_table(:objects, :primary_key => 'guid') do |t| t.column :name, :string, :limit => 80 end
generates:
CREATE TABLE objects ( guid int(11) DEFAULT NULL auto_increment PRIMARY KEY, name varchar(80) )
Do not add a primary key column
create_table(:categories_suppliers, :id => false) do |t| t.column :category_id, :integer t.column :supplier_id, :integer end
generates:
CREATE TABLE categories_suppliers ( category_id int, supplier_id int )
See also TableDefinition#column for details on how to create columns.
# File activerecord/lib/active_record/connection_adapters/abstract/schema_statements.rb, line 156 156: def create_table(table_name, options = {}) 157: td = table_definition 158: td.primary_key(options[:primary_key] || Base.get_primary_key(table_name.to_s.singularize)) unless options[:id] == false 159: 160: yield td if block_given? 161: 162: if options[:force] && table_exists?(table_name) 163: drop_table(table_name, options) 164: end 165: 166: create_sql = "CREATE#{' TEMPORARY' if options[:temporary]} TABLE " 167: create_sql << "#{quote_table_name(table_name)} (" 168: create_sql << td.to_sql 169: create_sql << ") #{options[:options]}" 170: execute create_sql 171: end
SELECT DISTINCT clause for a given set of columns and a given ORDER BY clause. Both PostgreSQL and Oracle overrides this for custom DISTINCT syntax.
distinct("posts.id", "posts.created_at desc")
Drops a table from the database.
Checks to see if an index exists on a table for a given index definition.
Examples
# Check an index exists index_exists?(:suppliers, :company_id) # Check an index on multiple columns exists index_exists?(:suppliers, [:company_id, :company_type]) # Check a unique index exists index_exists?(:suppliers, :company_id, :unique => true) # Check an index with a custom name exists index_exists?(:suppliers, :company_id, :name => "idx_company_id"
# File activerecord/lib/active_record/connection_adapters/abstract/schema_statements.rb, line 44 44: def index_exists?(table_name, column_name, options = {}) 45: column_names = Array.wrap(column_name) 46: index_name = options.key?(:name) ? options[:name].to_s : index_name(table_name, :column => column_names) 47: if options[:unique] 48: indexes(table_name).any?{ |i| i.unique && i.name == index_name } 49: else 50: indexes(table_name).any?{ |i| i.name == index_name } 51: end 52: end
Verify the existence of an index with a given name.
The default argument is returned if the underlying implementation does not define the indexes method, as there’s no way to determine the correct answer in that case.
# File activerecord/lib/active_record/connection_adapters/abstract/schema_statements.rb, line 403 403: def index_name_exists?(table_name, index_name, default) 404: return default unless respond_to?(:indexes) 405: index_name = index_name.to_s 406: indexes(table_name).detect { |i| i.name == index_name } 407: end
Should not be called normally, but this operation is non-destructive. The migrations module handles this automatically.
# File activerecord/lib/active_record/connection_adapters/abstract/schema_statements.rb, line 422 422: def initialize_schema_migrations_table 423: sm_table = ActiveRecord::Migrator.schema_migrations_table_name 424: 425: unless table_exists?(sm_table) 426: create_table(sm_table, :id => false) do |schema_migrations_table| 427: schema_migrations_table.column :version, :string, :null => false 428: end 429: add_index sm_table, :version, :unique => true, 430: :name => "#{Base.table_name_prefix}unique_schema_migrations#{Base.table_name_suffix}" 431: 432: # Backwards-compatibility: if we find schema_info, assume we've 433: # migrated up to that point: 434: si_table = Base.table_name_prefix + 'schema_info' + Base.table_name_suffix 435: 436: if table_exists?(si_table) 437: ActiveSupport::Deprecation.warn "Usage of the schema table `#{si_table}` is deprecated. Please switch to using `schema_migrations` table" 438: 439: old_version = select_value("SELECT version FROM #{quote_table_name(si_table)}").to_i 440: assume_migrated_upto_version(old_version) 441: drop_table(si_table) 442: end 443: end 444: end
Returns a Hash of mappings from the abstract data types to the native database types. See TableDefinition#column for details on the recognized abstract data types.
Removes the column(s) from the table definition.
Examples
remove_column(:suppliers, :qualification) remove_columns(:suppliers, :qualification, :experience)
# File activerecord/lib/active_record/connection_adapters/abstract/schema_statements.rb, line 271 271: def remove_column(table_name, *column_names) 272: if column_names.flatten! 273: message = 'Passing array to remove_columns is deprecated, please use ' + 274: 'multiple arguments, like: `remove_columns(:posts, :foo, :bar)`' 275: ActiveSupport::Deprecation.warn message, caller 276: end 277: 278: columns_for_remove(table_name, *column_names).each do |column_name| 279: execute "ALTER TABLE #{quote_table_name(table_name)} DROP #{column_name}" 280: end 281: end
Alias for remove_column
Remove the given index from the table.
Remove the index_accounts_on_column in the accounts table.
remove_index :accounts, :column
Remove the index named index_accounts_on_branch_id in the accounts table.
remove_index :accounts, :column => :branch_id
Remove the index named index_accounts_on_branch_id_and_party_id in the accounts table.
remove_index :accounts, :column => [:branch_id, :party_id]
Remove the index named by_branch_party in the accounts table.
remove_index :accounts, :name => :by_branch_party
Removes the timestamp columns (created_at and updated_at) from the table definition.
Examples
remove_timestamps(:suppliers)
Renames a column.
Example
rename_column(:suppliers, :description, :name)
Rename an index.
Rename the index_people_on_last_name index to index_users_on_last_name
rename_index :people, 'index_people_on_last_name', 'index_users_on_last_name'
# File activerecord/lib/active_record/connection_adapters/abstract/schema_statements.rb, line 377 377: def rename_index(table_name, old_name, new_name) 378: # this is a naive implementation; some DBs may support this more efficiently (Postgres, for instance) 379: old_index_def = indexes(table_name).detect { |i| i.name == old_name } 380: return unless old_index_def 381: remove_index(table_name, :name => old_name) 382: add_index(table_name, old_index_def.columns, :name => new_name, :unique => old_index_def.unique) 383: end
Renames a table.
Example
rename_table('octopuses', 'octopi')
Returns a string of CREATE TABLE SQL statement(s) for recreating the entire structure of the database.
Truncates a table alias according to the limits of the current adapter.
Checks to see if the table table_name exists on the database.
Example
table_exists?(:developers)
# File activerecord/lib/active_record/connection_adapters/abstract/schema_statements.rb, line 561 561: def add_index_options(table_name, column_name, options = {}) 562: column_names = Array.wrap(column_name) 563: index_name = index_name(table_name, :column => column_names) 564: 565: if Hash === options # legacy support, since this param was a string 566: index_type = options[:unique] ? "UNIQUE" : "" 567: index_name = options[:name].to_s if options.key?(:name) 568: else 569: index_type = options 570: end 571: 572: if index_name.length > index_name_length 573: raise ArgumentError, "Index name '#{index_name}' on table '#{table_name}' is too long; the limit is #{index_name_length} characters" 574: end 575: if index_name_exists?(table_name, index_name, false) 576: raise ArgumentError, "Index name '#{index_name}' on table '#{table_name}' already exists" 577: end 578: index_columns = quoted_columns_for_index(column_names, options).join(", ") 579: 580: [index_name, index_type, index_columns] 581: end
# File activerecord/lib/active_record/connection_adapters/abstract/schema_statements.rb, line 532 532: def add_index_sort_order(option_strings, column_names, options = {}) 533: if options.is_a?(Hash) && order = options[:order] 534: case order 535: when Hash 536: column_names.each {|name| option_strings[name] += " #{order[name].to_s.upcase}" if order.has_key?(name)} 537: when String 538: column_names.each {|name| option_strings[name] += " #{order.upcase}"} 539: end 540: end 541: 542: return option_strings 543: end
# File activerecord/lib/active_record/connection_adapters/abstract/schema_statements.rb, line 593 593: def columns_for_remove(table_name, *column_names) 594: column_names = column_names.flatten 595: 596: raise ArgumentError.new("You must specify at least one column name. Example: remove_column(:people, :first_name)") if column_names.blank? 597: column_names.map {|column_name| quote_column_name(column_name) } 598: end
# File activerecord/lib/active_record/connection_adapters/abstract/schema_statements.rb, line 583 583: def index_name_for_remove(table_name, options = {}) 584: index_name = index_name(table_name, options) 585: 586: unless index_name_exists?(table_name, index_name, true) 587: raise ArgumentError, "Index name '#{index_name}' on table '#{table_name}' does not exist" 588: end 589: 590: index_name 591: end
Overridden by the mysql adapter for supporting index lengths
# File activerecord/lib/active_record/connection_adapters/abstract/schema_statements.rb, line 546 546: def quoted_columns_for_index(column_names, options = {}) 547: option_strings = Hash[column_names.map {|name| [name, '']}] 548: 549: # add index sort order if supported 550: if supports_index_sort_order? 551: option_strings = add_index_sort_order(option_strings, column_names, options) 552: end 553: 554: column_names.map {|name| quote_column_name(name) + option_strings[name]} 555: end