hae-vault 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (160) hide show
  1. package/.env.example +7 -0
  2. package/CLAUDE.md +220 -0
  3. package/README.md +206 -0
  4. package/SKILL.md +60 -0
  5. package/dist/cli/dashboard.d.ts +3 -0
  6. package/dist/cli/dashboard.d.ts.map +1 -0
  7. package/dist/cli/dashboard.js +206 -0
  8. package/dist/cli/dashboard.js.map +1 -0
  9. package/dist/cli/import.d.ts +3 -0
  10. package/dist/cli/import.d.ts.map +1 -0
  11. package/dist/cli/import.js +78 -0
  12. package/dist/cli/import.js.map +1 -0
  13. package/dist/cli/index.d.ts +3 -0
  14. package/dist/cli/index.d.ts.map +1 -0
  15. package/dist/cli/index.js +31 -0
  16. package/dist/cli/index.js.map +1 -0
  17. package/dist/cli/info.d.ts +5 -0
  18. package/dist/cli/info.d.ts.map +1 -0
  19. package/dist/cli/info.js +34 -0
  20. package/dist/cli/info.js.map +1 -0
  21. package/dist/cli/metrics.d.ts +3 -0
  22. package/dist/cli/metrics.d.ts.map +1 -0
  23. package/dist/cli/metrics.js +20 -0
  24. package/dist/cli/metrics.js.map +1 -0
  25. package/dist/cli/query.d.ts +3 -0
  26. package/dist/cli/query.d.ts.map +1 -0
  27. package/dist/cli/query.js +18 -0
  28. package/dist/cli/query.js.map +1 -0
  29. package/dist/cli/serve.d.ts +3 -0
  30. package/dist/cli/serve.d.ts.map +1 -0
  31. package/dist/cli/serve.js +19 -0
  32. package/dist/cli/serve.js.map +1 -0
  33. package/dist/cli/sleep.d.ts +3 -0
  34. package/dist/cli/sleep.d.ts.map +1 -0
  35. package/dist/cli/sleep.js +19 -0
  36. package/dist/cli/sleep.js.map +1 -0
  37. package/dist/cli/summary.d.ts +3 -0
  38. package/dist/cli/summary.d.ts.map +1 -0
  39. package/dist/cli/summary.js +53 -0
  40. package/dist/cli/summary.js.map +1 -0
  41. package/dist/cli/trends.d.ts +3 -0
  42. package/dist/cli/trends.d.ts.map +1 -0
  43. package/dist/cli/trends.js +77 -0
  44. package/dist/cli/trends.js.map +1 -0
  45. package/dist/cli/watch.d.ts +12 -0
  46. package/dist/cli/watch.d.ts.map +1 -0
  47. package/dist/cli/watch.js +89 -0
  48. package/dist/cli/watch.js.map +1 -0
  49. package/dist/cli/workouts.d.ts +3 -0
  50. package/dist/cli/workouts.d.ts.map +1 -0
  51. package/dist/cli/workouts.js +19 -0
  52. package/dist/cli/workouts.js.map +1 -0
  53. package/dist/config.d.ts +9 -0
  54. package/dist/config.d.ts.map +1 -0
  55. package/dist/config.js +25 -0
  56. package/dist/config.js.map +1 -0
  57. package/dist/db/importLog.d.ts +5 -0
  58. package/dist/db/importLog.d.ts.map +1 -0
  59. package/dist/db/importLog.js +10 -0
  60. package/dist/db/importLog.js.map +1 -0
  61. package/dist/db/metrics.d.ts +4 -0
  62. package/dist/db/metrics.d.ts.map +1 -0
  63. package/dist/db/metrics.js +14 -0
  64. package/dist/db/metrics.js.map +1 -0
  65. package/dist/db/schema.d.ts +5 -0
  66. package/dist/db/schema.d.ts.map +1 -0
  67. package/dist/db/schema.js +100 -0
  68. package/dist/db/schema.js.map +1 -0
  69. package/dist/db/sleep.d.ts +4 -0
  70. package/dist/db/sleep.d.ts.map +1 -0
  71. package/dist/db/sleep.js +13 -0
  72. package/dist/db/sleep.js.map +1 -0
  73. package/dist/db/workouts.d.ts +4 -0
  74. package/dist/db/workouts.d.ts.map +1 -0
  75. package/dist/db/workouts.js +11 -0
  76. package/dist/db/workouts.js.map +1 -0
  77. package/dist/index.d.ts +3 -0
  78. package/dist/index.d.ts.map +1 -0
  79. package/dist/index.js +5 -0
  80. package/dist/index.js.map +1 -0
  81. package/dist/parse/metrics.d.ts +17 -0
  82. package/dist/parse/metrics.d.ts.map +1 -0
  83. package/dist/parse/metrics.js +33 -0
  84. package/dist/parse/metrics.js.map +1 -0
  85. package/dist/parse/sleep.d.ts +23 -0
  86. package/dist/parse/sleep.d.ts.map +1 -0
  87. package/dist/parse/sleep.js +58 -0
  88. package/dist/parse/sleep.js.map +1 -0
  89. package/dist/parse/time.d.ts +4 -0
  90. package/dist/parse/time.d.ts.map +1 -0
  91. package/dist/parse/time.js +41 -0
  92. package/dist/parse/time.js.map +1 -0
  93. package/dist/parse/workouts.d.ts +17 -0
  94. package/dist/parse/workouts.d.ts.map +1 -0
  95. package/dist/parse/workouts.js +24 -0
  96. package/dist/parse/workouts.js.map +1 -0
  97. package/dist/server/app.d.ts +5 -0
  98. package/dist/server/app.d.ts.map +1 -0
  99. package/dist/server/app.js +39 -0
  100. package/dist/server/app.js.map +1 -0
  101. package/dist/server/ingest.d.ts +15 -0
  102. package/dist/server/ingest.d.ts.map +1 -0
  103. package/dist/server/ingest.js +41 -0
  104. package/dist/server/ingest.js.map +1 -0
  105. package/dist/types/hae.d.ts +103 -0
  106. package/dist/types/hae.d.ts.map +1 -0
  107. package/dist/types/hae.js +2 -0
  108. package/dist/types/hae.js.map +1 -0
  109. package/dist/util/zip.d.ts +3 -0
  110. package/dist/util/zip.d.ts.map +1 -0
  111. package/dist/util/zip.js +24 -0
  112. package/dist/util/zip.js.map +1 -0
  113. package/docs/COMMANDS.md +315 -0
  114. package/docs/plans/2026-02-18-hae-vault-initial-implementation.md +2015 -0
  115. package/docs/plans/2026-02-18-readme-dashboard-design.md +213 -0
  116. package/docs/plans/2026-02-18-readme-dashboard-plan.md +1306 -0
  117. package/docs/plans/2026-02-18-zip-env-watch-design.md +213 -0
  118. package/docs/plans/2026-02-18-zip-env-watch.md +966 -0
  119. package/package.json +57 -0
  120. package/src/cli/dashboard.ts +242 -0
  121. package/src/cli/import.ts +85 -0
  122. package/src/cli/index.ts +32 -0
  123. package/src/cli/info.ts +36 -0
  124. package/src/cli/metrics.ts +20 -0
  125. package/src/cli/query.ts +17 -0
  126. package/src/cli/serve.ts +18 -0
  127. package/src/cli/sleep.ts +19 -0
  128. package/src/cli/summary.ts +58 -0
  129. package/src/cli/trends.ts +103 -0
  130. package/src/cli/watch.ts +111 -0
  131. package/src/cli/workouts.ts +19 -0
  132. package/src/config.ts +28 -0
  133. package/src/db/importLog.ts +18 -0
  134. package/src/db/metrics.ts +15 -0
  135. package/src/db/schema.ts +105 -0
  136. package/src/db/sleep.ts +15 -0
  137. package/src/db/workouts.ts +13 -0
  138. package/src/index.ts +4 -0
  139. package/src/parse/metrics.ts +50 -0
  140. package/src/parse/sleep.ts +82 -0
  141. package/src/parse/time.ts +43 -0
  142. package/src/parse/workouts.ts +42 -0
  143. package/src/server/app.ts +46 -0
  144. package/src/server/ingest.ts +68 -0
  145. package/src/types/hae.ts +94 -0
  146. package/src/util/zip.ts +24 -0
  147. package/tests/cli-watch.test.ts +64 -0
  148. package/tests/db-import-log.test.ts +40 -0
  149. package/tests/db-metrics.test.ts +44 -0
  150. package/tests/db-schema.test.ts +55 -0
  151. package/tests/db-sleep.test.ts +36 -0
  152. package/tests/db-workouts.test.ts +34 -0
  153. package/tests/ingest.test.ts +99 -0
  154. package/tests/parse-metrics.test.ts +55 -0
  155. package/tests/parse-sleep.test.ts +65 -0
  156. package/tests/parse-time.test.ts +48 -0
  157. package/tests/parse-workouts.test.ts +43 -0
  158. package/tests/types.test.ts +27 -0
  159. package/tests/util-zip.test.ts +46 -0
  160. package/tsconfig.json +19 -0
