mirror of
https://github.com/mastodon/mastodon.git
synced 2025-01-25 11:31:40 +01:00
186 lines
5.1 KiB
Ruby
186 lines
5.1 KiB
Ruby
# 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 'GET /api/v1/push/subscription' do
|
|
subject { get '/api/v1/push/subscription', headers: headers }
|
|
|
|
before { create_subscription_with_token }
|
|
|
|
it 'shows subscription details' do
|
|
subject
|
|
|
|
expect(response)
|
|
.to have_http_status(200)
|
|
expect(response.content_type)
|
|
.to start_with('application/json')
|
|
expect(response.parsed_body)
|
|
.to include(endpoint: endpoint)
|
|
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
|