neko/internal/config/capture.go

231 lines
8.4 KiB
Go
Raw Normal View History

package config
import (
2021-03-11 17:55:13 +01:00
"os"
2021-12-01 22:34:36 +01:00
"github.com/pion/webrtc/v3"
2021-03-16 15:24:58 +01:00
"github.com/rs/zerolog/log"
"github.com/spf13/cobra"
"github.com/spf13/viper"
2021-02-01 23:50:18 +01:00
2022-03-20 11:43:00 +01:00
"gitlab.com/demodesk/neko/server/pkg/types"
"gitlab.com/demodesk/neko/server/pkg/types/codec"
"gitlab.com/demodesk/neko/server/pkg/utils"
)
2020-11-01 16:09:48 +01:00
type Capture struct {
2021-03-11 17:55:13 +01:00
Display string
VideoCodec codec.RTPCodec
VideoIDs []string
VideoPipelines map[string]types.VideoConfig
2021-03-29 00:58:51 +02:00
2021-03-11 17:55:13 +01:00
AudioDevice string
2021-02-14 14:40:17 +01:00
AudioCodec codec.RTPCodec
AudioPipeline string
2021-02-05 15:10:41 +01:00
2021-03-18 13:17:10 +01:00
BroadcastAudioBitrate int
BroadcastVideoBitrate int
BroadcastPreset string
BroadcastPipeline string
2021-02-05 15:10:41 +01:00
2021-03-16 15:24:58 +01:00
ScreencastEnabled bool
2021-02-05 15:10:41 +01:00
ScreencastRate string
ScreencastQuality string
2021-01-22 18:13:32 +01:00
ScreencastPipeline string
2021-12-09 23:22:24 +01:00
WebcamEnabled bool
WebcamDevice string
2022-01-08 23:53:45 +01:00
WebcamWidth int
WebcamHeight int
2021-12-09 23:22:24 +01:00
MicrophoneEnabled bool
MicrophoneDevice string
}
2020-11-01 16:09:48 +01:00
func (Capture) Init(cmd *cobra.Command) error {
2021-03-11 17:55:13 +01:00
// audio
2021-12-09 23:22:24 +01:00
cmd.PersistentFlags().String("capture.audio.device", "audio_output.monitor", "pulseaudio device to capture")
2021-03-16 15:24:58 +01:00
if err := viper.BindPFlag("capture.audio.device", cmd.PersistentFlags().Lookup("capture.audio.device")); err != nil {
return err
}
2021-03-16 15:24:58 +01:00
cmd.PersistentFlags().String("capture.audio.codec", "opus", "audio codec to be used")
if err := viper.BindPFlag("capture.audio.codec", cmd.PersistentFlags().Lookup("capture.audio.codec")); err != nil {
return err
}
2021-03-16 15:24:58 +01:00
cmd.PersistentFlags().String("capture.audio.pipeline", "", "gstreamer pipeline used for audio streaming")
if err := viper.BindPFlag("capture.audio.pipeline", cmd.PersistentFlags().Lookup("capture.audio.pipeline")); err != nil {
return err
}
// videos
cmd.PersistentFlags().String("capture.video.codec", "vp8", "video codec to be used")
if err := viper.BindPFlag("capture.video.codec", cmd.PersistentFlags().Lookup("capture.video.codec")); err != nil {
return err
}
cmd.PersistentFlags().StringSlice("capture.video.ids", []string{}, "ordered list of video ids")
if err := viper.BindPFlag("capture.video.ids", cmd.PersistentFlags().Lookup("capture.video.ids")); err != nil {
return err
}
cmd.PersistentFlags().String("capture.video.pipelines", "", "pipelines config in JSON used for video streaming")
if err := viper.BindPFlag("capture.video.pipelines", cmd.PersistentFlags().Lookup("capture.video.pipelines")); err != nil {
return err
}
2020-11-01 16:09:48 +01:00
// broadcast
2021-12-09 22:58:12 +01:00
cmd.PersistentFlags().Int("capture.broadcast.audio_bitrate", 128, "broadcast audio bitrate in KB/s")
if err := viper.BindPFlag("capture.broadcast.audio_bitrate", cmd.PersistentFlags().Lookup("capture.broadcast.audio_bitrate")); err != nil {
2021-03-18 13:17:10 +01:00
return err
}
2021-12-09 22:58:12 +01:00
cmd.PersistentFlags().Int("capture.broadcast.video_bitrate", 4096, "broadcast video bitrate in KB/s")
if err := viper.BindPFlag("capture.broadcast.video_bitrate", cmd.PersistentFlags().Lookup("capture.broadcast.video_bitrate")); err != nil {
2021-03-18 13:17:10 +01:00
return err
}
2021-12-09 22:58:12 +01:00
cmd.PersistentFlags().String("capture.broadcast.preset", "veryfast", "broadcast speed preset for h264 encoding")
if err := viper.BindPFlag("capture.broadcast.preset", cmd.PersistentFlags().Lookup("capture.broadcast.preset")); err != nil {
2021-03-18 13:17:10 +01:00
return err
}
2021-03-16 15:24:58 +01:00
cmd.PersistentFlags().String("capture.broadcast.pipeline", "", "gstreamer pipeline used for broadcasting")
if err := viper.BindPFlag("capture.broadcast.pipeline", cmd.PersistentFlags().Lookup("capture.broadcast.pipeline")); err != nil {
2020-11-01 16:09:48 +01:00
return err
}
2021-01-22 18:13:32 +01:00
// screencast
2021-03-16 15:24:58 +01:00
cmd.PersistentFlags().Bool("capture.screencast.enabled", false, "enable screencast")
if err := viper.BindPFlag("capture.screencast.enabled", cmd.PersistentFlags().Lookup("capture.screencast.enabled")); err != nil {
2021-01-22 18:13:32 +01:00
return err
}
2021-03-16 15:24:58 +01:00
cmd.PersistentFlags().String("capture.screencast.rate", "10/1", "screencast frame rate")
if err := viper.BindPFlag("capture.screencast.rate", cmd.PersistentFlags().Lookup("capture.screencast.rate")); err != nil {
2021-01-22 18:44:27 +01:00
return err
}
2021-03-16 15:24:58 +01:00
cmd.PersistentFlags().String("capture.screencast.quality", "60", "screencast JPEG quality")
if err := viper.BindPFlag("capture.screencast.quality", cmd.PersistentFlags().Lookup("capture.screencast.quality")); err != nil {
2021-01-22 18:44:27 +01:00
return err
}
2021-03-16 15:24:58 +01:00
cmd.PersistentFlags().String("capture.screencast.pipeline", "", "gstreamer pipeline used for screencasting")
if err := viper.BindPFlag("capture.screencast.pipeline", cmd.PersistentFlags().Lookup("capture.screencast.pipeline")); err != nil {
2021-01-22 18:13:32 +01:00
return err
}
2021-12-09 23:22:24 +01:00
// webcam
cmd.PersistentFlags().Bool("capture.webcam.enabled", false, "enable webcam stream")
if err := viper.BindPFlag("capture.webcam.enabled", cmd.PersistentFlags().Lookup("capture.webcam.enabled")); err != nil {
return err
}
2022-01-06 22:35:12 +01:00
// sudo apt install v4l2loopback-dkms v4l2loopback-utils
// sudo apt-get install linux-headers-`uname -r` linux-modules-extra-`uname -r`
// sudo modprobe v4l2loopback exclusive_caps=1
2021-12-09 23:22:24 +01:00
cmd.PersistentFlags().String("capture.webcam.device", "/dev/video0", "v4l2sink device used for webcam")
if err := viper.BindPFlag("capture.webcam.device", cmd.PersistentFlags().Lookup("capture.webcam.device")); err != nil {
return err
}
2022-01-08 23:53:45 +01:00
cmd.PersistentFlags().Int("capture.webcam.width", 1280, "webcam stream width")
if err := viper.BindPFlag("capture.webcam.width", cmd.PersistentFlags().Lookup("capture.webcam.width")); err != nil {
return err
}
cmd.PersistentFlags().Int("capture.webcam.height", 720, "webcam stream height")
if err := viper.BindPFlag("capture.webcam.height", cmd.PersistentFlags().Lookup("capture.webcam.height")); err != nil {
return err
}
2021-12-09 23:22:24 +01:00
// microphone
cmd.PersistentFlags().Bool("capture.microphone.enabled", true, "enable microphone stream")
if err := viper.BindPFlag("capture.microphone.enabled", cmd.PersistentFlags().Lookup("capture.microphone.enabled")); err != nil {
return err
}
cmd.PersistentFlags().String("capture.microphone.device", "audio_input", "pulseaudio device used for microphone")
if err := viper.BindPFlag("capture.microphone.device", cmd.PersistentFlags().Lookup("capture.microphone.device")); err != nil {
return err
}
return nil
}
2020-11-01 16:09:48 +01:00
func (s *Capture) Set() {
2021-12-01 22:34:36 +01:00
var ok bool
2021-03-11 17:55:13 +01:00
// Display is provided by env variable
s.Display = os.Getenv("DISPLAY")
2021-03-29 00:58:51 +02:00
// video
videoCodec := viper.GetString("capture.video.codec")
2021-12-01 22:34:36 +01:00
s.VideoCodec, ok = codec.ParseStr(videoCodec)
if !ok || s.VideoCodec.Type != webrtc.RTPCodecTypeVideo {
log.Warn().Str("codec", videoCodec).Msgf("unknown video codec, using Vp8")
s.VideoCodec = codec.VP8()
}
s.VideoIDs = viper.GetStringSlice("capture.video.ids")
if err := viper.UnmarshalKey("capture.video.pipelines", &s.VideoPipelines, viper.DecodeHook(
utils.JsonStringAutoDecode(s.VideoPipelines),
2021-03-29 00:58:51 +02:00
)); err != nil {
log.Warn().Err(err).Msgf("unable to parse video pipelines")
}
// default video
if len(s.VideoPipelines) == 0 {
log.Warn().Msgf("no video pipelines specified, using defaults")
2021-03-30 00:37:06 +02:00
s.VideoCodec = codec.VP8()
s.VideoPipelines = map[string]types.VideoConfig{
2021-12-01 22:34:36 +01:00
"main": {
2021-03-30 00:37:06 +02:00
GstPipeline: "ximagesrc display-name={display} show-pointer=false use-damage=false " +
"! video/x-raw " +
"! videoconvert " +
"! queue " +
"! vp8enc end-usage=cbr cpu-used=4 threads=4 deadline=1 keyframe-max-dist=25 " +
"! appsink name=appsink",
},
}
s.VideoIDs = []string{"main"}
2021-03-29 00:58:51 +02:00
}
// audio
2021-03-16 15:24:58 +01:00
s.AudioDevice = viper.GetString("capture.audio.device")
s.AudioPipeline = viper.GetString("capture.audio.pipeline")
2021-03-11 17:55:13 +01:00
2021-03-16 15:24:58 +01:00
audioCodec := viper.GetString("capture.audio.codec")
2021-12-01 22:34:36 +01:00
s.AudioCodec, ok = codec.ParseStr(audioCodec)
if !ok || s.AudioCodec.Type != webrtc.RTPCodecTypeAudio {
2021-03-16 15:24:58 +01:00
log.Warn().Str("codec", audioCodec).Msgf("unknown audio codec, using Opus")
2021-03-11 17:55:13 +01:00
s.AudioCodec = codec.Opus()
}
2021-03-29 00:58:51 +02:00
// broadcast
2021-03-18 13:17:10 +01:00
s.BroadcastAudioBitrate = viper.GetInt("capture.broadcast.audio_bitrate")
s.BroadcastVideoBitrate = viper.GetInt("capture.broadcast.video_bitrate")
s.BroadcastPreset = viper.GetString("capture.broadcast.preset")
2021-03-17 11:19:35 +01:00
s.BroadcastPipeline = viper.GetString("capture.broadcast.pipeline")
2021-02-05 15:10:41 +01:00
2021-03-29 00:58:51 +02:00
// screencast
2021-03-16 15:24:58 +01:00
s.ScreencastEnabled = viper.GetBool("capture.screencast.enabled")
s.ScreencastRate = viper.GetString("capture.screencast.rate")
s.ScreencastQuality = viper.GetString("capture.screencast.quality")
s.ScreencastPipeline = viper.GetString("capture.screencast.pipeline")
2021-12-09 23:22:24 +01:00
// webcam
s.WebcamEnabled = viper.GetBool("capture.webcam.enabled")
s.WebcamDevice = viper.GetString("capture.webcam.device")
2022-01-08 23:53:45 +01:00
s.WebcamWidth = viper.GetInt("capture.webcam.width")
s.WebcamHeight = viper.GetInt("capture.webcam.height")
2021-12-09 23:22:24 +01:00
// microphone
s.MicrophoneEnabled = viper.GetBool("capture.microphone.enabled")
s.MicrophoneDevice = viper.GetString("capture.microphone.device")
}