parent
fe57f6330f
commit
6c4c84b161
@ -0,0 +1,27 @@ |
|||||||
|
class Feed |
||||||
|
def initialize(type, account) |
||||||
|
@type = type |
||||||
|
@account = account |
||||||
|
end |
||||||
|
|
||||||
|
def get(limit, offset = 0) |
||||||
|
unhydrated = redis.zrevrange(key, offset, limit) |
||||||
|
status_map = Hash.new |
||||||
|
|
||||||
|
# If we're after most recent items and none are there, we need to precompute the feed |
||||||
|
return PrecomputeFeedService.new.(@type, @account).take(limit) if unhydrated.empty? && offset == 0 |
||||||
|
|
||||||
|
Status.where(id: unhydrated).each { |status| status_map[status.id.to_s] = status } |
||||||
|
return unhydrated.map { |id| status_map[id] } |
||||||
|
end |
||||||
|
|
||||||
|
private |
||||||
|
|
||||||
|
def key |
||||||
|
"feed:#{@type}:#{@account.id}" |
||||||
|
end |
||||||
|
|
||||||
|
def redis |
||||||
|
$redis |
||||||
|
end |
||||||
|
end |
@ -0,0 +1,46 @@ |
|||||||
|
class FanOutOnWriteService < BaseService |
||||||
|
MAX_FEED_SIZE = 800 |
||||||
|
|
||||||
|
# Push a status into home and mentions feeds |
||||||
|
# @param [Status] status |
||||||
|
def call(status) |
||||||
|
replied_to_user = status.reply? ? status.thread.account : nil |
||||||
|
|
||||||
|
# Deliver to local self |
||||||
|
push(:home, status.account.id, status) if status.account.local? |
||||||
|
|
||||||
|
# Deliver to local followers |
||||||
|
status.account.followers.each do |follower| |
||||||
|
next if (status.reply? && !follower.following?(replied_to_user)) || !follower.local? |
||||||
|
push(:home, follower.id, status) |
||||||
|
end |
||||||
|
|
||||||
|
# Deliver to local mentioned |
||||||
|
status.mentions.each do |mentioned_account| |
||||||
|
next unless mentioned_account.local? |
||||||
|
push(:mentions, mentioned_account.id, status) |
||||||
|
end |
||||||
|
end |
||||||
|
|
||||||
|
private |
||||||
|
|
||||||
|
def push(type, receiver_id, status) |
||||||
|
redis.zadd(key(type, receiver_id), status.created_at.to_i, status.id) |
||||||
|
trim(type, receiver_id) |
||||||
|
end |
||||||
|
|
||||||
|
def trim(type, receiver_id) |
||||||
|
return unless redis.zcard(key(type, receiver_id)) > MAX_FEED_SIZE |
||||||
|
|
||||||
|
last = redis.zrevrange(key(type, receiver_id), MAX_FEED_SIZE - 1, MAX_FEED_SIZE - 1) |
||||||
|
redis.zremrangebyscore(key(type, receiver_id), '-inf', "(#{last.last}") |
||||||
|
end |
||||||
|
|
||||||
|
def key(type, id) |
||||||
|
"feed:#{type}:#{id}" |
||||||
|
end |
||||||
|
|
||||||
|
def redis |
||||||
|
$redis |
||||||
|
end |
||||||
|
end |
@ -0,0 +1,35 @@ |
|||||||
|
class PrecomputeFeedService < BaseService |
||||||
|
MAX_FEED_SIZE = 800 |
||||||
|
|
||||||
|
# Fill up a user's home/mentions feed from DB and return it |
||||||
|
# @param [Symbol] type :home or :mentions |
||||||
|
# @param [Account] account |
||||||
|
# @return [Array] |
||||||
|
def call(type, account) |
||||||
|
statuses = send(type.to_s, account).order('created_at desc').limit(MAX_FEED_SIZE) |
||||||
|
statuses.each { |status| push(type, account.id, status) } |
||||||
|
statuses |
||||||
|
end |
||||||
|
|
||||||
|
private |
||||||
|
|
||||||
|
def push(type, receiver_id, status) |
||||||
|
redis.zadd(key(type, receiver_id), status.created_at.to_i, status.id) |
||||||
|
end |
||||||
|
|
||||||
|
def home(account) |
||||||
|
Status.where(account: [account] + account.following) |
||||||
|
end |
||||||
|
|
||||||
|
def mentions(account) |
||||||
|
Status.where(id: Mention.where(account: account).pluck(:status_id)) |
||||||
|
end |
||||||
|
|
||||||
|
def key(type, id) |
||||||
|
"feed:#{type}:#{id}" |
||||||
|
end |
||||||
|
|
||||||
|
def redis |
||||||
|
$redis |
||||||
|
end |
||||||
|
end |
@ -0,0 +1 @@ |
|||||||
|
$redis = Redis.new(host: ENV['REDIS_HOST'] || 'localhost', port: ENV['REDIS_PORT'] || 6379) |
Loading…
Reference in new issue