Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Change slave to device_id and slave= to device_id=. #2600

Merged
merged 1 commit into from
Mar 6, 2025
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
10 changes: 6 additions & 4 deletions API_changes.rst
Original file line number Diff line number Diff line change
Expand Up @@ -2,13 +2,15 @@ API changes
===========
Versions (X.Y.Z) where Z > 0 e.g. 3.0.1 do NOT have API changes!

-----------------
API changes 3.9.0
API changes 4.0.0
-----------------
- Python 3.9 is reaching end of life, and no longer supported.
Depending on the usage the code might still work
- Start*Server, custom_functions -> custom_pdu (handled by Modbus<x>Server)
- payload removed (replaced by "convert_combined_to/from_registers")
- Start*Server, custom_functions is now custom_pdu (handled by Modbus<x>Server)
- ModbusSlaveContext replaced by ModbusDeviceContext
- payload removed (replaced by "convert_to/from_registers")
- slave=, slaves= replaced by device_id=, device_ids=
- slave request names changed to device

API changes 3.8.0
-----------------
Expand Down
1 change: 1 addition & 0 deletions doc/index.rst
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@ Please select a topic in the left hand column.
:caption: Contents:
:hidden:

source/upgrade_40
source/api_changes
source/client
source/server
Expand Down
Binary file modified doc/source/_static/examples.tgz
Binary file not shown.
Binary file modified doc/source/_static/examples.zip
Binary file not shown.
38 changes: 19 additions & 19 deletions doc/source/client.rst
Original file line number Diff line number Diff line change
Expand Up @@ -127,12 +127,12 @@ Synchronous example

from pymodbus.client import ModbusTcpClient

client = ModbusTcpClient('MyDevice.lan') # Create client object
client.connect() # connect to device
client.write_coil(1, True, slave=1) # set information in device
result = client.read_coils(2, 3, slave=1) # get information from device
print(result.bits[0]) # use information
client.close() # Disconnect device
client = ModbusTcpClient('MyDevice.lan') # Create client object
client.connect() # connect to device
client.write_coil(1, True, device_id=1) # set information in device
result = client.read_coils(2, 3, device_id=1) # get information from device
print(result.bits[0]) # use information
client.close() # Disconnect device

The line :mod:`client.connect()` connects to the device (or comm port). If this cannot connect successfully within
the timeout it throws an exception. After this initial connection, further
Expand All @@ -147,22 +147,22 @@ Asynchronous example

from pymodbus.client import AsyncModbusTcpClient

client = AsyncModbusTcpClient('MyDevice.lan') # Create client object
await client.connect() # connect to device, reconnect automatically
await client.write_coil(1, True, slave=1) # set information in device
result = await client.read_coils(2, 3, slave=1) # get information from device
print(result.bits[0]) # use information
client.close() # Disconnect device
client = AsyncModbusTcpClient('MyDevice.lan') # Create client object
await client.connect() # connect to device, reconnect automatically
await client.write_coil(1, True, device_id=1) # set information in device
result = await client.read_coils(2, 3, device_id=1) # get information from device
print(result.bits[0]) # use information
client.close() # Disconnect device

The line :mod:`client = AsyncModbusTcpClient('MyDevice.lan')` only creates the object; it does not activate
anything.

The line :mod:`await client.connect()` connects to the device (or comm port), if this cannot connect successfully within
the timeout it throws an exception. If connected successfully reconnecting later is handled automatically

The line :mod:`await client.write_coil(1, True, slave=1)` is an example of a write request, set address 1 to True on device 1 (slave).
The line :mod:`await client.write_coil(1, True, device_id=1)` is an example of a write request, set address 1 to True on device 1.

The line :mod:`result = await client.read_coils(2, 3, slave=1)` is an example of a read request, get the value of address 2, 3 and 4 (count = 3) from device 1 (slave).
The line :mod:`result = await client.read_coils(2, 3, device_id=1)` is an example of a read request, get the value of address 2, 3 and 4 (count = 3) from device 1.

The last line :mod:`client.close()` closes the connection and render the object inactive.

Expand Down Expand Up @@ -194,13 +194,13 @@ Client device addressing
------------------------

With **TCP**, **TLS** and **UDP**, the tcp/ip address of the physical device is defined when creating the object.
Logical devices represented by the device is addressed with the :mod:`slave=` parameter.
Logical devices represented by the device is addressed with the :mod:`device_id=` parameter.

With **Serial**, the comm port is defined when creating the object.
The physical devices are addressed with the :mod:`slave=` parameter.
The physical devices are addressed with the :mod:`device_id=` parameter.

:mod:`slave=0` is defined as broadcast in the modbus standard, but pymodbus treats it as a normal device.
please note :mod:`slave=0` can only be used to address devices that truly have id=0 ! Using :mod:`slave=0` to
:mod:`device_id=0` is defined as broadcast in the modbus standard, but pymodbus treats it as a normal device.
please note :mod:`device_id=0` can only be used to address devices that truly have id=0 ! Using :mod:`device_id=0` to
address a single device with id not 0 is against the protocol.

If an application is expecting multiple responses to a broadcast request, it must call :mod:`client.execute` and deal with the responses.
Expand All @@ -217,7 +217,7 @@ All simple request calls (mixin) return a unified result independent whether it
The application should evaluate the result generically::

