Skip to main content

Track time spent on projects, backed by Git Annex

Project description

coverage report PyPI version REUSE status

⚠️ This tool is early development. The most basic time tracking feature (recording, deletion, time frame search) as well as syncing are implemented though.

annextimelog - ⏱️ Git Annex-backed Time Tracking

This is a brainstorm for a Git Annex-backed time tracker. The idea originated across some of my Mastodon threads:

The gist is that I was (and still am) unhappy with the existing time tracking solutions. I worked with hledger's timeclock and timewarrior each for quite some time and built my own workflow and scripts around them.

✅ Requirements

Over the years, the below features turned out to be my personal requirements for a time-tracking system (TL;DR: easy and intuitive recording, hassle-free syncing, data export for further analysis). Here is a table comparing annextimelog with timewarrior and hledger timeclock:

✅ = feature available, 🟡 = partly available, ❌ = not available

feature timewarrior hledger timeclock annextimelog
precise start and end times ✅ as git-annex metadata
tracking of overlapping/simultaneous periods 🟡 (separate files) ✅ backend can do it
nice, colourful, graphical summary 🟡 ✅ with Python rich, more planned
plain text data storage 🟡 buried in git-annex branch
git-friendly, merge conflict free data format 🟡¹ 🟡¹ ✅ git-annex’ own merge strategy
arbitrary tags attachable to tracked periods 🟡 hledger tags² ✅ just git-annex metadata
arbitrary notes attachable to tracked periods 🟡³ 🟡 hledger tags² ✅ just git-annex metadata
tags can have values ✅ hledger tags² ✅ just git-annex metadata
files attach-/linkable to tracked periods 🟡 path as file: tag 🟡 annexed files, linking is planned
cli to start, stop, edit, etc. tracked periods ✅⁴ ❌ own scripts needed 🟡 very basic, more planned
plugin system 🟡⁵ 🟡⁶ (hledger’s own) ❌ git-style plugin system planned
data export to common format ✅ (JSON) ✅ (CSV, JSON) ✅ as timeclock, JSON, cli commands
syncing functionality built-in ✅ git-annex’s purpose is syncing
multi-user support 🟡 e.g. use tag user=NAME

¹last line is always modified, merge conflicts can arise when working from different machines

²hledger tags have limitations, e.g. no spaces, colons, commas, etc.

³timewarrior annotations can't contain newlines for example. I wrote an extension to edit your annotation in your $EDITOR and optionally GPG-encrypt it, which lets you add newlines. Quite an inconvenience.

⁴timewarrior’s cli has some nasty inconveniences (e.g. no shortcut for ‘yesterday’, must painfully type out the full date, no intelligence to operate only on yesterday, gets confused and errors out in certain combinations of start/end times, etc…)

⁵timewarrior extensions (here mine) are just fed the data via STDIN, not other command-line arguments. Not as useful as the git-style plugin system.

⁶for the analysis part, hledger plugins can be used. But as there is no actual cli to manage the data, there’s no plugin system for that.

🛠️ Implementation