@@ -0,0 +1,206 @@
1
+ import { Command } from 'commander';
2
+ import { openDb } from '../db/schema.js';
3
+ // ── helpers ──────────────────────────────────────────────────────────────────
4
+ function pad(s, width) {
5
+ return s.padEnd(width);
6
+ }
7
+ function fmt1(n) {
8
+ if (n == null)
9
+ return '—';
10
+ return n.toFixed(1);
11
+ }
12
+ function fmtInt(n) {
13
+ if (n == null)
14
+ return '—';
15
+ return Math.round(n).toLocaleString();
16
+ }
17
+ function sec2min(s) {
18
+ if (s == null)
19
+ return '—';
20
+ return `${Math.round(s / 60)}min`;
21
+ }
22
+ function kj2kcal(kj) {
23
+ if (kj == null)
24
+ return '—';
25
+ return `${Math.round(kj / 4.184)} kcal`;
26
+ }
27
+ function arrow(first, last) {
28
+ const delta = last - first;
29
+ if (Math.abs(delta) < 0.01 * Math.abs(first || 1))
30
+ return '→';
31
+ return delta > 0 ? '↑' : '↓';
32
+ }
33
+ function ruler(label, width = 43) {
34
+ const inner = `── ${label} `;
35
+ return inner + '─'.repeat(Math.max(0, width - inner.length));
36
+ }
37
+ function workoutEmoji(name) {
38
+ const n = name.toLowerCase();
39
+ if (n.includes('run'))
40
+ return '🏃';
41
+ if (n.includes('walk'))
42
+ return '🚶';
43
+ if (n.includes('cycl') || n.includes('bike'))
44
+ return '🚴';
45
+ if (n.includes('swim'))
46
+ return '🏊';
47
+ if (n.includes('yoga'))
48
+ return '🧘';
49
+ if (n.includes('strength') || n.includes('weight') || n.includes('lift'))
50
+ return '🏋️';
51
+ if (n.includes('hike'))
52
+ return '🥾';
53
+ return '🏅';
54
+ }
55
+ function latestMetric(db, metricName, days = 7) {
56
+ const since = new Date();
57
+ since.setDate(since.getDate() - days);
58
+ const row = db.prepare(`SELECT qty FROM metrics WHERE metric = ? AND date >= ? AND qty IS NOT NULL ORDER BY ts DESC LIMIT 1`).get(metricName, since.toISOString().slice(0, 10));
59
+ return row?.qty ?? null;
60
+ }
61
+ function dailyAvgs(db, metricName, days) {
62
+ const since = new Date();
63
+ since.setDate(since.getDate() - days);
64
+ const rows = db.prepare(`SELECT date, AVG(qty) as avg_qty FROM metrics
65
+ WHERE metric = ? AND date >= ? AND qty IS NOT NULL
66
+ GROUP BY date ORDER BY date ASC`).all(metricName, since.toISOString().slice(0, 10));
67
+ return rows.map(r => r.avg_qty);
68
+ }
69
+ function sleepDailyAvgs(db, days) {
70
+ const since = new Date();
71
+ since.setDate(since.getDate() - days);
72
+ const rows = db.prepare(`SELECT asleep_h FROM sleep WHERE date >= ? AND asleep_h IS NOT NULL ORDER BY date ASC`).all(since.toISOString().slice(0, 10));
73
+ return rows.map(r => r.asleep_h);
74
+ }
75
+ function trendLine(values, label, unit, round = false) {
76
+ if (values.length < 2)
77
+ return '';
78
+ const avg = values.reduce((a, b) => a + b, 0) / values.length;
79
+ const first = values[0];
80
+ const last = values[values.length - 1];
81
+ const dir = arrow(first, last);
82
+ const fmt = round ? fmtInt : fmt1;
83
+ return ` ${pad(label + ':', 14)} ${fmt(first)} → ${fmt(last)}${unit} ${dir} (avg ${fmt(avg)})`;
84
+ }
85
+ // ── command ───────────────────────────────────────────────────────────────────
86
+ export const dashboardCommand = new Command('dashboard')
87
+ .description('Terminal dashboard: sleep, activity, heart health, workouts, trends')
88
+ .option('--days <n>', 'Trend window in days', '7')
89
+ .option('--json', 'Output raw JSON')
90
+ .action((opts) => {
91
+ const db = openDb();
92
+ const trendDays = parseInt(opts.days, 10);
93
+ const today = new Date().toISOString().slice(0, 10);
94
+ // ── sleep (last night) ─────────────────────────────────────────────────
95
+ const sleep = db.prepare(`SELECT * FROM sleep ORDER BY date DESC LIMIT 1`).get();
96
+ // ── activity (today, fallback last 2 days) ─────────────────────────────
97
+ const steps = latestMetric(db, 'step_count', 2);
98
+ const activeCal = latestMetric(db, 'active_energy_burned', 2);
99
+ const standHours = latestMetric(db, 'apple_stand_hour', 2);
100
+ // ── heart health (last 7 days) ─────────────────────────────────────────
101
+ const restingHR = latestMetric(db, 'resting_heart_rate', 7);
102
+ const hrv = latestMetric(db, 'heart_rate_variability_sdnn', 7);
103
+ // ── recent workouts ────────────────────────────────────────────────────
104
+ const workouts = db.prepare(`SELECT date, name, duration_s, calories_kj, avg_hr FROM workouts ORDER BY ts DESC LIMIT 5`).all();
105
+ // ── trends ────────────────────────────────────────────────────────────
106
+ const stepTrend = dailyAvgs(db, 'step_count', trendDays);
107
+ const hrTrend = dailyAvgs(db, 'resting_heart_rate', trendDays);
108
+ const hrvTrend = dailyAvgs(db, 'heart_rate_variability_sdnn', trendDays);
109
+ const sleepTrend = sleepDailyAvgs(db, trendDays);
110
+ // ── vault stats ───────────────────────────────────────────────────────
111
+ const metricsCount = db.prepare('SELECT COUNT(*) as c FROM metrics').get().c;
112
+ const sleepCount = db.prepare('SELECT COUNT(*) as c FROM sleep').get().c;
113
+ const workoutsCount = db.prepare('SELECT COUNT(*) as c FROM workouts').get().c;
114
+ const lastSync = db.prepare('SELECT received_at FROM sync_log ORDER BY received_at DESC LIMIT 1').get();
115
+ if (opts.json) {
116
+ console.log(JSON.stringify({
117
+ date: today,
118
+ sleep: sleep ?? null,
119
+ activity: { steps, activeCal, standHours },
120
+ heartHealth: { restingHR, hrv },
121
+ workouts,
122
+ trends: { steps: stepTrend, restingHR: hrTrend, hrv: hrvTrend, sleep: sleepTrend },
123
+ vault: { metricsCount, sleepCount, workoutsCount, lastSync: lastSync?.received_at ?? null }
124
+ }, null, 2));
125
+ return;
126
+ }
127
+ const lines = [];
128
+ lines.push(`📅 ${today} | Apple Health Vault`);
129
+ lines.push('');
130
+ // sleep
131
+ lines.push(ruler('Sleep (last night)'));
132
+ if (sleep) {
133
+ const eff = sleep.in_bed_h && sleep.asleep_h
134
+ ? Math.round((sleep.asleep_h / sleep.in_bed_h) * 100) : null;
135
+ lines.push(`😴 ${fmt1(sleep.asleep_h)}h | Efficiency: ${eff != null ? eff + '%' : '—'}`);
136
+ const light = (sleep.asleep_h ?? 0) - (sleep.deep_h ?? 0) - (sleep.rem_h ?? 0);
137
+ const deepPct = sleep.asleep_h ? Math.round(((sleep.deep_h ?? 0) / sleep.asleep_h) * 100) : 0;
138
+ const remPct = sleep.asleep_h ? Math.round(((sleep.rem_h ?? 0) / sleep.asleep_h) * 100) : 0;
139
+ const lightPct = sleep.asleep_h ? Math.round((light / sleep.asleep_h) * 100) : 0;
140
+ lines.push(` Deep: ${fmt1(sleep.deep_h)}h (${deepPct}%) | REM: ${fmt1(sleep.rem_h)}h (${remPct}%) | Light: ${fmt1(light)}h (${lightPct}%)`);
141
+ lines.push(` Awake: ${fmt1(sleep.awake_h)}h | Source: ${sleep.source ?? '—'}`);
142
+ }
143
+ else {
144
+ lines.push(' No sleep data');
145
+ }
146
+ lines.push('');
147
+ // activity
148
+ lines.push(ruler('Activity (recent)'));
149
+ const actParts = [];
150
+ if (steps != null)
151
+ actParts.push(`👟 ${fmtInt(steps)} steps`);
152
+ if (activeCal != null)
153
+ actParts.push(`🔥 ${fmtInt(activeCal)} kcal active`);
154
+ if (actParts.length > 0) {
155
+ lines.push(actParts.join(' | '));
156
+ if (standHours != null)
157
+ lines.push(` Stand hours: ${Math.round(standHours)}`);
158
+ }
159
+ else {
160
+ lines.push(' No activity data');
161
+ }
162
+ lines.push('');
163
+ // heart health
164
+ lines.push(ruler('Heart Health'));
165
+ const hh = [];
166
+ if (restingHR != null)
167
+ hh.push(`💓 Resting HR: ${Math.round(restingHR)}bpm`);
168
+ if (hrv != null)
169
+ hh.push(`HRV: ${Math.round(hrv)}ms`);
170
+ lines.push(hh.length > 0 ? hh.join(' | ') : ' No heart data');
171
+ lines.push('');
172
+ // workouts
173
+ lines.push(ruler('Recent Workouts'));
174
+ if (workouts.length > 0) {
175
+ for (const w of workouts) {
176
+ const emoji = workoutEmoji(w.name);
177
+ const namePadded = pad(w.name, 18);
178
+ const dur = sec2min(w.duration_s);
179
+ const cal = kj2kcal(w.calories_kj);
180
+ lines.push(`${emoji} ${w.date} ${namePadded} ${dur} ${cal}`);
181
+ }
182
+ }
183
+ else {
184
+ lines.push(' No workout data');
185
+ }
186
+ lines.push('');
187
+ // trends
188
+ lines.push(ruler(`${trendDays}-Day Trends`));
189
+ const tl = [
190
+ trendLine(stepTrend, 'Steps', '', true),
191
+ trendLine(sleepTrend, 'Sleep', 'h'),
192
+ trendLine(hrTrend, 'Resting HR', 'bpm', true),
193
+ trendLine(hrvTrend, 'HRV', 'ms', true),
194
+ ].filter(Boolean);
195
+ if (tl.length > 0)
196
+ lines.push(...tl);
197
+ else
198
+ lines.push(' Insufficient data for trends');
199
+ lines.push('');
200
+ // vault stats
201
+ lines.push(ruler('Vault Stats'));
202
+ lines.push(` Metrics: ${metricsCount.toLocaleString()} | Sleep: ${sleepCount} | Workouts: ${workoutsCount}`);
203
+ lines.push(` Last sync: ${lastSync?.received_at ? new Date(lastSync.received_at).toISOString().replace('T', ' ').slice(0, 16) + ' UTC' : 'never'}`);
204
+ console.log(lines.join('\n'));
205
+ });
206
+ //# sourceMappingURL=dashboard.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"dashboard.js","sourceRoot":"","sources":["../../src/cli/dashboard.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,3 @@
1
+ import { Command } from 'commander';
2
+ export declare const importCommand: Command;
3
+ //# sourceMappingURL=import.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"import.d.ts","sourceRoot":"","sources":["../../src/cli/import.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AAgDpC,eAAO,MAAM,aAAa,SAoCtB,CAAC"}
@@ -0,0 +1,78 @@
1
+ import { Command } from 'commander';
2
+ import { readFileSync } from 'node:fs';
3
+ import { createHash } from 'node:crypto';
4
+ import { openDb } from '../db/schema.js';
5
+ import { ingest } from '../server/ingest.js';
6
+ import { hasBeenImported, logImport } from '../db/importLog.js';
7
+ import { extractPayloadFromZip } from '../util/zip.js';
8
+ import { config } from '../config.js';
9
+ function sha256(buf) {
10
+ return createHash('sha256').update(buf).digest('hex');
11
+ }
12
+ function loadFile(file) {
13
+ let buf;
14
+ try {
15
+ buf = readFileSync(file);
16
+ }
17
+ catch (err) {
18
+ console.error(JSON.stringify({ error: `Cannot read file: ${String(err)}` }));
19
+ process.exit(1);
20
+ }
21
+ const hash = sha256(buf);
22
+ let payload;
23
+ if (file.toLowerCase().endsWith('.zip')) {
24
+ payload = extractPayloadFromZip(buf);
25
+ if (!payload) {
26
+ console.error(JSON.stringify({ error: 'No valid HealthAutoExport-*.json found in zip' }));
27
+ process.exit(1);
28
+ }
29
+ }
30
+ else {
31
+ try {
32
+ payload = JSON.parse(buf.toString('utf-8'));
33
+ }
34
+ catch (err) {
35
+ console.error(JSON.stringify({ error: `Invalid JSON: ${String(err)}` }));
36
+ process.exit(1);
37
+ }
38
+ if (!payload?.data) {
39
+ console.error(JSON.stringify({ error: 'Missing data field — not a valid HAE export' }));
40
+ process.exit(1);
41
+ }
42
+ }
43
+ return { payload, hash };
44
+ }
45
+ export const importCommand = new Command('import')
46
+ .description('Import a Health Auto Export JSON or ZIP file into the database')
47
+ .argument('<file>', 'Path to the HAE JSON or ZIP export file')
48
+ .option('--target <name>', 'Target name (device/person identifier)', config.target)
49
+ .option('--pretty', 'Pretty-print summary JSON', false)
50
+ .action((file, opts) => {
51
+ const db = openDb(config.dbPath);
52
+ const { payload, hash } = loadFile(file);
53
+ if (hasBeenImported(db, hash)) {
54
+ const result = { skipped: true, reason: 'already imported', file, hash };
55
+ console.log(opts.pretty ? JSON.stringify(result, null, 2) : JSON.stringify(result));
56
+ return;
57
+ }
58
+ const ingestResult = ingest(db, payload, {
59
+ target: opts.target,
60
+ sessionId: null,
61
+ automationName: 'file-import',
62
+ automationPeriod: 'manual',
63
+ });
64
+ logImport(db, file, hash, ingestResult);
65
+ const result = {
66
+ ok: true,
67
+ file,
68
+ target: opts.target,
69
+ hash,
70
+ added: {
71
+ metrics: ingestResult.metricsAdded,
72
+ sleep: ingestResult.sleepAdded,
73
+ workouts: ingestResult.workoutsAdded,
74
+ },
75
+ };
76
+ console.log(opts.pretty ? JSON.stringify(result, null, 2) : JSON.stringify(result));
77
+ });
78
+ //# sourceMappingURL=import.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"import.js","sourceRoot":"","sources":["../../src/cli/import.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,3 @@
1
+ import { Command } from 'commander';
2
+ export declare const program: Command;
3
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/cli/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AAapC,eAAO,MAAM,OAAO,SAAgB,CAAC"}
@@ -0,0 +1,31 @@
1
+ import { Command } from 'commander';
2
+ import { serveCommand } from './serve.js';
3
+ import { metricsCommand } from './metrics.js';
4
+ import { sleepCommand } from './sleep.js';
5
+ import { workoutsCommand } from './workouts.js';
6
+ import { summaryCommand } from './summary.js';
7
+ import { queryCommand } from './query.js';
8
+ import { sourcesCommand, lastSyncCommand, statsCommand } from './info.js';
9
+ import { importCommand } from './import.js';
10
+ import { watchCommand } from './watch.js';
11
+ import { dashboardCommand } from './dashboard.js';
12
+ import { trendsCommand } from './trends.js';
13
+ export const program = new Command();
14
+ program
15
+ .name('hvault')
16
+ .description('Apple Health data vault — ingest + query')
17
+ .version('0.1.0');
18
+ program.addCommand(serveCommand);
19
+ program.addCommand(importCommand);
20
+ program.addCommand(watchCommand);
21
+ program.addCommand(metricsCommand);
22
+ program.addCommand(sleepCommand);
23
+ program.addCommand(workoutsCommand);
24
+ program.addCommand(summaryCommand);
25
+ program.addCommand(queryCommand);
26
+ program.addCommand(dashboardCommand);
27
+ program.addCommand(trendsCommand);
28
+ program.addCommand(sourcesCommand);
29
+ program.addCommand(lastSyncCommand);
30
+ program.addCommand(statsCommand);
31
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/cli/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AACpC,OAAO,EAAE,YAAY,EAAE,MAAM,YAAY,CAAC;AAC1C,OAAO,EAAE,cAAc,EAAE,MAAM,cAAc,CAAC;AAC9C,OAAO,EAAE,YAAY,EAAE,MAAM,YAAY,CAAC;AAC1C,OAAO,EAAE,eAAe,EAAE,MAAM,eAAe,CAAC;AAChD,OAAO,EAAE,cAAc,EAAE,MAAM,cAAc,CAAC;AAC9C,OAAO,EAAE,YAAY,EAAE,MAAM,YAAY,CAAC;AAC1C,OAAO,EAAE,cAAc,EAAE,eAAe,EAAE,YAAY,EAAE,MAAM,WAAW,CAAC;AAC1E,OAAO,EAAE,aAAa,EAAE,MAAM,aAAa,CAAC;AAC5C,OAAO,EAAE,YAAY,EAAE,MAAM,YAAY,CAAC;AAC1C,OAAO,EAAE,gBAAgB,EAAE,MAAM,gBAAgB,CAAC;AAClD,OAAO,EAAE,aAAa,EAAE,MAAM,aAAa,CAAC;AAE5C,MAAM,CAAC,MAAM,OAAO,GAAG,IAAI,OAAO,EAAE,CAAC;AACrC,OAAO;KACJ,IAAI,CAAC,QAAQ,CAAC;KACd,WAAW,CAAC,0CAA0C,CAAC;KACvD,OAAO,CAAC,OAAO,CAAC,CAAC;AAEpB,OAAO,CAAC,UAAU,CAAC,YAAY,CAAC,CAAC;AACjC,OAAO,CAAC,UAAU,CAAC,aAAa,CAAC,CAAC;AAClC,OAAO,CAAC,UAAU,CAAC,YAAY,CAAC,CAAC;AACjC,OAAO,CAAC,UAAU,CAAC,cAAc,CAAC,CAAC;AACnC,OAAO,CAAC,UAAU,CAAC,YAAY,CAAC,CAAC;AACjC,OAAO,CAAC,UAAU,CAAC,eAAe,CAAC,CAAC;AACpC,OAAO,CAAC,UAAU,CAAC,cAAc,CAAC,CAAC;AACnC,OAAO,CAAC,UAAU,CAAC,YAAY,CAAC,CAAC;AACjC,OAAO,CAAC,UAAU,CAAC,gBAAgB,CAAC,CAAC;AACrC,OAAO,CAAC,UAAU,CAAC,aAAa,CAAC,CAAC;AAClC,OAAO,CAAC,UAAU,CAAC,cAAc,CAAC,CAAC;AACnC,OAAO,CAAC,UAAU,CAAC,eAAe,CAAC,CAAC;AACpC,OAAO,CAAC,UAAU,CAAC,YAAY,CAAC,CAAC"}
@@ -0,0 +1,5 @@
1
+ import { Command } from 'commander';
2
+ export declare const sourcesCommand: Command;
3
+ export declare const lastSyncCommand: Command;
4
+ export declare const statsCommand: Command;
5
+ //# sourceMappingURL=info.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"info.d.ts","sourceRoot":"","sources":["../../src/cli/info.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AAGpC,eAAO,MAAM,cAAc,SAUvB,CAAC;AAEL,eAAO,MAAM,eAAe,SAOxB,CAAC;AAEL,eAAO,MAAM,YAAY,SAWrB,CAAC"}
@@ -0,0 +1,34 @@
1
+ import { Command } from 'commander';
2
+ import { openDb } from '../db/schema.js';
3
+ export const sourcesCommand = new Command('sources')
4
+ .description('Show what metrics are in the DB and their date coverage')
5
+ .option('--pretty', 'Pretty-print JSON', false)
6
+ .action((opts) => {
7
+ const db = openDb();
8
+ const rows = db.prepare(`
9
+ SELECT metric, units, COUNT(*) as count, MIN(date) as first_date, MAX(date) as last_date
10
+ FROM metrics GROUP BY metric, units ORDER BY metric
11
+ `).all();
12
+ console.log(opts.pretty ? JSON.stringify(rows, null, 2) : JSON.stringify(rows));
13
+ });
14
+ export const lastSyncCommand = new Command('last-sync')
15
+ .description('Show when the last HAE push was received')
16
+ .option('--pretty', 'Pretty-print JSON', false)
17
+ .action((opts) => {
18
+ const db = openDb();
19
+ const row = db.prepare(`SELECT * FROM sync_log ORDER BY received_at DESC LIMIT 1`).get() ?? null;
20
+ console.log(opts.pretty ? JSON.stringify(row, null, 2) : JSON.stringify(row));
21
+ });
22
+ export const statsCommand = new Command('stats')
23
+ .description('Show row counts per table')
24
+ .option('--pretty', 'Pretty-print JSON', false)
25
+ .action((opts) => {
26
+ const db = openDb();
27
+ const metrics = db.prepare('SELECT COUNT(*) as count FROM metrics').get().count;
28
+ const sleep = db.prepare('SELECT COUNT(*) as count FROM sleep').get().count;
29
+ const workouts = db.prepare('SELECT COUNT(*) as count FROM workouts').get().count;
30
+ const syncs = db.prepare('SELECT COUNT(*) as count FROM sync_log').get().count;
31
+ const result = { metrics, sleep, workouts, syncs };
32
+ console.log(opts.pretty ? JSON.stringify(result, null, 2) : JSON.stringify(result));
33
+ });
34
+ //# sourceMappingURL=info.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"info.js","sourceRoot":"","sources":["../../src/cli/info.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AACpC,OAAO,EAAE,MAAM,EAAE,MAAM,iBAAiB,CAAC;AAEzC,MAAM,CAAC,MAAM,cAAc,GAAG,IAAI,OAAO,CAAC,SAAS,CAAC;KACjD,WAAW,CAAC,yDAAyD,CAAC;KACtE,MAAM,CAAC,UAAU,EAAE,mBAAmB,EAAE,KAAK,CAAC;KAC9C,MAAM,CAAC,CAAC,IAAI,EAAE,EAAE;IACf,MAAM,EAAE,GAAG,MAAM,EAAE,CAAC;IACpB,MAAM,IAAI,GAAG,EAAE,CAAC,OAAO,CAAC;;;KAGvB,CAAC,CAAC,GAAG,EAAE,CAAC;IACT,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,CAAC;AAClF,CAAC,CAAC,CAAC;AAEL,MAAM,CAAC,MAAM,eAAe,GAAG,IAAI,OAAO,CAAC,WAAW,CAAC;KACpD,WAAW,CAAC,0CAA0C,CAAC;KACvD,MAAM,CAAC,UAAU,EAAE,mBAAmB,EAAE,KAAK,CAAC;KAC9C,MAAM,CAAC,CAAC,IAAI,EAAE,EAAE;IACf,MAAM,EAAE,GAAG,MAAM,EAAE,CAAC;IACpB,MAAM,GAAG,GAAG,EAAE,CAAC,OAAO,CAAC,0DAA0D,CAAC,CAAC,GAAG,EAAE,IAAI,IAAI,CAAC;IACjG,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,GAAG,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC,CAAC;AAChF,CAAC,CAAC,CAAC;AAEL,MAAM,CAAC,MAAM,YAAY,GAAG,IAAI,OAAO,CAAC,OAAO,CAAC;KAC7C,WAAW,CAAC,2BAA2B,CAAC;KACxC,MAAM,CAAC,UAAU,EAAE,mBAAmB,EAAE,KAAK,CAAC;KAC9C,MAAM,CAAC,CAAC,IAAI,EAAE,EAAE;IACf,MAAM,EAAE,GAAG,MAAM,EAAE,CAAC;IACpB,MAAM,OAAO,GAAI,EAAE,CAAC,OAAO,CAAC,uCAAuC,CAAC,CAAC,GAAG,EAAwB,CAAC,KAAK,CAAC;IACvG,MAAM,KAAK,GAAI,EAAE,CAAC,OAAO,CAAC,qCAAqC,CAAC,CAAC,GAAG,EAAwB,CAAC,KAAK,CAAC;IACnG,MAAM,QAAQ,GAAI,EAAE,CAAC,OAAO,CAAC,wCAAwC,CAAC,CAAC,GAAG,EAAwB,CAAC,KAAK,CAAC;IACzG,MAAM,KAAK,GAAI,EAAE,CAAC,OAAO,CAAC,wCAAwC,CAAC,CAAC,GAAG,EAAwB,CAAC,KAAK,CAAC;IACtG,MAAM,MAAM,GAAG,EAAE,OAAO,EAAE,KAAK,EAAE,QAAQ,EAAE,KAAK,EAAE,CAAC;IACnD,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,CAAC,CAAC;AACtF,CAAC,CAAC,CAAC"}
@@ -0,0 +1,3 @@
1
+ import { Command } from 'commander';
2
+ export declare const metricsCommand: Command;
3
+ //# sourceMappingURL=metrics.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"metrics.d.ts","sourceRoot":"","sources":["../../src/cli/metrics.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AAGpC,eAAO,MAAM,cAAc,SAgBvB,CAAC"}
@@ -0,0 +1,20 @@
1
+ import { Command } from 'commander';
2
+ import { openDb } from '../db/schema.js';
3
+ export const metricsCommand = new Command('metrics')
4
+ .description('Query health metrics')
5
+ .requiredOption('--metric <name>', 'Metric name (e.g. step_count, heart_rate)')
6
+ .option('--days <n>', 'Last N days', '30')
7
+ .option('--pretty', 'Pretty-print JSON', false)
8
+ .action((opts) => {
9
+ const db = openDb();
10
+ const since = new Date();
11
+ since.setDate(since.getDate() - parseInt(opts.days, 10));
12
+ const rows = db.prepare(`
13
+ SELECT ts, date, qty, min, avg, max, units, source, target
14
+ FROM metrics
15
+ WHERE metric = ? AND date >= ?
16
+ ORDER BY ts ASC
17
+ `).all(opts.metric, since.toISOString().slice(0, 10));
18
+ console.log(opts.pretty ? JSON.stringify(rows, null, 2) : JSON.stringify(rows));
19
+ });
20
+ //# sourceMappingURL=metrics.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"metrics.js","sourceRoot":"","sources":["../../src/cli/metrics.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AACpC,OAAO,EAAE,MAAM,EAAE,MAAM,iBAAiB,CAAC;AAEzC,MAAM,CAAC,MAAM,cAAc,GAAG,IAAI,OAAO,CAAC,SAAS,CAAC;KACjD,WAAW,CAAC,sBAAsB,CAAC;KACnC,cAAc,CAAC,iBAAiB,EAAE,2CAA2C,CAAC;KAC9E,MAAM,CAAC,YAAY,EAAE,aAAa,EAAE,IAAI,CAAC;KACzC,MAAM,CAAC,UAAU,EAAE,mBAAmB,EAAE,KAAK,CAAC;KAC9C,MAAM,CAAC,CAAC,IAAI,EAAE,EAAE;IACf,MAAM,EAAE,GAAG,MAAM,EAAE,CAAC;IACpB,MAAM,KAAK,GAAG,IAAI,IAAI,EAAE,CAAC;IACzB,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC,OAAO,EAAE,GAAG,QAAQ,CAAC,IAAI,CAAC,IAAI,EAAE,EAAE,CAAC,CAAC,CAAC;IACzD,MAAM,IAAI,GAAG,EAAE,CAAC,OAAO,CAAC;;;;;KAKvB,CAAC,CAAC,GAAG,CAAC,IAAI,CAAC,MAAM,EAAE,KAAK,CAAC,WAAW,EAAE,CAAC,KAAK,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC;IACtD,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,CAAC;AAClF,CAAC,CAAC,CAAC"}
@@ -0,0 +1,3 @@
1
+ import { Command } from 'commander';
2
+ export declare const queryCommand: Command;
3
+ //# sourceMappingURL=query.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"query.d.ts","sourceRoot":"","sources":["../../src/cli/query.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AAGpC,eAAO,MAAM,YAAY,SAarB,CAAC"}
@@ -0,0 +1,18 @@
1
+ import { Command } from 'commander';
2
+ import { openDb } from '../db/schema.js';
3
+ export const queryCommand = new Command('query')
4
+ .description('Run raw SQL against the health database (returns JSON)')
5
+ .argument('<sql>', 'SQL query to run')
6
+ .option('--pretty', 'Pretty-print JSON', false)
7
+ .action((sql, opts) => {
8
+ const db = openDb();
9
+ try {
10
+ const rows = db.prepare(sql).all();
11
+ console.log(opts.pretty ? JSON.stringify(rows, null, 2) : JSON.stringify(rows));
12
+ }
13
+ catch (err) {
14
+ console.error(JSON.stringify({ error: String(err) }));
15
+ process.exit(1);
16
+ }
17
+ });
18
+ //# sourceMappingURL=query.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"query.js","sourceRoot":"","sources":["../../src/cli/query.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AACpC,OAAO,EAAE,MAAM,EAAE,MAAM,iBAAiB,CAAC;AAEzC,MAAM,CAAC,MAAM,YAAY,GAAG,IAAI,OAAO,CAAC,OAAO,CAAC;KAC7C,WAAW,CAAC,wDAAwD,CAAC;KACrE,QAAQ,CAAC,OAAO,EAAE,kBAAkB,CAAC;KACrC,MAAM,CAAC,UAAU,EAAE,mBAAmB,EAAE,KAAK,CAAC;KAC9C,MAAM,CAAC,CAAC,GAAW,EAAE,IAAI,EAAE,EAAE;IAC5B,MAAM,EAAE,GAAG,MAAM,EAAE,CAAC;IACpB,IAAI,CAAC;QACH,MAAM,IAAI,GAAG,EAAE,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,CAAC;QACnC,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,CAAC;IAClF,CAAC;IAAC,OAAO,GAAG,EAAE,CAAC;QACb,OAAO,CAAC,KAAK,CAAC,IAAI,CAAC,SAAS,CAAC,EAAE,KAAK,EAAE,MAAM,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC,CAAC;QACtD,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAClB,CAAC;AACH,CAAC,CAAC,CAAC"}
@@ -0,0 +1,3 @@
1
+ import { Command } from 'commander';
2
+ export declare const serveCommand: Command;
3
+ //# sourceMappingURL=serve.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"serve.d.ts","sourceRoot":"","sources":["../../src/cli/serve.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AAKpC,eAAO,MAAM,YAAY,SAYrB,CAAC"}
@@ -0,0 +1,19 @@
1
+ import { Command } from 'commander';
2
+ import { createApp } from '../server/app.js';
3
+ import { openDb } from '../db/schema.js';
4
+ import { config } from '../config.js';
5
+ export const serveCommand = new Command('serve')
6
+ .description('Start HTTP server to receive Health Auto Export pushes')
7
+ .option('-p, --port <number>', 'Port to listen on', String(config.port))
8
+ .option('--token <secret>', 'Require Authorization: Bearer <secret>', config.token)
9
+ .action((opts) => {
10
+ const db = openDb(config.dbPath);
11
+ const app = createApp(db, { token: opts.token });
12
+ const port = parseInt(opts.port, 10);
13
+ app.listen(port, () => {
14
+ console.log(`hvault server listening on http://0.0.0.0:${port}/api/ingest`);
15
+ if (opts.token)
16
+ console.log('Auth: Bearer token required');
17
+ });
18
+ });
19
+ //# sourceMappingURL=serve.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"serve.js","sourceRoot":"","sources":["../../src/cli/serve.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AACpC,OAAO,EAAE,SAAS,EAAE,MAAM,kBAAkB,CAAC;AAC7C,OAAO,EAAE,MAAM,EAAE,MAAM,iBAAiB,CAAC;AACzC,OAAO,EAAE,MAAM,EAAE,MAAM,cAAc,CAAC;AAEtC,MAAM,CAAC,MAAM,YAAY,GAAG,IAAI,OAAO,CAAC,OAAO,CAAC;KAC7C,WAAW,CAAC,wDAAwD,CAAC;KACrE,MAAM,CAAC,qBAAqB,EAAE,mBAAmB,EAAE,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;KACvE,MAAM,CAAC,kBAAkB,EAAE,wCAAwC,EAAE,MAAM,CAAC,KAAK,CAAC;KAClF,MAAM,CAAC,CAAC,IAAI,EAAE,EAAE;IACf,MAAM,EAAE,GAAG,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC;IACjC,MAAM,GAAG,GAAG,SAAS,CAAC,EAAE,EAAE,EAAE,KAAK,EAAE,IAAI,CAAC,KAAK,EAAE,CAAC,CAAC;IACjD,MAAM,IAAI,GAAG,QAAQ,CAAC,IAAI,CAAC,IAAI,EAAE,EAAE,CAAC,CAAC;IACrC,GAAG,CAAC,MAAM,CAAC,IAAI,EAAE,GAAG,EAAE;QACpB,OAAO,CAAC,GAAG,CAAC,6CAA6C,IAAI,aAAa,CAAC,CAAC;QAC5E,IAAI,IAAI,CAAC,KAAK;YAAE,OAAO,CAAC,GAAG,CAAC,6BAA6B,CAAC,CAAC;IAC7D,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC"}
@@ -0,0 +1,3 @@
1
+ import { Command } from 'commander';
2
+ export declare const sleepCommand: Command;
3
+ //# sourceMappingURL=sleep.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"sleep.d.ts","sourceRoot":"","sources":["../../src/cli/sleep.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AAGpC,eAAO,MAAM,YAAY,SAerB,CAAC"}
@@ -0,0 +1,19 @@
1
+ import { Command } from 'commander';
2
+ import { openDb } from '../db/schema.js';
3
+ export const sleepCommand = new Command('sleep')
4
+ .description('Query sleep data')
5
+ .option('--days <n>', 'Last N days', '14')
6
+ .option('--pretty', 'Pretty-print JSON', false)
7
+ .action((opts) => {
8
+ const db = openDb();
9
+ const since = new Date();
10
+ since.setDate(since.getDate() - parseInt(opts.days, 10));
11
+ const rows = db.prepare(`
12
+ SELECT date, sleep_start, sleep_end, core_h, deep_h, rem_h, awake_h, asleep_h, in_bed_h, schema_ver, source
13
+ FROM sleep
14
+ WHERE date >= ?
15
+ ORDER BY date ASC
16
+ `).all(since.toISOString().slice(0, 10));
17
+ console.log(opts.pretty ? JSON.stringify(rows, null, 2) : JSON.stringify(rows));
18
+ });
19
+ //# sourceMappingURL=sleep.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"sleep.js","sourceRoot":"","sources":["../../src/cli/sleep.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AACpC,OAAO,EAAE,MAAM,EAAE,MAAM,iBAAiB,CAAC;AAEzC,MAAM,CAAC,MAAM,YAAY,GAAG,IAAI,OAAO,CAAC,OAAO,CAAC;KAC7C,WAAW,CAAC,kBAAkB,CAAC;KAC/B,MAAM,CAAC,YAAY,EAAE,aAAa,EAAE,IAAI,CAAC;KACzC,MAAM,CAAC,UAAU,EAAE,mBAAmB,EAAE,KAAK,CAAC;KAC9C,MAAM,CAAC,CAAC,IAAI,EAAE,EAAE;IACf,MAAM,EAAE,GAAG,MAAM,EAAE,CAAC;IACpB,MAAM,KAAK,GAAG,IAAI,IAAI,EAAE,CAAC;IACzB,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC,OAAO,EAAE,GAAG,QAAQ,CAAC,IAAI,CAAC,IAAI,EAAE,EAAE,CAAC,CAAC,CAAC;IACzD,MAAM,IAAI,GAAG,EAAE,CAAC,OAAO,CAAC;;;;;KAKvB,CAAC,CAAC,GAAG,CAAC,KAAK,CAAC,WAAW,EAAE,CAAC,KAAK,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC;IACzC,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,CAAC;AAClF,CAAC,CAAC,CAAC"}
@@ -0,0 +1,3 @@
1
+ import { Command } from 'commander';
2
+ export declare const summaryCommand: Command;
3
+ //# sourceMappingURL=summary.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"summary.d.ts","sourceRoot":"","sources":["../../src/cli/summary.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AAGpC,eAAO,MAAM,cAAc,SAsDvB,CAAC"}
@@ -0,0 +1,53 @@
1
+ import { Command } from 'commander';
2
+ import { openDb } from '../db/schema.js';
3
+ export const summaryCommand = new Command('summary')
4
+ .description('Summarise metrics (averages) over N days')
5
+ .option('--days <n>', 'Last N days', '90')
6
+ .option('--pretty', 'Pretty-print JSON', false)
7
+ .option('-c, --color', 'Pretty terminal output with emoji indicators', false)
8
+ .action((opts) => {
9
+ const db = openDb();
10
+ const since = new Date();
11
+ since.setDate(since.getDate() - parseInt(opts.days, 10));
12
+ const sinceStr = since.toISOString().slice(0, 10);
13
+ if (opts.color) {
14
+ const days = parseInt(opts.days, 10);
15
+ function avgMetric(metricName) {
16
+ const row = db.prepare(`SELECT AVG(qty) as avg FROM metrics WHERE metric = ? AND date >= ? AND qty IS NOT NULL`).get(metricName, sinceStr);
17
+ return row?.avg ?? null;
18
+ }
19
+ const steps = avgMetric('step_count');
20
+ const restingHR = avgMetric('resting_heart_rate');
21
+ const hrv = avgMetric('heart_rate_variability_sdnn');
22
+ const activeCal = avgMetric('active_energy_burned');
23
+ const sleepRow = db.prepare(`SELECT AVG(asleep_h) as avg FROM sleep WHERE date >= ? AND asleep_h IS NOT NULL`).get(sinceStr);
24
+ const sleep = sleepRow?.avg ?? null;
25
+ const lines = [`📊 ${days}-Day Summary`, ''];
26
+ if (steps != null)
27
+ lines.push(`👟 Avg Steps: ${Math.round(steps).toLocaleString()}`);
28
+ if (restingHR != null)
29
+ lines.push(`💓 Avg Resting HR: ${Math.round(restingHR)}bpm`);
30
+ if (hrv != null)
31
+ lines.push(`🧠 Avg HRV: ${Math.round(hrv)}ms`);
32
+ if (sleep != null)
33
+ lines.push(`😴 Avg Sleep: ${sleep.toFixed(1)}h`);
34
+ if (activeCal != null)
35
+ lines.push(`🔥 Avg Active Cal: ${Math.round(activeCal).toLocaleString()} kcal`);
36
+ if (lines.length === 2)
37
+ lines.push(' No summary data available');
38
+ console.log(lines.join('\n'));
39
+ return;
40
+ }
41
+ const rows = db.prepare(`
42
+ SELECT metric, units,
43
+ AVG(qty) as avg_qty, MIN(qty) as min_qty, MAX(qty) as max_qty,
44
+ COUNT(*) as count,
45
+ MIN(date) as first_date, MAX(date) as last_date
46
+ FROM metrics
47
+ WHERE date >= ? AND qty IS NOT NULL
48
+ GROUP BY metric, units
49
+ ORDER BY metric ASC
50
+ `).all(sinceStr);
51
+ console.log(opts.pretty ? JSON.stringify(rows, null, 2) : JSON.stringify(rows));
52
+ });
53
+ //# sourceMappingURL=summary.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"summary.js","sourceRoot":"","sources":["../../src/cli/summary.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,3 @@
1
+ import { Command } from 'commander';
2
+ export declare const trendsCommand: Command;
3
+ //# sourceMappingURL=trends.d.ts.map