209 lines
5.1 KiB
Go
209 lines
5.1 KiB
Go
// Copyright 2024 Joshua Rich <joshua.rich@gmail.com>.
|
|
// SPDX-License-Identifier: MIT
|
|
|
|
//revive:disable:unused-receiver
|
|
package net
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"log/slog"
|
|
"slices"
|
|
"strings"
|
|
"sync"
|
|
|
|
"github.com/godbus/dbus/v5"
|
|
|
|
"github.com/joshuar/go-hass-agent/internal/hass/sensor"
|
|
"github.com/joshuar/go-hass-agent/internal/linux"
|
|
"github.com/joshuar/go-hass-agent/internal/logging"
|
|
"github.com/joshuar/go-hass-agent/internal/preferences"
|
|
"github.com/joshuar/go-hass-agent/pkg/linux/dbusx"
|
|
)
|
|
|
|
const (
|
|
dBusNMPath = "/org/freedesktop/NetworkManager"
|
|
dBusNMObj = "org.freedesktop.NetworkManager"
|
|
dbusNMActiveConnPath = dBusNMPath + "/ActiveConnection"
|
|
dbusNMActiveConnIntr = dBusNMObj + ".Connection.Active"
|
|
|
|
connStateChangedSignal = "StateChanged"
|
|
ipv4ConfigPropName = "Ip4Config"
|
|
ipv6ConfigPropName = "Ip6Config"
|
|
statePropName = "State"
|
|
activeConnectionsProp = "ActivatingConnection"
|
|
|
|
netConnWorkerID = "network_connection_sensors"
|
|
)
|
|
|
|
type ConnectionsWorker struct {
|
|
bus *dbusx.Bus
|
|
list map[string]*connection
|
|
logger *slog.Logger
|
|
prefs *WorkerPrefs
|
|
linux.EventSensorWorker
|
|
mu sync.Mutex
|
|
}
|
|
|
|
func (w *ConnectionsWorker) track(conn *connection) {
|
|
w.mu.Lock()
|
|
w.list[conn.name] = conn
|
|
w.mu.Unlock()
|
|
}
|
|
|
|
func (w *ConnectionsWorker) untrack(id string) {
|
|
w.mu.Lock()
|
|
delete(w.list, id)
|
|
w.mu.Unlock()
|
|
}
|
|
|
|
func (w *ConnectionsWorker) isTracked(id string) bool {
|
|
w.mu.Lock()
|
|
defer w.mu.Unlock()
|
|
|
|
if _, found := w.list[id]; found {
|
|
return true
|
|
}
|
|
|
|
return false
|
|
}
|
|
|
|
func (w *ConnectionsWorker) Sensors(_ context.Context) ([]sensor.Entity, error) {
|
|
return nil, linux.ErrUnimplemented
|
|
}
|
|
|
|
//nolint:mnd
|
|
//revive:disable:function-length
|
|
func (w *ConnectionsWorker) Events(ctx context.Context) (<-chan sensor.Entity, error) {
|
|
sensorCh := make(chan sensor.Entity)
|
|
connCtx, connCancel := context.WithCancel(ctx)
|
|
|
|
triggerCh, err := dbusx.NewWatch(
|
|
dbusx.MatchPathNamespace(dbusNMActiveConnPath),
|
|
dbusx.MatchInterface(dbusNMActiveConnIntr),
|
|
dbusx.MatchMembers("StateChanged"),
|
|
).Start(connCtx, w.bus)
|
|
if err != nil {
|
|
close(sensorCh)
|
|
connCancel()
|
|
|
|
return sensorCh, fmt.Errorf("failed to create network connections D-Bus watch: %w", err)
|
|
}
|
|
|
|
go func() {
|
|
defer close(sensorCh)
|
|
w.logger.Debug("Watching for network connections.")
|
|
|
|
for event := range triggerCh {
|
|
connectionPath := dbus.ObjectPath(event.Path)
|
|
// If this connection is in the process of deactivating, don't
|
|
// start tracking it.
|
|
if state, stateChange := event.Content[0].(uint32); stateChange {
|
|
if state > 2 {
|
|
continue
|
|
}
|
|
}
|
|
// Track all activating/new connections.
|
|
if err = w.handleConnection(connCtx, connectionPath, sensorCh); err != nil {
|
|
w.logger.Debug("Could not monitor connection.", slog.Any("error", err))
|
|
}
|
|
}
|
|
|
|
w.logger.Debug("Stopped watching network connections.")
|
|
}()
|
|
|
|
go func() {
|
|
defer connCancel()
|
|
<-ctx.Done()
|
|
w.logger.Debug("Stopped events.")
|
|
}()
|
|
|
|
// monitor all current active connections
|
|
connectionlist, err := dbusx.NewProperty[[]dbus.ObjectPath](w.bus, dBusNMPath, dBusNMObj, dBusNMObj+".ActiveConnections").Get()
|
|
if err != nil {
|
|
w.logger.Debug("Error getting active connections from D-Bus", slog.Any("error", err))
|
|
} else {
|
|
for _, path := range connectionlist {
|
|
if err := w.handleConnection(connCtx, path, sensorCh); err != nil {
|
|
w.logger.Debug("Could not monitor connection.", slog.Any("error", err))
|
|
}
|
|
}
|
|
}
|
|
|
|
return sensorCh, nil
|
|
}
|
|
|
|
func (w *ConnectionsWorker) PreferencesID() string {
|
|
return preferencesID
|
|
}
|
|
|
|
func (w *ConnectionsWorker) DefaultPreferences() WorkerPrefs {
|
|
return WorkerPrefs{
|
|
IgnoredDevices: defaultIgnoredDevices,
|
|
}
|
|
}
|
|
|
|
func (w *ConnectionsWorker) handleConnection(ctx context.Context, path dbus.ObjectPath, sensorCh chan sensor.Entity) error {
|
|
conn, err := newConnection(w.bus, path)
|
|
if err != nil {
|
|
return fmt.Errorf("could not create connection: %w", err)
|
|
}
|
|
// Ignore loopback or already tracked connections.
|
|
if conn.name == "lo" || w.isTracked(conn.name) {
|
|
slog.Debug("Ignoring connection.", slog.String("connection", conn.name))
|
|
|
|
return nil
|
|
}
|
|
// Ignore user-defined devices.
|
|
if slices.ContainsFunc(w.prefs.IgnoredDevices, func(e string) bool {
|
|
return strings.HasPrefix(conn.name, e)
|
|
}) {
|
|
return nil
|
|
}
|
|
|
|
// Start monitoring the connection. Pass any sensor updates from the
|
|
// connection through the sensor channel.
|
|
go func() {
|
|
w.track(conn)
|
|
|
|
for s := range conn.monitor(ctx, w.bus) {
|
|
sensorCh <- s
|
|
}
|
|
|
|
w.untrack(conn.name)
|
|
}()
|
|
|
|
return nil
|
|
}
|
|
|
|
func NewConnectionWorker(ctx context.Context) (*linux.EventSensorWorker, error) {
|
|
var err error
|
|
|
|
worker := linux.NewEventSensorWorker(netConnWorkerID)
|
|
|
|
bus, ok := linux.CtxGetSystemBus(ctx)
|
|
if !ok {
|
|
return worker, linux.ErrNoSystemBus
|
|
}
|
|
|
|
connectionsWorker := &ConnectionsWorker{
|
|
bus: bus,
|
|
list: make(map[string]*connection),
|
|
logger: logging.FromContext(ctx).
|
|
With(slog.String("worker", netConnWorkerID)),
|
|
}
|
|
|
|
connectionsWorker.prefs, err = preferences.LoadWorker(connectionsWorker)
|
|
if err != nil {
|
|
return worker, fmt.Errorf("could not load preferences: %w", err)
|
|
}
|
|
|
|
if connectionsWorker.prefs.Disabled {
|
|
return worker, nil
|
|
}
|
|
|
|
worker.EventSensorType = connectionsWorker
|
|
|
|
return worker, nil
|
|
}
|