Modbus Controller Select ======================== .. seo:: :description: Instructions for setting up Modbus Controller Select(s) with ESPHome. The ``modbus_controller`` Select platform allows you to create a Select from modbus registers. Configuration variables: ------------------------ - **name** (**Required**, string): The name of the Select. - **address** (**Required**, int): The start address of the first or only register of the Select (can be decimal or hexadecimal). - **optionsmap** (**Required**, Map[str, int]): Provide a mapping from options (str) of this Select to values (int) of the modbus register and vice versa. All options and all values have to be unique. - **value_type** (*Optional*): The datatype of the modbus data. Defaults to ``U_WORD``. - ``U_WORD`` (unsigned 16 bit integer from 1 register = 16bit) - ``S_WORD`` (signed 16 bit integer from 1 register = 16bit) - ``U_DWORD`` (unsigned 32 bit integer from 2 registers = 32bit) - ``S_DWORD`` (signed 32 bit integer from 2 registers = 32bit) - ``U_DWORD_R`` (unsigned 32 bit integer from 2 registers low word first) - ``S_DWORD_R`` (signed 32 bit integer from 2 registers low word first) - ``U_QWORD`` (unsigned 64 bit integer from 4 registers = 64bit) - ``S_QWORD`` (signed 64 bit integer from 4 registers = 64bit) - ``U_QWORD_R`` (unsigned 64 bit integer from 4 registers low word first) - ``U_QWORD_R`` (signed 64 bit integer from 4 registers low word first) - **register_count** (*Optional*): The number of registers which are used for this Select. Only required for uncommon response encodings or to :ref:`optimize modbus communications<modbus_register_count>`. Overrides the defaults determined by ``value_type``. - **skip_updates** (*Optional*, int): By default, all sensors of a modbus_controller are updated together. For data points that don't change very frequently, updates can be skipped. A value of 5 would only update this sensor range in every 5th update cycle. Note: The modbus_controller groups components by address ranges to reduce number of transactions. All components with the same starting address will be updated in one request. ``skip_updates`` applies for *all* components in the same range. - **register_count** (*Optional*, int): The number of consecutive registers this read request should span or skip in a single command. Default is 1. See :ref:`modbus_register_count` for more details. - **force_new_range** (*Optional*, boolean): If possible sensors with sequential addresses are grouped together and requested in one range. Setting this to ``true`` enforces the start of a new range at that address. - **id** (*Optional*, :ref:`config-id`): Manually specify the ID used for code generation. - **lambda** (*Optional*, :ref:`lambda <config-lambda>`): Lambda to be evaluated every update interval to get the current option of the select. Parameters passed into lambda - **x** (``int64_t``): The parsed integer value of the modbus data. - **data** (``const std::vector<uint8_t>&``): vector containing the complete raw modbus response bytes for this sensor. Note: because the response contains data for all registers in the same range you have to use ``data[item->offset]`` to get the first response byte for your sensor. - **item** (``ModbusSelect*const``): The sensor object itself. Possible return values for the lambda: - ``return <std::string>;`` The new option for this Select. - ``return {};`` Use default mapping (see ``optionsmap``). - **write_lambda** (*Optional*, :ref:`lambda <config-lambda>`): Lambda to be evaluated on every update of the Sensor, before the new value is written to the modbus registers. - **use_write_multiple** (*Optional*, boolean): By default the modbus command *Function Code 6 (Preset Single Registers)* is used for setting the holding register if only one register is set. If your device only supports *Function Code 16 (Preset Multiple Registers)* set this option to ``true``. - **optimistic** (*Optional*, boolean): Whether to operate in optimistic mode - when in this mode, any command sent to the Modbus Select will immediately update the reported state. Defaults to ``false``. - All other options from :ref:`Select <config-select>`. .. code-block:: yaml # example lambda: |- ESP_LOGD("Reg1000", "Received value %lld", x); ESP_LOGD("Reg1000", "Parsed from bytes 0x%x;0x%x", data[item->offset], data[item->offset + 1]); if (x > 3) { return std::string("Three"); } Parameters passed into ``write_lambda`` --------------------------------------- - **x** (``const std::string&``): The option value to set for this Select. - **value** (``int64_t``): The mapping value of ``x`` using ``optionsmap``. - **payload** (``std::vector<uint16_t>& payload``): Empty vector for the payload. The lamdba can add 16 bit raw modbus register words which are send to the modbus device. - **item** (``ModbusSelect*const``): The sensor object itself. Possible return values for the lambda: - ``return <int64_t>;`` the value which should be written to the configured modbus registers. If there were data written to ``payload`` this value is ignored. - ``return {};`` Skip updating the register. .. code-block:: yaml # example write_lambda: |- ESP_LOGD("Reg1000", "Set option to %s (%lld)", x.c_str(), value); // re-use default option value from optionsmap if (value == 0) { return value; } // return own option value if (x == "One") { return 2; } // write payload if (x == "Two") { payload.push_back(0x0001); return 0; // any value will do } // ignore update return {}; Example: -------- .. code-block:: yaml # Example configuration entry select: - platform: modbus_controller name: "Modbus Select Register 1000" address: 1000 value_type: U_WORD optionsmap: "Zero": 0 "One": 1 "Two": 2 "Three": 3 See Also -------- - :doc:`/components/modbus` - :doc:`/components/modbus_controller` - :doc:`/components/sensor/modbus_controller` - :doc:`/components/binary_sensor/modbus_controller` - :doc:`/components/output/modbus_controller` - :doc:`/components/switch/modbus_controller` - :doc:`/components/number/modbus_controller` - :doc:`/components/text_sensor/modbus_controller` - :ref:`automation` - https://www.modbustools.com/modbus.html - :ghedit:`Edit`