74 lines
1.7 KiB
Ruby
74 lines
1.7 KiB
Ruby
# frozen_string_literal: true
|
|
|
|
module Gitlab
|
|
module CycleAnalytics
|
|
class UsageData
|
|
PROJECTS_LIMIT = 10
|
|
|
|
attr_reader :projects, :options
|
|
|
|
def initialize
|
|
@projects = Project.sorted_by_activity.limit(PROJECTS_LIMIT)
|
|
@options = { from: 7.days.ago }
|
|
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.new(project, options).all_medians_per_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
|