Cache relationships in API (#6482)
* Cache relationships in API * Fetch relationships for search results in UI * Only save one account's maps in each cache itemgh/stable
parent
9b8a448477
commit
cba2897108
|
@ -1,4 +1,5 @@
|
||||||
import api from '../api';
|
import api from '../api';
|
||||||
|
import { fetchRelationships } from './accounts';
|
||||||
|
|
||||||
export const SEARCH_CHANGE = 'SEARCH_CHANGE';
|
export const SEARCH_CHANGE = 'SEARCH_CHANGE';
|
||||||
export const SEARCH_CLEAR = 'SEARCH_CLEAR';
|
export const SEARCH_CLEAR = 'SEARCH_CLEAR';
|
||||||
|
@ -38,6 +39,7 @@ export function submitSearch() {
|
||||||
},
|
},
|
||||||
}).then(response => {
|
}).then(response => {
|
||||||
dispatch(fetchSearchSuccess(response.data));
|
dispatch(fetchSearchSuccess(response.data));
|
||||||
|
dispatch(fetchRelationships(response.data.accounts.map(item => item.id)));
|
||||||
}).catch(error => {
|
}).catch(error => {
|
||||||
dispatch(fetchSearchFail(error));
|
dispatch(fetchSearchFail(error));
|
||||||
});
|
});
|
||||||
|
|
|
@ -16,12 +16,16 @@ class AccountDomainBlock < ApplicationRecord
|
||||||
belongs_to :account
|
belongs_to :account
|
||||||
validates :domain, presence: true, uniqueness: { scope: :account_id }
|
validates :domain, presence: true, uniqueness: { scope: :account_id }
|
||||||
|
|
||||||
after_create :remove_blocking_cache
|
after_commit :remove_blocking_cache
|
||||||
after_destroy :remove_blocking_cache
|
after_commit :remove_relationship_cache
|
||||||
|
|
||||||
private
|
private
|
||||||
|
|
||||||
def remove_blocking_cache
|
def remove_blocking_cache
|
||||||
Rails.cache.delete("exclude_domains_for:#{account_id}")
|
Rails.cache.delete("exclude_domains_for:#{account_id}")
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def remove_relationship_cache
|
||||||
|
Rails.cache.delete_matched("relationship:#{account_id}:*")
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -12,14 +12,14 @@
|
||||||
|
|
||||||
class Block < ApplicationRecord
|
class Block < ApplicationRecord
|
||||||
include Paginable
|
include Paginable
|
||||||
|
include RelationshipCacheable
|
||||||
|
|
||||||
belongs_to :account
|
belongs_to :account
|
||||||
belongs_to :target_account, class_name: 'Account'
|
belongs_to :target_account, class_name: 'Account'
|
||||||
|
|
||||||
validates :account_id, uniqueness: { scope: :target_account_id }
|
validates :account_id, uniqueness: { scope: :target_account_id }
|
||||||
|
|
||||||
after_create :remove_blocking_cache
|
after_commit :remove_blocking_cache
|
||||||
after_destroy :remove_blocking_cache
|
|
||||||
|
|
||||||
private
|
private
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,16 @@
|
||||||
|
# frozen_string_literal: true
|
||||||
|
|
||||||
|
module RelationshipCacheable
|
||||||
|
extend ActiveSupport::Concern
|
||||||
|
|
||||||
|
included do
|
||||||
|
after_commit :remove_relationship_cache
|
||||||
|
end
|
||||||
|
|
||||||
|
private
|
||||||
|
|
||||||
|
def remove_relationship_cache
|
||||||
|
Rails.cache.delete("relationship:#{account_id}:#{target_account_id}")
|
||||||
|
Rails.cache.delete("relationship:#{target_account_id}:#{account_id}")
|
||||||
|
end
|
||||||
|
end
|
|
@ -13,6 +13,7 @@
|
||||||
|
|
||||||
class Follow < ApplicationRecord
|
class Follow < ApplicationRecord
|
||||||
include Paginable
|
include Paginable
|
||||||
|
include RelationshipCacheable
|
||||||
|
|
||||||
belongs_to :account, counter_cache: :following_count
|
belongs_to :account, counter_cache: :following_count
|
||||||
|
|
||||||
|
|
|
@ -13,6 +13,7 @@
|
||||||
|
|
||||||
class FollowRequest < ApplicationRecord
|
class FollowRequest < ApplicationRecord
|
||||||
include Paginable
|
include Paginable
|
||||||
|
include RelationshipCacheable
|
||||||
|
|
||||||
belongs_to :account
|
belongs_to :account
|
||||||
belongs_to :target_account, class_name: 'Account'
|
belongs_to :target_account, class_name: 'Account'
|
||||||
|
|
|
@ -13,14 +13,14 @@
|
||||||
|
|
||||||
class Mute < ApplicationRecord
|
class Mute < ApplicationRecord
|
||||||
include Paginable
|
include Paginable
|
||||||
|
include RelationshipCacheable
|
||||||
|
|
||||||
belongs_to :account
|
belongs_to :account
|
||||||
belongs_to :target_account, class_name: 'Account'
|
belongs_to :target_account, class_name: 'Account'
|
||||||
|
|
||||||
validates :account_id, uniqueness: { scope: :target_account_id }
|
validates :account_id, uniqueness: { scope: :target_account_id }
|
||||||
|
|
||||||
after_create :remove_blocking_cache
|
after_commit :remove_blocking_cache
|
||||||
after_destroy :remove_blocking_cache
|
|
||||||
|
|
||||||
private
|
private
|
||||||
|
|
||||||
|
|
|
@ -5,11 +5,67 @@ class AccountRelationshipsPresenter
|
||||||
:muting, :requested, :domain_blocking
|
:muting, :requested, :domain_blocking
|
||||||
|
|
||||||
def initialize(account_ids, current_account_id, **options)
|
def initialize(account_ids, current_account_id, **options)
|
||||||
@following = Account.following_map(account_ids, current_account_id).merge(options[:following_map] || {})
|
@account_ids = account_ids.map { |a| a.is_a?(Account) ? a.id : a }
|
||||||
@followed_by = Account.followed_by_map(account_ids, current_account_id).merge(options[:followed_by_map] || {})
|
@current_account_id = current_account_id
|
||||||
@blocking = Account.blocking_map(account_ids, current_account_id).merge(options[:blocking_map] || {})
|
|
||||||
@muting = Account.muting_map(account_ids, current_account_id).merge(options[:muting_map] || {})
|
@following = cached[:following].merge(Account.following_map(@uncached_account_ids, @current_account_id))
|
||||||
@requested = Account.requested_map(account_ids, current_account_id).merge(options[:requested_map] || {})
|
@followed_by = cached[:followed_by].merge(Account.followed_by_map(@uncached_account_ids, @current_account_id))
|
||||||
@domain_blocking = Account.domain_blocking_map(account_ids, current_account_id).merge(options[:domain_blocking_map] || {})
|
@blocking = cached[:blocking].merge(Account.blocking_map(@uncached_account_ids, @current_account_id))
|
||||||
|
@muting = cached[:muting].merge(Account.muting_map(@uncached_account_ids, @current_account_id))
|
||||||
|
@requested = cached[:requested].merge(Account.requested_map(@uncached_account_ids, @current_account_id))
|
||||||
|
@domain_blocking = cached[:domain_blocking].merge(Account.domain_blocking_map(@uncached_account_ids, @current_account_id))
|
||||||
|
|
||||||
|
cache_uncached!
|
||||||
|
|
||||||
|
@following.merge!(options[:following_map] || {})
|
||||||
|
@followed_by.merge!(options[:followed_by_map] || {})
|
||||||
|
@blocking.merge!(options[:blocking_map] || {})
|
||||||
|
@muting.merge!(options[:muting_map] || {})
|
||||||
|
@requested.merge!(options[:requested_map] || {})
|
||||||
|
@domain_blocking.merge!(options[:domain_blocking_map] || {})
|
||||||
|
end
|
||||||
|
|
||||||
|
private
|
||||||
|
|
||||||
|
def cached
|
||||||
|
return @cached if defined?(@cached)
|
||||||
|
|
||||||
|
@cached = {
|
||||||
|
following: {},
|
||||||
|
followed_by: {},
|
||||||
|
blocking: {},
|
||||||
|
muting: {},
|
||||||
|
requested: {},
|
||||||
|
domain_blocking: {},
|
||||||
|
}
|
||||||
|
|
||||||
|
@uncached_account_ids = []
|
||||||
|
|
||||||
|
@account_ids.each do |account_id|
|
||||||
|
maps_for_account = Rails.cache.read("relationship:#{@current_account_id}:#{account_id}")
|
||||||
|
|
||||||
|
if maps_for_account.is_a?(Hash)
|
||||||
|
@cached.merge!(maps_for_account)
|
||||||
|
else
|
||||||
|
@uncached_account_ids << account_id
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
@cached
|
||||||
|
end
|
||||||
|
|
||||||
|
def cache_uncached!
|
||||||
|
@uncached_account_ids.each do |account_id|
|
||||||
|
maps_for_account = {
|
||||||
|
following: { account_id => following[account_id] },
|
||||||
|
followed_by: { account_id => followed_by[account_id] },
|
||||||
|
blocking: { account_id => blocking[account_id] },
|
||||||
|
muting: { account_id => muting[account_id] },
|
||||||
|
requested: { account_id => requested[account_id] },
|
||||||
|
domain_blocking: { account_id => domain_blocking[account_id] },
|
||||||
|
}
|
||||||
|
|
||||||
|
Rails.cache.write("relationship:#{@current_account_id}:#{account_id}", maps_for_account, expires_in: 1.day)
|
||||||
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
Reference in New Issue