455 lines
7.9 KiB
Go
455 lines
7.9 KiB
Go
// Package mdns is a multicast dns registry
|
|
package registry
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"net"
|
|
"strconv"
|
|
"strings"
|
|
"sync"
|
|
"time"
|
|
|
|
"github.com/google/uuid"
|
|
"github.com/micro/mdns"
|
|
)
|
|
|
|
var (
|
|
// use a .micro domain rather than .local
|
|
mdnsDomain = "micro"
|
|
)
|
|
|
|
type mdnsTxt struct {
|
|
Service string
|
|
Version string
|
|
Endpoints []*Endpoint
|
|
Metadata map[string]string
|
|
}
|
|
|
|
type mdnsEntry struct {
|
|
id string
|
|
node *mdns.Server
|
|
}
|
|
|
|
type mdnsRegistry struct {
|
|
opts Options
|
|
// the mdns domain
|
|
domain string
|
|
|
|
sync.Mutex
|
|
services map[string][]*mdnsEntry
|
|
|
|
mtx sync.RWMutex
|
|
|
|
// watchers
|
|
watchers map[string]*mdnsWatcher
|
|
|
|
// listener
|
|
listener chan *mdns.ServiceEntry
|
|
}
|
|
|
|
func newRegistry(opts ...Option) Registry {
|
|
options := Options{
|
|
Context: context.Background(),
|
|
Timeout: time.Millisecond * 100,
|
|
}
|
|
|
|
for _, o := range opts {
|
|
o(&options)
|
|
}
|
|
|
|
// set the domain
|
|
domain := mdnsDomain
|
|
|
|
d, ok := options.Context.Value("mdns.domain").(string)
|
|
if ok {
|
|
domain = d
|
|
}
|
|
|
|
return &mdnsRegistry{
|
|
opts: options,
|
|
domain: domain,
|
|
services: make(map[string][]*mdnsEntry),
|
|
watchers: make(map[string]*mdnsWatcher),
|
|
}
|
|
}
|
|
|
|
func (m *mdnsRegistry) Init(opts ...Option) error {
|
|
for _, o := range opts {
|
|
o(&m.opts)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (m *mdnsRegistry) Options() Options {
|
|
return m.opts
|
|
}
|
|
|
|
func (m *mdnsRegistry) Register(service *Service, opts ...RegisterOption) error {
|
|
m.Lock()
|
|
defer m.Unlock()
|
|
|
|
entries, ok := m.services[service.Name]
|
|
// first entry, create wildcard used for list queries
|
|
if !ok {
|
|
s, err := mdns.NewMDNSService(
|
|
service.Name,
|
|
"_services",
|
|
m.domain+".",
|
|
"",
|
|
9999,
|
|
[]net.IP{net.ParseIP("0.0.0.0")},
|
|
nil,
|
|
)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
srv, err := mdns.NewServer(&mdns.Config{Zone: &mdns.DNSSDService{s}})
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// append the wildcard entry
|
|
entries = append(entries, &mdnsEntry{id: "*", node: srv})
|
|
}
|
|
|
|
var gerr error
|
|
|
|
for _, node := range service.Nodes {
|
|
var seen bool
|
|
var e *mdnsEntry
|
|
|
|
for _, entry := range entries {
|
|
if node.Id == entry.id {
|
|
seen = true
|
|
e = entry
|
|
break
|
|
}
|
|
}
|
|
|
|
// already registered, continue
|
|
if seen {
|
|
continue
|
|
// doesn't exist
|
|
} else {
|
|
e = &mdnsEntry{}
|
|
}
|
|
|
|
txt, err := encode(&mdnsTxt{
|
|
Service: service.Name,
|
|
Version: service.Version,
|
|
Endpoints: service.Endpoints,
|
|
Metadata: node.Metadata,
|
|
})
|
|
|
|
if err != nil {
|
|
gerr = err
|
|
continue
|
|
}
|
|
|
|
//
|
|
host, pt, err := net.SplitHostPort(node.Address)
|
|
if err != nil {
|
|
gerr = err
|
|
continue
|
|
}
|
|
port, _ := strconv.Atoi(pt)
|
|
|
|
// we got here, new node
|
|
s, err := mdns.NewMDNSService(
|
|
node.Id,
|
|
service.Name,
|
|
m.domain+".",
|
|
"",
|
|
port,
|
|
[]net.IP{net.ParseIP(host)},
|
|
txt,
|
|
)
|
|
if err != nil {
|
|
gerr = err
|
|
continue
|
|
}
|
|
|
|
srv, err := mdns.NewServer(&mdns.Config{Zone: s})
|
|
if err != nil {
|
|
gerr = err
|
|
continue
|
|
}
|
|
|
|
e.id = node.Id
|
|
e.node = srv
|
|
entries = append(entries, e)
|
|
}
|
|
|
|
// save
|
|
m.services[service.Name] = entries
|
|
|
|
return gerr
|
|
}
|
|
|
|
func (m *mdnsRegistry) Deregister(service *Service) error {
|
|
m.Lock()
|
|
defer m.Unlock()
|
|
|
|
var newEntries []*mdnsEntry
|
|
|
|
// loop existing entries, check if any match, shutdown those that do
|
|
for _, entry := range m.services[service.Name] {
|
|
var remove bool
|
|
|
|
for _, node := range service.Nodes {
|
|
if node.Id == entry.id {
|
|
entry.node.Shutdown()
|
|
remove = true
|
|
break
|
|
}
|
|
}
|
|
|
|
// keep it?
|
|
if !remove {
|
|
newEntries = append(newEntries, entry)
|
|
}
|
|
}
|
|
|
|
// last entry is the wildcard for list queries. Remove it.
|
|
if len(newEntries) == 1 && newEntries[0].id == "*" {
|
|
newEntries[0].node.Shutdown()
|
|
delete(m.services, service.Name)
|
|
} else {
|
|
m.services[service.Name] = newEntries
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (m *mdnsRegistry) GetService(service string) ([]*Service, error) {
|
|
serviceMap := make(map[string]*Service)
|
|
entries := make(chan *mdns.ServiceEntry, 10)
|
|
done := make(chan bool)
|
|
|
|
p := mdns.DefaultParams(service)
|
|
// set context with timeout
|
|
p.Context, _ = context.WithTimeout(context.Background(), m.opts.Timeout)
|
|
// set entries channel
|
|
p.Entries = entries
|
|
// set the domain
|
|
p.Domain = m.domain
|
|
|
|
go func() {
|
|
for {
|
|
select {
|
|
case e := <-entries:
|
|
// list record so skip
|
|
if p.Service == "_services" {
|
|
continue
|
|
}
|
|
if p.Domain != m.domain {
|
|
continue
|
|
}
|
|
if e.TTL == 0 {
|
|
continue
|
|
}
|
|
|
|
txt, err := decode(e.InfoFields)
|
|
if err != nil {
|
|
continue
|
|
}
|
|
|
|
if txt.Service != service {
|
|
continue
|
|
}
|
|
|
|
s, ok := serviceMap[txt.Version]
|
|
if !ok {
|
|
s = &Service{
|
|
Name: txt.Service,
|
|
Version: txt.Version,
|
|
Endpoints: txt.Endpoints,
|
|
}
|
|
}
|
|
|
|
s.Nodes = append(s.Nodes, &Node{
|
|
Id: strings.TrimSuffix(e.Name, "."+p.Service+"."+p.Domain+"."),
|
|
Address: fmt.Sprintf("%s:%d", e.AddrV4.String(), e.Port),
|
|
Metadata: txt.Metadata,
|
|
})
|
|
|
|
serviceMap[txt.Version] = s
|
|
case <-p.Context.Done():
|
|
close(done)
|
|
return
|
|
}
|
|
}
|
|
}()
|
|
|
|
// execute the query
|
|
if err := mdns.Query(p); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// wait for completion
|
|
<-done
|
|
|
|
// create list and return
|
|
services := make([]*Service, 0, len(serviceMap))
|
|
|
|
for _, service := range serviceMap {
|
|
services = append(services, service)
|
|
}
|
|
|
|
return services, nil
|
|
}
|
|
|
|
func (m *mdnsRegistry) ListServices() ([]*Service, error) {
|
|
serviceMap := make(map[string]bool)
|
|
entries := make(chan *mdns.ServiceEntry, 10)
|
|
done := make(chan bool)
|
|
|
|
p := mdns.DefaultParams("_services")
|
|
// set context with timeout
|
|
p.Context, _ = context.WithTimeout(context.Background(), m.opts.Timeout)
|
|
// set entries channel
|
|
p.Entries = entries
|
|
// set domain
|
|
p.Domain = m.domain
|
|
|
|
var services []*Service
|
|
|
|
go func() {
|
|
for {
|
|
select {
|
|
case e := <-entries:
|
|
if e.TTL == 0 {
|
|
continue
|
|
}
|
|
if !strings.HasSuffix(e.Name, p.Domain+".") {
|
|
continue
|
|
}
|
|
name := strings.TrimSuffix(e.Name, "."+p.Service+"."+p.Domain+".")
|
|
if !serviceMap[name] {
|
|
serviceMap[name] = true
|
|
services = append(services, &Service{Name: name})
|
|
}
|
|
case <-p.Context.Done():
|
|
close(done)
|
|
return
|
|
}
|
|
}
|
|
}()
|
|
|
|
// execute query
|
|
if err := mdns.Query(p); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// wait till done
|
|
<-done
|
|
|
|
return services, nil
|
|
}
|
|
|
|
func (m *mdnsRegistry) Watch(opts ...WatchOption) (Watcher, error) {
|
|
var wo WatchOptions
|
|
for _, o := range opts {
|
|
o(&wo)
|
|
}
|
|
|
|
md := &mdnsWatcher{
|
|
id: uuid.New().String(),
|
|
wo: wo,
|
|
ch: make(chan *mdns.ServiceEntry, 32),
|
|
exit: make(chan struct{}),
|
|
domain: m.domain,
|
|
registry: m,
|
|
}
|
|
|
|
m.mtx.Lock()
|
|
defer m.mtx.Unlock()
|
|
|
|
// save the watcher
|
|
m.watchers[md.id] = md
|
|
|
|
// check of the listener exists
|
|
if m.listener != nil {
|
|
return md, nil
|
|
}
|
|
|
|
// start the listener
|
|
go func() {
|
|
// go to infinity
|
|
for {
|
|
m.mtx.Lock()
|
|
|
|
// just return if there are no watchers
|
|
if len(m.watchers) == 0 {
|
|
m.listener = nil
|
|
m.mtx.Unlock()
|
|
return
|
|
}
|
|
|
|
// check existing listener
|
|
if m.listener != nil {
|
|
m.mtx.Unlock()
|
|
return
|
|
}
|
|
|
|
// reset the listener
|
|
exit := make(chan struct{})
|
|
ch := make(chan *mdns.ServiceEntry, 32)
|
|
m.listener = ch
|
|
|
|
m.mtx.Unlock()
|
|
|
|
// send messages to the watchers
|
|
go func() {
|
|
send := func(w *mdnsWatcher, e *mdns.ServiceEntry) {
|
|
select {
|
|
case w.ch <- e:
|
|
default:
|
|
}
|
|
}
|
|
|
|
for {
|
|
select {
|
|
case <-exit:
|
|
return
|
|
case e, ok := <-ch:
|
|
if !ok {
|
|
return
|
|
}
|
|
m.mtx.RLock()
|
|
// send service entry to all watchers
|
|
for _, w := range m.watchers {
|
|
send(w, e)
|
|
}
|
|
m.mtx.RUnlock()
|
|
}
|
|
}
|
|
|
|
}()
|
|
|
|
// start listening, blocking call
|
|
mdns.Listen(ch, exit)
|
|
|
|
// mdns.Listen has unblocked
|
|
// kill the saved listener
|
|
m.mtx.Lock()
|
|
m.listener = nil
|
|
close(ch)
|
|
m.mtx.Unlock()
|
|
}
|
|
}()
|
|
|
|
return md, nil
|
|
}
|
|
|
|
func (m *mdnsRegistry) String() string {
|
|
return "mdns"
|
|
}
|
|
|
|
// NewRegistry returns a new default registry which is mdns
|
|
func NewRegistry(opts ...Option) Registry {
|
|
return newRegistry(opts...)
|
|
}
|