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.
- package/.env.example +7 -0
- package/CLAUDE.md +220 -0
- package/README.md +206 -0
- package/SKILL.md +60 -0
- package/dist/cli/dashboard.d.ts +3 -0
- package/dist/cli/dashboard.d.ts.map +1 -0
- package/dist/cli/dashboard.js +206 -0
- package/dist/cli/dashboard.js.map +1 -0
- package/dist/cli/import.d.ts +3 -0
- package/dist/cli/import.d.ts.map +1 -0
- package/dist/cli/import.js +78 -0
- package/dist/cli/import.js.map +1 -0
- package/dist/cli/index.d.ts +3 -0
- package/dist/cli/index.d.ts.map +1 -0
- package/dist/cli/index.js +31 -0
- package/dist/cli/index.js.map +1 -0
- package/dist/cli/info.d.ts +5 -0
- package/dist/cli/info.d.ts.map +1 -0
- package/dist/cli/info.js +34 -0
- package/dist/cli/info.js.map +1 -0
- package/dist/cli/metrics.d.ts +3 -0
- package/dist/cli/metrics.d.ts.map +1 -0
- package/dist/cli/metrics.js +20 -0
- package/dist/cli/metrics.js.map +1 -0
- package/dist/cli/query.d.ts +3 -0
- package/dist/cli/query.d.ts.map +1 -0
- package/dist/cli/query.js +18 -0
- package/dist/cli/query.js.map +1 -0
- package/dist/cli/serve.d.ts +3 -0
- package/dist/cli/serve.d.ts.map +1 -0
- package/dist/cli/serve.js +19 -0
- package/dist/cli/serve.js.map +1 -0
- package/dist/cli/sleep.d.ts +3 -0
- package/dist/cli/sleep.d.ts.map +1 -0
- package/dist/cli/sleep.js +19 -0
- package/dist/cli/sleep.js.map +1 -0
- package/dist/cli/summary.d.ts +3 -0
- package/dist/cli/summary.d.ts.map +1 -0
- package/dist/cli/summary.js +53 -0
- package/dist/cli/summary.js.map +1 -0
- package/dist/cli/trends.d.ts +3 -0
- package/dist/cli/trends.d.ts.map +1 -0
- package/dist/cli/trends.js +77 -0
- package/dist/cli/trends.js.map +1 -0
- package/dist/cli/watch.d.ts +12 -0
- package/dist/cli/watch.d.ts.map +1 -0
- package/dist/cli/watch.js +89 -0
- package/dist/cli/watch.js.map +1 -0
- package/dist/cli/workouts.d.ts +3 -0
- package/dist/cli/workouts.d.ts.map +1 -0
- package/dist/cli/workouts.js +19 -0
- package/dist/cli/workouts.js.map +1 -0
- package/dist/config.d.ts +9 -0
- package/dist/config.d.ts.map +1 -0
- package/dist/config.js +25 -0
- package/dist/config.js.map +1 -0
- package/dist/db/importLog.d.ts +5 -0
- package/dist/db/importLog.d.ts.map +1 -0
- package/dist/db/importLog.js +10 -0
- package/dist/db/importLog.js.map +1 -0
- package/dist/db/metrics.d.ts +4 -0
- package/dist/db/metrics.d.ts.map +1 -0
- package/dist/db/metrics.js +14 -0
- package/dist/db/metrics.js.map +1 -0
- package/dist/db/schema.d.ts +5 -0
- package/dist/db/schema.d.ts.map +1 -0
- package/dist/db/schema.js +100 -0
- package/dist/db/schema.js.map +1 -0
- package/dist/db/sleep.d.ts +4 -0
- package/dist/db/sleep.d.ts.map +1 -0
- package/dist/db/sleep.js +13 -0
- package/dist/db/sleep.js.map +1 -0
- package/dist/db/workouts.d.ts +4 -0
- package/dist/db/workouts.d.ts.map +1 -0
- package/dist/db/workouts.js +11 -0
- package/dist/db/workouts.js.map +1 -0
- package/dist/index.d.ts +3 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +5 -0
- package/dist/index.js.map +1 -0
- package/dist/parse/metrics.d.ts +17 -0
- package/dist/parse/metrics.d.ts.map +1 -0
- package/dist/parse/metrics.js +33 -0
- package/dist/parse/metrics.js.map +1 -0
- package/dist/parse/sleep.d.ts +23 -0
- package/dist/parse/sleep.d.ts.map +1 -0
- package/dist/parse/sleep.js +58 -0
- package/dist/parse/sleep.js.map +1 -0
- package/dist/parse/time.d.ts +4 -0
- package/dist/parse/time.d.ts.map +1 -0
- package/dist/parse/time.js +41 -0
- package/dist/parse/time.js.map +1 -0
- package/dist/parse/workouts.d.ts +17 -0
- package/dist/parse/workouts.d.ts.map +1 -0
- package/dist/parse/workouts.js +24 -0
- package/dist/parse/workouts.js.map +1 -0
- package/dist/server/app.d.ts +5 -0
- package/dist/server/app.d.ts.map +1 -0
- package/dist/server/app.js +39 -0
- package/dist/server/app.js.map +1 -0
- package/dist/server/ingest.d.ts +15 -0
- package/dist/server/ingest.d.ts.map +1 -0
- package/dist/server/ingest.js +41 -0
- package/dist/server/ingest.js.map +1 -0
- package/dist/types/hae.d.ts +103 -0
- package/dist/types/hae.d.ts.map +1 -0
- package/dist/types/hae.js +2 -0
- package/dist/types/hae.js.map +1 -0
- package/dist/util/zip.d.ts +3 -0
- package/dist/util/zip.d.ts.map +1 -0
- package/dist/util/zip.js +24 -0
- package/dist/util/zip.js.map +1 -0
- package/docs/COMMANDS.md +315 -0
- package/docs/plans/2026-02-18-hae-vault-initial-implementation.md +2015 -0
- package/docs/plans/2026-02-18-readme-dashboard-design.md +213 -0
- package/docs/plans/2026-02-18-readme-dashboard-plan.md +1306 -0
- package/docs/plans/2026-02-18-zip-env-watch-design.md +213 -0
- package/docs/plans/2026-02-18-zip-env-watch.md +966 -0
- package/package.json +57 -0
- package/src/cli/dashboard.ts +242 -0
- package/src/cli/import.ts +85 -0
- package/src/cli/index.ts +32 -0
- package/src/cli/info.ts +36 -0
- package/src/cli/metrics.ts +20 -0
- package/src/cli/query.ts +17 -0
- package/src/cli/serve.ts +18 -0
- package/src/cli/sleep.ts +19 -0
- package/src/cli/summary.ts +58 -0
- package/src/cli/trends.ts +103 -0
- package/src/cli/watch.ts +111 -0
- package/src/cli/workouts.ts +19 -0
- package/src/config.ts +28 -0
- package/src/db/importLog.ts +18 -0
- package/src/db/metrics.ts +15 -0
- package/src/db/schema.ts +105 -0
- package/src/db/sleep.ts +15 -0
- package/src/db/workouts.ts +13 -0
- package/src/index.ts +4 -0
- package/src/parse/metrics.ts +50 -0
- package/src/parse/sleep.ts +82 -0
- package/src/parse/time.ts +43 -0
- package/src/parse/workouts.ts +42 -0
- package/src/server/app.ts +46 -0
- package/src/server/ingest.ts +68 -0
- package/src/types/hae.ts +94 -0
- package/src/util/zip.ts +24 -0
- package/tests/cli-watch.test.ts +64 -0
- package/tests/db-import-log.test.ts +40 -0
- package/tests/db-metrics.test.ts +44 -0
- package/tests/db-schema.test.ts +55 -0
- package/tests/db-sleep.test.ts +36 -0
- package/tests/db-workouts.test.ts +34 -0
- package/tests/ingest.test.ts +99 -0
- package/tests/parse-metrics.test.ts +55 -0
- package/tests/parse-sleep.test.ts +65 -0
- package/tests/parse-time.test.ts +48 -0
- package/tests/parse-workouts.test.ts +43 -0
- package/tests/types.test.ts +27 -0
- package/tests/util-zip.test.ts +46 -0
- 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 @@
|
|
|
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 @@
|
|
|
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 @@
|
|
|
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"}
|
package/dist/cli/info.js
ADDED
|
@@ -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 @@
|
|
|
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 @@
|
|
|
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 @@
|
|
|
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 @@
|
|
|
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 @@
|
|
|
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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"}
|