require File.expand_path("../test_helper", File.dirname(__FILE__)) class DomainTest < ActiveSupport::TestCase # Domain =================================================================== test "generate should return domain" do assert_kind_of Domain, Domain.generate end test "name should return rails application name" do begin Object::Quux = Module.new Object::Quux::Application = Class.new Object::Rails = Struct.new(:application).new(Object::Quux::Application.new) assert_equal "Quux", Domain.generate.name ensure Object::Quux.send :remove_const, :Application Object.send :remove_const, :Quux Object.send :remove_const, :Rails end end test "name should return nil outside rails" do assert_nil Domain.generate.name end test "inspect should display object id only" do create_model "Foo", :bar => :references do belongs_to :bar end create_model "Bar" assert_match %r{#}, Domain.generate.inspect end # Entity processing ======================================================== test "entity_by_name should return associated entity for given name" do create_model "Foo" assert_equal Foo, Domain.generate.entity_by_name("Foo").model end test "entities should return domain entities" do create_models "Foo", "Bar" assert_equal [Domain::Entity] * 2, Domain.generate.entities.collect(&:class) end test "entities should return all domain entities sorted by name" do create_models "Foo", "Bar", "Baz", "Qux" assert_equal [Bar, Baz, Foo, Qux], Domain.generate.entities.collect(&:model) end test "entities should include abstract entities" do create_model "Stronghold" do has_many :cannons, :as => :defensible end assert_equal ["Defensible", "Stronghold"], Domain.generate.entities.collect(&:name) end test "entities should include abstract entities only once" do create_model "Stronghold" do has_many :cannons, :as => :defensible end create_model "Galleon" do has_many :cannons, :as => :defensible end assert_equal ["Defensible", "Galleon", "Stronghold"], Domain.generate.entities.collect(&:name) end test "entities should omit abstract models" do Object.const_set :Foo, Class.new(ActiveRecord::Base) { self.abstract_class = true } create_model "Bar", Foo do set_table_name "bars" end create_table "foos", {}, true create_table "bars", {}, true assert_equal ["Bar"], Domain.generate.entities.collect(&:name) end # Relationship processing ================================================== test "relationships should return empty array for empty domain" do assert_equal [], Domain.generate.relationships end test "relationships should return relationships in domain model" do create_models "Baz", "Qux" create_model "Foo", :bar => :references, :qux => :references do belongs_to :bar belongs_to :qux end create_model "Bar", :baz => :references do belongs_to :baz end assert_equal [Domain::Relationship] * 3, Domain.generate.relationships.collect(&:class) end test "relationships should count mutual relationship as one" do create_model "Foo", :bar => :references do belongs_to :bar end create_model "Bar" do has_many :foos end assert_equal [Domain::Relationship], Domain.generate.relationships.collect(&:class) end test "relationships should count mutual indirect relationship as one" do create_model "Wizard" do has_many :spell_masteries has_many :spells, :through => :spell_masteries end create_model "Spell" do has_many :spell_masteries has_many :wizards, :through => :spell_masteries end create_model "SpellMastery", :wizard => :references, :spell => :references do belongs_to :wizard belongs_to :spell end assert_equal [Domain::Relationship], Domain.generate.relationships.select(&:indirect?).collect(&:class) end test "relationships should count relationship between same models with distinct foreign key seperately" do create_model "Foo", :bar => :references, :special_bar => :references do belongs_to :bar end create_model "Bar" do has_many :foos, :foreign_key => :special_bar_id end assert_equal [Domain::Relationship] * 2, Domain.generate.relationships.collect(&:class) end # Specialization processing ================================================ test "specializations should return empty array for empty domain" do assert_equal [], Domain.generate.specializations end test "specializations should return empty array for domain without single table inheritance" do create_simple_domain assert_equal [], Domain.generate.specializations end test "specializations should return specializations in domain model" do create_specialization assert_equal [Domain::Specialization], Domain.generate.specializations.collect(&:class) end test "specializations should return specializations of specializations in domain model" do create_specialization Object.const_set :BelgianBeer, Class.new(Beer) assert_equal [Domain::Specialization] * 2, Domain.generate.specializations.collect(&:class) end test "specializations should return generalizations in domain model" do create_model "Post" do has_many :assets, :as => :attachable end create_model "Asset", :attachable => :references do belongs_to :attachable, :polymorphic => true end assert_equal [Domain::Specialization], Domain.generate.specializations.collect(&:class) end test "specializations should return generalizations and specializations in domain model" do create_model "Content", :type => :string do has_many :assets, :as => :attachable end Object.const_set :Post, Class.new(Content) create_model "Asset", :attachable => :references do belongs_to :attachable, :polymorphic => true end assert_equal [Domain::Specialization] * 2, Domain.generate.specializations.collect(&:class) end # test "generalizations should ..." do # # TODO # create_model "Post" do # has_many :assets, :as => :attachable # end # create_model "Asset", :attachable => :references do # belongs_to :attachable, :polymorphic => true # end # assert_equal [], Domain.generate.relationships # end # Erroneous associations =================================================== test "relationships should omit bad has_many associations" do create_model "Foo" do has_many :flabs end assert_equal [], Domain.generate(:warn => false).relationships end test "relationships should omit bad has_many through association" do create_model "Foo" do has_many :flabs, :through => :bars end assert_equal [], Domain.generate(:warn => false).relationships end test "relationships should omit association to model outside domain" do create_model "Foo" do has_many :bars end create_model "Bar", :foo => :references assert_equal [], Domain.new([Foo], :warn => false).relationships end test "relationships should output a warning when a bad association is encountered" do create_model "Foo" do has_many :flabs end output = collect_stdout do Domain.generate.relationships end assert_match /Ignoring invalid association :flabs on Foo/, output end test "relationships should output a warning when an association to model outside domain is encountered" do create_model "Foo" do has_many :bars end create_model "Bar", :foo => :references output = collect_stdout do Domain.new([Foo]).relationships end assert_match /model Bar exists, but is not included in domain/, output end test "relationships should output a warning when an association to a non existent generalization is encountere" do create_model "Foo" do has_many :bars, :as => :foo end create_model "Bar", :foobar => :references do belongs_to :foo_bar, :polymorphic => true end output = collect_stdout do Domain.generate.relationships end assert_match /polymorphic interface FooBar does not exist/, output end test "relationships should not warn when a bad association is encountered if warnings are disabled" do create_model "Foo" do has_many :flabs end output = collect_stdout do Domain.generate(:warn => false).relationships end assert_equal "", output end # Erroneous models ========================================================= test "entities should omit bad models" do Object.const_set :Foo, Class.new(ActiveRecord::Base) assert_equal [], Domain.generate(:warn => false).entities end test "entities should output a warning when a model table does not exist" do Object.const_set :Foo, Class.new(ActiveRecord::Base) output = collect_stdout do Domain.generate.entities end assert_match /Ignoring invalid model Foo \(table foos does not exist\)/, output end end