# Copyright (c) 2010, Diaspora Inc. This file is # licensed under the Affero General Public License version 3 or later. See # the COPYRIGHT file. require 'spec_helper' describe Diaspora::UserModules::Connecting do let(:user) { alice } let(:aspect) { user.aspects.create(:name => 'heroes') } let(:aspect1) { user.aspects.create(:name => 'other') } let(:person) { Factory.create(:person) } let(:person_one) { Factory.create :person } let(:person_two) { Factory.create :person } let(:person_three) { Factory.create :person } let(:user2) { eve } let(:aspect2) { user2.aspects.create(:name => "aspect two") } describe '#send_contact_request_to' do it 'should not be able to contact request an existing contact' do user.activate_contact(user2.person, aspect1) proc { user.send_contact_request_to(user2.person, aspect1) }.should raise_error(ActiveRecord::RecordInvalid) end it 'should not be able to contact request no-one' do proc { user.send_contact_request_to(nil, aspect) }.should raise_error(ActiveRecord::RecordInvalid) end it 'creates a pending contact' do proc { user.send_contact_request_to(user2.person, aspect1) }.should change(Contact, :count).by(1) user.contact_for(user2.person).pending.should == true user.contact_for(user2.person).should be_pending end it 'persists no request for requester' do proc { user.send_contact_request_to(user2.person, aspect1) }.should_not change{Request.where(:recipient_id => user.person.id).count} end it 'persists a request for the recipient' do user.send_contact_request_to(user2.person, aspect1) user2.request_from(user.person).should_not be_nil end end context 'contact requesting' do describe '#receive_contact_request' do before do @r = Request.diaspora_initialize(:to => user.person, :from => person) end it 'adds a request to pending if it was not sent by user' do user.receive_contact_request(@r) Request.where(:recipient_id => user.person.id).all.should include @r end it 'creates no contact' do lambda { received_req = @r.receive(user, person_one) }.should_not change(Contact, :count) end it 'enqueues a mail job' do Resque.should_receive(:enqueue).with(Jobs::MailRequestReceived, user.id, person.id) zord = Postzord::Receiver.new(user, :object => @r, :person => person) zord.receive_object end end describe '#receive_request_acceptance' do before do @original_request = user.send_contact_request_to(user2.person, aspect) @acceptance = @original_request.reverse_for(user2) end it 'connects to the acceptor' do @acceptance.receive(user, user2.person) user.contact_for(user2.person).should_not be_nil end it 'deletes the acceptance' do @acceptance.receive(user, user2.person) Request.where(:sender_id => user2.person.id, :recipient_id => user.person.id).should be_empty end it 'enqueues a mail job' do Resque.should_receive(:enqueue).with(Jobs::MailRequestAcceptance, user.id, user2.person.id).once zord = Postzord::Receiver.new(user, :object => @acceptance, :person => user2.person) zord.receive_object end end context 'received a contact request' do let(:request_for_user) {Request.diaspora_initialize(:to => user.person, :from => person)} let(:request2_for_user) {Request.diaspora_initialize(:to => user.person, :from => person_one)} let(:request_from_myself) {Request.diaspora_initialize(:to => user.person, :from => user.person)} before do Request.diaspora_initialize(:from => person, :to => user.person).save Request.diaspora_initialize(:from => person_one, :to => user.person).save @received_request = Request.where(:sender_id => person.id, :recipient_id => user.person.id).first @received_request2 = Request.where(:sender_id => person_one.id, :recipient_id => user.person.id).first end it "should delete an accepted contact request" do proc { user.accept_contact_request(@received_request, aspect) }.should change(Request, :count ).by(-1) end it "should mark the corresponding notification as 'read'" do notification = Notification.create(:target_id => @received_request.id, :target_type => 'new_request', :unread => true) Notification.where(:target_id=>@received_request.id).first.unread.should be_true user.accept_contact_request(@received_request, aspect) Notification.where(:target_id=>@received_request.id).first.unread.should be_false end it 'should be able to ignore a pending contact request' do proc { user.ignore_contact_request(@received_request.id) }.should change(Request, :count ).by(-1) end it 'should ignore a contact request from yourself' do reversed_request = request_from_myself.reverse_for(user) user.receive_contact_request(reversed_request) reversed_request.persisted?.should be false end end describe 'multiple users accepting/rejecting the same person' do before do @request1 = Request.diaspora_initialize(:to => user.person, :from => person_one) @request2 = Request.diaspora_initialize(:to => user2.person, :from => person_one) @request3 = Request.diaspora_initialize(:to => user2.person, :from => user.person) @req1_xml = @request1.to_diaspora_xml @req2_xml = @request2.to_diaspora_xml @req3_xml = @request3.to_diaspora_xml @request1.destroy @request2.destroy @request3.destroy end context 'request from one remote person to one local user' do before do zord = Postzord::Receiver.new(user, :person => user.person) @received_request = zord.parse_and_receive(@req3_xml) @received_request.reload end it 'should connect the user other user on the same pod' do proc { user2.accept_contact_request(@received_request, aspect2) }.should_not change(Person, :count) user2.contact_for(user.person).should_not be_nil end it 'should not delete the ignored user on the same pod' do proc { user2.ignore_contact_request(@received_request.id) }.should_not change(Person, :count) user2.contact_for(user.person).should be_nil end end context 'Two users receiving requests from one person' do before do zord1 = Postzord::Receiver.new(user, :person => person_one) zord2 = Postzord::Receiver.new(user, :person => person_one) @req_to_user = zord1.parse_and_receive(@req1_xml) @req_to_user2 = zord2.parse_and_receive(@req2_xml) end describe '#accept_contact_request' do it 'should both users should connect the same person' do user.accept_contact_request @req_to_user, aspect user.contact_for(person_one).should_not be_nil user2.accept_contact_request @req_to_user2, aspect2 user2.contact_for(person_one).should_not be_nil end it 'should keep the person around if one of the users rejects him' do user.accept_contact_request @req_to_user, aspect user.contact_for(person_one).should_not be_nil user2.ignore_contact_request @req_to_user2.id user2.contact_for(person_one).should be_nil end end it 'should keep the person around if the users ignores them' do user.ignore_contact_request Request.where(:recipient_id => user.person.id).first.id user.contact_for(person_one).should be_nil user2.ignore_contact_request Request.where(:recipient_id => user2.person.id).first.id user2.contact_for(person_one).should be_nil end end end describe 'a user accepting rejecting multiple people' do before do request = Request.diaspora_initialize(:to => user.person, :from => person_one) @received_request = request.receive(user, person_one) end describe '#accept_contact_request' do it "deletes the received request" do lambda { user.accept_contact_request(@received_request, aspect) }.should change(Request, :count).by(-1) end it "creates a new contact" do lambda { user.accept_contact_request(@received_request, aspect) }.should change(Contact, :count).by(1) user.contact_for(person_one).should_not be_nil end end describe '#ignore_contact_request' do it "removes the request" do lambda { user.ignore_contact_request(@received_request.id) }.should change(Request, :count).by(-1) end it "creates no new contact" do lambda { user.ignore_contact_request(@received_request) }.should_not change(Contact, :count) end end end describe 'disconnecting' do before do connect_users(user, aspect, user2, aspect2) end it 'should disconnect the other user on the same seed' do lambda { user2.disconnect user.person }.should change { user2.reload.contacts.count }.by(-1) aspect2.reload.contacts.count.should == 0 end it 'is disconnected by another user' do lambda { user.disconnected_by user2.person }.should change { user.contacts.count }.by(-1) aspect.reload.contacts.count.should == 0 end it 'should remove the contact from all aspects they are in' do user.add_contact_to_aspect( user.contact_for(user2.person), aspect1) aspect.reload.contacts.count.should == 1 aspect1.reload.contacts.count.should == 1 lambda { user.disconnected_by user2.person }.should change { user.contacts.count }.by(-1) aspect.reload.contacts.count.should == 0 aspect1.reload.contacts.count.should == 0 end context 'with a post' do before do @message = user.post(:status_message, :message => "hi", :to => aspect.id) end it "deletes the disconnected user's posts from visible_posts" do user2.reload.raw_visible_posts.include?(@message).should be_true user2.disconnect user.person user2.reload.raw_visible_posts.include?(@message).should be_false end it "deletes the disconnected user's posts from the aspect's posts" do Post.count.should == 1 aspect2.reload.posts.include?(@message).should be_true user2.disconnect user.person aspect2.reload.posts.include?(@message).should be_false Post.count.should == 1 end end end end end