dawarich/app/services/google_maps/semantic_history_parser.rb

92 lines
3.5 KiB
Ruby
Raw Normal View History

2024-03-15 18:27:31 -04:00
# frozen_string_literal: true
2024-05-18 09:00:44 -04:00
class GoogleMaps::SemanticHistoryParser
include Imports::Broadcaster
2024-05-25 07:36:15 -04:00
attr_reader :import, :user_id
2024-03-15 18:27:31 -04:00
2024-05-25 07:36:15 -04:00
def initialize(import, user_id)
@import = import
2024-05-25 07:36:15 -04:00
@user_id = user_id
2024-03-15 18:27:31 -04:00
end
def call
points_data = parse_json
points_data.each.with_index(1) do |point_data, index|
next if Point.exists?(
timestamp: point_data[:timestamp],
latitude: point_data[:latitude],
longitude: point_data[:longitude],
user_id:
)
2024-03-15 18:27:31 -04:00
Point.create(
latitude: point_data[:latitude],
longitude: point_data[:longitude],
timestamp: point_data[:timestamp],
raw_data: point_data[:raw_data],
topic: 'Google Maps Timeline Export',
tracker_id: 'google-maps-timeline-export',
2024-05-25 07:36:15 -04:00
import_id: import.id,
user_id:
2024-03-15 18:27:31 -04:00
)
broadcast_import_progress(import, index)
2024-12-24 07:08:14 -05:00
puts "Created point: #{location['latitudeE7'] / 1e7}, #{location['longitudeE7'] / 1e7} at #{location['timestamp']}"
2024-03-15 18:27:31 -04:00
end
end
private
def parse_json
import.raw_data['timelineObjects'].flat_map do |timeline_object|
2024-03-15 18:27:31 -04:00
if timeline_object['activitySegment'].present?
if timeline_object['activitySegment']['startLocation'].blank?
next if timeline_object['activitySegment']['waypointPath'].blank?
timeline_object['activitySegment']['waypointPath']['waypoints'].map do |waypoint|
{
latitude: waypoint['latE7'].to_f / 10**7,
longitude: waypoint['lngE7'].to_f / 10**7,
timestamp: Timestamps.parse_timestamp(timeline_object['activitySegment']['duration']['startTimestamp'] || timeline_object['activitySegment']['duration']['startTimestampMs']),
2024-03-15 18:27:31 -04:00
raw_data: timeline_object
}
end
else
{
latitude: timeline_object['activitySegment']['startLocation']['latitudeE7'].to_f / 10**7,
longitude: timeline_object['activitySegment']['startLocation']['longitudeE7'].to_f / 10**7,
timestamp: Timestamps.parse_timestamp(timeline_object['activitySegment']['duration']['startTimestamp'] || timeline_object['activitySegment']['duration']['startTimestampMs']),
2024-03-15 18:27:31 -04:00
raw_data: timeline_object
}
end
elsif timeline_object['placeVisit'].present?
if timeline_object.dig('placeVisit', 'location', 'latitudeE7').present? &&
timeline_object.dig('placeVisit', 'location', 'longitudeE7').present?
{
latitude: timeline_object['placeVisit']['location']['latitudeE7'].to_f / 10**7,
longitude: timeline_object['placeVisit']['location']['longitudeE7'].to_f / 10**7,
timestamp: Timestamps.parse_timestamp(timeline_object['placeVisit']['duration']['startTimestamp'] || timeline_object['placeVisit']['duration']['startTimestampMs']),
raw_data: timeline_object
}
elsif timeline_object.dig('placeVisit', 'otherCandidateLocations')&.any?
point = timeline_object['placeVisit']['otherCandidateLocations'][0]
next unless point['latitudeE7'].present? && point['longitudeE7'].present?
{
latitude: point['latitudeE7'].to_f / 10**7,
longitude: point['longitudeE7'].to_f / 10**7,
timestamp: Timestamps.parse_timestamp(timeline_object['placeVisit']['duration']['startTimestamp'] || timeline_object['placeVisit']['duration']['startTimestampMs']),
raw_data: timeline_object
}
else
next
end
2024-03-15 18:27:31 -04:00
end
end.reject(&:blank?)
end
end