You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
mastodon/spec/requests/api/web/push_subscriptions_spec.rb

202 lines
5.7 KiB
Ruby

# frozen_string_literal: true
require 'rails_helper'
RSpec.describe 'API Web Push Subscriptions' do
let(:create_payload) do
{
subscription: {
endpoint: 'https://fcm.googleapis.com/fcm/send/fiuH06a27qE:APA91bHnSiGcLwdaxdyqVXNDR9w1NlztsHb6lyt5WDKOC_Z_Q8BlFxQoR8tWFSXUIDdkyw0EdvxTu63iqamSaqVSevW5LfoFwojws8XYDXv_NRRLH6vo2CdgiN4jgHv5VLt2A8ah6lUX',
keys: {
p256dh: 'BEm_a0bdPDhf0SOsrnB2-ategf1hHoCnpXgQsFj5JCkcoMrMt2WHoPfEYOYPzOIs9mZE8ZUaD7VA5vouy0kEkr8=',
auth: 'eH_C8rq2raXqlcBVDa1gLg==',
},
standard: standard,
},
}
end
let(:alerts_payload) do
{
data: {
policy: 'all',
alerts: {
follow: true,
follow_request: false,
favourite: false,
reblog: true,
mention: false,
poll: true,
status: false,
},
},
}
end
let(:standard) { '1' }
describe 'DELETE /api/web/push_subscriptions/:id' do
subject { delete api_web_push_subscription_path(token) }
context 'when the subscription exists' do
let!(:web_push_subscription) do
Fabricate(:web_push_subscription)
end
let(:token) do
web_push_subscription.generate_token_for(:unsubscribe)
end
it 'deletes the subscription' do
expect { subject }
.to change(Web::PushSubscription, :count).by(-1)
expect(response).to have_http_status(200)
end
end
context 'when the subscription does not exist' do
let(:web_push_subscription) do
Fabricate(:web_push_subscription)
end
let(:token) do
web_push_subscription.generate_token_for(:unsubscribe)
end
before do
token # memoize before destroying the record
web_push_subscription.destroy!
end
it 'does nothing' do
subject
expect(response).to have_http_status(200)
end
end
context 'when the token is invalid' do
let(:token) { 'invalid--invalid' }
it 'does nothing' do
subject
expect(response).to have_http_status(200)
end
end
end
describe 'POST /api/web/push_subscriptions' do
before { sign_in(user) }
let(:user) { Fabricate :user }
it 'gracefully handles invalid nested params' do
post api_web_push_subscriptions_path, params: { subscription: 'invalid' }
expect(response)
.to have_http_status(400)
end
it 'saves push subscriptions with valid params' do
post api_web_push_subscriptions_path, params: create_payload
expect(response)
.to have_http_status(200)
expect(created_push_subscription)
.to have_attributes(
endpoint: eq(create_payload[:subscription][:endpoint]),
key_p256dh: eq(create_payload[:subscription][:keys][:p256dh]),
key_auth: eq(create_payload[:subscription][:keys][:auth])
)
.and be_standard
expect(user.session_activations.first.web_push_subscription)
.to eq(created_push_subscription)
end
context 'when standard is provided as false value' do
let(:standard) { '0' }
it 'saves push subscription with standard as false' do
post api_web_push_subscriptions_path, params: create_payload
expect(created_push_subscription)
.to_not be_standard
end
end
context 'with a user who has a session with a prior subscription' do
before do
# Trigger creation of a `SessionActivation` for the user so that the
# prior_subscription setup and verification works as expected
get about_path
end
let!(:prior_subscription) { Fabricate(:web_push_subscription, user:, session_activation: user.session_activations.last) }
it 'destroys prior subscription when creating new one' do
post api_web_push_subscriptions_path, params: create_payload
expect(response)
.to have_http_status(200)
expect { prior_subscription.reload }
.to raise_error(ActiveRecord::RecordNotFound)
end
end
context 'with initial data' do
it 'saves alert settings' do
post api_web_push_subscriptions_path, params: create_payload.merge(alerts_payload)
expect(response)
.to have_http_status(200)
expect(created_push_subscription.data['policy'])
.to eq 'all'
alert_types.each do |type|
expect(created_push_subscription.data['alerts'][type])
.to eq(alerts_payload[:data][:alerts][type.to_sym].to_s)
end
end
end
end
describe 'PUT /api/web/push_subscriptions' do
before { sign_in Fabricate :user }
let(:subscription) { Fabricate :web_push_subscription }
it 'gracefully handles invalid nested params' do
put api_web_push_subscription_path(subscription), params: { data: 'invalid' }
expect(response)
.to have_http_status(400)
end
it 'changes existing alert settings' do
# Create record this way to correctly associate a `SessionActivation`
# during full POST->create cycle
post api_web_push_subscriptions_path params: create_payload
expect(response)
.to have_http_status(200)
put api_web_push_subscription_path(created_push_subscription), params: alerts_payload
expect(created_push_subscription.data['policy'])
.to eq 'all'
alert_types.each do |type|
expect(created_push_subscription.data['alerts'][type])
.to eq(alerts_payload[:data][:alerts][type.to_sym].to_s)
end
end
end
def created_push_subscription
Web::PushSubscription
.find_by(endpoint: create_payload[:subscription][:endpoint])
end
def alert_types
Notification::LEGACY_TYPE_CLASS_MAP.values.map(&:to_s)
end
end