Skip to content

Commit

Permalink
Fix leak of arbitrary statuses through unfavourite action in REST API (
Browse files Browse the repository at this point in the history
  • Loading branch information
Gargron authored and abcang committed Feb 28, 2020
1 parent a687dcb commit 5383f7a
Show file tree
Hide file tree
Showing 8 changed files with 265 additions and 85 deletions.
32 changes: 32 additions & 0 deletions app/controllers/api/v1/statuses/bookmarks_controller.rb
Original file line number Diff line number Diff line change
@@ -0,0 +1,32 @@
# frozen_string_literal: true

class Api::V1::Statuses::BookmarksController < Api::BaseController
include Authorization

before_action -> { doorkeeper_authorize! :write, :'write:bookmarks' }
before_action :require_user!
before_action :set_status

respond_to :json

def create
current_account.bookmarks.find_or_create_by!(account: current_account, status: @status)
render json: @status, serializer: REST::StatusSerializer
end

def destroy
bookmark = current_account.bookmarks.find_by(status: @status)
bookmark&.destroy!

render json: @status, serializer: REST::StatusSerializer, relationships: StatusRelationshipsPresenter.new([@status], current_account.id, bookmarks_map: { @status.id => false })
end

private

def set_status
@status = Status.find(params[:status_id])
authorize @status, :show?
rescue Mastodon::NotPermittedError
not_found
end
end
Original file line number Diff line number Diff line change
Expand Up @@ -67,8 +67,7 @@ def set_status
@status = Status.find(params[:status_id])
authorize @status, :show?
rescue Mastodon::NotPermittedError
# Reraise in order to get a 404 instead of a 403 error code
raise ActiveRecord::RecordNotFound
not_found
end

def authorize_if_got_token
Expand Down
26 changes: 9 additions & 17 deletions app/controllers/api/v1/statuses/favourites_controller.rb
Original file line number Diff line number Diff line change
Expand Up @@ -5,34 +5,26 @@ class Api::V1::Statuses::FavouritesController < Api::BaseController

before_action -> { doorkeeper_authorize! :write }
before_action :require_user!
before_action :set_status

respond_to :json

def create
@status = favourited_status
FavouriteService.new.call(current_account, @status)
render json: @status, serializer: REST::StatusSerializer
end

def destroy
@status = requested_status
@favourites_map = { @status.id => false }

UnfavouriteWorker.perform_async(current_user.account_id, @status.id)

render json: @status, serializer: REST::StatusSerializer, relationships: StatusRelationshipsPresenter.new([@status], current_user&.account_id, favourites_map: @favourites_map)
UnfavouriteWorker.perform_async(current_account.id, @status.id)
render json: @status, serializer: REST::StatusSerializer, relationships: StatusRelationshipsPresenter.new([@status], current_account.id, favourites_map: { @status.id => false })
end

private

def favourited_status
service_result.status.reload
end

def service_result
FavouriteService.new.call(current_user.account, requested_status)
end

def requested_status
Status.find(params[:status_id])
def set_status
@status = Status.find(params[:status_id])
authorize @status, :show?
rescue Mastodon::NotPermittedError
not_found
end
end
Original file line number Diff line number Diff line change
Expand Up @@ -64,8 +64,7 @@ def set_status
@status = Status.find(params[:status_id])
authorize @status, :show?
rescue Mastodon::NotPermittedError
# Reraise in order to get a 404 instead of a 403 error code
raise ActiveRecord::RecordNotFound
not_found
end

def authorize_if_got_token
Expand Down
26 changes: 16 additions & 10 deletions app/controllers/api/v1/statuses/reblogs_controller.rb
Original file line number Diff line number Diff line change
Expand Up @@ -5,31 +5,37 @@ class Api::V1::Statuses::ReblogsController < Api::BaseController

before_action -> { doorkeeper_authorize! :write }
before_action :require_user!
before_action :set_reblog

respond_to :json

def create
@status = ReblogService.new.call(current_user.account, status_for_reblog)
@status = ReblogService.new.call(current_account, @reblog, reblog_params)
render json: @status, serializer: REST::StatusSerializer
end

def destroy
@status = status_for_destroy.reblog
@reblogs_map = { @status.id => false }
@status = current_account.statuses.find_by(reblog_of_id: @reblog.id)

authorize status_for_destroy, :unreblog?
RemovalWorker.perform_async(status_for_destroy.id)
if @status
authorize @status, :unreblog?
@status.discard
RemovalWorker.perform_async(@status.id)
end

render json: @status, serializer: REST::StatusSerializer, relationships: StatusRelationshipsPresenter.new([@status], current_user&.account_id, reblogs_map: @reblogs_map)
render json: @reblog, serializer: REST::StatusSerializer, relationships: StatusRelationshipsPresenter.new([@status], current_account.id, reblogs_map: { @reblog.id => false })
end

private

def status_for_reblog
Status.find params[:status_id]
def set_reblog
@reblog = Status.find(params[:status_id])
authorize @reblog, :show?
rescue Mastodon::NotPermittedError
not_found
end

def status_for_destroy
current_user.account.statuses.where(reblog_of_id: params[:status_id]).first!
def reblog_params
params.permit(:visibility)
end
end
88 changes: 88 additions & 0 deletions spec/controllers/api/v1/statuses/bookmarks_controller_spec.rb
Original file line number Diff line number Diff line change
@@ -0,0 +1,88 @@
# frozen_string_literal: true

require 'rails_helper'

describe Api::V1::Statuses::BookmarksController do
render_views

let(:user) { Fabricate(:user, account: Fabricate(:account, username: 'alice')) }
let(:app) { Fabricate(:application, name: 'Test app', website: 'http://testapp.com') }
let(:token) { Fabricate(:accessible_access_token, resource_owner_id: user.id, scopes: 'write:bookmarks', application: app) }

context 'with an oauth token' do
before do
allow(controller).to receive(:doorkeeper_token) { token }
end

describe 'POST #create' do
let(:status) { Fabricate(:status, account: user.account) }

before do
post :create, params: { status_id: status.id }
end

context 'with public status' do
it 'returns http success' do
expect(response).to have_http_status(:success)
end

it 'updates the bookmarked attribute' do
expect(user.account.bookmarked?(status)).to be true
end

it 'returns json with updated attributes' do
hash_body = body_as_json

expect(hash_body[:id]).to eq status.id.to_s
expect(hash_body[:bookmarked]).to be true
end
end

context 'with private status of not-followed account' do
let(:status) { Fabricate(:status, visibility: :private) }

it 'returns http not found' do
expect(response).to have_http_status(404)
end
end
end

describe 'POST #destroy' do
context 'with public status' do
let(:status) { Fabricate(:status, account: user.account) }

before do
Bookmark.find_or_create_by!(account: user.account, status: status)
post :destroy, params: { status_id: status.id }
end

it 'returns http success' do
expect(response).to have_http_status(:success)
end

it 'updates the bookmarked attribute' do
expect(user.account.bookmarked?(status)).to be false
end

it 'returns json with updated attributes' do
hash_body = body_as_json

expect(hash_body[:id]).to eq status.id.to_s
expect(hash_body[:bookmarked]).to be false
end
end

context 'with private status that was not bookmarked' do
let(:status) { Fabricate(:status, visibility: :private) }

before do
post :destroy, params: { status_id: status.id }
end

it 'returns http not found' do
expect(response).to have_http_status(404)
end
end
end
end
end
86 changes: 59 additions & 27 deletions spec/controllers/api/v1/statuses/favourites_controller_spec.rb
Original file line number Diff line number Diff line change
Expand Up @@ -21,45 +21,77 @@
post :create, params: { status_id: status.id }
end

it 'returns http success' do
expect(response).to have_http_status(200)
context 'with public status' do
it 'returns http success' do
expect(response).to have_http_status(200)
end

it 'updates the favourites count' do
expect(status.favourites.count).to eq 1
end

it 'updates the favourited attribute' do
expect(user.account.favourited?(status)).to be true
end

it 'returns json with updated attributes' do
hash_body = body_as_json

expect(hash_body[:id]).to eq status.id.to_s
expect(hash_body[:favourites_count]).to eq 1
expect(hash_body[:favourited]).to be true
end
end

it 'updates the favourites count' do
expect(status.favourites.count).to eq 1
end

it 'updates the favourited attribute' do
expect(user.account.favourited?(status)).to be true
end

it 'return json with updated attributes' do
hash_body = body_as_json
context 'with private status of not-followed account' do
let(:status) { Fabricate(:status, visibility: :private) }

expect(hash_body[:id]).to eq status.id.to_s
expect(hash_body[:favourites_count]).to eq 1
expect(hash_body[:favourited]).to be true
it 'returns http not found' do
expect(response).to have_http_status(404)
end
end
end

describe 'POST #destroy' do
let(:status) { Fabricate(:status, account: user.account) }
context 'with public status' do
let(:status) { Fabricate(:status, account: user.account) }

before do
FavouriteService.new.call(user.account, status)
post :destroy, params: { status_id: status.id }
end
before do
FavouriteService.new.call(user.account, status)
post :destroy, params: { status_id: status.id }
end

it 'returns http success' do
expect(response).to have_http_status(200)
end
it 'returns http success' do
expect(response).to have_http_status(200)
end

it 'updates the favourites count' do
expect(status.favourites.count).to eq 0
end

it 'updates the favourited attribute' do
expect(user.account.favourited?(status)).to be false
end

it 'updates the favourites count' do
expect(status.favourites.count).to eq 0
it 'returns json with updated attributes' do
hash_body = body_as_json

expect(hash_body[:id]).to eq status.id.to_s
expect(hash_body[:favourites_count]).to eq 0
expect(hash_body[:favourited]).to be false
end
end

it 'updates the favourited attribute' do
expect(user.account.favourited?(status)).to be false
context 'with private status that was not favourited' do
let(:status) { Fabricate(:status, visibility: :private) }

before do
post :destroy, params: { status_id: status.id }
end

it 'returns http not found' do
expect(response).to have_http_status(404)
end
end
end
end
Expand Down
Loading

0 comments on commit 5383f7a

Please sign in to comment.