2020-10-14 14:07:38 -07:00
|
|
|
package storageproviders
|
|
|
|
|
|
|
|
import (
|
2023-05-31 11:10:04 -07:00
|
|
|
"os"
|
|
|
|
"path/filepath"
|
|
|
|
"sort"
|
2020-10-14 14:07:38 -07:00
|
|
|
|
2023-05-31 11:10:04 -07:00
|
|
|
"github.com/pkg/errors"
|
2020-10-14 14:07:38 -07:00
|
|
|
log "github.com/sirupsen/logrus"
|
|
|
|
|
|
|
|
"github.com/owncast/owncast/config"
|
2023-05-30 14:05:24 -07:00
|
|
|
"github.com/owncast/owncast/core/data"
|
2023-08-09 16:19:09 -07:00
|
|
|
"github.com/owncast/owncast/utils"
|
2020-10-14 14:07:38 -07:00
|
|
|
)
|
|
|
|
|
2021-09-12 00:18:15 -07:00
|
|
|
// LocalStorage represents an instance of the local storage provider for HLS video.
|
2023-08-09 16:19:09 -07:00
|
|
|
type LocalStorage struct {
|
|
|
|
streamID string
|
2023-09-22 02:58:02 +02:00
|
|
|
host string
|
2023-08-09 16:19:09 -07:00
|
|
|
}
|
2020-10-14 14:07:38 -07:00
|
|
|
|
2021-10-11 18:29:36 -05:00
|
|
|
// NewLocalStorage returns a new LocalStorage instance.
|
|
|
|
func NewLocalStorage() *LocalStorage {
|
|
|
|
return &LocalStorage{}
|
|
|
|
}
|
2020-10-15 17:50:17 -07:00
|
|
|
|
2023-08-09 16:19:09 -07:00
|
|
|
// SetStreamId sets the stream id for this storage provider.
|
|
|
|
func (s *LocalStorage) SetStreamId(streamID string) {
|
|
|
|
s.streamID = streamID
|
|
|
|
}
|
|
|
|
|
2020-11-13 00:14:59 +01:00
|
|
|
// Setup configures this storage provider.
|
2020-10-14 14:07:38 -07:00
|
|
|
func (s *LocalStorage) Setup() error {
|
2023-09-22 02:58:02 +02:00
|
|
|
s.host = data.GetVideoServingEndpoint()
|
2020-10-14 14:07:38 -07:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2020-11-13 00:14:59 +01:00
|
|
|
// SegmentWritten is called when a single segment of video is written.
|
2023-08-09 16:19:09 -07:00
|
|
|
func (s *LocalStorage) SegmentWritten(localFilePath string) (string, int, error) {
|
|
|
|
if s.streamID == "" {
|
|
|
|
log.Fatalln("stream id must be set when handling video segments")
|
|
|
|
}
|
|
|
|
|
|
|
|
destinationPath, err := s.Save(localFilePath, localFilePath, 0)
|
|
|
|
if err != nil {
|
2020-11-14 18:39:53 -08:00
|
|
|
log.Warnln(err)
|
2023-08-09 16:19:09 -07:00
|
|
|
return "", 0, err
|
2020-11-14 18:39:53 -08:00
|
|
|
}
|
2023-08-09 16:19:09 -07:00
|
|
|
|
|
|
|
return destinationPath, 0, nil
|
2020-10-14 14:07:38 -07:00
|
|
|
}
|
|
|
|
|
2020-11-13 00:14:59 +01:00
|
|
|
// VariantPlaylistWritten is called when a variant hls playlist is written.
|
2020-10-14 14:07:38 -07:00
|
|
|
func (s *LocalStorage) VariantPlaylistWritten(localFilePath string) {
|
2023-08-09 16:19:09 -07:00
|
|
|
if s.streamID == "" {
|
|
|
|
log.Fatalln("stream id must be set when handling video playlists")
|
|
|
|
}
|
|
|
|
|
|
|
|
if _, err := s.Save(localFilePath, localFilePath, 0); err != nil {
|
2020-10-16 15:04:31 -07:00
|
|
|
log.Errorln(err)
|
2020-10-14 14:07:38 -07:00
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-13 00:14:59 +01:00
|
|
|
// MasterPlaylistWritten is called when the master hls playlist is written.
|
2020-10-14 14:07:38 -07:00
|
|
|
func (s *LocalStorage) MasterPlaylistWritten(localFilePath string) {
|
2023-08-09 16:19:09 -07:00
|
|
|
if s.streamID == "" {
|
|
|
|
log.Fatalln("stream id must be set when handling video playlists")
|
|
|
|
}
|
|
|
|
|
|
|
|
masterPlaylistDestinationLocation := filepath.Join(config.HLSStoragePath, "/stream.m3u8")
|
|
|
|
if err := rewriteLocalPlaylist(localFilePath, s.streamID, masterPlaylistDestinationLocation); err != nil {
|
|
|
|
log.Errorln(err)
|
|
|
|
return
|
2020-11-14 18:39:53 -08:00
|
|
|
}
|
2020-10-14 14:07:38 -07:00
|
|
|
}
|
|
|
|
|
2020-11-13 00:14:59 +01:00
|
|
|
// Save will save a local filepath using the storage provider.
|
2023-08-09 16:19:09 -07:00
|
|
|
func (s *LocalStorage) Save(filePath, destinationPath string, retryCount int) (string, error) {
|
|
|
|
if filePath != destinationPath {
|
|
|
|
if err := utils.Move(filePath, destinationPath); err != nil {
|
|
|
|
return "", errors.Wrap(err, "unable to move file")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return destinationPath, nil
|
2020-10-14 14:07:38 -07:00
|
|
|
}
|
2023-05-31 11:10:04 -07:00
|
|
|
|
|
|
|
func (s *LocalStorage) Cleanup() error {
|
2023-08-09 16:19:09 -07:00
|
|
|
// If we're recording, don't perform the cleanup.
|
2023-08-09 16:35:42 -07:00
|
|
|
if config.EnableReplayFeatures {
|
2023-08-09 16:19:09 -07:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2023-05-31 11:10:04 -07:00
|
|
|
// Determine how many files we should keep on disk
|
|
|
|
maxNumber := data.GetStreamLatencyLevel().SegmentCount
|
|
|
|
buffer := 10
|
2023-08-20 20:22:00 -07:00
|
|
|
baseDirectory := filepath.Join(config.HLSStoragePath, s.streamID)
|
2023-05-31 11:10:04 -07:00
|
|
|
|
|
|
|
files, err := getAllFilesRecursive(baseDirectory)
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "unable find old video files for cleanup")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Delete old private HLS files on disk
|
|
|
|
for directory := range files {
|
|
|
|
files := files[directory]
|
|
|
|
if len(files) < maxNumber+buffer {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
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 {
|
|
|
|
return errors.Wrap(err, "unable to delete old video files")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2023-08-09 16:19:09 -07:00
|
|
|
func (s *LocalStorage) GetRemoteDestinationPathFromLocalFilePath(localFilePath string) string {
|
|
|
|
return localFilePath
|
|
|
|
}
|
|
|
|
|
2023-05-31 11:10:04 -07:00
|
|
|
func getAllFilesRecursive(baseDirectory string) (map[string][]os.FileInfo, error) {
|
|
|
|
files := make(map[string][]os.FileInfo)
|
|
|
|
|
|
|
|
var directory string
|
|
|
|
err := filepath.Walk(baseDirectory, func(path string, info os.FileInfo, err error) error {
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if info.IsDir() {
|
|
|
|
directory = info.Name()
|
|
|
|
}
|
|
|
|
|
|
|
|
if filepath.Ext(info.Name()) == ".ts" {
|
|
|
|
files[directory] = append(files[directory], info)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// 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
|
|
|
|
}
|