209 lines
5.1 KiB
Go
209 lines
5.1 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
|
|
package apps
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
"log/slog"
|
|
|
|
"github.com/godbus/dbus/v5"
|
|
|
|
"github.com/joshuar/go-hass-agent/internal/hass/sensor"
|
|
"github.com/joshuar/go-hass-agent/internal/hass/sensor/types"
|
|
"github.com/joshuar/go-hass-agent/internal/linux"
|
|
"github.com/joshuar/go-hass-agent/internal/preferences"
|
|
"github.com/joshuar/go-hass-agent/pkg/linux/dbusx"
|
|
)
|
|
|
|
const (
|
|
appStateDBusMethod = "org.freedesktop.impl.portal.Background.GetAppState"
|
|
appStateDBusPath = "/org/freedesktop/portal/desktop"
|
|
appStateDBusInterface = "org.freedesktop.impl.portal.Background"
|
|
appStateDBusEvent = "org.freedesktop.impl.portal.Background.RunningApplicationsChanged"
|
|
|
|
workerID = "app_sensors"
|
|
|
|
activeAppsIcon = "mdi:application"
|
|
activeAppsName = "Active App"
|
|
activeAppsID = "active_app"
|
|
|
|
runningAppsIcon = "mdi:apps"
|
|
runningAppsUnits = "apps"
|
|
runningAppsName = "Running Apps"
|
|
runningAppsID = "running_apps"
|
|
)
|
|
|
|
var ErrNoApps = errors.New("no running apps")
|
|
|
|
type WorkerPrefs preferences.CommonWorkerPrefs
|
|
|
|
func (w *sensorWorker) PreferencesID() string {
|
|
return workerID
|
|
}
|
|
|
|
func (w *sensorWorker) DefaultPreferences() WorkerPrefs {
|
|
return WorkerPrefs{}
|
|
}
|
|
|
|
type sensorWorker struct {
|
|
getAppStates func() (map[string]dbus.Variant, error)
|
|
triggerCh chan dbusx.Trigger
|
|
runningApp string
|
|
totalRunningApps int
|
|
}
|
|
|
|
func (w *sensorWorker) Events(ctx context.Context) (<-chan sensor.Entity, error) {
|
|
sensorCh := make(chan sensor.Entity)
|
|
logger := slog.Default().With(slog.String("worker", workerID))
|
|
|
|
sendSensors := func(ctx context.Context, sensorCh chan sensor.Entity) {
|
|
appSensors, err := w.Sensors(ctx)
|
|
if err != nil {
|
|
logger.Debug("Failed to update app sensors.", slog.Any("error", err))
|
|
|
|
return
|
|
}
|
|
|
|
for _, s := range appSensors {
|
|
sensorCh <- s
|
|
}
|
|
}
|
|
|
|
// Send an initial update.
|
|
go func() {
|
|
sendSensors(ctx, sensorCh)
|
|
}()
|
|
|
|
// Listen for and process updates from D-Bus.
|
|
go func() {
|
|
defer close(sensorCh)
|
|
|
|
for range w.triggerCh {
|
|
sendSensors(ctx, sensorCh)
|
|
}
|
|
}()
|
|
|
|
return sensorCh, nil
|
|
}
|
|
|
|
func (w *sensorWorker) Sensors(_ context.Context) ([]sensor.Entity, error) {
|
|
var (
|
|
sensors []sensor.Entity
|
|
runningApps []string
|
|
)
|
|
|
|
appStates, err := w.getAppStates()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
for name, variant := range appStates {
|
|
// Convert the state to something we understand.
|
|
state, err := dbusx.VariantToValue[uint32](variant)
|
|
if err != nil {
|
|
continue
|
|
}
|
|
// If the state is greater than 0, this app is running.
|
|
if state > 0 {
|
|
runningApps = append(runningApps, name)
|
|
}
|
|
// If the state is 2 this app is running and the currently active app.
|
|
if state == 2 && w.runningApp != name {
|
|
w.runningApp = name
|
|
sensors = append(sensors,
|
|
sensor.NewSensor(
|
|
sensor.WithName(activeAppsName),
|
|
sensor.WithID(activeAppsID),
|
|
sensor.AsTypeSensor(),
|
|
sensor.WithState(
|
|
sensor.WithIcon(activeAppsIcon),
|
|
sensor.WithValue(name),
|
|
sensor.WithDataSourceAttribute(linux.DataSrcDbus),
|
|
),
|
|
),
|
|
)
|
|
}
|
|
}
|
|
|
|
// Update the running apps sensor.
|
|
if w.totalRunningApps != len(runningApps) {
|
|
sensors = append(sensors,
|
|
sensor.NewSensor(
|
|
sensor.WithName(runningAppsName),
|
|
sensor.WithID(runningAppsID),
|
|
sensor.WithUnits(runningAppsUnits),
|
|
sensor.WithStateClass(types.StateClassMeasurement),
|
|
sensor.WithState(
|
|
sensor.WithIcon(runningAppsIcon),
|
|
sensor.WithValue(len(runningApps)),
|
|
sensor.WithDataSourceAttribute(linux.DataSrcDbus),
|
|
sensor.WithAttribute("apps", runningApps),
|
|
),
|
|
),
|
|
)
|
|
w.totalRunningApps = len(runningApps)
|
|
}
|
|
|
|
return sensors, nil
|
|
}
|
|
|
|
func NewAppWorker(ctx context.Context) (*linux.EventSensorWorker, error) {
|
|
worker := linux.NewEventSensorWorker(workerID)
|
|
|
|
// If we cannot find a portal interface, we cannot monitor the active app.
|
|
portalDest, ok := linux.CtxGetDesktopPortal(ctx)
|
|
if !ok {
|
|
return worker, linux.ErrNoDesktopPortal
|
|
}
|
|
|
|
// Connect to the D-Bus session bus. Bail if we can't.
|
|
bus, ok := linux.CtxGetSessionBus(ctx)
|
|
if !ok {
|
|
return worker, linux.ErrNoSessionBus
|
|
}
|
|
|
|
triggerCh, err := dbusx.NewWatch(
|
|
dbusx.MatchPath(appStateDBusPath),
|
|
dbusx.MatchInterface(appStateDBusInterface),
|
|
dbusx.MatchMembers("RunningApplicationsChanged"),
|
|
).Start(ctx, bus)
|
|
if err != nil {
|
|
return worker, fmt.Errorf("could not watch D-Bus for app state events: %w", err)
|
|
}
|
|
|
|
appsWorker := &sensorWorker{
|
|
triggerCh: triggerCh,
|
|
getAppStates: func() (map[string]dbus.Variant, error) {
|
|
apps, err := dbusx.GetData[map[string]dbus.Variant](bus, appStateDBusPath, portalDest, appStateDBusMethod)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("could not retrieve app list from D-Bus: %w", err)
|
|
}
|
|
|
|
if apps == nil {
|
|
return nil, ErrNoApps
|
|
}
|
|
|
|
return apps, nil
|
|
},
|
|
}
|
|
|
|
prefs, err := preferences.LoadWorker(appsWorker)
|
|
if err != nil {
|
|
return worker, fmt.Errorf("could not load preferences: %w", err)
|
|
}
|
|
|
|
if prefs.Disabled {
|
|
return worker, nil
|
|
}
|
|
|
|
worker.EventSensorType = appsWorker
|
|
|
|
return worker, nil
|
|
}
|