Skip to content
Extraits de code Groupes Projets
receiving_spec.rb 11,5 ko
Newer Older
Raphael's avatar
Raphael a validé
#   Copyright (c) 2010, Diaspora Inc.  This file is
Raphael's avatar
Raphael a validé
#   licensed under the Affero General Public License version 3 or later.  See
Raphael's avatar
Raphael a validé
#   the COPYRIGHT file.
require 'spec_helper'
describe 'a user receives a post' do
Raphael's avatar
Raphael a validé
  def receive_with_zord(user, person, xml)
    zord = Postzord::Receiver.new(user, :person => person)
    zord.parse_and_receive(xml)
  end

    @user1  = alice
    @aspect = @user1.aspects.first

    @user2   = bob
    @aspect2 = @user2.aspects.first

    @user3   = eve
    @aspect3 = @user3.aspects.first

    @contact = @user1.contact_for(@user2.person)
  it 'streams only one message to the everyone aspect when a multi-aspected contacts posts' do
    contact = @user1.contact_for(@user2.person)
    @user1.add_contact_to_aspect(contact, @user1.aspects.create(:name => "villains"))
    status = @user2.build_post(:status_message, :message => "Users do things", :to => @aspect2.id)
Derrick Camerino's avatar
Derrick Camerino a validé
    Diaspora::WebSocket.should_receive(:queue_to_user).exactly(:once)
    zord = Postzord::Receiver.new(@user1, :object => status, :person => @user2.person)
    zord.receive_object
  it 'should be able to parse and store a status message from xml' do
    status_message = @user2.post :status_message, :message => 'store this!', :to => @aspect2.id

    xml = status_message.to_diaspora_xml
    status_message.destroy
      receive_with_zord(@user1, @user2.person, xml)
    }.should change(Post,:count).by(1)
Raphael's avatar
Raphael a validé
  it 'should not create new aspects on message receive' do
    num_aspects = @user1.aspects.size
    2.times do |n|
      status_message = @user2.post :status_message, :message => "store this #{n}!", :to => @aspect2.id
    end
    @user1.aspects.size.should == num_aspects
maxwell's avatar
maxwell a validé
  context 'update posts' do
    it 'does not update posts not marked as mutable' do
      status = @user1.post :status_message, :message => "store this!", :to => @aspect.id
      xml = status.to_diaspora_xml
     receive_with_zord(@user2, @user1.person, xml)
      status.reload.message.should == 'store this!'
      photo = @user1.post(:photo, :user_file => uploaded_photo, :caption => "Original", :to => @aspect.id)
danielvincent's avatar
danielvincent a validé
      photo.caption = 'foo'
      xml = photo.to_diaspora_xml
      receive_with_zord(@user2, @user1.person, xml)
danielvincent's avatar
danielvincent a validé
      photo.reload.caption.should match(/foo/)
  describe 'post refs' do
      @status_message = @user2.post :status_message, :message => "hi", :to => @aspect2.id
      @user1.reload
      @aspect.reload
    it "adds a received post to the aspect and visible_posts array" do
      @user1.raw_visible_posts.include?(@status_message).should be_true
      @aspect.posts.include?(@status_message).should be_true
    it 'removes posts upon disconnecting' do
      @user1.disconnect(@contact)
      @user1.reload
      @user1.raw_visible_posts.should_not include @status_message
    it 'deletes a post if the noone links to it' do
      person = Factory(:person)
      @user1.activate_contact(person, @aspect)
      post = Factory.create(:status_message, :person => person)
      post.post_visibilities.should be_empty
      receive_with_zord(@user1, person, post.to_diaspora_xml)
      @aspect.post_visibilities.reset
      @aspect.posts(true).should include(post)
      post.post_visibilities.reset
      post.post_visibilities.length.should == 1
        @user1.disconnected_by(person)
      }.should change(Post, :count).by(-1)
    it 'deletes post_visibilities on disconnected by' do
      person = Factory(:person)
      @user1.activate_contact(person, @aspect)
      post = Factory.create(:status_message, :person => person)
      post.post_visibilities.should be_empty
      receive_with_zord(@user1, person, post.to_diaspora_xml)
      @aspect.post_visibilities.reset
      @aspect.posts(true).should include(post)
      post.post_visibilities.reset
      post.post_visibilities.length.should == 1
        @user1.disconnected_by(person)
      }.should change{post.post_visibilities(true).count}.by(-1)
    end
    it 'should keep track of user references for one person ' do
      @status_message.reload
      @status_message.user_refs.should == 3
      @user1.disconnect(@contact)
      @status_message.reload
      @status_message.user_refs.should == 2
    end

    it 'should not override userrefs on receive by another person' do
      new_user = Factory(:user_with_aspect)
      @status_message.post_visibilities.reset
      @status_message.user_refs.should == 3
      new_user.activate_contact(@user2.person, new_user.aspects.first)
      xml = @status_message.to_diaspora_xml
      receive_with_zord(new_user, @user2.person, xml)
      @status_message.post_visibilities.reset
      @status_message.user_refs.should == 4
      @user1.disconnect(@contact)
      @status_message.post_visibilities.reset
      @status_message.user_refs.should == 3
