dawarich/app/services/visits/creator.rb
2025-03-07 23:32:56 +01:00

96 lines
3 KiB
Ruby

# frozen_string_literal: true
module Visits
# Creates visit records from detected visit data
class Creator
attr_reader :user
def initialize(user)
@user = user
end
def create_visits(visits)
visits.map do |visit_data|
# Variables to store data outside the transaction
visit_instance = nil
place_data = nil
# First transaction to create the visit
ActiveRecord::Base.transaction do
# Try to find matching area or place
area = find_matching_area(visit_data)
# Only find/create place if no area was found
place_data = PlaceFinder.new(user).find_or_create_place(visit_data) unless area
main_place = place_data&.dig(:main_place)
visit_instance = Visit.create!(
user: user,
area: area,
place: main_place,
started_at: Time.zone.at(visit_data[:start_time]),
ended_at: Time.zone.at(visit_data[:end_time]),
duration: visit_data[:duration] / 60, # Convert to minutes
name: generate_visit_name(area, main_place, visit_data[:suggested_name]),
status: :suggested
)
Point.where(id: visit_data[:points].map(&:id)).update_all(visit_id: visit_instance.id)
end
# Associate suggested places outside the main transaction
# to avoid deadlocks when multiple processes run simultaneously
if place_data&.dig(:suggested_places).present?
associate_suggested_places(visit_instance, place_data[:suggested_places])
end
visit_instance
end
end
private
# Create place_visits records directly to avoid deadlocks
def associate_suggested_places(visit, suggested_places)
existing_place_ids = visit.place_visits.pluck(:place_id)
# Only create associations that don't already exist
place_ids_to_add = suggested_places.map(&:id) - existing_place_ids
# Skip if there's nothing to add
return if place_ids_to_add.empty?
# Batch create place_visit records
place_visits_attrs = place_ids_to_add.map do |place_id|
{ visit_id: visit.id, place_id: place_id, created_at: Time.current, updated_at: Time.current }
end
# Use insert_all for efficient bulk insertion without callbacks
PlaceVisit.insert_all(place_visits_attrs) if place_visits_attrs.any?
end
def find_matching_area(visit_data)
user.areas.find do |area|
near_area?([visit_data[:center_lat], visit_data[:center_lon]], area)
end
end
def near_area?(center, area)
distance = Geocoder::Calculations.distance_between(
center,
[area.latitude, area.longitude],
units: :km
)
distance * 1000 <= area.radius # Convert to meters
end
def generate_visit_name(area, place, suggested_name)
return area.name if area
return place.name if place
return suggested_name if suggested_name.present?
'Unknown Location'
end
end
end