joshuar-go-hass-agent/internal/linux/system/dbusCommand.go

124 lines
3.2 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 system
import (
"context"
"encoding/json"
"fmt"
"log/slog"
"github.com/eclipse/paho.golang/paho"
mqtthass "github.com/joshuar/go-hass-anything/v12/pkg/hass"
mqttapi "github.com/joshuar/go-hass-anything/v12/pkg/mqtt"
"github.com/joshuar/go-hass-agent/internal/components/logging"
"github.com/joshuar/go-hass-agent/internal/components/preferences"
"github.com/joshuar/go-hass-agent/internal/linux"
"github.com/joshuar/go-hass-agent/pkg/linux/dbusx"
)
const (
dbusCmdPreferencesID = "dbus_commands"
)
type dbusCommandMsg struct {
Bus string `json:"bus"`
Destination string `json:"destination"`
Path string `json:"path"`
Method string `json:"method"`
Args []any `json:"args"`
UseSessionPath bool `json:"use_session_path"`
}
type dbusCmdWorker struct{}
func (w *dbusCmdWorker) PreferencesID() string {
return basePreferencesID + "." + dbusCmdPreferencesID
}
func (w *dbusCmdWorker) DefaultPreferences() preferences.CommonWorkerPrefs {
return preferences.CommonWorkerPrefs{}
}
func NewDBusCommandSubscription(ctx context.Context, device *mqtthass.Device) (*mqttapi.Subscription, error) {
logger := logging.FromContext(ctx).With(slog.String("controller", "dbus_command"))
worker := &dbusCmdWorker{}
prefs, err := preferences.LoadWorker(ctx, worker)
if err != nil {
return nil, fmt.Errorf("could not load preferences: %w", err)
}
//nolint:nilnil
if prefs.IsDisabled() {
return nil, nil
}
systemBus, ok := linux.CtxGetSystemBus(ctx)
if !ok {
return nil, linux.ErrNoSystemBus
}
sessionBus, ok := linux.CtxGetSessionBus(ctx)
if !ok {
return nil, linux.ErrNoSessionBus
}
busMap := map[string]*dbusx.Bus{"session": sessionBus, "system": systemBus}
return &mqttapi.Subscription{
Callback: func(packet *paho.Publish) {
var (
dbusMsg dbusCommandMsg
err error
)
logger = logger.With(
slog.String("bus", dbusMsg.Bus),
slog.String("destination", dbusMsg.Destination),
slog.String("path", dbusMsg.Path),
slog.String("method", dbusMsg.Method),
)
// Unmarshal the request.
if err = json.Unmarshal(packet.Payload, &dbusMsg); err != nil {
logger.Error("Could not unmarshal D-Bus MQTT message.", slog.Any("error", err))
return
}
// Check which bus type was requested.
bus, ok := busMap[dbusMsg.Bus]
if !ok {
logger.Error("Unsupported D-Bus type.")
return
}
// Fetch the session path if requested.
if dbusMsg.UseSessionPath {
dbusMsg.Path, err = busMap["session"].GetSessionPath()
if err != nil {
logger.Error("Could not determine session path.", slog.Any("error", err))
return
}
}
logger.Info("Dispatching D-Bus command.")
// Call the method.
err = dbusx.NewMethod(bus, dbusMsg.Destination, dbusMsg.Path, dbusMsg.Method).Call(ctx, dbusMsg.Args...)
if err != nil {
logger.Warn("Error dispatching D-Bus command.", slog.Any("error", err))
}
},
Topic: "gohassagent/" + device.Name + "/dbuscommand",
},
nil
}