From 6298ac705da92cc793cbf40dd91484d34e01d255 Mon Sep 17 00:00:00 2001 From: Anton Date: Sat, 24 Jan 2015 00:20:18 +0300 Subject: Add `foreign_key_exists?` method. --- .../abstract/schema_definitions.rb | 4 ++++ .../abstract/schema_statements.rb | 23 ++++++++++++++++++++++ .../test/cases/migration/foreign_key_test.rb | 21 ++++++++++++++++++++ 3 files changed, 48 insertions(+) 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 7eaa89c9a7..a70d93c28b 100644 --- a/activerecord/lib/active_record/connection_adapters/abstract/schema_definitions.rb +++ b/activerecord/lib/active_record/connection_adapters/abstract/schema_definitions.rb @@ -650,6 +650,10 @@ module ActiveRecord def foreign_key(*args) # :nodoc: @base.add_foreign_key(name, *args) end + + def foreign_key_exists?(*args) # :nodoc: + @base.foreign_key_exists?(name, *args) + end private def native 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 f905669a24..0ca8570d00 100644 --- a/activerecord/lib/active_record/connection_adapters/abstract/schema_statements.rb +++ b/activerecord/lib/active_record/connection_adapters/abstract/schema_statements.rb @@ -671,6 +671,29 @@ module ActiveRecord def foreign_keys(table_name) raise NotImplementedError, "foreign_keys is not implemented" end + + # Checks to see if a foreign key exists on a table for a given foreign key definition. + # + # # Check a foreign key exists + # foreign_key_exists?(:accounts, :branches) + # + # # Check a foreign key on specified column exists + # foreign_key_exists?(:accounts, column: :owner_id) + # + # # Check a foreign key with a custom name exists + # foreign_key_exists?(:accounts, name: "special_fk_name") + # + def foreign_key_exists?(from_table, options_or_to_table = {}) + return unless supports_foreign_keys? + + if options_or_to_table.is_a?(Hash) + options = options_or_to_table + else + options = { column: foreign_key_column_for(options_or_to_table) } + end + + foreign_keys(from_table).any? {|fk| options.keys.all? {|key| fk.options[key].to_s == options[key].to_s } } + end # Adds a new foreign key. +from_table+ is the table with the key column, # +to_table+ contains the referenced primary key. diff --git a/activerecord/test/cases/migration/foreign_key_test.rb b/activerecord/test/cases/migration/foreign_key_test.rb index 78d9dd90a3..8cf2ead661 100644 --- a/activerecord/test/cases/migration/foreign_key_test.rb +++ b/activerecord/test/cases/migration/foreign_key_test.rb @@ -145,6 +145,27 @@ module ActiveRecord fk = foreign_keys.first assert_equal :nullify, fk.on_update end + + def test_foreign_key_exists + @connection.add_foreign_key :astronauts, :rockets + + assert @connection.foreign_key_exists?(:astronauts, :rockets) + assert_not @connection.foreign_key_exists?(:astronauts, :stars) + end + + def test_foreign_key_exists_by_column + @connection.add_foreign_key :astronauts, :rockets, column: "rocket_id" + + assert @connection.foreign_key_exists?(:astronauts, column: "rocket_id") + assert_not @connection.foreign_key_exists?(:astronauts, column: "star_id") + end + + def test_foreign_key_exists_by_name + @connection.add_foreign_key :astronauts, :rockets, column: "rocket_id", name: "fancy_named_fk" + + assert @connection.foreign_key_exists?(:astronauts, name: "fancy_named_fk") + assert_not @connection.foreign_key_exists?(:astronauts, name: "other_fancy_named_fk") + end def test_remove_foreign_key_inferes_column @connection.add_foreign_key :astronauts, :rockets -- cgit v1.2.3 From f0ae503d1931a2f269db3437550be3173ba06bce Mon Sep 17 00:00:00 2001 From: Anton Date: Thu, 29 Jan 2015 23:30:53 +0300 Subject: Add methods to get foreign key matching arguments --- .../abstract/schema_definitions.rb | 8 +++ .../abstract/schema_statements.rb | 68 +++++++++------------- 2 files changed, 36 insertions(+), 40 deletions(-) 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 a70d93c28b..a9f276453e 100644 --- a/activerecord/lib/active_record/connection_adapters/abstract/schema_definitions.rb +++ b/activerecord/lib/active_record/connection_adapters/abstract/schema_definitions.rb @@ -50,6 +50,14 @@ module ActiveRecord options[:primary_key] != default_primary_key end + def defined_for?(options_or_to_table = {}) + if options_or_to_table.is_a?(Hash) + options_or_to_table.all? {|assoc| options[assoc[0]].to_s == assoc[1].to_s } + else + to_table == options_or_to_table.to_s + end + end + private def default_primary_key "id" 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 0ca8570d00..a740c53158 100644 --- a/activerecord/lib/active_record/connection_adapters/abstract/schema_statements.rb +++ b/activerecord/lib/active_record/connection_adapters/abstract/schema_statements.rb @@ -671,29 +671,6 @@ module ActiveRecord def foreign_keys(table_name) raise NotImplementedError, "foreign_keys is not implemented" end - - # Checks to see if a foreign key exists on a table for a given foreign key definition. - # - # # Check a foreign key exists - # foreign_key_exists?(:accounts, :branches) - # - # # Check a foreign key on specified column exists - # foreign_key_exists?(:accounts, column: :owner_id) - # - # # Check a foreign key with a custom name exists - # foreign_key_exists?(:accounts, name: "special_fk_name") - # - def foreign_key_exists?(from_table, options_or_to_table = {}) - return unless supports_foreign_keys? - - if options_or_to_table.is_a?(Hash) - options = options_or_to_table - else - options = { column: foreign_key_column_for(options_or_to_table) } - end - - foreign_keys(from_table).any? {|fk| options.keys.all? {|key| fk.options[key].to_s == options[key].to_s } } - end # Adds a new foreign key. +from_table+ is the table with the key column, # +to_table+ contains the referenced primary key. @@ -771,29 +748,40 @@ module ActiveRecord # def remove_foreign_key(from_table, options_or_to_table = {}) return unless supports_foreign_keys? - - if options_or_to_table.is_a?(Hash) - options = options_or_to_table - else - options = { column: foreign_key_column_for(options_or_to_table) } - end - - fk_name_to_delete = options.fetch(:name) do - fk_to_delete = foreign_keys(from_table).detect {|fk| fk.column == options[:column].to_s } - - if fk_to_delete - fk_to_delete.name - else - raise ArgumentError, "Table '#{from_table}' has no foreign key on column '#{options[:column]}'" - end - end - + + fk_name_to_delete = foreign_key_for!(from_table, options_or_to_table).try(:name) + at = create_alter_table from_table at.drop_foreign_key fk_name_to_delete execute schema_creation.accept(at) end + # Checks to see if a foreign key exists on a table for a given foreign key definition. + # + # # Check a foreign key exists + # foreign_key_exists?(:accounts, :branches) + # + # # Check a foreign key on specified column exists + # foreign_key_exists?(:accounts, column: :owner_id) + # + # # Check a foreign key with a custom name exists + # foreign_key_exists?(:accounts, name: "special_fk_name") + # + def foreign_key_exists?(from_table, options_or_to_table = {}) + foreign_key_for(from_table, options_or_to_table).present? + end + + def foreign_key_for(from_table, options_or_to_table = {}) # :nodoc: + return unless supports_foreign_keys? + foreign_keys(from_table).detect {|fk| fk.defined_for? options_or_to_table } + end + + def foreign_key_for!(from_table, options_or_to_table = {}) # :nodoc: + foreign_key_for(from_table, options_or_to_table) or \ + raise ArgumentError, "Table '#{from_table}' has no foreign key for #{options_or_to_table}" + end + def foreign_key_column_for(table_name) # :nodoc: "#{table_name.to_s.singularize}_id" end -- cgit v1.2.3