# frozen_string_literal: true class ScoreCalculator def self.default_score_weights { reply_count: 5, like_score: 15, incoming_link_count: 5, bookmark_count: 2, reads: 0.2 } end def initialize(weightings = nil) @weightings = weightings || ScoreCalculator.default_score_weights end # Calculate the score for all posts based on the weightings def calculate(opts = nil) update_posts_score(opts) update_posts_rank(opts) update_topics_rank(opts) end private def update_posts_score(opts) limit = 20_000 components = [] @weightings.each_key { |k| components << "COALESCE(posts.#{k}, 0) * :#{k}" } components = components.join(" + ") builder = DB.build <<SQL UPDATE posts p SET score = x.score FROM ( SELECT posts.id, #{components} as score FROM posts join topics on posts.topic_id = topics.id /*where*/ limit #{limit} ) AS x WHERE x.id = p.id SQL builder.where("posts.score IS NULL OR posts.score <> #{components}", @weightings) filter_topics(builder, opts) while builder.exec == limit end end def update_posts_rank(opts) limit = 20_000 builder = DB.build <<~SQL UPDATE posts SET percent_rank = X.percent_rank FROM ( SELECT posts.id, Y.percent_rank FROM posts JOIN ( SELECT id, percent_rank() OVER (PARTITION BY topic_id ORDER BY SCORE DESC) as percent_rank FROM posts ) Y ON Y.id = posts.id JOIN topics ON posts.topic_id = topics.id /*where*/ LIMIT #{limit} ) AS X WHERE posts.id = X.id SQL builder.where("posts.percent_rank IS NULL OR Y.percent_rank <> posts.percent_rank") filter_topics(builder, opts) while builder.exec == limit end end def update_topics_rank(opts) builder = DB.build <<~SQL UPDATE topics AS topics SET has_summary = (topics.like_count >= :likes_required AND topics.posts_count >= :posts_required AND x.max_score >= :score_required), score = x.avg_score FROM (SELECT p.topic_id, MAX(p.score) AS max_score, AVG(p.score) AS avg_score FROM posts AS p GROUP BY p.topic_id) AS x /*where*/ SQL defaults = { likes_required: SiteSetting.summary_likes_required, posts_required: SiteSetting.summary_posts_required, score_required: SiteSetting.summary_score_threshold, } builder.where(<<~SQL, defaults) x.topic_id = topics.id AND ( (topics.score <> x.avg_score OR topics.score IS NULL) OR (topics.has_summary IS NULL OR topics.has_summary <> ( topics.like_count >= :likes_required AND topics.posts_count >= :posts_required AND x.max_score >= :score_required )) ) SQL filter_topics(builder, opts) builder.exec end def filter_topics(builder, opts) return builder unless opts if min_topic_age = opts[:min_topic_age] builder.where("topics.bumped_at > :bumped_at ", bumped_at: min_topic_age) end if max_topic_length = opts[:max_topic_length] builder.where("topics.posts_count < :max_topic_length", max_topic_length: max_topic_length) end builder end end