This component adds electricity prices from stock exchange EPEX Spot to Home Assistant. EPEX Spot does not provide free access to the data, so this component uses different ways to retrieve the data.
January 2024: Hi there, I'm eperimenting with a new feature:
https://github.com/mampfes/ha_epex_spot_sensor
Please let me know if this is a useful addition to EPEX Spit.
You can choose between multiple sources:
-
Awattar
Awattar provides a free of charge service for their customers. Market price data is available for Germany and Austria. So far no user identifiation is required.
-
EPEX Spot Web Scraper
This source uses web scraping technologies to retrieve publicly available data from its website.
-
SMARD.de
SMARD.de provides a free of charge API to retrieve a lot of information about electricity market including market prices. SMARD.de is serviced by the Bundesnetzagentur, Germany.
-
smartENERGY.at
smartENERGY.at provides a free of charge service for their customers. Market price data is available for Austria. So far no user identifiation is required.
If you like this component, please give it a star on github.
-
Ensure that HACS is installed.
-
Install EPEX Spot integration via HACS:
-
Add EPEX Spot integration to Home Assistant:
In case you would like to install manually:
-
Copy the folder
custom_components/epex_spot
tocustom_components
in your Home Assistantconfig
folder. -
Add EPEX Spot integration to Home Assistant:
This integration provides the following sensors:
- Net market price
- Market price
- Average market price during the day
- Median market price during the day
- Lowest market price during the day
- Highest market price during the day
- Current market price quantile during the day
- Rank of the current market price during the day
The EPEX Spot Web Scraper provides some additional sensors:
- Buy Volume
- Sell Volume
- Volume
NOTE: For GB data, the prices will be shown in GBP instead of EUR. The sensor attribute names are adjusted accordingly.
The sensor value reports the net market price in ct/kWh. The price value will be updated every hour to reflect the current net market price.
The sensor attributes contains a list of all available net market prices (for today and tomorrow if available) in ct/kWh.
data:
- start_time: "2022-12-15T23:00:00+00:00"
end_time: "2022-12-16T00:00:00+00:00"
price_ct_per_kwh: 29.63
- start_time: "2022-12-16T00:00:00+00:00"
end_time: "2022-12-16T01:00:00+00:00"
price_ct_per_kwh: 28.812
- start_time: "2022-12-16T01:00:00+00:00"
end_time: "2022-12-16T02:00:00+00:00"
price_ct_per_kwh: 28.019
The net market price will be calculated as follows:
<Net Price>
= <Market Price>
+ <Surcharges>
+ <Tax>
The values for surcharges and tax can be adjusted in the integration configuration. 2 different types of surcharges can be adjusted:
- Percentage Surcharge, stated in % of the EPEX Spot market price.
- Absolute Surcharge, stated in ct/kWh.
Example:
Percentage Surchage = 3%
Absolute Surcharge = 12ct
Tax = 19%
Net Price = ((Market Price * 1.03) + 12ct) * 1.19
The sensor value reports the EPEX Spot market price in EUR/MWh. The price value will be updated every hour to reflect the current market price.
The sensor attributes contains additional values:
- The market price in ct/kWh.
- A list of all available market prices (for today and tomorrow if available) in EUR/MWh and ct/kWh.
price_ct_per_kwh: 29.63
data:
- start_time: "2022-12-15T23:00:00+00:00"
end_time: "2022-12-16T00:00:00+00:00"
price_eur_per_mwh: 296.3
price_ct_per_kwh: 29.63
- start_time: "2022-12-16T00:00:00+00:00"
end_time: "2022-12-16T01:00:00+00:00"
price_eur_per_mwh: 288.12
price_ct_per_kwh: 28.812
- start_time: "2022-12-16T01:00:00+00:00"
end_time: "2022-12-16T02:00:00+00:00"
price_eur_per_mwh: 280.19
price_ct_per_kwh: 28.019
The sensor value reports the average EPEX Spot market price during the day. The sensor value reports the market price in EUR/MWh. The market price in ct/kWh is available as sensor attribute.
price_ct_per_kwh: 29.63
The sensor value reports the median EPEX Spot market price during the day. The sensor value reports the market price in EUR/MWh. The market price in ct/kWh is available as sensor attribute.
price_ct_per_kwh: 29.63
The sensor value reports the lowest EPEX Spot market price during the day. The sensor value reports the market price in EUR/MWh. The market price in ct/kWh is available as sensor attribute.
The sensor attributes contains the start and endtime of the lowest market price timeframe.
price_ct_per_kwh: 29.63
start_time: "2023-02-15T22:00:00+00:00"
end_time: "2023-02-15T23:00:00+00:00"
The sensor value reports the highest EPEX Spot market price during the day. The sensor value reports the market price in EUR/MWh. The market price in ct/kWh is available as sensor attribute.
The sensor attributes contains the start and endtime of the highest market price timeframe.
price_ct_per_kwh: 29.63
start_time: "2023-02-15T22:00:00+00:00"
end_time: "2023-02-15T23:00:00+00:00"
The sensor value reports the quantile between the lowest market price and the highest market price during the day in the range between 0 .. 1.
Examples:
- The sensor reports 0 if the current market price is the lowest during the day.
- The sensor reports 1 if the current market price is the highest during the day.
- If the sensor reports e.g., 0.25, then the current market price is 25% of the range between the lowest and the highest market price.
The sensor value reports the rank of the current market price during the day. Or in other words: The number of hours in which the price is lower than the current price.
Examples:
- The sensor reports 0 if the current market price is the lowest during the day. There is no lower market price during the day.
- The sensor reports 23 if the current market price is the highest during the day (if the market price will be updated hourly). There are 23 hours which are cheaper than the current hour market price.
- The sensor reports 1 if the current market price is the 2nd cheapest during the day. There is 1 one which is cheaper than the current hour market price.
List of Service Calls:
- Get Lowest Price Interval
- Get Highest Price Interval
- Fetch Data
Requires Release >= 2.0.0
Get the time interval during which the price is at its lowest/highest point.
Knowing the hours with the lowest / highest consecutive prices during the day could be an interesting use case. This might be of value when looking for the most optimum time to start your washing machine, dishwasher, dryer, etc.
With this service call, you can let the integration calculate the optimal start time. The only mandatory attribute is the duration of your appliance. Optionally you can limit start- and end-time, e.g. to start your appliance only during night hours.
epex_spot.get_lowest_price_interval
epex_spot.get_highest_price_interval
Service data attribute | Optional | Description | Example |
---|---|---|---|
device_id |
yes | A EPEX Spot service instance ID. In case you have multiple EPEX Spot instances. | 9d44d8ce9b19e0863cf574c2763749ac |
earliest_start |
yes | Earliest time to start the appliance. | "14:00:00" |
earliest_start_post |
yes | Postponement of earliest_start in days: 0 = today (default), 1= tomorrow |
0 |
latest_end |
yes | Latest time to end the appliance. | "16:00:00" |
latest_end_post |
yes | Postponement of latest_end in days: 0 = today (default), 1= tomorrow |
0 |
duration |
no | Required duration to complete appliance. | See below... |
Notes:
- If
earliest_start
is omitted, the current time is used instead. - If
latest_end
is omitted, the end of all available market data is used. earliest_start
refers to today ifearliest_start_post
is omitted or set to 0.latest_end
will be automatically trimmed to the available market area.- If
earliest_start
andlatest_end
are present andlatest_end
is earlier than (or equal to)earliest_start
, thenlatest_end
refers to tomorrow. device_id
is only required if have have setup multiple EPEX Spot instances. The easiest way to get the unique device id, is to use the Developer Tools -> Services.
Service Call Examples:
service: epex_spot.get_lowest_price_interval
data:
device_id: 9d44d8ce9b19e0863cf574c2763749ac
earliest_start: "14:00:00"
latest_end: "16:00:00"
duration:
hours: 1
minutes: 0
seconds: 0
service: epex_spot.get_lowest_price_interval
data:
earliest_start: "14:00:00"
latest_end: "16:00:00"
duration: "00:30:00" # 30 minutes
service: epex_spot.get_lowest_price_interval
data:
duration: "00:30" # 30 minutes
service: epex_spot.get_lowest_price_interval
data:
duration: 120 # in seconds -> 2 minutes
# get the lowest price all day tomorrow:
service: epex_spot.get_lowest_price_interval
data:
earliest_start: "00:00:00"
earliest_start_post: 1
latest_end: "00:00:00"
latest_end_post: 2
duration: "01:30:00" # 1h, 30 minutes
The response contains the calculated start and end-time and the average price per MWh/KWh.
Example:
start: "2023-09-29T12:00:00+00:00"
end: "2023-09-29T13:00:00+00:00"
price_eur_per_mwh: 77.04
price_ct_per_kwh: 7.704000000000001
net_price_ct_per_kwh: 23.6394928
With Home Assistant release >= 2023.9 you can use the Template Integration to create a sensor that shows the start time:
template:
- trigger:
- platform: time
at: "00:00:00"
action:
- service: epex_spot.get_lowest_price_interval
data:
earliest_start: "20:00:00"
latest_end: "23:00:00"
duration:
hours: 1
minutes: 5
response_variable: resp
sensor:
- name: Start Appliance
device_class: timestamp
state: "{{ resp.start is defined and resp.start }}"
This sensor can be used to trigger automations:
trigger:
- platform: time
at: sensor.start_appliance
condition: []
action: []
Requires Release >= 2.1.0
Fetch data from all services or a specific service.
epex_spot.fetch_data
Service data attribute | Optional | Description | Example |
---|---|---|---|
device_id |
yes | A EPEX Spot service instance ID. In case you have multiple EPEX Spot instances. | 9d44d8ce9b19e0863cf574c2763749ac |
With ApexCharts, you can easily show a chart like this:
You just have to install ApexCharts (via HACS) and enter the following data in the card configuration:
type: custom:apexcharts-card
header:
show: true
title: Electricity Prices
graph_span: 48h
span:
start: day
now:
show: true
label: Now
series:
- entity: sensor.epex_spot_de_price
name: Electricity Price
type: column
extend_to: end
data_generator: >
return entity.attributes.data.map((entry, index) => { return [new
Date(entry.start_time).getTime(), entry.price_eur_per_mwh]; });
NOTE: Deprecated since release 2.0.0. Use Service call instead!
It might be an interesting use case to know what the hours with lowest consecutive prices during the day are. This might be of value when looking for the most optimum time to start your washing machine, dishwasher, dryer, etc.
The template below determines when the 3 hours with lowest consecutive prices start, between 06:00 and 22:00.
You can change these hours in the template below, if you want hours before 06:00 and after 22:00 also to be considered.
Remove {%- set ns.combo = ns.combo[6:22] %}
do disable this filtering completely.
template:
- sensor:
- name: epex_start_low_period
state: >-
{% set ns = namespace(attr_dict=[]) %}
{% for item in (state_attr('sensor.epex_spot_be_price', 'data'))[0:24] %}
{%- set ns.attr_dict = ns.attr_dict + [(loop.index-1,item["price_eur_per_mwh"])] %}
{% endfor %}
{%- set price_map = dict(ns.attr_dict) %}
{%- set price_sort = price_map.values()|list %}
{%- set keys_list = price_map.keys()|list %}
{%- set ns = namespace(combo=[]) %}
{%- for p in keys_list %}
{%- set p = p|int %}
{%- if p < 22 %}
{%- set ns.combo = ns.combo + [(p, ((price_sort)[p] + (price_sort)[p+1] + (price_sort)[p+2])|round(2))] %}
{%- endif %}
{%- endfor %}
{%- set ns.combo = ns.combo[6:22] %}
{%- set mapper = dict(ns.combo) %}
{%- set key = mapper.keys()|list %}
{%- set val = mapper.values()|list %}
{%- set val_min = mapper.values()|min %}
{{ key[val.index(val_min)]|string + ":00" }}
Here's another ApexCharts example.
It shows the price for the current day, the next day and the min/max
value for each day.
Furthermore, it also fills the hours during which prices are lowest (see 2.)
type: custom:apexcharts-card
header:
show: false
graph_span: 48h
span:
start: day
now:
show: true
label: Now
color_list:
- var(--primary-color)
series:
- entity: sensor.epex_spot_be_price
yaxis_id: uurprijs
float_precision: 2
type: line
curve: stepline
extend_to: false
show:
extremas: true
data_generator: >
return entity.attributes.data.map((entry, index) => { return [new
Date(entry.start_time).getTime(), entry.price_eur_per_mwh]; }).slice(0,24);
color_threshold:
- value: 0
color: "#186ddc"
- value: 0.155
color: "#04822e"
- value: 0.2
color: "#12A141"
- value: 0.25
color: "#79B92C"
- value: 0.3
color: "#C4D81D"
- value: 0.35
color: "#F3DC0C"
- value: 0.4
color: red
- value: 0.5
color: magenta
- entity: sensor.epex_spot_be_price
yaxis_id: uurprijs
float_precision: 2
type: line
curve: stepline
extend_to: end
show:
extremas: true
data_generator: >
return entity.attributes.data.map((entry, index) => { return [new
Date(entry.start_time).getTime(), entry.price_eur_per_mwh]; }).slice(23,47);
color_threshold:
- value: 0
color: "#186ddc"
- value: 0.155
color: "#04822e"
- value: 0.2
color: "#12A141"
- value: 0.25
color: "#79B92C"
- value: 0.3
color: "#C4D81D"
- value: 0.35
color: "#F3DC0C"
- value: 0.4
color: red
- value: 0.5
color: magenta
- entity: sensor.epex_spot_be_price
yaxis_id: uurprijs
color: green
float_precision: 2
type: area
curve: stepline
extend_to: false
data_generator: >
return entity.attributes.data.map((entry, index) => { return [new
Date(entry.start_time).getTime(), entry.price_eur_per_mwh];}).slice(parseInt(hass.states['sensor.epex_start_low_period'].state.substring(0,2)),parseInt(hass.states['sensor.epex_start_low_period'].state.substring(0,2))+4);
experimental:
color_threshold: true
yaxis:
- id: uurprijs
decimals: 2
apex_config:
title:
text: €/MWh
tickAmount: 4
apex_config:
legend:
show: false
tooltip:
x:
show: true
format: HH:00 - HH:59