require 'spec_helper' describe Ripple::Associations do require 'support/models/invoice' require 'support/models/customer' require 'support/models/note' it "should provide access to the associations hash" do Invoice.should respond_to(:associations) Invoice.associations.should be_kind_of(Hash) end it "should collect the embedded associations" do Invoice.embedded_associations.should == Array(Invoice.associations[:note]) end it "should collect the linked associations" do Invoice.linked_associations.should == Array(Invoice.associations[:customer]) end it "should collect the stored_key associations" do Account.stored_key_associations.should == Array(Account.associations[:transactions]) Transaction.stored_key_associations.should == Array(Transaction.associations[:account]) end it "should copy associations to a subclass" do Invoice.associations[:foo] = "bar" class SubInvoice < Invoice; end SubInvoice.associations[:foo].should == "bar" end describe "when adding a :many association" do it "should add accessor and mutator methods" do Invoice.many :items Invoice.instance_methods.map(&:to_sym).should include(:items) Invoice.instance_methods.map(&:to_sym).should include(:items=) end end describe "when adding a :one association" do it "should add accessor, mutator, and query methods" do Invoice.one :payee Invoice.instance_methods.map(&:to_sym).should include(:payee) Invoice.instance_methods.map(&:to_sym).should include(:payee=) Invoice.instance_methods.map(&:to_sym).should include(:payee?) end end before(:all) do @orig_invoice = Invoice Object.send(:remove_const, :Invoice) class Invoice < @orig_invoice; end end after(:all) do Object.send(:remove_const, :SubInvoice) if defined?(SubInvoice) Object.send(:remove_const, :Invoice) Object.send(:const_set, :Invoice, @orig_invoice) end end describe Ripple::Association do it "should initialize with a type and name" do lambda { Ripple::Association.new(:many, :pages) }.should_not raise_error end describe "determining the class name" do it "should default to the camelized class name on :one relationships" do @association = Ripple::Association.new(:one, :page) @association.class_name.should == "Page" end it "should default to the singularized camelized class name on :many relationships" do @association = Ripple::Association.new(:many, :pages) @association.class_name.should == "Page" end it "should use the :class_name option when given" do @association = Ripple::Association.new(:many, :pages, :class_name => "Note") @association.class_name.should == "Note" end end describe "determining the target class" do require 'support/models/tree' it "should default to the constantized class name" do @association = Ripple::Association.new(:one, :t, :class_name => "Trunk") @association.klass.should == Trunk end it "should be determined by the derived class name" do @association = Ripple::Association.new(:many, :branches) @association.klass.should == Branch end it "should use the :class option when given" do @association = Ripple::Association.new(:many, :pages, :class => Leaf) @association.klass.should == Leaf end end it "should be many when type is :many" do Ripple::Association.new(:many, :pages).should be_many end it "should be one when type is :one" do Ripple::Association.new(:one, :pages).should be_one end it "should determine an instance variable based on the name" do Ripple::Association.new(:many, :pages).ivar.should == "@_pages" end describe "key correspondence" do require 'support/models/profile' require 'support/models/user' it "should raise an exception if trying to use a many association when :using => :key" do expect { Profile.many :user, :using => :key }.to raise_error(ArgumentError, "You cannot use a many association using :key") end it "should add the key_delegate attr_accessor to the model declaring the association" do Profile.one :user, :using => :key Profile.new.should respond_to(:key_delegate) Profile.new.should respond_to(:key_delegate=) end end end