Files
smart-home-tablet/lib/timers.ts
Cosmo e96e7a1342
All checks were successful
Deploy / deploy (push) Successful in 3m18s
feat(voice): tool endpoints, timer widget, clean Siri-style overlay
Adds the infrastructure for Claude tool use + visual timer.

Tablet API surface (all bearer-authed with VOICE_API_KEY, middleware bypassed):
- /api/voice/tools/weather    — current + short forecast via Open-Meteo
- /api/voice/tools/transport  — tram arrivals by direction / route filter
- /api/voice/tools/events     — Google Calendar today/week
- /api/voice/tools/notes      — notes + shopping lists
- /api/voice/timer            — start (with seconds+label), cancel; GET list (cookie ok)
  Active timers persisted at /data/tablet-timers.json

UI:
- VoiceOverlay stripped to minimal Siri look: no agent emoji/name, just the
  pulsing orb (3-layer radial gradient, independent breath animations),
  subtle status label on wake only, transcription/response text centered.
  Agents distinguished by orb color (Cosmo indigo/violet, Люся pink).
- TimerWidget: bottom-right chip stack with countdown, progress bar, turns
  amber in last 10s. On expiry, fires fullscreen alarm overlay with beep
  (WebAudio osc) + Остановить button.

Other:
- lib/timers.ts — persistent timer store in /data
- lib/voice-tools.ts — shared bearer-auth helper
- middleware — bypass list now covers /api/voice/tools/* and /api/voice/timer
2026-04-23 13:33:31 +00:00

58 lines
1.3 KiB
TypeScript

import * as fs from 'fs'
import * as path from 'path'
const DATA_DIR = fs.existsSync('/data') ? '/data' : '/tmp'
const TIMERS_PATH = path.join(DATA_DIR, 'tablet-timers.json')
export interface Timer {
id: string
label: string
startedAt: string // ISO
endsAt: string // ISO
agent?: 'cosmo' | 'lusya'
}
function load(): Timer[] {
try {
if (fs.existsSync(TIMERS_PATH)) {
return JSON.parse(fs.readFileSync(TIMERS_PATH, 'utf-8'))
}
} catch {}
return []
}
function save(list: Timer[]) {
try {
fs.writeFileSync(TIMERS_PATH, JSON.stringify(list, null, 2))
} catch {}
}
// Mutative helpers, used by timer API route
export function listActive(): Timer[] {
const now = Date.now()
// Drop any that expired over 30 min ago — stale garbage
const list = load().filter(t => new Date(t.endsAt).getTime() > now - 30 * 60 * 1000)
save(list)
return list
}
export function addTimer(t: Omit<Timer, 'id' | 'startedAt'>): Timer {
const full: Timer = {
id: Date.now().toString(36) + Math.random().toString(36).slice(2, 6),
startedAt: new Date().toISOString(),
...t,
}
const list = load()
list.push(full)
save(list)
return full
}
export function removeTimer(id: string): boolean {
const list = load()
const next = list.filter(t => t.id !== id)
if (next.length === list.length) return false
save(next)
return true
}