mirror of
https://github.com/gen2brain/cam2ip.git
synced 2026-01-10 08:38:34 +00:00
Add video file reader
This commit is contained in:
39
cam2ip.go
39
cam2ip.go
@@ -7,6 +7,7 @@ import (
|
|||||||
|
|
||||||
"github.com/gen2brain/cam2ip/camera"
|
"github.com/gen2brain/cam2ip/camera"
|
||||||
"github.com/gen2brain/cam2ip/server"
|
"github.com/gen2brain/cam2ip/server"
|
||||||
|
"github.com/gen2brain/cam2ip/video"
|
||||||
)
|
)
|
||||||
|
|
||||||
const (
|
const (
|
||||||
@@ -19,11 +20,12 @@ func main() {
|
|||||||
|
|
||||||
flag.IntVar(&srv.Index, "index", 0, "Camera index")
|
flag.IntVar(&srv.Index, "index", 0, "Camera index")
|
||||||
flag.IntVar(&srv.Delay, "delay", 10, "Delay between frames, in milliseconds")
|
flag.IntVar(&srv.Delay, "delay", 10, "Delay between frames, in milliseconds")
|
||||||
flag.Float64Var(&srv.FrameWidth, "frame-width", 640, "Frame width")
|
flag.Float64Var(&srv.FrameWidth, "frame-width", 640, "Camera frame width")
|
||||||
flag.Float64Var(&srv.FrameHeight, "frame-height", 480, "Frame height")
|
flag.Float64Var(&srv.FrameHeight, "frame-height", 480, "Camera frame height")
|
||||||
flag.BoolVar(&srv.NoWebGL, "nowebgl", false, "Disable WebGL drawing of images (html handler)")
|
flag.BoolVar(&srv.NoWebGL, "nowebgl", false, "Disable WebGL drawing of images (html handler)")
|
||||||
flag.StringVar(&srv.Bind, "bind-addr", ":56000", "Bind address")
|
flag.StringVar(&srv.Bind, "bind-addr", ":56000", "Bind address")
|
||||||
flag.StringVar(&srv.Htpasswd, "htpasswd-file", "", "Path to htpasswd file, if empty auth is disabled")
|
flag.StringVar(&srv.Htpasswd, "htpasswd-file", "", "Path to htpasswd file, if empty auth is disabled")
|
||||||
|
flag.StringVar(&srv.FileName, "video-file", "", "Use video file instead of camera")
|
||||||
flag.Parse()
|
flag.Parse()
|
||||||
|
|
||||||
srv.Name = name
|
srv.Name = name
|
||||||
@@ -38,16 +40,35 @@ func main() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
srv.Camera, err = camera.New(srv.Index)
|
if srv.FileName != "" {
|
||||||
if err != nil {
|
if _, err = os.Stat(srv.FileName); err != nil {
|
||||||
fmt.Fprintf(os.Stderr, "%s\n", err.Error())
|
fmt.Fprintf(os.Stderr, "%s\n", err.Error())
|
||||||
os.Exit(1)
|
os.Exit(1)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
srv.Camera.SetProperty(camera.PropFrameWidth, srv.FrameWidth)
|
if srv.FileName != "" {
|
||||||
srv.Camera.SetProperty(camera.PropFrameHeight, srv.FrameHeight)
|
vid, err := video.New(srv.FileName)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Fprintf(os.Stderr, "%s\n", err.Error())
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
|
||||||
defer srv.Camera.Close()
|
srv.Reader = vid
|
||||||
|
} else {
|
||||||
|
cam, err := camera.New(srv.Index)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Fprintf(os.Stderr, "%s\n", err.Error())
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
|
||||||
|
cam.SetProperty(camera.PropFrameWidth, srv.FrameWidth)
|
||||||
|
cam.SetProperty(camera.PropFrameHeight, srv.FrameHeight)
|
||||||
|
|
||||||
|
srv.Reader = cam
|
||||||
|
}
|
||||||
|
|
||||||
|
defer srv.Reader.Close()
|
||||||
|
|
||||||
fmt.Fprintf(os.Stderr, "Listening on %s\n", srv.Bind)
|
fmt.Fprintf(os.Stderr, "Listening on %s\n", srv.Bind)
|
||||||
|
|
||||||
|
|||||||
@@ -53,14 +53,12 @@ const (
|
|||||||
|
|
||||||
// Camera represents camera.
|
// Camera represents camera.
|
||||||
type Camera struct {
|
type Camera struct {
|
||||||
Index int
|
|
||||||
camera *opencv.Capture
|
camera *opencv.Capture
|
||||||
}
|
}
|
||||||
|
|
||||||
// New returns new Camera for given camera index.
|
// New returns new Camera for given camera index.
|
||||||
func New(index int) (camera *Camera, err error) {
|
func New(index int) (camera *Camera, err error) {
|
||||||
camera = &Camera{}
|
camera = &Camera{}
|
||||||
camera.Index = index
|
|
||||||
|
|
||||||
camera.camera = opencv.NewCameraCapture(index)
|
camera.camera = opencv.NewCameraCapture(index)
|
||||||
if camera.camera == nil {
|
if camera.camera == nil {
|
||||||
|
|||||||
@@ -1,3 +1,4 @@
|
|||||||
|
// Package encoder.
|
||||||
package encoder
|
package encoder
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
|||||||
@@ -4,18 +4,18 @@ import (
|
|||||||
"log"
|
"log"
|
||||||
"net/http"
|
"net/http"
|
||||||
|
|
||||||
"github.com/gen2brain/cam2ip/camera"
|
|
||||||
"github.com/gen2brain/cam2ip/encoder"
|
"github.com/gen2brain/cam2ip/encoder"
|
||||||
|
"github.com/gen2brain/cam2ip/reader"
|
||||||
)
|
)
|
||||||
|
|
||||||
// JPEG handler.
|
// JPEG handler.
|
||||||
type JPEG struct {
|
type JPEG struct {
|
||||||
camera *camera.Camera
|
reader reader.ImageReader
|
||||||
}
|
}
|
||||||
|
|
||||||
// NewJPEG returns new JPEG handler.
|
// NewJPEG returns new JPEG handler.
|
||||||
func NewJPEG(camera *camera.Camera) *JPEG {
|
func NewJPEG(reader reader.ImageReader) *JPEG {
|
||||||
return &JPEG{camera}
|
return &JPEG{reader}
|
||||||
}
|
}
|
||||||
|
|
||||||
// ServeHTTP handles requests on incoming connections.
|
// ServeHTTP handles requests on incoming connections.
|
||||||
@@ -29,7 +29,7 @@ func (j *JPEG) ServeHTTP(w http.ResponseWriter, r *http.Request) {
|
|||||||
w.Header().Add("Cache-Control", "no-store, no-cache")
|
w.Header().Add("Cache-Control", "no-store, no-cache")
|
||||||
w.Header().Add("Content-Type", "image/jpeg")
|
w.Header().Add("Content-Type", "image/jpeg")
|
||||||
|
|
||||||
img, err := j.camera.Read()
|
img, err := j.reader.Read()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Printf("jpeg: read: %v", err)
|
log.Printf("jpeg: read: %v", err)
|
||||||
return
|
return
|
||||||
|
|||||||
@@ -9,19 +9,19 @@ import (
|
|||||||
"net/textproto"
|
"net/textproto"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/gen2brain/cam2ip/camera"
|
|
||||||
"github.com/gen2brain/cam2ip/encoder"
|
"github.com/gen2brain/cam2ip/encoder"
|
||||||
|
"github.com/gen2brain/cam2ip/reader"
|
||||||
)
|
)
|
||||||
|
|
||||||
// MJPEG handler.
|
// MJPEG handler.
|
||||||
type MJPEG struct {
|
type MJPEG struct {
|
||||||
camera *camera.Camera
|
reader reader.ImageReader
|
||||||
delay int
|
delay int
|
||||||
}
|
}
|
||||||
|
|
||||||
// NewMJPEG returns new MJPEG handler.
|
// NewMJPEG returns new MJPEG handler.
|
||||||
func NewMJPEG(camera *camera.Camera, delay int) *MJPEG {
|
func NewMJPEG(reader reader.ImageReader, delay int) *MJPEG {
|
||||||
return &MJPEG{camera, delay}
|
return &MJPEG{reader, delay}
|
||||||
}
|
}
|
||||||
|
|
||||||
// ServeHTTP handles requests on incoming connections.
|
// ServeHTTP handles requests on incoming connections.
|
||||||
@@ -56,7 +56,7 @@ loop:
|
|||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
img, err := m.camera.Read()
|
img, err := m.reader.Read()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Printf("mjpeg: read: %v", err)
|
log.Printf("mjpeg: read: %v", err)
|
||||||
continue
|
continue
|
||||||
|
|||||||
@@ -8,29 +8,29 @@ import (
|
|||||||
|
|
||||||
"golang.org/x/net/websocket"
|
"golang.org/x/net/websocket"
|
||||||
|
|
||||||
"github.com/gen2brain/cam2ip/camera"
|
|
||||||
"github.com/gen2brain/cam2ip/encoder"
|
"github.com/gen2brain/cam2ip/encoder"
|
||||||
|
"github.com/gen2brain/cam2ip/reader"
|
||||||
)
|
)
|
||||||
|
|
||||||
// Socket handler.
|
// Socket handler.
|
||||||
type Socket struct {
|
type Socket struct {
|
||||||
camera *camera.Camera
|
reader reader.ImageReader
|
||||||
delay int
|
delay int
|
||||||
}
|
}
|
||||||
|
|
||||||
// NewSocket returns new socket handler.
|
// NewSocket returns new socket handler.
|
||||||
func NewSocket(camera *camera.Camera, delay int) websocket.Handler {
|
func NewSocket(reader reader.ImageReader, delay int) websocket.Handler {
|
||||||
s := &Socket{camera, delay}
|
s := &Socket{reader, delay}
|
||||||
return websocket.Handler(s.write)
|
return websocket.Handler(s.write)
|
||||||
}
|
}
|
||||||
|
|
||||||
// write writes images to socket
|
// write writes images to socket
|
||||||
func (s *Socket) write(ws *websocket.Conn) {
|
func (s *Socket) write(ws *websocket.Conn) {
|
||||||
for {
|
for {
|
||||||
img, err := s.camera.Read()
|
img, err := s.reader.Read()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Printf("socket: read: %v", err)
|
log.Printf("socket: read: %v", err)
|
||||||
continue
|
break
|
||||||
}
|
}
|
||||||
|
|
||||||
w := new(bytes.Buffer)
|
w := new(bytes.Buffer)
|
||||||
|
|||||||
15
reader/reader.go
Normal file
15
reader/reader.go
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
// Package reader.
|
||||||
|
package reader
|
||||||
|
|
||||||
|
import (
|
||||||
|
"image"
|
||||||
|
)
|
||||||
|
|
||||||
|
// ImageReader interface
|
||||||
|
type ImageReader interface {
|
||||||
|
// Read reads next frame from video and returns image.
|
||||||
|
Read() (img image.Image, err error)
|
||||||
|
|
||||||
|
// Close closes camera/video.
|
||||||
|
Close() error
|
||||||
|
}
|
||||||
@@ -8,8 +8,8 @@ import (
|
|||||||
|
|
||||||
"github.com/abbot/go-http-auth"
|
"github.com/abbot/go-http-auth"
|
||||||
|
|
||||||
"github.com/gen2brain/cam2ip/camera"
|
|
||||||
"github.com/gen2brain/cam2ip/handlers"
|
"github.com/gen2brain/cam2ip/handlers"
|
||||||
|
"github.com/gen2brain/cam2ip/reader"
|
||||||
)
|
)
|
||||||
|
|
||||||
// Server struct.
|
// Server struct.
|
||||||
@@ -20,13 +20,17 @@ type Server struct {
|
|||||||
Bind string
|
Bind string
|
||||||
Htpasswd string
|
Htpasswd string
|
||||||
|
|
||||||
Index int
|
Index int
|
||||||
Delay int
|
Delay int
|
||||||
|
|
||||||
FrameWidth float64
|
FrameWidth float64
|
||||||
FrameHeight float64
|
FrameHeight float64
|
||||||
NoWebGL bool
|
|
||||||
|
|
||||||
Camera *camera.Camera
|
NoWebGL bool
|
||||||
|
|
||||||
|
FileName string
|
||||||
|
|
||||||
|
Reader reader.ImageReader
|
||||||
}
|
}
|
||||||
|
|
||||||
// NewServer returns new Server.
|
// NewServer returns new Server.
|
||||||
@@ -44,10 +48,10 @@ func (s *Server) ListenAndServe() error {
|
|||||||
}
|
}
|
||||||
|
|
||||||
http.Handle("/html", newAuthHandler(handlers.NewHTML(s.Bind, s.FrameWidth, s.FrameHeight, s.NoWebGL), basic))
|
http.Handle("/html", newAuthHandler(handlers.NewHTML(s.Bind, s.FrameWidth, s.FrameHeight, s.NoWebGL), basic))
|
||||||
http.Handle("/jpeg", newAuthHandler(handlers.NewJPEG(s.Camera), basic))
|
http.Handle("/jpeg", newAuthHandler(handlers.NewJPEG(s.Reader), basic))
|
||||||
http.Handle("/mjpeg", newAuthHandler(handlers.NewMJPEG(s.Camera, s.Delay), basic))
|
http.Handle("/mjpeg", newAuthHandler(handlers.NewMJPEG(s.Reader, s.Delay), basic))
|
||||||
|
|
||||||
http.Handle("/socket", handlers.NewSocket(s.Camera, s.Delay))
|
http.Handle("/socket", handlers.NewSocket(s.Reader, s.Delay))
|
||||||
|
|
||||||
http.HandleFunc("/favicon.ico", func(w http.ResponseWriter, r *http.Request) {
|
http.HandleFunc("/favicon.ico", func(w http.ResponseWriter, r *http.Request) {
|
||||||
w.WriteHeader(http.StatusOK)
|
w.WriteHeader(http.StatusOK)
|
||||||
|
|||||||
50
video/video.go
Normal file
50
video/video.go
Normal file
@@ -0,0 +1,50 @@
|
|||||||
|
// Package video.
|
||||||
|
package video
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"image"
|
||||||
|
|
||||||
|
"github.com/lazywei/go-opencv/opencv"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Video represents video.
|
||||||
|
type Video struct {
|
||||||
|
video *opencv.Capture
|
||||||
|
}
|
||||||
|
|
||||||
|
// New returns new Video for given path.
|
||||||
|
func New(filename string) (video *Video, err error) {
|
||||||
|
video = &Video{}
|
||||||
|
|
||||||
|
video.video = opencv.NewFileCapture(filename)
|
||||||
|
if video.video == nil {
|
||||||
|
err = fmt.Errorf("video: can not open video %s", filename)
|
||||||
|
}
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// Read reads next frame from video and returns image.
|
||||||
|
func (v *Video) Read() (img image.Image, err error) {
|
||||||
|
if v.video.GrabFrame() {
|
||||||
|
frame := v.video.RetrieveFrame(1)
|
||||||
|
img = frame.ToImage()
|
||||||
|
} else {
|
||||||
|
err = fmt.Errorf("video: can not grab frame")
|
||||||
|
}
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// Close closes video.
|
||||||
|
func (v *Video) Close() (err error) {
|
||||||
|
if v.video == nil {
|
||||||
|
err = fmt.Errorf("video: video is not opened")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
v.video.Release()
|
||||||
|
v.video = nil
|
||||||
|
return
|
||||||
|
}
|
||||||
62
video/video_test.go
Normal file
62
video/video_test.go
Normal file
@@ -0,0 +1,62 @@
|
|||||||
|
package video
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"image/jpeg"
|
||||||
|
"io/ioutil"
|
||||||
|
"os"
|
||||||
|
"path/filepath"
|
||||||
|
"testing"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestVideo(t *testing.T) {
|
||||||
|
video, err := New("ranko.mp4")
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
defer video.Close()
|
||||||
|
|
||||||
|
tmpdir, err := ioutil.TempDir(os.TempDir(), "cam2ip")
|
||||||
|
if err != nil {
|
||||||
|
t.Error(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
defer os.RemoveAll(tmpdir)
|
||||||
|
|
||||||
|
var i int
|
||||||
|
var n int = 10
|
||||||
|
|
||||||
|
timeout := time.After(time.Duration(n) * time.Second)
|
||||||
|
|
||||||
|
for {
|
||||||
|
select {
|
||||||
|
case <-timeout:
|
||||||
|
//fmt.Printf("Fps: %d\n", i/n)
|
||||||
|
return
|
||||||
|
default:
|
||||||
|
i += 1
|
||||||
|
|
||||||
|
img, err := video.Read()
|
||||||
|
if err != nil {
|
||||||
|
t.Error(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
file, err := os.Create(filepath.Join(tmpdir, fmt.Sprintf("%03d.jpg", i)))
|
||||||
|
if err != nil {
|
||||||
|
t.Error(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
err = jpeg.Encode(file, img, &jpeg.Options{Quality: 75})
|
||||||
|
if err != nil {
|
||||||
|
t.Error(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
err = file.Close()
|
||||||
|
if err != nil {
|
||||||
|
t.Error(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user