Zwift, Home Assistant и LED-подсветка

Да, подключил и Звифт к “Умному дому” 🙂

Пока ограничился только LED-подсветкой на мониторе, меняющей цвет в зависимости от пульса и мощности. Позже, возможно, сделаю управление вентиляторами – хотя мне они обычно требуются всегда на максимальной мощности.

LED-лента на мониторе – уже вторая реализация; первая была на мониторе на рабочем месте. Правда, та пока только предупреждает при воздушных тревогах, потом расширю автоматизацию.

Вернемся к Zwift’у. Для интеграции в Home Assistant (Core):

  • Качаем это.
  • Размещаем в $HOME у Home Assistant по пути ~/.homeassistant/custom_components/zwift/ содержимое каталога по ссылке выше (~/.homeassistant/ – то, где лежит configuration.yaml, база и так далее).
  • На момент написания плагин дефолтно добавляет User ID в Zwift’e к именам и названиям сенсоров. Завел Issue, будет движение или нет – не знаю, но у себя поправил код, чтобы этого не было:
diff

$ diff sensor.py sensor.py.orig
187,188c187,188
< self._unique_id = “{}_{}”.format(self._base_name, SENSOR_TYPES[self._type].get(
< ‘name’)).replace(” “, “”).lower()

> self._unique_id = “{}_{}_{}”.format(self._base_name, SENSOR_TYPES[self._type].get(
> ‘name’), self._player.player_id).replace(” “, “”).lower()
198c198
< return “{} {}”.format(self._base_name, SENSOR_TYPES[self._type].get(‘name’))

> return “{} {} ({})”.format(self._base_name, SENSOR_TYPES[self._type].get(‘name’), self._player.player_id)
203c203
< return “{} {}”.format(self._base_name, SENSOR_TYPES[self._type].get(‘name’))

> return “{} {} ({})”.format(self._base_name, SENSOR_TYPES[self._type].get(‘name’), self._player.friendly_player_id)

[свернуть]
  • В configuration.yaml добавляем

sensor:
  - platform: zwift
    username: !secret my_zwift_username
    password: !secret my_zwift_password

