Active record associations

What’s wrong with ruby associations?

Somehow association fields don’t have any support for dirty objects
nor optimistic locking.

I would expect dirty objects to work like this:

class Person
has_many :cars
end

person = Person.find_by_name(“Jack”)
person.cars

=> [#<Car id: 2, name: “BMW”, created_at: “2008-11-04 03:04:46”,
updated_at: “2008-11-04 03:04:46”>]

person.cars=[Car.find_by_name(“Porsche”), Car.find_by_name(“VW”)]
person.cars

=> [#<Car id: 1, name: “Porsche”, created_at: “2008-11-04 03:04:46”,
updated_at: “2008-11-04 03:04:46”>, #<Car id: 3, name: “VW”,
created_at: “2008-11-04 03:04:46”, updated_at: “2008-11-04 03:04:46”>]

person.changed?
=> true

person.changes

=> {“cars”=>[[#<Car id: 2, name: “BMW”, created_at: “2008-11-04
03:04:46”, updated_at: “2008-11-04 03:04:46”>], [#<Car id: 1, name:
“Porsche”, created_at: “2008-11-04 03:04:46”, updated_at: “2008-11-04
03:04:46”>, #<Car id: 3, name: “VW”, created_at: “2008-11-04
03:04:46”, updated_at: “2008-11-04 03:04:46”>]]}

person.save #associations are saved in the database

However ror returns the following

person = Person.find_by_name(“Jack”)
person.cars

=> [#<Car id: 2, name: “BMW”, created_at: “2008-11-04 03:04:46”,
updated_at: “2008-11-04 03:04:46”>]

person.cars=[Car.find_by_name(“Porsche”), Car.find_by_name(“VW”)] #associations are saved in the database
person.cars

=> [#<Car id: 1, name: “Porsche”, created_at: “2008-11-04 03:04:46”,
updated_at: “2008-11-04 03:04:46”>, #<Car id: 3, name: “VW”,
created_at: “2008-11-04 03:04:46”, updated_at: “2008-11-04 03:04:46”>]

person.changed?
=> false

person.changes

=> {}

The same problem happens with optimistic locking

p1 = Person.find_by_name(“Jack”)
p2 = Person.find_by_name(“Jack”)

p1.cars = [Car.find_by_name(“Porsche”)]
p1.save

p2.cars = [Car.find_by_name(“BMW”)]
p2.save

I would expect the object to raise an exception, but it doesn’t.

Isn’t there a way to unify the behaviour of regular fields and
associations? Most objects in my app don’t represent flat tables, but
tables that associate each other. Right now I’m forced to having to
wrap many attribute updates into a transaction and write my own object
comparisons.