Simplify HLS storage paths (#1393)

* Remove private vs public HLS paths and add a HLS controller. Closes #875

* Use http.ServeFile instead
This commit is contained in:
Gabe Kangas 2021-09-12 11:32:42 -07:00 committed by GitHub
parent b92ad00926
commit 1b053ffd1b
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
11 changed files with 71 additions and 67 deletions

View file

@ -7,8 +7,6 @@ const (
StaticVersionNumber = "0.0.9" // Shown when you build from develop StaticVersionNumber = "0.0.9" // Shown when you build from develop
// WebRoot is the web server root directory. // WebRoot is the web server root directory.
WebRoot = "webroot" WebRoot = "webroot"
// PrivateHLSStoragePath is the HLS write directory.
PrivateHLSStoragePath = "hls"
// FfmpegSuggestedVersion is the version of ffmpeg we suggest. // FfmpegSuggestedVersion is the version of ffmpeg we suggest.
FfmpegSuggestedVersion = "v4.1.5" // Requires the v FfmpegSuggestedVersion = "v4.1.5" // Requires the v
// DataDirectory is the directory we save data to. // DataDirectory is the directory we save data to.
@ -18,8 +16,9 @@ const (
) )
var ( var (
// PublicHLSStoragePath is the directory we write public HLS files to for distribution.
PublicHLSStoragePath = filepath.Join(WebRoot, "hls")
// BackupDirectory is the directory we write backup files to. // BackupDirectory is the directory we write backup files to.
BackupDirectory = filepath.Join(DataDirectory, "backup") BackupDirectory = filepath.Join(DataDirectory, "backup")
// HLSStoragePath is the directory HLS video is written to.
HLSStoragePath = filepath.Join(DataDirectory, "hls")
) )

50
controllers/hls.go Normal file
View file

@ -0,0 +1,50 @@
package controllers
import (
"net/http"
"path"
"path/filepath"
"strconv"
"strings"
"github.com/owncast/owncast/config"
"github.com/owncast/owncast/core"
"github.com/owncast/owncast/core/data"
"github.com/owncast/owncast/router/middleware"
"github.com/owncast/owncast/utils"
)
// HandleHLSRequest will manage all requests to HLS content.
func HandleHLSRequest(w http.ResponseWriter, r *http.Request) {
// Sanity check to limit requests to HLS file types.
if filepath.Ext(r.URL.Path) != ".m3u8" && filepath.Ext(r.URL.Path) != ".ts" {
w.WriteHeader(http.StatusNotFound)
return
}
requestedPath := r.URL.Path
relativePath := strings.Replace(requestedPath, "/hls/", "", 1)
fullPath := filepath.Join(config.HLSStoragePath, relativePath)
// If using external storage then only allow requests for the
// master playlist at stream.m3u8, no variants or segments.
if data.GetS3Config().Enabled && relativePath != "stream.m3u8" {
w.WriteHeader(http.StatusNotFound)
return
}
// Handle playlists
if path.Ext(r.URL.Path) == ".m3u8" {
// Playlists should never be cached.
middleware.DisableCache(w)
// Use this as an opportunity to mark this viewer as active.
id := utils.GenerateClientIDFromRequest(r)
core.SetViewerIDActive(id)
} else {
cacheTime := utils.GetCacheDurationSecondsForPath(relativePath)
w.Header().Set("Cache-Control", "public, max-age="+strconv.Itoa(cacheTime))
}
http.ServeFile(w, r, fullPath)
}

View file

@ -62,14 +62,6 @@ func IndexHandler(w http.ResponseWriter, r *http.Request) {
return return
} }
if path.Ext(r.URL.Path) == ".m3u8" {
middleware.DisableCache(w)
// Use this as an opportunity to mark this viewer as active.
id := utils.GenerateClientIDFromRequest(r)
core.SetViewerIDActive(id)
}
// Set a cache control max-age header // Set a cache control max-age header
middleware.SetCachingHeaders(w, r) middleware.SetCachingHeaders(w, r)

View file

