From b8fed490c275487f14bff4f5b49a6072d27e5b5f Mon Sep 17 00:00:00 2001 From: xegineering Date: Thu, 3 Oct 2024 19:16:56 +0200 Subject: Add soundbox.StreamURLContext() This should be the primary public API of the library to stream web radio to soundbox devices. --- README.md | 4 +++- example_test.go | 38 ++++++++++++++++++++++++++++++++++++++ stream.go | 40 ++++++++++++++++++++++++++++++++++++++++ 3 files changed, 81 insertions(+), 1 deletion(-) create mode 100644 example_test.go diff --git a/README.md b/README.md index c9cbb06..1fc718d 100644 --- a/README.md +++ b/README.md @@ -1,9 +1,11 @@ # soundbox-go -[soundbox-go][1] is a Go library to interface with [soundbox][2] devices. +[soundbox-go][1] is a Go library to interface with [soundbox][2] devices. It +currently requires that the [ffmpeg][3] command line utility is installed. While the repository and project name is `soundbox-go` the Go module name is only `soundbox` to make the resulting code more readable. [1]: https://xengineering.eu/git/soundbox-go [2]: https://xengineering.eu/git/soundbox +[3]: https://ffmpeg.org diff --git a/example_test.go b/example_test.go new file mode 100644 index 0000000..78b2f27 --- /dev/null +++ b/example_test.go @@ -0,0 +1,38 @@ +package soundbox_test + +import ( + "context" + "log" + "net" + "time" + + "xengineering.eu/soundbox" +) + +func ExampleStreamURLContext() { + ctx, cancel := context.WithCancel(context.Background()) + + // all soundboxes are referenced by their MAC address + soundboxes := []net.HardwareAddr{ + {0x00, 0x00, 0x5E, 0x00, 0x53, 0x01}, + {0x00, 0x00, 0x5E, 0x00, 0x53, 0x02}, + {0x00, 0x00, 0x5E, 0x00, 0x53, 0x03}, + } + + // currently only web radio is supported + url := "https://example.org/radio.mp3" + + // start streaming + go func() { + err := soundbox.StreamURLContext(ctx, url, soundboxes) + if err != nil { + log.Fatal(err) + } + }() + + // let it play for some time + time.Sleep(time.Minute) + + // stop it + cancel() +} diff --git a/stream.go b/stream.go index 38b3727..ac0a189 100644 --- a/stream.go +++ b/stream.go @@ -1,5 +1,45 @@ package soundbox +import ( + "context" + "fmt" + "os/exec" + "net" +) + // streamingPort is the default network port a soundbox is listening to for // incoming audio stream data. const streamingPort = 5316 + +func StreamURLContext(ctx context.Context, url string, targets []net.HardwareAddr) error { + iface, err := getInterface() + if err != nil { + return err + } + + cmd := []string{ + "-re", + "-i", + url, + } + + for _, target := range targets { + ip, err := toLinkLocal(target) + if err != nil { + return err + } + + cmd = append(cmd, "-acodec") + cmd = append(cmd, "flac") + cmd = append(cmd, "-f") + cmd = append(cmd, "ogg") + cmd = append(cmd, fmt.Sprintf( + "tcp://[%s%%%s]:%d", + ip, + iface.Name, + streamingPort, + )) + } + + return exec.CommandContext(ctx, "ffmpeg", cmd...).Run() +} -- cgit v1.2.3-70-g09d2