generated from CubeCraft-Creations/Tracehound
CUB-187+188+191+193: Recording handlers, status ingestion, SSE endpoint
ci/verify Branch verified
ci/verify Branch verified
This commit is contained in:
@@ -0,0 +1,88 @@
|
||||
// Package api provides HTTP handlers for camera operations.
|
||||
package api
|
||||
|
||||
import (
|
||||
"log"
|
||||
"net/http"
|
||||
|
||||
"github.com/cubecraft/remoterig/internal/db"
|
||||
"github.com/go-chi/chi/v5"
|
||||
)
|
||||
|
||||
// StartRecording returns a handler for POST /cameras/{id}/start.
|
||||
func StartRecording(database *db.DB) http.HandlerFunc {
|
||||
return func(w http.ResponseWriter, r *http.Request) {
|
||||
cameraID := chi.URLParam(r, "id")
|
||||
if cameraID == "" {
|
||||
respondJSON(w, http.StatusBadRequest, map[string]string{"error": "camera_id required"})
|
||||
return
|
||||
}
|
||||
|
||||
// Check if camera is registered
|
||||
var exists int
|
||||
err := database.QueryRowContext(r.Context(),
|
||||
"SELECT COUNT(*) FROM cameras WHERE camera_id = ?", cameraID).Scan(&exists)
|
||||
if err != nil || exists == 0 {
|
||||
respondJSON(w, http.StatusNotFound, map[string]string{"error": "camera not registered"})
|
||||
return
|
||||
}
|
||||
|
||||
// Open recording event
|
||||
result, err := database.ExecContext(r.Context(), `
|
||||
INSERT INTO recording_events (camera_id, started_at, reason)
|
||||
VALUES (?, datetime('now'), 'manual')
|
||||
`, cameraID)
|
||||
if err != nil {
|
||||
log.Printf("Error starting recording: %v", err)
|
||||
respondJSON(w, http.StatusInternalServerError, map[string]string{"error": "database error"})
|
||||
return
|
||||
}
|
||||
|
||||
rows, _ := result.RowsAffected()
|
||||
log.Printf("Recording started on %s (%d rows affected)", cameraID, rows)
|
||||
|
||||
respondJSON(w, http.StatusOK, map[string]string{
|
||||
"status": "recording_started",
|
||||
"camera_id": cameraID,
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
// StopRecording returns a handler for POST /cameras/{id}/stop.
|
||||
func StopRecording(database *db.DB) http.HandlerFunc {
|
||||
return func(w http.ResponseWriter, r *http.Request) {
|
||||
cameraID := chi.URLParam(r, "id")
|
||||
if cameraID == "" {
|
||||
respondJSON(w, http.StatusBadRequest, map[string]string{"error": "camera_id required"})
|
||||
return
|
||||
}
|
||||
|
||||
// Check if camera is registered
|
||||
var exists int
|
||||
err := database.QueryRowContext(r.Context(),
|
||||
"SELECT COUNT(*) FROM cameras WHERE camera_id = ?", cameraID).Scan(&exists)
|
||||
if err != nil || exists == 0 {
|
||||
respondJSON(w, http.StatusNotFound, map[string]string{"error": "camera not registered"})
|
||||
return
|
||||
}
|
||||
|
||||
// Close the most recent open recording event
|
||||
result, err := database.ExecContext(r.Context(), `
|
||||
UPDATE recording_events SET stopped_at = datetime('now'), reason = 'manual'
|
||||
WHERE camera_id = ? AND stopped_at IS NULL
|
||||
`, cameraID)
|
||||
if err != nil {
|
||||
log.Printf("Error stopping recording: %v", err)
|
||||
respondJSON(w, http.StatusInternalServerError, map[string]string{"error": "database error"})
|
||||
return
|
||||
}
|
||||
|
||||
rows, _ := result.RowsAffected()
|
||||
log.Printf("Recording stopped on %s (%d rows affected)", cameraID, rows)
|
||||
|
||||
respondJSON(w, http.StatusOK, map[string]string{
|
||||
"status": "recording_stopped",
|
||||
"camera_id": cameraID,
|
||||
})
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,106 @@
|
||||
// Package api provides HTTP handlers for camera operations.
|
||||
package api
|
||||
|
||||
import (
|
||||
"encoding/json"
|
||||
"log"
|
||||
"net/http"
|
||||
|
||||
"github.com/cubecraft/remoterig/internal/db"
|
||||
"github.com/go-chi/chi/v5"
|
||||
)
|
||||
|
||||
// PushStatus accepts a status update from an ESP32 node and persists it.
|
||||
func PushStatus(database *db.DB) http.HandlerFunc {
|
||||
return func(w http.ResponseWriter, r *http.Request) {
|
||||
cameraID := chi.URLParam(r, "id")
|
||||
if cameraID == "" {
|
||||
respondJSON(w, http.StatusBadRequest, map[string]string{"error": "camera_id required"})
|
||||
return
|
||||
}
|
||||
|
||||
var req struct {
|
||||
BatteryPct *int `json:"battery_pct"`
|
||||
VideoRemainingSec *int `json:"video_remaining_sec"`
|
||||
Recording bool `json:"recording"`
|
||||
Mode string `json:"mode"`
|
||||
Resolution string `json:"resolution"`
|
||||
FPS int `json:"fps"`
|
||||
Online bool `json:"online"`
|
||||
RawBatteryPct *float64 `json:"raw_battery_pct"`
|
||||
Timestamp *string `json:"ts"`
|
||||
}
|
||||
if err := json.NewDecoder(r.Body).Decode(&req); err != nil {
|
||||
respondJSON(w, http.StatusBadRequest, map[string]string{"error": "invalid request body"})
|
||||
return
|
||||
}
|
||||
|
||||
// Check if camera is registered
|
||||
var exists int
|
||||
err := database.QueryRowContext(r.Context(),
|
||||
"SELECT COUNT(*) FROM cameras WHERE camera_id = ?", cameraID).Scan(&exists)
|
||||
if err != nil || exists == 0 {
|
||||
respondJSON(w, http.StatusNotFound, map[string]string{"error": "camera not registered"})
|
||||
return
|
||||
}
|
||||
|
||||
// Insert status log
|
||||
result, err := database.ExecContext(r.Context(), `
|
||||
INSERT INTO status_logs (camera_id, battery_pct, video_remaining_sec,
|
||||
recording_state, mode, resolution, fps, online, raw_battery_pct)
|
||||
VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?)
|
||||
`, cameraID, req.BatteryPct, req.VideoRemainingSec,
|
||||
boolToInt(req.Recording), req.Mode, req.Resolution,
|
||||
req.FPS, boolToInt(req.Online), req.RawBatteryPct)
|
||||
if err != nil {
|
||||
log.Printf("Error inserting status log: %v", err)
|
||||
respondJSON(w, http.StatusInternalServerError, map[string]string{"error": "database error"})
|
||||
return
|
||||
}
|
||||
|
||||
// Check if recording state changed - update recording_events if so
|
||||
var prevRecording int
|
||||
err = database.QueryRowContext(r.Context(), `
|
||||
SELECT recording_state FROM status_logs
|
||||
WHERE camera_id = ? AND recorded_at > datetime('now', '-60 seconds')
|
||||
ORDER BY recorded_at DESC LIMIT 1
|
||||
`, cameraID).Scan(&prevRecording)
|
||||
if err == nil && prevRecording != boolToInt(req.Recording) {
|
||||
reason := "manual"
|
||||
if req.Recording {
|
||||
// Start recording - open a new event
|
||||
_, err := database.ExecContext(r.Context(), `
|
||||
INSERT INTO recording_events (camera_id, started_at, reason)
|
||||
VALUES (?, datetime('now'), ?)
|
||||
`, cameraID, reason)
|
||||
if err != nil {
|
||||
log.Printf("Error inserting recording event: %v", err)
|
||||
}
|
||||
} else {
|
||||
// Stop recording - close the most recent open event
|
||||
_, err := database.ExecContext(r.Context(), `
|
||||
UPDATE recording_events SET stopped_at = datetime('now')
|
||||
WHERE camera_id = ? AND stopped_at IS NULL
|
||||
ORDER BY started_at DESC LIMIT 1
|
||||
`, cameraID)
|
||||
if err != nil {
|
||||
log.Printf("Error updating recording event: %v", err)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
_, _ = result.RowsAffected() // consume the result
|
||||
|
||||
respondJSON(w, http.StatusOK, map[string]string{
|
||||
"status": "accepted",
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
// boolToInt converts a bool to 0 or 1 for SQLite storage.
|
||||
func boolToInt(b bool) int {
|
||||
if b {
|
||||
return 1
|
||||
}
|
||||
return 0
|
||||
}
|
||||
Reference in New Issue
Block a user