Skip to content

Commit

Permalink
Add endpoints for unread notifications count (mastodon#31191)
Browse files Browse the repository at this point in the history
  • Loading branch information
ClearlyClaire authored Jul 30, 2024
1 parent 2ce99c5 commit 598ae4f
Show file tree
Hide file tree
Showing 6 changed files with 186 additions and 2 deletions.
4 changes: 2 additions & 2 deletions app/controllers/api/base_controller.rb
Original file line number Diff line number Diff line change
Expand Up @@ -30,10 +30,10 @@ def doorkeeper_forbidden_render_options(*)

protected

def limit_param(default_limit)
def limit_param(default_limit, max_limit = nil)
return default_limit unless params[:limit]

[params[:limit].to_i.abs, default_limit * 2].min
[params[:limit].to_i.abs, max_limit || (default_limit * 2)].min
end

def params_slice(*keys)
Expand Down
14 changes: 14 additions & 0 deletions app/controllers/api/v1/notifications_controller.rb
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,8 @@ class Api::V1::NotificationsController < Api::BaseController
after_action :insert_pagination_headers, only: :index

DEFAULT_NOTIFICATIONS_LIMIT = 40
DEFAULT_NOTIFICATIONS_COUNT_LIMIT = 100
MAX_NOTIFICATIONS_COUNT_LIMIT = 1_000

def index
with_read_replica do
Expand All @@ -17,6 +19,14 @@ def index
render json: @notifications, each_serializer: REST::NotificationSerializer, relationships: @relationships
end

def unread_count
limit = limit_param(DEFAULT_NOTIFICATIONS_COUNT_LIMIT, MAX_NOTIFICATIONS_COUNT_LIMIT)

with_read_replica do
render json: { count: browserable_account_notifications.paginate_by_min_id(limit, notification_marker&.last_read_id).count }
end
end

def show
@notification = current_account.notifications.without_suspended.find(params[:id])
render json: @notification, serializer: REST::NotificationSerializer
Expand Down Expand Up @@ -54,6 +64,10 @@ def browserable_account_notifications
)
end

def notification_marker
current_user.markers.find_by(timeline: 'notifications')
end

def target_statuses_from_notifications
@notifications.reject { |notification| notification.target_status.nil? }.map(&:target_status)
end
Expand Down
14 changes: 14 additions & 0 deletions app/controllers/api/v2_alpha/notifications_controller.rb
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,8 @@ class Api::V2Alpha::NotificationsController < Api::BaseController
after_action :insert_pagination_headers, only: :index

DEFAULT_NOTIFICATIONS_LIMIT = 40
DEFAULT_NOTIFICATIONS_COUNT_LIMIT = 100
MAX_NOTIFICATIONS_COUNT_LIMIT = 1_000

def index
with_read_replica do
Expand Down Expand Up @@ -35,6 +37,14 @@ def index
end
end

def unread_count
limit = limit_param(DEFAULT_NOTIFICATIONS_COUNT_LIMIT, MAX_NOTIFICATIONS_COUNT_LIMIT)

with_read_replica do
render json: { count: browserable_account_notifications.paginate_groups_by_min_id(limit, min_id: notification_marker&.last_read_id).count }
end
end

def show
@notification = current_account.notifications.without_suspended.find_by!(group_key: params[:id])
render json: NotificationGroup.from_notification(@notification), serializer: REST::NotificationGroupSerializer
Expand Down Expand Up @@ -92,6 +102,10 @@ def browserable_account_notifications
)
end

def notification_marker
current_user.markers.find_by(timeline: 'notifications')
end

def target_statuses_from_notifications
@notifications.filter_map(&:target_status)
end
Expand Down
2 changes: 2 additions & 0 deletions config/routes/api.rb
Original file line number Diff line number Diff line change
Expand Up @@ -167,6 +167,7 @@
resources :notifications, only: [:index, :show] do
collection do
post :clear
get :unread_count
end

member do
Expand Down Expand Up @@ -336,6 +337,7 @@
resources :notifications, only: [:index, :show] do
collection do
post :clear
get :unread_count
end

member do
Expand Down
77 changes: 77 additions & 0 deletions spec/requests/api/v1/notifications_spec.rb
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,83 @@
let(:scopes) { 'read:notifications write:notifications' }
let(:headers) { { 'Authorization' => "Bearer #{token.token}" } }

describe 'GET /api/v1/notifications/unread_count', :inline_jobs do
subject do
get '/api/v1/notifications/unread_count', headers: headers, params: params
end

