souplesse/README.md

153 lines
5.2 KiB
Markdown
Raw Normal View History

2024-10-29 21:49:06 +00:00
# Souplesse
2024-11-13 22:10:01 +00:00
![screenshot](screenshot.png)
2024-10-29 21:49:06 +00:00
_This readme describes what may someday be, not what is today_
Reads a GPX file which you generated by cycling around, and tells you
interesting(sic) things about it.
The principle we aspire to is that the measurement is subsidary to the
ride, not the purpose of the ride. The purpose of the ride is to enjoy
cycling, or to see new places, or to get from A to B, and the purpose
of Souplesse is to see if we can get any useful numbers out of the
riding you were doing anyway without making you do more of it.
So, the general idea is that given some ride data it can tell you how
long/how often you spent at a given level of effort (e.g. heart rate),
or output (power, speed, cadence, rate of ascent).
## Canned views
### ride view
graph of (selected variables) / time, with buttons to select variables
slider for threshold level
all points above threshold are highlighted and interval times above
threshold shown
zoom in/out on time range
show the selected points on a map
### calendar view
note this will need some kind of server-side storage so that the
system remembers all your gpx files
shows dates that you rode
for each ride, show time at effort
some kind of slider for effort level
## Query view
Not yet decided if this is useful, something that allows graphs of
arbitrary functions of properties (e.g. to look at power/cadence
ratio, or ... some other weirdness)
2024-10-30 17:17:48 +00:00
# Tech notes
2024-11-01 00:02:32 +00:00
Use `nix-shell`. Inside the shell
2024-11-12 00:20:48 +00:00
* use `make` to build frontend (Elm) and backend (Haskell/Yesod)
2024-11-01 00:02:32 +00:00
* run tests with `cabal test --test-show-details=always`: if you don't
2024-10-30 17:17:48 +00:00
ask for details it won't tell you about incomplete pattern matches
2024-11-01 00:02:32 +00:00
* run the app with `cabal run`
2024-10-30 17:17:48 +00:00
2024-10-29 21:49:06 +00:00
----
_Do not look below this line_
## WIP, Puzzles and TODO
2024-11-12 00:20:48 +00:00
* rename Track to Gpx, it deals only with parsing.
* can we lose this "if isJust lat && isJust lon && isJust ts" wart?
* probably we should store points in a more efficient form than
a singly-linked list
2024-11-01 00:00:36 +00:00
2024-11-14 09:47:40 +00:00
* boring stuff like auth[zn]
2024-11-01 00:00:36 +00:00
* need a web server in haskell that
2024-11-14 09:47:40 +00:00
* [done] accepts file upload and parses the gpx file
* [done] serves the data points in some format elm can digest easily
* [done] need a database of some kind so the data can be saved
2024-11-11 21:21:49 +00:00
* [done] frontend can get data from backend
* [done] for DX, backend can serve the js files needed by frontend
* [ad hoc] we only have yesod-core, may need other parts as well
2024-11-14 09:47:40 +00:00
* [done] detect and refuse uploads which overlap an existing time frame (http 409) so that we can script upload-all-the-tracks.
2024-11-10 13:13:22 +00:00
* could we converge the Point and Trkpt to make sql better?
2024-11-11 21:21:49 +00:00
* [done] move Store into Point
2024-11-14 09:47:40 +00:00
* on timeline, show power, cadence, speed, height, ascent (checkboxes)
* done some
* need speed and ascent
* need checkboxes
* zoom gesture on graphs causes map to adjust
* zooming map causes graphs to adjust
* threshold display: adjust vertical slider to show time spent at
or above a particular intensity. Indicate somehow the length of
each continuous stretch at that intensity
2024-11-11 21:21:49 +00:00
2024-11-10 13:13:22 +00:00
* calendar displays sessions. a session is a sequence of measurements
describing a ride or a race or a trip. we can extract potential
sessions from the data by looking for series of points not more than
x (10?) minutes apart, but the rider may override that. Consider: I
ride solo to the start point of a group ride, join a tandem partner
to do the group ride, then ride solo home. There is not necessarily
ten minutes between them.
after a new track is uploaded, we look at all the points covered by
draft sessions, and rearrange them to cover the new points. Draft
sessions are then presented to the rider who may approve them
as-is - perhaps involving other data collection as well ("perceived
effort" or "which bike setup was this" or ...) - or chop them up
2024-11-21 21:34:44 +00:00
using information they have but that the computer doesn't
2024-11-10 13:13:22 +00:00
in theory we don't even need draft sessions and we could have the
2024-11-12 00:20:48 +00:00
rider create sessions from the calendar page. However, that's a GET
and might be slow if it has to figure out what all the sessions would
be every time someone looks at it. So the draft session is just to
precompute that and make the view easier
2024-11-10 13:13:22 +00:00
the summary of a session is for display on the calendar and might
change depending on the nature of the training effort. e.g.
for a long slow ride we show total distance, for interval training
we show time spent in HR zones ...
2024-11-17 16:16:46 +00:00
----
2024-11-21 21:34:44 +00:00
start and end marks can be drag targets but we also need to know where
they are when they're not being dragged
2024-11-17 16:16:46 +00:00
2024-11-21 21:34:44 +00:00
selectedRange start, duration
2024-11-17 16:16:46 +00:00
## Postgres
I run the postgresql devel server using Docker instead of changing my
global NixOS configuration, so that it's self-contained and I can
start and stop it when I want to
2024-11-04 23:37:48 +00:00
```
docker run -p 5432:5432 --name souplesse-postgres -e POSTGRES_USER=souplesse -e POSTGRES_PASSWORD=secret -d postgres
nix-shell -p postgresql --run "psql -h localhost -U souplesse -p 5432"
```
2024-11-07 18:59:56 +00:00
## Sample data
The upload form deliberately doesn't have CSRF (for now, at least) so
that you can chuck a bunch of GPX files at it using curl
```
for i in tmp/Tracks/*.gpx ; do curl --form f1=@$i 'http://localhost:3000/upload'; done
```
This should be safe to do repeatedly because it will refuse upload of
tracks where the database already contains any points in the time
range of the uploaded track