[locations] Refactor finding locations

This commit is contained in:
2024-02-10 16:57:52 -05:00
parent 74f672a2fd
commit 2661aee915
2 changed files with 37 additions and 56 deletions

View File

@ -1,3 +1,4 @@
from decimal import Decimal
import logging
from typing import Dict
from uuid import uuid4
@ -83,6 +84,28 @@ class GeoLocation(ScrobblableMixin):
)
return location
def loc_diff(self, old_lat_lon: tuple) -> tuple:
return (
abs(Decimal(old_lat_lon[0]) - Decimal(self.lat)),
abs(Decimal(old_lat_lon[1]) - Decimal(self.lon)),
)
def has_moved(self, past_points) -> bool:
has_moved = False
all_moves = []
for point in past_points:
loc_diff = self.loc_diff((point.lat, point.lon))
if loc_diff and loc_diff[0] < 0.001 and loc_diff[1] > 0.001:
all_moves.append(True)
else:
all_moves.append(False)
if not False in all_moves:
has_moved = True
return has_moved
class RawGeoLocation(TimeStampedModel):
user = models.ForeignKey(User, on_delete=models.CASCADE)

View File

@ -687,8 +687,7 @@ class Scrobble(TimeStampedModel):
return percent
@property
def can_be_updated(self) -> bool:
def can_be_updated(self, media, user_id) -> bool:
updatable = True
if self.media_obj.__class__.__name__ in LONG_PLAY_MEDIA.values():
@ -700,62 +699,26 @@ class Scrobble(TimeStampedModel):
if self.is_stale:
logger.info(f"No - stale - {self.id} - {self.source}")
updatable = False
if (
self.media_obj.__class__.__name__ in ["GeoLocation"]
and not self.has_moved
):
if self.media_obj.__class__.__name__ in [
"GeoLocation"
] and not self.has_moved(media, user_id):
logger.info(f"Yes - in the same place - {self.id} - {self.source}")
updatable = True
return updatable
@property
def loc_diff(self) -> tuple:
if self.media_type != self.MediaType.GEO_LOCATION:
logger.warn("Non-location scrobble, no diff")
return tuple()
if not self.previous_by_media:
logger.warn(f"No previous location scrobble for {self}, no diff")
return tuple()
if self.media_obj == self.previous_by_media.media_obj:
logger.warn("Previous scrobble is same location, no diff")
return tuple()
return (
abs(
Decimal(self.media_obj.lat)
- Decimal(self.previous_by_media.media_obj.lat)
),
abs(
Decimal(self.media_obj.lon)
- Decimal(self.previous_by_media.media_obj.lon)
),
)
@property
def has_moved(self) -> bool:
@classmethod
def has_moved(cls, new_location: GeoLocation, user_id: int) -> bool:
"""Given a new location, let us know if we've moved from there"""
has_moved = False
if self.media_type != self.MediaType.GEO_LOCATION:
logger.warn("Non-location scrobble means nothing")
return has_moved
past_scrobbles = Scrobble.objects.filter(
media_type="GeoLocation",
user_id=user_id,
).order_by("-timestamp")[1:POINTS_FOR_MOVEMENT_HISTORY]
past_points = [s.media_obj for s in past_scrobbles]
scrobble = self
all_moves = []
for i in range(POINTS_FOR_MOVEMENT_HISTORY):
loc_diff = self.loc_diff
if loc_diff and loc_diff[0] < 0.001 and loc_diff[1] > 0.001:
all_moves.append(True)
else:
all_moves.append(False)
scrobble = self.previous_by_media
if not False in all_moves:
has_moved = True
return has_moved
return new_location.has_moved(past_points)
@property
def media_obj(self):
@ -807,11 +770,6 @@ class Scrobble(TimeStampedModel):
media_query = models.Q(**{key: media})
scrobble_data[key + "_id"] = media.id
if key == "geo_location":
# For geo locations, it's a time sequence, not per location, so
# just get the last location we know of
media_query = models.Q(media_type="GeoLocation")
scrobble = (
cls.objects.filter(
media_query,
@ -821,7 +779,7 @@ class Scrobble(TimeStampedModel):
.first()
)
if scrobble and scrobble.can_be_updated:
if scrobble and scrobble.can_be_updated(media, user_id):
source = scrobble_data["source"]
mtype = media.__class__.__name__
logger.info(