README.rdoc 10.5 KB
Newer Older
D
Initial  
David Heinemeier Hansson 已提交
1 2 3
= Active Record -- Object-relation mapping put on rails

Active Record connects business objects and database tables to create a persistable
4
domain model where logic and data are presented in one wrapping. It's an implementation 
D
Initial  
David Heinemeier Hansson 已提交
5 6 7 8 9 10
of the object-relational mapping (ORM) pattern[http://www.martinfowler.com/eaaCatalog/activeRecord.html] 
by the same name as described by Martin Fowler:

  "An object that wraps a row in a database table or view, encapsulates 
       the database access, and adds domain logic on that data."

11
Active Record's main contribution to the pattern is to relieve the original of two stunting problems:
D
Initial  
David Heinemeier Hansson 已提交
12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31
lack of associations and inheritance. By adding a simple domain language-like set of macros to describe
the former and integrating the Single Table Inheritance pattern for the latter, Active Record narrows the
gap of functionality between the data mapper and active record approach.

A short rundown of the major features:

* Automated mapping between classes and tables, attributes and columns.

   class Product < ActiveRecord::Base; end
   
   ...is automatically mapped to the table named "products", such as:
   
   CREATE TABLE products (
     id int(11) NOT NULL auto_increment,
     name varchar(255),
     PRIMARY KEY  (id)
   );

   ...which again gives Product#name and Product#name=(new_name) 
   
32
  {Learn more}[link:classes/ActiveRecord/Base.html]
D
Initial  
David Heinemeier Hansson 已提交
33 34 35 36 37 38 39 40 41 42


* Associations between objects controlled by simple meta-programming macros. 

   class Firm < ActiveRecord::Base
     has_many   :clients
     has_one    :account
     belongs_to :conglomorate
   end

43
  {Learn more}[link:classes/ActiveRecord/Associations/ClassMethods.html]
D
Initial  
David Heinemeier Hansson 已提交
44 45 46 47 48 49 50 51 52 53 54


* Aggregations of value objects controlled by simple meta-programming macros. 

   class Account < ActiveRecord::Base
     composed_of :balance, :class_name => "Money",
                 :mapping => %w(balance amount)
     composed_of :address, 
                 :mapping => [%w(address_street street), %w(address_city city)]
   end

55
  {Learn more}[link:classes/ActiveRecord/Aggregations/ClassMethods.html]
D
Initial  
David Heinemeier Hansson 已提交
56 57 58 59


* Validation rules that can differ for new or existing objects.

60 61 62 63 64 65
    class Account < ActiveRecord::Base
      validates_presence_of     :subdomain, :name, :email_address, :password
      validates_uniqueness_of   :subdomain
      validates_acceptance_of   :terms_of_service, :on => :create
      validates_confirmation_of :password, :email_address, :on => :create
    end
D
Initial  
David Heinemeier Hansson 已提交
66

67
  {Learn more}[link:classes/ActiveRecord/Validations.html]
D
Initial  
David Heinemeier Hansson 已提交
68 69 70 71 72 73 74 75 76 77 78 79 80
 
* Callbacks as methods or queues on the entire lifecycle (instantiation, saving, destroying, validating, etc).

   class Person < ActiveRecord::Base
     def before_destroy # is called just before Person#destroy
       CreditCard.find(credit_card_id).destroy
     end
   end

   class Account < ActiveRecord::Base
     after_find :eager_load, 'self.class.announce(#{id})'
   end

81
  {Learn more}[link:classes/ActiveRecord/Callbacks.html]
D
Initial  
David Heinemeier Hansson 已提交
82 83 84 85 86 87


* Observers for the entire lifecycle

   class CommentObserver < ActiveRecord::Observer
     def after_create(comment) # is called just after Comment#save
88
       Notifications.deliver_new_comment("david@loudthinking.com", comment)
D
Initial  
David Heinemeier Hansson 已提交
89 90 91
     end
   end

92
  {Learn more}[link:classes/ActiveRecord/Observer.html]
D
Initial  
David Heinemeier Hansson 已提交
93 94 95 96 97 98 99 100 101


* Inheritance hierarchies 

   class Company < ActiveRecord::Base; end
   class Firm < Company; end
   class Client < Company; end
   class PriorityClient < Client; end

102
  {Learn more}[link:classes/ActiveRecord/Base.html]
D
Initial  
David Heinemeier Hansson 已提交
103 104


P
Pratik Naik 已提交
105
* Transactions
D
Initial  
David Heinemeier Hansson 已提交
106

P
Pratik Naik 已提交
107
    # Database transaction
D
Initial  
David Heinemeier Hansson 已提交
108 109 110 111 112
    Account.transaction do
      david.withdrawal(100)
      mary.deposit(100)
    end

113
  {Learn more}[link:classes/ActiveRecord/Transactions/ClassMethods.html]
D
Initial  
David Heinemeier Hansson 已提交
114 115 116 117 118 119 120 121


* Reflections on columns, associations, and aggregations

    reflection = Firm.reflect_on_association(:clients)
    reflection.klass # => Client (class)
    Firm.columns # Returns an array of column descriptors for the firms table

122
  {Learn more}[link:classes/ActiveRecord/Reflection/ClassMethods.html]
D
Initial  
David Heinemeier Hansson 已提交
123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138


* Direct manipulation (instead of service invocation)

  So instead of (Hibernate[http://www.hibernate.org/] example):

     long pkId = 1234;
     DomesticCat pk = (DomesticCat) sess.load( Cat.class, new Long(pkId) );
     // something interesting involving a cat...
     sess.save(cat);
     sess.flush(); // force the SQL INSERT

  Active Record lets you:

     pkId = 1234
     cat = Cat.find(pkId)
139
     # something even more interesting involving the same cat...
D
Initial  
David Heinemeier Hansson 已提交
140 141
     cat.save

142
  {Learn more}[link:classes/ActiveRecord/Base.html]
D
Initial  
David Heinemeier Hansson 已提交
143 144 145 146


* Database abstraction through simple adapters (~100 lines) with a shared connector

147
   ActiveRecord::Base.establish_connection(:adapter => "sqlite", :database => "dbfile")
D
Initial  
David Heinemeier Hansson 已提交
148 149 150 151 152 153 154 155 156

   ActiveRecord::Base.establish_connection(
     :adapter  => "mysql", 
     :host     => "localhost", 
     :username => "me", 
     :password => "secret", 
     :database => "activerecord"
   )

157
  {Learn more}[link:classes/ActiveRecord/Base.html#M000081] and read about the built-in support for
158
  MySQL[link:classes/ActiveRecord/ConnectionAdapters/MysqlAdapter.html], PostgreSQL[link:classes/ActiveRecord/ConnectionAdapters/PostgreSQLAdapter.html], SQLite[link:classes/ActiveRecord/ConnectionAdapters/SQLiteAdapter.html], Oracle[link:classes/ActiveRecord/ConnectionAdapters/OracleAdapter.html], SQLServer[link:classes/ActiveRecord/ConnectionAdapters/SQLServerAdapter.html], and DB2[link:classes/ActiveRecord/ConnectionAdapters/DB2Adapter.html].
D
Initial  
David Heinemeier Hansson 已提交
159 160 161 162 163 164 165 166


* Logging support for Log4r[http://log4r.sourceforge.net] and Logger[http://www.ruby-doc.org/stdlib/libdoc/logger/rdoc]

    ActiveRecord::Base.logger = Logger.new(STDOUT)
    ActiveRecord::Base.logger = Log4r::Logger.new("Application Log")


167 168 169 170 171
* Database agnostic schema management with Migrations

    class AddSystemSettings < ActiveRecord::Migration
      def self.up
        create_table :system_settings do |t|
172 173 174 175
          t.string :name
          t.string :label
          t.text :value
          t.string :type
176 177 178 179 180 181 182 183 184 185 186 187 188
          t.integer  :position
        end

        SystemSetting.create :name => "notice", :label => "Use notice?", :value => 1
      end

      def self.down
        drop_table :system_settings
      end
    end

  {Learn more}[link:classes/ActiveRecord/Migration.html]

D
Initial  
David Heinemeier Hansson 已提交
189 190 191 192
== Simple example (1/2): Defining tables and classes (using MySQL)

Data definitions are specified only in the database. Active Record queries the database for 
the column names (that then serves to determine which attributes are valid) on regular
193
object instantiation through the new constructor and relies on the column names in the rows
D
Initial  
David Heinemeier Hansson 已提交
194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238
with the finders.
 
   # CREATE TABLE companies (
   #   id int(11) unsigned NOT NULL auto_increment,
   #   client_of int(11),
   #   name varchar(255),
   #   type varchar(100),
   #   PRIMARY KEY  (id)
   # )

Active Record automatically links the "Company" object to the "companies" table

   class Company < ActiveRecord::Base
     has_many :people, :class_name => "Person"
   end

   class Firm < Company
     has_many :clients
  
     def people_with_all_clients
      clients.inject([]) { |people, client| people + client.people }
     end
   end

The foreign_key is only necessary because we didn't use "firm_id" in the data definition
 
   class Client < Company
     belongs_to :firm, :foreign_key => "client_of"
   end

   # CREATE TABLE people (
   #   id int(11) unsigned NOT NULL auto_increment,
   #   name text,
   #   company_id text,
   #   PRIMARY KEY  (id)
   # )

Active Record will also automatically link the "Person" object to the "people" table

   class Person < ActiveRecord::Base
     belongs_to :company
   end

== Simple example (2/2): Using the domain

239
Picking a database connection for all the Active Records
D
Initial  
David Heinemeier Hansson 已提交
240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267

   ActiveRecord::Base.establish_connection(
     :adapter  => "mysql", 
     :host     => "localhost", 
     :username => "me", 
     :password => "secret", 
     :database => "activerecord"
   )

Create some fixtures

   firm = Firm.new("name" => "Next Angle")
   # SQL: INSERT INTO companies (name, type) VALUES("Next Angle", "Firm")
   firm.save

   client = Client.new("name" => "37signals", "client_of" => firm.id)
   # SQL: INSERT INTO companies (name, client_of, type) VALUES("37signals", 1, "Firm")
   client.save

Lots of different finders

   # SQL: SELECT * FROM companies WHERE id = 1
   next_angle = Company.find(1)

   # SQL: SELECT * FROM companies WHERE id = 1 AND type = 'Firm'
   next_angle = Firm.find(1)    

   # SQL: SELECT * FROM companies WHERE id = 1 AND name = 'Next Angle'
268
   next_angle = Company.find(:first, :conditions => "name = 'Next Angle'")
D
Initial  
David Heinemeier Hansson 已提交
269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293

   next_angle = Firm.find_by_sql("SELECT * FROM companies WHERE id = 1").first

The supertype, Company, will return subtype instances

   Firm === next_angle

All the dynamic methods added by the has_many macro

  next_angle.clients.empty?  # true
  next_angle.clients.size    # total number of clients
  all_clients = next_angle.clients

Constrained finds makes access security easier when ID comes from a web-app

   # SQL: SELECT * FROM companies WHERE client_of = 1 AND type = 'Client' AND id = 2
   thirty_seven_signals = next_angle.clients.find(2)

Bi-directional associations thanks to the "belongs_to" macro

   thirty_seven_signals.firm.nil? # true


== Philosophy 

294
Active Record attempts to provide a coherent wrapper as a solution for the inconvenience that is 
D
Initial  
David Heinemeier Hansson 已提交
295
object-relational mapping. The prime directive for this mapping has been to minimize
296
the amount of code needed to build a real-world domain model. This is made possible
D
Initial  
David Heinemeier Hansson 已提交
297 298 299 300 301 302 303 304 305 306 307 308 309 310 311
by relying on a number of conventions that make it easy for Active Record to infer
complex relations and structures from a minimal amount of explicit direction.

Convention over Configuration:
* No XML-files!
* Lots of reflection and run-time extension
* Magic is not inherently a bad word 

Admit the Database:
* Lets you drop down to SQL for odd cases and performance
* Doesn't attempt to duplicate or replace data definitions


== Download

J
José Valim 已提交
312
The latest version of Active Record can be installed with Rubygems:
D
Initial  
David Heinemeier Hansson 已提交
313

J
José Valim 已提交
314
* gem install activerecord
D
Initial  
David Heinemeier Hansson 已提交
315 316 317

Documentation can be found at 

J
José Valim 已提交
318
* http://api.rubyonrails.org
D
Initial  
David Heinemeier Hansson 已提交
319 320 321 322


== License

323
Active Record is released under the MIT license.
D
Initial  
David Heinemeier Hansson 已提交
324 325 326 327


== Support

328
The Active Record homepage is http://www.rubyonrails.com. You can find the Active Record
D
Initial  
David Heinemeier Hansson 已提交
329 330 331 332 333 334
RubyForge page at http://rubyforge.org/projects/activerecord. And as Jim from Rake says:

   Feel free to submit commits or feature requests.  If you send a patch,
   remember to update the corresponding unit tests.  If fact, I prefer
   new feature to be submitted in the form of new unit tests.

335 336
For other information, feel free to ask on the rubyonrails-talk 
(http://groups.google.com/group/rubyonrails-talk) mailing list.