Cleanup private and optinally public segments on a timer instead of after every save. Closes #254
This commit is contained in:
parent
2d71337146
commit
8f740c4cb6
@ -10,54 +10,65 @@ import (
|
|||||||
"github.com/owncast/owncast/config"
|
"github.com/owncast/owncast/config"
|
||||||
)
|
)
|
||||||
|
|
||||||
// Cleanup will delete old files off disk that are no longer being referenced
|
// CleanupOldContent will delete old files from the private dir that are no longer being referenced
|
||||||
// in the stream.
|
// in the stream.
|
||||||
func Cleanup(directoryPath string) {
|
func CleanupOldContent(baseDirectory string) {
|
||||||
// Determine how many files we should keep on disk
|
// Determine how many files we should keep on disk
|
||||||
maxNumber := config.Config.GetMaxNumberOfReferencedSegmentsInPlaylist()
|
maxNumber := config.Config.GetMaxNumberOfReferencedSegmentsInPlaylist()
|
||||||
buffer := 10
|
buffer := 10
|
||||||
|
|
||||||
files, err := getSegmentFiles(directoryPath)
|
files, err := getAllFilesRecursive(baseDirectory)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatal(err)
|
log.Fatal(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Delete old private HLS files on disk
|
||||||
|
for directory := range files {
|
||||||
|
files := files[directory]
|
||||||
if len(files) < maxNumber+buffer {
|
if len(files) < maxNumber+buffer {
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
// Delete old files on disk
|
|
||||||
filesToDelete := files[maxNumber+buffer:]
|
|
||||||
for _, file := range filesToDelete {
|
|
||||||
os.Remove(filepath.Join(directoryPath, file.Name()))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func getSegmentFiles(dirname string) ([]os.FileInfo, error) {
|
|
||||||
f, err := os.Open(dirname)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
list, err := f.Readdir(-1) // -1 says to get a list of all files
|
|
||||||
f.Close()
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
filteredList := make([]os.FileInfo, 0)
|
|
||||||
|
|
||||||
// Filter out playlists because we don't want to clean them up
|
|
||||||
for _, file := range list {
|
|
||||||
if filepath.Ext(file.Name()) == ".m3u8" {
|
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
filteredList = append(filteredList, file)
|
|
||||||
|
filesToDelete := files[maxNumber+buffer:]
|
||||||
|
log.Traceln("Deleting", len(filesToDelete), "old files from", baseDirectory, "for video variant", directory)
|
||||||
|
|
||||||
|
for _, file := range filesToDelete {
|
||||||
|
fileToDelete := filepath.Join(baseDirectory, directory, file.Name())
|
||||||
|
err := os.Remove(fileToDelete)
|
||||||
|
if err != nil {
|
||||||
|
log.Errorln(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func getAllFilesRecursive(baseDirectory string) (map[string][]os.FileInfo, error) {
|
||||||
|
var files = make(map[string][]os.FileInfo)
|
||||||
|
|
||||||
|
var directory string
|
||||||
|
filepath.Walk(baseDirectory, func(path string, info os.FileInfo, err error) error {
|
||||||
|
if err != nil {
|
||||||
|
log.Fatalf(err.Error())
|
||||||
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
// Sort by date so we can delete old files
|
if info.IsDir() {
|
||||||
sort.Slice(filteredList, func(i, j int) bool {
|
directory = info.Name()
|
||||||
return filteredList[i].ModTime().UnixNano() > filteredList[j].ModTime().UnixNano()
|
}
|
||||||
|
|
||||||
|
if filepath.Ext(info.Name()) == ".ts" {
|
||||||
|
files[directory] = append(files[directory], info)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
})
|
})
|
||||||
|
|
||||||
return filteredList, nil
|
// Sort by date so we can delete old files
|
||||||
|
for directory := range files {
|
||||||
|
sort.Slice(files[directory], func(i, j int) bool {
|
||||||
|
return files[directory][i].ModTime().UnixNano() > files[directory][j].ModTime().UnixNano()
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
return files, nil
|
||||||
}
|
}
|
||||||
|
@ -2,6 +2,7 @@ package storageproviders
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
|
"time"
|
||||||
|
|
||||||
log "github.com/sirupsen/logrus"
|
log "github.com/sirupsen/logrus"
|
||||||
|
|
||||||
@ -13,9 +14,22 @@ import (
|
|||||||
type LocalStorage struct {
|
type LocalStorage struct {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Cleanup old public HLS content every N min from the webroot.
|
||||||
|
var _onlineCleanupTicker *time.Ticker
|
||||||
|
|
||||||
// Setup configures this storage provider
|
// Setup configures this storage provider
|
||||||
func (s *LocalStorage) Setup() error {
|
func (s *LocalStorage) Setup() error {
|
||||||
// no-op
|
// NOTE: This cleanup timer will have to be disabled to support recordings in the future
|
||||||
|
// as all HLS segments have to be publicly available on disk to keep a recording of them.
|
||||||
|
_onlineCleanupTicker = time.NewTicker(1 * time.Minute)
|
||||||
|
go func() {
|
||||||
|
for {
|
||||||
|
select {
|
||||||
|
case <-_onlineCleanupTicker.C:
|
||||||
|
ffmpeg.CleanupOldContent(config.PublicHLSStoragePath)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}()
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -49,15 +63,7 @@ func (s *LocalStorage) Save(filePath string, retryCount int) (string, error) {
|
|||||||
newPath = filepath.Join(config.WebRoot, filePath)
|
newPath = filepath.Join(config.WebRoot, filePath)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Move video segments to the destination directory.
|
|
||||||
// Copy playlists to the destination directory so they can still be referenced in
|
|
||||||
// the private hls working directory.
|
|
||||||
if filepath.Ext(filePath) == ".m3u8" {
|
|
||||||
utils.Copy(filePath, newPath)
|
utils.Copy(filePath, newPath)
|
||||||
} else {
|
|
||||||
utils.Move(filePath, newPath)
|
|
||||||
ffmpeg.Cleanup(filepath.Dir(newPath))
|
|
||||||
}
|
|
||||||
|
|
||||||
return newPath, nil
|
return newPath, nil
|
||||||
}
|
}
|
||||||
|
@ -6,7 +6,6 @@ import (
|
|||||||
"os"
|
"os"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
|
|
||||||
"github.com/owncast/owncast/core/ffmpeg"
|
|
||||||
"github.com/owncast/owncast/core/playlist"
|
"github.com/owncast/owncast/core/playlist"
|
||||||
"github.com/owncast/owncast/utils"
|
"github.com/owncast/owncast/utils"
|
||||||
log "github.com/sirupsen/logrus"
|
log "github.com/sirupsen/logrus"
|
||||||
@ -160,8 +159,6 @@ func (s *S3Storage) Save(filePath string, retryCount int) (string, error) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
ffmpeg.Cleanup(filepath.Dir(filePath))
|
|
||||||
|
|
||||||
return response.Location, nil
|
return response.Location, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -16,7 +16,11 @@ import (
|
|||||||
"github.com/grafov/m3u8"
|
"github.com/grafov/m3u8"
|
||||||
)
|
)
|
||||||
|
|
||||||
var _cleanupTimer *time.Timer
|
// After the stream goes offline this timer fires a full cleanup after N min.
|
||||||
|
var _offlineCleanupTimer *time.Timer
|
||||||
|
|
||||||
|
// While a stream takes place cleanup old HLS content every N min.
|
||||||
|
var _onlineCleanupTicker *time.Ticker
|
||||||
|
|
||||||
//SetStreamAsConnected sets the stream as connected
|
//SetStreamAsConnected sets the stream as connected
|
||||||
func SetStreamAsConnected() {
|
func SetStreamAsConnected() {
|
||||||
@ -25,6 +29,8 @@ func SetStreamAsConnected() {
|
|||||||
_stats.LastDisconnectTime = utils.NullTime{time.Now(), false}
|
_stats.LastDisconnectTime = utils.NullTime{time.Now(), false}
|
||||||
|
|
||||||
StopOfflineCleanupTimer()
|
StopOfflineCleanupTimer()
|
||||||
|
startOnlineCleanupTimer()
|
||||||
|
|
||||||
if _yp != nil {
|
if _yp != nil {
|
||||||
_yp.Start()
|
_yp.Start()
|
||||||
}
|
}
|
||||||
@ -112,15 +118,16 @@ func SetStreamAsDisconnected() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
StartOfflineCleanupTimer()
|
StartOfflineCleanupTimer()
|
||||||
|
stopOnlineCleanupTimer()
|
||||||
}
|
}
|
||||||
|
|
||||||
// StartOfflineCleanupTimer will fire a cleanup after n minutes being disconnected
|
// StartOfflineCleanupTimer will fire a cleanup after n minutes being disconnected
|
||||||
func StartOfflineCleanupTimer() {
|
func StartOfflineCleanupTimer() {
|
||||||
_cleanupTimer = time.NewTimer(5 * time.Minute)
|
_offlineCleanupTimer = time.NewTimer(5 * time.Minute)
|
||||||
go func() {
|
go func() {
|
||||||
for {
|
for {
|
||||||
select {
|
select {
|
||||||
case <-_cleanupTimer.C:
|
case <-_offlineCleanupTimer.C:
|
||||||
// Reset the session count since the session is over
|
// Reset the session count since the session is over
|
||||||
_stats.SessionMaxViewerCount = 0
|
_stats.SessionMaxViewerCount = 0
|
||||||
resetDirectories()
|
resetDirectories()
|
||||||
@ -132,7 +139,25 @@ func StartOfflineCleanupTimer() {
|
|||||||
|
|
||||||
// StopOfflineCleanupTimer will stop the previous cleanup timer
|
// StopOfflineCleanupTimer will stop the previous cleanup timer
|
||||||
func StopOfflineCleanupTimer() {
|
func StopOfflineCleanupTimer() {
|
||||||
if _cleanupTimer != nil {
|
if _offlineCleanupTimer != nil {
|
||||||
_cleanupTimer.Stop()
|
_offlineCleanupTimer.Stop()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func startOnlineCleanupTimer() {
|
||||||
|
_onlineCleanupTicker = time.NewTicker(1 * time.Minute)
|
||||||
|
go func() {
|
||||||
|
for {
|
||||||
|
select {
|
||||||
|
case <-_onlineCleanupTicker.C:
|
||||||
|
ffmpeg.CleanupOldContent(config.PrivateHLSStoragePath)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
}
|
||||||
|
|
||||||
|
func stopOnlineCleanupTimer() {
|
||||||
|
if _onlineCleanupTicker != nil {
|
||||||
|
_onlineCleanupTicker.Stop()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user