147 lines
3.8 KiB
Go
147 lines
3.8 KiB
Go
package main
|
|
|
|
import (
|
|
"log"
|
|
"sync"
|
|
|
|
"git.sr.ht/~michalr/go-satel"
|
|
)
|
|
|
|
func isBasicEventElementOkay(basicEventElement satel.BasicEventElement, allowedTypes []SatelChangeType, allowedIndexes []int) bool {
|
|
for _, allowedType := range allowedTypes {
|
|
if allowedType.GetChangeType() == basicEventElement.Type {
|
|
return true
|
|
}
|
|
}
|
|
for _, allowedIndex := range allowedIndexes {
|
|
if allowedIndex == basicEventElement.Index {
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|
|
|
|
func FilterByTypeOrIndex(ev <-chan satel.Event, wg *sync.WaitGroup, allowedTypes []SatelChangeType, allowedIndexes []int) <-chan satel.Event {
|
|
returnChan := make(chan satel.Event)
|
|
|
|
if (len(allowedTypes) == 0) && (len(allowedIndexes) == 0) {
|
|
// no allowed types == all types are allowed
|
|
wg.Add(1)
|
|
go func() {
|
|
defer wg.Done()
|
|
defer close(returnChan)
|
|
|
|
for e := range ev {
|
|
returnChan <- e
|
|
}
|
|
}()
|
|
} else {
|
|
wg.Add(1)
|
|
go func() {
|
|
defer wg.Done()
|
|
defer close(returnChan)
|
|
|
|
for e := range ev {
|
|
retEv := satel.Event{BasicEvents: make([]satel.BasicEventElement, 0)}
|
|
for _, basicEventElement := range e.BasicEvents {
|
|
if isBasicEventElementOkay(basicEventElement, allowedTypes, allowedIndexes) {
|
|
retEv.BasicEvents = append(retEv.BasicEvents, basicEventElement)
|
|
}
|
|
}
|
|
if len(retEv.BasicEvents) != 0 {
|
|
returnChan <- retEv
|
|
}
|
|
}
|
|
}()
|
|
}
|
|
|
|
return returnChan
|
|
}
|
|
|
|
func FilterByLastSeen(ev <-chan satel.Event, wg *sync.WaitGroup, dataStore *DataStore, logger *log.Logger) <-chan satel.Event {
|
|
returnChan := make(chan satel.Event)
|
|
|
|
wg.Add(1)
|
|
go func() {
|
|
defer wg.Done()
|
|
defer close(returnChan)
|
|
|
|
for e := range ev {
|
|
retEv := satel.Event{BasicEvents: make([]satel.BasicEventElement, 0)}
|
|
for _, basicEventElement := range e.BasicEvents {
|
|
lastSeen := dataStore.GetSystemState()
|
|
val, ok := lastSeen[EventKey{basicEventElement.Type, basicEventElement.Index}]
|
|
if !ok || val.Value != basicEventElement.Value {
|
|
retEv.BasicEvents = append(retEv.BasicEvents, basicEventElement)
|
|
// TODO: flush to disk only after the loop finishes
|
|
dataStore.SetSystemState(EventKey{basicEventElement.Type, basicEventElement.Index}, EventValue{basicEventElement.Value})
|
|
}
|
|
}
|
|
if len(retEv.BasicEvents) != 0 {
|
|
returnChan <- retEv
|
|
}
|
|
}
|
|
logger.Print("Satel disconnected.")
|
|
}()
|
|
|
|
return returnChan
|
|
}
|
|
|
|
func appendToGenericMessage(msg *GenericMessage, new *GenericMessage) *GenericMessage {
|
|
if msg == nil {
|
|
msg = &GenericMessage{make([]satel.BasicEventElement, 0)}
|
|
}
|
|
|
|
throughNewMessages:
|
|
for _, newEv := range new.Messages {
|
|
for i, oldEv := range msg.Messages {
|
|
if oldEv.Index == newEv.Index && oldEv.Type == newEv.Type {
|
|
// this message was seen - update its value
|
|
msg.Messages[i].Value = newEv.Value
|
|
continue throughNewMessages
|
|
}
|
|
}
|
|
// apparently this type of message was not yet seen, save it
|
|
msg.Messages = append(msg.Messages, newEv)
|
|
}
|
|
return msg
|
|
}
|
|
|
|
func Throttle(inputEvents <-chan GenericMessage, wg *sync.WaitGroup, sleeper Sleeper, logger *log.Logger) <-chan GenericMessage {
|
|
returnChan := make(chan GenericMessage)
|
|
timeoutEvents := make(chan interface{})
|
|
|
|
wg.Add(1)
|
|
go func() {
|
|
defer wg.Done()
|
|
defer close(returnChan)
|
|
|
|
var currentEvent *GenericMessage = nil
|
|
loop:
|
|
for {
|
|
select {
|
|
case ev, ok := <-inputEvents:
|
|
if !ok {
|
|
break loop
|
|
}
|
|
if currentEvent == nil {
|
|
logger.Print("Waiting for more messages to arrive before sending...")
|
|
sleeper.Sleep(timeoutEvents)
|
|
}
|
|
currentEvent = appendToGenericMessage(currentEvent, &ev)
|
|
case <-timeoutEvents:
|
|
logger.Print("Time's up, sending all messages we've got for now.")
|
|
returnChan <- *currentEvent
|
|
currentEvent = nil
|
|
}
|
|
}
|
|
|
|
// If anything is left to be sent, send it now
|
|
if currentEvent != nil {
|
|
returnChan <- *currentEvent
|
|
}
|
|
}()
|
|
|
|
return returnChan
|
|
}
|