Read the statement by Michael Teeuw here.
Airplay 'Now Playing' module - help badly needed
-
I’ve really been struggling to write my first module.
I’ve used Shairport-Sync to allow my mirror to act as an Airplay device - something which works a treat.
I’ve added Shairport-Sync-Metadata-Reader in order to base64-decode the playing track metadata, which Shairport outputs in xml format to a FiFo file.What I’m trying to do is create a module to read the FiFo stream, base64-decode the values and it sees, and display a couple of the fields (Artist/Track)
I think I’ll need a node-helper to do the watch/decode/filter and then send any Artist & Track fields it sees to the module which then displays it.
I’ve created this which pops up the decoded result for a hardcoded string
<script> var string = 'RGFycmVu'; var decodedString = atob(string); window.alert(decodedString); </script>I’ve tried without success though to incoporate this as a function into a .js file, so that I can display my test-decode on the mirror.
I’ve been looking through all the modules I can find to find something I can use as an example but no luck. I know I need to pass the decodedString variable somehow but have struggled to work out how :( -
The first step of my module, which simply displays a base64-decoded string in the mirror reads like this.
/* global Module */ /* Magic Mirror * Module: base64 * * By Michael Teeuw http://michaelteeuw.nl * MIT Licensed. */ Module.register("base64",{ // Default module config. defaults: { text: "Hello World!" }, frombase64: function() { var string = 'RGFycmVu'; var decodedString = atob(string); return decodedString; }, // Override dom generator. getDom: function() { var wrapper = document.createElement("div"); var decstr = this.frombase64 var msgstr = document.createTextNode(decstr); wrapper.appendChild(msgstr); return wrapper; } });It seems like it should work, but I get a literal string rather than a value returned…

-
@darrene You’re missing parenthesis in your call to frombase64:
this.frombase64();
-
Ah! Thanks so much @alexyak - that’s fantastic. I can now move forwards :)
So parenthesis are needed to call the function and explicitly not pass it a parameter. Is that right?
-
So step 2, I have made a short javascript file that will read a fifo file and decode what it sees in the buffer.
var fs = require('fs') var fifo = fs.createReadStream('shairport-sync-metadata') fifo.on('data', function(fifoHexBuffer) { fifoHexBuffer = fifoHexBuffer.toString(); console.log(fifoHexBuffer) })At the moment, the code runs once and stops once it reads. My next task is to have it continue, then I’ll have to make it into a node-helper script, to interact with my base64 decoder/display
-
@darrene checkout the file watcher of the nodejs filesystem https://nodejs.org/docs/latest/api/fs.html#fs_class_fs_fswatcher
-
Luckily there is a fifo-js package available, which makes things super-easy!
npm install fifo-jsconst FIFO = require('fifo-js') let fifo = new FIFO('fifo-testfile') var count = 1 while (count == 1) { let text = fifo.readSync(fifo) console.log(text) } fifo.close()I’m getting there. I have an IN and I have an OUT in javascript. Now the hard work of joining the two together - interpreting and the decoding the appropriate metadata begins!
-
So a bit more testing and things aren’t looking as rosy as I’d first thought. Before trying fifo-js, I’d experimented with the filesystem readStream - If i try looping that I end up with a java out of memory condition.
The fifo-js approach works with a fifo file that I echo things into to test, but doesn’t return anything while the Shairport service has it open for some reason?
There is a metadata reader script which I could use to read the metadata and parse it. I’ve compiled the executable and it runs fine. I should be able to call that from my javascript module through child process, right?
-
Hmm, using a child process doesn’t return anything either :(
The shairport metadata reader is a compiled C program which uses printf to return the metadata to the console:
pi@raspberrypi:~/fifo-js $ ~/shairport-sync-metadata-reader/shairport-sync-metadata-reader < /tmp/shairport-sync-metadata Artist: "Sohn". Title: "Tremors". ^C pi@raspberrypi:~/fifo-js $But I dont’ seem to be able to parse what’s being sent to stdout from my javascript
If anyone has any examples that work, I’d be grateful.
-
Hi @darrene ,
How did you ever make out with your module? I’ve started down the same path, I’m also using shairport-sync but in conjunction with forked-daapd (an airplay server for Raspbian).
As forked-daapd can also use mpd commands I’ve been trying to use komponist, which I’ve managed to make return the playing song, album, etc. print out to the console but as I’ve never really done any javascript before am trying to piece together building a module for MM.
If you’ve made any progress on your module, I’d love to see it as it may help me finish mine.
-
I’m just dropping by to express my encouragement for this effort! I would love to have a now-playing module that used the shairport-sync metadata :)
-
Are there any updates to this module?
-
I’m also interested in the development of this module.
I’ve started to recompile Shairport-sync so it can write a plain .TXT file with the metadata each time a new song is detected, now I have to read the last line of this TXT file through a MM-module. But I don’t know if this is the right direction, editing the core of Shairport-sync seems like a bad idea (no updates, etc). I wish I could read metadatas via the pipe in a less complex way. Anyone has worked on this ? Thanks ! -
ANNOUNCE: Check out https://github.com/surekap/MMM-ShairportMetadata
Its my first attempt at writing anything in node.js so I basically wrote a python script to translate the XML from shairport-sync into json objects.
The module manages a child process which pipes the XML to the Py script and reads the json and displays it. -
Thanks a lot @surekap for this amazing module. It works !
Quick suggestions :- hide the module if no airplay device is connected / no song playing
- align text & image on left/right depending on the module position (currently both centered in its container)
-
@Brice changes made. Check out the “alignment” config option.
The hide behaviour is not perfect - takes a few seconds to hide the module - this is because we issue the hide command in case of ‘pfls’ [play flush] and ‘pend’ [play end] metadata commands. Unfortunately, the ‘pfls’ command is followed up with fresh metadata and it takes a few seconds for the ‘pend’ to come from the iPhone.
Tested with playing and pausing repeatedly.
Hello! It looks like you're interested in this conversation, but you don't have an account yet.
Getting fed up of having to scroll through the same posts each visit? When you register for an account, you'll always come back to exactly where you were before, and choose to be notified of new replies (either via email, or push notification). You'll also be able to save bookmarks and upvote posts to show your appreciation to other community members.
With your input, this post could be even better 💗
Register Login