# frozen_string_literal: true class Stats::CalculateMonth def initialize(user_id, year, month) @user = User.find(user_id) @year = year @month = month end def call return if points.empty? update_month_stats(year, month) rescue StandardError => e create_stats_update_failed_notification(user, e) end private attr_reader :user, :year, :month def start_timestamp = DateTime.new(year, month, 1).to_i def end_timestamp DateTime.new(year, month, -1).to_i # -1 returns last day of month end def update_month_stats(year, month) Stat.transaction do stat = Stat.find_or_initialize_by(year:, month:, user:) distance_by_day = stat.distance_by_day stat.assign_attributes( daily_distance: distance_by_day, distance: distance(distance_by_day), toponyms: toponyms ) stat.save end end def points return @points if defined?(@points) @points = user .tracked_points .without_raw_data .where(timestamp: start_timestamp..end_timestamp) .select(:latitude, :longitude, :timestamp, :city, :country) .order(timestamp: :asc) end def distance(distance_by_day) distance_by_day.sum { |day| day[1] } end def toponyms CountriesAndCities.new(points).call end def create_stats_update_failed_notification(user, error) Notifications::Create.new( user:, kind: :error, title: 'Stats update failed', content: "#{error.message}, stacktrace: #{error.backtrace.join("\n")}" ).call end end