joshuar-go-hass-agent/internal/agent/agent.go

326 lines
8.0 KiB
Go

// Copyright (c) 2024 Joshua Rich <joshua.rich@gmail.com>
//
// This software is released under the MIT License.
// https://opensource.org/licenses/MIT
// revive:disable:unused-receiver
//
//go:generate moq -out agent_mocks_test.go . UI Registry Tracker SensorController MQTTController Worker
package agent
import (
"context"
"errors"
"fmt"
"log/slog"
"os"
"os/signal"
"path/filepath"
"sync"
"syscall"
"time"
"github.com/adrg/xdg"
fyneui "github.com/joshuar/go-hass-agent/internal/agent/ui/fyneUI"
"github.com/joshuar/go-hass-agent/internal/agent/ui"
"github.com/joshuar/go-hass-agent/internal/hass"
"github.com/joshuar/go-hass-agent/internal/hass/sensor"
"github.com/joshuar/go-hass-agent/internal/logging"
"github.com/joshuar/go-hass-agent/internal/preferences"
)
const (
defaultTimeout = 30 * time.Second
)
var ErrInvalidPrefernces = errors.New("invalid agent preferences")
// UI are the methods required for the agent to display its windows, tray
// and notifications.
type UI interface {
DisplayNotification(n ui.Notification)
DisplayTrayIcon(ctx context.Context, agent ui.Agent, client ui.HassClient, doneCh chan struct{})
DisplayRegistrationWindow(prefs *preferences.Preferences, doneCh chan struct{}) chan struct{}
Run(agent ui.Agent, doneCh chan struct{})
}
type Registry interface {
SetDisabled(id string, state bool) error
SetRegistered(id string, state bool) error
IsDisabled(id string) bool
IsRegistered(id string) bool
}
type Tracker interface {
SensorList() []string
Add(details sensor.Details) error
Get(key string) (sensor.Details, error)
Reset()
}
// Agent holds the options of the running agent, the UI object and a channel for
// closing the agent down.
type Agent struct {
ui UI
hass HassClient
done chan struct{}
prefs *preferences.Preferences
logger *slog.Logger
id string
headless bool
forceRegister bool
}
// Option is a functional parameter that will configure a feature of the agent.
type Option func(*Agent)
// newDefaultAgent returns an agent with default options.
func newDefaultAgent(ctx context.Context, id string) *Agent {
return &Agent{
done: make(chan struct{}),
id: id,
logger: logging.FromContext(ctx),
}
}
// NewAgent creates a new agent with the options specified.
func NewAgent(ctx context.Context, id string, options ...Option) (*Agent, error) {
agent := newDefaultAgent(ctx, id)
// Load the agent preferences.
prefs, err := preferences.Load(agent.GetPreferencesPath())
if err != nil && !errors.Is(err, preferences.ErrNoPreferences) {
return nil, fmt.Errorf("could not create agent: %w", err)
}
agent.prefs = prefs
for _, option := range options {
option(agent)
}
agent.ui = fyneui.NewFyneUI(ctx, preferences.AppName)
return agent, nil
}
// Headless sets whether the agent should run in a headless mode, without any
// GUI.
func Headless(value bool) Option {
return func(a *Agent) {
a.headless = value
}
}
// WithRegistrationInfo will set the info required for registering the agent.
// Only used when the Register command is run.
func WithRegistrationInfo(server, token string, ignoreURLs bool) Option {
return func(a *Agent) {
a.prefs.Registration = &preferences.Registration{
Server: server,
Token: token,
}
a.prefs.Hass = &preferences.Hass{
IgnoreHassURLs: ignoreURLs,
}
}
}
// ForceRegister will force the agent to register against Home Assistant,
// regardless of whether it is already registered. Only used when the Register
// command is run.
func ForceRegister(value bool) Option {
return func(a *Agent) {
a.forceRegister = value
}
}
// Run is the "main loop" of the agent. It sets up the agent, loads the config
// then spawns a sensor tracker and the workers to gather sensor data and
// publish it to Home Assistant.
//
//revive:disable:function-length
func (agent *Agent) Run(ctx context.Context, trk Tracker, reg Registry) error {
var (
wg sync.WaitGroup
regWait sync.WaitGroup
)
ctx, cancelFunc := signal.NotifyContext(ctx, os.Interrupt, syscall.SIGTERM)
go func() {
<-ctx.Done()
cancelFunc()
}()
agent.hass = hass.NewClient(ctx, trk, reg)
// agent.handleSignals()
regWait.Add(1)
go func() {
defer regWait.Done()
if err := agent.checkRegistration(ctx, trk); err != nil {
agent.logger.Log(ctx, logging.LevelFatal, "Error checking registration status.", slog.Any("error", err))
close(agent.done)
}
}()
wg.Add(1)
go func() {
defer wg.Done()
regWait.Wait()
agent.hass.Endpoint(agent.prefs.RestAPIURL(), defaultTimeout)
// Create a context for runners
controllerCtx, cancelFunc := context.WithCancel(ctx)
// Cancel the runner context when the agent is done.
go func() {
<-agent.done
cancelFunc()
agent.logger.Debug("Agent done.")
}()
var (
sensorControllers []SensorController
mqttControllers []MQTTController
)
// Setup and sort all controllers by type.
for _, c := range agent.setupControllers(controllerCtx) {
switch controller := c.(type) {
case SensorController:
sensorControllers = append(sensorControllers, controller)
case MQTTController:
mqttControllers = append(mqttControllers, controller)
}
}
wg.Add(1)
// Run workers for any sensor controllers.
go func() {
defer wg.Done()
agent.runSensorWorkers(controllerCtx, sensorControllers...)
}()
if len(mqttControllers) > 0 {
wg.Add(1)
// Run workers for any MQTT controllers.
go func() {
defer wg.Done()
agent.runMQTTWorkers(controllerCtx, mqttControllers...)
}()
}
wg.Add(1)
// Listen for notifications from Home Assistant.
go func() {
defer wg.Done()
agent.runNotificationsWorker(controllerCtx)
}()
}()
agent.ui.DisplayTrayIcon(ctx, agent, agent.hass, agent.done)
agent.ui.Run(agent, agent.done)
wg.Wait()
return nil
}
func (agent *Agent) Register(ctx context.Context, trk Tracker) {
var wg sync.WaitGroup
wg.Add(1)
go func() {
defer wg.Done()
if err := agent.checkRegistration(ctx, trk); err != nil {
agent.logger.Log(ctx, logging.LevelFatal, "Error checking registration status", slog.Any("error", err))
}
close(agent.done)
}()
agent.ui.Run(agent, agent.done)
wg.Wait()
}
// Stop will close the agent's done channel which indicates to any goroutines it
// is time to clean up and exit.
func (agent *Agent) Stop() {
defer close(agent.done)
agent.logger.Debug("Stopping Agent.")
if err := agent.prefs.Save(); err != nil {
agent.logger.Warn("Could not save agent preferences", slog.Any("error", err))
}
}
// Reset will remove any agent related files and configuration.
func (agent *Agent) Reset(ctx context.Context) error {
prefs := agent.GetMQTTPreferences()
if prefs != nil && prefs.IsMQTTEnabled() {
if err := agent.resetMQTTControllers(ctx); err != nil {
agent.logger.Warn("Problems occurred resetting MQTT configuration.", slog.Any("error", err))
}
}
return nil
}
// Headless returns a boolean indicating whether the agent is running in
// headless mode or not.
func (agent *Agent) Headless() bool {
return agent.headless
}
// GetMQTTPreferences returns the subset of agent preferences to do with MQTT.
func (agent *Agent) GetMQTTPreferences() *preferences.MQTT {
return agent.prefs.GetMQTTPreferences()
}
// SaveMQTTPreferences takes the given preferences and saves them to disk as
// part of all agent preferences.
func (agent *Agent) SaveMQTTPreferences(prefs *preferences.MQTT) error {
if agent.prefs != nil {
agent.prefs.MQTT = prefs
err := agent.prefs.Save()
if err != nil {
return fmt.Errorf("failed to save mqtt preferences: %w", err)
}
return nil
}
return ErrInvalidPrefernces
}
func (agent *Agent) GetRestAPIURL() string {
return agent.prefs.RestAPIURL()
}
func (agent *Agent) GetRegistryPath() string {
if agent != nil {
return filepath.Join(xdg.ConfigHome, agent.id, "sensorRegistry")
}
return filepath.Join(xdg.ConfigHome, preferences.AppID, "sensorRegistry")
}
func (agent *Agent) GetPreferencesPath() string {
if agent != nil {
return filepath.Join(xdg.ConfigHome, agent.id)
}
return filepath.Join(xdg.ConfigHome, preferences.AppID)
}