Surprisingly, many of the above requirements can be fulfilled without reinventing the wheel by employing git-annex, an extension to git that enables (among many other powerful file syncing things) attaching metadata to files. Git Annex provides sophisticated mechanisms to sync (multiple) git repositories without interaction and can also resolve certain merge conflicts on its own. I had the following design in mind, of which most is already implemented:

  • Every tracked period (working on a project, sleeping, whatever) is represented by an annexed file.
    • This file is created with some random, unique content and name without further meaning (e.g. just a UUID).
    • The file's content will never change. In fact, it doesn't even matter and the file might be git annex drop --forced right after creation.
    • The file's name also stays the same. Renames could cause git merge conflicts.
    • Period files would have a common extension (e.g. .t to keep it short) so that git-annex can be instructed to not want to copy them around and complain if they are missing. (e.g. git annex wanted . 'not include=*.t')
    • Period files would be (automatically/regularly) sorted into a YYYY/MM/DD as well as YYYY/WW (calendar week) folder structure and may appear in multiple locations if they overlap (e.g. when tracking sleep, a period likely spans two days). This allows for optimization of some queries and operations (e.g. showing only the current day, week or month). Git Annex still knows they're the same thing even if they're scattered all over the place.
    • Period metadata is stored as git annex metadata:
      • Metadata can be merged without conflicts by git annex. The most recent change wins, but the history stays available.
      • Fields:
        • start and end time (ISO format UTC)
        • tags (tag metadata field)
        • arbitray metadata with values (cli with atl track 08:00 work @home would start tracking a work period starting at 08:00 local time and set the location metadata field to home)
          • one of those fields is the note field, which can be edited in the $EDITOR to add arbitrary tex, even with newlines, emojis, whatever, possibly encrypted (but not with git-annex's mechanism)
  • The cli would be a bit similar to timewarrior's, and could look like this:
    • atl track 08:00 work @home            # start working from home (tags=work, location=home)
      atl track 08:00 work location=home    # same
      atl tag project1              # add a tag to currently running period
      atl note "bla"                # add a note to currently running period
      atl note                      # opens $EDITOR to edit note
      atl note -e "bla"             # same, but encrypt store note encrypted
      atl stop                      # stop current period tracking
      
      atl day|week|month            # show list/chart of tracked periods, with dynamic number (newest=1, etc.)
      atl note :3                   # edit note of the third-last tracked period
      
      atl track yesterday18:00 - 21:00 "coding session"   # track 3h of past event yesterday
      atl track y18:00 - 21:00 "coding session"           # shorter form
      atl track 2023-12-01T18:00 - 2023-12-01T21:00 "coding session" # timewarrior-style form
      
      atl sync                      # run 'git annex assist' to sync up
      

📦 Installation

You can run this tool if you have nix installed:

# drop into a temporary shell with the command available
nix shell gitlab:nobodyinperson/annextimelog

# install it
nix profile install gitlab:nobodyinperson/annextimelog

On Arch Linux you can install from the AUR with your favorite helper, or directly with pacman from this user repository.

# use an AUR helper to install
paru -S annextimelog

Otherwise, you can install it like any other Python package, e.g. with pip or better pipx:

pipx install annextimelog

# latest development version
pipx install git+https://gitlab.com/nobodyinperson/annextimelog

Note that in this case you will need to install git-annex manually.

Any of the above makes the annextimelog (or atl) command available.

❓ Usage

# Show help page
atl --help

# Show exactly what's going on underneath (more -v → more detail)
atl -vvvvv ...

# Add a remote to sync to
atl git remote add myserver git@myserver.com:...

# Sync status with git annex
atl sync

# Track a time period with metadata
atl track 10:00 15:00 work @home ="make a title with the equals sign" :"make a note with a colon"
atl tr 15:00 - 10min ago code @home # timewarrior-style time ranges

# Show events from this week
atl summary --week
atl su -w # short form
atl su -b 2023-01-01 -e 2023-02-01  # specific time period

# Delete an event
atl rm 3QicA4G4

# other Output formats
atl -O timeclock  # output hledger timeclock format
atl -O timeclock | hledger -f timeclock:- bal --daily # analyse with hledger
atl -O json       # output JSON

There's some example periods in the doc folder you can try out.

⚙️ Configuration

annextimelog reuses git's configuration system. You can configure annextimelog per-repo (atl git config ...), per-user (atl git config --global ...) or system-wide (typically in /etc/gitconfig).

# list all annextimelog configs known to git
atl git config --get-regexp '^annextimelog'

# open the git config in your $EDITOR
atl git config --edit

# let the week start at Sunday
atl git config annextimelog.weekstartssunday true

# Don't commit, except in 'annextimelog sync'.
# This speeds up things but reduces granularity to undo changes
atl git config annextimelog.commit false

# Skip steps that are not strictly necessary.
# This speeds up things but might leave the repository in a less ordered state.
atl git config annextimelog.fast true

🛠️ Development

This project uses poetry, so you can run the following in this repository to get into a development environment:

poetry install
poetry shell
# now you're in a shell with everything set up

Other:

# Auto-run mypy when file changes:
watchexec -w . 'poetry run mypy .'

# Auto-run tests when file changes:
watchexec -w . 'poetry run coverage run --source=annextimelog -m unittest -v -b;coverage html;coverage report'

# Test how a sequence of command-line args is interpreted as event metadata
poetry run python -c 'from annextimelog.token import *;import sys;from rich import print;print(list(Token.from_strings(sys.argv[1:])))' work @home note=bla myfield+=one,two,three 2h ago until now

# Run tests against a different Python version with nix: (careful to update both versions below!)
LC_ALL=C.UTF-8 nix-shell -p poetry python310 --run 'poetry env use python3.10;poetry install;poetry run python -m unittest -v'

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

annextimelog-0.5.0.tar.gz (28.2 kB view hashes)

Uploaded Source

Built Distribution

annextimelog-0.5.0-py3-none-any.whl (39.1 kB view hashes)

Uploaded Python 3

Supported by

AWS AWS Cloud computing and Security Sponsor Datadog Datadog Monitoring Fastly Fastly CDN Google Google Download Analytics Microsoft Microsoft PSF Sponsor Pingdom Pingdom Monitoring Sentry Sentry Error logging StatusPage StatusPage Status page