2018-09-15 01:42:22 +10:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2023-05-24 19:55:40 +10:00
|
|
|
require_relative 'base'
|
2018-09-15 01:42:22 +10:00
|
|
|
|
2023-05-24 00:08:26 +10:00
|
|
|
module Mastodon::CLI
|
2023-05-24 19:55:40 +10:00
|
|
|
class Feeds < Base
|
2022-04-29 01:47:34 +10:00
|
|
|
include Redisable
|
2024-11-30 01:08:57 +11:00
|
|
|
include DatabaseHelper
|
2019-09-10 21:48:48 +10:00
|
|
|
|
2018-09-15 01:42:22 +10:00
|
|
|
option :all, type: :boolean, default: false
|
2019-09-10 21:48:48 +10:00
|
|
|
option :concurrency, type: :numeric, default: 5, aliases: [:c]
|
|
|
|
option :verbose, type: :boolean, aliases: [:v]
|
2018-09-15 01:42:22 +10:00
|
|
|
option :dry_run, type: :boolean, default: false
|
|
|
|
desc 'build [USERNAME]', 'Build home and list feeds for one or all users'
|
|
|
|
long_desc <<-LONG_DESC
|
|
|
|
Build home and list feeds that are stored in Redis from the database.
|
|
|
|
|
|
|
|
With the --all option, all active users will be processed.
|
|
|
|
Otherwise, a single user specified by USERNAME.
|
|
|
|
LONG_DESC
|
|
|
|
def build(username = nil)
|
|
|
|
if options[:all] || username.nil?
|
2023-06-11 02:37:36 +10:00
|
|
|
processed, = parallelize_with_progress(active_user_accounts) do |account|
|
2023-05-31 00:07:44 +10:00
|
|
|
PrecomputeFeedService.new.call(account) unless dry_run?
|
2018-09-15 01:42:22 +10:00
|
|
|
end
|
|
|
|
|
2023-05-31 00:07:44 +10:00
|
|
|
say("Regenerated feeds for #{processed} accounts #{dry_run_mode_suffix}", :green, true)
|
2018-09-15 01:42:22 +10:00
|
|
|
elsif username.present?
|
|
|
|
account = Account.find_local(username)
|
|
|
|
|
2024-01-26 19:53:44 +11:00
|
|
|
fail_with_message 'No such account' if account.nil?
|
2018-10-22 01:42:22 +11:00
|
|
|
|
2023-05-31 00:07:44 +10:00
|
|
|
PrecomputeFeedService.new.call(account) unless dry_run?
|
2018-09-15 01:42:22 +10:00
|
|
|
|
2023-05-31 00:07:44 +10:00
|
|
|
say("OK #{dry_run_mode_suffix}", :green, true)
|
2018-09-15 01:42:22 +10:00
|
|
|
else
|
2024-01-26 19:53:44 +11:00
|
|
|
fail_with_message 'No account(s) given'
|
2018-09-15 01:42:22 +10:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
desc 'clear', 'Remove all home and list feeds from Redis'
|
|
|
|
def clear
|
2022-04-29 01:47:34 +10:00
|
|
|
keys = redis.keys('feed:*')
|
2023-03-05 02:38:28 +11:00
|
|
|
redis.del(keys)
|
2018-09-15 01:42:22 +10:00
|
|
|
say('OK', :green)
|
|
|
|
end
|
2023-06-11 02:37:36 +10:00
|
|
|
|
2024-11-30 01:08:57 +11:00
|
|
|
desc 'vacuum', 'Remove home feeds of inactive users from Redis'
|
|
|
|
long_desc <<-LONG_DESC
|
|
|
|
Running this task should not be needed in most cases, as Mastodon will
|
|
|
|
automatically clean up feeds from inactive accounts every day.
|
|
|
|
|
|
|
|
However, this task is more aggressive in order to clean up feeds that
|
|
|
|
may have been missed because of bugs or database mishaps.
|
|
|
|
LONG_DESC
|
|
|
|
def vacuum
|
|
|
|
with_read_replica do
|
|
|
|
say('Deleting orphaned home feeds…')
|
|
|
|
redis.scan_each(match: 'feed:home:*').each_slice(1000) do |keys|
|
|
|
|
ids = keys.map { |key| key.split(':')[2] }.compact_blank
|
|
|
|
|
|
|
|
known_ids = User.confirmed.signed_in_recently.where(account_id: ids).pluck(:account_id)
|
|
|
|
|
|
|
|
keys_to_delete = keys.filter { |key| known_ids.exclude?(key.split(':')[2]&.to_i) }
|
|
|
|
redis.del(keys_to_delete)
|
|
|
|
end
|
|
|
|
|
|
|
|
say('Deleting orphaned list feeds…')
|
|
|
|
redis.scan_each(match: 'feed:list:*').each_slice(1000) do |keys|
|
|
|
|
ids = keys.map { |key| key.split(':')[2] }.compact_blank
|
|
|
|
|
|
|
|
known_ids = List.where(account_id: User.confirmed.signed_in_recently.select(:account_id)).where(id: ids).pluck(:id)
|
|
|
|
|
|
|
|
keys_to_delete = keys.filter { |key| known_ids.exclude?(key.split(':')[2]&.to_i) }
|
|
|
|
redis.del(keys_to_delete)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2023-06-11 02:37:36 +10:00
|
|
|
private
|
|
|
|
|
|
|
|
def active_user_accounts
|
|
|
|
Account.joins(:user).merge(User.active)
|
|
|
|
end
|
2018-09-15 01:42:22 +10:00
|
|
|
end
|
|
|
|
end
|