FIX: restrict other user's notification routes (#14442)
It was possible to see notifications of other users using routes: - notifications/responses - notifications/likes-received - notifications/mentions - notifications/edits We weren't showing anything private (like notifications about private messages), only things that're publicly available in other places. But anyway, it feels strange that it's possible to look at notifications of someone else. Additionally, there is a risk that we can unintentionally leak something on these pages in the future. This commit restricts these routes.
This commit is contained in:
parent
2d4b9595e8
commit
b609f6c11c
|
@ -1,18 +1,18 @@
|
||||||
# frozen_string_literal: true
|
# frozen_string_literal: true
|
||||||
|
|
||||||
class UserActionsController < ApplicationController
|
class UserActionsController < ApplicationController
|
||||||
|
|
||||||
def index
|
def index
|
||||||
params.require(:username)
|
params.require(:username)
|
||||||
params.permit(:filter, :offset, :acting_username, :limit)
|
params.permit(:filter, :offset, :acting_username, :limit)
|
||||||
|
|
||||||
user = fetch_user_from_params(include_inactive: current_user.try(:staff?) || (current_user && SiteSetting.show_inactive_accounts))
|
user = fetch_user_from_params(include_inactive: current_user.try(:staff?) || (current_user && SiteSetting.show_inactive_accounts))
|
||||||
raise Discourse::NotFound unless guardian.can_see_profile?(user)
|
|
||||||
|
|
||||||
offset = [0, params[:offset].to_i].max
|
offset = [0, params[:offset].to_i].max
|
||||||
action_types = (params[:filter] || "").split(",").map(&:to_i)
|
action_types = (params[:filter] || "").split(",").map(&:to_i)
|
||||||
limit = params.fetch(:limit, 30).to_i
|
limit = params.fetch(:limit, 30).to_i
|
||||||
|
|
||||||
|
raise Discourse::NotFound unless guardian.can_see_profile?(user)
|
||||||
|
raise Discourse::NotFound unless guardian.can_see_user_actions?(user, action_types)
|
||||||
|
|
||||||
opts = {
|
opts = {
|
||||||
user_id: user.id,
|
user_id: user.id,
|
||||||
user: user,
|
user: user,
|
||||||
|
|
|
@ -55,6 +55,16 @@ class UserAction < ActiveRecord::Base
|
||||||
assigned: 16)
|
assigned: 16)
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def self.private_types
|
||||||
|
@private_types ||= [
|
||||||
|
WAS_LIKED,
|
||||||
|
RESPONSE,
|
||||||
|
MENTION,
|
||||||
|
QUOTE,
|
||||||
|
EDIT
|
||||||
|
]
|
||||||
|
end
|
||||||
|
|
||||||
def self.last_action_in_topic(user_id, topic_id)
|
def self.last_action_in_topic(user_id, topic_id)
|
||||||
UserAction.where(user_id: user_id,
|
UserAction.where(user_id: user_id,
|
||||||
target_topic_id: topic_id,
|
target_topic_id: topic_id,
|
||||||
|
|
|
@ -129,6 +129,11 @@ module UserGuardian
|
||||||
true
|
true
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def can_see_user_actions?(user, action_types)
|
||||||
|
return true if !@user.anonymous? && (@user.id == user.id || is_admin?)
|
||||||
|
(action_types & UserAction.private_types).empty?
|
||||||
|
end
|
||||||
|
|
||||||
def allowed_user_field_ids(user)
|
def allowed_user_field_ids(user)
|
||||||
@allowed_user_field_ids ||= {}
|
@allowed_user_field_ids ||= {}
|
||||||
|
|
||||||
|
|
|
@ -99,5 +99,54 @@ describe UserActionsController do
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
context "other users' activity" do
|
||||||
|
fab!(:another_user) { Fabricate(:user) }
|
||||||
|
|
||||||
|
UserAction.private_types.each do |action_type|
|
||||||
|
action_name = UserAction.types.key(action_type)
|
||||||
|
it "anonymous users cannot list other users' actions of type: #{action_name}" do
|
||||||
|
list_and_check(action_type, 404)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
UserAction.private_types.each do |action_type|
|
||||||
|
fab!(:user) { Fabricate(:user) }
|
||||||
|
action_name = UserAction.types.key(action_type)
|
||||||
|
|
||||||
|
it "logged in users cannot list other users' actions of type: #{action_name}" do
|
||||||
|
sign_in(user)
|
||||||
|
list_and_check(action_type, 404)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
UserAction.private_types.each do |action_type|
|
||||||
|
fab!(:moderator) { Fabricate(:moderator) }
|
||||||
|
action_name = UserAction.types.key(action_type)
|
||||||
|
|
||||||
|
it "moderators cannot list other users' actions of type: #{action_name}" do
|
||||||
|
sign_in(moderator)
|
||||||
|
list_and_check(action_type, 404)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
UserAction.private_types.each do |action_type|
|
||||||
|
fab!(:admin) { Fabricate(:admin) }
|
||||||
|
action_name = UserAction.types.key(action_type)
|
||||||
|
|
||||||
|
it "admins can list other users' actions of type: #{action_name}" do
|
||||||
|
sign_in(admin)
|
||||||
|
list_and_check(action_type, 200)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def list_and_check(action_type, expected_response)
|
||||||
|
get "/user_actions.json", params: {
|
||||||
|
filter: action_type,
|
||||||
|
username: another_user.username
|
||||||
|
}
|
||||||
|
|
||||||
|
expect(response.status).to eq(expected_response)
|
||||||
|
end
|
||||||
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
Loading…
Reference in New Issue