primary_key.rb 4.2 KB
Newer Older
1
require "set"
2

3 4 5 6 7
module ActiveRecord
  module AttributeMethods
    module PrimaryKey
      extend ActiveSupport::Concern

8
      # Returns this record's primary key value wrapped in an array if one is
9
      # available.
10
      def to_key
11
        sync_with_transaction_state
12
        key = id
13
        [key] if key
14 15
      end

16
      # Returns the primary key value.
17
      def id
18 19
        if pk = self.class.primary_key
          sync_with_transaction_state
20
          _read_attribute(pk)
21
        end
22 23
      end

24
      # Sets the primary key value.
25
      def id=(value)
26
        sync_with_transaction_state
27
        write_attribute(self.class.primary_key, value) if self.class.primary_key
28 29
      end

30
      # Queries the primary key value.
31
      def id?
32
        sync_with_transaction_state
33 34 35
        query_attribute(self.class.primary_key)
      end

36
      # Returns the primary key value before type cast.
37
      def id_before_type_cast
38
        sync_with_transaction_state
39 40 41
        read_attribute_before_type_cast(self.class.primary_key)
      end

42 43 44 45 46 47
      # Returns the primary key previous value.
      def id_was
        sync_with_transaction_state
        attribute_was(self.class.primary_key)
      end

48 49 50 51 52
      def id_in_database
        sync_with_transaction_state
        attribute_in_database(self.class.primary_key)
      end

53
      private
54

55
        def attribute_method?(attr_name)
56 57
          attr_name == "id" || super
        end
58

59
        module ClassMethods
60
          ID_ATTRIBUTE_METHODS = %w(id id= id? id_before_type_cast id_was id_in_database).to_set
61

62 63
          def instance_method_already_implemented?(method_name)
            super || primary_key && ID_ATTRIBUTE_METHODS.include?(method_name)
64 65
          end

66 67 68
          def dangerous_attribute_method?(method_name)
            super && !ID_ATTRIBUTE_METHODS.include?(method_name)
          end
69

70 71 72 73 74 75 76
          # Defines the primary key field -- can be overridden in subclasses.
          # Overwriting will negate any effect of the +primary_key_prefix_type+
          # setting, though.
          def primary_key
            @primary_key = reset_primary_key unless defined? @primary_key
            @primary_key
          end
77

78 79 80 81 82
          # Returns a quoted version of the primary key name, used to construct
          # SQL statements.
          def quoted_primary_key
            @quoted_primary_key ||= connection.quote_column_name(primary_key)
          end
83

84 85 86 87 88 89
          def reset_primary_key #:nodoc:
            if self == base_class
              self.primary_key = get_primary_key(base_class.name)
            else
              self.primary_key = base_class.primary_key
            end
90
          end
91

92 93 94 95 96
          def get_primary_key(base_name) #:nodoc:
            if base_name && primary_key_prefix_type == :table_name
              base_name.foreign_key(false)
            elsif base_name && primary_key_prefix_type == :table_name_with_underscore
              base_name.foreign_key
97
            else
98 99 100 101 102 103
              if ActiveRecord::Base != self && table_exists?
                pk = connection.schema_cache.primary_keys(table_name)
                suppress_composite_primary_key(pk)
              else
                "id"
              end
104
            end
105 106
          end

107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126
          # Sets the name of the primary key column.
          #
          #   class Project < ActiveRecord::Base
          #     self.primary_key = 'sysid'
          #   end
          #
          # You can also define the #primary_key method yourself:
          #
          #   class Project < ActiveRecord::Base
          #     def self.primary_key
          #       'foo_' + super
          #     end
          #   end
          #
          #   Project.primary_key # => "foo_id"
          def primary_key=(value)
            @primary_key        = value && value.to_s
            @quoted_primary_key = nil
            @attributes_builder = nil
          end
127

128
          private
129

130 131
            def suppress_composite_primary_key(pk)
              return pk unless pk.is_a?(Array)
132

133
              warn <<-WARNING.strip_heredoc
134
                WARNING: Active Record does not support composite primary key.
135

136 137
                #{table_name} has composite primary key. Composite primary key is ignored.
              WARNING
138
            end
139
        end
140 141 142
    end
  end
end