Welcome to mirror list, hosted at ThFree Co, Russian Federation.

open_issues_count_service.rb « groups « services « app - gitlab.com/gitlab-org/gitlab-foss.git - Unnamed repository; edit this file 'description' to name the repository.
summaryrefslogtreecommitdiff
blob: db1ca09212a8d7ca95f6fde720f966dc90c454a5 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
# frozen_string_literal: true

module Groups
  # Service class for counting and caching the number of open issues of a group.
  class OpenIssuesCountService < BaseCountService
    include Gitlab::Utils::StrongMemoize

    VERSION = 1
    PUBLIC_COUNT_KEY = 'group_public_open_issues_count'
    TOTAL_COUNT_KEY = 'group_total_open_issues_count'
    CACHED_COUNT_THRESHOLD = 1000
    EXPIRATION_TIME = 24.hours

    attr_reader :group, :user

    def initialize(group, user = nil)
      @group = group
      @user = user
    end

    # Reads count value from cache and return it if present.
    # If empty or expired, #uncached_count will calculate the issues count for the group and
    # compare it with the threshold. If it is greater, it will be written to the cache and returned.
    # If below, it will be returned without being cached.
    # This results in only caching large counts and calculating the rest with every call to maintain
    # accuracy.
    def count
      cached_count = Rails.cache.read(cache_key)
      return cached_count unless cached_count.blank?

      refreshed_count = uncached_count
      update_cache_for_key(cache_key) { refreshed_count } if refreshed_count > CACHED_COUNT_THRESHOLD
      refreshed_count
    end

    def cache_key(key = nil)
      ['groups', 'open_issues_count_service', VERSION, group.id, cache_key_name]
    end

    private

    def cache_options
      super.merge({ expires_in: EXPIRATION_TIME })
    end

    def cache_key_name
      public_only? ? PUBLIC_COUNT_KEY : TOTAL_COUNT_KEY
    end

    def public_only?
      !user_is_at_least_reporter?
    end

    def user_is_at_least_reporter?
      strong_memoize(:user_is_at_least_reporter) do
        group.member?(user, Gitlab::Access::REPORTER)
      end
    end

    def relation_for_count
      IssuesFinder.new(user, group_id: group.id, state: 'opened', non_archived: true, include_subgroups: true, public_only: public_only?).execute
    end
  end
end