2023-09-16 15:27:07 +00:00
|
|
|
package record
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"io/fs"
|
|
|
|
"os"
|
|
|
|
"path/filepath"
|
|
|
|
"strings"
|
|
|
|
"time"
|
|
|
|
|
2023-10-14 20:52:10 +00:00
|
|
|
"github.com/bluenviron/mediamtx/internal/conf"
|
2023-09-16 15:27:07 +00:00
|
|
|
"github.com/bluenviron/mediamtx/internal/logger"
|
|
|
|
)
|
|
|
|
|
|
|
|
func commonPath(v string) string {
|
|
|
|
common := ""
|
|
|
|
remaining := v
|
|
|
|
|
|
|
|
for {
|
|
|
|
i := strings.IndexAny(remaining, "\\/")
|
|
|
|
if i < 0 {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
|
|
|
|
var part string
|
|
|
|
part, remaining = remaining[:i+1], remaining[i+1:]
|
|
|
|
|
|
|
|
if strings.Contains(part, "%") {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
|
|
|
|
common += part
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(common) > 0 {
|
|
|
|
common = common[:len(common)-1]
|
|
|
|
}
|
|
|
|
|
|
|
|
return common
|
|
|
|
}
|
|
|
|
|
2023-10-07 21:48:37 +00:00
|
|
|
// CleanerEntry is a cleaner entry.
|
|
|
|
type CleanerEntry struct {
|
2023-12-02 14:35:21 +00:00
|
|
|
PathFormat string
|
|
|
|
Format conf.RecordFormat
|
|
|
|
DeleteAfter time.Duration
|
2023-10-07 21:48:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Cleaner removes expired recording segments from disk.
|
2023-09-16 15:27:07 +00:00
|
|
|
type Cleaner struct {
|
2023-12-02 14:15:17 +00:00
|
|
|
Entries []CleanerEntry
|
|
|
|
Parent logger.Writer
|
|
|
|
|
2023-10-07 21:48:37 +00:00
|
|
|
ctx context.Context
|
|
|
|
ctxCancel func()
|
2023-09-16 15:27:07 +00:00
|
|
|
|
|
|
|
done chan struct{}
|
|
|
|
}
|
|
|
|
|
2023-12-02 14:15:17 +00:00
|
|
|
// Initialize initializes a Cleaner.
|
|
|
|
func (c *Cleaner) Initialize() {
|
|
|
|
c.ctx, c.ctxCancel = context.WithCancel(context.Background())
|
|
|
|
c.done = make(chan struct{})
|
2023-09-16 15:27:07 +00:00
|
|
|
|
|
|
|
go c.run()
|
|
|
|
}
|
|
|
|
|
|
|
|
// Close closes the Cleaner.
|
|
|
|
func (c *Cleaner) Close() {
|
|
|
|
c.ctxCancel()
|
|
|
|
<-c.done
|
|
|
|
}
|
|
|
|
|
|
|
|
// Log is the main logging function.
|
|
|
|
func (c *Cleaner) Log(level logger.Level, format string, args ...interface{}) {
|
2023-12-02 14:15:17 +00:00
|
|
|
c.Parent.Log(level, "[record cleaner]"+format, args...)
|
2023-09-16 15:27:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Cleaner) run() {
|
|
|
|
defer close(c.done)
|
|
|
|
|
|
|
|
interval := 30 * 60 * time.Second
|
2023-12-02 14:15:17 +00:00
|
|
|
for _, e := range c.Entries {
|
|
|
|
if interval > (e.DeleteAfter / 2) {
|
|
|
|
interval = e.DeleteAfter / 2
|
2023-10-07 21:48:37 +00:00
|
|
|
}
|
2023-09-16 15:27:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
c.doRun() //nolint:errcheck
|
|
|
|
|
|
|
|
for {
|
|
|
|
select {
|
|
|
|
case <-time.After(interval):
|
2023-10-07 21:48:37 +00:00
|
|
|
c.doRun()
|
2023-09-16 15:27:07 +00:00
|
|
|
|
|
|
|
case <-c.ctx.Done():
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-10-07 21:48:37 +00:00
|
|
|
func (c *Cleaner) doRun() {
|
2023-12-02 14:15:17 +00:00
|
|
|
for _, e := range c.Entries {
|
2023-10-07 21:48:37 +00:00
|
|
|
c.doRunEntry(&e) //nolint:errcheck
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Cleaner) doRunEntry(e *CleanerEntry) error {
|
2023-12-02 14:35:21 +00:00
|
|
|
pathFormat := e.PathFormat
|
2023-10-14 20:52:10 +00:00
|
|
|
|
2023-12-02 14:15:17 +00:00
|
|
|
switch e.Format {
|
2023-10-14 20:52:10 +00:00
|
|
|
case conf.RecordFormatMPEGTS:
|
2023-12-02 14:35:21 +00:00
|
|
|
pathFormat += ".ts"
|
2023-10-14 20:52:10 +00:00
|
|
|
|
|
|
|
default:
|
2023-12-02 14:35:21 +00:00
|
|
|
pathFormat += ".mp4"
|
2023-10-14 20:52:10 +00:00
|
|
|
}
|
|
|
|
|
2023-12-02 14:15:17 +00:00
|
|
|
// we have to convert to absolute paths
|
|
|
|
// otherwise, commonPath and fpath inside Walk() won't have common elements
|
2023-12-02 14:35:21 +00:00
|
|
|
pathFormat, _ = filepath.Abs(pathFormat)
|
2023-12-02 14:15:17 +00:00
|
|
|
|
2023-12-02 14:35:21 +00:00
|
|
|
commonPath := commonPath(pathFormat)
|
2023-09-16 15:27:07 +00:00
|
|
|
now := timeNow()
|
|
|
|
|
2023-11-10 15:03:04 +00:00
|
|
|
filepath.Walk(commonPath, func(fpath string, info fs.FileInfo, err error) error { //nolint:errcheck
|
2023-09-16 15:27:07 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if !info.IsDir() {
|
2023-12-02 14:35:21 +00:00
|
|
|
var pa path
|
|
|
|
ok := pa.decode(pathFormat, fpath)
|
2023-12-02 14:15:17 +00:00
|
|
|
if ok {
|
2023-12-02 14:35:21 +00:00
|
|
|
if now.Sub(time.Time(pa)) > e.DeleteAfter {
|
2023-11-10 15:03:04 +00:00
|
|
|
c.Log(logger.Debug, "removing %s", fpath)
|
|
|
|
os.Remove(fpath)
|
2023-09-16 15:27:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
|
2023-11-10 15:03:04 +00:00
|
|
|
filepath.Walk(commonPath, func(fpath string, info fs.FileInfo, err error) error { //nolint:errcheck
|
2023-09-16 15:27:07 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if info.IsDir() {
|
2023-11-10 15:03:04 +00:00
|
|
|
os.Remove(fpath)
|
2023-09-16 15:27:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|