my.location.home: simplify config format, make it a bit more robust + tests
This commit is contained in:
parent
d8ed780e36
commit
4666378f7e
3 changed files with 68 additions and 34 deletions
|
@ -2,52 +2,63 @@
|
|||
Simple location provider, serving as a fallback when more detailed data isn't available
|
||||
'''
|
||||
from dataclasses import dataclass
|
||||
from datetime import datetime, date
|
||||
from datetime import datetime, date, time, timezone
|
||||
from functools import lru_cache
|
||||
from typing import Optional, Sequence, Tuple, Union
|
||||
from typing import Optional, Sequence, Tuple, Union, cast
|
||||
|
||||
from ..core.common import fromisoformat
|
||||
|
||||
from my.config import location as L
|
||||
user_config = L.home
|
||||
from my.config import location as user_config
|
||||
|
||||
|
||||
DateIsh = Union[datetime, str]
|
||||
DateIsh = Union[datetime, date, str]
|
||||
|
||||
# todo hopefully reasonable? might be nice to add name or something too
|
||||
LatLon = Tuple[float, float]
|
||||
|
||||
@dataclass
|
||||
class home(user_config):
|
||||
class Config(user_config):
|
||||
home: Union[
|
||||
LatLon, # either single, 'current' location
|
||||
Sequence[Tuple[ # or, a sequence of location history
|
||||
DateIsh, # date when you moved to
|
||||
LatLon, # the location
|
||||
]]
|
||||
]
|
||||
# TODO could make current Optional and somehow determine from system settings?
|
||||
# todo possibly also could be core config.. but not sure
|
||||
current: LatLon
|
||||
|
||||
'''
|
||||
First element is location, the second is the date when you left it (datetime/ISO string)
|
||||
'''
|
||||
past: Sequence[Tuple[LatLon, DateIsh]] = ()
|
||||
# todo test for proper localized/not localized handling as well
|
||||
# todo make sure they are increasing
|
||||
|
||||
|
||||
@property
|
||||
def _past(self) -> Sequence[Tuple[LatLon, datetime]]:
|
||||
def _history(self) -> Sequence[Tuple[datetime, LatLon]]:
|
||||
home1 = self.home
|
||||
# tood ugh, can't test for isnstance LatLon, it's a tuple itself
|
||||
home2: Sequence[Tuple[DateIsh, LatLon]]
|
||||
if isinstance(home1[0], tuple):
|
||||
# already a sequence
|
||||
home2 = cast(Sequence[Tuple[DateIsh, LatLon]], home1)
|
||||
else:
|
||||
# must be a pair of coordinates. also doesn't really matter which date to pick?
|
||||
loc = cast(LatLon, home1)
|
||||
home2 = [(datetime.min, loc)]
|
||||
|
||||
# todo cache?
|
||||
res = []
|
||||
for loc, x in self.past:
|
||||
for x, loc in home2:
|
||||
dt: datetime
|
||||
if isinstance(x, str):
|
||||
dt = fromisoformat(x)
|
||||
else:
|
||||
elif isinstance(x, datetime):
|
||||
dt = x
|
||||
res.append((loc, dt))
|
||||
else:
|
||||
dt = datetime.combine(x, time.min)
|
||||
# todo not sure about doing it here, but makes it easier to compare..
|
||||
if dt.tzinfo is None:
|
||||
dt = dt.replace(tzinfo=timezone.utc)
|
||||
res.append((dt, loc))
|
||||
res = list(sorted(res, key=lambda p: p[0]))
|
||||
return res
|
||||
|
||||
|
||||
|
||||
from ..core.cfg import make_config
|
||||
config = make_config(home)
|
||||
config = make_config(Config)
|
||||
|
||||
|
||||
@lru_cache(maxsize=None)
|
||||
|
@ -55,7 +66,12 @@ def get_location(dt: datetime) -> LatLon:
|
|||
'''
|
||||
Interpolates the location at dt
|
||||
'''
|
||||
for loc, pdt in config._past:
|
||||
if dt <= pdt:
|
||||
if dt.tzinfo is None:
|
||||
dt = dt.replace(tzinfo=timezone.utc)
|
||||
hist = list(reversed(config._history))
|
||||
for pdt, loc in hist:
|
||||
if dt >= pdt:
|
||||
return loc
|
||||
return config.current
|
||||
else:
|
||||
# I guess the most reasonable is to fallback on the first location
|
||||
return hist[-1][1]
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue