emit /recevie samples on demand.

This commit is contained in:
Miroslav Šedivý 2022-01-22 17:58:05 +01:00
parent 26ab7fa12d
commit c2cf9277dc
2 changed files with 67 additions and 71 deletions

View File

@ -25,10 +25,8 @@ type ScreencastManagerCtx struct {
pipelineStr string pipelineStr string
pipelineMu sync.Mutex pipelineMu sync.Mutex
image types.Sample image types.Sample
sample chan types.Sample tickerStop chan struct{}
sampleStop chan struct{}
sampleUpdate chan struct{}
enabled bool enabled bool
started bool started bool
@ -42,18 +40,16 @@ func screencastNew(enabled bool, pipelineStr string) *ScreencastManagerCtx {
Logger() Logger()
manager := &ScreencastManagerCtx{ manager := &ScreencastManagerCtx{
logger: logger, logger: logger,
pipelineStr: pipelineStr, pipelineStr: pipelineStr,
sampleStop: make(chan struct{}), tickerStop: make(chan struct{}),
sampleUpdate: make(chan struct{}), enabled: enabled,
enabled: enabled, started: false,
started: false,
} }
manager.wg.Add(1) manager.wg.Add(1)
go func() { go func() {
manager.logger.Debug().Msg("started emitting samples")
defer manager.wg.Done() defer manager.wg.Done()
ticker := time.NewTicker(screencastTimeout) ticker := time.NewTicker(screencastTimeout)
@ -61,15 +57,10 @@ func screencastNew(enabled bool, pipelineStr string) *ScreencastManagerCtx {
for { for {
select { select {
case <-manager.sampleStop: case <-manager.tickerStop:
manager.logger.Debug().Msg("stopped emitting samples")
return return
case <-manager.sampleUpdate:
manager.logger.Debug().Msg("update emitting samples")
case sample := <-manager.sample:
manager.image = sample
case <-ticker.C: case <-ticker.C:
if manager.started && !atomic.CompareAndSwapInt32(&manager.expired, 0, 1) { if manager.Started() && !atomic.CompareAndSwapInt32(&manager.expired, 0, 1) {
manager.stop() manager.stop()
} }
} }
@ -84,7 +75,7 @@ func (manager *ScreencastManagerCtx) shutdown() {
manager.destroyPipeline() manager.destroyPipeline()
close(manager.sampleStop) close(manager.tickerStop)
manager.wg.Wait() manager.wg.Wait()
} }
@ -105,22 +96,13 @@ func (manager *ScreencastManagerCtx) Started() bool {
func (manager *ScreencastManagerCtx) Image() ([]byte, error) { func (manager *ScreencastManagerCtx) Image() ([]byte, error) {
atomic.StoreInt32(&manager.expired, 0) atomic.StoreInt32(&manager.expired, 0)
if !manager.started { err := manager.start()
err := manager.start() if err != nil && !errors.Is(err, types.ErrCapturePipelineAlreadyExists) {
if err != nil { return nil, err
return nil, err
}
select {
case sample := <-manager.sample:
return sample.Data, nil
case <-time.After(1 * time.Second):
return nil, errors.New("timeouted")
}
} }
if manager.image.Data == nil { if manager.image.Data == nil {
return nil, errors.New("image sample not found") return nil, errors.New("image data not found")
} }
return manager.image.Data, nil return manager.image.Data, nil
@ -173,8 +155,34 @@ func (manager *ScreencastManagerCtx) createPipeline() error {
manager.pipeline.AttachAppsink("appsink") manager.pipeline.AttachAppsink("appsink")
manager.pipeline.Play() manager.pipeline.Play()
manager.sample = manager.pipeline.Sample // get first image
manager.sampleUpdate <- struct{}{} var ok bool
select {
case manager.image, ok = <-manager.pipeline.Sample:
if !ok {
return errors.New("unable to get first image")
}
case <-time.After(1 * time.Second):
return errors.New("timeouted while waiting for first image")
}
manager.wg.Add(1)
go func() {
manager.logger.Debug().Msg("started receiving images")
defer manager.wg.Done()
for {
image, ok := <-manager.pipeline.Sample
if !ok {
manager.logger.Debug().Msg("stopped receiving images")
return
}
manager.image = image
}
}()
return nil return nil
} }

View File

@ -25,10 +25,6 @@ type StreamSinkManagerCtx struct {
pipelineMu sync.Mutex pipelineMu sync.Mutex
pipelineStr func() string pipelineStr func() string
sample chan types.Sample
sampleStop chan struct{}
sampleUpdate chan struct{}
listeners map[uintptr]*func(sample types.Sample) listeners map[uintptr]*func(sample types.Sample)
listenersMu sync.Mutex listenersMu sync.Mutex
} }
@ -40,37 +36,12 @@ func streamSinkNew(codec codec.RTPCodec, pipelineStr func() string, video_id str
Str("video_id", video_id).Logger() Str("video_id", video_id).Logger()
manager := &StreamSinkManagerCtx{ manager := &StreamSinkManagerCtx{
logger: logger, logger: logger,
codec: codec, codec: codec,
pipelineStr: pipelineStr, pipelineStr: pipelineStr,
sampleStop: make(chan struct{}), listeners: map[uintptr]*func(sample types.Sample){},
sampleUpdate: make(chan struct{}),
listeners: map[uintptr]*func(sample types.Sample){},
} }
manager.wg.Add(1)
go func() {
manager.logger.Debug().Msg("started emitting samples")
defer manager.wg.Done()
for {
select {
case <-manager.sampleStop:
manager.logger.Debug().Msg("stopped emitting samples")
return
case <-manager.sampleUpdate:
manager.logger.Debug().Msg("update emitting samples")
case sample := <-manager.sample:
manager.listenersMu.Lock()
for _, emit := range manager.listeners {
(*emit)(sample)
}
manager.listenersMu.Unlock()
}
}
}()
return manager return manager
} }
@ -84,8 +55,6 @@ func (manager *StreamSinkManagerCtx) shutdown() {
manager.listenersMu.Unlock() manager.listenersMu.Unlock()
manager.destroyPipeline() manager.destroyPipeline()
close(manager.sampleStop)
manager.wg.Wait() manager.wg.Wait()
} }
@ -249,8 +218,27 @@ func (manager *StreamSinkManagerCtx) createPipeline() error {
manager.pipeline.AttachAppsink("appsink") manager.pipeline.AttachAppsink("appsink")
manager.pipeline.Play() manager.pipeline.Play()
manager.sample = manager.pipeline.Sample manager.wg.Add(1)
manager.sampleUpdate <- struct{}{}
go func() {
manager.logger.Debug().Msg("started emitting samples")
defer manager.wg.Done()
for {
sample, ok := <-manager.pipeline.Sample
if !ok {
manager.logger.Debug().Msg("stopped emitting samples")
return
}
manager.listenersMu.Lock()
for _, emit := range manager.listeners {
(*emit)(sample)
}
manager.listenersMu.Unlock()
}
}()
return nil return nil
} }