try:
rr = await client.read_coils(1, 1, slave=1)
rr = await client.read_coils(1, 1, device_id=1)
except ModbusException as exc:
_logger.error(f"ERROR: exception in pymodbus {exc}")
raise exc
Expand Down
2 changes: 1 addition & 1 deletion doc/source/library/datastore.rst
Original file line number Diff line number Diff line change
Expand Up @@ -11,7 +11,7 @@ Datastore classes
:members:
:member-order: bysource

.. autoclass:: pymodbus.datastore.ModbusSlaveContext
.. autoclass:: pymodbus.datastore.ModbusDeviceContext
:members:
:member-order: bysource

Expand Down
8 changes: 4 additions & 4 deletions doc/source/library/simulator/calls_response.rst
Original file line number Diff line number Diff line change
Expand Up @@ -72,7 +72,7 @@
},
{
"value": 17,
"text": "report_slave_id",
"text": "report_device_id",
"selected": false
},
{
Expand Down Expand Up @@ -131,7 +131,7 @@
},
{
"value": 4,
"text": "SLAVE_FAILURE",
"text": "DEVICE_FAILURE",
"selected": false
},
{
Expand All @@ -141,7 +141,7 @@
},
{
"value": 6,
"text": "SLAVE_BUSY",
"text": "DEVICE_BUSY",
"selected": false
},
{
Expand All @@ -164,4 +164,4 @@
"call_rows": [],
"foot": "not active",
"result": "ok"
}
}
6 changes: 3 additions & 3 deletions doc/source/library/simulator/config.rst
Original file line number Diff line number Diff line change
Expand Up @@ -87,7 +87,7 @@ Server configuration examples
"comm": "tcp",
"host": "0.0.0.0",
"port": 5020,
"ignore_missing_slaves": false,
"ignore_missing_devices": false,
"framer": "socket",
"identity": {
"VendorName": "pymodbus",
Expand Down Expand Up @@ -123,7 +123,7 @@ Server configuration examples
"port": 5020,
"certfile": "certificates/pymodbus.crt",
"keyfile": "certificates/pymodbus.key",
"ignore_missing_slaves": false,
"ignore_missing_devices": false,
"framer": "tls",
"identity": {
"VendorName": "pymodbus",
Expand All @@ -138,7 +138,7 @@ Server configuration examples
"comm": "udp",
"host": "0.0.0.0",
"port": 5020,
"ignore_missing_slaves": false,
"ignore_missing_devices": false,
"framer": "socket",
"identity": {
"VendorName": "pymodbus",
Expand Down
3 changes: 1 addition & 2 deletions doc/source/roadmap.rst
Original file line number Diff line number Diff line change
Expand Up @@ -20,12 +20,11 @@ The following bullet points are what the maintainers focus on:
- 4.0.0, with:
- Simulator datastore, with simple configuration
- Remove remote_datastore
- Remove BinaryPayload
- Server becomes Simulator
- client async with sync/async API
- Only one datastore, but with different API`s
- 4.1.0, with:
- ModbusControlBlock pr slave
- ModbusControlBlock pr device
- New custom PDU (function codes)
- New serial forwarder
- GUI client, to analyze devices
Expand Down
47 changes: 47 additions & 0 deletions doc/source/upgrade_40.rst
Original file line number Diff line number Diff line change
@@ -0,0 +1,47 @@
Pymodbus 4.0 upgrade procedure
==============================

Pymodbus 4.0 contains a number of incompatibilities with Pymodbus 3.x, however
most of these are simple edits.

Python 3.9
----------
Python 3.9 is reaching end of life and from october 2025 no longer receives security updates.

Pymodbus starting with v4.0 start using python 3.10 features, and thus users need to update to
at least python v3.10

Users that cannot upgrade the python version, should not upgrade pymodbus to v4.X


Start<x>Server
--------------
custom_funcion= is changed to custom_pdu= and is handled by Modbus<x>Server.


payload classes removed
-----------------------
Please replace by result.convert_from_registers() and/or convert_to_registers()


Simple replacements
-------------------

please replace
- slave= with device_id=
- slaves= with device_ids=
- ModbusServerContext(slaves=) with ModbusServerContext(devices=)

please rename
- ModbusSlaveContext to ModbusDeviceContext
- RemoteSlaveContext to RemoteDeviceContext
- report_slave_id() with report_device_id()
- diag_read_slave_message_count with diag_read_device_message_count
- diag_read_slave_no_response_count with diag_read_device_no_response_count
- diag_read_slave_nak_count with diag_read_device_nak_count
- diag_read_slave_busy_count with diag_read_device_busy_count
- ReturnSlaveMessageCountRequest with ReturnDeviceMessageCountRequest
- ReturnSlaveNoResponseCountRequest with ReturnDeviceNoResponseCountRequest
- ReturnSlaveNAKCountRequest with ReturnDeviceNAKCountRequest
- ReturnSlaveBusyCountRequest with ReturnDeviceBusyCountRequest
- ReturnSlaveBusCharacterOverrunCountRequest with ReturnDeviceBusCharacterOverrunCountRequest
4 changes: 2 additions & 2 deletions examples/client_async.py
Original file line number Diff line number Diff line change
Expand Up @@ -125,9 +125,9 @@ async def run_async_client(client, modbus_calls=None):

async def run_a_few_calls(client):
"""Test connection works."""
rr = await client.read_coils(32, count=1, slave=1)
rr = await client.read_coils(32, count=1, device_id=1)
assert len(rr.bits) == 8
rr = await client.read_holding_registers(4, count=2, slave=1)
rr = await client.read_holding_registers(4, count=2, device_id=1)
assert rr.registers[0] == 17
assert rr.registers[1] == 17

Expand Down
Loading