mirror of
https://github.com/owncast/owncast.git
synced 2024-11-24 05:38:58 +03:00
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:
parent
b92ad00926
commit
1b053ffd1b
11 changed files with 71 additions and 67 deletions
|
@ -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
50
controllers/hls.go
Normal 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)
|
||||||
|
}
|
|
@ -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)
|
||||||
|
|
||||||
|
|
|
@ -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()
|
||||||
|
|
|
@ -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
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -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()
|
||||||
|
|
||||||
|
|
|
@ -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)
|
||||||
}
|
}
|
||||||
}()
|
}()
|
||||||
}
|
}
|
||||||
|
|
|
@ -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()
|
||||||
|
|
|
@ -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
|
||||||
|
|
||||||
|
|
|
@ -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)
|
||||||
|
|
|
@ -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))
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue