diff --git a/Changelog.md b/Changelog.md index a9189ff05e9e65e361808149402a3a2ee665bf24..182a9cd06f1fe451301ebfee80973ef1b2efeb92 100644 --- a/Changelog.md +++ b/Changelog.md @@ -60,6 +60,7 @@ With the port to Bootstrap 3, app/views/terms/default.haml has a new structure. ## Refactor * Drop broken correlations from the admin pages [#6223](https://github.com/diaspora/diaspora/pull/6223) +* Extract PostService from PostsController [#6208](https://github.com/diaspora/diaspora/pull/6208) ## Bug fixes * Fix indentation and a link title on the default home page [#6212](https://github.com/diaspora/diaspora/pull/6212) diff --git a/app/assets/javascripts/app/models/post.js b/app/assets/javascripts/app/models/post.js index 8335f5a0d66e85c71f6a1f3ad8b0d53e270c1f4c..c351ee787214b9500d0023458e2e9f2358023c69 100644 --- a/app/assets/javascripts/app/models/post.js +++ b/app/assets/javascripts/app/models/post.js @@ -38,13 +38,6 @@ app.models.Post = Backbone.Model.extend(_.extend({}, app.models.formatDateMixin, .done(function(){ app.events.trigger('person:block:'+personId); }); }, - toggleFavorite : function(options){ - this.set({favorite : !this.get("favorite")}); - - /* guard against attempting to save a model that a user doesn't own */ - if(options.save){ this.save() } - }, - headline : function() { var headline = this.get("text").trim() , newlineIdx = headline.indexOf("\n"); diff --git a/app/controllers/posts_controller.rb b/app/controllers/posts_controller.rb index ec31639b3bc93207c63d60d51587fe0cc765d52f..1d1abb93121a8475a00cdde243975824f08a969f 100644 --- a/app/controllers/posts_controller.rb +++ b/app/controllers/posts_controller.rb @@ -5,94 +5,58 @@ class PostsController < ApplicationController include PostsHelper - before_action :authenticate_user!, :except => [:show, :iframe, :oembed, :interactions] - before_action :set_format_if_malformed_from_status_net, :only => :show - before_action :find_post, :only => [:show, :interactions] + before_action :authenticate_user!, only: :destroy + before_action :set_format_if_malformed_from_status_net, only: :show - respond_to :html, - :mobile, - :json, - :xml + respond_to :html, :mobile, :json, :xml - rescue_from Diaspora::NonPublic do |exception| + rescue_from Diaspora::NonPublic do respond_to do |format| - format.all { render :template=>'errors/not_public', :status=>404, :layout => "application"} + format.all { render template: "errors/not_public", status: 404, layout: "application" } end end def show - mark_corresponding_notifications_read if user_signed_in? - + post_service = PostService.new(id: params[:id], user: current_user) + post_service.mark_user_notifications + @post = post_service.post respond_to do |format| - format.html { - gon.post = PostPresenter.new(@post, current_user) - render "posts/show" - } + format.html { gon.post = post_service.present_json } format.xml { render xml: @post.to_diaspora_xml } - format.mobile { render "posts/show" } - format.json { render json: PostPresenter.new(@post, current_user) } + format.json { render json: post_service.present_json } end end def iframe - render :text => post_iframe_url(params[:id]), :layout => false + render text: post_iframe_url(params[:id]), layout: false end def oembed post_id = OEmbedPresenter.id_from_url(params.delete(:url)) - post = Post.find_by_guid_or_id_with_user(post_id, current_user) - if post.present? - oembed = OEmbedPresenter.new(post, params.slice(:format, :maxheight, :minheight)) - render :json => oembed - else - render :nothing => true, :status => 404 - end + post_service = PostService.new(id: post_id, user: current_user, + oembed: params.slice(:format, :maxheight, :minheight)) + render json: post_service.present_oembed end def interactions - respond_with(PostInteractionPresenter.new(@post, current_user)) + post_service = PostService.new(id: params[:id], user: current_user) + respond_with post_service.present_interactions_json end def destroy - find_current_user_post(params[:id]) - current_user.retract(@post) - + post_service = PostService.new(id: params[:id], user: current_user) + post_service.retract_post + @post = post_service.post respond_to do |format| - format.js { render 'destroy',:layout => false, :format => :js } - format.json { render :nothing => true, :status => 204 } + format.js { render "destroy", layout: false, format: :js } + format.json { render nothing: true, status: 204 } format.any { redirect_to stream_path } end end - def update - find_current_user_post(params[:id]) - @post.favorite = !@post.favorite - @post.save - render :nothing => true, :status => 202 - end - - protected - - def find_post #checks whether current user can see it - @post = Post.find_by_guid_or_id_with_user(params[:id], current_user) - end - - def find_current_user_post(id) #makes sure current_user can modify - @post = current_user.posts.find(id) - end + private def set_format_if_malformed_from_status_net - request.format = :html if request.format == 'application/html+xml' - end - - def mark_corresponding_notifications_read - # For comments, reshares, likes - Notification.where(recipient_id: current_user.id, target_type: "Post", target_id: @post.id, unread: true).each do |n| - n.set_read_state( true ) - end - - # For mentions - mention = @post.mentions.where(person_id: current_user.person_id).first - Notification.where(recipient_id: current_user.id, target_type: "Mention", target_id: mention.id, unread: true).first.try(:set_read_state, true) if mention + request.format = :html if request.format == "application/html+xml" end end diff --git a/app/models/post.rb b/app/models/post.rb index 79d06078cd1981758ec082aea5ef056f7515ee92..e59576085e380eedd2e7f5bd0137db888a691d18 100644 --- a/app/models/post.rb +++ b/app/models/post.rb @@ -148,17 +148,20 @@ class Post < ActiveRecord::Base self.author.profile.nsfw? end - def self.find_by_guid_or_id_with_user(id, user=nil) - key = id.to_s.length <= 8 ? :id : :guid - post = if user - user.find_visible_shareable_by_id(Post, id, :key => key) - else - Post.where(key => id).includes(:author, :comments => :author).first - end - - # is that a private post? - raise(Diaspora::NonPublic) unless user || post.try(:public?) - - post || raise(ActiveRecord::RecordNotFound.new("could not find a post with id #{id}")) + def self.find_public(id) + where(post_key(id) => id).includes(:author, comments: :author).first.tap do |post| + raise ActiveRecord::RecordNotFound.new("could not find a post with id #{id}") unless post + raise Diaspora::NonPublic unless post.public? + end + end + + def self.find_non_public_by_guid_or_id_with_user(id, user) + user.find_visible_shareable_by_id(Post, id, key: post_key(id)).tap do |post| + raise ActiveRecord::RecordNotFound.new("could not find a post with id #{id}") unless post + end + end + + def self.post_key(id) + id.to_s.length <= 8 ? :id : :guid end end diff --git a/app/presenters/post_interaction_presenter.rb b/app/presenters/post_interaction_presenter.rb index c67b7528a704757aef92cccb3a8bdb17b1adc44d..2616b0f76e77ff1fc3c71c2133891a104fadd962 100644 --- a/app/presenters/post_interaction_presenter.rb +++ b/app/presenters/post_interaction_presenter.rb @@ -16,6 +16,8 @@ class PostInteractionPresenter } end + private + def participations return @post.participations.none unless @current_user @post.participations.where(author: @current_user.person) diff --git a/app/presenters/post_presenter.rb b/app/presenters/post_presenter.rb index 5a763dcee682d56996e42a4be234b951b20e58bb..f17e6bb8c33a3e1fd74a94c419c3eb28de01559f 100644 --- a/app/presenters/post_presenter.rb +++ b/app/presenters/post_presenter.rb @@ -29,7 +29,6 @@ class PostPresenter :post_type => @post.post_type, :image_url => @post.image_url, :object_url => @post.object_url, - :favorite => @post.favorite, :nsfw => @post.nsfw, :author => @post.author.as_api_response(:backbone), :o_embed_cache => @post.o_embed_cache.try(:as_api_response, :backbone), diff --git a/app/services/post_service.rb b/app/services/post_service.rb new file mode 100644 index 0000000000000000000000000000000000000000..1408d064297cb471770aa34a8c647fac728f9823 --- /dev/null +++ b/app/services/post_service.rb @@ -0,0 +1,65 @@ +class PostService + attr_reader :post + + def initialize(params) + @id = params[:id] + @user = params[:user] + @oembed = params[:oembed] || {} + assign_post + end + + def assign_post + if user + @post = Post.find_non_public_by_guid_or_id_with_user(id, user) + else + @post = Post.find_public(id) + end + end + + def present_json + PostPresenter.new(post, user) + end + + def present_interactions_json + PostInteractionPresenter.new(post, user) + end + + def present_oembed + OEmbedPresenter.new(post, oembed) + end + + def mark_user_notifications + mark_corresponding_notifications_read if user + end + + def retract_post + raise Diaspora::NotMine unless user_owns_post? + user.retract(@post) + end + + private + + attr_reader :user, :id, :oembed + + def user_owns_post? + post.author == user.person + end + + def mark_corresponding_notifications_read + mark_comment_reshare_like_notifications_read + mark_mention_notifications_read + end + + def mark_comment_reshare_like_notifications_read + notification = Notification.where(recipient_id: user.id, target_type: "Post", target_id: post.id, unread: true) + notification.each do |notification| + notification.set_read_state(true) + end + end + + def mark_mention_notifications_read + mention = post.mentions.where(person_id: user.person_id).first + Notification.where(recipient_id: user.id, target_type: "Mention", target_id: mention.id, unread: true) + .first.try(:set_read_state, true) if mention + end +end diff --git a/db/migrate/20150724152052_remove_favorites_from_posts.rb b/db/migrate/20150724152052_remove_favorites_from_posts.rb new file mode 100644 index 0000000000000000000000000000000000000000..8d4bd873f5debb26e2ee3b1fb97536a20f8a6fc8 --- /dev/null +++ b/db/migrate/20150724152052_remove_favorites_from_posts.rb @@ -0,0 +1,9 @@ +class RemoveFavoritesFromPosts < ActiveRecord::Migration + def self.up + remove_column :posts, :favorite + end + + def self.down + add_column :posts, :favorite, :boolean, default: false + end +end diff --git a/db/schema.rb b/db/schema.rb index 2a4e925917134e6abcb19d4604b07f0367fb9e51..a0dffc8dc711df8ee97d7713c99cc265c664b546 100644 --- a/db/schema.rb +++ b/db/schema.rb @@ -11,7 +11,7 @@ # # It's strongly recommended that you check this file into your version control system. -ActiveRecord::Schema.define(version: 20150630221004) do +ActiveRecord::Schema.define(version: 20150724152052) do create_table "account_deletions", force: :cascade do |t| t.string "diaspora_handle", limit: 255 @@ -375,7 +375,6 @@ ActiveRecord::Schema.define(version: 20150630221004) do t.integer "reshares_count", limit: 4, default: 0 t.datetime "interacted_at" t.string "frame_name", limit: 255 - t.boolean "favorite", default: false t.string "facebook_id", limit: 255 t.string "tweet_id", limit: 255 t.integer "open_graph_cache_id", limit: 4 diff --git a/spec/controllers/posts_controller_spec.rb b/spec/controllers/posts_controller_spec.rb index 246238484e26e51d84589a831a25a308aac7690b..c6ed28fbbb6a2411d32ccb8005a8a50100452363 100644 --- a/spec/controllers/posts_controller_spec.rb +++ b/spec/controllers/posts_controller_spec.rb @@ -2,191 +2,112 @@ # licensed under the Affero General Public License version 3 or later. See # the COPYRIGHT file. -require 'spec_helper' +require "spec_helper" + +describe PostsController, type: :controller do + let!(:post_service_double) { double("post_service") } -describe PostsController, :type => :controller do before do aspect = alice.aspects.first - @message = alice.build_post :status_message, :text => "ohai", :to => aspect.id + @message = alice.build_post :status_message, text: "ohai", to: aspect.id @message.save! alice.add_to_streams(@message, [aspect]) - alice.dispatch_post @message, :to => aspect.id + alice.dispatch_post @message, to: aspect.id + + allow(PostService).to receive(:new).and_return(post_service_double) end - describe '#show' do - context 'user signed in' do + describe "#show" do + before do + expect(post_service_double).to receive(:mark_user_notifications) + allow(post_service_double).to receive(:present_json) + end + + context "user signed in" do before do sign_in :user, alice + expect(post_service_double).to receive(:post).and_return(@message) end - it 'succeeds' do - get :show, "id" => @message.id - expect(response).to be_success - end - - it 'succeeds on mobile' do - get :show, "id" => @message.id + it "succeeds" do + get :show, id: @message.id expect(response).to be_success end it 'succeeds after removing a mention when closing the mentioned user\'s account' do - user = FactoryGirl.create(:user, :username => "user") + user = FactoryGirl.create(:user, username: "user") alice.share_with(user.person, alice.aspects.first) - msg = alice.build_post :status_message, text: "Mention @{User ; #{user.diaspora_handle}}", :public => true, :to => 'all' + msg = alice.build_post :status_message, + text: "Mention @{User ; #{user.diaspora_handle}}", public: true, to: "all" msg.save! expect(msg.mentioned_people.count).to eq(1) user.destroy - get :show, "id" => msg.id + get :show, id: msg.id expect(response).to be_success end - it 'renders the application layout on mobile' do - get :show, :id => @message.id, :format => :mobile - expect(response).to render_template('layouts/application') + it "renders the application layout on mobile" do + get :show, id: @message.id, format: :mobile + expect(response).to render_template("layouts/application") end - it 'succeeds on mobile with a reshare' do - get :show, "id" => FactoryGirl.create(:reshare, :author => alice.person).id, :format => :mobile + it "succeeds on mobile with a reshare" do + get :show, id: FactoryGirl.create(:reshare, author: alice.person).id, format: :mobile expect(response).to be_success end - - it 'marks a corresponding notifications as read' do - FactoryGirl.create(:notification, :recipient => alice, :target => @message, :unread => true) - note = FactoryGirl.create(:notification, :recipient => alice, :target => @message, :unread => true) - - expect { - get :show, :id => @message.id - note.reload - }.to change(Notification.where(:unread => true), :count).by(-2) - end - - it 'marks a corresponding mention notification as read' do - status_msg = bob.post(:status_message, {text: "this is a text mentioning @{Mention User ; #{alice.diaspora_handle}} ... have fun testing!", :public => true, :to => 'all'}) - mention = status_msg.mentions.where(person_id: alice.person.id).first - note = FactoryGirl.create(:notification, :recipient => alice, :target_type => "Mention", :target_id => mention.id, :unread => true) - - expect { - get :show, :id => status_msg.id - note.reload - }.to change(Notification.where(:unread => true), :count).by(-1) - end - - it '404 if the post is missing' do - expect { - get :show, :id => 1234567 - }.to raise_error(ActiveRecord::RecordNotFound) - end end - context 'user not signed in' do - context 'given a public post' do + context "user not signed in" do + context "given a public post" do before :each do - @status = alice.post(:status_message, :text => "hello", :public => true, :to => 'all') + @status = alice.post(:status_message, text: "hello", public: true, to: "all") + expect(post_service_double).to receive(:post).and_return(@status) end - it 'shows a public post' do - get :show, :id => @status.id - expect(response.status).to eq(200) + it "shows a public post" do + get :show, id: @status.id + expect(response.body).to match "hello" end - it 'succeeds for statusnet' do + it "succeeds for statusnet" do @request.env["HTTP_ACCEPT"] = "application/html+xml,text/html" - get :show, :id => @status.id - expect(response).to be_success + get :show, id: @status.id + expect(response.body).to match "hello" end - it 'responds with diaspora xml if format is xml' do - get :show, :id => @status.guid, :format => :xml + it "responds with diaspora xml if format is xml" do + get :show, id: @status.guid, format: :xml expect(response.body).to eq(@status.to_diaspora_xml) end end - - it 'does not show a private post' do - status = alice.post(:status_message, :text => "hello", :public => false, :to => 'all') - get :show, :id => status.id - expect(response.status).to eq(404) - end - - # We want to be using guids from now on for this post route, but do not want to break - # pre-exisiting permalinks. We can assume a guid is 8 characters long as we have - # guids set to hex(8) since we started using them. - context 'id/guid switch' do - before do - @status = alice.post(:status_message, :text => "hello", :public => true, :to => 'all') - end - - it 'assumes guids less than 8 chars are ids and not guids' do - p = Post.where(:id => @status.id.to_s) - expect(Post).to receive(:where) - .with(hash_including(:id => @status.id.to_s)) - .and_return(p) - get :show, :id => @status.id - expect(response).to be_success - end - - it 'assumes guids more than (or equal to) 8 chars are actually guids' do - p = Post.where(:guid => @status.guid) - expect(Post).to receive(:where) - .with(hash_including(:guid => @status.guid)) - .and_return(p) - get :show, :id => @status.guid - expect(response).to be_success - end - end end end - describe 'iframe' do - it 'contains an iframe' do - get :iframe, :id => @message.id + describe "iframe" do + it "contains an iframe" do + get :iframe, id: @message.id expect(response.body).to match /iframe/ end end - describe 'oembed' do - it 'works when you can see it' do - sign_in alice - get :oembed, :url => "/posts/#{@message.id}" - expect(response.body).to match /iframe/ - end - - it 'returns a 404 response when the post is not found' do - get :oembed, :url => "/posts/#{@message.id}" - expect(response.status).to eq(404) + describe "oembed" do + it "receives a present oembed" do + expect(post_service_double).to receive(:present_oembed) + get :oembed, url: "/posts/#{@message.id}" end end - describe '#destroy' do + describe "#destroy" do before do sign_in alice end - it 'let a user delete his message' do - message = alice.post(:status_message, :text => "hey", :to => alice.aspects.first.id) - delete :destroy, :format => :js, :id => message.id - expect(response).to be_success - expect(StatusMessage.find_by_id(message.id)).to be_nil - end - - it 'sends a retraction on delete' do - allow(controller).to receive(:current_user).and_return alice - message = alice.post(:status_message, :text => "hey", :to => alice.aspects.first.id) - expect(alice).to receive(:retract).with(message) - delete :destroy, :format => :js, :id => message.id - expect(response).to be_success - end - - it 'will not let you destroy posts visible to you' do - message = bob.post(:status_message, :text => "hey", :to => bob.aspects.first.id) - expect { delete :destroy, :format => :js, :id => message.id }.to raise_error(ActiveRecord::RecordNotFound) - expect(StatusMessage.exists?(message.id)).to be true - end - - it 'will not let you destory posts you do not own' do - message = eve.post(:status_message, :text => "hey", :to => eve.aspects.first.id) - expect { delete :destroy, :format => :js, :id => message.id }.to raise_error(ActiveRecord::RecordNotFound) - expect(StatusMessage.exists?(message.id)).to be true + it "will receive a retract post" do + expect(post_service_double).to receive(:retract_post) + expect(post_service_double).to receive(:post).and_return(@message) + message = alice.post(:status_message, text: "hey", to: alice.aspects.first.id) + delete :destroy, format: :js, id: message.id end end end diff --git a/spec/models/post_spec.rb b/spec/models/post_spec.rb index 00fbe98e2d38f4ab67e699b306d40ba866f8ac6e..d35d4d19ff434d2fad641dfa6d36600baa2e2e08 100644 --- a/spec/models/post_spec.rb +++ b/spec/models/post_spec.rb @@ -400,43 +400,55 @@ describe Post, :type => :model do end end - describe "#find_by_guid_or_id_with_user" do + describe "#find_public" do it "succeeds with an id" do post = FactoryGirl.create :status_message, public: true - expect(Post.find_by_guid_or_id_with_user(post.id)).to eq(post) + expect(Post.find_public post.id).to eq(post) end it "succeeds with an guid" do post = FactoryGirl.create :status_message, public: true - expect(Post.find_by_guid_or_id_with_user(post.guid)).to eq(post) + expect(Post.find_public post.guid).to eq(post) end - it "looks up on the passed user object if it's non-nil" do - post = FactoryGirl.create :status_message - user = double - expect(user).to receive(:find_visible_shareable_by_id).with(Post, post.id, key: :id).and_return(post) - Post.find_by_guid_or_id_with_user post.id, user - end - - it "raises ActiveRecord::RecordNotFound with a non-existing id and a user" do - user = double(find_visible_shareable_by_id: nil) + it "raises ActiveRecord::RecordNotFound for a non-existing id without a user" do + allow(Post).to receive_messages where: double(includes: double(first: nil)) expect { - Post.find_by_guid_or_id_with_user 123, user + Post.find_public 123 }.to raise_error ActiveRecord::RecordNotFound end - it "raises Diaspora::NonPublic for a non-existing id without a user" do - allow(Post).to receive_messages where: double(includes: double(first: nil)) + it "raises Diaspora::NonPublic for a private post without a user" do + post = FactoryGirl.create :status_message expect { - Post.find_by_guid_or_id_with_user 123 + Post.find_public post.id }.to raise_error Diaspora::NonPublic end + end - it "raises Diaspora::NonPublic for a private post without a user" do + describe "#find_non_public_by_guid_or_id_with_user" do + it "succeeds with an id" do + post = FactoryGirl.create :status_message_in_aspect + expect(Post.find_non_public_by_guid_or_id_with_user(post.id, post.author.owner)).to eq(post) + end + + it "succeeds with an guid" do + post = FactoryGirl.create :status_message_in_aspect + expect(Post.find_non_public_by_guid_or_id_with_user(post.guid, post.author.owner)).to eq(post) + end + + it "looks up on the passed user object if it's non-nil" do post = FactoryGirl.create :status_message + user = double + expect(user).to receive(:find_visible_shareable_by_id).with(Post, post.id, key: :id).and_return(post) + Post.find_non_public_by_guid_or_id_with_user(post.id, user) + end + + it "raises ActiveRecord::RecordNotFound with a non-existing id and a user" do + user = double(find_visible_shareable_by_id: nil) expect { - Post.find_by_guid_or_id_with_user post.id - }.to raise_error Diaspora::NonPublic + Post.find_non_public_by_guid_or_id_with_user(123, user) + }.to raise_error ActiveRecord::RecordNotFound end end end diff --git a/spec/services/post_service_spec.rb b/spec/services/post_service_spec.rb new file mode 100644 index 0000000000000000000000000000000000000000..47f57d750c0734a3f510c18f80b7bdb81480cb77 --- /dev/null +++ b/spec/services/post_service_spec.rb @@ -0,0 +1,127 @@ +require "spec_helper" + +describe PostService do + before do + aspect = alice.aspects.first + @message = alice.build_post :status_message, text: "ohai", to: aspect.id + @message.save! + + alice.add_to_streams(@message, [aspect]) + alice.dispatch_post @message, to: aspect.id + end + + describe "#assign_post" do + context "when the post is private" do + it "RecordNotFound if the post cannot be found" do + expect { PostService.new(id: 1_234_567, user: alice) }.to raise_error(ActiveRecord::RecordNotFound) + end + it "NonPublic if there is no user" do + expect { PostService.new(id: @message.id) }.to raise_error(Diaspora::NonPublic) + end + it "RecordNotFound if user cannot see post" do + expect { PostService.new(id: @message.id, user: eve) }.to raise_error(ActiveRecord::RecordNotFound) + end + end + + context "when the post is public" do + it "RecordNotFound if the post cannot be found" do + expect { PostService.new(id: 1_234_567) }.to raise_error(ActiveRecord::RecordNotFound) + end + end + + # We want to be using guids from now on for this post route, but do not want to break + # pre-exisiting permalinks. We can assume a guid is 8 characters long as we have + # guids set to hex(8) since we started using them. + context "id/guid switch" do + before do + @status = alice.post(:status_message, text: "hello", public: true, to: "all") + end + + it "assumes guids less than 8 chars are ids and not guids" do + post = Post.where(id: @status.id.to_s) + expect(Post).to receive(:where).with(hash_including(id: @status.id)).and_return(post).at_least(:once) + PostService.new(id: @status.id, user: alice) + end + + it "assumes guids more than (or equal to) 8 chars are actually guids" do + post = Post.where(guid: @status.guid) + expect(Post).to receive(:where).with(hash_including(guid: @status.guid)).and_return(post).at_least(:once) + PostService.new(id: @status.guid, user: alice) + end + end + end + + describe "#mark_user_notifications" do + it "marks a corresponding notifications as read" do + FactoryGirl.create(:notification, recipient: alice, target: @message, unread: true) + FactoryGirl.create(:notification, recipient: alice, target: @message, unread: true) + post_service = PostService.new(id: @message.id, user: alice) + expect { post_service.mark_user_notifications }.to change(Notification.where(unread: true), :count).by(-2) + end + + it "marks a corresponding mention notification as read" do + status_text = "this is a text mentioning @{Mention User ; #{alice.diaspora_handle}} ... have fun testing!" + status_msg = + bob.post(:status_message, text: status_text, public: true, to: "all") + mention = status_msg.mentions.where(person_id: alice.person.id).first + FactoryGirl.create(:notification, recipient: alice, target_type: "Mention", target_id: mention.id, unread: true) + post_service = PostService.new(id: status_msg.id, user: alice) + expect { post_service.mark_user_notifications }.to change(Notification.where(unread: true), :count).by(-1) + end + end + + describe "#present_json" do + it "works for a private post" do + post_service = PostService.new(id: @message.id, user: alice) + expect(post_service.present_json.to_json).to match(/\"text\"\:\"ohai\"/) + end + + it "works for a public post " do + status = alice.post(:status_message, text: "hello", public: true, to: "all") + post_service = PostService.new(id: status.id) + expect(post_service.present_json.to_json).to match(/\"text\"\:\"hello\"/) + end + end + + describe "#present_oembed" do + it "works for a private post" do + post_service = PostService.new(id: @message.id, user: alice) + expect(post_service.present_oembed.to_json).to match(/iframe/) + end + + it "works for a public post" do + status = alice.post(:status_message, text: "hello", public: true, to: "all") + post_service = PostService.new(id: status.id) + expect(post_service.present_oembed.to_json).to match(/iframe/) + end + end + + describe "#retract_post" do + it "let a user delete his message" do + message = alice.post(:status_message, text: "hey", to: alice.aspects.first.id) + post_service = PostService.new(id: message.id, user: alice) + post_service.retract_post + expect(StatusMessage.find_by_id(message.id)).to be_nil + end + + it "sends a retraction on delete" do + message = alice.post(:status_message, text: "hey", to: alice.aspects.first.id) + post_service = PostService.new(id: message.id, user: alice) + expect(alice).to receive(:retract).with(message) + post_service.retract_post + end + + it "will not let you destroy posts visible to you but that you do not own" do + message = bob.post(:status_message, text: "hey", to: bob.aspects.first.id) + post_service = PostService.new(id: message.id, user: alice) + expect { post_service.retract_post }.to raise_error(Diaspora::NotMine) + expect(StatusMessage.exists?(message.id)).to be true + end + + it "will not let you destroy posts that are not visible to you" do + message = eve.post(:status_message, text: "hey", to: eve.aspects.first.id) + expect { PostService.new(id: message.id, user: alice) }.to raise_error(ActiveRecord::RecordNotFound) + expect(StatusMessage.exists?(message.id)).to be true + end + end +end