Files
smart-home-tablet/lib/voice-tools.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

20 lines
722 B
TypeScript

/**
* Helper для /api/voice/tools/* — общий bearer-check и forwarding к внутренним endpoint'ам.
* Позволяет голосовому скрипту вызывать tools через один и тот же токен (VOICE_API_KEY).
*/
export function isBearerAuthorized(req: Request): boolean {
const expected = process.env.VOICE_API_KEY
if (!expected) return false
const auth = req.headers.get('authorization') || ''
const token = auth.replace(/^Bearer\s+/i, '').trim()
return token === expected
}
export function unauthorized() {
return new Response(JSON.stringify({ error: 'unauthorized' }), {
status: 401,
headers: { 'Content-Type': 'application/json' },
})
}