mirror of
https://github.com/diamondburned/arikawa.git
synced 2025-01-20 19:47:12 +00:00
61 lines
1.2 KiB
Go
61 lines
1.2 KiB
Go
|
// Package handleloop provides clean abstractions to handle listening to
|
||
|
// channels and passing them onto event handlers.
|
||
|
package handleloop
|
||
|
|
||
|
import "github.com/diamondburned/arikawa/v2/utils/handler"
|
||
|
|
||
|
// Loop provides a reusable event looper abstraction. It is thread-safe to use
|
||
|
// concurrently.
|
||
|
type Loop struct {
|
||
|
dst *handler.Handler
|
||
|
run chan struct{}
|
||
|
stop chan struct{}
|
||
|
}
|
||
|
|
||
|
func NewLoop(dst *handler.Handler) *Loop {
|
||
|
return &Loop{
|
||
|
dst: dst,
|
||
|
run: make(chan struct{}, 1), // intentional 1 buffer
|
||
|
stop: make(chan struct{}), // intentional unbuffer
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Start starts a new event loop. It will try to stop existing loops before.
|
||
|
func (l *Loop) Start(src <-chan interface{}) {
|
||
|
// Ensure we're stopped.
|
||
|
l.Stop()
|
||
|
|
||
|
// Mark that we're running.
|
||
|
l.run <- struct{}{}
|
||
|
|
||
|
go func() {
|
||
|
for {
|
||
|
select {
|
||
|
case event := <-src:
|
||
|
l.dst.Call(event)
|
||
|
|
||
|
case <-l.stop:
|
||
|
l.stop <- struct{}{}
|
||
|
return
|
||
|
}
|
||
|
}
|
||
|
}()
|
||
|
}
|
||
|
|
||
|
// Stop tries to stop the Loop. If the Loop is not running, then it does
|
||
|
// nothing; thus, it can be called multiple times.
|
||
|
func (l *Loop) Stop() {
|
||
|
// Ensure that we are running before stopping.
|
||
|
select {
|
||
|
case <-l.run:
|
||
|
// running
|
||
|
default:
|
||
|
return
|
||
|
}
|
||
|
|
||
|
// send a close request
|
||
|
l.stop <- struct{}{}
|
||
|
// wait for a reply
|
||
|
<-l.stop
|
||
|
}
|