How I built a Sonos replacement app in 2014
Back in 2014 in my spare time I built an app to replace the default Sonos app used to control multiple speakers in a home. It was a little challenging so I thought I’d share the process. You can check out most of the code on Github.
data:image/s3,"s3://crabby-images/8d90c/8d90c448a1cb42437bc8350b8101eef3efc7e5f1" alt=""
Finding the API
First off I needed to understand how the existing app was communicating with speakers on the local network. Sonos doesn’t really have an official API so I had to do some digging with Wireshark.
data:image/s3,"s3://crabby-images/6db4a/6db4ae01a031bc35305240c80451531c4eefa087" alt=""
Wireshark reveals all the network traffic, which allowed me to see what requests were being passed between the speakers and the app. Sonos uses UPnP and this established a base understanding of what was needed to build a new app.
Making a better experience
What bothered me most about the current app was it didn’t give me the controls I needed quickly. Most of the time all I wanted to do was change the volume and pair speakers (I also really wanted to make my living room play out of the Line-In input quickly without any fuss). Here were a couple solutions:
data:image/s3,"s3://crabby-images/f9d71/f9d71bed41f83a93625ae7705813ed169856fd06" alt=""
I went through a bunch of iterations, mostly because it was fun. iOS 7 shipped in the middle of it all which forced me to simplify things and learn a new emerging visual language.
data:image/s3,"s3://crabby-images/4cb57/4cb57b21969610a6d0c921417203e6975a7117ae" alt=""
data:image/s3,"s3://crabby-images/ecc41/ecc41364f4e04fcaf1f0877e2eb4317942246a1a" alt=""
data:image/s3,"s3://crabby-images/b87df/b87df2ec1a50f29ec13c78e0944f56aab12fd109" alt=""
data:image/s3,"s3://crabby-images/19631/19631072ffe9fa06a00396f1037528ac67dc1eaa" alt=""
data:image/s3,"s3://crabby-images/2f340/2f34033ba8c952c57f06930adcdbce3b93a94322" alt=""
data:image/s3,"s3://crabby-images/d0410/d04106a316987081f1ad083a7d1f1bb659447124" alt=""
Each mocked version above was built to some degree. You can see all the iterations on Github. A lot of this is still in development and I’ll try to push improvements when they’re ready.
What about desktop?
Then I realized I wanted more than just an app on my phone, sometimes I’d be on my laptop and want to quickly adjust volume. So I decided to build a menu bar app. Volume is all I really needed here so I didn’t implement anything else.
data:image/s3,"s3://crabby-images/75311/75311dd5a2244b0628a291e17e8bd762938a4f5e" alt=""
This was a lot easier once I abstracted out all the bits needed to control Sonos into a little library called SonosKit.
Can it be smarter?
The above was all nice but I began to notice another annoyance—I didn’t like having to turn my speakers off when I left my apartment. I had already figured out how to turn all my lights off with IFTTT so why not make this work with Sonos. Using a Raspberry Pi and a Sonos command-line interface I hooked up a very basic home REST API where I could detect my phone leaving a location using a geofence.
Now when I leave it makes a request to my home and the music turns off.
data:image/s3,"s3://crabby-images/c52dd/c52dd840900663ee950a2f57513c07b7ee8a784b" alt=""
Next up
The next thing I’m working on (aside from finishing some of the above stuff) is a way to determine an optimal state without having to interact with anything. By learning more about my listening habits I could automatically start playing one of three podcasts before going to bed or automatically start playing jazz on the weekends. All things I manually do on a pretty regular basis. Then I’m curious if I can use music to effect my mood—after a stressful day (i.e. lots of meetings on my calendar) I could let Sonos know and maybe start playing some chill music when I walk into my apartment.