2022-09-12 22:12:47 +02:00
|
|
|
package capture
|
2020-04-05 22:34:51 +00:00
|
|
|
|
|
|
|
import (
|
2022-09-17 12:43:17 +02:00
|
|
|
"errors"
|
|
|
|
|
|
|
|
"github.com/rs/zerolog"
|
|
|
|
"github.com/rs/zerolog/log"
|
2020-04-05 22:34:51 +00:00
|
|
|
|
2022-09-12 22:18:18 +02:00
|
|
|
"m1k1o/neko/internal/config"
|
2021-10-05 22:38:24 +02:00
|
|
|
"m1k1o/neko/internal/types"
|
2020-04-05 22:34:51 +00:00
|
|
|
)
|
|
|
|
|
2022-09-12 22:12:47 +02:00
|
|
|
type CaptureManagerCtx struct {
|
2022-09-17 12:43:17 +02:00
|
|
|
logger zerolog.Logger
|
|
|
|
desktop types.DesktopManager
|
2020-04-05 22:34:51 +00:00
|
|
|
|
2022-09-17 12:43:17 +02:00
|
|
|
// sinks
|
|
|
|
broadcast *BroacastManagerCtx
|
|
|
|
audio *StreamSinkManagerCtx
|
|
|
|
video *StreamSinkManagerCtx
|
2020-04-05 22:34:51 +00:00
|
|
|
}
|
|
|
|
|
2022-09-17 12:43:17 +02:00
|
|
|
func New(desktop types.DesktopManager, config *config.Capture) *CaptureManagerCtx {
|
|
|
|
logger := log.With().Str("module", "capture").Logger()
|
2023-01-29 17:51:39 +01:00
|
|
|
|
2023-01-29 17:40:07 +01:00
|
|
|
return &CaptureManagerCtx{
|
2022-09-17 12:43:17 +02:00
|
|
|
logger: logger,
|
|
|
|
desktop: desktop,
|
|
|
|
|
|
|
|
// sinks
|
2022-09-21 18:58:28 +02:00
|
|
|
broadcast: broadcastNew(func(url string) (string, error) {
|
2022-09-17 12:43:17 +02:00
|
|
|
return NewBroadcastPipeline(config.AudioDevice, config.Display, config.BroadcastPipeline, url)
|
2022-09-21 18:59:38 +02:00
|
|
|
}, config.BroadcastUrl),
|
2022-09-21 18:58:28 +02:00
|
|
|
audio: streamSinkNew(config.AudioCodec, func() (string, error) {
|
2022-09-17 12:43:17 +02:00
|
|
|
return NewAudioPipeline(config.AudioCodec, config.AudioDevice, config.AudioPipeline, config.AudioBitrate)
|
|
|
|
}, "audio"),
|
2022-09-21 18:58:28 +02:00
|
|
|
video: streamSinkNew(config.VideoCodec, func() (string, error) {
|
2023-01-29 17:40:07 +01:00
|
|
|
// use screen fps as default
|
|
|
|
fps := desktop.GetScreenSize().Rate
|
|
|
|
// if max fps is set, cap it to that value
|
|
|
|
if config.VideoMaxFPS > 0 && config.VideoMaxFPS < fps {
|
|
|
|
fps = config.VideoMaxFPS
|
|
|
|
}
|
|
|
|
return NewVideoPipeline(config.VideoCodec, config.Display, config.VideoPipeline, fps, config.VideoBitrate, config.VideoHWEnc)
|
2022-09-24 14:21:32 +02:00
|
|
|
}, "video"),
|
2022-09-17 12:43:17 +02:00
|
|
|
}
|
2020-04-05 22:34:51 +00:00
|
|
|
}
|
|
|
|
|
2022-09-12 22:12:47 +02:00
|
|
|
func (manager *CaptureManagerCtx) Start() {
|
2022-09-17 12:43:17 +02:00
|
|
|
if manager.broadcast.Started() {
|
|
|
|
if err := manager.broadcast.createPipeline(); err != nil {
|
|
|
|
manager.logger.Panic().Err(err).Msg("unable to create broadcast pipeline")
|
|
|
|
}
|
2021-08-15 15:37:27 +02:00
|
|
|
}
|
2020-04-05 22:34:51 +00:00
|
|
|
|
2023-01-21 23:43:04 +01:00
|
|
|
go func() {
|
|
|
|
for {
|
2023-01-29 18:00:56 +01:00
|
|
|
before, ok := <-manager.desktop.GetScreenSizeChangeChannel()
|
2023-01-29 13:28:03 +01:00
|
|
|
if !ok {
|
2023-01-29 18:00:56 +01:00
|
|
|
manager.logger.Info().Msg("screen size change channel was closed")
|
2023-01-29 13:28:03 +01:00
|
|
|
return
|
|
|
|
}
|
2023-01-28 22:08:36 +01:00
|
|
|
|
2023-01-29 18:00:56 +01:00
|
|
|
if before {
|
|
|
|
// before screen size change, we need to destroy all pipelines
|
2020-09-24 08:09:02 +02:00
|
|
|
|
2023-01-29 18:00:56 +01:00
|
|
|
if manager.video.Started() {
|
|
|
|
manager.video.destroyPipeline()
|
|
|
|
}
|
2023-01-28 22:08:36 +01:00
|
|
|
|
2023-01-29 18:00:56 +01:00
|
|
|
if manager.broadcast.Started() {
|
|
|
|
manager.broadcast.destroyPipeline()
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// after screen size change, we need to recreate all pipelines
|
|
|
|
|
|
|
|
if manager.video.Started() {
|
|
|
|
err := manager.video.createPipeline()
|
|
|
|
if err != nil && !errors.Is(err, types.ErrCapturePipelineAlreadyExists) {
|
|
|
|
manager.logger.Panic().Err(err).Msg("unable to recreate video pipeline")
|
|
|
|
}
|
2023-01-21 23:43:04 +01:00
|
|
|
}
|
2020-04-05 22:34:51 +00:00
|
|
|
|
2023-01-29 18:00:56 +01:00
|
|
|
if manager.broadcast.Started() {
|
|
|
|
err := manager.broadcast.createPipeline()
|
|
|
|
if err != nil && !errors.Is(err, types.ErrCapturePipelineAlreadyExists) {
|
|
|
|
manager.logger.Panic().Err(err).Msg("unable to recreate broadcast pipeline")
|
|
|
|
}
|
2023-01-21 23:43:04 +01:00
|
|
|
}
|
2022-09-17 12:43:17 +02:00
|
|
|
}
|
|
|
|
}
|
2023-01-21 23:43:04 +01:00
|
|
|
}()
|
2020-04-05 22:34:51 +00:00
|
|
|
}
|
|
|
|
|
2022-09-17 12:43:17 +02:00
|
|
|
func (manager *CaptureManagerCtx) Shutdown() error {
|
|
|
|
manager.logger.Info().Msgf("shutdown")
|
2020-04-05 22:34:51 +00:00
|
|
|
|
2022-09-17 12:43:17 +02:00
|
|
|
manager.broadcast.shutdown()
|
2020-04-05 22:34:51 +00:00
|
|
|
|
2022-09-17 12:43:17 +02:00
|
|
|
manager.audio.shutdown()
|
|
|
|
manager.video.shutdown()
|
2020-04-05 23:07:25 +00:00
|
|
|
|
2022-09-17 12:43:17 +02:00
|
|
|
return nil
|
2020-04-05 23:07:25 +00:00
|
|
|
}
|
|
|
|
|
2022-09-17 12:43:17 +02:00
|
|
|
func (manager *CaptureManagerCtx) Broadcast() types.BroadcastManager {
|
|
|
|
return manager.broadcast
|
2020-04-05 22:34:51 +00:00
|
|
|
}
|
|
|
|
|
2022-09-17 12:43:17 +02:00
|
|
|
func (manager *CaptureManagerCtx) Audio() types.StreamSinkManager {
|
|
|
|
return manager.audio
|
|
|
|
}
|
2020-09-24 08:09:02 +02:00
|
|
|
|
2022-09-17 12:43:17 +02:00
|
|
|
func (manager *CaptureManagerCtx) Video() types.StreamSinkManager {
|
|
|
|
return manager.video
|
2020-04-05 22:34:51 +00:00
|
|
|
}
|