@ -118,9 +118,8 @@ func transitionToOfflineVideoStreamContent() {
func resetDirectories() { func resetDirectories() {
log.Trace("Resetting file directories to a clean slate.") log.Trace("Resetting file directories to a clean slate.")
// Wipe the public, web-accessible hls data directory // Wipe hls data directory
utils.CleanupDirectory(config.PublicHLSStoragePath) utils.CleanupDirectory(config.HLSStoragePath)
utils.CleanupDirectory(config.PrivateHLSStoragePath)
// Remove the previous thumbnail // Remove the previous thumbnail
logo := data.GetLogoPath() logo := data.GetLogoPath()

View file

@ -1,14 +1,12 @@
package storageproviders package storageproviders
import ( import (
"path/filepath"
"time" "time"
log "github.com/sirupsen/logrus" log "github.com/sirupsen/logrus"
"github.com/owncast/owncast/config" "github.com/owncast/owncast/config"
"github.com/owncast/owncast/core/transcoder" "github.com/owncast/owncast/core/transcoder"
"github.com/owncast/owncast/utils"
) )
// LocalStorage represents an instance of the local storage provider for HLS video. // LocalStorage represents an instance of the local storage provider for HLS video.
@ -25,7 +23,7 @@ func (s *LocalStorage) Setup() error {
_onlineCleanupTicker = time.NewTicker(1 * time.Minute) _onlineCleanupTicker = time.NewTicker(1 * time.Minute)
go func() { go func() {
for range _onlineCleanupTicker.C { for range _onlineCleanupTicker.C {
transcoder.CleanupOldContent(config.PublicHLSStoragePath) transcoder.CleanupOldContent(config.HLSStoragePath)
} }
}() }()
return nil return nil
@ -55,15 +53,5 @@ func (s *LocalStorage) MasterPlaylistWritten(localFilePath string) {
// Save will save a local filepath using the storage provider. // Save will save a local filepath using the storage provider.
func (s *LocalStorage) Save(filePath string, retryCount int) (string, error) { func (s *LocalStorage) Save(filePath string, retryCount int) (string, error) {
newPath := "" return filePath, nil
// This is a hack
if filePath == "hls/stream.m3u8" {
newPath = filepath.Join(config.PublicHLSStoragePath, filepath.Base(filePath))
} else {
newPath = filepath.Join(config.WebRoot, filePath)
}
err := utils.Copy(filePath, newPath)
return newPath, err
} }

View file

@ -199,7 +199,7 @@ func (s *S3Storage) rewriteRemotePlaylist(filePath string) error {
item.URI = s.host + filepath.Join("/hls", item.URI) item.URI = s.host + filepath.Join("/hls", item.URI)
} }
publicPath := filepath.Join(config.PublicHLSStoragePath, filepath.Base(filePath)) publicPath := filepath.Join(config.HLSStoragePath, filepath.Base(filePath))
newPlaylist := p.String() newPlaylist := p.String()

View file

@ -50,17 +50,12 @@ func setStreamAsConnected(rtmpOut *io.PipeReader) {
go _yp.Start() go _yp.Start()
} }
segmentPath := config.PublicHLSStoragePath segmentPath := config.HLSStoragePath
s3Config := data.GetS3Config()
if err := setupStorage(); err != nil { if err := setupStorage(); err != nil {
log.Fatalln("failed to setup the storage", err) log.Fatalln("failed to setup the storage", err)
} }
if s3Config.Enabled {
segmentPath = config.PrivateHLSStoragePath
}
go func() { go func() {
_transcoder = transcoder.NewTranscoder() _transcoder = transcoder.NewTranscoder()
_transcoder.TranscoderCompleted = func(error) { _transcoder.TranscoderCompleted = func(error) {
@ -100,8 +95,8 @@ func SetStreamAsDisconnected() {
} }
for index := range _currentBroadcast.OutputSettings { for index := range _currentBroadcast.OutputSettings {
playlistFilePath := fmt.Sprintf(filepath.Join(config.PrivateHLSStoragePath, "%d/stream.m3u8"), index) playlistFilePath := fmt.Sprintf(filepath.Join(config.HLSStoragePath, "%d/stream.m3u8"), index)
segmentFilePath := fmt.Sprintf(filepath.Join(config.PrivateHLSStoragePath, "%d/%s"), index, offlineFilename) segmentFilePath := fmt.Sprintf(filepath.Join(config.HLSStoragePath, "%d/%s"), index, offlineFilename)
if err := utils.Copy(offlineFilePath, segmentFilePath); err != nil { if err := utils.Copy(offlineFilePath, segmentFilePath); err != nil {
log.Warnln(err) log.Warnln(err)
@ -191,7 +186,7 @@ func startOnlineCleanupTimer() {
_onlineCleanupTicker = time.NewTicker(1 * time.Minute) _onlineCleanupTicker = time.NewTicker(1 * time.Minute)
go func() { go func() {
for range _onlineCleanupTicker.C { for range _onlineCleanupTicker.C {
transcoder.CleanupOldContent(config.PrivateHLSStoragePath) transcoder.CleanupOldContent(config.HLSStoragePath)
} }
}() }()
} }

View file

@ -59,7 +59,7 @@ func (s *FileWriterReceiverService) uploadHandler(w http.ResponseWriter, r *http
} }
path := r.URL.Path path := r.URL.Path
writePath := filepath.Join(config.PrivateHLSStoragePath, path) writePath := filepath.Join(config.HLSStoragePath, path)
var buf bytes.Buffer var buf bytes.Buffer
defer r.Body.Close() defer r.Body.Close()

View file

@ -233,19 +233,8 @@ func NewTranscoder() *Transcoder {
transcoder.currentStreamOutputSettings = data.GetStreamOutputVariants() transcoder.currentStreamOutputSettings = data.GetStreamOutputVariants()
transcoder.currentLatencyLevel = data.GetStreamLatencyLevel() transcoder.currentLatencyLevel = data.GetStreamLatencyLevel()
transcoder.codec = getCodec(data.GetVideoCodec()) transcoder.codec = getCodec(data.GetVideoCodec())
transcoder.segmentOutputPath = config.HLSStoragePath
var outputPath string transcoder.playlistOutputPath = config.HLSStoragePath
if data.GetS3Config().Enabled {
// Segments are not available via the local HTTP server
outputPath = config.PrivateHLSStoragePath
} else {
// Segments are available via the local HTTP server
outputPath = config.PublicHLSStoragePath
}
transcoder.segmentOutputPath = outputPath
// Playlists are available via the local HTTP server
transcoder.playlistOutputPath = config.PublicHLSStoragePath
transcoder.input = "pipe:0" // stdin transcoder.input = "pipe:0" // stdin

View file

@ -94,27 +94,16 @@ func handleTranscoderMessage(message string) {
func createVariantDirectories() { func createVariantDirectories() {
// Create private hls data dirs // Create private hls data dirs
utils.CleanupDirectory(config.PublicHLSStoragePath) utils.CleanupDirectory(config.HLSStoragePath)
utils.CleanupDirectory(config.PrivateHLSStoragePath)
if len(data.GetStreamOutputVariants()) != 0 { if len(data.GetStreamOutputVariants()) != 0 {
for index := range data.GetStreamOutputVariants() { for index := range data.GetStreamOutputVariants() {
if err := os.MkdirAll(path.Join(config.PrivateHLSStoragePath, strconv.Itoa(index)), 0750); err != nil { if err := os.MkdirAll(path.Join(config.HLSStoragePath, strconv.Itoa(index)), 0750); err != nil {
log.Fatalln(err)
}
dir := path.Join(config.PublicHLSStoragePath, strconv.Itoa(index))
log.Traceln("Creating", dir)
if err := os.MkdirAll(dir, 0750); err != nil {
log.Fatalln(err) log.Fatalln(err)
} }
} }
} else { } else {
dir := path.Join(config.PrivateHLSStoragePath, strconv.Itoa(0)) dir := path.Join(config.HLSStoragePath, strconv.Itoa(0))
log.Traceln("Creating", dir)
if err := os.MkdirAll(dir, 0750); err != nil {
log.Fatalln(err)
}
dir = path.Join(config.PublicHLSStoragePath, strconv.Itoa(0))
log.Traceln("Creating", dir) log.Traceln("Creating", dir)
if err := os.MkdirAll(dir, 0750); err != nil { if err := os.MkdirAll(dir, 0750); err != nil {
log.Fatalln(err) log.Fatalln(err)

View file

@ -74,6 +74,9 @@ func Start() error {
// Current inbound broadcaster // Current inbound broadcaster
http.HandleFunc("/api/admin/status", middleware.RequireAdminAuth(admin.Status)) http.HandleFunc("/api/admin/status", middleware.RequireAdminAuth(admin.Status))
// Return HLS video
http.HandleFunc("/hls/", controllers.HandleHLSRequest)
// Disconnect inbound stream // Disconnect inbound stream
http.HandleFunc("/api/admin/disconnect", middleware.RequireAdminAuth(admin.DisconnectInboundConnection)) http.HandleFunc("/api/admin/disconnect", middleware.RequireAdminAuth(admin.DisconnectInboundConnection))