2026-05-18 17:52:48 -04:00
|
|
|
// 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")
|
2026-05-23 08:50:21 -04:00
|
|
|
if !validateCameraID(w, cameraID) {
|
2026-05-18 17:52:48 -04:00
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Check if camera is registered
|
|
|
|
|
var exists int
|
|
|
|
|
err := database.QueryRowContext(r.Context(),
|
|
|
|
|
"SELECT COUNT(*) FROM cameras WHERE camera_id = ?", cameraID).Scan(&exists)
|
2026-05-23 08:50:21 -04:00
|
|
|
if err != nil {
|
|
|
|
|
log.Printf("Error checking camera existence: %v", err)
|
|
|
|
|
respondError(w, http.StatusInternalServerError, "database error", err.Error())
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
if exists == 0 {
|
|
|
|
|
respondError(w, http.StatusNotFound, "camera not found")
|
2026-05-18 17:52:48 -04:00
|
|
|
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)
|
2026-05-23 08:50:21 -04:00
|
|
|
respondError(w, http.StatusInternalServerError, "database error", err.Error())
|
2026-05-18 17:52:48 -04:00
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2026-05-23 08:50:21 -04:00
|
|
|
rowsAffected, _ := result.RowsAffected()
|
|
|
|
|
log.Printf("Recording started on %s (%d rows affected)", cameraID, rowsAffected)
|
2026-05-18 17:52:48 -04:00
|
|
|
|
|
|
|
|
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")
|
2026-05-23 08:50:21 -04:00
|
|
|
if !validateCameraID(w, cameraID) {
|
2026-05-18 17:52:48 -04:00
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Check if camera is registered
|
|
|
|
|
var exists int
|
|
|
|
|
err := database.QueryRowContext(r.Context(),
|
|
|
|
|
"SELECT COUNT(*) FROM cameras WHERE camera_id = ?", cameraID).Scan(&exists)
|
2026-05-23 08:50:21 -04:00
|
|
|
if err != nil {
|
|
|
|
|
log.Printf("Error checking camera existence: %v", err)
|
|
|
|
|
respondError(w, http.StatusInternalServerError, "database error", err.Error())
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
if exists == 0 {
|
|
|
|
|
respondError(w, http.StatusNotFound, "camera not found")
|
2026-05-18 17:52:48 -04:00
|
|
|
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)
|
2026-05-23 08:50:21 -04:00
|
|
|
respondError(w, http.StatusInternalServerError, "database error", err.Error())
|
2026-05-18 17:52:48 -04:00
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2026-05-23 08:50:21 -04:00
|
|
|
rowsAffected, _ := result.RowsAffected()
|
|
|
|
|
log.Printf("Recording stopped on %s (%d rows affected)", cameraID, rowsAffected)
|
2026-05-18 17:52:48 -04:00
|
|
|
|
|
|
|
|
respondJSON(w, http.StatusOK, map[string]string{
|
|
|
|
|
"status": "recording_stopped",
|
|
|
|
|
"camera_id": cameraID,
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
}
|