Upgrading Radiant Heating with Home Assistant

Like most houses in the UK, I have “wet radiant heating”. If you have radiators on your walls or an underfloor heating system that uses pipes, chances are you do too. There are several commercial systems for upgrading these to smart heating including the Google Nest (you’ll also need the Heat Link), Hive, and Tado. Out of the three, only Tado can control each radiator individually, though all are rather expensive at ~£200 ($256) each.

As a second wave of winter approaches the country, I set out to build my own automated heating system for a fraction of the cost. This is obviously customised to my particular setup, though you can easily add more or less to suit yours.

Caution. This project involves mains electricity which can kill you. If you have any doubts as to what you’re doing, stop and contact a qualified electrician. I can guarantee that they will be happy to help you avoid getting injured.

Caution. Many heating systems involve gas boilers. Faulty gas systems can, and will, kill you and others. If you are not formally qualified do deal with these systems (i.e. CORGI registered), do not attempt any alterations to the gas side of your system. Stop and contact a qualified plumber or gas technician who will be happy to help.

Boiler Control

With the formalities taken care of, let’s get started. At the heart of my heating system is a boiler, this heats water which gets pumped around the radiators to give it’s heat out to different parts of the house. Any reliable heating system will need to be able to turn on the boiler when somewhere starts getting cold (this is known as a “Call for Heat”).

In the UK, many houses use heating designs origanaly created in the 80s and mine is no different. From various repairs over the years, I know mine is based on Honeywell’s Y system, and from this I was able to find full schematics and wiring diagrams online. This is a rather enginious system, reminiscent of the relay logic common at the time. Yours may be different, so please do some research and don’t assume you can wire things into the same places.

The existing system uses an ST699 timer for both heating and hot water, which will need to be left in place to keep the hot water control. Luckily, the Y system also uses (in my case, cheap copy) a T6360 room thermostat. This is a great place to add our control, as it means we can override the automations to turn the heating on (turn up the room thermostat) or off (turn off the timer) in case there are any problems.

Because all the existing components used a switch mains signal, we’ll need to do the same to make things easier. For this I turned to the Sonoff Basic R2, which can switch up to 10A at 230v. This will be plenty for the few components that run off the switched supply directly (in my case, the selector valve and pump) with a generous safety margin, as the control systems are on a 3A fuse. Note that this Sonoff does require a neutral connection for it’s control system, but there’s no need to run the neutral output back through the Sonoff.

For controlling this Sonoff, I’m a big fan of ESPHome. This is highly customisable and can be used with a huge variety of devices. Another popular firmware for these is Tasmota. I won’t go over the exact details for setting up ESPHome as this is well covered elsewhere, but here’s the config used on mine:


Though the boiler is the heart of our heating system, it would be nothing without limbs to actually do work. In this case, those limbs are the radiators. Radiators come in many shapes and sizes, but we can use a cleverly designed feature of most modern radiators to easily control them. Well, kind of. The Thermostatic Radiator Valve (TRV) it attached between the flow pipe and the radiator, and controls the flow of hot water into the radiator. This is not to be confused with the lockshield valve on the other end of the radiator, which is used to restrict the flow for radiators closer the the boiler (known as balancing) to even out the available heat throughout the building.

The particularly cleaver part or a TRV is that they’re designed in two parts; the valve body, attached to the pipes to actually control the water, and the valve head, containing a wax element and dial to set the temperature. More importantly, these two parts can easily be separated by unscrewing then without the need to drain the radiator first. This means we can easily replace the valve head to add our control without risking anything getting wet.

Here’s where things get even better, because nobody likes re-inventing the wheel, you can buy off-the-shelf electronic valves to replace the TRV heads. These are known as thermal actuators, and are really designed for larger systems with seperate timers and themostats, so they’re available in 24v or 230v versions as well as relay-style normally open and normally closed. I picked one on amazon for no particular reason, and it worked great.

To make the control as easy as possible, I chose a valve that ran on 230v and closed when power was removed. This means we can use another Sonoff to control them. In this case, the Sonoff TH10, which includes a handy temperature sensor that can be used as a room thermostat.

A quick note on these, while the sensor socket looks like a normal headphone jack, it’s actually a 2.5mm TRRS socket. I learnt this the hard way after ordering half a dozen headphone extension leads and finding they don’t fit.

As before, I’m using ESPHome for the firmware on these. The configuration is very similar to the boiler switch, but with the added temperature sensor. I may rethink this in the future to make the button work as a temperature boost for an hour or so, as Home Assistant will turn the valve off again if it’s changed manually. As before, here’s my current config for one of them:


The automations really tie this system together, and the various elements don’t really function without them. Ultimately, this is going to need some heavy customisation to suit your needs, but there are some basics you’ll definitely want.

First off is the climate component, this will let you set a target temperature either manually or through an automation, and Home Assistant will turn on and off the radiator as needed to get to this temperature. Each entry needs a name, a heater entity that gets turned on when the room needs heating, and a target sensor entity that reports the current temperature of the room.

We can also add an automation to turn the boiler on and off whenever any radiator is on, because a radiator with cold water isn’t going to do much for heating the room. These are set to trigger when any radiator goes on or off, but the off automation has an extra condition to make sure all the rooms are up to temperature before allowing the boiler to turn off

Bonus Round: Underfloor Heating

I’ve also got a small underfloor heating loop running off the boiler. This was always a bit annoying to use as it required the timer and thermostat at the opposite end of the house to be on, as well as the ones for the underfloor heating. Because of this, now is a great time to upgrade it. I used the same Sonoff TH10 as the radiators, but this time wired it to the separate pump that runs the underfloor loop.

