mirror of
https://github.com/m1k1o/neko.git
synced 2024-07-24 14:40:50 +12:00
emit /recevie samples on demand.
This commit is contained in:
parent
26ab7fa12d
commit
c2cf9277dc
@ -26,9 +26,7 @@ type ScreencastManagerCtx struct {
|
|||||||
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
|
||||||
@ -44,8 +42,7 @@ func screencastNew(enabled bool, pipelineStr string) *ScreencastManagerCtx {
|
|||||||
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,
|
||||||
}
|
}
|
||||||
@ -53,7 +50,6 @@ func screencastNew(enabled bool, pipelineStr string) *ScreencastManagerCtx {
|
|||||||
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 {
|
if err != nil && !errors.Is(err, types.ErrCapturePipelineAlreadyExists) {
|
||||||
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
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -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
|
||||||
}
|
}
|
||||||
@ -43,34 +39,9 @@ func streamSinkNew(codec codec.RTPCodec, pipelineStr func() string, video_id str
|
|||||||
logger: logger,
|
logger: logger,
|
||||||
codec: codec,
|
codec: codec,
|
||||||
pipelineStr: pipelineStr,
|
pipelineStr: pipelineStr,
|
||||||
sampleStop: make(chan struct{}),
|
|
||||||
sampleUpdate: make(chan struct{}),
|
|
||||||
listeners: map[uintptr]*func(sample types.Sample){},
|
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
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user