require 'spec_helper' @@last_pid = 0 describe ActiveFedora::Base do it_behaves_like "An ActiveModel" describe 'descendants' do it "should record the decendants" do ActiveFedora::Base.descendants.should include(ModsArticle, SpecialThing) end end describe "sharding" do it "should have a shard_index" do ActiveFedora::Base.shard_index(@this_pid).should == 0 end context "When the repository is NOT sharded" do subject {ActiveFedora::Base.connection_for_pid('test:bar')} before(:each) do ActiveFedora.config.stub(:sharded?).and_return(false) ActiveFedora::Base.fedora_connection = {} ActiveFedora.config.stub(:credentials).and_return(:url=>'myfedora') end it { should be_kind_of Rubydora::Repository} it "should be the standard connection" do subject.client.url.should == 'myfedora' end describe "assign_pid" do it "should use fedora to generate pids" do # TODO: This juggling of Fedora credentials & establishing connections should be handled by an establish_fedora_connection method, # possibly wrap it all into a fedora_connection method - MZ 06-05-2012 stubfedora = double("Fedora") stubfedora.should_receive(:connection).and_return(double("Connection", :mint =>"sample:newpid")) # Should use ActiveFedora.config.credentials as a single hash rather than an array of shards ActiveFedora::RubydoraConnection.should_receive(:new).with(ActiveFedora.config.credentials).and_return(stubfedora) ActiveFedora::Base.assign_pid(ActiveFedora::Base.new.inner_object) end end describe "shard_index" do it "should always return zero (the first and only connection)" do ActiveFedora::Base.shard_index('test:bar').should == 0 end end end context "When the repository is sharded" do before :each do ActiveFedora.config.stub(:sharded?).and_return(true) ActiveFedora::Base.fedora_connection = {} ActiveFedora.config.stub(:credentials).and_return([{:url=>'shard1'}, {:url=>'shard2'} ]) end describe "assign_pid" do it "should always use the first shard to generate pids" do stubhard1 = double("Shard") stubhard2 = double("Shard") stubhard1.should_receive(:connection).and_return(double("Connection", :mint =>"sample:newpid")) stubhard2.should_receive(:connection).never ActiveFedora::Base.fedora_connection = {0 => stubhard1, 1 => stubhard2} ActiveFedora::Base.assign_pid(ActiveFedora::Base.new.inner_object) end end describe "shard_index" do it "should use modulo of md5 of the pid to distribute objects across shards" do ActiveFedora::Base.shard_index('test:bar').should == 0 ActiveFedora::Base.shard_index('test:nanana').should == 1 end end describe "the repository" do describe "for test:bar" do subject {ActiveFedora::Base.connection_for_pid('test:bar')} it "should be shard1" do subject.client.url.should == 'shard1' end end describe "for test:baz" do subject {ActiveFedora::Base.connection_for_pid('test:nanana')} it "should be shard1" do subject.client.url.should == 'shard2' end end end end end describe "reindex_everything" do it "should call update_index on every object except for the fedora-system objects" do Rubydora::Repository.any_instance.should_receive(:search). and_yield(double(pid:'XXX')).and_yield(double(pid:'YYY')).and_yield(double(pid:'ZZZ')). and_yield(double(pid:'fedora-system:ServiceDeployment-3.0')). and_yield(double(pid:'fedora-system:ServiceDefinition-3.0')). and_yield(double(pid:'fedora-system:FedoraObject-3.0')) mock_update = double(:mock_obj) mock_update.should_receive(:update_index).exactly(3).times ActiveFedora::Base.should_receive(:find).with('XXX').and_return(mock_update) ActiveFedora::Base.should_receive(:find).with('YYY').and_return(mock_update) ActiveFedora::Base.should_receive(:find).with('ZZZ').and_return(mock_update) ActiveFedora::Base.reindex_everything end it "should accept a query param for the search" do query_string = "pid~*" Rubydora::Repository.any_instance.should_receive(:search).with(query_string). and_yield(double(pid:'XXX')).and_yield(double(pid:'YYY')).and_yield(double(pid:'ZZZ')) mock_update = double(:mock_obj) mock_update.should_receive(:update_index).exactly(3).times ActiveFedora::Base.should_receive(:find).with('XXX').and_return(mock_update) ActiveFedora::Base.should_receive(:find).with('YYY').and_return(mock_update) ActiveFedora::Base.should_receive(:find).with('ZZZ').and_return(mock_update) ActiveFedora::Base.reindex_everything(query_string) end end describe "With a test class" do before :all do class FooHistory < ActiveFedora::Base has_metadata :type=>ActiveFedora::SimpleDatastream, :name=>"someData", :autocreate => true do |m| m.field "fubar", :string m.field "swank", :text end has_metadata :type=>ActiveFedora::SimpleDatastream, :name=>"withText", :autocreate => true do |m| m.field "fubar", :text end has_metadata :type=>ActiveFedora::SimpleDatastream, :name=>"withText2", :label=>"withLabel", :autocreate => true do |m| m.field "fubar", :text end has_attributes :fubar, datastream: 'withText', multiple: true has_attributes :swank, datastream: 'someData', multiple: true end class FooAdaptation < ActiveFedora::Base has_metadata :type=>ActiveFedora::OmDatastream, :name=>'someData' end class FooInherited < FooHistory end end after :all do Object.send(:remove_const, :FooHistory) Object.send(:remove_const, :FooAdaptation) Object.send(:remove_const, :FooInherited) end def increment_pid @@last_pid += 1 end before(:each) do @this_pid = increment_pid.to_s stub_get(@this_pid) Rubydora::Repository.any_instance.stub(:client).and_return(@mock_client) ActiveFedora::Base.stub(:assign_pid).and_return(@this_pid) @test_object = ActiveFedora::Base.new end after(:each) do begin ActiveFedora::SolrService.stub(:instance) #@test_object.delete rescue end end describe '#new' do it "should create an inner object" do # for doing AFObject.new(params[:foo]) when nothing is in params[:foo] Rubydora::DigitalObject.any_instance.should_receive(:save).never result = ActiveFedora::Base.new(nil) result.inner_object.should be_kind_of(ActiveFedora::UnsavedDigitalObject) end it "should not save or get an pid on init" do Rubydora::DigitalObject.any_instance.should_receive(:save).never ActiveFedora::Base.should_receive(:assign_pid).never f = FooHistory.new end it "should be able to create with a custom pid" do f = FooHistory.new(:pid=>'numbnuts:1') f.pid.should == 'numbnuts:1' end end describe ".datastream_class_for_name" do it "should return the specifed class" do FooAdaptation.datastream_class_for_name('someData').should == ActiveFedora::OmDatastream end it "should return the specifed class" do FooAdaptation.datastream_class_for_name('content').should == ActiveFedora::Datastream end end describe ".internal_uri" do it "should return pid as fedors uri" do @test_object.internal_uri.should eql("info:fedora/#{@test_object.pid}") end end ### Methods for ActiveModel::Conversions it "should have to_param once it's saved" do @test_object.to_param.should be_nil @test_object.inner_object.stub(:new? => false, :pid => 'foo:123') @test_object.to_param.should == 'foo:123' end it "should have to_key once it's saved" do @test_object.to_key.should be_nil @test_object.inner_object.stub(new?: false, pid: 'foo:123') @test_object.to_key.should == ['foo:123'] end it "should have to_model when it's saved" do @test_object.to_model.should be @test_object end ### end ActiveModel::Conversions ### Methods for ActiveModel::Naming it "Should know the model_name" do FooHistory.model_name.should == 'FooHistory' FooHistory.model_name.human.should == 'Foo history' end ### End ActiveModel::Naming describe ".datastreams" do let(:test_history) { FooHistory.new } it "should create accessors for datastreams declared with has_metadata" do test_history.withText.should == test_history.datastreams['withText'] end describe "dynamic accessors" do before do test_history.add_datastream(ds) test_history.class.build_datastream_accessor(ds.dsid) end describe "when the datastream is named with dash" do let(:ds) {double('datastream', :dsid=>'eac-cpf')} it "should convert dashes to underscores" do test_history.eac_cpf.should == ds end end describe "when the datastream is named with underscore" do let (:ds) { double('datastream', :dsid=>'foo_bar') } it "should preserve the underscore" do test_history.foo_bar.should == ds end end end end it 'should provide #find' do ActiveFedora::Base.should respond_to(:find) end it "should provide .create_date" do @test_object.should respond_to(:create_date) end it "should provide .modified_date" do @test_object.should respond_to(:modified_date) end it 'should respond to .rels_ext' do @test_object.should respond_to(:rels_ext) end describe '.rels_ext' do it 'should return the RelsExtDatastream object from the datastreams array' do @test_object.stub(:datastreams => {"RELS-EXT" => "foo"}) @test_object.rels_ext.should == "foo" end end it 'should provide #add_relationship' do @test_object.should respond_to(:add_relationship) end describe '#add_relationship' do it 'should call #add_relationship on the rels_ext datastream' do @test_object.add_relationship("predicate", "info:fedora/object") pred = ActiveFedora::Predicates.vocabularies["info:fedora/fedora-system:def/relations-external#"]["predicate"] @test_object.relationships.should have_statement(RDF::Statement.new(RDF::URI.new(@test_object.internal_uri), pred, RDF::URI.new("info:fedora/object"))) end it "should update the RELS-EXT datastream and set the datastream as dirty when relationships are added" do mock_ds = double("Rels-Ext") mock_ds.stub(:content_will_change!) @test_object.datastreams["RELS-EXT"] = mock_ds @test_object.add_relationship(:is_member_of, "info:fedora/demo:5") @test_object.add_relationship(:is_member_of, "info:fedora/demo:10") end it 'should add a relationship to an object only if it does not exist already' do next_pid = increment_pid.to_s ActiveFedora::Base.stub(:assign_pid).and_return(next_pid) stub_get(next_pid) @test_object3 = ActiveFedora::Base.new @test_object.add_relationship(:has_part,@test_object3) @test_object.ids_for_outbound(:has_part).should == [@test_object3.pid] #try adding again and make sure not there twice @test_object.add_relationship(:has_part,@test_object3) @test_object.ids_for_outbound(:has_part).should == [@test_object3.pid] end it 'should add literal relationships if requested' do @test_object.add_relationship(:conforms_to,"AnInterface",true) @test_object.ids_for_outbound(:conforms_to).should == ["AnInterface"] end end it 'should provide #remove_relationship' do @test_object.should respond_to(:remove_relationship) end describe '#remove_relationship' do it 'should remove a relationship from the relationships hash' do @test_object3 = ActiveFedora::Base.new() @test_object3.stub(:pid=>'7') @test_object4 = ActiveFedora::Base.new() @test_object4.stub(:pid=>'8') @test_object.add_relationship(:has_part,@test_object3) @test_object.add_relationship(:has_part,@test_object4) #check both are there @test_object.ids_for_outbound(:has_part).should == [@test_object3.pid,@test_object4.pid] @test_object.remove_relationship(:has_part,@test_object3) #check only one item removed @test_object.ids_for_outbound(:has_part).should == [@test_object4.pid] @test_object.remove_relationship(:has_part,@test_object4) #check last item removed and predicate removed since now emtpy @test_object.relationships.size.should == 0 end end it 'should provide #relationships' do @test_object.should respond_to(:relationships) end describe '#relationships' do it 'should return a graph' do @test_object.relationships.kind_of?(RDF::Graph).should be_true @test_object.relationships.size.should == 0 end end describe '.assert_content_model' do it "should default to the name of the class" do stub_get(@this_pid) stub_add_ds(@this_pid, ['RELS-EXT']) @test_object.assert_content_model @test_object.relationships(:has_model).should == ["info:fedora/afmodel:ActiveFedora_Base"] end end describe '.save' do it "should create a new record" do @test_object.stub(:new_record? => true) @test_object.should_receive(:assign_pid) @test_object.should_receive(:serialize_datastreams) @test_object.inner_object.should_receive(:save) @test_object.should_receive(:update_index) @test_object.save end it "should update an existing record" do @test_object.stub(:new_record? => false) @test_object.should_receive(:serialize_datastreams) @test_object.inner_object.should_receive(:save) @test_object.should_receive(:update_index) @test_object.save end end describe "#create" do it "should build a new record and save it" do obj = double() obj.should_receive(:save) FooHistory.should_receive(:new).and_return(obj) @hist = FooHistory.create(:fubar=>'ta', :swank=>'da') end end describe ".adapt_to" do it "should return an adapted object of the requested type" do @test_object = FooHistory.new() @test_object.adapt_to(FooAdaptation).class.should == FooAdaptation end it "should not make an additional call to fedora to create the adapted object" do @test_object = FooHistory.new() adapted = @test_object.adapt_to(FooAdaptation) end it "should propagate new datastreams to the adapted object" do @test_object = FooHistory.new() @test_object.add_file_datastream("XXX", :dsid=>'MY_DSID') adapted = @test_object.adapt_to(FooAdaptation) adapted.datastreams.keys.should include 'MY_DSID' adapted.datastreams['MY_DSID'].content.should == "XXX" adapted.datastreams['MY_DSID'].changed?.should be_true end it "should propagate modified datastreams to the adapted object" do @test_object = FooHistory.new() orig_ds = @test_object.datastreams['someData'] orig_ds.content="" adapted = @test_object.adapt_to(FooAdaptation) adapted.datastreams.keys.should include 'someData' adapted.datastreams['someData'].should == orig_ds adapted.datastreams['someData'].content.strip.should == "" adapted.datastreams['someData'].changed?.should be_true end it "should use the datastream definitions from the adapted object" do @test_object = FooHistory.new() adapted = @test_object.adapt_to(FooAdaptation) adapted.datastreams.keys.should include 'someData' adapted.datastreams['someData'].class.should == ActiveFedora::OmDatastream end end describe ".adapt_to_cmodel with implemented (non-ActiveFedora::Base) cmodel" do subject { FooHistory.new } it "should not cast to a random first cmodel if it has a specific cmodel already" do ActiveFedora::ContentModel.should_receive(:known_models_for).with(subject).and_return([FooAdaptation]) subject.adapt_to_cmodel.should be_kind_of FooHistory end it "should cast to an inherited model over a random one" do ActiveFedora::ContentModel.should_receive(:known_models_for).with(subject).and_return([FooAdaptation, FooInherited]) subject.adapt_to_cmodel.should be_kind_of FooInherited end it "should not cast when a cmodel is same as the class" do ActiveFedora::ContentModel.should_receive(:known_models_for).with(subject).and_return([FooHistory]) subject.adapt_to_cmodel.should === subject end end describe ".adapt_to_cmodel with ActiveFedora::Base" do subject { ActiveFedora::Base.new } it "should cast to the first cmodel if ActiveFedora::Base (or no specified cmodel)" do ActiveFedora::ContentModel.should_receive(:known_models_for).with(subject).and_return([FooAdaptation, FooHistory]) subject.adapt_to_cmodel.should be_kind_of FooAdaptation end end describe ".to_solr" do it "should provide .to_solr" do @test_object.should respond_to(:to_solr) end it "should add pid, system_create_date, system_modified_date and object_state from object attributes" do @test_object.should_receive(:create_date).and_return("2012-03-04T03:12:02Z") @test_object.should_receive(:modified_date).and_return("2012-03-07T03:12:02Z") @test_object.stub(pid: 'changeme:123') @test_object.state = "D" solr_doc = @test_object.to_solr solr_doc[ActiveFedora::SolrService.solr_name("system_create", :stored_sortable, type: :date)].should eql("2012-03-04T03:12:02Z") solr_doc[ActiveFedora::SolrService.solr_name("system_modified", :stored_sortable, type: :date)].should eql("2012-03-07T03:12:02Z") solr_doc[ActiveFedora::SolrService.solr_name("object_state", :stored_sortable)].should eql("D") solr_doc[:id].should eql("changeme:123") end it "should omit base metadata and RELS-EXT if :model_only==true" do @test_object.add_relationship(:has_part, "foo", true) solr_doc = @test_object.to_solr(Hash.new, :model_only => true) solr_doc[ActiveFedora::SolrService.solr_name("system_create", type: :date)].should be_nil solr_doc[ActiveFedora::SolrService.solr_name("system_modified", type: :date)].should be_nil solr_doc["id"].should be_nil solr_doc[ActiveFedora::SolrService.solr_name("has_part", :symbol)].should be_nil end it "should add self.class as the :active_fedora_model" do stub_get(@this_pid) stub_get_content(@this_pid, ['RELS-EXT', 'someData', 'withText2', 'withText']) @test_history = FooHistory.new() solr_doc = @test_history.to_solr solr_doc[ActiveFedora::SolrService.solr_name("active_fedora_model", :stored_sortable)].should eql("FooHistory") end it "should call .to_solr on all SimpleDatastreams and OmDatastreams, passing the resulting document to solr" do mock1 = double("ds1", :to_solr => {}) mock2 = double("ds2", :to_solr => {}) ngds = double("ngds", :to_solr => {}) ngds.should_receive(:solrize_profile) mock1.should_receive(:solrize_profile) mock2.should_receive(:solrize_profile) @test_object.should_receive(:datastreams).twice.and_return({:ds1 => mock1, :ds2 => mock2, :ngds => ngds}) @test_object.should_receive(:solrize_relationships) @test_object.to_solr end it "should call .to_solr on all RDFDatastreams, passing the resulting document to solr" do mock = double("ds1", :to_solr => {}) mock.should_receive(:solrize_profile) @test_object.should_receive(:datastreams).twice.and_return({:ds1 => mock}) @test_object.should_receive(:solrize_relationships) @test_object.to_solr end it "should call .to_solr on the relationships rels-ext is dirty" do @test_object.add_relationship(:has_collection_member, "info:fedora/test:member") rels_ext = @test_object.rels_ext rels_ext.should be_changed @test_object.should_receive(:solrize_relationships) @test_object.to_solr end end describe ".label" do it "should return the label of the inner object" do @test_object.inner_object.should_receive(:label).and_return("foo label") @test_object.label.should == "foo label" end end describe ".label=" do it "should set the label of the inner object" do @test_object.label.should_not == "foo label" @test_object.label = "foo label" @test_object.label.should == "foo label" end end describe "update_attributes" do it "should set the attributes and save" do m = FooHistory.new att= {"fubar"=> '1234', "baz" =>'stuff'} m.should_receive(:fubar=).with('1234') m.should_receive(:baz=).with('stuff') m.should_receive(:save) m.update_attributes(att) end end describe "update" do it "should set the attributes and save" do m = FooHistory.new att= {"fubar"=> '1234', "baz" =>'stuff'} m.should_receive(:fubar=).with('1234') m.should_receive(:baz=).with('stuff') m.should_receive(:save) m.update(att) end end describe ".solrize_relationships" do it "should serialize the relationships into a Hash" do graph = RDF::Graph.new subject = RDF::URI.new "info:fedora/test:sample_pid" graph.insert RDF::Statement.new(subject, ActiveFedora::Predicates.find_graph_predicate(:is_member_of), RDF::URI.new('info:fedora/demo:10')) graph.insert RDF::Statement.new(subject, ActiveFedora::Predicates.find_graph_predicate(:is_part_of), RDF::URI.new('info:fedora/demo:11')) graph.insert RDF::Statement.new(subject, ActiveFedora::Predicates.find_graph_predicate(:has_part), RDF::URI.new('info:fedora/demo:12')) graph.insert RDF::Statement.new(subject, ActiveFedora::Predicates.find_graph_predicate(:conforms_to), "AnInterface") @test_object.should_receive(:relationships).and_return(graph) solr_doc = @test_object.solrize_relationships solr_doc[ActiveFedora::SolrService.solr_name("is_member_of", :symbol)].should == "info:fedora/demo:10" solr_doc[ActiveFedora::SolrService.solr_name("is_part_of", :symbol)].should == "info:fedora/demo:11" solr_doc[ActiveFedora::SolrService.solr_name("has_part", :symbol)].should == "info:fedora/demo:12" solr_doc[ActiveFedora::SolrService.solr_name("conforms_to", :symbol)].should == "AnInterface" end end end end