700 lines
33 KiB
Python
700 lines
33 KiB
Python
"""One-shot builder for notebooks/05_intra_run.ipynb.
|
||
|
||
Run with: uv run python notebooks/_build_05.py
|
||
"""
|
||
|
||
from __future__ import annotations
|
||
|
||
import json
|
||
from pathlib import Path
|
||
|
||
|
||
def md(*lines: str) -> dict:
|
||
return {"cell_type": "markdown", "metadata": {}, "source": _join(lines)}
|
||
|
||
|
||
def code(*lines: str) -> dict:
|
||
return {
|
||
"cell_type": "code",
|
||
"execution_count": None,
|
||
"metadata": {},
|
||
"outputs": [],
|
||
"source": _join(lines),
|
||
}
|
||
|
||
|
||
def _join(lines):
|
||
text = "\n".join(lines)
|
||
# Jupyter expects a list where each entry ends with \n except possibly the last.
|
||
parts = text.split("\n")
|
||
return [p + ("\n" if i < len(parts) - 1 else "") for i, p in enumerate(parts)]
|
||
|
||
|
||
cells: list[dict] = []
|
||
|
||
cells.append(md(
|
||
"# 05 — Intra-run dynamics",
|
||
"",
|
||
"Within-run signals from lap-level splits: cardiac drift, cadence/stride, route-controlled pace, HR-zone distribution.",
|
||
"",
|
||
"**Data note.** This project's sync was via the Garmin live API (`sync.py`), not the official zip export, so `activity_fit_files` is empty and per-second FIT data isn't available. Everything here runs on `activity_splits` (per-mile laps, ~2 000 rows). When FIT files arrive via `ingest_export.py`, these same analyses upgrade to per-second resolution — only the loader needs to change.",
|
||
))
|
||
|
||
cells.append(code(
|
||
"import sys",
|
||
"sys.path.insert(0, '..')",
|
||
"import numpy as np",
|
||
"import pandas as pd",
|
||
"import matplotlib.pyplot as plt",
|
||
"from analysis import (",
|
||
" open_conn, load_splits, decoupling,",
|
||
" assign_hr_zone, cluster_routes, haversine_km,",
|
||
")",
|
||
"",
|
||
"conn = open_conn()",
|
||
"splits = load_splits(conn) # running only by default",
|
||
"print(f'{len(splits):,} splits across {splits.activity_id.nunique():,} runs, "
|
||
"{splits.start_time_local.min().date()} → {splits.start_time_local.max().date()}')",
|
||
))
|
||
|
||
# ----- Section 1: cardiac drift -----
|
||
cells.append(md(
|
||
"## 1. Cardiac drift (Pa:Hr decoupling)",
|
||
"",
|
||
"Within a single run, divide the laps into first half and second half. For each half compute the duration-weighted ratio `speed / HR` — essentially \"pace per heartbeat,\" the gold-standard aerobic-fitness index. Decoupling = how much that ratio falls between halves.",
|
||
"",
|
||
"$$\\text{decoupling}\\;\\% = \\left(\\frac{(\\text{speed}/\\text{HR})_{1st}}{(\\text{speed}/\\text{HR})_{2nd}} - 1\\right) \\times 100$$",
|
||
"",
|
||
"Friel's rule of thumb for steady aerobic runs:",
|
||
"- **< 5 %** — aerobically developed",
|
||
"- **5–10 %** — moderate drift; sustainable",
|
||
"- **> 10 %** — significant drift; pace was unsustainable or it was a hot/hard day",
|
||
"",
|
||
"Negative values mean you ran *more efficiently* in the second half (a negative split or conservative opener).",
|
||
))
|
||
|
||
cells.append(code(
|
||
"dec = decoupling(splits, min_splits=6)",
|
||
"print(f'{len(dec)} runs with ≥ 6 splits')",
|
||
"dec['decoupling_pct'].describe().round(2)",
|
||
))
|
||
|
||
cells.append(code(
|
||
"fig, axes = plt.subplots(1, 2, figsize=(13, 4.5))",
|
||
"",
|
||
"ax = axes[0]",
|
||
"ax.hist(dec['decoupling_pct'], bins=30, edgecolor='white')",
|
||
"for x, lab, color in [(5, 'good <5%', '#2a9d8f'), (10, 'caution 10%', '#e76f51')]:",
|
||
" ax.axvline(x, color=color, ls='--', lw=1.2, label=lab)",
|
||
"ax.axvline(0, color='gray', lw=0.8)",
|
||
"ax.set_xlabel('decoupling (%)'); ax.set_ylabel('runs')",
|
||
"ax.set_title(f'Pa:Hr decoupling distribution (n={len(dec)})')",
|
||
"ax.legend()",
|
||
"",
|
||
"ax = axes[1]",
|
||
"sc = ax.scatter(dec['start_time_local'], dec['decoupling_pct'],",
|
||
" c=dec['distance_km'], cmap='viridis', alpha=0.75, s=28)",
|
||
"ax.axhline(5, color='#2a9d8f', ls='--', lw=1)",
|
||
"ax.axhline(10, color='#e76f51', ls='--', lw=1)",
|
||
"ax.axhline(0, color='gray', lw=0.6)",
|
||
"ax.set_ylabel('decoupling (%)'); ax.set_title('decoupling over time (color = distance km)')",
|
||
"plt.colorbar(sc, ax=ax, label='distance (km)')",
|
||
"fig.autofmt_xdate()",
|
||
"fig.tight_layout()",
|
||
))
|
||
|
||
cells.append(md(
|
||
"### Aerobic runs only — the clean view",
|
||
"",
|
||
"Decoupling is only interpretable on **steady aerobic** efforts. Filter to runs ≥ 8 km with avg HR < 165 (well below threshold for a sub-3:30 marathoner) and look at the trend. Less drift over time = better aerobic conditioning.",
|
||
))
|
||
|
||
cells.append(code(
|
||
"aerobic = dec[(dec['distance_km'] >= 8) & (dec['avg_hr'] < 165)].copy()",
|
||
"aerobic['quarter'] = aerobic['start_time_local'].dt.to_period('Q').dt.to_timestamp()",
|
||
"print(f'{len(aerobic)} aerobic runs')",
|
||
"",
|
||
"fig, ax = plt.subplots(figsize=(11, 4.5))",
|
||
"ax.scatter(aerobic['start_time_local'], aerobic['decoupling_pct'],",
|
||
" c=aerobic['avg_hr'], cmap='magma_r', s=40, alpha=0.85)",
|
||
"",
|
||
"# rolling median (smooth trend)",
|
||
"aerobic_sorted = aerobic.sort_values('start_time_local')",
|
||
"rolling = aerobic_sorted.set_index('start_time_local')['decoupling_pct'].rolling('120D', min_periods=5).median()",
|
||
"ax.plot(rolling.index, rolling.values, color='black', lw=2, label='120-day rolling median')",
|
||
"",
|
||
"ax.axhline(5, color='#2a9d8f', ls='--', lw=1)",
|
||
"ax.axhline(10, color='#e76f51', ls='--', lw=1)",
|
||
"ax.set_ylabel('decoupling (%)')",
|
||
"ax.set_title('Cardiac drift on aerobic runs (≥8 km, avg HR < 165)')",
|
||
"ax.legend(loc='upper right')",
|
||
"fig.autofmt_xdate()",
|
||
"fig.tight_layout()",
|
||
))
|
||
|
||
cells.append(code(
|
||
"# Year-over-year aerobic decoupling summary",
|
||
"aerobic.groupby('year').agg(",
|
||
" n=('decoupling_pct', 'size'),",
|
||
" median_drift=('decoupling_pct', 'median'),",
|
||
" mean_drift=('decoupling_pct', 'mean'),",
|
||
" pct_under_5=('decoupling_pct', lambda s: (s < 5).mean() * 100),",
|
||
").round(2)",
|
||
))
|
||
|
||
# ----- Section 1b: per-second race-day decoupling from FIT files -----
|
||
cells.append(md(
|
||
"## 1b. Per-second decoupling — race-day deep dive",
|
||
"",
|
||
"Lap-level decoupling (above) is coarse. With FIT files linked (since the takeout-export ingest), we can read the per-second `heart_rate` and `enhanced_speed` directly and compute Friel's decoupling without the noise from aid-station stops and lap rounding.",
|
||
"",
|
||
"**Method:**",
|
||
"1. Drop the first 5 min (warmup) and last 2 min (cooldown / finish sprint).",
|
||
"2. Drop records with speed < 0.5 m/s — aid-station pauses don't drag the mean.",
|
||
"3. Slice the moving time into equal-time chunks (halves or quartiles).",
|
||
"4. For each chunk: `efficiency = mean(speed) / mean(HR)`.",
|
||
"5. `decoupling % = (eff_first / eff_chunk − 1) × 100` — positive = drift.",
|
||
"",
|
||
"Friel's rule: < 5% on a steady aerobic run = aerobically developed; > 10% = unsustainable pacing or fueling deficit. Race-day numbers are expected to be higher than training (you push the back half), but *how much* higher matters.",
|
||
))
|
||
|
||
cells.append(code(
|
||
"from analysis import load_fit_records, fit_decoupling, fit_rolling_efficiency",
|
||
"",
|
||
"race_meta = pd.read_sql('''",
|
||
" SELECT a.activity_id, a.start_time_local, a.distance_m/1000 AS km, a.avg_hr",
|
||
" FROM activities a JOIN activity_fit_files f USING(activity_id)",
|
||
" WHERE a.distance_m >= 45000 AND a.distance_m <= 60000",
|
||
" AND a.activity_type='running'",
|
||
" ORDER BY a.start_time_local",
|
||
"''', conn, parse_dates=['start_time_local'])",
|
||
"print(f'{len(race_meta)} prior 50K-class races with FIT linked:')",
|
||
"race_meta",
|
||
))
|
||
|
||
cells.append(code(
|
||
"# Load all four FITs once, cache the records frames",
|
||
"race_records = {}",
|
||
"for _, r in race_meta.iterrows():",
|
||
" aid = int(r['activity_id'])",
|
||
" race_records[aid] = load_fit_records(conn, aid)",
|
||
" print(f\" {r['start_time_local'].date()} aid={aid} records={len(race_records[aid]):,}\")",
|
||
))
|
||
|
||
cells.append(md(
|
||
"### Halves and quartiles — when does the drift start?",
|
||
))
|
||
|
||
cells.append(code(
|
||
"halves = []",
|
||
"quarts = []",
|
||
"for _, r in race_meta.iterrows():",
|
||
" aid = int(r['activity_id'])",
|
||
" h = fit_decoupling(race_records[aid], segments=2)",
|
||
" h.insert(0, 'race', r['start_time_local'].date())",
|
||
" halves.append(h)",
|
||
" q = fit_decoupling(race_records[aid], segments=4)",
|
||
" q.insert(0, 'race', r['start_time_local'].date())",
|
||
" quarts.append(q)",
|
||
"halves_df = pd.concat(halves, ignore_index=True)",
|
||
"quarts_df = pd.concat(quarts, ignore_index=True)",
|
||
"",
|
||
"print('Per-race half-by-half decoupling:')",
|
||
"(halves_df.pivot(index='race', columns='segment', values='decoupling_pct')",
|
||
" .round(1).rename(columns={1:'Q1+Q2', 2:'Q3+Q4'}))",
|
||
))
|
||
|
||
cells.append(code(
|
||
"print('Quartile decoupling — where the drift actually starts:')",
|
||
"(quarts_df.pivot(index='race', columns='segment', values='decoupling_pct')",
|
||
" .round(1).rename(columns={1:'Q1',2:'Q2',3:'Q3',4:'Q4'}))",
|
||
))
|
||
|
||
cells.append(code(
|
||
"# Visualise quartile decoupling — bars per race, grouped by quartile",
|
||
"fig, ax = plt.subplots(figsize=(11, 4.5))",
|
||
"races = sorted(quarts_df['race'].unique())",
|
||
"x = np.arange(len(races))",
|
||
"w = 0.2",
|
||
"colors = ['#2a9d8f', '#e9c46a', '#f4a261', '#e76f51'] # cool → hot",
|
||
"for i in range(1, 5):",
|
||
" vals = [quarts_df[(quarts_df.race == r) & (quarts_df.segment == i)]['decoupling_pct'].iloc[0]",
|
||
" for r in races]",
|
||
" ax.bar(x + (i - 2.5) * w, vals, w, color=colors[i-1], label=f'Q{i}')",
|
||
"ax.axhline(0, color='black', lw=0.5)",
|
||
"ax.axhline(10, color='gray', ls='--', lw=1, label='Friel \"unsustainable\" threshold')",
|
||
"ax.set_xticks(x)",
|
||
"ax.set_xticklabels([str(r) for r in races])",
|
||
"ax.set_ylabel('decoupling (%)')",
|
||
"ax.set_title('Per-second decoupling by race quartile — the wall lands in Q3 every time')",
|
||
"ax.legend(loc='upper left', ncol=5)",
|
||
"fig.tight_layout()",
|
||
))
|
||
|
||
cells.append(md(
|
||
"### Rolling efficiency curves — when does the wheels-come-off moment hit?",
|
||
"",
|
||
"5-minute rolling speed/HR over elapsed time. Flat = pacing matches HR. Falling curve = decoupling in progress. The y-axis is the same physical quantity Friel's method aggregates, just plotted continuously.",
|
||
))
|
||
|
||
cells.append(code(
|
||
"fig, axes = plt.subplots(len(race_meta), 1, figsize=(13, 2.5 * len(race_meta)),",
|
||
" sharex=True, squeeze=False)",
|
||
"axes = axes.flatten()",
|
||
"for ax, (_, r) in zip(axes, race_meta.iterrows()):",
|
||
" aid = int(r['activity_id'])",
|
||
" rolled = fit_rolling_efficiency(race_records[aid], window_s=300)",
|
||
" valid = rolled.dropna(subset=['rolling_efficiency'])",
|
||
" ax.plot(valid['elapsed_min'], valid['rolling_efficiency'], color='#264653', lw=1.5)",
|
||
" # Normalise against the first 30 minutes' mean to show % drop",
|
||
" base = valid.loc[valid['elapsed_min'] < 30, 'rolling_efficiency'].mean()",
|
||
" if base and base > 0:",
|
||
" ax2 = ax.twinx()",
|
||
" ax2.plot(valid['elapsed_min'], (valid['rolling_efficiency'] / base - 1) * 100,",
|
||
" color='#e76f51', lw=1, alpha=0.6)",
|
||
" ax2.set_ylabel('% vs first 30 min', color='#e76f51', fontsize=9)",
|
||
" ax2.axhline(0, color='#e76f51', ls=':', lw=0.6, alpha=0.5)",
|
||
" ax.set_ylabel('speed / HR')",
|
||
" ax.set_title(f\"{r['start_time_local'].date()} — {r['km']:.1f} km, avg HR {r['avg_hr']:.0f}\",",
|
||
" fontsize=10)",
|
||
"axes[-1].set_xlabel('elapsed minutes')",
|
||
"fig.suptitle('Rolling efficiency through each race (5-min window)', y=1.01)",
|
||
"fig.tight_layout()",
|
||
))
|
||
|
||
cells.append(md(
|
||
"### HR and pace traces, side by side",
|
||
"",
|
||
"Same data, separated: HR (left axis, magma colour-scale) and pace (right axis, inverted so faster is up). The interesting moments are where the curves *diverge* — HR climbing while pace stays flat (drift) or HR steady while pace falls (just tired legs).",
|
||
))
|
||
|
||
cells.append(code(
|
||
"fig, axes = plt.subplots(len(race_meta), 1, figsize=(13, 2.8 * len(race_meta)),",
|
||
" sharex=True, squeeze=False)",
|
||
"axes = axes.flatten()",
|
||
"for ax, (_, r) in zip(axes, race_meta.iterrows()):",
|
||
" aid = int(r['activity_id'])",
|
||
" rec = race_records[aid].dropna(subset=['heart_rate','speed_mps','elapsed_s'])",
|
||
" rec = rec[rec['speed_mps'] > 0.5]",
|
||
" em = rec['elapsed_s'] / 60",
|
||
" rolled_hr = rec['heart_rate'].rolling(300, min_periods=30).mean()",
|
||
" rolled_pace = (1 / rec['speed_mps']) * 1000 / 60",
|
||
" rolled_pace = rolled_pace.rolling(300, min_periods=30).mean()",
|
||
" ax.plot(em, rolled_hr, color='#9b2226', lw=1.4, label='HR (5-min avg)')",
|
||
" ax.set_ylabel('HR (bpm)', color='#9b2226')",
|
||
" ax.tick_params(axis='y', labelcolor='#9b2226')",
|
||
" ax2 = ax.twinx()",
|
||
" ax2.plot(em, rolled_pace, color='#264653', lw=1.4, label='pace (5-min avg)')",
|
||
" ax2.set_ylabel('pace (min/km)', color='#264653')",
|
||
" ax2.tick_params(axis='y', labelcolor='#264653')",
|
||
" ax2.invert_yaxis() # faster = up",
|
||
" ax.set_title(f\"{r['start_time_local'].date()} — {r['km']:.1f} km\", fontsize=10)",
|
||
"axes[-1].set_xlabel('elapsed minutes')",
|
||
"fig.suptitle('HR (red) and pace (dark) — divergence = decoupling', y=1.01)",
|
||
"fig.tight_layout()",
|
||
))
|
||
|
||
cells.append(md(
|
||
"### Per-second vs per-mile decoupling — sanity check",
|
||
"",
|
||
"How does the FIT-derived number compare to the lap-level decoupling we computed in §1? Per-second is correctly excluding stopped time and lap rounding, so should be **lower** than the per-mile number for the same race — but the qualitative ranking should agree.",
|
||
))
|
||
|
||
cells.append(code(
|
||
"# Pull the per-mile (§1) value for each race and compare to per-second",
|
||
"lap_dec = dec.set_index('activity_id')['decoupling_pct']",
|
||
"rows = []",
|
||
"for _, r in race_meta.iterrows():",
|
||
" aid = int(r['activity_id'])",
|
||
" ps = halves_df[(halves_df.race == r['start_time_local'].date()) & (halves_df.segment == 2)]['decoupling_pct'].iloc[0]",
|
||
" lap = lap_dec.get(aid, float('nan'))",
|
||
" rows.append({'race': r['start_time_local'].date(), 'km': r['km'],",
|
||
" 'per_mile_decoupling_pct': round(lap, 1),",
|
||
" 'per_second_decoupling_pct': round(ps, 1),",
|
||
" 'delta': round(lap - ps, 1) if not pd.isna(lap) else None})",
|
||
"pd.DataFrame(rows)",
|
||
))
|
||
|
||
cells.append(md(
|
||
"### What this means for the 50-mile",
|
||
"",
|
||
"The per-second view localises the drift: in every prior race the wheels come off around the **4-hour mark** (between Q2 and Q3). For the 50-mile that's roughly halfway through the race — exactly when fueling errors stop being recoverable.",
|
||
"",
|
||
"Three concrete implications:",
|
||
"",
|
||
"1. **Front-load fueling.** The textbook glycogen depletion curve says 90 min of running on stored glycogen, then performance falls off without external carbs. Q1 (the easy half) shouldn't be a fueling holiday — every aid station, every hour, from the start.",
|
||
"2. **Recalibrate pace by HR, not by feel.** The rolling-efficiency plots show HR rising while pace falls. Setting an HR ceiling (e.g. Z2 top = 143 bpm for the long run, slightly higher for race) and *enforcing it* would flatten the Q3 collapse.",
|
||
"3. **What success looks like on Sept 12.** A 50-mile race executed cleanly should look like the *first half* of these 50K curves repeated twice. If the Q3 wall reappears around hour 4–5, treat it as a planned aid-station break to top up calories before continuing.",
|
||
))
|
||
|
||
# ----- Section 2: cadence + stride -----
|
||
cells.append(md(
|
||
"## 2. Cadence and stride length",
|
||
"",
|
||
"At a given pace, faster runners tend to have **higher cadence and shorter stride**. Watching cadence-vs-pace and stride-vs-pace by year shows whether form is shifting independently of fitness.",
|
||
"",
|
||
"Garmin's `averageRunCadence` per split is already **both-legs** steps-per-minute (typical running range 150–185). `strideLength` is in cm.",
|
||
))
|
||
|
||
cells.append(code(
|
||
"form = splits.dropna(subset=['averageRunCadence', 'strideLength', 'pace_min_per_km']).copy()",
|
||
"form = form[form['averageRunCadence'] > 0].copy() # zeros = walking/standing intervals",
|
||
"form['cadence_spm'] = form['averageRunCadence'] # already both-legs SPM",
|
||
"form['stride_m'] = form['strideLength'] / 100",
|
||
"form = form[(form['cadence_spm'] >= 140) & (form['cadence_spm'] <= 200)] # drop walks/junk",
|
||
"print(f'{len(form):,} clean form splits, {form.activity_id.nunique()} runs')",
|
||
"",
|
||
"fig, axes = plt.subplots(1, 2, figsize=(13, 5), sharex=True)",
|
||
"years = sorted(form['year'].unique())",
|
||
"cmap = plt.cm.viridis(np.linspace(0, 0.9, len(years)))",
|
||
"",
|
||
"for c, y in zip(cmap, years):",
|
||
" d = form[form['year'] == y]",
|
||
" axes[0].scatter(d['pace_min_per_km'], d['cadence_spm'], s=8, alpha=0.35, color=c, label=str(y))",
|
||
" axes[1].scatter(d['pace_min_per_km'], d['stride_m'], s=8, alpha=0.35, color=c, label=str(y))",
|
||
"",
|
||
"axes[0].set_ylabel('cadence (steps/min, both legs)')",
|
||
"axes[1].set_ylabel('stride length (m)')",
|
||
"for ax in axes:",
|
||
" ax.set_xlabel('pace (min/km)')",
|
||
" ax.invert_xaxis() # faster runs to the right",
|
||
"axes[0].legend(title='year', loc='lower left', fontsize=8)",
|
||
"axes[0].set_title('Cadence vs pace')",
|
||
"axes[1].set_title('Stride length vs pace')",
|
||
"fig.tight_layout()",
|
||
))
|
||
|
||
cells.append(md(
|
||
"### Form at a controlled pace",
|
||
"",
|
||
"Bin splits into a narrow easy-pace band (5:30–6:30 min/km) and look at cadence / stride / vertical metrics year-over-year. Holding pace constant strips out the obvious \"faster = higher cadence\" effect and isolates technique drift.",
|
||
))
|
||
|
||
cells.append(code(
|
||
"band = form[(form['pace_min_per_km'] >= 5.5) & (form['pace_min_per_km'] <= 6.5)].copy()",
|
||
"vert = splits.dropna(subset=['verticalOscillation', 'verticalRatio', 'groundContactTime'])",
|
||
"vert = vert[(vert['pace_min_per_km'] >= 5.5) & (vert['pace_min_per_km'] <= 6.5)]",
|
||
"",
|
||
"summary = band.groupby('year').agg(",
|
||
" n_splits=('cadence_spm', 'size'),",
|
||
" cadence_med=('cadence_spm', 'median'),",
|
||
" stride_med=('stride_m', 'median'),",
|
||
")",
|
||
"vsum = vert.groupby('year').agg(",
|
||
" vert_osc_cm=('verticalOscillation', 'median'),",
|
||
" vert_ratio=('verticalRatio', 'median'),",
|
||
" gct_ms=('groundContactTime', 'median'),",
|
||
")",
|
||
"summary = summary.join(vsum).round(2)",
|
||
"summary",
|
||
))
|
||
|
||
cells.append(code(
|
||
"fig, axes = plt.subplots(1, 3, figsize=(14, 4))",
|
||
"summary['cadence_med'].plot(kind='bar', ax=axes[0], color='#264653')",
|
||
"axes[0].set_ylabel('cadence (spm)'); axes[0].set_title('Cadence at easy pace')",
|
||
"axes[0].set_ylim(summary['cadence_med'].min() - 3, summary['cadence_med'].max() + 3)",
|
||
"",
|
||
"summary['stride_med'].plot(kind='bar', ax=axes[1], color='#2a9d8f')",
|
||
"axes[1].set_ylabel('stride length (m)'); axes[1].set_title('Stride at easy pace')",
|
||
"axes[1].set_ylim(summary['stride_med'].min() - 0.05, summary['stride_med'].max() + 0.05)",
|
||
"",
|
||
"if summary['vert_osc_cm'].notna().any():",
|
||
" summary['vert_osc_cm'].plot(kind='bar', ax=axes[2], color='#e76f51')",
|
||
" axes[2].set_ylabel('vertical oscillation (cm)'); axes[2].set_title('Vertical bounce at easy pace')",
|
||
" axes[2].set_ylim(summary['vert_osc_cm'].min() - 0.5, summary['vert_osc_cm'].max() + 0.5)",
|
||
"else:",
|
||
" axes[2].text(0.5, 0.5, 'no vertical-osc data', ha='center', va='center', transform=axes[2].transAxes)",
|
||
"",
|
||
"for ax in axes:",
|
||
" ax.set_xlabel('year')",
|
||
"fig.suptitle('Form metrics in the 5:30–6:30 min/km band, year over year')",
|
||
"fig.tight_layout()",
|
||
))
|
||
|
||
# ----- Section 3: GPS route clustering -----
|
||
cells.append(md(
|
||
"## 3. Route clustering — pace controlled for terrain",
|
||
"",
|
||
"Raw pace year-over-year mixes terrain, weather, intent. Cluster runs by their **start coordinates** (greedy haversine, 250 m radius) and you get \"my usual routes.\" Within a cluster the route is roughly the same, so pace differences are mostly fitness, not geography.",
|
||
))
|
||
|
||
cells.append(code(
|
||
"starts = (splits.dropna(subset=['startLatitude', 'startLongitude'])",
|
||
" .groupby('activity_id')",
|
||
" .agg(lat=('startLatitude', 'first'),",
|
||
" lon=('startLongitude', 'first'),",
|
||
" start_time=('start_time_local', 'first'),",
|
||
" distance_km=('distance_m', lambda s: s.sum() / 1000),",
|
||
" avg_hr=('avg_hr', 'mean'),",
|
||
" avg_pace=('pace_min_per_km', 'mean')))",
|
||
"starts['cluster'] = cluster_routes(starts['lat'].values, starts['lon'].values, radius_km=0.25)",
|
||
"print(f'{len(starts)} runs with start coords; {(starts.cluster >= 0).sum()} clustered, "
|
||
"{(starts.cluster == -1).sum()} singletons')",
|
||
"",
|
||
"top = (starts[starts.cluster >= 0]",
|
||
" .groupby('cluster')",
|
||
" .agg(n=('cluster', 'size'),",
|
||
" lat=('lat', 'median'),",
|
||
" lon=('lon', 'median'),",
|
||
" first=('start_time', 'min'),",
|
||
" last=('start_time', 'max'),",
|
||
" med_dist=('distance_km', 'median'))",
|
||
" .sort_values('n', ascending=False))",
|
||
"top.head(10)",
|
||
))
|
||
|
||
cells.append(code(
|
||
"fig, ax = plt.subplots(figsize=(9, 7))",
|
||
"",
|
||
"# unclustered points faint",
|
||
"unc = starts[starts.cluster == -1]",
|
||
"ax.scatter(unc['lon'], unc['lat'], color='lightgray', s=12, alpha=0.6, label='singletons')",
|
||
"",
|
||
"# top-10 clusters colored",
|
||
"top10 = top.head(10).index.tolist()",
|
||
"cmap = plt.cm.tab10(np.linspace(0, 1, len(top10)))",
|
||
"for c, cl in zip(cmap, top10):",
|
||
" pts = starts[starts.cluster == cl]",
|
||
" ax.scatter(pts['lon'], pts['lat'], color=c, s=40, alpha=0.8,",
|
||
" label=f'route #{cl} (n={len(pts)})')",
|
||
"",
|
||
"ax.set_xlabel('longitude'); ax.set_ylabel('latitude')",
|
||
"ax.set_title('Run start points — top 10 recurring routes')",
|
||
"ax.legend(loc='best', fontsize=8)",
|
||
"ax.set_aspect('equal', adjustable='datalim')",
|
||
"fig.tight_layout()",
|
||
))
|
||
|
||
cells.append(md(
|
||
"### Pace progression within each frequent route",
|
||
"",
|
||
"Now restrict to clusters with ≥5 runs and plot pace over time per route. Slopes here are much cleaner than the global pace trend because terrain is held constant.",
|
||
))
|
||
|
||
cells.append(code(
|
||
"frequent = top[top['n'] >= 5].index.tolist()",
|
||
"print(f'{len(frequent)} routes with ≥5 runs')",
|
||
"",
|
||
"if frequent:",
|
||
" n_cols = min(3, len(frequent))",
|
||
" n_rows = (len(frequent) + n_cols - 1) // n_cols",
|
||
" fig, axes = plt.subplots(n_rows, n_cols, figsize=(5 * n_cols, 3.2 * n_rows),",
|
||
" squeeze=False, sharey=True)",
|
||
" for ax, cl in zip(axes.flat, frequent):",
|
||
" d = (starts[starts.cluster == cl]",
|
||
" .dropna(subset=['avg_pace'])",
|
||
" .sort_values('start_time'))",
|
||
" ax.scatter(d['start_time'], d['avg_pace'], s=30, alpha=0.75, color='#264653')",
|
||
" if len(d) >= 3:",
|
||
" # rolling median by date",
|
||
" rd = d.set_index('start_time')['avg_pace'].rolling('120D', min_periods=2).median()",
|
||
" ax.plot(rd.index, rd.values, color='#e76f51', lw=1.5)",
|
||
" ax.invert_yaxis() # faster up",
|
||
" ax.set_title(f'route #{cl} — n={len(d)}, ~{top.loc[cl, \"med_dist\"]:.1f} km')",
|
||
" ax.set_ylabel('pace (min/km)')",
|
||
" # blank unused axes",
|
||
" for ax in axes.flat[len(frequent):]:",
|
||
" ax.set_visible(False)",
|
||
" fig.suptitle('Per-route pace over time (terrain held roughly constant)')",
|
||
" fig.tight_layout()",
|
||
"else:",
|
||
" print('No clusters with ≥5 runs.')",
|
||
))
|
||
|
||
# ----- Section 4: HR zones (Garmin-configured, FIT-based when available) -----
|
||
cells.append(md(
|
||
"## 4. HR-zone time-in-zone (Garmin-configured zones, per-second when possible)",
|
||
"",
|
||
"**Zones come from Garmin's `heartRateZones.json` (training method: HR_MAX),** not estimated from observed HR. Lactate-threshold HR sits at 182 inside Z4.",
|
||
"",
|
||
"| Zone | range (bpm) | feel | role |",
|
||
"|------|-------------|------|------|",
|
||
"| Z1 | 102–122 | walk / recovery | active rest |",
|
||
"| Z2 | 123–143 | conversational | **long-run target** |",
|
||
"| Z3 | 144–164 | tempo | the \"junk-miles middle\" |",
|
||
"| Z4 | 165–185 | threshold (LTHR=182) | hard sustained |",
|
||
"| Z5 | 186–209 | VO₂ max | intervals |",
|
||
"",
|
||
"For each activity, time-in-zone comes from `activity_time_in_zone` (precomputed by `compute_time_in_zone.py`):",
|
||
"- **`source='fit'`** — per-second HR from the FIT file. Each record's `dt` (typically 1 s) goes into whichever zone its HR falls in. Accurate even when laps span zone boundaries.",
|
||
"- **`source='lap'`** — fallback for activities without a linked FIT. The whole lap's duration is assigned to whichever zone the *lap's average* HR sits in. Smears across boundaries, biases toward middle zones.",
|
||
"",
|
||
"**Polarized-training rule (Seiler):** elites accumulate ~80% of weekly time in Z1+Z2 and ~20% in Z4+Z5, with little Z3.",
|
||
))
|
||
|
||
cells.append(code(
|
||
"from analysis import HR_ZONES_USER",
|
||
"",
|
||
"tiz = pd.read_sql('''",
|
||
" SELECT t.activity_id, t.z1_s, t.z2_s, t.z3_s, t.z4_s, t.z5_s, t.total_s, t.source,",
|
||
" a.start_time_local, a.activity_type, a.distance_m, a.duration_s",
|
||
" FROM activity_time_in_zone t",
|
||
" JOIN activities a USING(activity_id)",
|
||
" WHERE a.activity_type IN ('running','trail_running')",
|
||
"''', conn, parse_dates=['start_time_local'])",
|
||
"tiz['week'] = tiz['start_time_local'].dt.to_period('W-MON').dt.start_time",
|
||
"tiz['year'] = tiz['start_time_local'].dt.year",
|
||
"",
|
||
"print(f'{len(tiz)} activities with cached time-in-zone')",
|
||
"print(' source breakdown:')",
|
||
"print(tiz['source'].value_counts().to_string())",
|
||
"print(f' fit coverage: {(tiz.source==\"fit\").mean()*100:.0f}% of running activities')",
|
||
))
|
||
|
||
cells.append(md(
|
||
"### Sanity check: FIT vs lap method on the same race",
|
||
"",
|
||
"On the same activity, how different are the two estimates? Take the 2025-09-20 race (8 hours, 28k FIT records) and compute both, then compare. The lap method should over-weight whichever zone the typical lap average falls in (here, Z3) and under-count time spent in adjacent zones because boundary-crossing laps get rounded to one zone.",
|
||
))
|
||
|
||
cells.append(code(
|
||
"from analysis import load_fit_records, time_in_zone_from_fit, time_in_zone_from_splits",
|
||
"",
|
||
"race_aid = race_meta.iloc[-1]['activity_id'] # most recent 50K (2025-09-20)",
|
||
"race_date = race_meta.iloc[-1]['start_time_local'].date()",
|
||
"",
|
||
"fit_tiz = time_in_zone_from_fit(race_records[int(race_aid)])",
|
||
"race_splits = pd.read_sql(",
|
||
" 'SELECT avg_hr, duration_s FROM activity_splits WHERE activity_id = ?',",
|
||
" conn, params=[int(race_aid)]",
|
||
")",
|
||
"lap_tiz = time_in_zone_from_splits(race_splits)",
|
||
"",
|
||
"compare = pd.DataFrame({",
|
||
" 'FIT (per-sec) min': {k: round(v / 60, 1) for k, v in fit_tiz.items()},",
|
||
" 'lap (avg-HR) min': {k: round(v / 60, 1) for k, v in lap_tiz.items()},",
|
||
"}).reindex(['Z1','Z2','Z3','Z4','Z5']).fillna(0)",
|
||
"compare.loc['total'] = compare.sum()",
|
||
"compare['delta (min)'] = (compare['FIT (per-sec) min'] - compare['lap (avg-HR) min']).round(1)",
|
||
"print(f'Race {race_date} — FIT vs lap time-in-zone:')",
|
||
"compare",
|
||
))
|
||
|
||
cells.append(code(
|
||
"# Weekly time-in-zone (hours) using whichever method was available per activity",
|
||
"wk_cols = ['z1_s','z2_s','z3_s','z4_s','z5_s']",
|
||
"weekly = tiz.groupby('week')[wk_cols].sum() / 3600",
|
||
"weekly.columns = ['Z1','Z2','Z3','Z4','Z5']",
|
||
"",
|
||
"fig, ax = plt.subplots(figsize=(14, 4.8))",
|
||
"colors = ['#264653', '#2a9d8f', '#e9c46a', '#f4a261', '#e76f51']",
|
||
"ax.stackplot(weekly.index, weekly.T.values, labels=weekly.columns, colors=colors, alpha=0.92)",
|
||
"ax.set_ylabel('hours / week')",
|
||
"ax.set_title('Weekly running time by HR zone (Garmin-configured zones)')",
|
||
"ax.legend(loc='upper left', ncol=5, fontsize=9)",
|
||
"fig.autofmt_xdate()",
|
||
"fig.tight_layout()",
|
||
))
|
||
|
||
cells.append(md(
|
||
"### Polarized index over time",
|
||
"",
|
||
"Collapse to three buckets:",
|
||
"- **easy** = Z1 + Z2 (HR ≤ 143)",
|
||
"- **moderate \"junk\"** = Z3 (144–164)",
|
||
"- **hard** = Z4 + Z5 (HR ≥ 165, threshold and up)",
|
||
"",
|
||
"Polarized: high easy, low moderate, modest hard. Pyramidal: high easy, modest moderate, low hard. Threshold-heavy: lots of moderate.",
|
||
))
|
||
|
||
cells.append(code(
|
||
"buckets = pd.DataFrame({",
|
||
" 'easy': weekly[['Z1','Z2']].sum(axis=1),",
|
||
" 'moderate': weekly['Z3'],",
|
||
" 'hard': weekly[['Z4','Z5']].sum(axis=1),",
|
||
"})",
|
||
"totals = buckets.sum(axis=1)",
|
||
"pct = buckets.div(totals.replace(0, np.nan), axis=0) * 100",
|
||
"",
|
||
"fig, axes = plt.subplots(2, 1, figsize=(14, 7), sharex=True)",
|
||
"",
|
||
"ax = axes[0]",
|
||
"ax.stackplot(pct.index, pct[['easy','moderate','hard']].T.values,",
|
||
" labels=['easy (Z1+Z2)', 'moderate (Z3)', 'hard (Z4+Z5)'],",
|
||
" colors=['#2a9d8f', '#e9c46a', '#e76f51'], alpha=0.9)",
|
||
"ax.axhline(80, color='black', ls='--', lw=0.8, label='80% easy target')",
|
||
"ax.set_ylabel('% of weekly time'); ax.set_ylim(0, 100)",
|
||
"ax.set_title('Polarized-training split (weekly)')",
|
||
"ax.legend(loc='lower left', ncol=4, fontsize=9)",
|
||
"",
|
||
"ax = axes[1]",
|
||
"rolling_pct = pct.rolling(4, min_periods=2).mean()",
|
||
"for col, c in [('easy','#2a9d8f'), ('moderate','#e9c46a'), ('hard','#e76f51')]:",
|
||
" ax.plot(rolling_pct.index, rolling_pct[col], color=c, lw=2, label=col)",
|
||
"ax.axhline(80, color='#2a9d8f', ls='--', lw=0.8)",
|
||
"ax.axhline(20, color='#e76f51', ls='--', lw=0.8)",
|
||
"ax.set_ylabel('% (4-week rolling mean)')",
|
||
"ax.legend(loc='best', fontsize=9)",
|
||
"fig.autofmt_xdate()",
|
||
"fig.tight_layout()",
|
||
))
|
||
|
||
cells.append(md(
|
||
"### Yearly summary + FIT-method coverage",
|
||
"",
|
||
"`fit_coverage_%` shows what fraction of each year's activities had a linked FIT (and therefore got per-second zones). 2026's lower coverage reflects activities that synced via the live API but aren't in the takeout dump.",
|
||
))
|
||
|
||
cells.append(code(
|
||
"yearly_hours = (tiz.groupby('year')[wk_cols].sum() / 3600).round(1)",
|
||
"yearly_hours.columns = ['Z1','Z2','Z3','Z4','Z5']",
|
||
"yearly_pct = yearly_hours.div(yearly_hours.sum(axis=1), axis=0) * 100",
|
||
"",
|
||
"out = pd.concat({'hours': yearly_hours, '%': yearly_pct.round(1)}, axis=1)",
|
||
"out['easy_pct'] = (yearly_pct[['Z1','Z2']].sum(axis=1)).round(1)",
|
||
"out['hard_pct'] = (yearly_pct[['Z4','Z5']].sum(axis=1)).round(1)",
|
||
"fit_coverage = tiz.groupby('year').apply(",
|
||
" lambda g: (g['source']=='fit').mean() * 100, include_groups=False",
|
||
").round(0)",
|
||
"out['fit_coverage_%'] = fit_coverage",
|
||
"out",
|
||
))
|
||
|
||
cells.append(md(
|
||
"### Race-build vs base-period zone distribution",
|
||
"",
|
||
"Compare what training looked like in the 12 weeks before each prior 50K race vs the rest of the year. A serious build should shift time into Z2 (long aerobic) and Z4 (threshold/tempo) and away from Z3.",
|
||
))
|
||
|
||
cells.append(code(
|
||
"race_dates = race_meta['start_time_local']",
|
||
"tiz['phase'] = 'base'",
|
||
"for rd in race_dates:",
|
||
" build_start = rd - pd.Timedelta(weeks=12)",
|
||
" mask = (tiz['start_time_local'] >= build_start) & (tiz['start_time_local'] < rd)",
|
||
" tiz.loc[mask, 'phase'] = f'build {rd.year}'",
|
||
"",
|
||
"phase_hours = (tiz.groupby('phase')[wk_cols].sum() / 3600).round(1)",
|
||
"phase_hours.columns = ['Z1','Z2','Z3','Z4','Z5']",
|
||
"phase_pct = (phase_hours.div(phase_hours.sum(axis=1), axis=0) * 100).round(1)",
|
||
"phase_pct['easy_Z1+Z2'] = (phase_pct['Z1'] + phase_pct['Z2']).round(1)",
|
||
"phase_pct['junk_Z3'] = phase_pct['Z3']",
|
||
"phase_pct['hard_Z4+Z5'] = (phase_pct['Z4'] + phase_pct['Z5']).round(1)",
|
||
"phase_pct[['easy_Z1+Z2','junk_Z3','hard_Z4+Z5']].sort_index()",
|
||
))
|
||
|
||
cells.append(md(
|
||
"## What's left when FIT files are ingested",
|
||
"",
|
||
"All four sections now use per-second FIT data where it's linked (349 of 378 activities, 92%). Remaining lap-only activities are mostly old multi-sport / triathlon legs that no FIT was uploaded for. Useful follow-ups:",
|
||
"",
|
||
"- **Cadence stability** — plot cadence over elapsed time within a long run; quantify the drop in the final 15 %.",
|
||
"- **GPS polylines for route clustering** — current §3 uses start coordinates only; with full FIT GPS tracks, match routes by Hausdorff distance (more accurate than start-only).",
|
||
"- **Decoupling vs fueling protocol** — once the user logs even informal fueling notes for a few long runs, regress decoupling against carb intake.",
|
||
))
|
||
|
||
notebook = {
|
||
"cells": cells,
|
||
"metadata": {
|
||
"kernelspec": {"display_name": ".venv", "language": "python", "name": "python3"},
|
||
"language_info": {"name": "python"},
|
||
},
|
||
"nbformat": 4,
|
||
"nbformat_minor": 5,
|
||
}
|
||
|
||
out = Path(__file__).parent / "05_intra_run.ipynb"
|
||
out.write_text(json.dumps(notebook, indent=1))
|
||
print(f"wrote {out} ({out.stat().st_size:,} bytes, {len(cells)} cells)")
|