2020-06-22 20:11:56 -05:00
|
|
|
package config
|
|
|
|
|
|
|
|
import (
|
|
|
|
"errors"
|
|
|
|
"io/ioutil"
|
|
|
|
|
2020-07-06 19:45:58 -07:00
|
|
|
"github.com/gabek/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-07-15 17:20:47 -07:00
|
|
|
ChatDatabaseFilePath string `yaml:"chatDatabaseFile"`
|
|
|
|
EnableDebugFeatures bool `yaml:"-"`
|
|
|
|
FFMpegPath string `yaml:"ffmpegPath"`
|
|
|
|
Files files `yaml:"files"`
|
|
|
|
InstanceDetails InstanceDetails `yaml:"instanceDetails"`
|
2020-10-02 00:02:42 -07:00
|
|
|
S3 S3 `yaml:"s3"`
|
2020-10-03 14:35:03 -07:00
|
|
|
VersionInfo string `yaml:"-"` // For storing the version/build number
|
2020-07-15 17:20:47 -07:00
|
|
|
VideoSettings videoSettings `yaml:"videoSettings"`
|
|
|
|
WebServerPort int `yaml:"webServerPort"`
|
2020-10-01 23:55:38 -07:00
|
|
|
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-01 23:55:38 -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"`
|
|
|
|
ExtraInfoFile string `yaml:"extraUserInfoFileName" json:"extraUserInfoFileName"`
|
|
|
|
Version string `json:"version"`
|
|
|
|
NSFW bool `yaml:"nsfw" json:"nsfw"`
|
|
|
|
}
|
|
|
|
|
|
|
|
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"`
|
|
|
|
OfflineContent string `yaml:"offlineContent"`
|
2020-10-01 23:55:38 -07:00
|
|
|
HighestQualityStreamIndex int `yaml:"-"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// Registration to the central Owncast YP (Yellow pages) service operating as a directory.
|
|
|
|
type yp struct {
|
|
|
|
Enabled bool `yaml:"enabled"`
|
|
|
|
InstanceURL string `yaml:"instanceURL"` // The public URL the directory should link to
|
|
|
|
YPServiceURL string `yaml:"ypServiceURL"` // 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-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-13 15:13:24 -07:00
|
|
|
func (c *config) GetOfflineContentPath() string {
|
|
|
|
if c.VideoSettings.OfflineContent != "" {
|
|
|
|
return c.VideoSettings.OfflineContent
|
|
|
|
}
|
|
|
|
|
|
|
|
// This is relative to the webroot, not the project root.
|
2020-07-17 18:27:00 -07:00
|
|
|
return _default.VideoSettings.OfflineContent
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *config) GetFFMpegPath() string {
|
|
|
|
if c.FFMpegPath != "" {
|
|
|
|
return c.FFMpegPath
|
|
|
|
}
|
|
|
|
|
|
|
|
return _default.FFMpegPath
|
|
|
|
}
|
|
|
|
|
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-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-06-22 20:11:56 -05:00
|
|
|
//Load tries to load the configuration file
|
2020-06-30 17:48:26 -07:00
|
|
|
func Load(filePath string, versionInfo 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-07-13 15:10:16 -07:00
|
|
|
// Defaults
|
|
|
|
|
|
|
|
// This is relative to the webroot, not the project root.
|
2020-07-17 18:27:00 -07:00
|
|
|
// Has to be set here instead of pulled from a getter
|
|
|
|
// since it's serialized to JSON.
|
2020-07-13 15:10:16 -07:00
|
|
|
if Config.InstanceDetails.ExtraInfoFile == "" {
|
2020-07-17 18:27:00 -07:00
|
|
|
Config.InstanceDetails.ExtraInfoFile = _default.InstanceDetails.ExtraInfoFile
|
2020-07-13 15:10:16 -07:00
|
|
|
}
|
|
|
|
|
2020-06-22 20:11:56 -05:00
|
|
|
return Config.verifySettings()
|
|
|
|
}
|