mirror of
https://github.com/AvengeMedia/DankMaterialShell.git
synced 2026-04-04 04:42:05 -04:00
244 lines
5.7 KiB
Go
244 lines
5.7 KiB
Go
package geolocation
|
|
|
|
import (
|
|
"fmt"
|
|
"sync"
|
|
|
|
"github.com/AvengeMedia/DankMaterialShell/core/internal/log"
|
|
"github.com/AvengeMedia/DankMaterialShell/core/pkg/dbusutil"
|
|
"github.com/AvengeMedia/DankMaterialShell/core/pkg/syncmap"
|
|
"github.com/godbus/dbus/v5"
|
|
)
|
|
|
|
const (
|
|
dbusGeoClueService = "org.freedesktop.GeoClue2"
|
|
dbusGeoCluePath = "/org/freedesktop/GeoClue2"
|
|
dbusGeoClueInterface = dbusGeoClueService
|
|
|
|
dbusGeoClueManagerPath = dbusGeoCluePath + "/Manager"
|
|
dbusGeoClueManagerInterface = dbusGeoClueInterface + ".Manager"
|
|
dbusGeoClueManagerGetClient = dbusGeoClueManagerInterface + ".GetClient"
|
|
|
|
dbusGeoClueClientInterface = dbusGeoClueInterface + ".Client"
|
|
dbusGeoClueClientDesktopId = dbusGeoClueClientInterface + ".DesktopId"
|
|
dbusGeoClueClientTimeThreshold = dbusGeoClueClientInterface + ".TimeThreshold"
|
|
dbusGeoClueClientTimeStart = dbusGeoClueClientInterface + ".Start"
|
|
dbusGeoClueClientTimeStop = dbusGeoClueClientInterface + ".Stop"
|
|
dbusGeoClueClientLocationUpdated = dbusGeoClueClientInterface + ".LocationUpdated"
|
|
|
|
dbusGeoClueLocationInterface = dbusGeoClueInterface + ".Location"
|
|
dbusGeoClueLocationLatitude = dbusGeoClueLocationInterface + ".Latitude"
|
|
dbusGeoClueLocationLongitude = dbusGeoClueLocationInterface + ".Longitude"
|
|
)
|
|
|
|
type GeoClueClient struct {
|
|
currLocation *Location
|
|
locationMutex sync.RWMutex
|
|
|
|
dbusConn *dbus.Conn
|
|
clientPath dbus.ObjectPath
|
|
signals chan *dbus.Signal
|
|
|
|
stopChan chan struct{}
|
|
sigWG sync.WaitGroup
|
|
|
|
subscribers syncmap.Map[string, chan Location]
|
|
}
|
|
|
|
func newGeoClueClient() (*GeoClueClient, error) {
|
|
dbusConn, err := dbus.ConnectSystemBus()
|
|
if err != nil {
|
|
return nil, fmt.Errorf("system bus connection failed: %w", err)
|
|
}
|
|
|
|
c := &GeoClueClient{
|
|
dbusConn: dbusConn,
|
|
stopChan: make(chan struct{}),
|
|
signals: make(chan *dbus.Signal, 256),
|
|
|
|
currLocation: &Location{
|
|
Latitude: 0.0,
|
|
Longitude: 0.0,
|
|
},
|
|
}
|
|
|
|
if err := c.setupClient(); err != nil {
|
|
dbusConn.Close()
|
|
return nil, err
|
|
}
|
|
|
|
if err := c.startSignalPump(); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return c, nil
|
|
}
|
|
|
|
func (c *GeoClueClient) Close() {
|
|
close(c.stopChan)
|
|
|
|
c.sigWG.Wait()
|
|
|
|
if c.signals != nil {
|
|
c.dbusConn.RemoveSignal(c.signals)
|
|
close(c.signals)
|
|
}
|
|
|
|
c.subscribers.Range(func(key string, ch chan Location) bool {
|
|
close(ch)
|
|
c.subscribers.Delete(key)
|
|
return true
|
|
})
|
|
|
|
if c.dbusConn != nil {
|
|
c.dbusConn.Close()
|
|
}
|
|
}
|
|
|
|
func (c *GeoClueClient) Subscribe(id string) chan Location {
|
|
ch := make(chan Location, 64)
|
|
c.subscribers.Store(id, ch)
|
|
return ch
|
|
}
|
|
|
|
func (c *GeoClueClient) Unsubscribe(id string) {
|
|
if ch, ok := c.subscribers.LoadAndDelete(id); ok {
|
|
close(ch)
|
|
}
|
|
}
|
|
|
|
func (c *GeoClueClient) setupClient() error {
|
|
managerObj := c.dbusConn.Object(dbusGeoClueService, dbusGeoClueManagerPath)
|
|
|
|
if err := managerObj.Call(dbusGeoClueManagerGetClient, 0).Store(&c.clientPath); err != nil {
|
|
return fmt.Errorf("failed to create GeoClue2 client: %w", err)
|
|
}
|
|
|
|
clientObj := c.dbusConn.Object(dbusGeoClueService, c.clientPath)
|
|
if err := clientObj.SetProperty(dbusGeoClueClientDesktopId, "dms"); err != nil {
|
|
return fmt.Errorf("failed to set desktop ID: %w", err)
|
|
}
|
|
|
|
if err := clientObj.SetProperty(dbusGeoClueClientTimeThreshold, uint(10)); err != nil {
|
|
return fmt.Errorf("failed to set time threshold: %w", err)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (c *GeoClueClient) startSignalPump() error {
|
|
c.dbusConn.Signal(c.signals)
|
|
|
|
if err := c.dbusConn.AddMatchSignal(
|
|
dbus.WithMatchObjectPath(c.clientPath),
|
|
dbus.WithMatchInterface(dbusGeoClueClientInterface),
|
|
dbus.WithMatchSender(dbusGeoClueClientLocationUpdated),
|
|
); err != nil {
|
|
return err
|
|
}
|
|
|
|
c.sigWG.Add(1)
|
|
go func() {
|
|
defer c.sigWG.Done()
|
|
|
|
clientObj := c.dbusConn.Object(dbusGeoClueService, c.clientPath)
|
|
clientObj.Call(dbusGeoClueClientTimeStart, 0)
|
|
defer clientObj.Call(dbusGeoClueClientTimeStop, 0)
|
|
|
|
for {
|
|
select {
|
|
case <-c.stopChan:
|
|
return
|
|
case sig, ok := <-c.signals:
|
|
if !ok {
|
|
return
|
|
}
|
|
if sig == nil {
|
|
continue
|
|
}
|
|
|
|
c.handleSignal(sig)
|
|
}
|
|
}
|
|
}()
|
|
|
|
return nil
|
|
}
|
|
|
|
func (c *GeoClueClient) handleSignal(sig *dbus.Signal) {
|
|
switch sig.Name {
|
|
case dbusGeoClueClientLocationUpdated:
|
|
if len(sig.Body) != 2 {
|
|
return
|
|
}
|
|
|
|
newLocationPath, ok := sig.Body[1].(dbus.ObjectPath)
|
|
if !ok {
|
|
return
|
|
}
|
|
|
|
if err := c.handleLocationUpdated(newLocationPath); err != nil {
|
|
log.Warn("GeoClue: Failed to handle location update: %v", err)
|
|
return
|
|
}
|
|
}
|
|
}
|
|
|
|
func (c *GeoClueClient) handleLocationUpdated(path dbus.ObjectPath) error {
|
|
locationObj := c.dbusConn.Object(dbusGeoClueService, path)
|
|
|
|
lat, err := locationObj.GetProperty(dbusGeoClueLocationLatitude)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
long, err := locationObj.GetProperty(dbusGeoClueLocationLongitude)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
c.locationMutex.Lock()
|
|
c.currLocation.Latitude = dbusutil.AsOr(lat, 0.0)
|
|
c.currLocation.Longitude = dbusutil.AsOr(long, 0.0)
|
|
c.locationMutex.Unlock()
|
|
|
|
c.notifySubscribers()
|
|
return nil
|
|
}
|
|
|
|
func (c *GeoClueClient) notifySubscribers() {
|
|
currentLocation, err := c.GetLocation()
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
c.subscribers.Range(func(key string, ch chan Location) bool {
|
|
select {
|
|
case ch <- currentLocation:
|
|
default:
|
|
log.Warn("GeoClue: subscriber channel full, dropping update")
|
|
}
|
|
return true
|
|
})
|
|
}
|
|
|
|
func (c *GeoClueClient) SeedLocation(loc Location) {
|
|
c.locationMutex.Lock()
|
|
defer c.locationMutex.Unlock()
|
|
c.currLocation.Latitude = loc.Latitude
|
|
c.currLocation.Longitude = loc.Longitude
|
|
}
|
|
|
|
func (c *GeoClueClient) GetLocation() (Location, error) {
|
|
c.locationMutex.RLock()
|
|
defer c.locationMutex.RUnlock()
|
|
if c.currLocation == nil {
|
|
return Location{
|
|
Latitude: 0.0,
|
|
Longitude: 0.0,
|
|
}, nil
|
|
}
|
|
stateCopy := *c.currLocation
|
|
return stateCopy, nil
|
|
}
|