И в secrets.yaml – логин/пароль. Перезапуск HA (перечитывание конфигов мне почему-то не помогало :(). Дальше в “Настройки – Устройства – Объекты” появляются различные сенсоры. У “Zwift Online” дополнительно есть атрибуты с кучей самых разных данных из Zwift’а.

Исполнение в железе:

Корпус для конструкции решил (пока?) не делать. Подходящих нет; все равно ее там никто не видит, да и внешних воздействий нет. Питание берем прямо с монитора, там 12В. Лента – WS2812B с Али, вариант “60 шт/метр IP30”. На мониторе на нижней стороне разместились 32 штуки. Брал на 5В, как более универсальный. В итоге 12В с монитора через KIS 3R33 (да, надо их утилизировать наконец-то) получаем 5В, а для ESP01 дополнительно стоит l78l33. При настройке KIS 3R33 (резистор примерно на 9,1 кОм между входом регулировки и общим) желательно напряжение чуть-чуть завышать относительно 5В (для более четкой работы 78l33).

Конфиг для ESP01 для ESP Home.

Заводим девайс в HA, дальше автоматизация:

Dell power LED

alias: Zwift power led
description: ""
trigger:
  - platform: state
    entity_id:
      - sensor.zwift_power
condition: []
action:
  - if:
      - condition: template
        value_template: >-
          {{ states("sensor.zwift_power") | float < state_attr("sensor.zwift_online","ftp") | float * 0.6 }} then: - service: light.turn_on metadata: {} data: rgb_color: - 204 - 204 - 204 transition: 0 brightness_pct: 75 target: entity_id: light.dell_zwift_led_power - if: - condition: template value_template: >-
          {{ state_attr("sensor.zwift_online","ftp") | float * 0.6 <=
          states("sensor.zwift_power") | float < state_attr("sensor.zwift_online","ftp") | float * 0.75 }} then: - service: light.turn_on metadata: {} data: rgb_color: - 0 - 145 - 255 transition: 0 brightness_pct: 75 target: entity_id: light.dell_zwift_led_power - if: - condition: template value_template: >-
          {{ state_attr("sensor.zwift_online","ftp") | float * 0.75 <=
          states("sensor.zwift_power") | float < state_attr("sensor.zwift_online","ftp") | float * 0.89 }} then: - service: light.turn_on metadata: {} data: rgb_color: - 43 - 255 - 0 transition: 0 brightness_pct: 75 target: entity_id: light.dell_zwift_led_power - if: - condition: template value_template: >-
          {{ state_attr("sensor.zwift_online","ftp") | float * 0.89 <=
          states("sensor.zwift_power") | float < state_attr("sensor.zwift_online","ftp") | float * 1.04 }} then: - service: light.turn_on metadata: {} data: rgb_color: - 255 - 234 - 0 transition: 0 brightness_pct: 75 target: entity_id: light.dell_zwift_led_power - if: - condition: template value_template: >-
          {{ state_attr("sensor.zwift_online","ftp") | float * 1.04 <=
          states("sensor.zwift_power") | float < state_attr("sensor.zwift_online","ftp") | float * 1.18 }} then: - service: light.turn_on metadata: {} data: rgb_color: - 255 - 136 - 0 transition: 0 brightness_pct: 82 target: entity_id: light.dell_zwift_led_power - if: - condition: template value_template: >-
          {{ state_attr("sensor.zwift_online","ftp") | float * 1.18 <=
          states("sensor.zwift_power") | float }}
    then:
      - service: light.turn_on
        metadata: {}
        data:
          rgb_color:
            - 255
            - 0
            - 0
          transition: 0
          brightness_pct: 95
        target:
          entity_id: light.dell_zwift_led_power
mode: single


[свернуть]

Пока не знаю, как использовать zwift_online->ftp для расчетов зон – ругается на синтаксис, а то и вообще страница HA виснет (!). Разберусь – поменяю. Update: переделал автоматизацию целиком, теперь подтягивает FTP. Возможно, стоит задать (если его нет) какое-то дефолтное значение на тот случай, если FTP не указан.

Аналогичная автоматизация и для HR-подсветки:

Dell HR LED

alias: Zwift LED HR
description: ""
trigger:
  - platform: state
    entity_id:
      - sensor.zwift_heart_rate
condition: []
action:
  - if:
      - condition: numeric_state
        entity_id: sensor.zwift_heart_rate
        below: 108
    then:
      - service: light.turn_on
        metadata: {}
        data:
          rgb_color:
            - 0
            - 204
            - 255
          transition: 0
          brightness_pct: 75
          effect: Pulse 1
        target:
          entity_id:
            - light.dell_zwift_led_hr_r
            - light.dell_zwift_led_hr_l
  - if:
      - condition: numeric_state
        entity_id: sensor.zwift_heart_rate
        below: 126
        above: 108
    then:
      - service: light.turn_on
        metadata: {}
        data:
          rgb_color:
            - 4
            - 255
            - 0
          transition: 0
          brightness_pct: 75
          effect: Pulse 2
        target:
          entity_id:
            - light.dell_zwift_led_hr_r
            - light.dell_zwift_led_hr_l
  - if:
      - condition: numeric_state
        entity_id: sensor.zwift_heart_rate
        below: 144
        above: 126
    then:
      - service: light.turn_on
        metadata: {}
        data:
          rgb_color:
            - 255
            - 247
            - 0
          transition: 0
          brightness_pct: 75
          effect: Pulse 3
        target:
          entity_id:
            - light.dell_zwift_led_hr_r
            - light.dell_zwift_led_hr_l
  - if:
      - condition: numeric_state
        entity_id: sensor.zwift_heart_rate
        below: 162
        above: 144
    then:
      - service: light.turn_on
        metadata: {}
        data:
          rgb_color:
            - 255
            - 115
            - 0
          transition: 0
          brightness_pct: 75
          effect: Pulse 4
        target:
          entity_id:
            - light.dell_zwift_led_hr_r
            - light.dell_zwift_led_hr_l
  - if:
      - condition: numeric_state
        entity_id: sensor.zwift_heart_rate
        above: 162
    then:
      - service: light.turn_on
        metadata: {}
        data:
          rgb_color:
            - 255
            - 0
            - 0
          transition: 0
          brightness_pct: 82
          effect: Pulse 5
        target:
          entity_id:
            - light.dell_zwift_led_hr_r
            - light.dell_zwift_led_hr_l
mode: single


[свернуть]

Первый пример (до разделения ленты на 3 раздела):

После этого решил вывести мощность в центр, а пульс – по краям и сделать легкую пульсацию для HR (надеюсь, не будет мешать – еще не пробовал).

Да, для partition-сущностей в esphome эффекты надо описывать отдельно.

PS: вообще изначально понравилась эта реализация:

Но тут без каких-либо ссылок. Однако именно тут упоминалось подключение к HA, с чего все и началось.

Update: как-то так:

Update: добавил “Радугу” для like’ов и смены уровня, а также возможность предупреждения о воздушной тревоге. На деле эффект “Радуга” не сильно наглядный по отношению к штатной работе, лучше какой-нибудь “Фейерверк”. Так или иначе, чтобы все работало – в секцию эффектов в esphome добавляем “– addressable_rainbow:” для всей ленты целиком (!), а не разделов. Шьем, дальше можно эффект использовать из Home Assistant.

В свою очередь в HA нужен некий флаг, позволяющий прервать работу основной автоматизации и включить некое уведомление. Идем в “Настройки – Устройства – Вспомогательное“, создаем Template-переключатель. Я его назвал “Dell LED alert”. И дальше в созданных выше автоматизациях добавляем дополнительную проверку – “если созданный переключатель выключен”:

В коде это будет так:

Пример для Power

alias: Zwift power led
description: ""
trigger:
  - platform: state
    entity_id:
      - sensor.zwift_power
condition:
  - condition: state
    entity_id: input_boolean.dell_led_alert
    state: "off"
action:
  - if:
      - condition: template
        value_template: >-
          {{ states("sensor.zwift_power") | float <
          state_attr("sensor.zwift_online","ftp") | float * 0.6 }}
************ и так далее **************

[свернуть]

А дальше создаем еще одну автоматизацию и вспомогательный сенсор. Сенсор потребуется, чтобы вытаскивать текущее число лайков из Zwift-сенсора. Идем в “Настройки – устройства – вспомогательное” и создаем сенсор-шаблон вида

{{ state_attr("sensor.zwift_online","latest_activity").activityRideOnCount | float}}

Возможно, можно вытаскивать значение и прямо в автоматизации, но я остановился на таком варианте.

А дальше – в автоматизации используем изменения этого числа, как триггер. И по срабатыванию триггера включаем созданную ранее блокировку, мигаем лентой так, как нам надо и в конце – выключаем блокировку:

Zwift ride ons

alias: Zwift Ride On and LVLs
description: ""
trigger:
  - platform: state
    entity_id:
      - sensor.zwift_ride_ons
  - platform: state
    entity_id:
      - sensor.zwift_online
    attribute: playerLevel
condition: []
action:
  - service: input_boolean.turn_on
    metadata: {}
    data: {}
    target:
      entity_id: input_boolean.dell_led_alert
  - service: light.turn_on
    metadata: {}
    data:
      effect: Rainbow
    target:
      entity_id: light.dell_zwift_led_light
  - delay:
      hours: 0
      minutes: 0
      seconds: 5
      milliseconds: 0
  - service: light.turn_off
    metadata: {}
    data: {}
    target:
      entity_id: light.dell_zwift_led_light
  - service: input_boolean.turn_off
    metadata: {}
    data: {}
    target:
      entity_id: input_boolean.dell_led_alert
mode: single


[свернуть]

Аналогичным образом сделано и уведомление о воздушных тревогах: в автоматизации воздушки добавлено выключение основной автоматизации Звифта, мигаем несколько раз красным, дальше снимаем блок и по первому же событию у нас продолжается отображение мощности и пульса.

Update: при окончании trial-периода в 2 недели API перестает работать – т.е., перестает работать и подсветка, даже если в профиль добавлена платежная карта и сам по себе Zwift продолжает работать во время попыток списать деньги с карты.

Одна мысль про “Zwift, Home Assistant и LED-подсветка”

Добавить комментарий