require 'rspec' $:.unshift File.expand_path('../../', __FILE__) require 'logger' require 'active_mocker/logger' require 'string_reader' require 'active_mocker/table' require 'active_mocker/config' require 'active_mocker/reparameterize' require 'active_mocker/field' require 'active_mocker/active_record' require 'active_mocker/model_reader' require 'active_mocker/schema_reader' require 'active_mocker/active_record/schema' require 'active_mocker/base' require 'active_support/all' require 'active_hash/ar_api' describe ActiveMocker::Base do before(:each) do ActiveMocker::Base.configure do |config| # Required Options config.schema_file = 'file is being inject as string' config.model_dir = 'file is being inject as string' # Dependency injection config.schema_file_reader = schema_file config.model_file_reader = model_file # Additional Options #config.active_hash_as_base = false #default #config.schema_attributes = true #default #config.model_relationships = true #default #config.model_methods = true #default #config.mass_assignment = true #default # Logging config.log_level = Logger::WARN #default end end let(:mock_class){ described_class.mock('Person') } after(:each) do ActiveMocker::Base.reload_default end let(:model_file){ StringReader.new <<-eos class Person < ActiveRecord::Base end eos } let(:schema_file){ StringReader.new <<-eos ActiveRecord::Schema.define(version: 20140327205359) do create_table "people", force: true do |t| t.integer "account_id" t.string "first_name", limit: 128 t.string "last_name", limit: 128 t.string "address", limit: 200 t.string "city", limit: 100 end end eos } describe '::column_names' do it 'returns an array of column names found from the schema.rb file' do expect(mock_class.column_names).to eq(["account_id", "first_name", "last_name", "address", "city"]) end end describe 'mass_assignment' do it "can pass any or all attributes from schema in initializer" do result = mock_class.new(first_name: "Sam", last_name: 'Walton') expect(result.first_name).to eq 'Sam' expect(result.last_name).to eq 'Walton' end context 'set to false' do it 'will fail' do described_class.mass_assignment = false person = described_class.mock("Person") expect{ person.new(first_name: "Sam", last_name: 'Walton') }.to raise_error ArgumentError end end end describe '#mock_class' do it 'create a mock object after the active record' do expect(mock_class).to eq(PersonMock) end context 'private methods' do let(:model_file){ StringReader.new <<-eos class Person < ActiveRecord::Base private def bar end end eos } it 'will not have private methods' do expect{mock_class.bar}.to raise_error(NoMethodError) end end describe '#mock_of' do it 'return the name of the class that is being mocked' do expect(mock_class.new.mock_of).to eq 'Person' end end describe 'relationships' do let(:model_file){ StringReader.new <<-eos class Person < ActiveRecord::Base belongs_to :account end eos } it 'add instance methods from model relationships' do result = mock_class.new(account: 'Account') expect(result.account).to eq 'Account' end end describe 'conflict of instance mocks and class mocks' do let(:model_file){ StringReader.new <<-eos class Person < ActiveRecord::Base def bar(name, type=nil) end def self.bar end end eos } it 'can mock instance method and class method of the same name' do result = mock_class.new result.mock_instance_method(:bar) do |name, type=nil| "Now implemented with #{name} and #{type}" end result = result.bar('foo', 'type') expect(result).to eq "Now implemented with foo and type" expect{mock_class.bar}.to raise_error('::bar is not Implemented for Class: PersonMock') mock_class.mock_class_method(:bar) do "Now implemented" end expect{mock_class.bar}.to_not raise_error expect(mock_class.bar).to eq("Now implemented") end end describe 'instance methods' do let(:model_file){ StringReader.new <<-eos class Person < ActiveRecord::Base def bar(name, type=nil) end end eos } it 'will raise exception for unimplemented methods' do expect(mock_class.new.method(:bar).parameters).to eq [[:req, :name], [:opt, :type]] expect{mock_class.new.bar}.to raise_error ArgumentError expect{mock_class.new.bar('foo', 'type')}.to raise_error('#bar is not Implemented for Class: PersonMock') end it 'can be implemented dynamically' do mock_class.mock_instance_method(:bar) do |name, type=nil| "Now implemented with #{name} and #{type}" end result = mock_class.new result = result.bar('foo', 'type') expect(result).to eq "Now implemented with foo and type" end end describe 'class methods' do let(:model_file){ StringReader.new <<-eos class Person < ActiveRecord::Base scope :named, -> { } def self.class_method end end eos } it 'will raise exception for unimplemented methods' do expect{mock_class.class_method}.to raise_error('::class_method is not Implemented for Class: PersonMock') end it 'can be implemented as follows' do mock_class.mock_class_method(:class_method) do "Now implemented" end expect{mock_class.class_method}.to_not raise_error expect(mock_class.class_method).to eq("Now implemented") end it 'loads named scopes as class method' do expect{mock_class.named}.to raise_error('::named is not Implemented for Class: PersonMock') end end end context 'option active_hash_as_base' do describe 'true' do before(:each) do ActiveMocker::Base.configure do |config| config.active_hash_as_base = true end end require 'active_hash' let(:model_file){ StringReader.new <<-eos class Person < ActiveRecord::Base belongs_to :account def bar end end eos } it 'uses active_hash::base as superclass' do expect(mock_class.superclass.name).to eq 'ActiveHash::Base' end it 'can mass assign attributes to constructor' do result = mock_class.new(first_name: "Sam", last_name: 'Walton', account: 0) expect(result.first_name).to eq 'Sam' expect(result.last_name).to eq 'Walton' expect(result.account).to eq 0 end it 'can save to class and then find instance by attribute' do record = mock_class.create(first_name: "Sam", last_name: 'Walton') expect(mock_class.find_by_first_name("Sam")).to eq record end it '::column_names' do expect(mock_class.column_names).to eq(["account_id", "first_name", "last_name", "address", "city"]) end it '#mock_of' do expect(mock_class.new.mock_of).to eq 'Person' end it 'instance methods from model' do expect{mock_class.new.bar}.to raise_error '#bar is not Implemented for Class: PersonMock' end end describe 'option active_hash_ext' do before(:each) do ActiveMocker::Base.configure do |config| config.active_hash_as_base = true end end require 'active_hash' let(:model_file){ StringReader.new <<-eos class Person < ActiveRecord::Base belongs_to :account def bar end end eos } it '#update' do person = mock_class.create(first_name: 'Justin') person.update(first_name: 'Dustin') expect(person.first_name).to eq 'Dustin' end it '::destroy_all' do mock_class.create expect(mock_class.count).to eq 1 mock_class.destroy_all expect(mock_class.count).to eq 0 end it '::find_by' do person = mock_class.create(first_name: 'dustin') expect(mock_class.find_by(first_name: 'dustin')).to eq person end end describe 'false' do let(:base_options){{active_hash_as_base: false}} it 'has object as supper class' do expect(mock_class.superclass.name).to eq 'Object' end end end describe '::configure' do it 'requires schema_file' do ActiveMocker::Base.reload_default expect{ ActiveMocker::Base.configure { } }.to raise_error end it 'requires model_dir' do ActiveMocker::Base.reload_default expect{ ActiveMocker::Base.configure { |c| c.schema_file = 'dir' } }.to raise_error end end end