Documentation
¶
Overview ¶
Package zeroconf is a pure Golang library that employs Multicast DNS-SD for browsing and resolving services in your network and registering own services in the local network.
It basically implements aspects of the standards RFC 6762 (mDNS) and RFC 6763 (DNS-SD). Though it does not support all requirements yet, the aim is to provide a complient solution in the long-term with the community.
By now, it should be compatible to [Avahi](http://avahi.org/) (tested) and Apple's Bonjour (untested). Should work in the most office, home and private environments.
Index ¶
- Constants
- func Browse(ctx context.Context, service, domain string, ...) error
- func Lookup(ctx context.Context, instance, service, domain string, ...) error
- func NewConnectionFactory() api.ConnectionFactory
- func NewInterfaceProvider() api.InterfaceProvider
- type Client
- type ClientOption
- type IPType
- type InterfaceManager
- func (m *InterfaceManager) Activate(iface net.Interface)
- func (m *InterfaceManager) ActiveIndices() []int
- func (m *InterfaceManager) GetActiveInterfaces() []net.Interface
- func (m *InterfaceManager) MarkFailed(ifIndex int, err error) bool
- func (m *InterfaceManager) SetBackoff(ifName string)
- func (m *InterfaceManager) Sync(current []net.Interface) []net.Interface
- type Server
- type ServerOption
- type ServiceEntry
- type ServiceRecord
Constants ¶
const ( WSAENETDOWN syscall.Errno = 10050 // Network is down WSAEADDRNOTAVAIL syscall.Errno = 10049 // Cannot assign requested address WSAEINVAL syscall.Errno = 10022 // Invalid argument )
Windows socket error codes (not in standard syscall package). These constants are safe to define cross-platform because errors.Is() performs type comparison - on non-Windows systems, these simply won't match.
Variables ¶
This section is empty.
Functions ¶
func Browse ¶
func Browse(ctx context.Context, service, domain string, entries, removed chan<- *ServiceEntry, opts ...ClientOption) error
Browse for all services of a given type in a given domain. Received entries are sent on the entries channel. It blocks until the context is canceled (or an error occurs).
func Lookup ¶
func Lookup(ctx context.Context, instance, service, domain string, entries chan<- *ServiceEntry, opts ...ClientOption) error
Lookup a specific service by its name and type in a given domain. Received entries are sent on the entries channel. It blocks until the context is canceled (or an error occurs).
func NewConnectionFactory ¶
func NewConnectionFactory() api.ConnectionFactory
NewConnectionFactory creates a new default connection factory.
func NewInterfaceProvider ¶
func NewInterfaceProvider() api.InterfaceProvider
NewInterfaceProvider creates a new default interface provider.
Types ¶
type Client ¶
type Client struct {
// contains filtered or unexported fields
}
Client structure encapsulates both IPv4/IPv6 UDP connections.
func NewClient ¶
func NewClient(opts ...ClientOption) (*Client, error)
NewClient creates a new mDNS client with the given options. This is the low-level constructor. For most use cases, prefer Browse() or Lookup().
type ClientOption ¶
type ClientOption func(*clientOpts)
ClientOption fills the option struct to configure intefaces, etc.
func SelectIPTraffic ¶
func SelectIPTraffic(t IPType) ClientOption
SelectIPTraffic selects the type of IP packets (IPv4, IPv6, or both) this instance listens for. This does not guarantee that only mDNS entries of this sepcific type passes. E.g. typical mDNS packets distributed via IPv4, may contain both DNS A and AAAA entries.
func SelectIfaces ¶
func SelectIfaces(ifaces []net.Interface) ClientOption
SelectIfaces selects the interfaces to query for mDNS records
func WithClientConnFactory ¶
func WithClientConnFactory(factory api.ConnectionFactory) ClientOption
WithClientConnFactory sets a custom connection factory for the client. This is primarily useful for testing with mock connections.
func WithClientInterfaceProvider ¶
func WithClientInterfaceProvider(provider api.InterfaceProvider) ClientOption
WithClientInterfaceProvider sets a custom interface provider for the client. This is primarily useful for testing with mock interface lists.
type IPType ¶
type IPType uint8
IPType specifies the IP traffic the client listens for. This does not guarantee that only mDNS entries of this sepcific type passes. E.g. typical mDNS packets distributed via IPv4, often contain both DNS A and AAAA entries.
type InterfaceManager ¶
type InterfaceManager struct {
// contains filtered or unexported fields
}
InterfaceManager tracks active and failed interfaces for one IP version. Thread-safe. Create separate instances for IPv4 and IPv6.
Concurrency model:
- ActiveIndices() returns a snapshot; iteration is lock-free
- MarkFailed() is idempotent; safe to call even if already removed
- Sync() runs periodically in background; updates are atomic
func NewInterfaceManager ¶
func NewInterfaceManager(initial []net.Interface, requested []string) *InterfaceManager
NewInterfaceManager creates a manager with initial interfaces. If requested is nil, dynamic mode is used (accepts new interfaces). If requested is non-nil, only those interface names are ever used.
func (*InterfaceManager) Activate ¶
func (m *InterfaceManager) Activate(iface net.Interface)
Activate adds an interface to the active set. Called after successful JoinGroup. Clears failure history. Handles the case where interface reconnected with a different index.
func (*InterfaceManager) ActiveIndices ¶
func (m *InterfaceManager) ActiveIndices() []int
ActiveIndices returns current active interface indices. Call this in send loops - never cache the result.
The returned slice is a snapshot. The caller iterates over it while the sync goroutine may modify the active map. This is safe because:
- Sends to removed indices fail fast and call MarkFailed (idempotent)
- New indices are picked up on the next ActiveIndices() call
func (*InterfaceManager) GetActiveInterfaces ¶
func (m *InterfaceManager) GetActiveInterfaces() []net.Interface
GetActiveInterfaces returns full interface objects for all active indices. Used for IP address collection (avoids race between ActiveIndices and lookup).
func (*InterfaceManager) MarkFailed ¶
func (m *InterfaceManager) MarkFailed(ifIndex int, err error) bool
MarkFailed removes an interface from active set if error indicates it's gone. Uses adaptive backoff: first failure = 1s, second = 5s, third+ = 30s.
This method is IDEMPOTENT: safe to call even if the interface was already removed by a concurrent Sync() call.
Returns true if the error indicated the interface is gone.
func (*InterfaceManager) SetBackoff ¶
func (m *InterfaceManager) SetBackoff(ifName string)
SetBackoff marks an interface as temporarily failed (e.g., JoinGroup failed). Increments the failure counter for adaptive backoff.
func (*InterfaceManager) Sync ¶
func (m *InterfaceManager) Sync(current []net.Interface) []net.Interface
Sync updates state based on currently available interfaces. Returns interfaces that were recovered and need JoinGroup calls.
Handles:
- Disappeared interfaces (removes from active, sets backoff)
- Index changes (interface reconnects with different index)
- New interfaces in dynamic mode
- Recovery after backoff expires
type Server ¶
type Server struct {
// contains filtered or unexported fields
}
Server structure encapsulates both IPv4/IPv6 UDP connections
func Register ¶
func Register(instance, service, domain string, port int, text []string, ifaces []net.Interface, opts ...ServerOption) (*Server, error)
Register a service by given arguments. This call will take the system's hostname and lookup IP by that hostname.
func RegisterProxy ¶
func RegisterProxy(instance, service, domain string, port int, host string, ips []string, text []string, ifaces []net.Interface, opts ...ServerOption) (*Server, error)
RegisterProxy registers a service proxy. This call will skip the hostname/IP lookup and will use the provided values.
type ServerOption ¶
type ServerOption func(*serverOpts)
ServerOption fills the option struct.
func WithServerConnFactory ¶
func WithServerConnFactory(factory api.ConnectionFactory) ServerOption
WithServerConnFactory sets a custom connection factory for the server. This is primarily useful for testing with mock connections.
func WithServerInterfaceProvider ¶
func WithServerInterfaceProvider(provider api.InterfaceProvider) ServerOption
WithServerInterfaceProvider sets a custom interface provider for the server. This is primarily useful for testing with mock interface lists.
type ServiceEntry ¶
type ServiceEntry struct {
ServiceRecord
HostName string `json:"hostname"` // Host machine DNS name
Port int `json:"port"` // Service Port
Text []string `json:"text"` // Service info served as a TXT record
Expiry time.Time `json:"expiry"` // Expiry of the service entry, will be converted to a TTL value
AddrIPv4 []net.IP `json:"-"` // Host machine IPv4 address
AddrIPv6 []net.IP `json:"-"` // Host machine IPv6 address
}
ServiceEntry represents a browse/lookup result for client API. It is also used to configure service registration (server API), which is used to answer multicast queries.
type ServiceRecord ¶
type ServiceRecord struct {
Instance string `json:"name"` // Instance name (e.g. "My web page")
Service string `json:"type"` // Service name (e.g. _http._tcp.)
Subtypes []string `json:"subtypes"` // Service subtypes
Domain string `json:"domain"` // If blank, assumes "local"
// contains filtered or unexported fields
}
ServiceRecord contains the basic description of a service, which contains instance name, service type & domain
func (*ServiceRecord) ServiceInstanceName ¶
func (s *ServiceRecord) ServiceInstanceName() string
ServiceInstanceName returns a complete service instance name (e.g. MyDemo\ Service._foobar._tcp.local.), which is composed from service instance name, service name and a domain.
func (*ServiceRecord) ServiceName ¶
func (s *ServiceRecord) ServiceName() string
ServiceName returns a complete service name (e.g. _foobar._tcp.local.), which is composed of a service name (also referred as service type) and a domain.
func (*ServiceRecord) ServiceTypeName ¶
func (s *ServiceRecord) ServiceTypeName() string
ServiceTypeName returns the complete identifier for a DNS-SD query.