Gisle Aune
2 years ago
22 changed files with 368 additions and 43 deletions
-
3cmd/bustest/main.go
-
5cmd/lucifer4-server/main.go
-
8commands/edit.go
-
2device/pointer.go
-
14device/state.go
-
5effects/serializable.go
-
46effects/vrange.go
-
7interface.go
-
39internal/color/color.go
-
2internal/gentools/maps.go
-
68services/effectenforcer/service.go
-
9services/effectenforcer/service_test.go
-
6services/httpapiv1/service.go
-
13services/hue/bridge.go
-
2services/nanoleaf/client.go
-
13services/resolver.go
-
7services/uistate/data.go
-
8services/uistate/patch.go
-
7services/uistate/service.go
-
27services/variables/data.go
-
17services/variables/patch.go
-
103services/variables/service.go
@ -0,0 +1,46 @@ |
|||
package effects |
|||
|
|||
import ( |
|||
"fmt" |
|||
"git.aiterp.net/lucifer3/server/device" |
|||
"time" |
|||
) |
|||
|
|||
type VRange struct { |
|||
States []device.State `json:"states,omitempty"` |
|||
Variable string `json:"variable"` |
|||
Min float64 `json:"min"` |
|||
Max float64 `json:"max"` |
|||
} |
|||
|
|||
func (e VRange) VariableName() string { |
|||
return e.Variable |
|||
} |
|||
|
|||
func (e VRange) VariableState(_, _ int, value float64) device.State { |
|||
if value <= e.Min { |
|||
return e.States[0] |
|||
} else if value >= e.Max { |
|||
return e.States[len(e.States)-1] |
|||
} |
|||
|
|||
return gradientStateFactor(e.States, true, (value-e.Min)/(e.Max-e.Min)) |
|||
} |
|||
|
|||
func (e VRange) State(_, _, _ int) device.State { |
|||
if len(e.States) == 0 { |
|||
return device.State{} |
|||
} |
|||
return e.States[0] |
|||
} |
|||
|
|||
func (e VRange) Frequency() time.Duration { |
|||
return 0 |
|||
} |
|||
|
|||
func (e VRange) EffectDescription() string { |
|||
return fmt.Sprintf( |
|||
"VRange(states:%s, name:%s, range:%.1f..%.1f)", |
|||
statesDescription(e.States), e.Variable, e.Min, e.Max, |
|||
) |
|||
} |
@ -0,0 +1,27 @@ |
|||
package variables |
|||
|
|||
import "git.aiterp.net/lucifer3/server/internal/gentools" |
|||
|
|||
type Variables struct { |
|||
Temperature map[string]AvgMinMax `json:"temperature"` |
|||
Motion map[string]AvgMinMax `json:"motion"` |
|||
} |
|||
|
|||
func (v Variables) WithPropertyPatch(patch PropertyPatch) Variables { |
|||
if patch.Motion != nil { |
|||
v.Motion = gentools.CopyMap(v.Motion) |
|||
v.Motion[patch.Key] = *patch.Motion |
|||
} |
|||
if patch.Temperature != nil { |
|||
v.Temperature = gentools.CopyMap(v.Temperature) |
|||
v.Temperature[patch.Key] = *patch.Temperature |
|||
} |
|||
|
|||
return v |
|||
} |
|||
|
|||
type AvgMinMax struct { |
|||
Avg float64 `json:"avg"` |
|||
Min float64 `json:"min"` |
|||
Max float64 `json:"max"` |
|||
} |
@ -0,0 +1,17 @@ |
|||
package variables |
|||
|
|||
import "fmt" |
|||
|
|||
type PropertyPatch struct { |
|||
Key string `json:"id"` |
|||
Temperature *AvgMinMax `json:"temperature,omitempty"` |
|||
Motion *AvgMinMax `json:"motion,omitempty"` |
|||
} |
|||
|
|||
func (e PropertyPatch) VerboseKey() string { |
|||
return "variables.PropertyPatch" |
|||
} |
|||
|
|||
func (e PropertyPatch) EventDescription() string { |
|||
return fmt.Sprintf("variables.PropertyPatch(key=%s)", e.Key) |
|||
} |
@ -0,0 +1,103 @@ |
|||
package variables |
|||
|
|||
import ( |
|||
lucifer3 "git.aiterp.net/lucifer3/server" |
|||
"git.aiterp.net/lucifer3/server/events" |
|||
"git.aiterp.net/lucifer3/server/services" |
|||
) |
|||
|
|||
func NewService(resolver *services.Resolver) lucifer3.Service { |
|||
return &service{ |
|||
resolver: resolver, |
|||
temperatures: map[string]float64{}, |
|||
motions: map[string]float64{}, |
|||
} |
|||
} |
|||
|
|||
type service struct { |
|||
resolver *services.Resolver |
|||
temperatures map[string]float64 |
|||
motions map[string]float64 |
|||
} |
|||
|
|||
func (s *service) Active() bool { |
|||
return true |
|||
} |
|||
|
|||
func (s *service) HandleEvent(bus *lucifer3.EventBus, event lucifer3.Event) { |
|||
var patchEvents []lucifer3.Event |
|||
|
|||
switch event := event.(type) { |
|||
case events.MotionSensed: |
|||
s.motions[event.ID] = event.SecondsSince |
|||
patchEvents = s.updateDevice(event.ID) |
|||
case events.TemperatureChanged: |
|||
s.temperatures[event.ID] = event.Temperature |
|||
patchEvents = s.updateDevice(event.ID) |
|||
case events.AliasAdded: |
|||
patchEvents = s.updateDevice(event.ID) |
|||
case events.AliasRemoved: |
|||
patchEvents = s.updateDevice(event.ID) |
|||
} |
|||
|
|||
bus.RunEvents(patchEvents) |
|||
} |
|||
|
|||
func (s *service) updateDevice(id string) []lucifer3.Event { |
|||
ptr := s.resolver.GetByID(id) |
|||
if ptr == nil { |
|||
return nil |
|||
} |
|||
|
|||
res := make([]lucifer3.Event, 0, len(ptr.Aliases)+1) |
|||
|
|||
for _, alias := range append([]string{ptr.ID}, ptr.Aliases...) { |
|||
patch := PropertyPatch{Key: alias} |
|||
motionSamples := 0.0 |
|||
motionTotal := 0.0 |
|||
temperatureSamples := 0.0 |
|||
temperatureTotal := 0.0 |
|||
|
|||
for _, ptr := range s.resolver.Resolve(alias) { |
|||
if temp, ok := s.temperatures[ptr.ID]; ok { |
|||
if patch.Temperature == nil { |
|||
patch.Temperature = &AvgMinMax{} |
|||
if temp < patch.Temperature.Min || temperatureSamples == 0.0 { |
|||
patch.Temperature.Min = temp |
|||
} |
|||
if temp > patch.Temperature.Max { |
|||
patch.Temperature.Max = temp |
|||
} |
|||
|
|||
temperatureSamples += 1.0 |
|||
temperatureTotal += temp |
|||
} |
|||
} |
|||
if motion, ok := s.motions[ptr.ID]; ok { |
|||
if patch.Motion == nil { |
|||
patch.Motion = &AvgMinMax{} |
|||
if motion < patch.Motion.Min || motionSamples == 0.0 { |
|||
patch.Motion.Min = motion |
|||
} |
|||
if motion > patch.Motion.Max { |
|||
patch.Motion.Max = motion |
|||
} |
|||
|
|||
motionSamples += 1.0 |
|||
motionTotal += motion |
|||
} |
|||
} |
|||
} |
|||
|
|||
if temperatureSamples > 0.5 { |
|||
patch.Temperature.Avg = temperatureTotal / temperatureSamples |
|||
} |
|||
if motionSamples > 0.5 { |
|||
patch.Motion.Avg = motionTotal / motionSamples |
|||
} |
|||
|
|||
res = append(res, patch) |
|||
} |
|||
|
|||
return res |
|||
} |
Write
Preview
Loading…
Cancel
Save
Reference in new issue