summaryrefslogtreecommitdiff
path: root/lib/gitlab/analytics/cycle_analytics/aggregated/records_fetcher.rb
blob: 7dce757cdc8e7e11ca4a1200772e0187c1e05d95 (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
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
# frozen_string_literal: true

module Gitlab
  module Analytics
    module CycleAnalytics
      module Aggregated
        class RecordsFetcher
          include Gitlab::Utils::StrongMemoize
          include StageQueryHelpers

          MAX_RECORDS = 20

          MAPPINGS = {
            Issue => {
              serializer_class: AnalyticsIssueSerializer,
              includes_for_query: { project: { namespace: [:route] }, author: [] },
              columns_for_select: %I[title iid id created_at author_id project_id]
            },
            MergeRequest => {
              serializer_class: AnalyticsMergeRequestSerializer,
              includes_for_query: { target_project: [:namespace], author: [] },
              columns_for_select: %I[title iid id created_at author_id state_id target_project_id]
            }
          }.freeze

          def initialize(stage:, query:, params: {})
            @stage = stage
            @query = query
            @params = params
            @sort = params[:sort] || :end_event
            @direction = params[:direction] || :desc
            @page = params[:page] || 1
            @per_page = MAX_RECORDS
            @stage_event_model = query.model
          end

          def serialized_records
            strong_memoize(:serialized_records) do
              records = ordered_and_limited_query.select(stage_event_model.arel_table[Arel.star], duration.as('total_time'))

              yield records if block_given?
              issuables_and_records = load_issuables(records)

              preload_associations(issuables_and_records.map(&:first))

              issuables_and_records.map do |issuable, record|
                project = issuable.project
                attributes = issuable.attributes.merge({
                  project_path: project.path,
                  namespace_path: project.namespace.route.path,
                  author: issuable.author,
                  total_time: record.total_time
                })
                serializer.represent(attributes)
              end
            end
          end

          # rubocop: disable CodeReuse/ActiveRecord
          def ordered_and_limited_query
            sorting_options = {
              end_event: {
                asc: -> { query.order(end_event_timestamp: :asc) },
                desc: -> { query.order(end_event_timestamp: :desc) }
              },
              duration: {
                asc: -> { query.order(duration.asc) },
                desc: -> { query.order(duration.desc) }
              }
            }

            sort_lambda = sorting_options.dig(sort, direction) || sorting_options.dig(:end_event, :desc)

            sort_lambda.call
              .page(page)
              .per(per_page)
              .without_count
          end
          # rubocop: enable CodeReuse/ActiveRecord

          private

          attr_reader :stage, :query, :sort, :direction, :params, :page, :per_page, :stage_event_model

          delegate :subject_class, to: :stage

          def load_issuables(stage_event_records)
            stage_event_records_by_issuable_id = stage_event_records.index_by(&:issuable_id)

            issuable_model = stage_event_model.issuable_model
            issuables_by_id = issuable_model.id_in(stage_event_records_by_issuable_id.keys).index_by(&:id)

            stage_event_records_by_issuable_id.map do |issuable_id, record|
              [issuables_by_id[issuable_id], record] if issuables_by_id[issuable_id]
            end.compact
          end

          def serializer
            MAPPINGS.fetch(subject_class).fetch(:serializer_class).new
          end

          # rubocop: disable CodeReuse/ActiveRecord
          def preload_associations(records)
            ActiveRecord::Associations::Preloader.new.preload(
              records,
              MAPPINGS.fetch(subject_class).fetch(:includes_for_query)
            )

            records
          end
          # rubocop: enable CodeReuse/ActiveRecord
        end
      end
    end
  end
end