67 lines
1.6 KiB
Go
67 lines
1.6 KiB
Go
// Copyright 2024 Joshua Rich <joshua.rich@gmail.com>.
|
|
// SPDX-License-Identifier: MIT
|
|
|
|
package hass
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"log/slog"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/grandcat/zeroconf"
|
|
|
|
"github.com/joshuar/go-hass-agent/internal/logging"
|
|
)
|
|
|
|
const (
|
|
haDiscoveryTimeout = 5 * time.Second
|
|
)
|
|
|
|
// FindServers is a helper function to generate a list of Home Assistant servers
|
|
// via local network auto-discovery.
|
|
func FindServers(ctx context.Context) ([]string, error) {
|
|
var serverList []string
|
|
|
|
resolver, err := zeroconf.NewResolver(nil)
|
|
if err != nil {
|
|
return serverList, fmt.Errorf("failed to initialize resolver: %w", err)
|
|
}
|
|
|
|
entries := make(chan *zeroconf.ServiceEntry)
|
|
|
|
go func(results <-chan *zeroconf.ServiceEntry) {
|
|
for entry := range results {
|
|
var server string
|
|
|
|
for _, t := range entry.Text {
|
|
if value, found := strings.CutPrefix(t, "base_url="); found {
|
|
server = value
|
|
}
|
|
}
|
|
|
|
if server != "" {
|
|
serverList = append(serverList, server)
|
|
} else {
|
|
logging.FromContext(ctx).Log(ctx, logging.LevelTrace,
|
|
"Found a server malformed server, will not use.", slog.String("server", entry.HostName))
|
|
}
|
|
}
|
|
}(entries)
|
|
|
|
logging.FromContext(ctx).Info("Looking for Home Assistant servers on the local network...")
|
|
|
|
searchCtx, searchCancel := context.WithTimeout(ctx, haDiscoveryTimeout)
|
|
defer searchCancel()
|
|
|
|
err = resolver.Browse(searchCtx, "_home-assistant._tcp", "local.", entries)
|
|
if err != nil {
|
|
return serverList, fmt.Errorf("could not start search for Home Assistant servers: %w", err)
|
|
}
|
|
|
|
<-searchCtx.Done()
|
|
|
|
return serverList, nil
|
|
}
|