mirror of
https://github.com/AvengeMedia/DankMaterialShell.git
synced 2025-12-05 21:15:38 -05:00
1333 lines
36 KiB
Go
1333 lines
36 KiB
Go
package server
|
|
|
|
import (
|
|
"bufio"
|
|
"encoding/json"
|
|
"fmt"
|
|
"net"
|
|
"os"
|
|
"path/filepath"
|
|
"strconv"
|
|
"strings"
|
|
"sync"
|
|
"sync/atomic"
|
|
"syscall"
|
|
"time"
|
|
|
|
"github.com/AvengeMedia/DankMaterialShell/core/internal/log"
|
|
"github.com/AvengeMedia/DankMaterialShell/core/internal/server/bluez"
|
|
"github.com/AvengeMedia/DankMaterialShell/core/internal/server/brightness"
|
|
"github.com/AvengeMedia/DankMaterialShell/core/internal/server/cups"
|
|
"github.com/AvengeMedia/DankMaterialShell/core/internal/server/dwl"
|
|
"github.com/AvengeMedia/DankMaterialShell/core/internal/server/evdev"
|
|
"github.com/AvengeMedia/DankMaterialShell/core/internal/server/extworkspace"
|
|
"github.com/AvengeMedia/DankMaterialShell/core/internal/server/freedesktop"
|
|
"github.com/AvengeMedia/DankMaterialShell/core/internal/server/loginctl"
|
|
"github.com/AvengeMedia/DankMaterialShell/core/internal/server/models"
|
|
"github.com/AvengeMedia/DankMaterialShell/core/internal/server/network"
|
|
"github.com/AvengeMedia/DankMaterialShell/core/internal/server/wayland"
|
|
"github.com/AvengeMedia/DankMaterialShell/core/internal/server/wlcontext"
|
|
"github.com/AvengeMedia/DankMaterialShell/core/internal/server/wlroutput"
|
|
"github.com/AvengeMedia/DankMaterialShell/core/pkg/syncmap"
|
|
)
|
|
|
|
const APIVersion = 22
|
|
|
|
var CLIVersion = "dev"
|
|
|
|
type Capabilities struct {
|
|
Capabilities []string `json:"capabilities"`
|
|
}
|
|
|
|
type ServerInfo struct {
|
|
APIVersion int `json:"apiVersion"`
|
|
CLIVersion string `json:"cliVersion,omitempty"`
|
|
Capabilities []string `json:"capabilities"`
|
|
}
|
|
|
|
type ServiceEvent struct {
|
|
Service string `json:"service"`
|
|
Data interface{} `json:"data"`
|
|
}
|
|
|
|
var networkManager *network.Manager
|
|
var loginctlManager *loginctl.Manager
|
|
var freedesktopManager *freedesktop.Manager
|
|
var waylandManager *wayland.Manager
|
|
var bluezManager *bluez.Manager
|
|
var cupsManager *cups.Manager
|
|
var dwlManager *dwl.Manager
|
|
var extWorkspaceManager *extworkspace.Manager
|
|
var brightnessManager *brightness.Manager
|
|
var wlrOutputManager *wlroutput.Manager
|
|
var evdevManager *evdev.Manager
|
|
var wlContext *wlcontext.SharedContext
|
|
|
|
var capabilitySubscribers syncmap.Map[string, chan ServerInfo]
|
|
var cupsSubscribers syncmap.Map[string, bool]
|
|
var cupsSubscriberCount atomic.Int32
|
|
var extWorkspaceAvailable atomic.Bool
|
|
var extWorkspaceInitMutex sync.Mutex
|
|
|
|
func getSocketDir() string {
|
|
if runtime := os.Getenv("XDG_RUNTIME_DIR"); runtime != "" {
|
|
return runtime
|
|
}
|
|
|
|
if os.Getuid() == 0 {
|
|
if _, err := os.Stat("/run"); err == nil {
|
|
return "/run/dankdots"
|
|
}
|
|
return "/var/run/dankdots"
|
|
}
|
|
|
|
return os.TempDir()
|
|
}
|
|
|
|
func GetSocketPath() string {
|
|
return filepath.Join(getSocketDir(), fmt.Sprintf("danklinux-%d.sock", os.Getpid()))
|
|
}
|
|
|
|
func cleanupStaleSockets() {
|
|
dir := getSocketDir()
|
|
entries, err := os.ReadDir(dir)
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
for _, entry := range entries {
|
|
if !strings.HasPrefix(entry.Name(), "danklinux-") || !strings.HasSuffix(entry.Name(), ".sock") {
|
|
continue
|
|
}
|
|
|
|
pidStr := strings.TrimPrefix(entry.Name(), "danklinux-")
|
|
pidStr = strings.TrimSuffix(pidStr, ".sock")
|
|
pid, err := strconv.Atoi(pidStr)
|
|
if err != nil {
|
|
continue
|
|
}
|
|
|
|
process, err := os.FindProcess(pid)
|
|
if err != nil {
|
|
socketPath := filepath.Join(dir, entry.Name())
|
|
os.Remove(socketPath)
|
|
log.Debugf("Removed stale socket: %s", socketPath)
|
|
continue
|
|
}
|
|
|
|
err = process.Signal(syscall.Signal(0))
|
|
if err != nil {
|
|
socketPath := filepath.Join(dir, entry.Name())
|
|
os.Remove(socketPath)
|
|
log.Debugf("Removed stale socket: %s", socketPath)
|
|
}
|
|
}
|
|
}
|
|
|
|
func InitializeNetworkManager() error {
|
|
manager, err := network.NewManager()
|
|
if err != nil {
|
|
log.Warnf("Failed to initialize network manager: %v", err)
|
|
return err
|
|
}
|
|
|
|
networkManager = manager
|
|
|
|
log.Info("Network manager initialized")
|
|
return nil
|
|
}
|
|
|
|
func InitializeLoginctlManager() error {
|
|
manager, err := loginctl.NewManager()
|
|
if err != nil {
|
|
log.Warnf("Failed to initialize loginctl manager: %v", err)
|
|
return err
|
|
}
|
|
|
|
loginctlManager = manager
|
|
|
|
log.Info("Loginctl manager initialized")
|
|
return nil
|
|
}
|
|
|
|
func InitializeFreedeskManager() error {
|
|
manager, err := freedesktop.NewManager()
|
|
if err != nil {
|
|
log.Warnf("Failed to initialize freedesktop manager: %v", err)
|
|
return err
|
|
}
|
|
|
|
freedesktopManager = manager
|
|
|
|
log.Info("Freedesktop manager initialized")
|
|
return nil
|
|
}
|
|
|
|
func InitializeWaylandManager() error {
|
|
log.Info("Attempting to initialize Wayland gamma control...")
|
|
|
|
if wlContext == nil {
|
|
ctx, err := wlcontext.New()
|
|
if err != nil {
|
|
log.Errorf("Failed to create shared Wayland context: %v", err)
|
|
return err
|
|
}
|
|
wlContext = ctx
|
|
}
|
|
|
|
config := wayland.DefaultConfig()
|
|
manager, err := wayland.NewManager(wlContext.Display(), config)
|
|
if err != nil {
|
|
log.Errorf("Failed to initialize wayland manager: %v", err)
|
|
return err
|
|
}
|
|
|
|
waylandManager = manager
|
|
|
|
log.Info("Wayland gamma control initialized successfully")
|
|
return nil
|
|
}
|
|
|
|
func InitializeBluezManager() error {
|
|
manager, err := bluez.NewManager()
|
|
if err != nil {
|
|
log.Warnf("Failed to initialize bluez manager: %v", err)
|
|
return err
|
|
}
|
|
|
|
bluezManager = manager
|
|
|
|
log.Info("Bluez manager initialized")
|
|
return nil
|
|
}
|
|
|
|
func InitializeCupsManager() error {
|
|
manager, err := cups.NewManager()
|
|
if err != nil {
|
|
log.Warnf("Failed to initialize cups manager: %v", err)
|
|
return err
|
|
}
|
|
|
|
cupsManager = manager
|
|
|
|
log.Info("CUPS manager initialized")
|
|
return nil
|
|
}
|
|
|
|
func InitializeDwlManager() error {
|
|
log.Info("Attempting to initialize DWL IPC...")
|
|
|
|
if wlContext == nil {
|
|
ctx, err := wlcontext.New()
|
|
if err != nil {
|
|
log.Errorf("Failed to create shared Wayland context: %v", err)
|
|
return err
|
|
}
|
|
wlContext = ctx
|
|
}
|
|
|
|
manager, err := dwl.NewManager(wlContext.Display())
|
|
if err != nil {
|
|
log.Debug("Failed to initialize dwl manager: %v", err)
|
|
return err
|
|
}
|
|
|
|
dwlManager = manager
|
|
|
|
log.Info("DWL IPC initialized successfully")
|
|
return nil
|
|
}
|
|
|
|
func InitializeBrightnessManager() error {
|
|
manager, err := brightness.NewManager()
|
|
if err != nil {
|
|
log.Warnf("Failed to initialize brightness manager: %v", err)
|
|
return err
|
|
}
|
|
|
|
brightnessManager = manager
|
|
|
|
log.Info("Brightness manager initialized")
|
|
return nil
|
|
}
|
|
|
|
func InitializeExtWorkspaceManager() error {
|
|
log.Info("Attempting to initialize ExtWorkspace...")
|
|
|
|
if wlContext == nil {
|
|
ctx, err := wlcontext.New()
|
|
if err != nil {
|
|
log.Errorf("Failed to create shared Wayland context: %v", err)
|
|
return err
|
|
}
|
|
wlContext = ctx
|
|
}
|
|
|
|
manager, err := extworkspace.NewManager(wlContext.Display())
|
|
if err != nil {
|
|
log.Debug("Failed to initialize extworkspace manager: %v", err)
|
|
return err
|
|
}
|
|
|
|
extWorkspaceManager = manager
|
|
|
|
log.Info("ExtWorkspace initialized successfully")
|
|
return nil
|
|
}
|
|
|
|
func InitializeWlrOutputManager() error {
|
|
log.Info("Attempting to initialize WlrOutput management...")
|
|
|
|
if wlContext == nil {
|
|
ctx, err := wlcontext.New()
|
|
if err != nil {
|
|
log.Errorf("Failed to create shared Wayland context: %v", err)
|
|
return err
|
|
}
|
|
wlContext = ctx
|
|
}
|
|
|
|
manager, err := wlroutput.NewManager(wlContext.Display())
|
|
if err != nil {
|
|
log.Debug("Failed to initialize wlroutput manager: %v", err)
|
|
return err
|
|
}
|
|
|
|
wlrOutputManager = manager
|
|
|
|
log.Info("WlrOutput management initialized successfully")
|
|
return nil
|
|
}
|
|
|
|
func InitializeEvdevManager() error {
|
|
manager, err := evdev.InitializeManager()
|
|
if err != nil {
|
|
log.Warnf("Failed to initialize evdev manager: %v", err)
|
|
return err
|
|
}
|
|
|
|
evdevManager = manager
|
|
|
|
log.Info("Evdev manager initialized")
|
|
return nil
|
|
}
|
|
|
|
func handleConnection(conn net.Conn) {
|
|
defer conn.Close()
|
|
|
|
caps := getCapabilities()
|
|
capsData, _ := json.Marshal(caps)
|
|
conn.Write(capsData)
|
|
conn.Write([]byte("\n"))
|
|
scanner := bufio.NewScanner(conn)
|
|
for scanner.Scan() {
|
|
line := scanner.Bytes()
|
|
|
|
var req models.Request
|
|
if err := json.Unmarshal(line, &req); err != nil {
|
|
log.Warnf("handleConnection: Failed to unmarshal JSON: %v, line: %s", err, string(line))
|
|
models.RespondError(conn, 0, "invalid json")
|
|
continue
|
|
}
|
|
|
|
go RouteRequest(conn, req)
|
|
}
|
|
}
|
|
|
|
func getCapabilities() Capabilities {
|
|
caps := []string{"plugins"}
|
|
|
|
if networkManager != nil {
|
|
caps = append(caps, "network")
|
|
}
|
|
|
|
if loginctlManager != nil {
|
|
caps = append(caps, "loginctl")
|
|
}
|
|
|
|
if freedesktopManager != nil {
|
|
caps = append(caps, "freedesktop")
|
|
}
|
|
|
|
if waylandManager != nil {
|
|
caps = append(caps, "gamma")
|
|
}
|
|
|
|
if bluezManager != nil {
|
|
caps = append(caps, "bluetooth")
|
|
}
|
|
|
|
if cupsManager != nil {
|
|
caps = append(caps, "cups")
|
|
}
|
|
|
|
if dwlManager != nil {
|
|
caps = append(caps, "dwl")
|
|
}
|
|
|
|
if extWorkspaceAvailable.Load() {
|
|
caps = append(caps, "extworkspace")
|
|
}
|
|
|
|
if brightnessManager != nil {
|
|
caps = append(caps, "brightness")
|
|
}
|
|
|
|
if wlrOutputManager != nil {
|
|
caps = append(caps, "wlroutput")
|
|
}
|
|
|
|
if evdevManager != nil {
|
|
caps = append(caps, "evdev")
|
|
}
|
|
|
|
return Capabilities{Capabilities: caps}
|
|
}
|
|
|
|
func getServerInfo() ServerInfo {
|
|
caps := []string{"plugins"}
|
|
|
|
if networkManager != nil {
|
|
caps = append(caps, "network")
|
|
}
|
|
|
|
if loginctlManager != nil {
|
|
caps = append(caps, "loginctl")
|
|
}
|
|
|
|
if freedesktopManager != nil {
|
|
caps = append(caps, "freedesktop")
|
|
}
|
|
|
|
if waylandManager != nil {
|
|
caps = append(caps, "gamma")
|
|
}
|
|
|
|
if bluezManager != nil {
|
|
caps = append(caps, "bluetooth")
|
|
}
|
|
|
|
if cupsManager != nil {
|
|
caps = append(caps, "cups")
|
|
}
|
|
|
|
if dwlManager != nil {
|
|
caps = append(caps, "dwl")
|
|
}
|
|
|
|
if extWorkspaceAvailable.Load() {
|
|
caps = append(caps, "extworkspace")
|
|
}
|
|
|
|
if brightnessManager != nil {
|
|
caps = append(caps, "brightness")
|
|
}
|
|
|
|
if wlrOutputManager != nil {
|
|
caps = append(caps, "wlroutput")
|
|
}
|
|
|
|
if evdevManager != nil {
|
|
caps = append(caps, "evdev")
|
|
}
|
|
|
|
return ServerInfo{
|
|
APIVersion: APIVersion,
|
|
CLIVersion: CLIVersion,
|
|
Capabilities: caps,
|
|
}
|
|
}
|
|
|
|
func notifyCapabilityChange() {
|
|
info := getServerInfo()
|
|
capabilitySubscribers.Range(func(key string, ch chan ServerInfo) bool {
|
|
select {
|
|
case ch <- info:
|
|
default:
|
|
}
|
|
return true
|
|
})
|
|
}
|
|
|
|
func handleSubscribe(conn net.Conn, req models.Request) {
|
|
clientID := fmt.Sprintf("meta-client-%p", conn)
|
|
|
|
var services []string
|
|
if servicesParam, ok := req.Params["services"].([]interface{}); ok {
|
|
for _, s := range servicesParam {
|
|
if str, ok := s.(string); ok {
|
|
services = append(services, str)
|
|
}
|
|
}
|
|
}
|
|
|
|
if len(services) == 0 {
|
|
services = []string{"all"}
|
|
}
|
|
|
|
subscribeAll := false
|
|
for _, s := range services {
|
|
if s == "all" {
|
|
subscribeAll = true
|
|
break
|
|
}
|
|
}
|
|
|
|
var wg sync.WaitGroup
|
|
eventChan := make(chan ServiceEvent, 256)
|
|
stopChan := make(chan struct{})
|
|
|
|
capChan := make(chan ServerInfo, 64)
|
|
capabilitySubscribers.Store(clientID+"-capabilities", capChan)
|
|
|
|
wg.Add(1)
|
|
go func() {
|
|
defer wg.Done()
|
|
defer capabilitySubscribers.Delete(clientID + "-capabilities")
|
|
|
|
for {
|
|
select {
|
|
case info, ok := <-capChan:
|
|
if !ok {
|
|
return
|
|
}
|
|
select {
|
|
case eventChan <- ServiceEvent{Service: "server", Data: info}:
|
|
case <-stopChan:
|
|
return
|
|
}
|
|
case <-stopChan:
|
|
return
|
|
}
|
|
}
|
|
}()
|
|
|
|
shouldSubscribe := func(service string) bool {
|
|
if subscribeAll {
|
|
return true
|
|
}
|
|
for _, s := range services {
|
|
if s == service {
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|
|
|
|
if shouldSubscribe("network") && networkManager != nil {
|
|
wg.Add(1)
|
|
netChan := networkManager.Subscribe(clientID + "-network")
|
|
go func() {
|
|
defer wg.Done()
|
|
defer networkManager.Unsubscribe(clientID + "-network")
|
|
|
|
initialState := networkManager.GetState()
|
|
select {
|
|
case eventChan <- ServiceEvent{Service: "network", Data: initialState}:
|
|
case <-stopChan:
|
|
return
|
|
}
|
|
|
|
for {
|
|
select {
|
|
case state, ok := <-netChan:
|
|
if !ok {
|
|
return
|
|
}
|
|
select {
|
|
case eventChan <- ServiceEvent{Service: "network", Data: state}:
|
|
case <-stopChan:
|
|
return
|
|
}
|
|
case <-stopChan:
|
|
return
|
|
}
|
|
}
|
|
}()
|
|
}
|
|
|
|
if shouldSubscribe("network.credentials") && networkManager != nil {
|
|
wg.Add(1)
|
|
credChan := networkManager.SubscribeCredentials(clientID + "-credentials")
|
|
go func() {
|
|
defer wg.Done()
|
|
defer networkManager.UnsubscribeCredentials(clientID + "-credentials")
|
|
|
|
for {
|
|
select {
|
|
case prompt, ok := <-credChan:
|
|
if !ok {
|
|
return
|
|
}
|
|
select {
|
|
case eventChan <- ServiceEvent{Service: "network.credentials", Data: prompt}:
|
|
case <-stopChan:
|
|
return
|
|
}
|
|
case <-stopChan:
|
|
return
|
|
}
|
|
}
|
|
}()
|
|
}
|
|
|
|
if shouldSubscribe("loginctl") && loginctlManager != nil {
|
|
wg.Add(1)
|
|
loginChan := loginctlManager.Subscribe(clientID + "-loginctl")
|
|
go func() {
|
|
defer wg.Done()
|
|
defer loginctlManager.Unsubscribe(clientID + "-loginctl")
|
|
|
|
initialState := loginctlManager.GetState()
|
|
select {
|
|
case eventChan <- ServiceEvent{Service: "loginctl", Data: initialState}:
|
|
case <-stopChan:
|
|
return
|
|
}
|
|
|
|
for {
|
|
select {
|
|
case state, ok := <-loginChan:
|
|
if !ok {
|
|
return
|
|
}
|
|
select {
|
|
case eventChan <- ServiceEvent{Service: "loginctl", Data: state}:
|
|
case <-stopChan:
|
|
return
|
|
}
|
|
case <-stopChan:
|
|
return
|
|
}
|
|
}
|
|
}()
|
|
}
|
|
|
|
if shouldSubscribe("freedesktop") && freedesktopManager != nil {
|
|
wg.Add(1)
|
|
freedesktopChan := freedesktopManager.Subscribe(clientID + "-freedesktop")
|
|
go func() {
|
|
defer wg.Done()
|
|
defer freedesktopManager.Unsubscribe(clientID + "-freedesktop")
|
|
|
|
initialState := freedesktopManager.GetState()
|
|
select {
|
|
case eventChan <- ServiceEvent{Service: "freedesktop", Data: initialState}:
|
|
case <-stopChan:
|
|
return
|
|
}
|
|
|
|
for {
|
|
select {
|
|
case state, ok := <-freedesktopChan:
|
|
if !ok {
|
|
return
|
|
}
|
|
select {
|
|
case eventChan <- ServiceEvent{Service: "freedesktop", Data: state}:
|
|
case <-stopChan:
|
|
return
|
|
}
|
|
case <-stopChan:
|
|
return
|
|
}
|
|
}
|
|
}()
|
|
}
|
|
|
|
if shouldSubscribe("gamma") && waylandManager != nil {
|
|
wg.Add(1)
|
|
waylandChan := waylandManager.Subscribe(clientID + "-gamma")
|
|
go func() {
|
|
defer wg.Done()
|
|
defer waylandManager.Unsubscribe(clientID + "-gamma")
|
|
|
|
initialState := waylandManager.GetState()
|
|
select {
|
|
case eventChan <- ServiceEvent{Service: "gamma", Data: initialState}:
|
|
case <-stopChan:
|
|
return
|
|
}
|
|
|
|
for {
|
|
select {
|
|
case state, ok := <-waylandChan:
|
|
if !ok {
|
|
return
|
|
}
|
|
select {
|
|
case eventChan <- ServiceEvent{Service: "gamma", Data: state}:
|
|
case <-stopChan:
|
|
return
|
|
}
|
|
case <-stopChan:
|
|
return
|
|
}
|
|
}
|
|
}()
|
|
}
|
|
|
|
if shouldSubscribe("bluetooth") && bluezManager != nil {
|
|
wg.Add(1)
|
|
bluezChan := bluezManager.Subscribe(clientID + "-bluetooth")
|
|
go func() {
|
|
defer wg.Done()
|
|
defer bluezManager.Unsubscribe(clientID + "-bluetooth")
|
|
|
|
initialState := bluezManager.GetState()
|
|
select {
|
|
case eventChan <- ServiceEvent{Service: "bluetooth", Data: initialState}:
|
|
case <-stopChan:
|
|
return
|
|
}
|
|
|
|
for {
|
|
select {
|
|
case state, ok := <-bluezChan:
|
|
if !ok {
|
|
return
|
|
}
|
|
select {
|
|
case eventChan <- ServiceEvent{Service: "bluetooth", Data: state}:
|
|
case <-stopChan:
|
|
return
|
|
}
|
|
case <-stopChan:
|
|
return
|
|
}
|
|
}
|
|
}()
|
|
}
|
|
|
|
if shouldSubscribe("bluetooth.pairing") && bluezManager != nil {
|
|
wg.Add(1)
|
|
pairingChan := bluezManager.SubscribePairing(clientID + "-pairing")
|
|
go func() {
|
|
defer wg.Done()
|
|
defer bluezManager.UnsubscribePairing(clientID + "-pairing")
|
|
|
|
for {
|
|
select {
|
|
case prompt, ok := <-pairingChan:
|
|
if !ok {
|
|
return
|
|
}
|
|
select {
|
|
case eventChan <- ServiceEvent{Service: "bluetooth.pairing", Data: prompt}:
|
|
case <-stopChan:
|
|
return
|
|
}
|
|
case <-stopChan:
|
|
return
|
|
}
|
|
}
|
|
}()
|
|
}
|
|
|
|
if shouldSubscribe("cups") {
|
|
cupsSubscribers.Store(clientID+"-cups", true)
|
|
count := cupsSubscriberCount.Add(1)
|
|
|
|
if count == 1 {
|
|
if err := InitializeCupsManager(); err != nil {
|
|
log.Warnf("Failed to initialize CUPS manager for subscription: %v", err)
|
|
} else {
|
|
notifyCapabilityChange()
|
|
}
|
|
}
|
|
|
|
if cupsManager != nil {
|
|
wg.Add(1)
|
|
cupsChan := cupsManager.Subscribe(clientID + "-cups")
|
|
go func() {
|
|
defer wg.Done()
|
|
defer func() {
|
|
cupsManager.Unsubscribe(clientID + "-cups")
|
|
cupsSubscribers.Delete(clientID + "-cups")
|
|
count := cupsSubscriberCount.Add(-1)
|
|
|
|
if count == 0 {
|
|
log.Info("Last CUPS subscriber disconnected, shutting down CUPS manager")
|
|
if cupsManager != nil {
|
|
cupsManager.Close()
|
|
cupsManager = nil
|
|
notifyCapabilityChange()
|
|
}
|
|
}
|
|
}()
|
|
|
|
initialState := cupsManager.GetState()
|
|
select {
|
|
case eventChan <- ServiceEvent{Service: "cups", Data: initialState}:
|
|
case <-stopChan:
|
|
return
|
|
}
|
|
|
|
for {
|
|
select {
|
|
case state, ok := <-cupsChan:
|
|
if !ok {
|
|
return
|
|
}
|
|
select {
|
|
case eventChan <- ServiceEvent{Service: "cups", Data: state}:
|
|
case <-stopChan:
|
|
return
|
|
}
|
|
case <-stopChan:
|
|
return
|
|
}
|
|
}
|
|
}()
|
|
}
|
|
}
|
|
|
|
if shouldSubscribe("dwl") && dwlManager != nil {
|
|
wg.Add(1)
|
|
dwlChan := dwlManager.Subscribe(clientID + "-dwl")
|
|
go func() {
|
|
defer wg.Done()
|
|
defer dwlManager.Unsubscribe(clientID + "-dwl")
|
|
|
|
initialState := dwlManager.GetState()
|
|
select {
|
|
case eventChan <- ServiceEvent{Service: "dwl", Data: initialState}:
|
|
case <-stopChan:
|
|
return
|
|
}
|
|
|
|
for {
|
|
select {
|
|
case state, ok := <-dwlChan:
|
|
if !ok {
|
|
return
|
|
}
|
|
select {
|
|
case eventChan <- ServiceEvent{Service: "dwl", Data: state}:
|
|
case <-stopChan:
|
|
return
|
|
}
|
|
case <-stopChan:
|
|
return
|
|
}
|
|
}
|
|
}()
|
|
}
|
|
|
|
if shouldSubscribe("extworkspace") {
|
|
if extWorkspaceManager == nil && extWorkspaceAvailable.Load() {
|
|
extWorkspaceInitMutex.Lock()
|
|
if extWorkspaceManager == nil {
|
|
if err := InitializeExtWorkspaceManager(); err != nil {
|
|
log.Warnf("Failed to initialize ExtWorkspace manager for subscription: %v", err)
|
|
}
|
|
}
|
|
extWorkspaceInitMutex.Unlock()
|
|
}
|
|
|
|
if extWorkspaceManager != nil {
|
|
wg.Add(1)
|
|
extWorkspaceChan := extWorkspaceManager.Subscribe(clientID + "-extworkspace")
|
|
go func() {
|
|
defer wg.Done()
|
|
defer extWorkspaceManager.Unsubscribe(clientID + "-extworkspace")
|
|
|
|
initialState := extWorkspaceManager.GetState()
|
|
select {
|
|
case eventChan <- ServiceEvent{Service: "extworkspace", Data: initialState}:
|
|
case <-stopChan:
|
|
return
|
|
}
|
|
|
|
for {
|
|
select {
|
|
case state, ok := <-extWorkspaceChan:
|
|
if !ok {
|
|
return
|
|
}
|
|
select {
|
|
case eventChan <- ServiceEvent{Service: "extworkspace", Data: state}:
|
|
case <-stopChan:
|
|
return
|
|
}
|
|
case <-stopChan:
|
|
return
|
|
}
|
|
}
|
|
}()
|
|
}
|
|
}
|
|
|
|
if shouldSubscribe("brightness") && brightnessManager != nil {
|
|
wg.Add(2)
|
|
brightnessStateChan := brightnessManager.Subscribe(clientID + "-brightness-state")
|
|
brightnessUpdateChan := brightnessManager.SubscribeUpdates(clientID + "-brightness-updates")
|
|
|
|
go func() {
|
|
defer wg.Done()
|
|
defer brightnessManager.Unsubscribe(clientID + "-brightness-state")
|
|
|
|
initialState := brightnessManager.GetState()
|
|
select {
|
|
case eventChan <- ServiceEvent{Service: "brightness", Data: initialState}:
|
|
case <-stopChan:
|
|
return
|
|
}
|
|
|
|
for {
|
|
select {
|
|
case state, ok := <-brightnessStateChan:
|
|
if !ok {
|
|
return
|
|
}
|
|
select {
|
|
case eventChan <- ServiceEvent{Service: "brightness", Data: state}:
|
|
case <-stopChan:
|
|
return
|
|
}
|
|
case <-stopChan:
|
|
return
|
|
}
|
|
}
|
|
}()
|
|
|
|
go func() {
|
|
defer wg.Done()
|
|
defer brightnessManager.UnsubscribeUpdates(clientID + "-brightness-updates")
|
|
|
|
for {
|
|
select {
|
|
case update, ok := <-brightnessUpdateChan:
|
|
if !ok {
|
|
return
|
|
}
|
|
select {
|
|
case eventChan <- ServiceEvent{Service: "brightness.update", Data: update}:
|
|
case <-stopChan:
|
|
return
|
|
}
|
|
case <-stopChan:
|
|
return
|
|
}
|
|
}
|
|
}()
|
|
}
|
|
|
|
if shouldSubscribe("wlroutput") && wlrOutputManager != nil {
|
|
wg.Add(1)
|
|
wlrOutputChan := wlrOutputManager.Subscribe(clientID + "-wlroutput")
|
|
go func() {
|
|
defer wg.Done()
|
|
defer wlrOutputManager.Unsubscribe(clientID + "-wlroutput")
|
|
|
|
initialState := wlrOutputManager.GetState()
|
|
select {
|
|
case eventChan <- ServiceEvent{Service: "wlroutput", Data: initialState}:
|
|
case <-stopChan:
|
|
return
|
|
}
|
|
|
|
for {
|
|
select {
|
|
case state, ok := <-wlrOutputChan:
|
|
if !ok {
|
|
return
|
|
}
|
|
select {
|
|
case eventChan <- ServiceEvent{Service: "wlroutput", Data: state}:
|
|
case <-stopChan:
|
|
return
|
|
}
|
|
case <-stopChan:
|
|
return
|
|
}
|
|
}
|
|
}()
|
|
}
|
|
|
|
if shouldSubscribe("evdev") && evdevManager != nil {
|
|
wg.Add(1)
|
|
evdevChan := evdevManager.Subscribe(clientID + "-evdev")
|
|
go func() {
|
|
defer wg.Done()
|
|
defer evdevManager.Unsubscribe(clientID + "-evdev")
|
|
|
|
initialState := evdevManager.GetState()
|
|
select {
|
|
case eventChan <- ServiceEvent{Service: "evdev", Data: initialState}:
|
|
case <-stopChan:
|
|
return
|
|
}
|
|
|
|
for {
|
|
select {
|
|
case state, ok := <-evdevChan:
|
|
if !ok {
|
|
return
|
|
}
|
|
select {
|
|
case eventChan <- ServiceEvent{Service: "evdev", Data: state}:
|
|
case <-stopChan:
|
|
return
|
|
}
|
|
case <-stopChan:
|
|
return
|
|
}
|
|
}
|
|
}()
|
|
}
|
|
|
|
go func() {
|
|
wg.Wait()
|
|
close(eventChan)
|
|
}()
|
|
|
|
info := getServerInfo()
|
|
if err := json.NewEncoder(conn).Encode(models.Response[ServiceEvent]{
|
|
ID: req.ID,
|
|
Result: &ServiceEvent{Service: "server", Data: info},
|
|
}); err != nil {
|
|
close(stopChan)
|
|
return
|
|
}
|
|
|
|
for event := range eventChan {
|
|
if err := json.NewEncoder(conn).Encode(models.Response[ServiceEvent]{
|
|
ID: req.ID,
|
|
Result: &event,
|
|
}); err != nil {
|
|
close(stopChan)
|
|
return
|
|
}
|
|
}
|
|
}
|
|
|
|
func cleanupManagers() {
|
|
if networkManager != nil {
|
|
networkManager.Close()
|
|
}
|
|
if loginctlManager != nil {
|
|
loginctlManager.Close()
|
|
}
|
|
if freedesktopManager != nil {
|
|
freedesktopManager.Close()
|
|
}
|
|
if waylandManager != nil {
|
|
waylandManager.Close()
|
|
}
|
|
if bluezManager != nil {
|
|
bluezManager.Close()
|
|
}
|
|
if cupsManager != nil {
|
|
cupsManager.Close()
|
|
}
|
|
if dwlManager != nil {
|
|
dwlManager.Close()
|
|
}
|
|
if extWorkspaceManager != nil {
|
|
extWorkspaceManager.Close()
|
|
}
|
|
if brightnessManager != nil {
|
|
brightnessManager.Close()
|
|
}
|
|
if wlrOutputManager != nil {
|
|
wlrOutputManager.Close()
|
|
}
|
|
if evdevManager != nil {
|
|
evdevManager.Close()
|
|
}
|
|
if wlContext != nil {
|
|
wlContext.Close()
|
|
}
|
|
}
|
|
|
|
func Start(printDocs bool) error {
|
|
cleanupStaleSockets()
|
|
|
|
socketPath := GetSocketPath()
|
|
os.Remove(socketPath)
|
|
|
|
listener, err := net.Listen("unix", socketPath)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer listener.Close()
|
|
defer cleanupManagers()
|
|
|
|
log.Infof("DMS API Server listening on: %s", socketPath)
|
|
log.Infof("API Version: %d", APIVersion)
|
|
log.Info("Protocol: JSON over Unix socket")
|
|
log.Info("Request format: {\"id\": <any>, \"method\": \"...\", \"params\": {...}}")
|
|
log.Info("Response format: {\"id\": <any>, \"result\": {...}} or {\"id\": <any>, \"error\": \"...\"}")
|
|
log.Info("")
|
|
if printDocs {
|
|
log.Info("Available methods:")
|
|
log.Info(" ping - Test connection")
|
|
log.Info(" getServerInfo - Get server info (API version and capabilities)")
|
|
log.Info(" subscribe - Subscribe to multiple services (params: services [default: all])")
|
|
log.Info("Plugins:")
|
|
log.Info(" plugins.list - List all plugins")
|
|
log.Info(" plugins.listInstalled - List installed plugins")
|
|
log.Info(" plugins.install - Install plugin (params: name)")
|
|
log.Info(" plugins.uninstall - Uninstall plugin (params: name)")
|
|
log.Info(" plugins.update - Update plugin (params: name)")
|
|
log.Info(" plugins.search - Search plugins (params: query, category?, compositor?, capability?)")
|
|
log.Info("Network:")
|
|
log.Info(" network.getState - Get current network state")
|
|
log.Info(" network.wifi.scan - Scan for WiFi networks (params: device?)")
|
|
log.Info(" network.wifi.networks - Get WiFi network list")
|
|
log.Info(" network.wifi.connect - Connect to WiFi (params: ssid, password?, username?, device?, eapMethod?, phase2Auth?, caCertPath?, clientCertPath?, privateKeyPath?, useSystemCACerts?)")
|
|
log.Info(" network.wifi.disconnect - Disconnect WiFi (params: device?)")
|
|
log.Info(" network.wifi.forget - Forget network (params: ssid)")
|
|
log.Info(" network.wifi.toggle - Toggle WiFi radio")
|
|
log.Info(" network.wifi.enable - Enable WiFi")
|
|
log.Info(" network.wifi.disable - Disable WiFi")
|
|
log.Info(" network.wifi.setAutoconnect - Set network autoconnect (params: ssid, autoconnect)")
|
|
log.Info(" network.ethernet.connect - Connect Ethernet")
|
|
log.Info(" network.ethernet.connect.config - Connect Ethernet to a specific configuration")
|
|
log.Info(" network.ethernet.disconnect - Disconnect Ethernet")
|
|
log.Info(" network.vpn.profiles - List VPN profiles")
|
|
log.Info(" network.vpn.active - List active VPN connections")
|
|
log.Info(" network.vpn.connect - Connect VPN (params: uuidOrName|name|uuid, singleActive?)")
|
|
log.Info(" network.vpn.disconnect - Disconnect VPN (params: uuidOrName|name|uuid)")
|
|
log.Info(" network.vpn.disconnectAll - Disconnect all VPNs")
|
|
log.Info(" network.vpn.clearCredentials - Clear saved VPN credentials (params: uuidOrName|name|uuid)")
|
|
log.Info(" network.vpn.plugins - List available VPN plugins")
|
|
log.Info(" network.vpn.import - Import VPN from file (params: file|path, name?)")
|
|
log.Info(" network.vpn.getConfig - Get VPN configuration (params: uuid|name|uuidOrName)")
|
|
log.Info(" network.vpn.updateConfig - Update VPN configuration (params: uuid, name?, autoconnect?, data?)")
|
|
log.Info(" network.vpn.delete - Delete VPN connection (params: uuid|name|uuidOrName)")
|
|
log.Info(" network.preference.set - Set preference (params: preference [auto|wifi|ethernet])")
|
|
log.Info(" network.info - Get network info (params: ssid)")
|
|
log.Info(" network.credentials.submit - Submit credentials for prompt (params: token, secrets, save?)")
|
|
log.Info(" network.credentials.cancel - Cancel credential prompt (params: token)")
|
|
log.Info(" network.subscribe - Subscribe to network state changes (streaming)")
|
|
log.Info("Loginctl:")
|
|
log.Info(" loginctl.getState - Get current session state")
|
|
log.Info(" loginctl.lock - Lock session")
|
|
log.Info(" loginctl.unlock - Unlock session")
|
|
log.Info(" loginctl.activate - Activate session")
|
|
log.Info(" loginctl.setIdleHint - Set idle hint (params: idle)")
|
|
log.Info(" loginctl.setLockBeforeSuspend - Set lock before suspend (params: enabled)")
|
|
log.Info(" loginctl.setSleepInhibitorEnabled - Enable/disable sleep inhibitor (params: enabled)")
|
|
log.Info(" loginctl.lockerReady - Signal locker UI is ready (releases sleep inhibitor)")
|
|
log.Info(" loginctl.terminate - Terminate session")
|
|
log.Info(" loginctl.subscribe - Subscribe to session state changes (streaming)")
|
|
log.Info("Freedesktop:")
|
|
log.Info(" freedesktop.getState - Get accounts & settings state")
|
|
log.Info(" freedesktop.accounts.setIconFile - Set profile icon (params: path)")
|
|
log.Info(" freedesktop.accounts.setRealName - Set real name (params: name)")
|
|
log.Info(" freedesktop.accounts.setEmail - Set email (params: email)")
|
|
log.Info(" freedesktop.accounts.setLanguage - Set language (params: language)")
|
|
log.Info(" freedesktop.accounts.setLocation - Set location (params: location)")
|
|
log.Info(" freedesktop.accounts.getUserIconFile - Get user icon (params: username)")
|
|
log.Info(" freedesktop.settings.getColorScheme - Get color scheme")
|
|
log.Info(" freedesktop.settings.setIconTheme - Set icon theme (params: iconTheme)")
|
|
log.Info("Wayland:")
|
|
log.Info(" wayland.gamma.getState - Get current gamma control state")
|
|
log.Info(" wayland.gamma.setTemperature - Set temperature range (params: low, high)")
|
|
log.Info(" wayland.gamma.setLocation - Set location (params: latitude, longitude)")
|
|
log.Info(" wayland.gamma.setManualTimes - Set manual times (params: sunrise, sunset)")
|
|
log.Info(" wayland.gamma.setGamma - Set gamma value (params: gamma)")
|
|
log.Info(" wayland.gamma.setEnabled - Enable/disable gamma control (params: enabled)")
|
|
log.Info(" wayland.gamma.subscribe - Subscribe to gamma state changes (streaming)")
|
|
log.Info("Bluetooth:")
|
|
log.Info(" bluetooth.getState - Get current bluetooth state")
|
|
log.Info(" bluetooth.startDiscovery - Start device discovery")
|
|
log.Info(" bluetooth.stopDiscovery - Stop device discovery")
|
|
log.Info(" bluetooth.setPowered - Set adapter power state (params: powered)")
|
|
log.Info(" bluetooth.pair - Pair with device (params: device)")
|
|
log.Info(" bluetooth.connect - Connect to device (params: device)")
|
|
log.Info(" bluetooth.disconnect - Disconnect from device (params: device)")
|
|
log.Info(" bluetooth.remove - Remove/unpair device (params: device)")
|
|
log.Info(" bluetooth.trust - Trust device (params: device)")
|
|
log.Info(" bluetooth.untrust - Untrust device (params: device)")
|
|
log.Info(" bluetooth.pairing.submit - Submit pairing response (params: token, secrets?, accept?)")
|
|
log.Info(" bluetooth.pairing.cancel - Cancel pairing prompt (params: token)")
|
|
log.Info(" bluetooth.subscribe - Subscribe to bluetooth state changes (streaming)")
|
|
log.Info("CUPS:")
|
|
log.Info(" cups.getPrinters - Get printers list")
|
|
log.Info(" cups.getJobs - Get non-completed jobs list (params: printerName)")
|
|
log.Info(" cups.pausePrinter - Pause printer (params: printerName)")
|
|
log.Info(" cups.resumePrinter - Resume printer (params: printerName)")
|
|
log.Info(" cups.cancelJob - Cancel job (params: printerName, jobID)")
|
|
log.Info(" cups.purgeJobs - Cancel all jobs (params: printerName)")
|
|
log.Info("DWL:")
|
|
log.Info(" dwl.getState - Get current dwl state (tags, windows, layouts, keyboard)")
|
|
log.Info(" dwl.setTags - Set active tags (params: output, tagmask, toggleTagset)")
|
|
log.Info(" dwl.setClientTags - Set focused client tags (params: output, andTags, xorTags)")
|
|
log.Info(" dwl.setLayout - Set layout (params: output, index)")
|
|
log.Info(" dwl.subscribe - Subscribe to dwl state changes (streaming)")
|
|
log.Info(" Output state includes:")
|
|
log.Info(" - tags : Tag states (active, clients, focused)")
|
|
log.Info(" - layoutSymbol : Current layout name")
|
|
log.Info(" - title : Focused window title")
|
|
log.Info(" - appId : Focused window app ID")
|
|
log.Info(" - kbLayout : Current keyboard layout")
|
|
log.Info(" - keymode : Current keybind mode")
|
|
log.Info("ExtWorkspace:")
|
|
log.Info(" extworkspace.getState - Get current workspace state (groups, workspaces)")
|
|
log.Info(" extworkspace.activateWorkspace - Activate workspace (params: groupID, workspaceID)")
|
|
log.Info(" extworkspace.deactivateWorkspace - Deactivate workspace (params: groupID, workspaceID)")
|
|
log.Info(" extworkspace.removeWorkspace - Remove workspace (params: groupID, workspaceID)")
|
|
log.Info(" extworkspace.createWorkspace - Create workspace (params: groupID, name)")
|
|
log.Info(" extworkspace.subscribe - Subscribe to workspace state changes (streaming)")
|
|
log.Info("Brightness:")
|
|
log.Info(" brightness.getState - Get current brightness state for all devices")
|
|
log.Info(" brightness.setBrightness - Set device brightness (params: device, percent)")
|
|
log.Info(" brightness.increment - Increment device brightness (params: device, step?)")
|
|
log.Info(" brightness.decrement - Decrement device brightness (params: device, step?)")
|
|
log.Info(" brightness.rescan - Rescan for brightness devices (e.g., after plugging in monitor)")
|
|
log.Info(" brightness.subscribe - Subscribe to brightness state changes (streaming)")
|
|
log.Info(" Subscription events:")
|
|
log.Info(" - brightness : Full device list (on rescan, DDC discovery, device changes)")
|
|
log.Info(" - brightness.update: Single device update (on brightness change for efficiency)")
|
|
log.Info("WlrOutput:")
|
|
log.Info(" wlroutput.getState - Get current output configuration state")
|
|
log.Info(" wlroutput.applyConfiguration - Apply output configuration (params: heads)")
|
|
log.Info(" wlroutput.testConfiguration - Test output configuration without applying (params: heads)")
|
|
log.Info(" wlroutput.subscribe - Subscribe to output state changes (streaming)")
|
|
log.Info(" Head configuration params:")
|
|
log.Info(" - name : Output name (required)")
|
|
log.Info(" - enabled : Enable/disable output (required)")
|
|
log.Info(" - modeId : Mode ID from available modes (optional)")
|
|
log.Info(" - customMode : Custom mode {width, height, refresh} (optional)")
|
|
log.Info(" - position : Position {x, y} (optional)")
|
|
log.Info(" - transform : Transform value (optional)")
|
|
log.Info(" - scale : Scale value (optional)")
|
|
log.Info(" - adaptiveSync : Adaptive sync state (optional)")
|
|
log.Info("Evdev:")
|
|
log.Info(" evdev.getState - Get current evdev state (caps lock)")
|
|
log.Info(" evdev.subscribe - Subscribe to evdev state changes (streaming)")
|
|
log.Info("")
|
|
}
|
|
log.Info("Initializing managers...")
|
|
log.Info("")
|
|
|
|
go func() {
|
|
ticker := time.NewTicker(30 * time.Second)
|
|
defer ticker.Stop()
|
|
|
|
if err := InitializeNetworkManager(); err != nil {
|
|
log.Warnf("Network manager unavailable: %v", err)
|
|
} else {
|
|
notifyCapabilityChange()
|
|
return
|
|
}
|
|
|
|
for range ticker.C {
|
|
if networkManager != nil {
|
|
return
|
|
}
|
|
if err := InitializeNetworkManager(); err == nil {
|
|
log.Info("Network manager initialized")
|
|
notifyCapabilityChange()
|
|
return
|
|
}
|
|
}
|
|
}()
|
|
|
|
go func() {
|
|
if err := InitializeLoginctlManager(); err != nil {
|
|
log.Warnf("Loginctl manager unavailable: %v", err)
|
|
} else {
|
|
notifyCapabilityChange()
|
|
}
|
|
}()
|
|
|
|
go func() {
|
|
if err := InitializeFreedeskManager(); err != nil {
|
|
log.Warnf("Freedesktop manager unavailable: %v", err)
|
|
} else if freedesktopManager != nil {
|
|
freedesktopManager.NotifySubscribers()
|
|
notifyCapabilityChange()
|
|
}
|
|
}()
|
|
|
|
if err := InitializeWaylandManager(); err != nil {
|
|
log.Warnf("Wayland manager unavailable: %v", err)
|
|
}
|
|
|
|
go func() {
|
|
if err := InitializeBluezManager(); err != nil {
|
|
log.Warnf("Bluez manager unavailable: %v", err)
|
|
} else {
|
|
notifyCapabilityChange()
|
|
}
|
|
}()
|
|
|
|
if err := InitializeDwlManager(); err != nil {
|
|
log.Debugf("DWL manager unavailable: %v", err)
|
|
}
|
|
|
|
if extworkspace.CheckCapability() {
|
|
extWorkspaceAvailable.Store(true)
|
|
log.Info("ExtWorkspace capability detected and will be available on subscription")
|
|
} else {
|
|
log.Debug("ExtWorkspace capability not available")
|
|
extWorkspaceAvailable.Store(false)
|
|
}
|
|
|
|
if err := InitializeWlrOutputManager(); err != nil {
|
|
log.Debugf("WlrOutput manager unavailable: %v", err)
|
|
}
|
|
|
|
fatalErrChan := make(chan error, 1)
|
|
if wlrOutputManager != nil {
|
|
go func() {
|
|
err := <-wlrOutputManager.FatalError()
|
|
fatalErrChan <- fmt.Errorf("WlrOutput fatal error: %w", err)
|
|
}()
|
|
}
|
|
if wlContext != nil {
|
|
go func() {
|
|
err := <-wlContext.FatalError()
|
|
fatalErrChan <- fmt.Errorf("Wayland context fatal error: %w", err)
|
|
}()
|
|
}
|
|
|
|
go func() {
|
|
if err := InitializeBrightnessManager(); err != nil {
|
|
log.Warnf("Brightness manager unavailable: %v", err)
|
|
} else {
|
|
notifyCapabilityChange()
|
|
}
|
|
}()
|
|
|
|
go func() {
|
|
if err := InitializeEvdevManager(); err != nil {
|
|
log.Debugf("Evdev manager unavailable: %v", err)
|
|
} else {
|
|
notifyCapabilityChange()
|
|
}
|
|
}()
|
|
|
|
if wlContext != nil {
|
|
wlContext.Start()
|
|
log.Info("Wayland event dispatcher started")
|
|
}
|
|
|
|
log.Info("")
|
|
log.Infof("Ready! Capabilities: %v", getCapabilities().Capabilities)
|
|
|
|
listenerErrChan := make(chan error, 1)
|
|
|
|
go func() {
|
|
for {
|
|
conn, err := listener.Accept()
|
|
if err != nil {
|
|
listenerErrChan <- err
|
|
return
|
|
}
|
|
go handleConnection(conn)
|
|
}
|
|
}()
|
|
|
|
select {
|
|
case err := <-listenerErrChan:
|
|
return err
|
|
case err := <-fatalErrChan:
|
|
return err
|
|
}
|
|
}
|