123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168 |
- # frozen_string_literal: true
- require 'rails_helper'
- RSpec.describe 'API V1 Push Subscriptions' do
- let(:user) { Fabricate(:user) }
- let(:endpoint) { 'https://fcm.googleapis.com/fcm/send/fiuH06a27qE:APA91bHnSiGcLwdaxdyqVXNDR9w1NlztsHb6lyt5WDKOC_Z_Q8BlFxQoR8tWFSXUIDdkyw0EdvxTu63iqamSaqVSevW5LfoFwojws8XYDXv_NRRLH6vo2CdgiN4jgHv5VLt2A8ah6lUX' }
- let(:keys) do
- {
- p256dh: 'BEm_a0bdPDhf0SOsrnB2-ategf1hHoCnpXgQsFj5JCkcoMrMt2WHoPfEYOYPzOIs9mZE8ZUaD7VA5vouy0kEkr8=',
- auth: 'eH_C8rq2raXqlcBVDa1gLg==',
- }
- end
- let(:create_payload) do
- {
- subscription: {
- endpoint: endpoint,
- keys: keys,
- },
- }.with_indifferent_access
- end
- let(:alerts_payload) do
- {
- data: {
- policy: 'all',
- alerts: {
- follow: true,
- follow_request: true,
- favourite: false,
- reblog: true,
- mention: false,
- poll: true,
- status: false,
- },
- },
- }.with_indifferent_access
- end
- let(:scopes) { 'push' }
- let(:token) { Fabricate(:accessible_access_token, resource_owner_id: user.id, scopes: scopes) }
- let(:headers) { { 'Authorization' => "Bearer #{token.token}" } }
- shared_examples 'validation error' do
- it 'returns a validation error' do
- subject
- expect(response).to have_http_status(422)
- expect(response.content_type)
- .to start_with('application/json')
- expect(endpoint_push_subscriptions.count).to eq(0)
- expect(endpoint_push_subscription).to be_nil
- end
- end
- describe 'POST /api/v1/push/subscription' do
- subject { post '/api/v1/push/subscription', params: create_payload, headers: headers }
- it 'saves push subscriptions and returns expected JSON' do
- subject
- expect(endpoint_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]),
- user_id: eq(user.id),
- access_token_id: eq(token.id)
- )
- expect(response.parsed_body.with_indifferent_access)
- .to include(
- { endpoint: create_payload[:subscription][:endpoint], alerts: {}, policy: 'all' }
- )
- end
- it 'replaces old subscription on repeat calls' do
- 2.times { subject }
- expect(endpoint_push_subscriptions.count)
- .to eq(1)
- end
- context 'with invalid endpoint URL' do
- let(:endpoint) { 'app://example.foo' }
- it_behaves_like 'validation error'
- end
- context 'with invalid p256dh key' do
- let(:keys) do
- {
- p256dh: 'BEm_invalidf0SOsrnB2-ategf1hHoCnpXgQsFj5JCkcoMrMt2WHoPfEYOYPzOIs9mZE8ZUaD7VA5vouy0kEkr8=',
- auth: 'eH_C8rq2raXqlcBVDa1gLg==',
- }
- end
- it_behaves_like 'validation error'
- end
- context 'with invalid base64 p256dh key' do
- let(:keys) do
- {
- p256dh: 'not base64',
- auth: 'eH_C8rq2raXqlcBVDa1gLg==',
- }
- end
- it_behaves_like 'validation error'
- end
- end
- describe 'PUT /api/v1/push/subscription' do
- subject { put '/api/v1/push/subscription', params: alerts_payload, headers: headers }
- before { create_subscription_with_token }
- it 'changes data policy and alert settings and returns expected JSON' do
- expect { subject }
- .to change { endpoint_push_subscription.reload.data }
- .from(nil)
- .to(include('policy' => alerts_payload[:data][:policy]))
- %w(follow follow_request favourite reblog mention poll status).each do |type|
- expect(endpoint_push_subscription.data['alerts']).to include(
- type.to_s => eq(alerts_payload[:data][:alerts][type.to_sym].to_s)
- )
- end
- expect(response.parsed_body.with_indifferent_access)
- .to include(
- endpoint: create_payload[:subscription][:endpoint],
- alerts: alerts_payload[:data][:alerts],
- policy: alerts_payload[:data][:policy]
- )
- end
- end
- describe 'DELETE /api/v1/push/subscription' do
- subject { delete '/api/v1/push/subscription', headers: headers }
- before { create_subscription_with_token }
- it 'removes the subscription' do
- expect { subject }
- .to change { endpoint_push_subscription }.to(nil)
- end
- end
- private
- def endpoint_push_subscriptions
- Web::PushSubscription.where(
- endpoint: create_payload[:subscription][:endpoint]
- )
- end
- def endpoint_push_subscription
- endpoint_push_subscriptions.first
- end
- def create_subscription_with_token
- Fabricate(
- :web_push_subscription,
- endpoint: create_payload[:subscription][:endpoint],
- access_token_id: token.id
- )
- end
- end
|