Crate rsiot_modbus_client
source ·Expand description
Опрос устройств Modbus, используя библиотеку tokio-modbus
Ссылки:
§Тестирование
Готовый docker-образ для тестов - GitHub.
Запускается через docker compose в корне. Инструкция.
§Диаграмма
§Пример
//! Простейший пример использования клиента Modbus
//!
//! Для тестирования можно использовать образ docker oitc/modbus-server
//!
//! Выполняется две операции:
//! - раз в 2 секунды на сервер в регистр 0 записывается значение счетчика (`input_config`)
//! - раз в 2 секунды считывается значение регистра 0 (`periodic_config`) и отправляется в логгер
//!
//! Запуск:
//!
//! ```bash
//! cargo run -p rsiot-modbus-client --example modbus_tcp_client
//! ```
#[cfg(any(target_arch = "x86_64", target_arch = "aarch64"))]
#[tokio::main]
async fn main() -> anyhow::Result<()> {
use std::net::{IpAddr, Ipv4Addr};
use serde::{Deserialize, Serialize};
use tokio::time::Duration;
use tracing::Level;
use tracing_subscriber::fmt;
use rsiot_component_core::{ComponentExecutor, ComponentExecutorConfig};
use rsiot_extra_components::{cmp_inject_periodic, cmp_logger};
use rsiot_messages_core::{Message, MsgDataBound};
use rsiot_modbus_client::cmp_modbus_client::{self, *};
#[derive(Clone, Debug, Deserialize, PartialEq, Serialize)]
pub enum Messages {
ValueWrite(f64),
ValueRead(f64),
}
impl MsgDataBound for Messages {}
// логгирование
fmt().init();
// Конфигурация modbus клиента
let modbus_client_config = Config {
enabled: true,
unit_id: 1,
input_config: vec![InputConfig {
fn_input: |msg| match msg.get_data()? {
Messages::ValueWrite(val) => Some(Request::WriteSingleRegister(0, val as u16)),
Messages::ValueRead(_) => None,
},
fn_on_success: |_data| vec![],
fn_on_failure: Vec::new,
}],
periodic_config: vec![PeriodicConfig {
period: Duration::from_secs(2),
request: Request::ReadHoldingRegisters(0, 1),
fn_on_success: |data| {
let mut msgs = vec![];
if let Response::U16(data) = data {
msgs.push(Message::new_custom(Messages::ValueRead(data[0] as f64)));
}
msgs
},
fn_on_failure: Vec::new,
}],
client_type: ClientType::Tcp(TcpClientType {
host: IpAddr::V4(Ipv4Addr::new(127, 0, 0, 1)),
port: 5020,
}),
};
let executor_config = ComponentExecutorConfig {
buffer_size: 100,
executor_name: "modbus_tcp_client".into(),
fn_auth: |msg, _| Some(msg),
};
let mut counter = 0.0;
ComponentExecutor::new(executor_config)
// Периодическое генерирование сообщения для записи счетчика на сервер
.add_cmp(cmp_inject_periodic::Cmp::new(cmp_inject_periodic::Config {
period: Duration::from_secs(2),
fn_periodic: move || {
let msg = Message::new_custom(Messages::ValueWrite(counter));
counter += 1.0;
vec![msg]
},
}))
// Клиент modbus
.add_cmp(cmp_modbus_client::Cmp::new(modbus_client_config))
// Вывод сообщений в лог
.add_cmp(cmp_logger::Cmp::new(cmp_logger::Config {
level: Level::INFO,
header: "".into(),
}))
.wait_result()
.await?;
Ok(())
}
#[cfg(not(any(target_arch = "x86_64", target_arch = "aarch64")))]
fn main() {}
TODO - рестарт не работает
Modules§
- Обмен данными с устройством, поддерживающим Modbus TCP сервер.
- Модуль для конвертации числовых данных в регистры и наоборот.