| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171 | package mdnsimport (	"context"	"errors"	"net"	"strings"	"time"	"golang.org/x/net/ipv4"	"golib/v2/pkg/mdns")const (	DefaultTimout = 3 * time.Second)type Server struct {	Name    []string	Address *net.UDPAddr	server *mdns.Conn}func (s *Server) Close() error {	return s.server.Close()}func (s *Server) ListenAndServe() error {	conn, err := net.ListenUDP(mdns.NetType, s.Address)	if err != nil {		return err	}	cfg := &mdns.Config{		LocalNames: s.Name,	}	s.server, err = mdns.Server(ipv4.NewPacketConn(conn), cfg)	if err != nil {		return err	}	select {}}func ListenAndServe(name string) error {	return ListenAndServeNames([]string{name})}func ListenAndServeNames(name []string) error {	server := &Server{		Name:    name,		Address: mdns.Address,	}	return server.ListenAndServe()}type Handler func(name string, addr net.IP)type Client struct {	Name    []string	Address *net.UDPAddr	Timout  time.Duration	Handle  Handler	server *mdns.Conn}func (c *Client) initServer() error {	if c.server != nil {		return nil	}	conn, err := net.ListenUDP(mdns.NetType, c.Address)	if err != nil {		return err	}	if c.Timout <= 0 {		c.Timout = DefaultTimout	}	c.server, err = mdns.Server(ipv4.NewPacketConn(conn), &mdns.Config{QueryInterval: c.Timout})	if err != nil {		return err	}	return nil}func (c *Client) Lookup() ([]net.IP, error) {	if err := c.initServer(); err != nil {		return nil, err	}	ips := make([]net.IP, 0, len(c.Name))	for _, name := range c.Name {		ctx, cancel := context.WithTimeout(context.Background(), c.Timout)		var src net.Addr		_, src, err := c.server.Query(ctx, name)		cancel()		if err != nil {			continue // 忽略错误		}		ips = append(ips, src.(*net.IPAddr).IP)	}	return ips, nil}func (c *Client) LookupWithName() (map[string]net.IP, error) {	if err := c.initServer(); err != nil {		return nil, err	}	ips := make(map[string]net.IP)	for _, name := range c.Name {		ctx, cancel := context.WithTimeout(context.Background(), c.Timout)		answer, src, err := c.server.Query(ctx, name)		cancel()		if err != nil {			return nil, err		}		ips[mdns.UnFqdn(answer.Name.String())] = src.(*net.IPAddr).IP	}	return ips, nil}func (c *Client) ListenAndServe() error {	if err := c.initServer(); err != nil {		return err	}	timer := time.NewTimer(c.Timout)	for {		select {		case <-timer.C:			for _, name := range c.Name {				go func(name string) {					ctx, cancel := context.WithTimeout(context.Background(), c.Timout)					answer, src, err := c.server.Query(ctx, name)					cancel()					if err != nil {						return					}					if c.Handle != nil {						c.Handle(mdns.UnFqdn(answer.Name.String()), src.(*net.IPAddr).IP)					}				}(name)			}			timer.Reset(c.Timout)		}	}}func Lookup(name string) (net.IP, error) {	ips, err := Lookups([]string{name})	if err != nil {		return nil, err	}	if len(ips) > 0 {		return ips[0], nil	}	return nil, errors.New("not found")}func Lookups(name []string) ([]net.IP, error) {	client := &Client{		Name:    name,		Address: mdns.Address,	}	return client.Lookup()}func Fqdn(name string) string {	return UnFqdn(name) + ".local"}func UnFqdn(name string) string {	return strings.TrimSuffix(name, ".local")}
 |