ActivityPub migration procedure (#4617)

* ActivityPub migration procedure

Once one account is detected as going from OStatus to ActivityPub,
invalidate WebFinger cache for other accounts from the same domain

* Unsubscribe from PuSH updates once we receive an ActivityPub payload

* Re-subscribe to PuSH unless already unsubscribed, regardless of protocol
This commit is contained in:
Eugen Rochko 2017-08-21 01:14:40 +02:00 committed by GitHub
parent cbc3b13ef6
commit 1a19358b24
8 changed files with 47 additions and 8 deletions

View file

@ -7,6 +7,7 @@ class ActivityPub::InboxesController < Api::BaseController
def create def create
if signed_request_account if signed_request_account
upgrade_account
process_payload process_payload
head 201 head 201
else else
@ -24,6 +25,11 @@ class ActivityPub::InboxesController < Api::BaseController
@body ||= request.body.read @body ||= request.body.read
end end
def upgrade_account
return unless signed_request_account.subscribed?
Pubsubhubbub::UnsubscribeWorker.perform_async(signed_request_account.id)
end
def process_payload def process_payload
ActivityPub::ProcessingWorker.perform_async(signed_request_account.id, body.force_encoding('UTF-8')) ActivityPub::ProcessingWorker.perform_async(signed_request_account.id, body.force_encoding('UTF-8'))
end end

View file

@ -17,7 +17,7 @@ module Admin
end end
def unsubscribe def unsubscribe
UnsubscribeService.new.call(@account) Pubsubhubbub::UnsubscribeWorker.perform_async(@account.id)
redirect_to admin_account_path(@account.id) redirect_to admin_account_path(@account.id)
end end

View file

@ -12,7 +12,8 @@ class ActivityPub::ProcessAccountService < BaseService
@domain = domain @domain = domain
@account = Account.find_by(uri: @uri) @account = Account.find_by(uri: @uri)
create_account if @account.nil? create_account if @account.nil?
upgrade_account if @account.ostatus?
update_account update_account
@account @account
@ -24,6 +25,7 @@ class ActivityPub::ProcessAccountService < BaseService
def create_account def create_account
@account = Account.new @account = Account.new
@account.protocol = :activitypub
@account.username = @username @account.username = @username
@account.domain = @domain @account.domain = @domain
@account.uri = @uri @account.uri = @uri
@ -50,6 +52,10 @@ class ActivityPub::ProcessAccountService < BaseService
@account.save! @account.save!
end end
def upgrade_account
ActivityPub::PostUpgradeWorker.perform_async(@account.domain)
end
def image_url(key) def image_url(key)
value = first_of_value(@json[key]) value = first_of_value(@json[key])

View file

@ -2,7 +2,7 @@
class UnsubscribeService < BaseService class UnsubscribeService < BaseService
def call(account) def call(account)
return unless account.ostatus? return if account.hub_url.blank?
@account = account @account = account
@response = build_request.perform @response = build_request.perform

View file

@ -0,0 +1,15 @@
# frozen_string_literal: true
class ActivityPub::PostUpgradeWorker
include Sidekiq::Worker
sidekiq_options queue: 'pull'
def perform(domain)
Account.where(domain: domain)
.where(protocol: :ostatus)
.where.not(last_webfingered_at: nil)
.in_batches
.update_all(last_webfingered_at: nil)
end
end

View file

@ -0,0 +1,15 @@
# frozen_string_literal: true
class Pubsubhubbub::UnsubscribeWorker
include Sidekiq::Worker
sidekiq_options queue: 'push', retry: false, unique: :until_executed, dead: false
def perform(account_id)
account = Account.find(account_id)
logger.debug "PuSH unsubscribing from #{account.acct}"
::UnsubscribeService.new.call(account)
rescue ActiveRecord::RecordNotFound
true
end
end

View file

@ -14,6 +14,6 @@ class Scheduler::SubscriptionsScheduler
private private
def expiring_accounts def expiring_accounts
Account.where(protocol: :ostatus).expiring(1.day.from_now).partitioned Account.expiring(1.day.from_now).partitioned
end end
end end

View file

@ -111,10 +111,7 @@ namespace :mastodon do
namespace :push do namespace :push do
desc 'Unsubscribes from PuSH updates of feeds nobody follows locally' desc 'Unsubscribes from PuSH updates of feeds nobody follows locally'
task clear: :environment do task clear: :environment do
Account.remote.without_followers.where.not(subscription_expires_at: nil).find_each do |a| Pubsubhubbub::UnsubscribeWorker.push_bulk(Account.remote.without_followers.where.not(subscription_expires_at: nil).pluck(:id))
Rails.logger.debug "PuSH unsubscribing from #{a.acct}"
UnsubscribeService.new.call(a)
end
end end
desc 'Re-subscribes to soon expiring PuSH subscriptions (deprecated)' desc 'Re-subscribes to soon expiring PuSH subscriptions (deprecated)'