Some underfloor heating systems already have separate control for different zones using thermal actuators on the manifold, if so you’ll also need to wire these in to a Sonoff and use similar automations to the boiler switch to ensure the pump is running whenever one of the zones needs more heat. However mine doesn’t, so this wasn’t needed.

Final Thoughts

Overall this system gives a lot more control than the Nest or Hive systems. It’s also significantly cheaper than all three main competitors. My system which included the boiler, 4 radiators, and the underfloor heating came to just over £130. More importantly, it’s fully integrated with Home Assistant, which means you get wonderful automations like “when I leave work around 5pm, and I’m heading towards home, start warming the house up”.

Installing Home Assistant, Mosquitto & AppDaemon on Debian

As part of some hardware upgrades, I’m moving Home Assistant and it’s associated programs off it’s shared box onto a dedicated VM. This means a fresh install of everything. Whilst docker would usually be an ideal candidate for this, I decided against it because I use a few custom components and troubleshooting these is a lot easier without Docker.

Preparing the VM

Nothing too special here, I used the latest network install media for Debian Buster (10). The netinst media is particularly useful if you have a fast internet connection, as the latest version of each package is downloaded before installing). Unfortunately this does mean slowing internet connections or offline installs are out of the question. If you’re doing a large number of installs, a local debian repository should work with the netinst media, but I have yet to test this.

Once this was done there’s the usual routine of adding SSH keys (yay for Ansible), setting up backups (Veeam works nicely), and installing Monitoring (Zabbix is awesome). I didn’t pay much attention to the system Python version, as Debian still ships with 3.5 which Home Assistant no longer supports.

Installing Dependencies

There are a number of global dependencies we can install through Apt to save time in the future. Here’s the command I eventually came up with to get everything up and running

Installing Home Assistant

By this point, I hope I don’t have to explain what Home Assistant is. In short, it’s some of the best home automation and BMS software out there. Unfortunately the latest versions have some dependency issues with AppDaemon (despite being designed to work together) and has now dropped support for Python 3.5. Because of this, we’ll install a separate version of Python for each of them.

Create a system account

First, we’ll create a new user to run the service. Useradd can do most of the work for us, using these parameters:

  • -r Create a system account, disables password ageing and uses a lower ID
  • -m Create the user’s home directory
  • -d Specify the location of the home directory
  • -s The login shell (or lack thereof) for the user

Overall, that gives us this:

From here on, everything will be run from this user account, because nothing else needs to use it’s version of Python or the libraries. To make this easier, we can use sudo to run bash as Home Assistant’s account:

Install Python

It’s always handy when people scripts exist to do the work for you, and installing Python is no different. To handle the installation, we turn to PyEnv. Using these commands we can install PyEnv, and then use it to install the latest version of Python.

Install Home Assistant

Now we have a working install of Python, Home Assistant can be installed using it’s version of pip

Starting automatically with systemd

There are lots of arguments in favour of and against systemd, though from my point of view the benefits outweigh the disadvantages. For example, here we’re going to tell systemd to start home assistant when the system boots, but after the network and Z-Wave adapter are ready this was causing a few issues for me, so I’ve removed it for now.

Installing Mosquitto

Out of the three Mosquitto is by far the easiest to install, and that’s saying something

And that’s it. Told you it was simple. As with all of these, the configuration will need adjusting to suit your needs, but that’s well covered by the relevent documentation.

Installing AppDaemon

AppDaemon is a companion for home assistant, and is capable of doing much more powerful automations using python modules & scripts.

The first part of this is very similar to the Home Assistant installation. I’m doing them using separate Python versions and environments because there are some dependency issues between the two. Keeping the environments completely separate will make updating in the future much easier.

Create a system account

Same as for the Home Assistant user, Debian will do most of the work for us with this command:

Most of the following commands needs to be run as the appdaemon user. We can switch to it using sudo to explicitly run Bash

Install Python

Again, we’ll use pyenv to install the latest version of Python for the service account

Install AppDaemon

Again this is fairly simple. AppDaemon is available through PyPi, so it can be installed using the Pip command:

Starting automatically with systemd

Finally, we can use systemd to automatically start appdaemon with the server. The below unit file can be placed in /etc/systemd/system/appdaemon.service

There’s a couple of non-obvious options in here, so let’s go through them all:

  • After… – This makes sure appdaemon doesn’t start until Home Assistant is up and running
  • ExecStartPre… – Like many users I store my Home Assistant configuration in Git. AppDaemon requires permissions on it’s configuration file, and modifying them through git as another user may accidentally remove the permissions. This command explicitly grants those permissions each time AppDaemon starts.

Installing/configuring HASS Apps, or any other AppDaemon app

One of my main uses of AppDaemon is Schedy, part of HASS Apps. Though this is just an example, and should work for any AppDaemon-powered app. First we install HASS Apps:

Now we can configure AppDaemon, using the config file /var/lib/appdaemon/config/appdaemon.yaml and it’s matching secrets file.

Then we need to load the HASS Apps module into AppDaemon. This is done using a shim in /var/lib/appdaemon/config/apps/hass_apps_loader.py

Now the important bit, we tell AppDaemon to use Schedy and what schedules we want. This goes in /var/lib/appdaemon/config/apps/schedy.yaml

Turning Everything On

Now all of our software is in place, and has a basic starting configuration, let’s tell systemd to turn everything on!

Home assistant will take a while to start for the first time, so now is ideal to go and make a cup of tea. In my case, this took over 90 minutes due to the complexity of my particular setup.