2021-02-18 23:05:52 -08:00
|
|
|
package models
|
|
|
|
|
2021-03-11 12:51:43 -08:00
|
|
|
import (
|
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
|
|
|
"math"
|
|
|
|
)
|
2021-02-18 23:05:52 -08:00
|
|
|
|
|
|
|
// StreamOutputVariant defines the output specifics of a single HLS stream variant.
|
|
|
|
type StreamOutputVariant struct {
|
2021-03-11 12:51:43 -08:00
|
|
|
// Name is an optional human-readable label for this stream output.
|
|
|
|
Name string `json:"name"`
|
|
|
|
|
2021-02-18 23:05:52 -08:00
|
|
|
// 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.
|
|
|
|
IsVideoPassthrough bool `yaml:"videoPassthrough" json:"videoPassthrough"`
|
|
|
|
IsAudioPassthrough bool `yaml:"audioPassthrough" json:"audioPassthrough"`
|
|
|
|
|
|
|
|
VideoBitrate int `yaml:"videoBitrate" json:"videoBitrate"`
|
|
|
|
AudioBitrate int `yaml:"audioBitrate" json:"audioBitrate"`
|
|
|
|
|
|
|
|
// Set only one of these in order to keep your current aspect ratio.
|
|
|
|
// Or set neither to not scale the video.
|
|
|
|
ScaledWidth int `yaml:"scaledWidth" json:"scaledWidth,omitempty"`
|
|
|
|
ScaledHeight int `yaml:"scaledHeight" json:"scaledHeight,omitempty"`
|
|
|
|
|
|
|
|
Framerate int `yaml:"framerate" json:"framerate"`
|
|
|
|
EncoderPreset string `yaml:"encoderPreset" json:"encoderPreset"` // Remove after migration is no longer used
|
|
|
|
// CPUUsageLevel represents a codec preset to configure CPU usage.
|
|
|
|
CPUUsageLevel int `json:"cpuUsageLevel"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetFramerate returns the framerate or default.
|
|
|
|
func (q *StreamOutputVariant) GetFramerate() int {
|
|
|
|
if q.IsVideoPassthrough {
|
|
|
|
return 0
|
|
|
|
}
|
|
|
|
|
|
|
|
if q.Framerate > 0 {
|
|
|
|
return q.Framerate
|
|
|
|
}
|
|
|
|
|
|
|
|
return 24
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetEncoderPreset returns the preset or default.
|
|
|
|
func (q *StreamOutputVariant) GetEncoderPreset() string {
|
|
|
|
if q.IsVideoPassthrough {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
|
|
|
if q.EncoderPreset != "" {
|
|
|
|
return q.EncoderPreset
|
|
|
|
}
|
|
|
|
|
|
|
|
return "veryfast"
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetCPUUsageLevel will return the libx264 codec encoder preset that maps to a level.
|
|
|
|
func (q *StreamOutputVariant) GetCPUUsageLevel() int {
|
|
|
|
presetMapping := map[string]int{
|
|
|
|
"ultrafast": 1,
|
|
|
|
"superfast": 2,
|
|
|
|
"veryfast": 3,
|
|
|
|
"faster": 4,
|
|
|
|
"fast": 5,
|
|
|
|
}
|
|
|
|
|
|
|
|
return presetMapping[q.GetEncoderPreset()]
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetIsAudioPassthrough will return if this variant audio is passthrough.
|
|
|
|
func (q *StreamOutputVariant) GetIsAudioPassthrough() bool {
|
|
|
|
if q.IsAudioPassthrough {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
if q.AudioBitrate == 0 {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2021-03-11 12:51:43 -08:00
|
|
|
// GetName will return the human readable name for this stream output.
|
|
|
|
func (q *StreamOutputVariant) GetName() string {
|
|
|
|
bitrate := getBitrateString(q.VideoBitrate)
|
|
|
|
|
|
|
|
if q.Name != "" {
|
|
|
|
return q.Name
|
|
|
|
} else if q.IsVideoPassthrough {
|
|
|
|
return "Source"
|
|
|
|
} else if q.ScaledHeight == 720 && q.ScaledWidth == 1080 {
|
|
|
|
return fmt.Sprintf("720p @%s", bitrate)
|
|
|
|
} else if q.ScaledHeight == 1080 && q.ScaledWidth == 1920 {
|
|
|
|
return fmt.Sprintf("1080p @%s", bitrate)
|
|
|
|
} else if q.ScaledHeight != 0 {
|
|
|
|
return fmt.Sprintf("%dh", q.ScaledHeight)
|
|
|
|
} else if q.ScaledWidth != 0 {
|
|
|
|
return fmt.Sprintf("%dw", q.ScaledWidth)
|
|
|
|
} else {
|
|
|
|
return fmt.Sprintf("%s@%dfps", bitrate, q.Framerate)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func getBitrateString(bitrate int) string {
|
|
|
|
if bitrate == 0 {
|
|
|
|
return ""
|
|
|
|
} else if bitrate < 1000 {
|
|
|
|
return fmt.Sprintf("%dKbps", bitrate)
|
|
|
|
} else if bitrate >= 1000 {
|
|
|
|
if math.Mod(float64(bitrate), 1000) == 0 {
|
|
|
|
return fmt.Sprintf("%dMbps", bitrate/1000.0)
|
|
|
|
} else {
|
|
|
|
return fmt.Sprintf("%.1fMbps", float32(bitrate)/1000.0)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
2021-02-18 23:05:52 -08:00
|
|
|
// MarshalJSON is a custom JSON marshal function for video stream qualities.
|
|
|
|
func (q *StreamOutputVariant) MarshalJSON() ([]byte, error) {
|
|
|
|
type Alias StreamOutputVariant
|
|
|
|
return json.Marshal(&struct {
|
|
|
|
Framerate int `json:"framerate"`
|
|
|
|
*Alias
|
|
|
|
}{
|
|
|
|
Framerate: q.GetFramerate(),
|
|
|
|
Alias: (*Alias)(q),
|
|
|
|
})
|
|
|
|
}
|