let(:params) { {} }

before do
first_status = PostStatusService.new.call(user.account, text: 'Test')
ReblogService.new.call(Fabricate(:account), first_status)
PostStatusService.new.call(Fabricate(:account), text: 'Hello @alice')
FavouriteService.new.call(Fabricate(:account), first_status)
FavouriteService.new.call(Fabricate(:account), first_status)
FollowService.new.call(Fabricate(:account), user.account)
end

it_behaves_like 'forbidden for wrong scope', 'write write:notifications'

context 'with no options' do
it 'returns expected notifications count' do
subject

expect(response).to have_http_status(200)
expect(body_as_json[:count]).to eq 5
end
end

context 'with a read marker' do
before do
id = user.account.notifications.browserable.order(id: :desc).offset(2).first.id
user.markers.create!(timeline: 'notifications', last_read_id: id)
end

it 'returns expected notifications count' do
subject

expect(response).to have_http_status(200)
expect(body_as_json[:count]).to eq 2
end
end

context 'with exclude_types param' do
let(:params) { { exclude_types: %w(mention) } }

it 'returns expected notifications count' do
subject

expect(response).to have_http_status(200)
expect(body_as_json[:count]).to eq 4
end
end

context 'with a user-provided limit' do
let(:params) { { limit: 2 } }

it 'returns a capped value' do
subject

expect(response).to have_http_status(200)
expect(body_as_json[:count]).to eq 2
end
end

context 'when there are more notifications than the limit' do
before do
stub_const('Api::V1::NotificationsController::DEFAULT_NOTIFICATIONS_COUNT_LIMIT', 2)
end

it 'returns a capped value' do
subject

expect(response).to have_http_status(200)
expect(body_as_json[:count]).to eq Api::V1::NotificationsController::DEFAULT_NOTIFICATIONS_COUNT_LIMIT
end
end
end

describe 'GET /api/v1/notifications', :inline_jobs do
subject do
get '/api/v1/notifications', headers: headers, params: params
Expand Down
77 changes: 77 additions & 0 deletions spec/requests/api/v2_alpha/notifications_spec.rb
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,83 @@
let(:scopes) { 'read:notifications write:notifications' }
let(:headers) { { 'Authorization' => "Bearer #{token.token}" } }

describe 'GET /api/v2_alpha/notifications/unread_count', :inline_jobs do
subject do
get '/api/v2_alpha/notifications/unread_count', headers: headers, params: params
end

let(:params) { {} }

before do
first_status = PostStatusService.new.call(user.account, text: 'Test')
ReblogService.new.call(Fabricate(:account), first_status)
PostStatusService.new.call(Fabricate(:account), text: 'Hello @alice')
FavouriteService.new.call(Fabricate(:account), first_status)
FavouriteService.new.call(Fabricate(:account), first_status)
FollowService.new.call(Fabricate(:account), user.account)
end

it_behaves_like 'forbidden for wrong scope', 'write write:notifications'

context 'with no options' do
it 'returns expected notifications count' do
subject

expect(response).to have_http_status(200)
expect(body_as_json[:count]).to eq 4
end
end

context 'with a read marker' do
before do
id = user.account.notifications.browserable.order(id: :desc).offset(2).first.id
user.markers.create!(timeline: 'notifications', last_read_id: id)
end

it 'returns expected notifications count' do
subject

expect(response).to have_http_status(200)
expect(body_as_json[:count]).to eq 2
end
end

context 'with exclude_types param' do
let(:params) { { exclude_types: %w(mention) } }

it 'returns expected notifications count' do
subject

expect(response).to have_http_status(200)
expect(body_as_json[:count]).to eq 3
end
end

context 'with a user-provided limit' do
let(:params) { { limit: 2 } }

it 'returns a capped value' do
subject

expect(response).to have_http_status(200)
expect(body_as_json[:count]).to eq 2
end
end

context 'when there are more notifications than the limit' do
before do
stub_const('Api::V2Alpha::NotificationsController::DEFAULT_NOTIFICATIONS_COUNT_LIMIT', 2)
end

it 'returns a capped value' do
subject

expect(response).to have_http_status(200)
expect(body_as_json[:count]).to eq Api::V2Alpha::NotificationsController::DEFAULT_NOTIFICATIONS_COUNT_LIMIT
end
end
end

describe 'GET /api/v2_alpha/notifications', :inline_jobs do
subject do
get '/api/v2_alpha/notifications', headers: headers, params: params
Expand Down

0 comments on commit 598ae4f

Please sign in to comment.