Add Strix camera discovery system with comprehensive database
This commit adds the complete Strix IP camera stream discovery system: - Go-based API server with SSE support for real-time updates - 3,600+ camera brand database with stream URL patterns - Intelligent fuzzy search across camera models - ONVIF discovery and stream validation - RESTful API with health check, camera search, and stream discovery - Makefile for building and deployment - Comprehensive README documentation 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude <noreply@anthropic.com>
This commit is contained in:
@@ -0,0 +1,354 @@
|
||||
package stream
|
||||
|
||||
import (
|
||||
"bytes"
|
||||
"context"
|
||||
"encoding/json"
|
||||
"fmt"
|
||||
"net/http"
|
||||
"net/url"
|
||||
"os/exec"
|
||||
"strings"
|
||||
"time"
|
||||
)
|
||||
|
||||
// Tester validates stream URLs
|
||||
type Tester struct {
|
||||
httpClient *http.Client
|
||||
ffprobeTimeout time.Duration
|
||||
logger interface{ Debug(string, ...any); Error(string, error, ...any) }
|
||||
}
|
||||
|
||||
// NewTester creates a new stream tester
|
||||
func NewTester(ffprobeTimeout time.Duration, logger interface{ Debug(string, ...any); Error(string, error, ...any) }) *Tester {
|
||||
return &Tester{
|
||||
httpClient: &http.Client{
|
||||
Timeout: 10 * time.Second,
|
||||
},
|
||||
ffprobeTimeout: ffprobeTimeout,
|
||||
logger: logger,
|
||||
}
|
||||
}
|
||||
|
||||
// TestResult contains the test results for a stream
|
||||
type TestResult struct {
|
||||
URL string
|
||||
Working bool
|
||||
Protocol string
|
||||
Type string
|
||||
Resolution string
|
||||
Codec string
|
||||
FPS int
|
||||
Bitrate int
|
||||
HasAudio bool
|
||||
Error string
|
||||
TestTime time.Duration
|
||||
Metadata map[string]interface{}
|
||||
}
|
||||
|
||||
// TestStream tests if a stream URL is working
|
||||
func (t *Tester) TestStream(ctx context.Context, streamURL, username, password string) TestResult {
|
||||
startTime := time.Now()
|
||||
result := TestResult{
|
||||
URL: streamURL,
|
||||
Metadata: make(map[string]interface{}),
|
||||
}
|
||||
|
||||
// Parse URL to determine protocol
|
||||
u, err := url.Parse(streamURL)
|
||||
if err != nil {
|
||||
result.Error = fmt.Sprintf("invalid URL: %v", err)
|
||||
result.TestTime = time.Since(startTime)
|
||||
return result
|
||||
}
|
||||
|
||||
result.Protocol = u.Scheme
|
||||
|
||||
// Test based on protocol
|
||||
switch u.Scheme {
|
||||
case "rtsp", "rtsps":
|
||||
t.testRTSP(ctx, streamURL, username, password, &result)
|
||||
case "http", "https":
|
||||
t.testHTTP(ctx, streamURL, username, password, &result)
|
||||
default:
|
||||
result.Error = fmt.Sprintf("unsupported protocol: %s", u.Scheme)
|
||||
}
|
||||
|
||||
result.TestTime = time.Since(startTime)
|
||||
return result
|
||||
}
|
||||
|
||||
// testRTSP tests an RTSP stream using ffprobe
|
||||
func (t *Tester) testRTSP(ctx context.Context, streamURL, username, password string, result *TestResult) {
|
||||
// Build ffprobe command
|
||||
cmdCtx, cancel := context.WithTimeout(ctx, t.ffprobeTimeout)
|
||||
defer cancel()
|
||||
|
||||
// Build URL with credentials if provided
|
||||
testURL := streamURL
|
||||
if username != "" && password != "" {
|
||||
u, _ := url.Parse(streamURL)
|
||||
u.User = url.UserPassword(username, password)
|
||||
testURL = u.String()
|
||||
}
|
||||
|
||||
args := []string{
|
||||
"-v", "quiet",
|
||||
"-print_format", "json",
|
||||
"-show_streams",
|
||||
"-show_format",
|
||||
"-rtsp_transport", "tcp",
|
||||
testURL,
|
||||
}
|
||||
|
||||
cmd := exec.CommandContext(cmdCtx, "ffprobe", args...)
|
||||
|
||||
// Capture output
|
||||
var stdout, stderr bytes.Buffer
|
||||
cmd.Stdout = &stdout
|
||||
cmd.Stderr = &stderr
|
||||
|
||||
t.logger.Debug("testing RTSP stream", "url", streamURL)
|
||||
|
||||
// Execute command
|
||||
err := cmd.Run()
|
||||
if err != nil {
|
||||
if cmdCtx.Err() == context.DeadlineExceeded {
|
||||
result.Error = "timeout while testing stream"
|
||||
} else {
|
||||
result.Error = fmt.Sprintf("ffprobe failed: %v", err)
|
||||
if stderr.Len() > 0 {
|
||||
result.Error += fmt.Sprintf(" (stderr: %s)", stderr.String())
|
||||
}
|
||||
}
|
||||
return
|
||||
}
|
||||
|
||||
// Parse ffprobe output
|
||||
var probeResult struct {
|
||||
Streams []struct {
|
||||
CodecName string `json:"codec_name"`
|
||||
CodecType string `json:"codec_type"`
|
||||
Width int `json:"width"`
|
||||
Height int `json:"height"`
|
||||
AvgFrameRate string `json:"avg_frame_rate"`
|
||||
BitRate string `json:"bit_rate"`
|
||||
} `json:"streams"`
|
||||
Format struct {
|
||||
BitRate string `json:"bit_rate"`
|
||||
} `json:"format"`
|
||||
}
|
||||
|
||||
if err := json.Unmarshal(stdout.Bytes(), &probeResult); err != nil {
|
||||
result.Error = fmt.Sprintf("failed to parse ffprobe output: %v", err)
|
||||
return
|
||||
}
|
||||
|
||||
// Extract stream information
|
||||
result.Working = len(probeResult.Streams) > 0
|
||||
result.Type = "FFMPEG"
|
||||
|
||||
for _, stream := range probeResult.Streams {
|
||||
if stream.CodecType == "video" {
|
||||
result.Codec = stream.CodecName
|
||||
result.Resolution = fmt.Sprintf("%dx%d", stream.Width, stream.Height)
|
||||
|
||||
// Parse frame rate
|
||||
if stream.AvgFrameRate != "" {
|
||||
parts := strings.Split(stream.AvgFrameRate, "/")
|
||||
if len(parts) == 2 {
|
||||
// Calculate FPS from fraction
|
||||
var num, den int
|
||||
fmt.Sscanf(parts[0], "%d", &num)
|
||||
fmt.Sscanf(parts[1], "%d", &den)
|
||||
if den > 0 {
|
||||
result.FPS = num / den
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// Parse bitrate
|
||||
if stream.BitRate != "" {
|
||||
fmt.Sscanf(stream.BitRate, "%d", &result.Bitrate)
|
||||
}
|
||||
} else if stream.CodecType == "audio" {
|
||||
result.HasAudio = true
|
||||
}
|
||||
}
|
||||
|
||||
// Use format bitrate if stream bitrate not available
|
||||
if result.Bitrate == 0 && probeResult.Format.BitRate != "" {
|
||||
fmt.Sscanf(probeResult.Format.BitRate, "%d", &result.Bitrate)
|
||||
}
|
||||
|
||||
if !result.Working {
|
||||
result.Error = "no streams found"
|
||||
}
|
||||
}
|
||||
|
||||
// testHTTP tests an HTTP stream
|
||||
func (t *Tester) testHTTP(ctx context.Context, streamURL, username, password string, result *TestResult) {
|
||||
// Create request
|
||||
req, err := http.NewRequestWithContext(ctx, "GET", streamURL, nil)
|
||||
if err != nil {
|
||||
result.Error = fmt.Sprintf("failed to create request: %v", err)
|
||||
return
|
||||
}
|
||||
|
||||
// Add Basic Auth if credentials provided
|
||||
if username != "" && password != "" {
|
||||
req.SetBasicAuth(username, password)
|
||||
}
|
||||
|
||||
// Add headers
|
||||
req.Header.Set("User-Agent", "Strix/1.0")
|
||||
|
||||
t.logger.Debug("testing HTTP stream", "url", streamURL)
|
||||
|
||||
// Send request
|
||||
resp, err := t.httpClient.Do(req)
|
||||
if err != nil {
|
||||
result.Error = fmt.Sprintf("HTTP request failed: %v", err)
|
||||
return
|
||||
}
|
||||
defer resp.Body.Close()
|
||||
|
||||
// Check status code
|
||||
if resp.StatusCode != http.StatusOK {
|
||||
result.Error = fmt.Sprintf("HTTP %d: %s", resp.StatusCode, resp.Status)
|
||||
|
||||
// Special handling for 401
|
||||
if resp.StatusCode == http.StatusUnauthorized {
|
||||
result.Error = "authentication required"
|
||||
}
|
||||
return
|
||||
}
|
||||
|
||||
// Check content type
|
||||
contentType := resp.Header.Get("Content-Type")
|
||||
result.Metadata["content_type"] = contentType
|
||||
|
||||
// Determine stream type based on content type
|
||||
switch {
|
||||
case strings.Contains(contentType, "multipart"):
|
||||
result.Type = "MJPEG"
|
||||
result.Working = true
|
||||
|
||||
// Read first few bytes to verify
|
||||
buffer := make([]byte, 512)
|
||||
n, _ := resp.Body.Read(buffer)
|
||||
if n > 0 {
|
||||
// Check for MJPEG boundary
|
||||
if bytes.Contains(buffer[:n], []byte("--")) {
|
||||
result.Working = true
|
||||
}
|
||||
}
|
||||
|
||||
case strings.Contains(contentType, "image/jpeg"):
|
||||
result.Type = "JPEG"
|
||||
result.Working = true
|
||||
|
||||
// Read first few bytes to verify JPEG magic bytes
|
||||
buffer := make([]byte, 3)
|
||||
n, _ := resp.Body.Read(buffer)
|
||||
if n >= 3 && buffer[0] == 0xFF && buffer[1] == 0xD8 && buffer[2] == 0xFF {
|
||||
result.Working = true
|
||||
} else {
|
||||
result.Working = false
|
||||
result.Error = "invalid JPEG data"
|
||||
}
|
||||
|
||||
case strings.Contains(contentType, "video"):
|
||||
result.Type = "HTTP_VIDEO"
|
||||
result.Working = true
|
||||
|
||||
// Try to probe with ffprobe for more details
|
||||
t.probeHTTPVideo(ctx, streamURL, username, password, result)
|
||||
|
||||
default:
|
||||
result.Type = "HTTP_UNKNOWN"
|
||||
result.Working = true // Assume it works if we got 200 OK
|
||||
result.Metadata["note"] = "unknown content type, may still be valid"
|
||||
}
|
||||
}
|
||||
|
||||
// probeHTTPVideo uses ffprobe to get more details about HTTP video stream
|
||||
func (t *Tester) probeHTTPVideo(ctx context.Context, streamURL, username, password string, result *TestResult) {
|
||||
cmdCtx, cancel := context.WithTimeout(ctx, t.ffprobeTimeout)
|
||||
defer cancel()
|
||||
|
||||
// Build URL with credentials if needed
|
||||
testURL := streamURL
|
||||
if username != "" && password != "" && !strings.Contains(streamURL, "@") {
|
||||
u, _ := url.Parse(streamURL)
|
||||
u.User = url.UserPassword(username, password)
|
||||
testURL = u.String()
|
||||
}
|
||||
|
||||
args := []string{
|
||||
"-v", "quiet",
|
||||
"-print_format", "json",
|
||||
"-show_streams",
|
||||
testURL,
|
||||
}
|
||||
|
||||
cmd := exec.CommandContext(cmdCtx, "ffprobe", args...)
|
||||
var stdout bytes.Buffer
|
||||
cmd.Stdout = &stdout
|
||||
|
||||
if err := cmd.Run(); err == nil {
|
||||
var probeResult struct {
|
||||
Streams []struct {
|
||||
CodecName string `json:"codec_name"`
|
||||
CodecType string `json:"codec_type"`
|
||||
Width int `json:"width"`
|
||||
Height int `json:"height"`
|
||||
} `json:"streams"`
|
||||
}
|
||||
|
||||
if json.Unmarshal(stdout.Bytes(), &probeResult) == nil {
|
||||
for _, stream := range probeResult.Streams {
|
||||
if stream.CodecType == "video" {
|
||||
result.Codec = stream.CodecName
|
||||
result.Resolution = fmt.Sprintf("%dx%d", stream.Width, stream.Height)
|
||||
break
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// TestMultiple tests multiple URLs concurrently
|
||||
func (t *Tester) TestMultiple(ctx context.Context, urls []string, username, password string, maxConcurrent int) []TestResult {
|
||||
if maxConcurrent <= 0 {
|
||||
maxConcurrent = 10
|
||||
}
|
||||
|
||||
results := make([]TestResult, len(urls))
|
||||
sem := make(chan struct{}, maxConcurrent)
|
||||
|
||||
for i, url := range urls {
|
||||
i, url := i, url // Capture for goroutine
|
||||
|
||||
sem <- struct{}{} // Acquire semaphore
|
||||
go func() {
|
||||
defer func() { <-sem }() // Release semaphore
|
||||
|
||||
results[i] = t.TestStream(ctx, url, username, password)
|
||||
}()
|
||||
}
|
||||
|
||||
// Wait for all to complete
|
||||
for i := 0; i < maxConcurrent; i++ {
|
||||
sem <- struct{}{}
|
||||
}
|
||||
|
||||
return results
|
||||
}
|
||||
|
||||
// IsFFProbeAvailable checks if ffprobe is available
|
||||
func (t *Tester) IsFFProbeAvailable() bool {
|
||||
cmd := exec.Command("ffprobe", "-version")
|
||||
err := cmd.Run()
|
||||
return err == nil
|
||||
}
|
||||
Reference in New Issue
Block a user