Raphael's avatar
Raphael a validé

  describe 'comments' do

    context 'remote' do
      before do
        connect_users(@user1, @aspect, @user3, @aspect3)
        @post = @user1.post :status_message, :message => "hello", :to => @aspect.id

        xml = @post.to_diaspora_xml

        receive_with_zord(@user2, @user1.person, xml)
        receive_with_zord(@user3, @user1.person, xml)

        @comment = @user3.comment('tada',:on => @post)
        @comment.post_creator_signature = @comment.sign_with_key(@user1.encryption_key)
        @xml = @comment.to_diaspora_xml
        @comment.delete
      end

      it 'should correctly attach the user already on the pod' do
        @user2.reload.raw_visible_posts.size.should == 1
        post_in_db = StatusMessage.find(@post.id)
        post_in_db.comments.should == []
        receive_with_zord(@user2, @user1.person, @xml)

        post_in_db.comments(true).first.person.should == @user3.person
      end

      it 'should correctly marshal a stranger for the downstream user' do
        remote_person = @user3.person.dup
        @user3.person.delete
        @user3.delete
        Person.where(:id => remote_person.id).delete_all
        Profile.where(:person_id => remote_person.id).delete_all
        remote_person.id = nil

        Person.should_receive(:by_account_identifier).twice.and_return{ |handle|
          if handle == @user1.person.diaspora_handle
            @user1.person.save
            @user1.person
          else
            remote_person.save(:validate => false)
            remote_person.profile = Factory(:profile, :person => remote_person)
            remote_person
          end
        }

        @user2.reload.raw_visible_posts.size.should == 1
        post_in_db = StatusMessage.find(@post.id)
        post_in_db.comments.should == []

        receive_with_zord(@user2, @user1.person, @xml)

        post_in_db.comments(true).first.person.should == remote_person
      end
    end
    context 'local' do
      before do
        @post = @user1.post :status_message, :message => "hello", :to => @aspect.id
        xml = @post.to_diaspora_xml
        receive_with_zord(@user2, @user1.person, xml)
        receive_with_zord(@user3, @user1.person, xml)
      it 'does not raise a `Mysql2::Error: Duplicate entry...` exception on save' do
        @comment = @user2.comment('tada',:on => @post)
        @xml = @comment.to_diaspora_xml

        lambda {
            receive_with_zord(@user1, @user2.person, @xml)
        }.should_not raise_exception
      end
Raphael's avatar
Raphael a validé
    end
  end
