257 lines
6.8 KiB
Go
257 lines
6.8 KiB
Go
// Copyright (c) 2024 Joshua Rich <joshua.rich@gmail.com>
|
|
//
|
|
// This software is released under the MIT License.
|
|
// https://opensource.org/licenses/MIT
|
|
|
|
package agent
|
|
|
|
import (
|
|
"context"
|
|
"sync"
|
|
|
|
"github.com/robfig/cron/v3"
|
|
"github.com/rs/zerolog/log"
|
|
|
|
mqttapi "github.com/joshuar/go-hass-anything/v9/pkg/mqtt"
|
|
|
|
"github.com/joshuar/go-hass-agent/internal/device"
|
|
"github.com/joshuar/go-hass-agent/internal/hass"
|
|
"github.com/joshuar/go-hass-agent/internal/hass/sensor"
|
|
"github.com/joshuar/go-hass-agent/internal/preferences"
|
|
"github.com/joshuar/go-hass-agent/internal/scripts"
|
|
)
|
|
|
|
// Worker represents an object that is responsible for retrieving and returning
|
|
// sensors. Workers have a Name and Description for display and logging
|
|
// purposes. A worker will have a Sensors function that will return a list of
|
|
// sensors, including their latest values. A worker will also have a Updates
|
|
// function that returns a channel on which updates to its sensors will be
|
|
// published.
|
|
type Worker interface {
|
|
Name() string
|
|
Description() string
|
|
Sensors(ctx context.Context) []sensor.Details
|
|
Updates(ctx context.Context) <-chan sensor.Details
|
|
}
|
|
|
|
// runWorkers will call all the sensor worker functions that have been defined
|
|
// for this device.
|
|
func runWorkers(ctx context.Context, trk SensorTracker, reg sensor.Registry) {
|
|
workerFuncs := sensorWorkers()
|
|
workerFuncs = append(workerFuncs, device.ExternalIPUpdater, device.VersionUpdater)
|
|
|
|
var wg sync.WaitGroup
|
|
var outCh []<-chan sensor.Details
|
|
|
|
log.Debug().Msg("Starting worker funcs.")
|
|
for i := 0; i < len(workerFuncs); i++ {
|
|
outCh = append(outCh, workerFuncs[i](ctx))
|
|
}
|
|
|
|
sensorUpdates := sensor.MergeSensorCh(ctx, outCh...)
|
|
wg.Add(1)
|
|
go func() {
|
|
log.Debug().Msg("Listening for sensor updates.")
|
|
defer wg.Done()
|
|
for {
|
|
select {
|
|
case <-ctx.Done():
|
|
return
|
|
case s := <-sensorUpdates:
|
|
go func(s sensor.Details) {
|
|
if err := trk.UpdateSensor(ctx, reg, s); err != nil {
|
|
log.Warn().Err(err).Str("id", s.ID()).Msg("Update failed.")
|
|
} else {
|
|
log.Debug().
|
|
Str("name", s.Name()).
|
|
Str("id", s.ID()).
|
|
Interface("state", s.State()).
|
|
Str("units", s.Units()).
|
|
Msg("Sensor updated.")
|
|
}
|
|
}(s)
|
|
}
|
|
}
|
|
}()
|
|
|
|
locationUpdates := locationWorker()(ctx)
|
|
wg.Add(1)
|
|
go func() {
|
|
log.Debug().Msg("Listening for location updates.")
|
|
defer wg.Done()
|
|
for {
|
|
select {
|
|
case <-ctx.Done():
|
|
return
|
|
case l := <-locationUpdates:
|
|
go func(l *hass.LocationData) {
|
|
if err := hass.UpdateLocation(ctx, l); err != nil {
|
|
log.Warn().Err(err).Msg("Location update failed.")
|
|
}
|
|
}(l)
|
|
}
|
|
}
|
|
}()
|
|
|
|
wg.Wait()
|
|
}
|
|
|
|
// runScripts will retrieve all scripts that the agent can run and queue them up
|
|
// to be run on their defined schedule using the cron scheduler. It also sets up
|
|
// a channel to receive script output and send appropriate sensor objects to the
|
|
// sensor.
|
|
func runScripts(ctx context.Context, path string, trk SensorTracker, reg sensor.Registry) {
|
|
allScripts, err := scripts.FindScripts(path)
|
|
switch {
|
|
case err != nil:
|
|
log.Error().Err(err).Msg("Error getting scripts.")
|
|
return
|
|
case len(allScripts) == 0:
|
|
log.Debug().Msg("Could not find any script files.")
|
|
return
|
|
}
|
|
c := cron.New()
|
|
var outCh []<-chan sensor.Details
|
|
for _, s := range allScripts {
|
|
schedule := s.Schedule()
|
|
if schedule != "" {
|
|
_, err := c.AddJob(schedule, s)
|
|
if err != nil {
|
|
log.Warn().Err(err).Str("script", s.Path()).
|
|
Msg("Unable to schedule script.")
|
|
break
|
|
}
|
|
outCh = append(outCh, s.Output)
|
|
log.Debug().Str("schedule", schedule).Str("script", s.Path()).
|
|
Msg("Added script sensor.")
|
|
}
|
|
}
|
|
log.Debug().Msg("Starting cron scheduler for script sensors.")
|
|
c.Start()
|
|
go func() {
|
|
for s := range sensor.MergeSensorCh(ctx, outCh...) {
|
|
go func(s sensor.Details) {
|
|
if err := trk.UpdateSensor(ctx, reg, s); err != nil {
|
|
log.Warn().Err(err).Str("id", s.ID()).Msg("Update sensor failed.")
|
|
} else {
|
|
log.Debug().
|
|
Str("name", s.Name()).
|
|
Str("id", s.ID()).
|
|
Interface("state", s.State()).
|
|
Str("units", s.Units()).
|
|
Msg("Sensor updated.")
|
|
}
|
|
}(s)
|
|
}
|
|
}()
|
|
<-ctx.Done()
|
|
log.Debug().Msg("Stopping cron scheduler for script sensors.")
|
|
cronCtx := c.Stop()
|
|
<-cronCtx.Done()
|
|
}
|
|
|
|
// runNotificationsWorker will run a goroutine that is listening for
|
|
// notification messages from Home Assistant on a websocket connection. Any
|
|
// received notifications will be dipslayed on the device running the agent.
|
|
func (agent *Agent) runNotificationsWorker(ctx context.Context) {
|
|
log.Debug().Msg("Listening for notifications.")
|
|
|
|
notifyCh := hass.StartWebsocket(ctx)
|
|
|
|
var wg sync.WaitGroup
|
|
|
|
wg.Add(1)
|
|
go func() {
|
|
defer wg.Done()
|
|
for {
|
|
select {
|
|
case <-ctx.Done():
|
|
log.Debug().Msg("Stopping notification handler.")
|
|
return
|
|
case n := <-notifyCh:
|
|
agent.ui.DisplayNotification(n)
|
|
}
|
|
}
|
|
}()
|
|
|
|
wg.Wait()
|
|
}
|
|
|
|
// runMQTTWorker will set up a connection to MQTT and listen on topics for
|
|
// controlling this device from Home Assistant.
|
|
func runMQTTWorker(ctx context.Context) {
|
|
prefs, err := preferences.Load()
|
|
if err != nil {
|
|
log.Error().Err(err).Msg("Could not load MQTT preferences.")
|
|
return
|
|
}
|
|
if !prefs.MQTTEnabled {
|
|
return
|
|
}
|
|
|
|
mqttCtx, mqttCancel := context.WithCancel(ctx)
|
|
defer mqttCancel()
|
|
|
|
// Create an MQTT device for this operating system and run its Setup.
|
|
mqttDevice := newMQTTDevice(mqttCtx)
|
|
if err := mqttDevice.Setup(mqttCtx); err != nil {
|
|
log.Error().Err(err).Msg("Could not set up device MQTT functionality.")
|
|
return
|
|
}
|
|
|
|
// Create a new connection to the MQTT broker. This will also publish the
|
|
// device subscriptions.
|
|
client, err := mqttapi.NewClient(mqttCtx, prefs, mqttDevice.Subscriptions(), mqttDevice.Configs())
|
|
if err != nil {
|
|
log.Error().Err(err).Msg("Could not connect to MQTT broker.")
|
|
return
|
|
}
|
|
|
|
// Publish the device configs.
|
|
log.Debug().Msg("Publishing configs.")
|
|
if err := client.Publish(mqttDevice.Configs()...); err != nil {
|
|
log.Error().Err(err).Msg("Failed to publish configuration messages.")
|
|
}
|
|
|
|
go func() {
|
|
log.Debug().Msg("Listening for messages to publish to MQTT.")
|
|
for {
|
|
select {
|
|
case msg := <-mqttDevice.Msgs():
|
|
if err := client.Publish(msg); err != nil {
|
|
log.Warn().Err(err).Msg("Unable to publish message to MQTT.")
|
|
}
|
|
case <-ctx.Done():
|
|
mqttCancel()
|
|
log.Debug().Msg("Stopped listening for messages to publish to MQTT.")
|
|
return
|
|
}
|
|
}
|
|
}()
|
|
|
|
<-ctx.Done()
|
|
}
|
|
|
|
func resetMQTTWorker(ctx context.Context) {
|
|
prefs, err := preferences.Load()
|
|
if err != nil {
|
|
log.Error().Err(err).Msg("Could not load MQTT preferences.")
|
|
return
|
|
}
|
|
if !prefs.MQTTEnabled {
|
|
return
|
|
}
|
|
|
|
mqttDevice := newMQTTDevice(ctx)
|
|
|
|
c, err := mqttapi.NewClient(ctx, prefs, nil, nil)
|
|
if err != nil {
|
|
log.Error().Err(err).Msg("Could not connect to MQTT broker.")
|
|
return
|
|
}
|
|
|
|
if err := c.Unpublish(mqttDevice.Configs()...); err != nil {
|
|
log.Error().Err(err).Msg("Failed to reset MQTT.")
|
|
}
|
|
}
|