92 lines
2.5 KiB
Python
92 lines
2.5 KiB
Python
'''
|
|
Rescuetime (phone activity tracking) data.
|
|
'''
|
|
REQUIRES = [
|
|
'git+https://github.com/karlicoss/rescuexport',
|
|
]
|
|
|
|
from pathlib import Path
|
|
from datetime import datetime, timedelta
|
|
from typing import Sequence, Iterable
|
|
|
|
from .core import get_files, LazyLogger
|
|
from .core.common import mcachew
|
|
from .core.error import Res, split_errors
|
|
|
|
from my.config import rescuetime as config
|
|
|
|
|
|
log = LazyLogger(__name__, level='info')
|
|
|
|
|
|
def inputs() -> Sequence[Path]:
|
|
return get_files(config.export_path)
|
|
|
|
|
|
import rescuexport.dal as dal
|
|
DAL = dal.DAL
|
|
Entry = dal.Entry
|
|
|
|
|
|
@mcachew(depends_on=lambda: inputs())
|
|
def entries() -> Iterable[Res[Entry]]:
|
|
dal = DAL(inputs())
|
|
it = dal.entries()
|
|
yield from dal.entries()
|
|
|
|
|
|
def groups(gap: timedelta=timedelta(hours=3)) -> Iterable[Res[Sequence[Entry]]]:
|
|
vit, eit = split_errors(entries(), ET=Exception)
|
|
yield from eit
|
|
import more_itertools
|
|
from more_itertools import split_when
|
|
yield from split_when(vit, lambda a, b: (b.dt - a.dt) > gap)
|
|
|
|
|
|
# todo automatic dataframe interface?
|
|
from .core.pandas import DataFrameT, as_dataframe
|
|
def dataframe() -> DataFrameT:
|
|
return as_dataframe(entries())
|
|
|
|
|
|
from .core import stat, Stats
|
|
def stats() -> Stats:
|
|
return {
|
|
**stat(groups),
|
|
**stat(entries),
|
|
}
|
|
|
|
|
|
# basically, hack config and populate it with fake data? fake data generated by DAL, but the rest is handled by this?
|
|
|
|
from typing import Iterator
|
|
from contextlib import contextmanager
|
|
# todo take seed, or what?
|
|
@contextmanager
|
|
def fake_data(rows: int=1000) -> Iterator[None]:
|
|
# todo also disable cachew automatically for such things?
|
|
from .core.cachew import disabled_cachew
|
|
from .core.cfg import override_config
|
|
from tempfile import TemporaryDirectory
|
|
with disabled_cachew(), override_config(config) as cfg, TemporaryDirectory() as td:
|
|
tdir = Path(td)
|
|
cfg.export_path = tdir
|
|
f = tdir / 'rescuetime.json'
|
|
import json
|
|
f.write_text(json.dumps(dal.fake_data_generator(rows=rows)))
|
|
yield
|
|
# TODO ok, now it's something that actually could run on CI!
|
|
# todo would be kinda nice if doctor could run against the fake data, to have a basic health check of the module?
|
|
|
|
|
|
def fill_influxdb() -> None:
|
|
from .core import influxdb
|
|
it = (dict(
|
|
dt=e.dt,
|
|
duration_d=e.duration_s,
|
|
tags=dict(activity=e.activity),
|
|
) for e in entries() if isinstance(e, Entry)) # TODO handle errors in core.influxdb
|
|
influxdb.fill(it, measurement=__name__)
|
|
|
|
|
|
# TODO lots of garbage in dir()? maybe need to del the imports...
|