diff options
Diffstat (limited to 'soundbox')
-rw-r--r-- | soundbox/interfaces.go | 39 | ||||
-rw-r--r-- | soundbox/ipv6.go | 19 | ||||
-rw-r--r-- | soundbox/network.go | 58 | ||||
-rw-r--r-- | soundbox/network_test.go (renamed from soundbox/ipv6_test.go) | 0 | ||||
-rw-r--r-- | soundbox/url.go (renamed from soundbox/stream.go) | 24 | ||||
-rw-r--r-- | soundbox/url_test.go (renamed from soundbox/stream_test.go) | 0 |
6 files changed, 60 insertions, 80 deletions
diff --git a/soundbox/interfaces.go b/soundbox/interfaces.go deleted file mode 100644 index 04839c7..0000000 --- a/soundbox/interfaces.go +++ /dev/null @@ -1,39 +0,0 @@ -package soundbox - -import ( - "fmt" - "net" -) - -// getInterface is a function guessing which interface should be used for -// soundbox streaming. This is required since soundbox relies on communication -// via IPv6 link-local addresses which require a specified interface to allow -// communication. This function returns the first interface it finds which is -// up and has a link-local address assigned. -func getInterface() (net.Interface, error) { - all, err := net.Interfaces() - if err != nil { - return net.Interface{}, err - } - - for _, iface := range all { - if iface.Flags&net.FlagUp == 0 { - continue - } - addresses, err := iface.Addrs() - if err != nil { - return net.Interface{}, err - } - for _, addr := range addresses { - ip, _, err := net.ParseCIDR(addr.String()) - if err != nil { - return net.Interface{}, err - } - if ip.IsLinkLocalUnicast() { - return iface, nil - } - } - } - - return net.Interface{}, fmt.Errorf("No interface found for soundbox streaming") -} diff --git a/soundbox/ipv6.go b/soundbox/ipv6.go deleted file mode 100644 index 355e233..0000000 --- a/soundbox/ipv6.go +++ /dev/null @@ -1,19 +0,0 @@ -package soundbox - -import ( - "fmt" - "net" -) - -// toLinkLocal converts a MAC address to the corresponding IPv6 link-local -// address. -func toLinkLocal(ha net.HardwareAddr) (net.IP, error) { - switch len(ha) { - case 6: - ip := net.IP{0xfe, 0x80, 0, 0, 0, 0, 0, 0, - ha[0] ^ 0b10, ha[1], ha[2], 0xff, 0xfe, ha[3], ha[4], ha[5]} - return ip, nil - default: - return nil, fmt.Errorf("Only IEEE 802 MAC-48 addresses supported") - } -} diff --git a/soundbox/network.go b/soundbox/network.go new file mode 100644 index 0000000..6e7acea --- /dev/null +++ b/soundbox/network.go @@ -0,0 +1,58 @@ +package soundbox + +import ( + "context" + "fmt" + "net" + "time" +) + +const dialTimeoutSeconds = 3 + +// toLinkLocal converts a MAC address to the corresponding IPv6 link-local +// address. +func toLinkLocal(ha net.HardwareAddr) (net.IP, error) { + switch len(ha) { + case 6: + ip := net.IP{0xfe, 0x80, 0, 0, 0, 0, 0, 0, + ha[0] ^ 0b10, ha[1], ha[2], 0xff, 0xfe, ha[3], ha[4], ha[5]} + return ip, nil + default: + return nil, fmt.Errorf("Only IEEE 802 MAC-48 addresses supported") + } +} + +func dialContext(ctx context.Context, ha net.HardwareAddr) (net.Conn, error) { + ip, err := toLinkLocal(ha) + if err != nil { + return nil, err + } + + ifaces, err := net.Interfaces() + if err != nil { + return nil, err + } + + c := make(chan net.Conn) + dialContext, cancel := context.WithTimeout(ctx, dialTimeoutSeconds * time.Second) + defer cancel() + for _, iface := range ifaces { + go func() { + var d net.Dialer + conn, err := d.DialContext( + ctx, + "tcp6", + fmt.Sprintf("[%s%%%s]:%d", ip, iface.Name, streamingPort), + ) + if err == nil { + c <- conn + } + }() + } + select { + case conn := <-c: + return conn, nil + case <-dialContext.Done(): + return nil, fmt.Errorf("Could not dial TCP connection to %v on port %d on any interface.", ha, streamingPort) + } +} diff --git a/soundbox/ipv6_test.go b/soundbox/network_test.go index 93ac489..93ac489 100644 --- a/soundbox/ipv6_test.go +++ b/soundbox/network_test.go diff --git a/soundbox/stream.go b/soundbox/url.go index 910523d..4ad9908 100644 --- a/soundbox/stream.go +++ b/soundbox/url.go @@ -3,7 +3,6 @@ package soundbox import ( "context" "errors" - "fmt" "io" "net" "os/exec" @@ -22,28 +21,9 @@ const writeTimeout = 1 * time.Second // devices. The devices are referenced via their MAC addresses given by the // targets argument. The ctx argument is passed to cancel the streaming. func StreamURLContext(ctx context.Context, url string, targets []net.HardwareAddr) error { - iface, err := getInterface() - if err != nil { - return err - } - - ips := make([]net.IP, 0) - for _, target := range targets { - ip, err := toLinkLocal(target) - if err != nil { - return err - } - ips = append(ips, ip) - } - conns := make([]net.Conn, 0) - for _, ip := range ips { - var d net.Dialer - conn, err := d.DialContext( - ctx, - "tcp6", - fmt.Sprintf("[%s%%%s]:%d", ip, iface.Name, streamingPort), - ) + for _, target := range targets { + conn, err := dialContext(ctx, target) if err != nil { return err } diff --git a/soundbox/stream_test.go b/soundbox/url_test.go index 41907ad..41907ad 100644 --- a/soundbox/stream_test.go +++ b/soundbox/url_test.go |