Asking my home to play music

Asking my home to play music

  • 7 minutes to read
  • Saturday, 12th of June 2021

Don't you love open source and the capabilities it provides? My constant tinkering of my home automation has no end! The latest improvement is asking the system to play my favorite music. And I remembered that the advanced capabilities of using Rhasspy with Home Assistant with as little code as possible is not clear to everybody. So here is an explanation of how this works, and how I leveraged these capabilities to satisfy my music needs.

Pre-Requisites

In order to use Home Assistant intent handling with Rhasspy, you need to define the Hass URL and an Access Token via the GUI or as described here.

Home assistant needs to have intents enabled. I opt to keep intents in a seperate file, so I added these two lines in Home Assistant's configuration.yaml :

intent: intent_script: !include intentscripts.yaml

I do not think that it makes any difference, but my examples are based on my setup, which uses Fsticuffs for intent recognition.

Predefined Intents

Home Assistant has some built-in intents. These are mentioned here. These intents need one or more parameters. We will use the intent HassTurnOn as an example. This intent needs one parameter - the name as described in the bultin intent documentation.

Rhasspy will trigger intents based on sentences file(s). A sample sentences file for some of the built-in intents is provided here.

The actual intent recognition sentence is the last line:

[HassTurnOn] turn on [the] (<entities>){name}

and when the text "turn on the kitchen light" is recognized, the HassTurnOn intent is triggered on home assistant with the parameter "name" having a value of "kitchen light".

The "kitchen light" is a "slot". Something that needs to be defined so that Rhasspy can understand that it is a variable and pass it to Home Assistant. Slots in Rhasspy are defined in slots files. And although one can create slots files with all the relevant entities that are valid for Home Assistant, there is a program that can do that for you. This program is here and needs to be installed in the system that Rhasspy is running, in the profiles directory, under a directory named "slot_programs/hass".

This program can take parameters, so when you call the program entities with the parameter light, it will return only the entities of type Light that are defined in your Home Assistant instance instead of all entities. This is what the rest of the lines of this intent in the sample sentences file do.

lights = $hass/entities,light switches = $hass/entities,switch entities = <lights> | <switches>

Just remember to re-train Rhasspy whenever you add new lights or switches in Home Assistant. Rhasspy should pick it up automatically.

Modifying the predefined intents

Lights and switches are not the only entities that we have in Home Assistant. If you would like to turn on and off TVs and Radios connected to your home assistant, you can add the media_player entity type, which would modify the lines above to be:

[HassTurnOn] lights = $hass/entities,light media = $hass/entities,media_player switches = $hass/entities,switch entities = <lights> | <switches> | <media> turn on [the] (<entities>){name}

If you would like to use a different sentence for lights and TVs, and a difference for the switches, you could modify the sentences file:

[HassTurnOn] lights = $hass/entities,light media = $hass/entities,media_player entities = <lights> | <media> turn on [the] (<entities>){name} [HassTurnOn] switches = $hass/entities,switch switch on [the] (<switches>){name}

Creating new intents

The built in intents cover a lot of cases, but not all. They are mostly on/off and similar actions. Sometimes, we just need to ask Home Assistant about something.

Using the slots program

One of the entities that Home Assistant understands is that of person. If you have device trackers for people, you can find out were they are by adding this sentence in a Rhasspy sentences file:

[IsHome] persons = $hass/entities,person where [is] (<persons>){name}

In Home Assistant you need to define the intent, and the response. Add this to the intentscripts.yaml file (be careful of yaml indentation).

IsHome: speech: text: > {% set tracker = "device_tracker." + name %} {{ name }} is at {{ states(tracker) }}

Without using the slots program

The slot program provided as an example uses the REST API endpoint /api/states to get the state of all entities. This is an indirect way of getting the entities, because there is no direct way to get the entities through the API. There is no way to get many things through the API, and areas is one of them. At the moment there is no way to retrieve the areas from Home Assistant, nor is there a way to use the area name in a jinja2 template. But that should not limit you.

Naming the devices

It is imporant to name the devices taking the area into account. For example if you have an environmental sensor, you can name the device as Bedroom environment which would create entities such as Bedroom environment temperature. This first part can be used to identify which sensor you are asking about.

Hard coding slots

You must create a slots file with the area names, as they are used in the device naming. Let's call this file as "rooms.ini". It could be as simple as :

Kitchen Bedroom

Add all the rooms in which you have environmental sensors in this file.

Creating the sentences

A sentence to retrieve the temperature could be something like :

[WeatherTemperature] what [is] the temperature in the ($rooms){room_name}

This takes the names defined in the rooms.ini slots file and sends it as room_name to Home Assistant with the intent WeatherTemperature

Creating the intent

We need to let Home Assistant know what to do with this intent, so in the intentscripts.yaml file you need to add:

WeatherTemperature: speech: text: > {% set room_name = room_name | regex_replace('\s', '_') %} {% set measurement = "sensor."+room_name+"_environment_temperature" %} The temperature in the {{room_name}} is {{ states(measurement) |int }} degrees

The regex in the first line takes care of any rooms that may contain spaces, and if your rooms are single words only, you do not need this line.

Writing new slots programs

The slots program provided is a great starting point for you to understand how it works, and write your new slots programs. Let's say that you want to control a Logitech Squeezebox through Home Assistant and Rhasspy. Here is an example how you can set up these three components to play the music you want on the player you want. Obviously, the implementation is simpler if you only have one player.

Slots programs

Logitech Media Server can be queried through API to return the players and the playlists. Documentation about this API on your Logitech Media Server at the address: http://your_mediaserver_ip:9000/html/docs/cli-api.html

A slot program to retrieve 100 playlists would be :

#!/usr/bin/env bash set -e url="http://your_mediaserver_ip:9000/jsonrpc.js" player="FF:FF:FF:FF" curl -X POST \ -H 'Content-Type: application/json' \ "${url}" \ -d '{"id":1,"method":"slim.request","params":["'${player}'",["playlists",0,100,"tags:s"]]}' \ 2> /dev/null | \ jq --raw-output '.result as $out | "\($out.playlists_loop[].playlist)"' | \ while read -r playlist; do echo ${playlist} done

Save this under the directory slot_programs/lms in your Rhasspy server as playlists Create a slot_program to retrieve the players, using the same concepts.

Rhasspy sentences

A sentence that will play the selected playlist, over the selected player will be:

[PlayPlaylist] playlists = $lms/playlists players = $lms/players play (<playlists>){playlist} [on (<players>){player}]

This will trigger the intent PlayPlayList on HomeAssistant with up to two parameters: the playlist (mandatory) and the player (optional).

Home Assistant intent

The intent will receive the playlist parameter, and may receive the player parameter. It will use the squeezebox integration to play the playlist on the defined player.

In this example it is important to decide which player is the default. This will be used if the second optional parameter is not spoken / defined / sent to Home assistant. Let's assume that the default player's name is "lounge". The intent will be (be careful with yaml indentation):

PlayPlaylist: action: - service: squeezebox.call_method target: entity_id: > media_player.{{ player | default("lounge") }} data: command: playlist parameters: - play - > {{playlist}}

As I was not happy with the speakers from my creation, I set up a Raspberry Pi with some cheap speakers somewhere else and I'm using this for outputting squeezebox, as well as Rhasspy's output. I can finally understand what the temperature in my bedroom is, while I'm listening to some relaxing music.