2020-06-22 20:11:56 -05:00
|
|
|
package config
|
|
|
|
|
|
|
|
import (
|
|
|
|
"errors"
|
|
|
|
"io/ioutil"
|
2020-11-09 19:35:01 -08:00
|
|
|
"os/exec"
|
|
|
|
"strings"
|
2020-06-22 20:11:56 -05:00
|
|
|
|
2020-10-06 01:07:09 +08:00
|
|
|
"github.com/owncast/owncast/utils"
|
2020-06-22 20:11:56 -05:00
|
|
|
log "github.com/sirupsen/logrus"
|
|
|
|
"gopkg.in/yaml.v2"
|
|
|
|
)
|
|
|
|
|
|
|
|
//Config contains a reference to the configuration
|
|
|
|
var Config *config
|
2020-07-17 18:27:00 -07:00
|
|
|
var _default config
|
2020-06-22 20:11:56 -05:00
|
|
|
|
|
|
|
type config struct {
|
2020-11-05 18:29:16 -08:00
|
|
|
DatabaseFilePath string `yaml:"databaseFile"`
|
|
|
|
EnableDebugFeatures bool `yaml:"-"`
|
|
|
|
FFMpegPath string `yaml:"ffmpegPath"`
|
|
|
|
Files files `yaml:"files"`
|
|
|
|
InstanceDetails InstanceDetails `yaml:"instanceDetails"`
|
|
|
|
S3 S3 `yaml:"s3"`
|
|
|
|
VersionInfo string `yaml:"-"` // For storing the version/build number
|
|
|
|
VersionNumber string `yaml:"-"`
|
|
|
|
VideoSettings videoSettings `yaml:"videoSettings"`
|
|
|
|
WebServerPort int `yaml:"webServerPort"`
|
|
|
|
DisableUpgradeChecks bool `yaml:"disableUpgradeChecks"`
|
|
|
|
YP YP `yaml:"yp"`
|
2020-06-28 15:10:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
// InstanceDetails defines the user-visible information about this particular instance.
|
|
|
|
type InstanceDetails struct {
|
2020-10-13 16:45:52 -07:00
|
|
|
Name string `yaml:"name" json:"name"`
|
|
|
|
Title string `yaml:"title" json:"title"`
|
|
|
|
Summary string `yaml:"summary" json:"summary"`
|
|
|
|
Logo logo `yaml:"logo" json:"logo"`
|
|
|
|
Tags []string `yaml:"tags" json:"tags"`
|
|
|
|
SocialHandles []socialHandle `yaml:"socialHandles" json:"socialHandles"`
|
|
|
|
Version string `json:"version"`
|
|
|
|
NSFW bool `yaml:"nsfw" json:"nsfw"`
|
2020-10-14 09:38:48 -07:00
|
|
|
ExtraPageContent string `json:"extraPageContent"`
|
2020-10-01 23:55:38 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
type logo struct {
|
|
|
|
Large string `yaml:"large" json:"large"`
|
|
|
|
Small string `yaml:"small" json:"small"`
|
2020-06-22 20:11:56 -05:00
|
|
|
}
|
|
|
|
|
2020-06-30 17:11:24 -07:00
|
|
|
type socialHandle struct {
|
|
|
|
Platform string `yaml:"platform" json:"platform"`
|
|
|
|
URL string `yaml:"url" json:"url"`
|
|
|
|
}
|
|
|
|
|
2020-06-22 20:11:56 -05:00
|
|
|
type videoSettings struct {
|
2020-07-06 19:45:58 -07:00
|
|
|
ChunkLengthInSeconds int `yaml:"chunkLengthInSeconds"`
|
|
|
|
StreamingKey string `yaml:"streamingKey"`
|
|
|
|
StreamQualities []StreamQuality `yaml:"streamQualities"`
|
2020-10-01 23:55:38 -07:00
|
|
|
HighestQualityStreamIndex int `yaml:"-"`
|
|
|
|
}
|
|
|
|
|
2020-10-07 22:42:14 -07:00
|
|
|
// YP allows registration to the central Owncast YP (Yellow pages) service operating as a directory.
|
|
|
|
type YP struct {
|
|
|
|
Enabled bool `yaml:"enabled" json:"enabled"`
|
|
|
|
InstanceURL string `yaml:"instanceURL" json:"instanceUrl"` // The public URL the directory should link to
|
|
|
|
YPServiceURL string `yaml:"ypServiceURL" json:"-"` // The base URL to the YP API to register with (optional)
|
2020-06-22 20:11:56 -05:00
|
|
|
}
|
|
|
|
|
2020-06-28 15:10:00 -07:00
|
|
|
// StreamQuality defines the specifics of a single HLS stream variant.
|
2020-06-25 17:44:47 -07:00
|
|
|
type StreamQuality struct {
|
|
|
|
// Enable passthrough to copy the video and/or audio directly from the
|
|
|
|
// incoming stream and disable any transcoding. It will ignore any of
|
|
|
|
// the below settings.
|
2020-10-02 00:02:42 -07:00
|
|
|
IsVideoPassthrough bool `yaml:"videoPassthrough" json:"videoPassthrough"`
|
|
|
|
IsAudioPassthrough bool `yaml:"audioPassthrough" json:"audioPassthrough"`
|
2020-06-25 17:44:47 -07:00
|
|
|
|
2020-10-02 00:02:42 -07:00
|
|
|
VideoBitrate int `yaml:"videoBitrate" json:"videoBitrate"`
|
|
|
|
AudioBitrate int `yaml:"audioBitrate" json:"audioBitrate"`
|
2020-06-25 17:44:47 -07:00
|
|
|
|
|
|
|
// Set only one of these in order to keep your current aspect ratio.
|
|
|
|
// Or set neither to not scale the video.
|
2020-10-02 00:02:42 -07:00
|
|
|
ScaledWidth int `yaml:"scaledWidth" json:"scaledWidth,omitempty"`
|
|
|
|
ScaledHeight int `yaml:"scaledHeight" json:"scaledHeight,omitempty"`
|
2020-06-25 17:44:47 -07:00
|
|
|
|
2020-10-02 00:02:42 -07:00
|
|
|
Framerate int `yaml:"framerate" json:"framerate"`
|
|
|
|
EncoderPreset string `yaml:"encoderPreset" json:"encoderPreset"`
|
2020-06-22 20:11:56 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
type files struct {
|
|
|
|
MaxNumberInPlaylist int `yaml:"maxNumberInPlaylist"`
|
|
|
|
}
|
|
|
|
|
2020-10-02 00:02:42 -07:00
|
|
|
//S3 is for configuring the S3 integration
|
|
|
|
type S3 struct {
|
|
|
|
Enabled bool `yaml:"enabled" json:"enabled"`
|
|
|
|
Endpoint string `yaml:"endpoint" json:"endpoint,omitempty"`
|
|
|
|
ServingEndpoint string `yaml:"servingEndpoint" json:"servingEndpoint,omitempty"`
|
|
|
|
AccessKey string `yaml:"accessKey" json:"accessKey,omitempty"`
|
|
|
|
Secret string `yaml:"secret" json:"secret,omitempty"`
|
|
|
|
Bucket string `yaml:"bucket" json:"bucket,omitempty"`
|
|
|
|
Region string `yaml:"region" json:"region,omitempty"`
|
|
|
|
ACL string `yaml:"acl" json:"acl,omitempty"`
|
2020-06-22 20:11:56 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
func (c *config) load(filePath string) error {
|
|
|
|
if !utils.DoesFileExists(filePath) {
|
2020-07-12 13:12:30 -07:00
|
|
|
log.Fatal("ERROR: valid config.yaml is required. Copy config-example.yaml to config.yaml and edit")
|
2020-06-22 20:11:56 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
yamlFile, err := ioutil.ReadFile(filePath)
|
|
|
|
if err != nil {
|
|
|
|
log.Printf("yamlFile.Get err #%v ", err)
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := yaml.Unmarshal(yamlFile, c); err != nil {
|
|
|
|
log.Fatalf("Unmarshal: %v", err)
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2020-07-06 19:45:58 -07:00
|
|
|
c.VideoSettings.HighestQualityStreamIndex = findHighestQuality(c.VideoSettings.StreamQualities)
|
|
|
|
|
2020-10-13 16:45:52 -07:00
|
|
|
// Add custom page content to the instance details.
|
|
|
|
customContentMarkdownData, err := ioutil.ReadFile(ExtraInfoFile)
|
|
|
|
if err == nil {
|
|
|
|
customContentMarkdownString := string(customContentMarkdownData)
|
2020-10-14 09:38:48 -07:00
|
|
|
c.InstanceDetails.ExtraPageContent = utils.RenderSimpleMarkdown(customContentMarkdownString)
|
2020-10-13 16:45:52 -07:00
|
|
|
}
|
|
|
|
|
2020-06-22 20:11:56 -05:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *config) verifySettings() error {
|
2020-07-18 17:46:18 -07:00
|
|
|
if c.VideoSettings.StreamingKey == "" {
|
|
|
|
return errors.New("No stream key set. Please set one in your config file.")
|
|
|
|
}
|
|
|
|
|
2020-06-22 20:11:56 -05:00
|
|
|
if c.S3.Enabled {
|
|
|
|
if c.S3.AccessKey == "" || c.S3.Secret == "" {
|
|
|
|
return errors.New("s3 support requires an access key and secret")
|
|
|
|
}
|
|
|
|
|
|
|
|
if c.S3.Region == "" || c.S3.Endpoint == "" {
|
|
|
|
return errors.New("s3 support requires a region and endpoint")
|
|
|
|
}
|
|
|
|
|
|
|
|
if c.S3.Bucket == "" {
|
|
|
|
return errors.New("s3 support requires a bucket created for storing public video segments")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-01 23:55:38 -07:00
|
|
|
if c.YP.Enabled && c.YP.InstanceURL == "" {
|
|
|
|
return errors.New("YP is enabled but instance url is not set")
|
|
|
|
}
|
|
|
|
|
2020-07-13 14:32:12 -07:00
|
|
|
return nil
|
|
|
|
}
|
2020-06-22 20:11:56 -05:00
|
|
|
|
2020-07-13 14:39:44 -07:00
|
|
|
func (c *config) GetVideoSegmentSecondsLength() int {
|
|
|
|
if c.VideoSettings.ChunkLengthInSeconds != 0 {
|
|
|
|
return c.VideoSettings.ChunkLengthInSeconds
|
|
|
|
}
|
|
|
|
|
2020-07-17 18:27:00 -07:00
|
|
|
return _default.GetVideoSegmentSecondsLength()
|
2020-07-13 14:39:44 -07:00
|
|
|
}
|
|
|
|
|
2020-07-13 14:48:56 -07:00
|
|
|
func (c *config) GetPublicWebServerPort() int {
|
|
|
|
if c.WebServerPort != 0 {
|
|
|
|
return c.WebServerPort
|
|
|
|
}
|
|
|
|
|
2020-07-17 18:27:00 -07:00
|
|
|
return _default.WebServerPort
|
2020-07-13 14:48:56 -07:00
|
|
|
}
|
|
|
|
|
2020-07-13 14:55:21 -07:00
|
|
|
func (c *config) GetMaxNumberOfReferencedSegmentsInPlaylist() int {
|
|
|
|
if c.Files.MaxNumberInPlaylist > 0 {
|
|
|
|
return c.Files.MaxNumberInPlaylist
|
|
|
|
}
|
|
|
|
|
2020-07-17 18:27:00 -07:00
|
|
|
return _default.GetMaxNumberOfReferencedSegmentsInPlaylist()
|
2020-07-13 14:55:21 -07:00
|
|
|
}
|
|
|
|
|
2020-07-17 18:27:00 -07:00
|
|
|
func (c *config) GetFFMpegPath() string {
|
|
|
|
if c.FFMpegPath != "" {
|
|
|
|
return c.FFMpegPath
|
|
|
|
}
|
|
|
|
|
2020-11-09 19:35:01 -08:00
|
|
|
// First look to see if ffmpeg is in the current working directory
|
|
|
|
localCopy := "./ffmpeg"
|
|
|
|
hasLocalCopyError := verifyFFMpegPath(localCopy)
|
|
|
|
if hasLocalCopyError == nil {
|
|
|
|
// No error, so all is good. Use the local copy.
|
|
|
|
return localCopy
|
|
|
|
}
|
|
|
|
|
|
|
|
cmd := exec.Command("which", "ffmpeg")
|
|
|
|
out, err := cmd.CombinedOutput()
|
|
|
|
if err != nil {
|
|
|
|
log.Debugln("Unable to determine path to ffmpeg. Please specify it in the config file.")
|
|
|
|
}
|
|
|
|
|
|
|
|
path := strings.TrimSpace(string(out))
|
|
|
|
|
|
|
|
return path
|
2020-07-17 18:27:00 -07:00
|
|
|
}
|
|
|
|
|
2020-10-01 23:55:38 -07:00
|
|
|
func (c *config) GetYPServiceHost() string {
|
|
|
|
if c.YP.YPServiceURL != "" {
|
|
|
|
return c.YP.YPServiceURL
|
|
|
|
}
|
|
|
|
|
|
|
|
return _default.YP.YPServiceURL
|
|
|
|
}
|
|
|
|
|
2020-10-26 08:55:31 -07:00
|
|
|
func (c *config) GetDataFilePath() string {
|
|
|
|
if c.DatabaseFilePath != "" {
|
|
|
|
return c.DatabaseFilePath
|
|
|
|
}
|
|
|
|
|
|
|
|
return _default.DatabaseFilePath
|
|
|
|
}
|
|
|
|
|
2020-07-17 18:27:00 -07:00
|
|
|
func (c *config) GetVideoStreamQualities() []StreamQuality {
|
|
|
|
if len(c.VideoSettings.StreamQualities) > 0 {
|
|
|
|
return c.VideoSettings.StreamQualities
|
|
|
|
}
|
|
|
|
|
|
|
|
return _default.VideoSettings.StreamQualities
|
2020-07-13 15:13:24 -07:00
|
|
|
}
|
|
|
|
|
2020-08-06 12:19:35 -07:00
|
|
|
// GetFramerate returns the framerate or default
|
|
|
|
func (q *StreamQuality) GetFramerate() int {
|
|
|
|
if q.Framerate > 0 {
|
|
|
|
return q.Framerate
|
|
|
|
}
|
|
|
|
|
|
|
|
return _default.VideoSettings.StreamQualities[0].Framerate
|
|
|
|
}
|
|
|
|
|
2020-10-26 09:12:44 -07:00
|
|
|
// GetEncoderPreset returns the preset or default
|
|
|
|
func (q *StreamQuality) GetEncoderPreset() string {
|
|
|
|
if q.EncoderPreset != "" {
|
|
|
|
return q.EncoderPreset
|
|
|
|
}
|
|
|
|
|
|
|
|
return _default.VideoSettings.StreamQualities[0].EncoderPreset
|
|
|
|
}
|
|
|
|
|
2020-06-22 20:11:56 -05:00
|
|
|
//Load tries to load the configuration file
|
2020-11-03 17:34:25 -08:00
|
|
|
func Load(filePath string, versionInfo string, versionNumber string) error {
|
2020-06-22 20:11:56 -05:00
|
|
|
Config = new(config)
|
2020-07-17 18:27:00 -07:00
|
|
|
_default = getDefaults()
|
2020-06-22 20:11:56 -05:00
|
|
|
|
|
|
|
if err := Config.load(filePath); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2020-06-30 17:48:26 -07:00
|
|
|
Config.VersionInfo = versionInfo
|
2020-11-03 17:34:25 -08:00
|
|
|
Config.VersionNumber = versionNumber
|
2020-06-22 20:11:56 -05:00
|
|
|
return Config.verifySettings()
|
|
|
|
}
|