summaryrefslogtreecommitdiff
path: root/lib/gitlab/cycle_analytics/usage_data.rb
blob: e58def57e69e4cfc7dc76e17eb176e433edbd767 (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
# frozen_string_literal: true

module Gitlab
  module CycleAnalytics
    class UsageData
      include Gitlab::Utils::StrongMemoize
      PROJECTS_LIMIT = 10

      attr_reader :options

      def initialize
        @options = { from: 7.days.ago }
      end

      def projects
        strong_memoize(:projects) do
          projects = Project.where.not(last_activity_at: nil).order(last_activity_at: :desc).limit(10) +
                     Project.where.not(last_repository_updated_at: nil).order(last_repository_updated_at: :desc).limit(10)

          projects = projects.uniq.sort_by do |project|
            [project.last_activity_at, project.last_repository_updated_at].min
          end

          if projects.size < 10
            projects.concat(Project.where(last_activity_at: nil, last_repository_updated_at: nil).limit(10))
          end

          projects.uniq.first(10)
        end
      end

      def to_json(*)
        total = 0

        values =
          medians_per_stage.each_with_object({}) do |(stage_name, medians), hsh|
            calculations = stage_values(medians)

            total += calculations.values.compact.sum
            hsh[stage_name] = calculations
          end

        values[:total] = total

        { avg_cycle_analytics: values }
      end

      private

      def medians_per_stage
        projects.each_with_object({}) do |project, hsh|
          ::CycleAnalytics::ProjectLevel.new(project, options: options).all_medians_by_stage.each do |stage_name, median|
            hsh[stage_name] ||= []
            hsh[stage_name] << median
          end
        end
      end

      def stage_values(medians)
        medians = medians.map(&:presence).compact
        average = calc_average(medians)

        {
          average: average,
          sd: standard_deviation(medians, average),
          missing: projects.length - medians.length
        }
      end

      def calc_average(values)
        return if values.empty?

        (values.sum / values.length).to_i
      end

      def standard_deviation(values, average)
        Math.sqrt(sample_variance(values, average)).to_i
      end

      def sample_variance(values, average)
        return 0 if values.length <= 1

        sum = values.inject(0) do |acc, val|
          acc + (val - average)**2
        end

        sum / (values.length - 1)
      end
    end
  end
end