require 'spec_helper' class Build include CouchPotato::Persistence property :state property :time view :timeline, :key => :time view :count, :key => :time, :reduce => true view :minimal_timeline, :key => :time, :properties => [:state], :type => :properties view :key_array_timeline, :key => [:time, :state] view :custom_timeline, :map => "function(doc) { emit(doc._id, {state: 'custom_' + doc.state}); }", :type => :custom view :custom_timeline_returns_docs, :map => "function(doc) { emit(doc._id, null); }", :include_docs => true, :type => :custom view :custom_with_reduce, :map => "function(doc) {if(doc.foreign_key) {emit(doc.foreign_key, 1);} else {emit(doc._id, 1)}}", :reduce => "function(key, values) {return({\"count\": sum(values)});}", :group => true, :type => :custom view :custom_count_with_reduce, :map => "function(doc) {if(doc.foreign_key) {emit(doc.foreign_key, 1);} else {emit(doc._id, 1)}}", :reduce => "function(key, values) {return(sum(values));}", :group => true, :type => :custom view :raw, :type => :raw, :map => "function(doc) {emit(doc._id, doc.state)}" view :filtered_raw, :type => :raw, :map => "function(doc) {emit(doc._id, doc.state)}", :results_filter => lambda{|res| res['rows'].map{|row| row['value']}} view :with_view_options, :group => true, :key => :time end class CustomBuild < Build end describe 'view' do before(:each) do recreate_db @db = CouchPotato.database end it "should return instances of the class" do @db.save_document Build.new(:state => 'success', :time => '2008-01-01') results = @db.view(Build.timeline) results.map(&:class).should == [Build] end it "should pass the view options to the view query" do query = mock 'query' CouchPotato::View::ViewQuery.stub!(:new).and_return(query) query.should_receive(:query_view!).with(hash_including(:key => 1)).and_return('rows' => []) @db.view Build.timeline(:key => 1) end it "should not return documents that don't have a matching JSON.create_id" do CouchPotato.couchrest_database.save_doc({:time => 'x'}) @db.view(Build.timeline).should == [] end it "should count documents" do @db.save_document! Build.new(:state => 'success', :time => '2008-01-01') @db.view(Build.count(:reduce => true)).should == 1 end it "should count zero documents" do @db.view(Build.count(:reduce => true)).should == 0 end it "should return the total_rows" do @db.save_document! Build.new(:state => 'success', :time => '2008-01-01') @db.view(Build.count(:reduce => false)).total_rows.should == 1 end describe "with multiple keys" do it "should return the documents with matching keys" do build = Build.new(:state => 'success', :time => '2008-01-01') @db.save! build @db.view(Build.timeline(:keys => ['2008-01-01'])).should == [build] end it "should not return documents with non-matching keys" do build = Build.new(:state => 'success', :time => '2008-01-01') @db.save! build @db.view(Build.timeline(:keys => ['2008-01-02'])).should be_empty end end describe "properties defined" do it "should assign the configured properties" do CouchPotato.couchrest_database.save_doc(:state => 'success', :time => '2008-01-01', JSON.create_id.to_sym => 'Build') @db.view(Build.minimal_timeline).first.state.should == 'success' end it "should not assign the properties not configured" do CouchPotato.couchrest_database.save_doc(:state => 'success', :time => '2008-01-01', JSON.create_id.to_sym => 'Build') @db.view(Build.minimal_timeline).first.time.should be_nil end it "should assign the id even if it is not configured" do id = CouchPotato.couchrest_database.save_doc(:state => 'success', :time => '2008-01-01', JSON.create_id.to_sym => 'Build')['id'] @db.view(Build.minimal_timeline).first._id.should == id end end describe "no properties defined" do it "should assign all properties to the objects by default" do id = CouchPotato.couchrest_database.save_doc({:state => 'success', :time => '2008-01-01', JSON.create_id.to_sym => 'Build'})['id'] result = @db.view(Build.timeline).first result.state.should == 'success' result.time.should == '2008-01-01' result._id.should == id end end describe "map function given" do it "should still return instances of the class" do CouchPotato.couchrest_database.save_doc({:state => 'success', :time => '2008-01-01'}) @db.view(Build.custom_timeline).map(&:class).should == [Build] end it "should assign the properties from the value" do CouchPotato.couchrest_database.save_doc({:state => 'success', :time => '2008-01-01'}) @db.view(Build.custom_timeline).map(&:state).should == ['custom_success'] end it "should assign the id" do doc = CouchPotato.couchrest_database.save_doc({:state => 'success', :time => '2008-01-01'}) @db.view(Build.custom_timeline).map(&:_id).should == [doc['id']] end it "should leave the other properties blank" do CouchPotato.couchrest_database.save_doc({:state => 'success', :time => '2008-01-01'}) @db.view(Build.custom_timeline).map(&:time).should == [nil] end describe "that returns null documents" do it "should return instances of the class" do CouchPotato.couchrest_database.save_doc({:state => 'success', :time => '2008-01-01'}) @db.view(Build.custom_timeline_returns_docs).map(&:class).should == [Build] end it "should assign the properties from the value" do CouchPotato.couchrest_database.save_doc({:state => 'success', :time => '2008-01-01'}) @db.view(Build.custom_timeline_returns_docs).map(&:state).should == ['success'] end it "should still return instance of class if document included JSON.create_id" do CouchPotato.couchrest_database.save_doc({:state => 'success', :time => '2008-01-01', JSON.create_id.to_sym => "Build"}) view_data = @db.view(Build.custom_timeline_returns_docs) view_data.map(&:class).should == [Build] view_data.map(&:state).should == ['success'] end end describe "additional reduce function given" do it "should still assign the id" do doc = CouchPotato.couchrest_database.save_doc({}) CouchPotato.couchrest_database.save_doc({:foreign_key => doc['id']}) @db.view(Build.custom_with_reduce).map(&:_id).should == [doc['id']] end describe "when the additional reduce function is a typical count" do it "should parse the reduce count" do doc = CouchPotato.couchrest_database.save_doc({}) CouchPotato.couchrest_database.save_doc({:foreign_key => doc['id']}) @db.view(Build.custom_count_with_reduce(:reduce => true)).should == 2 end end end end describe "with array as key" do it "should create a map function with the composite key" do CouchPotato::View::ViewQuery.should_receive(:new) do |db, design_name, view, list| view['key_array_timeline'][:map].should match(/emit\(\[doc\['time'\], doc\['state'\]\]/) stub('view query', :query_view! => {'rows' => []}) end @db.view Build.key_array_timeline end end describe "raw view" do it "should return the raw data" do @db.save_document Build.new(:state => 'success', :time => '2008-01-01') @db.view(Build.raw)['rows'][0]['value'].should == 'success' end it "should return filtred raw data" do @db.save_document Build.new(:state => 'success', :time => '2008-01-01') @db.view(Build.filtered_raw).should == ['success'] end it "should pass view options declared in the view declaration to the query" do view_query = mock 'view_query' CouchPotato::View::ViewQuery.stub!(:new).and_return(view_query) view_query.should_receive(:query_view!).with(hash_including(:group => true)).and_return({'rows' => []}) @db.view(Build.with_view_options) end end describe "inherited views" do it "should support parent views for objects of the subclass" do @db.save_document CustomBuild.new(:state => 'success', :time => '2008-01-01') @db.view(CustomBuild.timeline).size.should == 1 @db.view(CustomBuild.timeline).first.should be_kind_of(CustomBuild) end end describe "list functions" do class Coworker include CouchPotato::Persistence property :name view :all_with_list, :key => :name, :list => :append_doe view :all, :key => :name list :append_doe, <<-JS function(head, req) { var row; send('{"rows": ['); while(row = getRow()) { row.doc.name = row.doc.name + ' doe'; send(JSON.stringify(row)); }; send(']}'); } JS end it "should use the list function declared at class level" do @db.save! Coworker.new(:name => 'joe') @db.view(Coworker.all_with_list).first.name.should == 'joe doe' end it "should use the list function passed at runtime" do @db.save! Coworker.new(:name => 'joe') @db.view(Coworker.all(:list => :append_doe)).first.name.should == 'joe doe' end end end