forked from cybrespace/mastodon
Add `policy` param to `POST /api/v1/push/subscriptions` (#16040)
With possible values `all`, `followed`, `follower`, and `none`, control from whom notifications will generate a Web Push alert
This commit is contained in:
parent
c968d22ee9
commit
ce2148c571
|
@ -3,13 +3,13 @@
|
||||||
class Api::V1::Push::SubscriptionsController < Api::BaseController
|
class Api::V1::Push::SubscriptionsController < Api::BaseController
|
||||||
before_action -> { doorkeeper_authorize! :push }
|
before_action -> { doorkeeper_authorize! :push }
|
||||||
before_action :require_user!
|
before_action :require_user!
|
||||||
before_action :set_web_push_subscription
|
before_action :set_push_subscription
|
||||||
before_action :check_web_push_subscription, only: [:show, :update]
|
before_action :check_push_subscription, only: [:show, :update]
|
||||||
|
|
||||||
def create
|
def create
|
||||||
@web_subscription&.destroy!
|
@push_subscription&.destroy!
|
||||||
|
|
||||||
@web_subscription = ::Web::PushSubscription.create!(
|
@push_subscription = Web::PushSubscription.create!(
|
||||||
endpoint: subscription_params[:endpoint],
|
endpoint: subscription_params[:endpoint],
|
||||||
key_p256dh: subscription_params[:keys][:p256dh],
|
key_p256dh: subscription_params[:keys][:p256dh],
|
||||||
key_auth: subscription_params[:keys][:auth],
|
key_auth: subscription_params[:keys][:auth],
|
||||||
|
@ -18,31 +18,31 @@ class Api::V1::Push::SubscriptionsController < Api::BaseController
|
||||||
access_token_id: doorkeeper_token.id
|
access_token_id: doorkeeper_token.id
|
||||||
)
|
)
|
||||||
|
|
||||||
render json: @web_subscription, serializer: REST::WebPushSubscriptionSerializer
|
render json: @push_subscription, serializer: REST::WebPushSubscriptionSerializer
|
||||||
end
|
end
|
||||||
|
|
||||||
def show
|
def show
|
||||||
render json: @web_subscription, serializer: REST::WebPushSubscriptionSerializer
|
render json: @push_subscription, serializer: REST::WebPushSubscriptionSerializer
|
||||||
end
|
end
|
||||||
|
|
||||||
def update
|
def update
|
||||||
@web_subscription.update!(data: data_params)
|
@push_subscription.update!(data: data_params)
|
||||||
render json: @web_subscription, serializer: REST::WebPushSubscriptionSerializer
|
render json: @push_subscription, serializer: REST::WebPushSubscriptionSerializer
|
||||||
end
|
end
|
||||||
|
|
||||||
def destroy
|
def destroy
|
||||||
@web_subscription&.destroy!
|
@push_subscription&.destroy!
|
||||||
render_empty
|
render_empty
|
||||||
end
|
end
|
||||||
|
|
||||||
private
|
private
|
||||||
|
|
||||||
def set_web_push_subscription
|
def set_push_subscription
|
||||||
@web_subscription = ::Web::PushSubscription.find_by(access_token_id: doorkeeper_token.id)
|
@push_subscription = Web::PushSubscription.find_by(access_token_id: doorkeeper_token.id)
|
||||||
end
|
end
|
||||||
|
|
||||||
def check_web_push_subscription
|
def check_push_subscription
|
||||||
not_found if @web_subscription.nil?
|
not_found if @push_subscription.nil?
|
||||||
end
|
end
|
||||||
|
|
||||||
def subscription_params
|
def subscription_params
|
||||||
|
@ -52,6 +52,6 @@ class Api::V1::Push::SubscriptionsController < Api::BaseController
|
||||||
def data_params
|
def data_params
|
||||||
return {} if params[:data].blank?
|
return {} if params[:data].blank?
|
||||||
|
|
||||||
params.require(:data).permit(alerts: [:follow, :follow_request, :favourite, :reblog, :mention, :poll, :status])
|
params.require(:data).permit(:policy, alerts: [:follow, :follow_request, :favourite, :reblog, :mention, :poll, :status])
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -2,6 +2,7 @@
|
||||||
|
|
||||||
class Api::Web::PushSubscriptionsController < Api::Web::BaseController
|
class Api::Web::PushSubscriptionsController < Api::Web::BaseController
|
||||||
before_action :require_user!
|
before_action :require_user!
|
||||||
|
before_action :set_push_subscription, only: :update
|
||||||
|
|
||||||
def create
|
def create
|
||||||
active_session = current_session
|
active_session = current_session
|
||||||
|
@ -15,9 +16,11 @@ class Api::Web::PushSubscriptionsController < Api::Web::BaseController
|
||||||
alerts_enabled = active_session.detection.device.mobile? || active_session.detection.device.tablet?
|
alerts_enabled = active_session.detection.device.mobile? || active_session.detection.device.tablet?
|
||||||
|
|
||||||
data = {
|
data = {
|
||||||
|
policy: 'all',
|
||||||
|
|
||||||
alerts: {
|
alerts: {
|
||||||
follow: alerts_enabled,
|
follow: alerts_enabled,
|
||||||
follow_request: false,
|
follow_request: alerts_enabled,
|
||||||
favourite: alerts_enabled,
|
favourite: alerts_enabled,
|
||||||
reblog: alerts_enabled,
|
reblog: alerts_enabled,
|
||||||
mention: alerts_enabled,
|
mention: alerts_enabled,
|
||||||
|
@ -28,7 +31,7 @@ class Api::Web::PushSubscriptionsController < Api::Web::BaseController
|
||||||
|
|
||||||
data.deep_merge!(data_params) if params[:data]
|
data.deep_merge!(data_params) if params[:data]
|
||||||
|
|
||||||
web_subscription = ::Web::PushSubscription.create!(
|
push_subscription = ::Web::PushSubscription.create!(
|
||||||
endpoint: subscription_params[:endpoint],
|
endpoint: subscription_params[:endpoint],
|
||||||
key_p256dh: subscription_params[:keys][:p256dh],
|
key_p256dh: subscription_params[:keys][:p256dh],
|
||||||
key_auth: subscription_params[:keys][:auth],
|
key_auth: subscription_params[:keys][:auth],
|
||||||
|
@ -37,27 +40,27 @@ class Api::Web::PushSubscriptionsController < Api::Web::BaseController
|
||||||
access_token_id: active_session.access_token_id
|
access_token_id: active_session.access_token_id
|
||||||
)
|
)
|
||||||
|
|
||||||
active_session.update!(web_push_subscription: web_subscription)
|
active_session.update!(web_push_subscription: push_subscription)
|
||||||
|
|
||||||
render json: web_subscription, serializer: REST::WebPushSubscriptionSerializer
|
render json: push_subscription, serializer: REST::WebPushSubscriptionSerializer
|
||||||
end
|
end
|
||||||
|
|
||||||
def update
|
def update
|
||||||
params.require([:id])
|
@push_subscription.update!(data: data_params)
|
||||||
|
render json: @push_subscription, serializer: REST::WebPushSubscriptionSerializer
|
||||||
web_subscription = ::Web::PushSubscription.find(params[:id])
|
|
||||||
web_subscription.update!(data: data_params)
|
|
||||||
|
|
||||||
render json: web_subscription, serializer: REST::WebPushSubscriptionSerializer
|
|
||||||
end
|
end
|
||||||
|
|
||||||
private
|
private
|
||||||
|
|
||||||
|
def set_push_subscription
|
||||||
|
@push_subscription = ::Web::PushSubscription.find(params[:id])
|
||||||
|
end
|
||||||
|
|
||||||
def subscription_params
|
def subscription_params
|
||||||
@subscription_params ||= params.require(:subscription).permit(:endpoint, keys: [:auth, :p256dh])
|
@subscription_params ||= params.require(:subscription).permit(:endpoint, keys: [:auth, :p256dh])
|
||||||
end
|
end
|
||||||
|
|
||||||
def data_params
|
def data_params
|
||||||
@data_params ||= params.require(:data).permit(alerts: [:follow, :follow_request, :favourite, :reblog, :mention, :poll, :status])
|
@data_params ||= params.require(:data).permit(:policy, alerts: [:follow, :follow_request, :favourite, :reblog, :mention, :poll, :status])
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -47,7 +47,7 @@ class Web::PushSubscription < ApplicationRecord
|
||||||
end
|
end
|
||||||
|
|
||||||
def pushable?(notification)
|
def pushable?(notification)
|
||||||
ActiveModel::Type::Boolean.new.cast(data&.dig('alerts', notification.type.to_s))
|
policy_allows_notification?(notification) && alert_enabled_for_notification_type?(notification)
|
||||||
end
|
end
|
||||||
|
|
||||||
def associated_user
|
def associated_user
|
||||||
|
@ -100,4 +100,25 @@ class Web::PushSubscription < ApplicationRecord
|
||||||
def contact_email
|
def contact_email
|
||||||
@contact_email ||= ::Setting.site_contact_email
|
@contact_email ||= ::Setting.site_contact_email
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def alert_enabled_for_notification_type?(notification)
|
||||||
|
truthy?(data&.dig('alerts', notification.type.to_s))
|
||||||
|
end
|
||||||
|
|
||||||
|
def policy_allows_notification?(notification)
|
||||||
|
case data&.dig('policy')
|
||||||
|
when nil, 'all'
|
||||||
|
true
|
||||||
|
when 'none'
|
||||||
|
false
|
||||||
|
when 'followed'
|
||||||
|
notification.account.following?(notification.from_account)
|
||||||
|
when 'follower'
|
||||||
|
notification.from_account.following?(notification.account)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def truthy?(val)
|
||||||
|
ActiveModel::Type::Boolean.new.cast(val)
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -27,20 +27,27 @@ describe Api::V1::Push::SubscriptionsController do
|
||||||
let(:alerts_payload) do
|
let(:alerts_payload) do
|
||||||
{
|
{
|
||||||
data: {
|
data: {
|
||||||
|
policy: 'all',
|
||||||
|
|
||||||
alerts: {
|
alerts: {
|
||||||
follow: true,
|
follow: true,
|
||||||
|
follow_request: true,
|
||||||
favourite: false,
|
favourite: false,
|
||||||
reblog: true,
|
reblog: true,
|
||||||
mention: false,
|
mention: false,
|
||||||
|
poll: true,
|
||||||
|
status: false,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}.with_indifferent_access
|
}.with_indifferent_access
|
||||||
end
|
end
|
||||||
|
|
||||||
describe 'POST #create' do
|
describe 'POST #create' do
|
||||||
it 'saves push subscriptions' do
|
before do
|
||||||
post :create, params: create_payload
|
post :create, params: create_payload
|
||||||
|
end
|
||||||
|
|
||||||
|
it 'saves push subscriptions' do
|
||||||
push_subscription = Web::PushSubscription.find_by(endpoint: create_payload[:subscription][:endpoint])
|
push_subscription = Web::PushSubscription.find_by(endpoint: create_payload[:subscription][:endpoint])
|
||||||
|
|
||||||
expect(push_subscription.endpoint).to eq(create_payload[:subscription][:endpoint])
|
expect(push_subscription.endpoint).to eq(create_payload[:subscription][:endpoint])
|
||||||
|
@ -52,31 +59,34 @@ describe Api::V1::Push::SubscriptionsController do
|
||||||
|
|
||||||
it 'replaces old subscription on repeat calls' do
|
it 'replaces old subscription on repeat calls' do
|
||||||
post :create, params: create_payload
|
post :create, params: create_payload
|
||||||
post :create, params: create_payload
|
|
||||||
|
|
||||||
expect(Web::PushSubscription.where(endpoint: create_payload[:subscription][:endpoint]).count).to eq 1
|
expect(Web::PushSubscription.where(endpoint: create_payload[:subscription][:endpoint]).count).to eq 1
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
describe 'PUT #update' do
|
describe 'PUT #update' do
|
||||||
it 'changes alert settings' do
|
before do
|
||||||
post :create, params: create_payload
|
post :create, params: create_payload
|
||||||
put :update, params: alerts_payload
|
put :update, params: alerts_payload
|
||||||
|
end
|
||||||
|
|
||||||
|
it 'changes alert settings' do
|
||||||
push_subscription = Web::PushSubscription.find_by(endpoint: create_payload[:subscription][:endpoint])
|
push_subscription = Web::PushSubscription.find_by(endpoint: create_payload[:subscription][:endpoint])
|
||||||
|
|
||||||
expect(push_subscription.data.dig('alerts', 'follow')).to eq(alerts_payload[:data][:alerts][:follow].to_s)
|
expect(push_subscription.data['policy']).to eq(alerts_payload[:data][:policy])
|
||||||
expect(push_subscription.data.dig('alerts', 'favourite')).to eq(alerts_payload[:data][:alerts][:favourite].to_s)
|
|
||||||
expect(push_subscription.data.dig('alerts', 'reblog')).to eq(alerts_payload[:data][:alerts][:reblog].to_s)
|
%w(follow follow_request favourite reblog mention poll status).each do |type|
|
||||||
expect(push_subscription.data.dig('alerts', 'mention')).to eq(alerts_payload[:data][:alerts][:mention].to_s)
|
expect(push_subscription.data['alerts'][type]).to eq(alerts_payload[:data][:alerts][type.to_sym].to_s)
|
||||||
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
describe 'DELETE #destroy' do
|
describe 'DELETE #destroy' do
|
||||||
it 'removes the subscription' do
|
before do
|
||||||
post :create, params: create_payload
|
post :create, params: create_payload
|
||||||
delete :destroy
|
delete :destroy
|
||||||
|
end
|
||||||
|
|
||||||
|
it 'removes the subscription' do
|
||||||
expect(Web::PushSubscription.find_by(endpoint: create_payload[:subscription][:endpoint])).to be_nil
|
expect(Web::PushSubscription.find_by(endpoint: create_payload[:subscription][:endpoint])).to be_nil
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -22,11 +22,16 @@ describe Api::Web::PushSubscriptionsController do
|
||||||
let(:alerts_payload) do
|
let(:alerts_payload) do
|
||||||
{
|
{
|
||||||
data: {
|
data: {
|
||||||
|
policy: 'all',
|
||||||
|
|
||||||
alerts: {
|
alerts: {
|
||||||
follow: true,
|
follow: true,
|
||||||
|
follow_request: false,
|
||||||
favourite: false,
|
favourite: false,
|
||||||
reblog: true,
|
reblog: true,
|
||||||
mention: false,
|
mention: false,
|
||||||
|
poll: true,
|
||||||
|
status: false,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -59,10 +64,11 @@ describe Api::Web::PushSubscriptionsController do
|
||||||
|
|
||||||
push_subscription = Web::PushSubscription.find_by(endpoint: create_payload[:subscription][:endpoint])
|
push_subscription = Web::PushSubscription.find_by(endpoint: create_payload[:subscription][:endpoint])
|
||||||
|
|
||||||
expect(push_subscription.data['alerts']['follow']).to eq(alerts_payload[:data][:alerts][:follow].to_s)
|
expect(push_subscription.data['policy']).to eq 'all'
|
||||||
expect(push_subscription.data['alerts']['favourite']).to eq(alerts_payload[:data][:alerts][:favourite].to_s)
|
|
||||||
expect(push_subscription.data['alerts']['reblog']).to eq(alerts_payload[:data][:alerts][:reblog].to_s)
|
%w(follow follow_request favourite reblog mention poll status).each do |type|
|
||||||
expect(push_subscription.data['alerts']['mention']).to eq(alerts_payload[:data][:alerts][:mention].to_s)
|
expect(push_subscription.data['alerts'][type]).to eq(alerts_payload[:data][:alerts][type.to_sym].to_s)
|
||||||
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
@ -81,10 +87,11 @@ describe Api::Web::PushSubscriptionsController do
|
||||||
|
|
||||||
push_subscription = Web::PushSubscription.find_by(endpoint: create_payload[:subscription][:endpoint])
|
push_subscription = Web::PushSubscription.find_by(endpoint: create_payload[:subscription][:endpoint])
|
||||||
|
|
||||||
expect(push_subscription.data['alerts']['follow']).to eq(alerts_payload[:data][:alerts][:follow].to_s)
|
expect(push_subscription.data['policy']).to eq 'all'
|
||||||
expect(push_subscription.data['alerts']['favourite']).to eq(alerts_payload[:data][:alerts][:favourite].to_s)
|
|
||||||
expect(push_subscription.data['alerts']['reblog']).to eq(alerts_payload[:data][:alerts][:reblog].to_s)
|
%w(follow follow_request favourite reblog mention poll status).each do |type|
|
||||||
expect(push_subscription.data['alerts']['mention']).to eq(alerts_payload[:data][:alerts][:mention].to_s)
|
expect(push_subscription.data['alerts'][type]).to eq(alerts_payload[:data][:alerts][type.to_sym].to_s)
|
||||||
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -1,16 +1,94 @@
|
||||||
require 'rails_helper'
|
require 'rails_helper'
|
||||||
|
|
||||||
RSpec.describe Web::PushSubscription, type: :model do
|
RSpec.describe Web::PushSubscription, type: :model do
|
||||||
let(:alerts) { { mention: true, reblog: false, follow: true, follow_request: false, favourite: true } }
|
let(:account) { Fabricate(:account) }
|
||||||
let(:push_subscription) { Web::PushSubscription.new(data: { alerts: alerts }) }
|
|
||||||
|
let(:policy) { 'all' }
|
||||||
|
|
||||||
|
let(:data) do
|
||||||
|
{
|
||||||
|
policy: policy,
|
||||||
|
|
||||||
|
alerts: {
|
||||||
|
mention: true,
|
||||||
|
reblog: false,
|
||||||
|
follow: true,
|
||||||
|
follow_request: false,
|
||||||
|
favourite: true,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
end
|
||||||
|
|
||||||
|
subject { described_class.new(data: data) }
|
||||||
|
|
||||||
describe '#pushable?' do
|
describe '#pushable?' do
|
||||||
it 'obeys alert settings' do
|
let(:notification_type) { :mention }
|
||||||
expect(push_subscription.send(:pushable?, Notification.new(activity_type: 'Mention'))).to eq true
|
let(:notification) { Fabricate(:notification, account: account, type: notification_type) }
|
||||||
expect(push_subscription.send(:pushable?, Notification.new(activity_type: 'Status'))).to eq false
|
|
||||||
expect(push_subscription.send(:pushable?, Notification.new(activity_type: 'Follow'))).to eq true
|
%i(mention reblog follow follow_request favourite).each do |type|
|
||||||
expect(push_subscription.send(:pushable?, Notification.new(activity_type: 'FollowRequest'))).to eq false
|
context "when notification is a #{type}" do
|
||||||
expect(push_subscription.send(:pushable?, Notification.new(activity_type: 'Favourite'))).to eq true
|
let(:notification_type) { type }
|
||||||
|
|
||||||
|
it "returns boolean corresonding to alert setting" do
|
||||||
|
expect(subject.pushable?(notification)).to eq data[:alerts][type]
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
context 'when policy is all' do
|
||||||
|
let(:policy) { 'all' }
|
||||||
|
|
||||||
|
it 'returns true' do
|
||||||
|
expect(subject.pushable?(notification)).to eq true
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
context 'when policy is none' do
|
||||||
|
let(:policy) { 'none' }
|
||||||
|
|
||||||
|
it 'returns false' do
|
||||||
|
expect(subject.pushable?(notification)).to eq false
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
context 'when policy is followed' do
|
||||||
|
let(:policy) { 'followed' }
|
||||||
|
|
||||||
|
context 'and notification is from someone you follow' do
|
||||||
|
before do
|
||||||
|
account.follow!(notification.from_account)
|
||||||
|
end
|
||||||
|
|
||||||
|
it 'returns true' do
|
||||||
|
expect(subject.pushable?(notification)).to eq true
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
context 'and notification is not from someone you follow' do
|
||||||
|
it 'returns false' do
|
||||||
|
expect(subject.pushable?(notification)).to eq false
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
context 'when policy is follower' do
|
||||||
|
let(:policy) { 'follower' }
|
||||||
|
|
||||||
|
context 'and notification is from someone who follows you' do
|
||||||
|
before do
|
||||||
|
notification.from_account.follow!(account)
|
||||||
|
end
|
||||||
|
|
||||||
|
it 'returns true' do
|
||||||
|
expect(subject.pushable?(notification)).to eq true
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
context 'and notification is not from someone who follows you' do
|
||||||
|
it 'returns false' do
|
||||||
|
expect(subject.pushable?(notification)).to eq false
|
||||||
|
end
|
||||||
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
Loading…
Reference in New Issue