Raphael's avatar
Raphael a validé


  describe 'receiving mulitple versions of the same post from a remote pod' do
    before do
      @local_luke, @local_leia, @remote_raphael = set_up_friends
      @post = Factory.build(:status_message, :message => 'hey', :guid => 12313123, :person => @remote_raphael, :created_at => 5.days.ago, :updated_at => 5.days.ago)
    end

    it 'does not update created_at or updated_at when two people save the same post' do
      @post = Factory.build(:status_message, :message => 'hey', :guid => 12313123, :person => @remote_raphael, :created_at => 5.days.ago, :updated_at => 5.days.ago)
      xml = @post.to_diaspora_xml
      receive_with_zord(@local_luke, @remote_raphael, xml)
      sleep(2)
      old_time = Time.now
      receive_with_zord(@local_leia, @remote_raphael, xml)
      (Post.find_by_guid @post.guid).updated_at.should be < old_time
      (Post.find_by_guid @post.guid).created_at.should be < old_time
    end

    it 'does not update the post if a new one is sent with a new created_at' do
      @post = Factory.build(:status_message, :message => 'hey', :guid => 12313123, :person => @remote_raphael, :created_at => 5.days.ago)
      old_time = @post.created_at
      xml = @post.to_diaspora_xml
      receive_with_zord(@local_luke, @remote_raphael, xml)
      @post = Factory.build(:status_message, :message => 'hey', :guid => 12313123, :person => @remote_raphael, :created_at => 2.days.ago)
      receive_with_zord(@local_luke, @remote_raphael, xml)
      (Post.find_by_guid @post.guid).created_at.day.should == old_time.day
    end
  end


Raphael's avatar
Raphael a validé
  describe 'salmon' do
    let(:post){@user1.post :status_message, :message => "hello", :to => @aspect.id}
    let(:salmon){@user1.salmon( post )}
Raphael's avatar
Raphael a validé

Raphael's avatar
Raphael a validé
    it 'processes a salmon for a post' do
      salmon_xml = salmon.xml_for(@user2.person)
      zord = Postzord::Receiver.new(@user2, :salmon_xml => salmon_xml)
      @user2.raw_visible_posts.include?(post).should be_true
Raphael's avatar
Raphael a validé
    end
  end


  context 'retractions' do
    it 'should accept retractions' do
      message = @user2.post(:status_message, :message => "cats", :to => @aspect2.id)
      retraction = Retraction.for(message)
      xml = retraction.to_diaspora_xml

      lambda {
        zord = Postzord::Receiver.new(@user1, :person => @user2.person)
        zord.parse_and_receive(xml)
      }.should change(StatusMessage, :count).by(-1)
    end

    it "should activate the Person if I initiated a request to that url" do
      begin
        @user1.send_contact_request_to(@user3.person, @aspect)
      rescue Exception => e
        raise e.original_exception
      end
      request = @user3.request_from(@user1.person)
      fantasy_resque do
        @user3.accept_and_respond(request.id, @aspect3.id)
      end
      @user1.reload
      @aspect.reload
      new_contact = @user1.contact_for(@user3.person)
      @aspect.contacts.include?(new_contact).should be true
      @user1.contacts.include?(new_contact).should be true
    end

    it 'should process retraction for a person' do
      retraction = Retraction.for(@user2)
      retraction_xml = retraction.to_diaspora_xml

      lambda {
        zord = Postzord::Receiver.new(@user1, :person => @user2.person)
        zord.parse_and_receive(retraction_xml)
      }.should change {
        @aspect.contacts(true).size }.by(-1)
    end

  end

  it 'should marshal a profile for a person' do
    #Create person
    person = @user2.person
    id = person.id
    person.profile = Profile.new(:first_name => 'bob', :last_name => 'billytown', :image_url => "http://clown.com")
    person.save

    #Cache profile for checking against marshaled profile
    new_profile = person.profile.dup
    new_profile.first_name = 'boo!!!'
    #Build xml for profile
    xml = new_profile.to_diaspora_xml

    #Marshal profile
    zord = Postzord::Receiver.new(@user1, :person => person)
    zord.parse_and_receive(xml)

    #Check that marshaled profile is the same as old profile
    person = Person.find(person.id)
    person.profile.first_name.should == new_profile.first_name
    person.profile.last_name.should == new_profile.last_name
    person.profile.image_url.should == new_profile.image_url